|
|
@ -1,20 +1,24 @@ |
|
|
|
from fastapi import FastAPI, Response, BackgroundTasks, Header |
|
|
|
from fastapi import FastAPI, Response, BackgroundTasks, Header, Request |
|
|
|
from fastapi.responses import StreamingResponse, FileResponse |
|
|
|
from fastapi.security import HTTPBasic, HTTPBasicCredentials |
|
|
|
from fastapi.exceptions import HTTPException |
|
|
|
import uvicorn |
|
|
|
import traceback |
|
|
|
import aiofiles |
|
|
|
|
|
|
|
from config_parser import Config as ConfigParser |
|
|
|
from config_parser import TranscodeStatus, TranscodeTools, Go2Rtc, NeedNVR |
|
|
|
from global_funcs import create_logger |
|
|
|
from global_funcs import create_logger, hexed |
|
|
|
from nvr_core import NVR |
|
|
|
from nvr_types import File |
|
|
|
import os |
|
|
|
import time |
|
|
|
|
|
|
|
class Server: |
|
|
|
app: FastAPI = FastAPI() |
|
|
|
config: ConfigParser = ConfigParser() |
|
|
|
go2rtc: Go2Rtc = Go2Rtc() |
|
|
|
security = HTTPBasic() |
|
|
|
|
|
|
|
API_BASE_REF = "/api/dvrip" |
|
|
|
|
|
|
@ -22,6 +26,7 @@ class Server: |
|
|
|
self.logger = create_logger(Server.__name__) |
|
|
|
self.setup_events() |
|
|
|
self.setup_routers() |
|
|
|
self.setup_middleware() |
|
|
|
|
|
|
|
def setup_events(self): |
|
|
|
@self.app.on_event('startup') |
|
|
@ -35,7 +40,34 @@ class Server: |
|
|
|
self.logger.info(f"{self.config.recorders[i]} channels count: {self.config.recorders[i].channels}") |
|
|
|
await self.go2rtc.start_go2rtc(self.config.recorders) |
|
|
|
|
|
|
|
def setup_middleware(self): |
|
|
|
if len(self.config.auth) != 0: |
|
|
|
self.logger.info("auth is enabled") |
|
|
|
|
|
|
|
@self.app.middleware("http") |
|
|
|
async def auth_handler(request: Request, call_next): |
|
|
|
try: |
|
|
|
s = await self.security(request) |
|
|
|
if s.username not in self.config.auth.mapping or hexed(s.password) != self.config.auth.mapping[s.username]: |
|
|
|
raise HTTPException(status_code=401, headers={"WWW-Authenticate": "Basic"}) |
|
|
|
except HTTPException as e: |
|
|
|
return Response(status_code=e.status_code, headers=e.headers) |
|
|
|
|
|
|
|
return await call_next(request) |
|
|
|
|
|
|
|
@self.app.get(self.API_BASE_REF + "/logout") |
|
|
|
async def logout(request: Request): |
|
|
|
request.scope.update(headers={"Authorization":"b"}) |
|
|
|
return Response(status_code=401) |
|
|
|
|
|
|
|
else: |
|
|
|
self.logger.warn("auth is disabled") |
|
|
|
|
|
|
|
def setup_routers(self): |
|
|
|
@self.app.get(self.API_BASE_REF + "/ping", status_code=200) |
|
|
|
async def getPing(): |
|
|
|
return {"pong":time.time()} |
|
|
|
|
|
|
|
@self.app.get(self.API_BASE_REF, status_code=200) |
|
|
|
async def getRecorders(response: Response): |
|
|
|
try: |
|
|
|