Pulling a video stream
Introduction
The following figure show the theory of mpegCoder.MpegClient
. Assuming that we have a video server from the remote side, the real-time stream is pushed continuously. Even we do not read the online stream, the data flow would not wait for reading. Therefore, we design the following two-thread workflow.
When connecting to the remote server, MpegClient
would create a sub-thread ("writer" in the figure). The writer thread would work as a backend service, and keep accepting frames from the remote side, even if we do not read the frames. The accepted frames are stored in a circular buffer (In the figure, the buffer size is 12). There are two cursors maintained by the writer and the reader respectively (shown as the arrows connected to the threads in the figure). The writting cursor is kept stepping each time a new frame is received.
The reading events would be triggered by the Python-C-API. When a new reading event comes from the main thread, the reader would lock the current position of the reading cursor, and read several frames from the buffer. After the reading results are collected, the lock would be released, and the reading cursor will be reset to the end of the read frames. When the writer is writing a new frame, the current written position will also be locked by the writer. A locked position would not be updated. For example, during the reading events, if the writting cursor moves to the locked position, the writer will wait until the reading is finished. Because the reading events are merely data-collecting operations, in most cases the reading events would not block the writer. If the writer is blocked for too long, the demuxing of the online stream may fail. So we recommend users to set a rational buffer size. For example, if we always read 5 frames each time, the buffer size is recommended to be double of the reading size, i.e. 10.
Codes
To test the following codes, we recommend users to use VLC or FFMpeg to push a remote stream, because the stream pushing without encoding is not supported by mpegCoder
currently. Using VLC or FFMpeg to serve the stream will occupy less system resources.
The following example codes would scale the remote frame to 480x360, and resample the frame rate to 5 FPS. The reading size and the buffer size are 5
and 12
respectively.
import os, sysimport timeimport mpegCodermpegCoder.setGlobal(dumpLevel=2) # show full log.if __name__ == '__main__': d = mpegCoder.MpegClient() # create the handle. d.setParameter(widthDst=480, heightDst=360, dstFrameRate=(5,1), readSize=5, cacheSize=12) # do basic settings. success = d.FFmpegSetup('rtsp://localhost:8554/video') # connect with the server. print(d) if not success: # exit the program if the server is not available. You could delete this checking and see what will happen. exit() d.start() # start the sub-thread for demuxing the stream. time.sleep(5) # wait for getting some frames. print('Get slept') p = d.ExtractFrame() # extract some frames from current cache. print(p.shape) # show information of extracted frames. for i in range(10): # wait for 50 seconds. time.sleep(5) p = d.ExtractFrame() # extract some frames from current cache. d.terminate() # shut down the current sub-thread. You could call start() and let it restart. d.clear() # but here we would like to clear the handle and exit.
After configuring the client, the codes contain the following key steps:
The
MpegClient.FFmpegSetup()
accepts a video stream address. The stream type would be detected from the protocol automacially. Currently, we supporthttp
,ftp
,sftp
,rtsp
, andrtmp
. Note that onlyrtsp
andrtmp
should be used for analyzing the real-time stream. Thehttp
,ftp
andsftp
protocols are mostly used for data transfer. This method will launch a connect to the remote server.When
MpegClient.start()
is called, the sub-thread "writer" will be created.Using
MpegClient.ExtractFrame()
to get the real-time data. The returned frame number is given byreadSize
during the configuration. However, user could override the configurtion by using an argument, for example,ExtractFrame(4)
would force the reader to read 4 frames.If the remote stream is closed,
d.ExtractFrame()
would returnNone
. However, user would terminate the client in any time. The methodMpegClient.terminate()
would stop the writing thread. But the connection would not be aborted untilMpegClient.clear()
is called.
Examples on Github
On Github, we provide the above example as a single branch.
In addition, we provide another example. This example is a simple video stream player based on PyQt5
and mpegCoder
.