The problem
I have created and started Red5 server-side stream in my custom application based on oflaDemo in this way:
serverStream = StreamUtils.createServerStream( app , "mystream" );
// repeats /streams/prometheus.flv from oflaDemo several times (20sec):
serverStream.addItem( SimplePlayItem.build( "prometheus" , 0 , 20000 ) );
serverStream.addItem( SimplePlayItem.build( "prometheus" , 0 , 20000 ) );
// now I want to add "camsource" stream, wich is published from Flash client:
serverStream.addItem( SimplePlayItem.build( "camsource" , 0 , 20000 ) );
...
serverStream.play();
prometheus.flv plays 2x nicely, but when times comes for "camsource" the result is:
[Red5_Scheduler_Worker-3] org.red5.server.stream.ProviderService - Exception attempting to lookup file: ServletContext resource [/streams/camsource.flv] cannot be resolved to URL because it does not exist
How can I create IServerStream that re-publishes user stream posted from flash client?
I can both see flv stream and cam stream by NetStream
.play(...)
in Flash, but I cannot create IServerStream programatically to publish from live source.
The comsource
publishing and serverStream
are in the same Scope.
My current working path
Due to This response on Red5 group, I have created my own ServerStream, in which I can push redirected/fake packets manually (I am not sure if this is done correctly):
public static class LiveTVServerStream extends ServerStream {
public void manuallyPushPacket( IStreamPacket packet ) {
for ( IStreamListener listener : getStreamListeners() ) {
listener.packetReceived( this , packet );
}
}
}
Then I create such stream and register it in appStart()
:
liveTVServerStream = new LiveTVServerStream();
liveTVServerStream.setScope( appScope );
liveTVServerStream.setName( "livetv" );
liveTVServerStream.setPublishedName( "livetv" );
// Add single starting video because ServerStream requires at least 1 item
liveTVServerStream.addItem( SimplePlayItem.build( "prometheus" , 0 , 20000 ) );
liveTVServerStream.start();
IContext context = scope.getContext();
IProviderService providerService = (IProviderService) context.getBean( IProviderService.BEAN_NAME );
providerService.registerBroadcastStream( scope , "livetv" , liveTVServerStream);
Then, if my "news reporter" (video source from webcam by Flash) starts broadcasting, I redirect packets to liveTVServerStream
:
public void streamPublishStart( final IBroadcastStream stream ) {
if ( !stream.getPublishedName().equals( "newsreporter" ) ) {
return;
}
logInfo( "News reporter connected - redirecting packets" );
stream.addStreamListener( new IStreamListener() {
volatile int dispatchedPacketCount = 0;
volatile int ignoredPacketCount = 0;
@Override
public void packetReceived( IBroadcastStream newsReporterStream , IStreamPacket packet ) {
if ( packet instanceof IRTMPEvent ) {
dispatchedPacketCount++;
if ( (dispatchedPacketCount % 50) == 0 ) {
logInfo( "Dispatched packets: " + dispatchedPacketCount + " ignored packets: " + ignoredPacketCount );
}
// delta-time calculations
long t1 = newsReporterStream.getCreationTime();
long t2 = liveTVServerStream.getCreationTime();
long delta = t2 - t1;
int t3 = ((IRTMPEvent) packet).getTimestamp();
t3 += delta;
((IRTMPEvent) packet).setTimestamp( t3 );
liveTVServerStream.manuallyPushPacket( packet );
} else {
ignoredPacketCount++;
}
}
} );
Current results
The log of Red5 console can be found here.
"News reporter" flash works ok - sending video from webcam. But when I subscribe to watch "livetv" stream, there is no video. Flash parts works - tested it with ServerStream
containing several .flv files in queue and Flash connected & streamed properly, like live TV.