You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
6.6 KiB
163 lines
6.6 KiB
from fastapi import FastAPI, Response, BackgroundTasks
|
|
from fastapi.responses import StreamingResponse, FileResponse
|
|
import uvicorn
|
|
import traceback
|
|
|
|
from config_parser import Config as ConfigParser
|
|
from config_parser import TranscodeStatus, TranscodeTools
|
|
from nvr_core import NVR
|
|
from nvr_types import File
|
|
|
|
class Server:
|
|
app: FastAPI = FastAPI()
|
|
config: ConfigParser = ConfigParser()
|
|
|
|
def __init__(self):
|
|
self.setup_events()
|
|
self.setup_routers()
|
|
|
|
def setup_events(self):
|
|
@self.app.on_event('startup')
|
|
async def on_startup():
|
|
print("i am alive")
|
|
|
|
def setup_routers(self):
|
|
@self.app.get("/api", status_code=200)
|
|
async def getRecorders(response: Response):
|
|
try:
|
|
return {"ok":True, "data":self.config.getRecorders()}
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return {"ok":False, "error":e}
|
|
|
|
@self.app.get("/api/channels/{recorder_index}", status_code=200)
|
|
async def getRecorder(response: Response, recorder_index:int):
|
|
try:
|
|
nvr:NVR = self.config.getRecorder(recorder_index).nvr
|
|
await nvr.login()
|
|
channels = await nvr.channels()
|
|
return {"ok":True, "data":channels}
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return {"ok":False, "error":e}
|
|
finally:
|
|
nvr.logout()
|
|
|
|
@self.app.get("/api/history/{recorder_index}/{channel}/{stream}")
|
|
async def getHistory(response: Response, recorder_index:int, channel: int, stream: int, start_date:str = None, end_date:str = None):
|
|
try:
|
|
nvr:NVR = self.config.getRecorder(recorder_index).nvr
|
|
await nvr.login()
|
|
return {"ok":True, "data":[f async for f in nvr.files(channel, start_date, end_date, stype=stream, json=True)]}
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return {"ok":False, "error":e}
|
|
finally:
|
|
nvr.logout()
|
|
|
|
@self.app.get("/api/snapshot/{recorder_index}/{channel}")
|
|
async def getSnapshot(response: Response, recorder_index:int, channel: int):
|
|
try:
|
|
nvr:NVR = self.config.getRecorder(recorder_index).nvr
|
|
await nvr.login()
|
|
async def image():
|
|
async for b in await nvr.client.snapshot(channel):
|
|
yield b
|
|
return StreamingResponse(image(), media_type="image/jpg")
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return b""
|
|
finally:
|
|
nvr.logout()
|
|
|
|
@self.app.get("/api/file/{recorder_index}")
|
|
async def getFile(response: Response, recorder_index:int, b64:str, background_tasks: BackgroundTasks):
|
|
try:
|
|
if len(b64) == 0:
|
|
response.status_code = 404
|
|
return ""
|
|
nvr:NVR = self.config.getRecorder(recorder_index).nvr
|
|
await nvr.login()
|
|
nvr.client.debug()
|
|
file: File = File.from_b64(b64 + "==")
|
|
print("open")
|
|
|
|
async def after():
|
|
try:
|
|
await nvr.client.busy.release()
|
|
except:
|
|
print("Already released")
|
|
nvr.logout()
|
|
|
|
background_tasks.add_task(after)
|
|
headers = {}
|
|
headers.update({"Content-Length":str(file.size)})
|
|
headers.update({"Content-Disposition": f'attachment; filename="{file.filename_cleared}.{file.type}"'})
|
|
return StreamingResponse(nvr.stream_file(file), media_type="application/octet-stream", headers=headers)
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return {"ok":False, "error":e}
|
|
|
|
@self.app.get("/api/transcode/status/{recorder_index}")
|
|
async def getTranscodeStatus(response: Response, recorder_index:int, b64:str, background_tasks: BackgroundTasks):
|
|
try:
|
|
if len(b64) == 0:
|
|
response.status_code = 404
|
|
return ""
|
|
|
|
if b64 in self.config.transcode_tools.statuses:
|
|
return {"ok":True, "data":self.config.transcode_tools.statuses[b64]}
|
|
|
|
nvr:NVR = self.config.getRecorder(recorder_index).nvr
|
|
await nvr.login()
|
|
nvr.client.debug()
|
|
file: File = File.from_b64(b64 + "==")
|
|
|
|
self.config.transcode_tools.statuses[b64] = TranscodeStatus(b64)
|
|
background_tasks.add_task(self.config.transcode_tools.processing_safe, status = self.config.transcode_tools.statuses[b64], file = file, nvr = nvr)
|
|
return {"ok":True, "data":self.config.transcode_tools.statuses[b64]}
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return {"ok":False, "error":e}
|
|
|
|
@self.app.get("/api/transcode/download")
|
|
async def getTranscodeDownload(response: Response, b64:str):
|
|
try:
|
|
if len(b64) == 0:
|
|
response.status_code = 404
|
|
return ""
|
|
|
|
if not b64 in self.config.transcode_tools.statuses:
|
|
response.status_code = 404
|
|
return ""
|
|
|
|
if self.config.transcode_tools.statuses[b64].done:
|
|
headers = {}
|
|
headers.update({"Content-Length":str(self.config.transcode_tools.statuses[b64].outSize)})
|
|
headers.update({"Content-Disposition": f'attachment; filename="{self.config.transcode_tools.statuses[b64].outName}"'})
|
|
return StreamingResponse(self.config.transcode_tools.statuses[b64].generate_bytes(), media_type="application/octet-stream", headers=headers)
|
|
else:
|
|
response.status_code = 429
|
|
return ""
|
|
|
|
except Exception as e:
|
|
traceback.print_exc()
|
|
response.status_code = 400
|
|
return {"ok":False, "error":e}
|
|
|
|
|
|
def run(self):
|
|
uvicorn.run(
|
|
self.app,
|
|
host=self.config.listen_address,
|
|
port=self.config.listen_port,
|
|
)
|
|
|
|
if __name__ == "__main__":
|
|
Server().run()
|