use FastAPI to build up a HTTP restful server. try followings:
- baseline: not go through different process, decode and draw frame in same process.
- file upload and download: use http multipart file upload and download. all frame data go through RAM not read/write Harddisk.
- frame rgb data encode and decode: convert to base64 or hex string. post the string to server, echo back then decode and draw frame.
- frame jpg encode and decode: use jpeg encode and decode then convert to base64 string. with jpeg compress and decompress which will consume CPU, but data size is much less than rgb.
pip install -r requirements_rest.txt
# start rest api server (fastapi)
python rest_server.py
# create anothor terminal and run rest client
# need desktop support, cv windows will pop up and show frame rate
pyhthon rest_client.py
manual change following variable in code to test different branch of rest_client.py
upload = True # test file upload and download
totext = False # baseline, direct draw frame
totext = True # echo image from fastapi server
rgb = True # use orginal rgb frame or compress to jpg
hex = True # use hex string or base64 string
use gRPC to send frame between server and client.
pip install -r requirements_grpc.txt
# generate python source file from proto file
python -m grpc_tools.protoc --python_out=. --grpc_python_out=. -I. ./protobuf/api.proto
# start rest api server (fastapi)
python grpc_server.py
# create anothor terminal and run rest client
# need desktop support, cv windows will pop up and show frame rate
pyhthon grpc_client.py
use ZMQ to send frame between server and client.
pip install -r requirements_zmq.txt
# start zmq server (ipc on linux, localhost on windows)
python zmq_server.py
# create anothor terminal and run zmq client
pyhthon zmq_client.py
- baseline: 105 fps
- zmq: 50 fps
- gRPC: 25 fps
- base64 encode and decode: 4.4 fps (pybase64)
- base64 encode and decode: 4.1 fps (build-in base64)
- hex encode and decode: 4.0 fps
- jpeg encode and decode: 8.2 fps
- file upload and download: 3.5 fps
- ZMQ is the fastest way, it just use socket to send binary data. (via Unix Domain Socket). But you need process the data structure yourself (RGB shape, etc.).
- gRPC is much faster than RestServer. gPRC go through binary data transfer, HTTP Rest is going through string transfer and HTTP multipart upload will NOT convert to base64.
- Python build-in base64 is super slow. pybase64 is a little faster. Both is slow, should try a C implementation base64 module. maybe due to python cannot run on multi core well.
- Going through jpeg compress then base64 is faster than orgianl rgb to base64. It should due to less data need to convert to base64, though the image quality will drop due to jpeg compress.
- HTTP upload and download is slower even than base64 encode and decode.
- With huge data trasmission, localhost (via virtual network driver) and uds (Unix Domain Socket, which not going through network stack) don't have too much different. Small packet and much requent data transmission should see the difference.
- None of above method is good enough for video transmission between processes.
Don't give up, HooTooVV! Explored and tried other methods, fianlly found following:
Use python shared memory to share frame data between processes and use ZMQ to send frame properties. (gPRC or Restful should be simmilar)
- shared memeory + zmq: 95 fps
- Python shared memory client/server process can cross python Interpreter. (Difference ptyhon version)
- Python shared memory is the fastest way to share video frame between processes.
- So far the only limitation is both side should be written in Python.
This way works. :).