Testing a Membrane Bin used in a WebRTC Engine Endpoint
I'm trying to test a membrane bin that I use in WebRTC engine endpoint. Though I'm having trouble getting my test setup properly.
I was trying to approach is this way:
1. Start my
Membrane.Testing.Pipeline
with a simple spec, just the conversation bin
2. Send a {:new_tracks} event to the bin, simulating what the Membrane.WebRTC.Engine does when a new bin is assed as an endpoint
3. Then update the spec to connect the Membrane.Testing.DynamicSource
and Membrane.Testing.Sink
But I'm never tripping the handle_pad_added
callback in my bin, which leads me to believe I'm doing something wrong.
I'm using the Membrane.Testing.Pipeline
so I can access the Membrane.Testing.Assertions
and assert that my conversation bin is emitting the right events, etc.
Is this the right approach for testing a custom bin used in the webrtc engine? Or is there a better / different approach? I checked out the integration tests in the repo, but none seemed to handle this exact case.
8 Replies
Hm, the code that you posted seems to look good. Could you share assertions that fail as well?
This is the assertion that's failing
I'm expecting this callback to be called:
I've even got a
handle_pad_added
without the guard
And it's never called either.
I assumed the notify_parent would trigger a notification to the test pipeline and I could assert on it.Do you implement
handle_setup
in any child that is spawned in your Pipeline?Yes.
I'm testing more, and the event returned from my
handle_init
I can assert on. But an event fired from this callback:
I can't assert on. Though I can see the IO.puts message in my logs, so I know the callback is being fired.
Though the children for this bin aren't spawned until we handle this callback:
And the assertions that are failing right now are prior to that in the lifecycle.Give me please the implementation of
handle_setup
that you implement. Maybe the execution of it doesn't end?
Please check as well if handle_playing
is executed in children of your pipeline.I checked and the handle_setup isn't being called. It's one of the child elements of the bin.
My hunch is that I'm setting the spec up wrong in my test file. My conversation bin doesn't actually return the child spec until we get the
handle_pad_added
callbacks.
Here's the relevant code from the conversation.ex bin that I'm trying to test:
Since neither of those is tripped, I think I'm not setting my testing spec up correctly, or the Testing pipeline isn't correctly notifying the element that new pads have been added.I see that:
1. You don't link any bin pad in
handle_pad_added
for :input
pad
2. You link :input
pad in handle_pad_added
for :output
pad
But where do you link :output
pad? I seems it is skippedWe setup the whole spec in the
handle_pad_added
for the output pad. I omitted it for brevity earlier. I
I think I realized at least part of the problem. My element is silently dying. When I send send in the :new_tracks
notification, I'm also trying to subscribe to the track published by the engine. I've got this line:
But the state.rtc_engine is the pid of my testing process, not an actual running RTC engine. So that call silently fails, and nothing else is executed, which is why my handle_pad_added calls weren't firing.
Adding this to my test, unblocked me. I'm successfully setting up the pipeline and mocking the RTC Engine pieces.
Thanks @Feliks !
FWIW - i think the failure is silent because my test assertions were using the default timeout of 2_000, but that Engine.subscribe call has a timeout of 5_000. So I never tripped the timeout and saw the error message.
@Feliks So now my issue is that the first element in my bin is Membrane.RTC.Engine.Endpoint.WebRTC.TrackReceiver
which has it's input pad flow_control: :push
; but the output pad of the Membrane.Testing.DynamicSource
that I'm using has an output pad flow_control: :manual
, which is incompatible.
I perused the Engine code, and I assume it's doing something to mediate between the two flow controls, though looking at the code in membrane webrtc engine ( engine.ex
tee.ex
and filter_tee.ex
) it wasn't obvious to me how that was being done. What do you advise? Should I build a dummy element in my testing pipeline that just mediates between the Testing.Source element and one that expects a push?
Or I could make a copy of TrackReceiver that has the pad definitions set to :auto
and just roll with it. Though I don't fully understand the implications here, so I'm a little worried about doing that and producing some subtle bug in the future.
I appreciate the advice.