add ffmpeger.py
This commit is contained in:
parent
a69bf7e6a0
commit
ec3ead528b
40
README.md
40
README.md
|
@ -1,9 +1,45 @@
|
|||
# utils
|
||||
|
||||
* [ffmpeger.py](https://git.hmp.today/pavel.muhortov/utils#ffmpeger-py)
|
||||
* [sendmail.py](https://git.hmp.today/pavel.muhortov/utils#sendmail-py)
|
||||
* [srchproc.py](https://git.hmp.today/pavel.muhortov/utils#srchproc-py)
|
||||
____
|
||||
## ffmpeger.py
|
||||
**Description:** FFmpeg management from Python
|
||||
**Dependencies:** Python 3 (tested version 3.9.5), installed or downloaded ffmpeg, srchproc.py in the same directory
|
||||
|
||||
| PARAMETERS | DESCRIPTION | DEFAULT|
|
||||
|-------------|-------------|--------|
|
||||
|**-s**, **--src**|sources urls|**REQUIRED**|
|
||||
|**[-h]**|print help and exit||
|
||||
|**[--preset]**|240p, 360p, 480p, 720p, 1080p, 1440p, 2160p|`None`|
|
||||
|**[--fps]**|frame per second encoding output|`None`|
|
||||
|**[--dst]**|destination url|`None`|
|
||||
|**[--ffpath]**|alternative path to bin|`None`|
|
||||
|**[--watchdog]**|detect ffmpeg freeze and terminate||
|
||||
|**[--sec]**|seconds to wait before the watchdog terminates|15|
|
||||
|**[--mono]**|detect ffmpeg running copy and terminate||
|
||||
|
||||
Example usage in cron with Python:
|
||||
```shell
|
||||
* * * * * /usr/bin/python3 ~/ffmpeger.py -s rtsp://user:pass@host:554/Streaming/Channels/101 --dst rtmp://a.rtmp.youtube.com/live2/YOUKEY --mono --watchdog --sec 30 >> /dev/null 2>&1
|
||||
```
|
||||
Example usage in terminal with make the script executable:
|
||||
```shell
|
||||
chmod u+x ./ffmpeger.py
|
||||
ffmpeger.py -s rtsp://user:pass@host:554/Streaming/Channels/101 --dst rtp://239.0.0.1:5554
|
||||
```
|
||||
Example usage in Python:
|
||||
```Python
|
||||
from ffmpeger import FFmpeg
|
||||
|
||||
FFmpeg.run(src='null, anull', preset='240p', fps=10)
|
||||
```
|
||||
____
|
||||
## sendmail.py
|
||||
**Description:** Sending email from Python
|
||||
**Dependencies:** Python 3 (tested version 3.9.5)
|
||||
|
||||
| PARAMETERS | DESCRIPTION | DEFAULT|
|
||||
|-------------|-------------|--------|
|
||||
|**-u**, **--user**|smtp valid user|**REQUIRED**|
|
||||
|
@ -38,6 +74,9 @@ print(log)
|
|||
```
|
||||
____
|
||||
## srchproc.py
|
||||
**Description:** Find a running process from Python
|
||||
**Dependencies:** Python 3 (tested version 3.9.5)
|
||||
|
||||
| PARAMETERS | DESCRIPTION | DEFAULT|
|
||||
|-------------|-------------|--------|
|
||||
|**[-h]**|print help and exit||
|
||||
|
@ -65,3 +104,4 @@ if processes:
|
|||
for process in processes:
|
||||
print(process)
|
||||
```
|
||||
|
||||
|
|
217
ffmpeger.py
Normal file
217
ffmpeger.py
Normal file
|
@ -0,0 +1,217 @@
|
|||
#!/usr/bin/env python3
|
||||
|
||||
|
||||
from multiprocessing import Process, Queue
|
||||
from os import path, environ
|
||||
from subprocess import Popen, PIPE, STDOUT
|
||||
from sys import platform
|
||||
from time import sleep
|
||||
from srchproc import Proc # or copy class Proc from file srchproc.py here for autonomy of this file
|
||||
|
||||
|
||||
class FFmpeg:
|
||||
"""
|
||||
FFmpeg management from Python
|
||||
"""
|
||||
@classmethod
|
||||
def run(cls, src: str, preset: str = None, fps: int = None, dst: str = None,
|
||||
ffpath: str = None, watchdog: bool = False, sec: int = 5, mono: bool = False):
|
||||
"""
|
||||
Running the installed ffmpeg
|
||||
:param src: sources urls (example: "rtsp://user:pass@host:554/Streaming/Channels/101, anull")
|
||||
:param preset: 240p, 360p, 480p, 720p, 1080p, 1440p, 2160p
|
||||
:param fps: frame per second encoding output
|
||||
:param dst: destination url (example: rtp://239.0.0.1:5554)
|
||||
:param ffpath: alternative path to bin (example: /usr/bin/ffmpeg)
|
||||
:param watchdog: detect ffmpeg freeze and terminate
|
||||
:param sec: seconds to wait before the watchdog terminates
|
||||
:param mono: detect ffmpeg running copy and terminate
|
||||
:return: None
|
||||
"""
|
||||
process = cls._bin(ffpath).split()+cls._src(src).split()+cls._preset(preset, fps).split()+cls._dst(dst).split()
|
||||
if mono and Proc.search(' '.join(process)):
|
||||
print('Process already exist, exit...')
|
||||
else:
|
||||
with Popen(process, stdout=PIPE, stderr=STDOUT) as proc:
|
||||
if watchdog:
|
||||
que = Queue()
|
||||
Process(target=cls._watchdog, args=(proc.pid, sec, que,), daemon=True).start()
|
||||
for line in proc.stdout:
|
||||
if not que:
|
||||
print(line, flush=True)
|
||||
else:
|
||||
que.put(line)
|
||||
exit()
|
||||
|
||||
@classmethod
|
||||
def _bin(cls, path_ffmpeg: str):
|
||||
"""
|
||||
Returns the path to the ffmpeg depending on the OS
|
||||
:param path_ffmpeg: alternative path to bin
|
||||
:return: path to ffmpeg
|
||||
"""
|
||||
faq = ('\n'
|
||||
'Main download page: https://ffmpeg.org/download.html\n'
|
||||
'\n'
|
||||
'Install on Linux (Debian):\n'
|
||||
'\tsudo apt install -y ffmpeg\n'
|
||||
'\tTarget: /usr/bin/ffmpeg\n'
|
||||
'\n'
|
||||
'Install on Windows:\n'
|
||||
'\tDownload and extract archive from: https://www.gyan.dev/ffmpeg/builds/ffmpeg-release-full.7z\n'
|
||||
'\tTarget: "%PROGRAMFILES%\\ffmpeg\\bin\\ffmpeg.exe"\n'
|
||||
'\n'
|
||||
'Install on MacOS:\n'
|
||||
'\tDownload and extract archive from: https://evermeet.cx/ffmpeg/\n'
|
||||
'\tTarget: /usr/bin/ffmpeg\n')
|
||||
if not path_ffmpeg:
|
||||
if platform.startswith('linux'):
|
||||
path_ffmpeg = '/usr/bin/ffmpeg'
|
||||
elif platform.startswith('win32'):
|
||||
path_ffmpeg = environ['PROGRAMFILES'] + "\\ffmpeg\\bin\\ffmpeg.exe"
|
||||
elif platform.startswith('darwin'):
|
||||
path_ffmpeg = '/usr/bin/ffmpeg'
|
||||
if path.exists(path_ffmpeg):
|
||||
return path_ffmpeg
|
||||
else:
|
||||
print('ON', platform, 'PLATFORM', 'not found ffmpeg', faq)
|
||||
return None
|
||||
|
||||
@classmethod
|
||||
def _src(cls, sources: str):
|
||||
"""
|
||||
Parsing sources into ffmpeg format
|
||||
:param sources: comma-separated list of sources in string format
|
||||
:return: ffmpeg format list of sources
|
||||
"""
|
||||
list_sources = []
|
||||
for src in sources.split(','):
|
||||
src = src.strip()
|
||||
if 'null' in src:
|
||||
src = ' '.join(['-f lavfi -i', src])
|
||||
elif 'rtsp' in src:
|
||||
src = ' '.join(['-rtsp_transport tcp -i', src])
|
||||
else:
|
||||
src = ' '.join(['-i', src])
|
||||
list_sources.append(src)
|
||||
return ' '.join(list_sources)
|
||||
|
||||
@classmethod
|
||||
def _preset(cls, choice: str, fps: int):
|
||||
"""
|
||||
Parsing preset into ffmpeg format
|
||||
:param choice: preset selection
|
||||
:param fps: frame per second encoding output
|
||||
:return: ffmpeg format encoding parameters
|
||||
"""
|
||||
tune = '-tune zerolatency'
|
||||
video = '-c:v copy'
|
||||
audio = '-c:a aac -b:a 128k'
|
||||
width, height, kbps = None, None, None
|
||||
if choice:
|
||||
if choice == '240p':
|
||||
width, height, kbps = 426, 240, 480
|
||||
if choice == '360p':
|
||||
width, height, kbps = 640, 360, 720
|
||||
if choice == '480p':
|
||||
width, height, kbps = 854, 480, 1920
|
||||
if choice == '720p':
|
||||
width, height, kbps = 1280, 720, 3960
|
||||
if choice == '1080p':
|
||||
width, height, kbps = 1920, 1080, 5940
|
||||
if choice == '1440p':
|
||||
width, height, kbps = 2560, 1440, 12960
|
||||
if choice == '2160p':
|
||||
width, height, kbps = 3840, 2160, 32400
|
||||
if width and height and kbps:
|
||||
video = ''.join(['-vf scale=', str(width), ':', str(height), ',setsar=1:1'])
|
||||
video = ' '.join([video, '-c:v libx264 -pix_fmt yuv420p -preset ultrafast'])
|
||||
if fps:
|
||||
video = ' '.join([video, '-r', str(fps), '-g', str(fps * 2)])
|
||||
video = ' '.join([video, '-b:v', str(kbps) + 'k'])
|
||||
return ' '.join([tune, video, audio])
|
||||
|
||||
@classmethod
|
||||
def _dst(cls, destination: str):
|
||||
"""
|
||||
Parsing destination into ffmpeg format
|
||||
:param destination:
|
||||
:return: ffmpeg format destination
|
||||
"""
|
||||
container = '-f null'
|
||||
stdout = '-v debug' # '-nostdin -nostats' # '-report'
|
||||
if destination:
|
||||
if 'rtmp' in destination:
|
||||
container = '-f flv'
|
||||
elif "rtp" in destination:
|
||||
container = '-f rtp_mpegts'
|
||||
else:
|
||||
destination = '-'
|
||||
return ' '.join([container, destination, stdout])
|
||||
|
||||
@classmethod
|
||||
def _watchdog(cls, pid: int, sec: int = 5, que: Queue = None):
|
||||
"""
|
||||
If no data arrives in the queue, kill the process
|
||||
:param pid: process ID
|
||||
:param sec: seconds to wait for data
|
||||
:param que: queue pointer
|
||||
:return: None
|
||||
"""
|
||||
if que:
|
||||
while True:
|
||||
while not que.empty():
|
||||
print(que.get())
|
||||
sleep(sec)
|
||||
if que.empty():
|
||||
cls._kill(pid)
|
||||
print('exit by watchdog')
|
||||
break
|
||||
exit()
|
||||
|
||||
@classmethod
|
||||
def _kill(cls, pid: int):
|
||||
"""
|
||||
Kill the process by means of the OS
|
||||
:param pid: process ID
|
||||
:return: None
|
||||
"""
|
||||
if platform.startswith('linux'):
|
||||
Popen(['kill', '-s', 'SIGKILL', str(pid)])
|
||||
elif platform.startswith('win32'):
|
||||
Popen(['taskkill', '/PID', str(pid), '/F'])
|
||||
elif platform.startswith('darwin'):
|
||||
Popen(['kill', '-s', 'SIGKILL', str(pid)])
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
from argparse import ArgumentParser
|
||||
|
||||
args = ArgumentParser(
|
||||
prog='FFmpeger',
|
||||
description='FFmpeg management from Python',
|
||||
epilog='Dependencies: '
|
||||
'Python 3 (tested version 3.9.5), '
|
||||
'installed or downloaded ffmpeg, '
|
||||
'srchproc.py in the same directory'
|
||||
)
|
||||
args.add_argument('-s', '--src', type=str, required=True,
|
||||
help='sources urls (example: "rtsp://user:pass@host:554/Streaming/Channels/101, anull")')
|
||||
args.add_argument('--preset', type=str, default=None, required=False,
|
||||
help='240p, 360p, 480p, 720p, 1080p, 1440p, 2160p')
|
||||
args.add_argument('--fps', type=int, default=None, required=False,
|
||||
help='frame per second encoding output')
|
||||
args.add_argument('--dst', type=str, default=None, required=False,
|
||||
help='destination url (example: rtp://239.0.0.1:5554)')
|
||||
args.add_argument('--ffpath', type=str, default=None, required=False,
|
||||
help='alternative path to bin (example: /usr/bin/ffmpeg)')
|
||||
args.add_argument('--watchdog', action='store_true', required=False,
|
||||
help='detect ffmpeg freeze and terminate')
|
||||
args.add_argument('--sec', type=int, default=15, required=False,
|
||||
help='seconds to wait before the watchdog terminates')
|
||||
args.add_argument('--mono', action='store_true', required=False,
|
||||
help='detect ffmpeg running copy and terminate')
|
||||
args = vars(args.parse_args())
|
||||
|
||||
FFmpeg.run(src=args['src'], preset=args['preset'], fps=args['fps'], dst=args['dst'],
|
||||
ffpath=args['ffpath'], watchdog=args['watchdog'], sec=args['sec'], mono=args['mono'])
|
Loading…
Reference in New Issue
Block a user