morphbr@453: import pygst morphbr@453: pygst.require("0.10") morphbr@453: import gst morphbr@453: import gobject renatofilho@483: import socket morphbr@453: morphbr@453: class Media: renatofilho@483: class StreamData: renatofilho@483: stream_count = 0 renatofilho@483: renatofilho@483: def __init__ (self, pipe, abin, vbin, sink): renatofilho@483: renatofilho@483: self.stream_count += 1 renatofilho@483: self.Id = self.stream_count renatofilho@483: self.Pipe = pipe renatofilho@483: self.Abin = abin renatofilho@483: self.Vbin = vbin renatofilho@483: self.Sink = sink renatofilho@483: self.Loop = gobject.MainLoop() renatofilho@483: self.ACaps = "" renatofilho@483: self.VCaps = "" renatofilho@483: self.Ready = False renatofilho@483: morphbr@453: morphbr@453: def __init__(self, config): morphbr@453: # set gstreamer basic options morphbr@453: self.config = config renatofilho@483: self.pipe = None renatofilho@483: self.streams = [] renatofilho@483: self.socket = None renatofilho@483: self.connection = None renatofilho@483: self.addr = None morphbr@453: morphbr@466: renatofilho@483: def setup(self, filename, mux, vcodec, vbitrate, renatofilho@483: fps, acodec, abitrate, width, height, port, options): morphbr@453: morphbr@453: ## Pipelines renatofilho@483: self.pipe = gst.Pipeline () renatofilho@483: uri = "file://" + filename renatofilho@483: print "Opening Uri:" + uri renatofilho@483: src = gst.element_make_from_uri (gst.URI_SRC, uri, "src") renatofilho@483: if (src is None): renatofilho@483: return None renatofilho@483: renatofilho@483: decode = gst.element_factory_make ("decodebin", "decode") renatofilho@483: if (decode is None): renatofilho@483: return None morphbr@453: renatofilho@483: mux = gst.element_factory_make ("avimux", "mux") renatofilho@483: if (mux is None): renatofilho@483: return None morphbr@453: renatofilho@483: sink = gst.element_factory_make ("fdsink", "sink") renatofilho@483: if (sink is None): renatofilho@483: return None morphbr@453: renatofilho@483: #Create socket renatofilho@483: self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) renatofilho@483: self.socket.bind(('', int (port))) renatofilho@483: renatofilho@483: #video encode renatofilho@483: #queue ! videoscale ! video/x-raw-yuv,width=240,height=144 ! videorate ! ffenc_h263p bitrate=256000 me-method=2 ! rtph263ppay ! udpsink host=224.0.0.1 port=5000 renatofilho@483: vbin = gst.Bin () renatofilho@483: vqueue = gst.element_factory_make ("queue", "vqueue") renatofilho@483: vscale = gst.element_factory_make ("videoscale", "vscale") renatofilho@483: vrate = gst.element_factory_make ("videorate", "vrate") renatofilho@483: vencode = gst.element_factory_make ("ffenc_h263p", "vencode") morphbr@453: renatofilho@483: if (None in [vbin, vqueue, vscale, vrate, vencode]): renatofilho@483: print "Fail to create video encode elements." renatofilho@483: return None morphbr@453: renatofilho@483: vscale_pad = vscale.get_pad("sink") renatofilho@483: if (vscale_pad is None): renatofilho@483: print "Fail to get vscale sink pad." renatofilho@483: return None morphbr@453: renatofilho@483: vscale_caps = gst.caps_from_string ("video/x-raw-yuv, width=%s, height=%s" % (width, height)) renatofilho@483: if (vscale_caps is None): renatofilho@483: print "Fail to create video caps" renatofilho@483: return None morphbr@453: renatofilho@483: if (not vscale_pad.set_caps (vscale_caps)): renatofilho@483: print "Fail to set video output caps" renatofilho@483: return None renatofilho@483: renatofilho@483: vbin.add (vqueue, vscale, vrate, vencode) renatofilho@483: if (not gst.element_link_many (vqueue, vscale, vrate, vencode)): renatofilho@483: print "Fail to link video elements" renatofilho@483: return None renatofilho@483: renatofilho@483: vbin.add_pad (gst.GhostPad ("sink", vqueue.get_pad ("sink"))) renatofilho@483: vbin.add_pad (gst.GhostPad ("src", vencode.get_pad ("src"))) renatofilho@483: renatofilho@483: #audio encode renatofilho@483: #audio/x-raw-int ! queue ! audioconvert ! faac ! rtpmp4gpay ! udpsink name=upd_audio host=224.0.0.1 port=5002 renatofilho@483: abin = gst.Bin () renatofilho@483: aqueue = gst.element_factory_make ("queue", "vqueue") renatofilho@483: aconvert = gst.element_factory_make ("audioconvert", "aconvert") renatofilho@483: aencode = gst.element_factory_make ("ffenc_ac3", "aencode") renatofilho@483: renatofilho@483: if (None in [abin, aqueue, aconvert, aencode]): renatofilho@483: print "Fail to create video encode elements." renatofilho@483: return None renatofilho@483: renatofilho@483: abin.add (aqueue, aconvert, aencode) renatofilho@483: if (not gst.element_link_many (aqueue, aconvert, aencode)): renatofilho@483: print "Fail to link video elements" renatofilho@483: return None renatofilho@483: renatofilho@483: abin.add_pad (gst.GhostPad ("sink", aqueue.get_pad ("sink"))) renatofilho@483: abin.add_pad (gst.GhostPad ("src", aencode.get_pad ("src"))) renatofilho@483: renatofilho@483: #Finish Pipeline renatofilho@483: renatofilho@483: self.pipe.add (src, decode, abin, vbin, mux, sink) renatofilho@483: gst.element_link_many (src, decode) renatofilho@483: gst.element_link_many (mux, sink) renatofilho@483: renatofilho@483: #Linking decode with mux renatofilho@483: mux_audio = mux.get_pad ("audio_0") renatofilho@483: mux_video = mux.get_pad ("video_0") renatofilho@483: renatofilho@483: audio_pad = abin.get_pad ("src") renatofilho@483: video_pad = vbin.get_pad ("src") renatofilho@483: renatofilho@483: if (audio_pad.link (mux_audio) != gst.PAD_LINK_OK): renatofilho@483: print "Fail to link audio with mux" renatofilho@483: return None renatofilho@483: renatofilho@483: if (video_pad.link (mux_video) != gst.PAD_LINK_OK): renatofilho@483: print "Fail to link audio with mux" renatofilho@483: return None renatofilho@483: renatofilho@483: stream_data = self.StreamData (self.pipe, abin, vbin, sink) renatofilho@483: renatofilho@483: bus = self.pipe.get_bus() renatofilho@483: bus.add_signal_watch() renatofilho@483: bus.connect("message", self.__on_bus_message, stream_data) renatofilho@483: renatofilho@483: decode.connect("new-decoded-pad", self.__on_decode_new_pad, stream_data) renatofilho@483: decode.connect("unknown-type", self.__on_decode_unknown_type, stream_data) renatofilho@483: renatofilho@483: renatofilho@483: self.pipe.set_state (gst.STATE_PAUSED) renatofilho@483: print "Running Pipe" renatofilho@483: stream_data.Loop.run () renatofilho@483: print "End run" renatofilho@483: renatofilho@483: a_caps = stream_data.ACaps renatofilho@483: v_caps = stream_data.VCaps renatofilho@483: stream_id = stream_data.Id renatofilho@483: renatofilho@483: self.streams.append (stream_data) morphbr@453: morphbr@453: def play(self): morphbr@453: morphbr@453: print "Trying to play pipeline: %s" % self.pipe morphbr@453: try: renatofilho@483: if (self.pipe): renatofilho@483: print "Waiting for connection" renatofilho@483: self.socket.listen(1) renatofilho@483: print "Connection Requested" renatofilho@483: #Create socket renatofilho@483: self.connection, self.addr = self.socket.accept () renatofilho@483: renatofilho@483: stream_data = self.streams[0] renatofilho@483: stream_data.Sink.set_property ("fd", self.connection.fileno()); renatofilho@483: print "Connected" renatofilho@483: renatofilho@483: self.pipe.set_state(gst.STATE_PLAYING) morphbr@453: except gobject.GError, e: morphbr@453: print "Error: " + str(e) morphbr@453: morphbr@453: morphbr@453: def stop(self): morphbr@453: morphbr@453: print "Trying to stop pipeline: %s" % self.pipe morphbr@453: try: morphbr@453: if (self.pipeline): renatofilho@483: self.connection.close () morphbr@453: self.pipeline.set_state(gst.STATE_NULL) morphbr@453: except gobject.GError, e: morphbr@453: print "Error: " + str(e) morphbr@453: renatofilho@483: def __on_bus_message (self, bus, message, stream_data): renatofilho@483: renatofilho@483: t = message.type renatofilho@483: if (t == gst.MESSAGE_STATE_CHANGED): renatofilho@483: oldstate = -1 renatofilho@483: newstate = -1 renatofilho@483: pending = -1 renatofilho@483: oldstate, newstate, pending = message.parse_state_changed () renatofilho@483: if ((oldstate == gst.STATE_READY) and \ renatofilho@483: (newstate == gst.STATE_PAUSED) and \ renatofilho@483: (pending == gst.STATE_VOID_PENDING) and \ renatofilho@483: (stream_data.Ready == False)): renatofilho@483: state_changed_status, current_state, pending_state = stream_data.Pipe.get_state () renatofilho@483: if ((current_state == gst.STATE_PAUSED) and \ renatofilho@483: (pending_state == gst.STATE_VOID_PENDING)): renatofilho@483: print "Pipe paused" renatofilho@483: stream_data.Loop.quit () renatofilho@483: stream_data.Ready = True renatofilho@483: elif (t == gst.MESSAGE_ERROR): renatofilho@483: err, debug = message.parse_error() renatofilho@483: print "Error: %s" % err, debug renatofilho@483: stream_data.Loop.quit () renatofilho@483: stream_data.Ready = False renatofilho@483: renatofilho@483: return True renatofilho@483: renatofilho@483: def __on_decode_unknown_type (self, decode, pad, caps, stream_data): renatofilho@483: renatofilho@483: print "Unknown Type" renatofilho@483: return None renatofilho@483: renatofilho@483: def __on_decode_new_pad (self, decode, pad, arg1, stream_data): renatofilho@483: renatofilho@483: caps = pad.get_caps().to_string() renatofilho@483: print "New pad " + caps renatofilho@483: if (caps.rfind ("audio") != -1): renatofilho@483: apad = stream_data.Abin.get_pad ("sink") renatofilho@483: if (pad.link (apad) != gst.PAD_LINK_OK): renatofilho@483: print "Error on link audio pad" renatofilho@483: return None renatofilho@483: elif (caps.rfind ("video") != -1): renatofilho@483: vpad = stream_data.Vbin.get_pad ("sink") renatofilho@483: if (pad.link (vpad) != gst.PAD_LINK_OK): renatofilho@483: print "Error on link video pad" renatofilho@483: return None renatofilho@483: else: renatofilho@483: print "Invalid caps" renatofilho@483: renatofilho@483: