aboutsummaryrefslogtreecommitdiff
path: root/example-scripts/gstreamer/ingest.py
blob: a9e06c475d18cde50eb662c1e794538fe906c239 (plain)
  1. #!/usr/bin/env python3
  2. # ingest.py
  3. """
  4. Source client for Voctomix.
  5. Features:
  6. Retrieves audio and video-caps config from core.
  7. Uses core's clock.
  8. Mix and match audio and video sources muxed into one streem.
  9. Can display video locally, including frame count and fps.
  10. Defaults to test audio and video sent to local core.
  11. """
  12. import argparse
  13. import gi
  14. import os
  15. import signal
  16. import socket
  17. import sys
  18. gi.require_version('Gst', '1.0')
  19. from gi.repository import Gst, GstNet, GObject
  20. # init GObject & Co. before importing local classes
  21. GObject.threads_init()
  22. Gst.init([])
  23. # this is to use the same code tha gui uses to get config from core
  24. sys.path.insert(0, '../..' )
  25. sys.path.insert(0, '.' )
  26. import voctogui.lib.connection as Connection
  27. def mk_video_src(args, videocaps):
  28. # make video soure part of pipeline
  29. video_device = "device={}".format(args.video_dev) \
  30. if args.video_dev else ""
  31. monitor = """tee name=t ! queue !
  32. videoconvert ! fpsdisplaysink sync=false
  33. t. ! queue !""" \
  34. if args.monitor else ""
  35. if args.video_source == 'dv':
  36. video_src = """
  37. dv1394src name=videosrc {video_device}!
  38. dvdemux name=demux !
  39. queue !
  40. dvdec !
  41. {monitor}
  42. deinterlace mode=1 !
  43. videoconvert !
  44. videorate !
  45. videoscale !
  46. """
  47. elif args.video_source == 'hdv':
  48. video_src = """
  49. hdv1394src {video_device} do-timestamp=true name=videosrc !
  50. tsdemux name=demux!
  51. queue !
  52. decodebin !
  53. {monitor}
  54. deinterlace mode=1 !
  55. videorate !
  56. videoscale !
  57. videoconvert !
  58. """
  59. elif args.video_source == 'hdmi2usb':
  60. # https://hdmi2usb.tv
  61. # Note: this code works with 720p
  62. video_src = """
  63. v4l2src {video_device} name=videosrc !
  64. queue !
  65. image/jpeg,width=1280,height=720 !
  66. jpegdec !
  67. {monitor}
  68. videoconvert !
  69. videorate !
  70. """
  71. elif args.video_source == 'ximage':
  72. video_src = """
  73. ximagesrc name=videosrc
  74. use-damage=false !
  75. {monitor}
  76. videoconvert !
  77. videorate !
  78. videoscale !
  79. """
  80. # startx=0 starty=0 endx=1919 endy=1079 !
  81. elif args.video_source == 'blackmagichdmi':
  82. video_src = """
  83. decklinkvideosrc mode=17 connection=2 !
  84. {monitor}
  85. videoconvert !
  86. videorate !
  87. videoscale !
  88. """
  89. elif args.video_source == 'test':
  90. video_src = """
  91. videotestsrc name=videosrc
  92. pattern=ball
  93. foreground-color=0x00ff0000 background-color=0x00440000 !
  94. {monitor}
  95. """
  96. video_src = video_src.format(
  97. video_device=video_device,
  98. monitor=monitor)
  99. video_src += videocaps + "!\n"
  100. return video_src
  101. def mk_audio_src(args, audiocaps):
  102. audio_device = "device={}".format(args.audio_dev) \
  103. if args.audio_dev else ""
  104. if args.audio_source in [ 'dv', 'hdv' ]:
  105. # this only works if video is from DV also.
  106. # or some gst source that gets demux ed
  107. audio_src = """
  108. demux. !
  109. audioconvert !
  110. """
  111. elif args.audio_source == 'pulse':
  112. audio_src = """
  113. pulsesrc {audio_device} name=audiosrc !
  114. """.format(audio_device=audio_device)
  115. elif args.audio_source == 'alsa':
  116. audio_src = """
  117. alsasrc {audio_device} name=audiosrc !
  118. """.format(audio_device=audio_device)
  119. elif args.audio_source == 'blackmagichdmi':
  120. audio_src = """
  121. decklinkaudiosrc !
  122. """
  123. elif args.audio_source == 'test':
  124. audio_src = """
  125. audiotestsrc name=audiosrc freq=330 !
  126. """
  127. audio_src += audiocaps + "!\n"
  128. return audio_src
  129. def mk_mux(args):
  130. mux = """
  131. mux.
  132. matroskamux name=mux !
  133. """
  134. return mux
  135. def mk_client(args):
  136. core_ip = socket.gethostbyname(args.host)
  137. client = """
  138. tcpclientsink host={host} port={port}
  139. """.format(host=core_ip, port=args.port)
  140. return client
  141. def mk_pipeline(args, server_caps):
  142. video_src = mk_video_src(args, server_caps['videocaps'])
  143. audio_src = mk_audio_src(args, server_caps['audiocaps'])
  144. mux = mk_mux(args)
  145. client = mk_client(args)
  146. pipeline = video_src + "mux.\n" + audio_src + mux + client
  147. # remove blank lines to make it more human readable
  148. pipeline = pipeline.replace("\n\n","\n")
  149. return pipeline
  150. def get_server_caps():
  151. # fetch config from server
  152. server_config = Connection.fetchServerConfig()
  153. server_caps = {'videocaps': server_config['mix']['videocaps'],
  154. 'audiocaps': server_config['mix']['audiocaps']}
  155. return server_caps
  156. def run_pipeline(pipeline, args):
  157. core_ip = socket.gethostbyname(args.host)
  158. clock = GstNet.NetClientClock.new('voctocore', core_ip, 9998, 0)
  159. print('obtained NetClientClock from host', clock)
  160. print('waiting for NetClientClock to sync…')
  161. clock.wait_for_sync(Gst.CLOCK_TIME_NONE)
  162. print('starting pipeline')
  163. senderPipeline = Gst.parse_launch(pipeline)
  164. senderPipeline.use_clock(clock)
  165. src = senderPipeline.get_by_name('src')
  166. def on_eos(self, bus, message):
  167. print('Received EOS-Signal')
  168. sys.exit(1)
  169. def on_error(self, bus, message):
  170. print('Received Error-Signal')
  171. (error, debug) = message.parse_error()
  172. print('Error-Details: #%u: %s' % (error.code, debug))
  173. sys.exit(1)
  174. # Binding End-of-Stream-Signal on Source-Pipeline
  175. senderPipeline.bus.add_signal_watch()
  176. senderPipeline.bus.connect("message::eos", on_eos)
  177. senderPipeline.bus.connect("message::error", on_error)
  178. print("playing")
  179. senderPipeline.set_state(Gst.State.PLAYING)
  180. mainloop = GObject.MainLoop()
  181. try:
  182. mainloop.run()
  183. except KeyboardInterrupt:
  184. print('Terminated via Ctrl-C')
  185. print('Shutting down...')
  186. senderPipeline.set_state(Gst.State.NULL)
  187. print('Done.')
  188. return
  189. def get_args():
  190. parser = argparse.ArgumentParser(
  191. description='''Vocto-ingest Client with Net-time support.
  192. Gst caps are retrieved from the server.
  193. Run without parameters: send test av to localhost:10000
  194. ''')
  195. parser.add_argument('-v', '--verbose', action='count', default=0,
  196. help="Also print INFO and DEBUG messages.")
  197. parser.add_argument( '--video-source', action='store',
  198. choices=[
  199. 'dv', 'hdv', 'hdmi2usb', 'blackmagichdmi',
  200. 'ximage',
  201. 'test', ],
  202. default='test',
  203. help="Where to get video from")
  204. parser.add_argument( '--video-dev', action='store',
  205. help="video device")
  206. parser.add_argument( '--audio-source', action='store',
  207. choices=['dv', 'alsa', 'pulse', 'blackmagichdmi', 'test'],
  208. default='test',
  209. help="Where to get audio from")
  210. parser.add_argument( '--audio-dev', action='store',
  211. default='hw:CARD=CODEC',
  212. help="for alsa/pulse, audio device")
  213. # maybe hw:1,0
  214. parser.add_argument( '--audio-delay', action='store',
  215. default='10',
  216. help="ms to delay audio")
  217. parser.add_argument('-m', '--monitor', action='store_true',
  218. help="fps display sink")
  219. parser.add_argument( '--host', action='store',
  220. default='localhost',
  221. help="hostname of vocto core")
  222. parser.add_argument( '--port', action='store',
  223. default='10000',
  224. help="port of vocto core")
  225. args = parser.parse_args()
  226. return args
  227. def main():
  228. args = get_args()
  229. core_ip = socket.gethostbyname(args.host)
  230. # establish a synchronus connection to server
  231. Connection.establish(core_ip)
  232. server_caps = get_server_caps()
  233. pipeline = mk_pipeline(args, server_caps)
  234. print(pipeline)
  235. run_pipeline(pipeline, args)
  236. if __name__ == '__main__':
  237. main()