← 返回首页
create_app - OPTIMADE Python tools
Skip to content
OPTIMADE Python tools
create_app
optimade-python-tools

create_app

add_major_version_base_url(app, index=False)

Add mandatory vMajor endpoints, i.e. all except /versions.

Source code in optimade/server/create_app.py
38 39 40 41 42 43
def add_major_version_base_url(app: FastAPI, index: bool = False): """Add mandatory vMajor endpoints, i.e. all except /versions.""" for endpoint in INDEX_ENDPOINTS if index else MAIN_ENDPOINTS: app.include_router( endpoint.router, prefix=BASE_URL_PREFIXES["major"], include_in_schema=False )

add_optional_versioned_base_urls(app, index=False)

Add the following OPTIONAL prefixes/base URLs to server:

/vMajor.Minor /vMajor.Minor.Patch

Source code in optimade/server/create_app.py
46 47 48 49 50 51 52 53 54 55 56 57 58 59
def add_optional_versioned_base_urls(app: FastAPI, index: bool = False): """Add the following OPTIONAL prefixes/base URLs to server: ``` /vMajor.Minor /vMajor.Minor.Patch ``` """ for version in ("minor", "patch"): for endpoint in INDEX_ENDPOINTS if index else MAIN_ENDPOINTS: app.include_router( endpoint.router, prefix=BASE_URL_PREFIXES[version], include_in_schema=False, )

create_app(config=None, index=False, logger_tag=None)

Create and configure a FastAPI app for the OPTIMADE API.

Sets up logging, middleware, exception handlers, routers, and optional test/JSONL data insertion. Can be used for both a regular OPTIMADE API or the index meta-database variant.

Note that the global ServerConfig instance is read from the "OPTIMADE_" env variables or the config json file, but this function allows to override config options for individual apps by passing a custom ServerConfig.

Parameters:

Name Type Description Default
config ServerConfig | None

ServerConfig instance to override config options specific to this app.

None
index bool

If True, create an index meta-database instance.

False
logger_tag str | None

Optional tag for the logger.

None

Returns:

Type Description
FastAPI

Configured FastAPI application.

Source code in optimade/server/create_app.py
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304
def create_app( config: ServerConfig | None = None, index: bool = False, logger_tag: str | None = None, ) -> FastAPI: """ Create and configure a FastAPI app for the OPTIMADE API. Sets up logging, middleware, exception handlers, routers, and optional test/JSONL data insertion. Can be used for both a regular OPTIMADE API or the index meta-database variant. Note that the global ServerConfig instance is read from the "OPTIMADE_" env variables or the config json file, but this function allows to override config options for individual apps by passing a custom ServerConfig. Args: config: ServerConfig instance to override config options specific to this app. index: If True, create an index meta-database instance. logger_tag: Optional tag for the logger. Returns: Configured FastAPI application. """ # create app-specific logger logger = create_logger(logger_tag, config) if config_warnings: logger.warning( f"Invalid config file or no config file provided, running server with default settings. Errors: " f"{[warnings.formatwarning(w.message, w.category, w.filename, w.lineno, '') for w in config_warnings]}" ) else: logger.info( f"Loaded settings from {os.getenv('OPTIMADE_CONFIG_FILE', DEFAULT_CONFIG_FILE_PATH)}" ) if config is None: config = ServerConfig() if config.debug: # pragma: no cover logger.info("DEBUG MODE") title = "OPTIMADE API" if not index else "OPTIMADE API - Index meta-database" description = """The [Open Databases Integration for Materials Design (OPTIMADE) consortium](https://www.optimade.org/) aims to make materials databases interoperational by developing a common REST API.\n""" if index: description += 'This is the "special" index meta-database.\n' description += f"\nThis specification is generated using [`optimade-python-tools`](https://github.com/Materials-Consortia/optimade-python-tools/tree/v{__version__}) v{__version__}." if index: config.is_index = True app = FastAPI( root_path=config.root_path, title=title, description=description, version=__api_version__, docs_url=f"{BASE_URL_PREFIXES['major']}/extensions/docs", redoc_url=f"{BASE_URL_PREFIXES['major']}/extensions/redoc", openapi_url=f"{BASE_URL_PREFIXES['major']}/extensions/openapi.json", default_response_class=JSONAPIResponse, separate_input_output_schemas=False, ) # Save the config in the app state for access in endpoints app.state.config = config # create entry collections and save in app state for access in endpoints entry_collections = create_entry_collections(config) app.state.entry_collections = entry_collections # store also the BaseResourceMapper app.state.base_resource_mapper = BaseResourceMapper() if not index: if config.insert_test_data or config.insert_from_jsonl: insert_main_data(config, entry_collections, logger) else: if config.insert_test_data and config.index_links_path.exists(): insert_index_data(config, entry_collections, logger) # Add middleware to set logging context @app.middleware("http") async def set_context(request, call_next): set_logging_context(logger_tag) response = await call_next(request) return response # Add CORS middleware app.add_middleware(CORSMiddleware, allow_origins=["*"]) # Then add required OPTIMADE middleware for middleware in OPTIMADE_MIDDLEWARE: app.add_middleware(middleware) # Enable GZIP after other middleware. if config.gzip.enabled: app.add_middleware( GZipMiddleware, minimum_size=config.gzip.minimum_size, compresslevel=config.gzip.compresslevel, ) # Add exception handlers for exception, handler in OPTIMADE_EXCEPTIONS: app.add_exception_handler(exception, handler) # Add various endpoints to unversioned URL endpoints = INDEX_ENDPOINTS if index else MAIN_ENDPOINTS endpoints += [versions] for endpoint in endpoints: app.include_router(endpoint.router) # add the versioned endpoints add_major_version_base_url(app, index=index) add_optional_versioned_base_urls(app, index=index) return app