2020-10-02 06:37:50 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
|
|
|
# Copyright (C) 2020 GNS3 Technologies Inc.
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""
|
|
|
|
FastAPI app
|
|
|
|
"""
|
|
|
|
|
2023-08-17 07:36:50 +00:00
|
|
|
from fastapi import FastAPI, Request, HTTPException, status
|
2020-10-02 06:37:50 +00:00
|
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
|
|
from fastapi.responses import JSONResponse
|
2023-08-17 07:36:50 +00:00
|
|
|
from fastapi.exceptions import RequestValidationError
|
2024-12-19 09:54:11 +00:00
|
|
|
from fastapi.staticfiles import StaticFiles
|
2021-05-16 09:07:17 +00:00
|
|
|
from sqlalchemy.exc import SQLAlchemyError
|
2021-04-17 09:03:20 +00:00
|
|
|
from uvicorn.main import Server as UvicornServer
|
2020-11-19 04:51:03 +00:00
|
|
|
|
2024-12-19 09:54:11 +00:00
|
|
|
from fastapi.openapi.docs import (
|
|
|
|
get_redoc_html,
|
|
|
|
get_swagger_ui_html,
|
|
|
|
get_swagger_ui_oauth2_redirect_html,
|
|
|
|
)
|
|
|
|
|
2020-10-02 06:37:50 +00:00
|
|
|
from gns3server.controller.controller_error import (
|
|
|
|
ControllerError,
|
|
|
|
ControllerNotFoundError,
|
2020-12-02 08:09:08 +00:00
|
|
|
ControllerBadRequestError,
|
2020-10-02 06:37:50 +00:00
|
|
|
ControllerTimeoutError,
|
|
|
|
ControllerForbiddenError,
|
2021-04-13 09:16:50 +00:00
|
|
|
ControllerUnauthorizedError,
|
2022-04-07 08:21:47 +00:00
|
|
|
ComputeConflictError
|
2020-10-02 06:37:50 +00:00
|
|
|
)
|
|
|
|
|
2020-11-19 04:51:03 +00:00
|
|
|
from gns3server.api.routes import controller, index
|
|
|
|
from gns3server.api.routes.compute import compute_api
|
|
|
|
from gns3server.core import tasks
|
2020-10-02 06:37:50 +00:00
|
|
|
from gns3server.version import __version__
|
|
|
|
|
|
|
|
import logging
|
2021-04-13 09:16:50 +00:00
|
|
|
|
2020-10-02 06:37:50 +00:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2020-11-19 04:51:03 +00:00
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
def get_application() -> FastAPI:
|
|
|
|
|
|
|
|
application = FastAPI(
|
2024-12-19 09:54:11 +00:00
|
|
|
title="GNS3 controller API",
|
|
|
|
description="This page describes the public controller API for GNS3",
|
|
|
|
version="v3",
|
|
|
|
docs_url=None,
|
|
|
|
redoc_url=None
|
2020-12-07 06:22:36 +00:00
|
|
|
)
|
2020-10-02 06:37:50 +00:00
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
application.add_middleware(
|
2021-04-13 09:16:50 +00:00
|
|
|
CORSMiddleware,
|
2021-12-19 07:40:15 +00:00
|
|
|
allow_origin_regex=r"http(s)?://(localhost|127.0.0.1)(:\d+)?",
|
2021-04-13 09:16:50 +00:00
|
|
|
allow_credentials=True,
|
|
|
|
allow_methods=["*"],
|
|
|
|
allow_headers=["*"],
|
|
|
|
)
|
2020-12-07 06:22:36 +00:00
|
|
|
|
|
|
|
application.add_event_handler("startup", tasks.create_startup_handler(application))
|
|
|
|
application.add_event_handler("shutdown", tasks.create_shutdown_handler(application))
|
|
|
|
application.include_router(index.router, tags=["Index"])
|
|
|
|
application.include_router(controller.router, prefix="/v3")
|
2024-12-19 09:54:11 +00:00
|
|
|
application.mount("/static", StaticFiles(packages=[('gns3server', 'static')]), name="static")
|
2021-10-21 11:08:36 +00:00
|
|
|
application.mount("/v3/compute", compute_api, name="compute")
|
2020-12-07 06:22:36 +00:00
|
|
|
|
|
|
|
return application
|
|
|
|
|
|
|
|
|
|
|
|
app = get_application()
|
2020-10-02 06:37:50 +00:00
|
|
|
|
2021-04-17 09:03:20 +00:00
|
|
|
# Monkey Patch uvicorn signal handler to detect the application is shutting down
|
|
|
|
app.state.exiting = False
|
|
|
|
unicorn_exit_handler = UvicornServer.handle_exit
|
|
|
|
|
|
|
|
|
|
|
|
def handle_exit(*args, **kwargs):
|
|
|
|
app.state.exiting = True
|
|
|
|
unicorn_exit_handler(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
UvicornServer.handle_exit = handle_exit
|
|
|
|
|
2024-12-19 09:54:11 +00:00
|
|
|
# Configure self-hosting JavaScript and CSS for docs
|
|
|
|
@app.get("/docs", include_in_schema=False)
|
|
|
|
async def custom_swagger_ui_html():
|
|
|
|
return get_swagger_ui_html(
|
|
|
|
openapi_url=app.openapi_url,
|
|
|
|
title=app.title + " - Swagger UI",
|
|
|
|
oauth2_redirect_url=app.swagger_ui_oauth2_redirect_url,
|
|
|
|
swagger_js_url="/static/swagger-ui-bundle.js",
|
|
|
|
swagger_css_url="/static/swagger-ui.css",
|
2024-12-19 10:25:42 +00:00
|
|
|
swagger_favicon_url="/static/favicon.ico"
|
2024-12-19 09:54:11 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
@app.get(app.swagger_ui_oauth2_redirect_url, include_in_schema=False)
|
|
|
|
async def swagger_ui_redirect():
|
|
|
|
return get_swagger_ui_oauth2_redirect_html()
|
|
|
|
|
|
|
|
|
|
|
|
@app.get("/redoc", include_in_schema=False)
|
|
|
|
async def redoc_html():
|
|
|
|
return get_redoc_html(
|
|
|
|
openapi_url=app.openapi_url,
|
|
|
|
title=app.title + " - ReDoc",
|
|
|
|
redoc_js_url="/static/redoc.standalone.js",
|
2024-12-19 10:25:42 +00:00
|
|
|
redoc_favicon_url="/static/favicon.ico"
|
2024-12-19 09:54:11 +00:00
|
|
|
)
|
2020-10-02 06:37:50 +00:00
|
|
|
|
|
|
|
@app.exception_handler(ControllerError)
|
|
|
|
async def controller_error_handler(request: Request, exc: ControllerError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller error in {request.url.path} ({request.method}): {exc}")
|
2020-10-02 06:37:50 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_409_CONFLICT,
|
2020-10-02 06:37:50 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@app.exception_handler(ControllerTimeoutError)
|
|
|
|
async def controller_timeout_error_handler(request: Request, exc: ControllerTimeoutError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller timeout error in {request.url.path} ({request.method}): {exc}")
|
2020-10-02 06:37:50 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_408_REQUEST_TIMEOUT,
|
2020-10-02 06:37:50 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@app.exception_handler(ControllerUnauthorizedError)
|
|
|
|
async def controller_unauthorized_error_handler(request: Request, exc: ControllerUnauthorizedError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller unauthorized error in {request.url.path} ({request.method}): {exc}")
|
2020-10-02 06:37:50 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
2020-10-02 06:37:50 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@app.exception_handler(ControllerForbiddenError)
|
|
|
|
async def controller_forbidden_error_handler(request: Request, exc: ControllerForbiddenError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller forbidden error in {request.url.path} ({request.method}): {exc}")
|
2020-10-02 06:37:50 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_403_FORBIDDEN,
|
2020-10-02 06:37:50 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@app.exception_handler(ControllerNotFoundError)
|
|
|
|
async def controller_not_found_error_handler(request: Request, exc: ControllerNotFoundError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller not found error in {request.url.path} ({request.method}): {exc}")
|
2020-10-02 06:37:50 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_404_NOT_FOUND,
|
2020-10-02 06:37:50 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-12-02 08:09:08 +00:00
|
|
|
@app.exception_handler(ControllerBadRequestError)
|
|
|
|
async def controller_bad_request_error_handler(request: Request, exc: ControllerBadRequestError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller bad request error in {request.url.path} ({request.method}): {exc}")
|
2020-12-02 08:09:08 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
2020-12-02 08:09:08 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-04-07 08:21:47 +00:00
|
|
|
@app.exception_handler(ComputeConflictError)
|
|
|
|
async def compute_conflict_error_handler(request: Request, exc: ComputeConflictError):
|
|
|
|
log.error(f"Controller received error from compute for request '{exc.url()}': {exc}")
|
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_409_CONFLICT,
|
2022-04-07 08:21:47 +00:00
|
|
|
content={"message": str(exc)},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-11-02 07:38:25 +00:00
|
|
|
# make sure the content key is "message", not "detail" per default
|
2021-12-24 02:35:39 +00:00
|
|
|
@app.exception_handler(HTTPException)
|
|
|
|
async def http_exception_handler(request: Request, exc: HTTPException):
|
2020-11-02 07:38:25 +00:00
|
|
|
return JSONResponse(
|
|
|
|
status_code=exc.status_code,
|
|
|
|
content={"message": exc.detail},
|
2021-12-24 02:35:39 +00:00
|
|
|
headers=exc.headers
|
2020-11-02 07:38:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-05-16 09:07:17 +00:00
|
|
|
@app.exception_handler(SQLAlchemyError)
|
|
|
|
async def sqlalchemry_error_handler(request: Request, exc: SQLAlchemyError):
|
2023-08-17 07:36:50 +00:00
|
|
|
log.error(f"Controller database error in {request.url.path} ({request.method}): {exc}")
|
2021-05-16 09:07:17 +00:00
|
|
|
return JSONResponse(
|
2023-08-17 07:36:50 +00:00
|
|
|
status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,
|
2021-05-16 09:07:17 +00:00
|
|
|
content={"message": "Database error detected, please check logs to find details"},
|
|
|
|
)
|
|
|
|
|
2023-08-17 07:36:50 +00:00
|
|
|
|
|
|
|
@app.exception_handler(RequestValidationError)
|
|
|
|
async def validation_exception_handler(request: Request, exc: RequestValidationError):
|
|
|
|
log.error(f"Request validation error in {request.url.path} ({request.method}): {exc}")
|
|
|
|
return JSONResponse(
|
|
|
|
status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,
|
|
|
|
content={"message": str(exc)}
|
|
|
|
)
|
|
|
|
|
2022-06-01 08:31:59 +00:00
|
|
|
# FIXME: do not use this middleware since it creates issue when using StreamingResponse
|
|
|
|
# see https://starlette-context.readthedocs.io/en/latest/middleware.html#why-are-there-two-middlewares-that-do-the-same-thing
|
|
|
|
|
|
|
|
# @app.middleware("http")
|
|
|
|
# async def add_extra_headers(request: Request, call_next):
|
|
|
|
# start_time = time.time()
|
|
|
|
# response = await call_next(request)
|
|
|
|
# process_time = time.time() - start_time
|
|
|
|
# response.headers["X-Process-Time"] = str(process_time)
|
|
|
|
# response.headers["X-GNS3-Server-Version"] = f"{__version__}"
|
|
|
|
# return response
|