All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.antmedia.streamsource.StreamFetcher Maven / Gradle / Ivy

package io.antmedia.streamsource;

import static org.bytedeco.javacpp.avcodec.av_packet_free;
import static org.bytedeco.javacpp.avcodec.av_packet_ref;
import static org.bytedeco.javacpp.avcodec.av_packet_unref;
import static org.bytedeco.javacpp.avformat.av_read_frame;
import static org.bytedeco.javacpp.avformat.avformat_close_input;
import static org.bytedeco.javacpp.avformat.avformat_find_stream_info;
import static org.bytedeco.javacpp.avformat.avformat_open_input;
import static org.bytedeco.javacpp.avutil.av_dict_free;
import static org.bytedeco.javacpp.avutil.av_dict_set;
import static org.bytedeco.javacpp.avutil.av_rescale_q;
import static org.bytedeco.javacpp.avutil.AVMEDIA_TYPE_AUDIO;

import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean;

import org.apache.commons.lang3.exception.ExceptionUtils;
import org.bytedeco.javacpp.avcodec;
import org.bytedeco.javacpp.avcodec.AVPacket;
import org.bytedeco.javacpp.avformat.AVFormatContext;
import org.bytedeco.javacpp.avutil;
import org.bytedeco.javacpp.avutil.AVDictionary;
import org.bytedeco.javacpp.avutil.AVRational;
import org.red5.server.api.scheduling.IScheduledJob;
import org.red5.server.api.scheduling.ISchedulingService;
import org.red5.server.api.scope.IScope;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import io.antmedia.AntMediaApplicationAdapter;
import io.antmedia.AppSettings;
import io.antmedia.datastore.db.types.Broadcast;
import io.antmedia.muxer.MuxAdaptor;
import io.antmedia.rest.model.Result;

public class StreamFetcher {

	protected static Logger logger = LoggerFactory.getLogger(StreamFetcher.class);
	private Broadcast stream;
	private WorkerThread thread;
	/**
	 * Connection setup timeout value
	 */
	private int timeout;
	private boolean exceptionInThread = false;

	/**
	 * Last packet received time
	 */
	private long lastPacketReceivedTime = 0;
	private boolean threadActive = false;
	private Result cameraError=new Result(false,"");
	private static final int PACKET_RECEIVED_INTERVAL_TIMEOUT = 3000;
	private IScope scope;
	private AntMediaApplicationAdapter appInstance;
	private long[] lastDTS;
	private MuxAdaptor muxAdaptor = null;

	/**
	 * If it is true, it restarts fetching everytime it disconnects
	 * if it is false, it does not restart
	 */
	private boolean restartStream = true;

	/**
	 * Buffer time in milliseconds
	 */
	private int bufferTime = 0;

	private ConcurrentLinkedQueue availableBufferQueue = new ConcurrentLinkedQueue<>();

	private ISchedulingService scheduler;
	private AVRational avRationalTimeBaseMS;
	private AppSettings appSettings;

	public StreamFetcher(Broadcast stream, IScope scope, ISchedulingService scheduler)  {
		if (stream == null || stream.getStreamId() == null || stream.getStreamUrl() == null) {
			String streamId = null;
			if (stream != null) {
				streamId = stream.getStreamId();
			}
			String streamUrl = null;
			if (stream != null) {
				streamUrl = stream.getStreamUrl();
			}
			throw new NullPointerException("Stream is not initialized properly. Check stream("+stream+"), "
					+ " stream id ("+ streamId +") and stream url ("+ streamUrl + ") values");
		}

		this.stream = stream;
		this.scope = scope;
		this.scheduler = scheduler;


		if (getAppSettings() == null) {
			throw new NullPointerException("App Settings is null in StreamFetcher");
		}

		this.bufferTime = getAppSettings().getStreamFetcherBufferTime();

		avRationalTimeBaseMS = new AVRational();
		avRationalTimeBaseMS.num(1);
		avRationalTimeBaseMS.den(1000);

		logger.debug(":::::::::::scope is {}" , scope.getName());

	}

	public Result prepareInput(AVFormatContext inputFormatContext) {

		setConnectionTimeout(5000);

		Result result = new Result(false);
		if (inputFormatContext == null) {
			logger.info("cannot allocate input context for {}", stream.getStreamId());
			return result;
		}

		AVDictionary optionsDictionary = new AVDictionary();

		String streamUrl = stream.getStreamUrl();
		if (streamUrl.startsWith("rtsp://")) {
			av_dict_set(optionsDictionary, "rtsp_transport", "tcp", 0);
		}

		String timeoutStr = String.valueOf(this.timeout);
		av_dict_set(optionsDictionary, "stimeout", timeoutStr, 0);

		int ret;

		logger.debug("stream url: {}  " , stream.getStreamUrl());

		if ((ret = avformat_open_input(inputFormatContext, stream.getStreamUrl(), null, optionsDictionary)) < 0) {

			byte[] data = new byte[1024];
			avutil.av_strerror(ret, data, data.length);

			String errorStr=new String(data, 0, data.length);

			result.setMessage(errorStr);		

			logger.debug("cannot open input context with error:: {}",  result.getMessage());
			return result;
		}

		av_dict_free(optionsDictionary);

		ret = avformat_find_stream_info(inputFormatContext, (AVDictionary) null);
		if (ret < 0) {

			result.setMessage("Could not find stream information\n");
			logger.info(result.getMessage());
			return result;
		}

		lastDTS = new long[inputFormatContext.nb_streams()];

		for (int i = 0; i < lastDTS.length; i++) {
			lastDTS[i] = -1;
		}

		result.setSuccess(true);
		return result;

	}

	public Result prepare(AVFormatContext inputFormatContext) {
		Result result = prepareInput(inputFormatContext);

		setCameraError(result);

		return result;

	}

	public class WorkerThread extends Thread implements IScheduledJob {

		private static final int PACKET_WRITER_PERIOD_IN_MS = 10;

		private volatile boolean stopRequestReceived = false;

		private volatile boolean streamPublished = false;
		protected AtomicBoolean isJobRunning = new AtomicBoolean(false);
		AVFormatContext inputFormatContext = null;

		private volatile boolean buffering = false;
		private ConcurrentLinkedQueue bufferQueue = new ConcurrentLinkedQueue<>();

		@Override
		public void run() {

			setThreadActive(true);
			long lastPacketTime = 0;
			long firstPacketTime = 0;
			long bufferDuration = 0;

			AVPacket pkt = null;
			String packetWriterJobName = null;
			try {
				inputFormatContext = new AVFormatContext(null); 
				pkt = avcodec.av_packet_alloc();
				Result result = prepare(inputFormatContext);


				if (result.isSuccess()) {
					boolean audioOnly = false;
					if(inputFormatContext.nb_streams() == 1) {
						audioOnly  = (inputFormatContext.streams(0).codecpar().codec_type() == AVMEDIA_TYPE_AUDIO);
					}

					muxAdaptor = MuxAdaptor.initializeMuxAdaptor(null,true, scope);
					// if there is only audio, firstKeyFrameReceivedChecked should be true in advance
					// because there is no video frame
					muxAdaptor.setFirstKeyFrameReceivedChecked(audioOnly); 


					muxAdaptor.init(scope, stream.getStreamId(), false);


					logger.info("{} stream count in stream {} is {}", stream.getStreamId(), stream.getStreamUrl(), inputFormatContext.nb_streams());

					if(muxAdaptor.prepareInternal(inputFormatContext)) {

						long currentTime = System.currentTimeMillis();
						muxAdaptor.setStartTime(currentTime);

						getInstance().startPublish(stream.getStreamId());

						if (bufferTime > 0) {
							packetWriterJobName = scheduler.addScheduledJob(PACKET_WRITER_PERIOD_IN_MS, this);
						}

						int bufferLogCounter = 0;
						while (av_read_frame(inputFormatContext, pkt) >= 0) {

							streamPublished = true;
							lastPacketReceivedTime = System.currentTimeMillis();

							/**
							 * Check that dts values are monotically increasing for each stream
							 */
							int packetIndex = pkt.stream_index();
							if (lastDTS[packetIndex] >= pkt.dts()) {
								pkt.dts(lastDTS[packetIndex] + 1);
							}
							lastDTS[packetIndex] = pkt.dts();
							if (pkt.dts() > pkt.pts()) {
								logger.info("dts ({}) is bigger than pts({})", pkt.dts(), pkt.pts());
								pkt.pts(pkt.dts());
							}

							/***************************************************
							 *  Memory of being paranoid or failing while looking for excellence without understanding the whole picture
							 *  
							 *  Increasing pkt.dts plus 1 is a simple hack for fixing dts error if current dts has a value lower 
							 *  than the last received dts. Because dts should be monotonically increasing. I made this simple hack and it is working. 
							 *  After that I thought the same may happen for the pts value as well and I have added below fix. 
							 *  Actually not a fix, it is a bug. Because pts values does not have to be monotonically increasing
							 *  and if stream has B-Frames then pts value can be lower than the last PTS value. So below
							 *  code snippet make the stream does not play smoothly. It took about 10 hours to find it this error.
							 *  
							 *  I have written this simple memory for me
							 *  and for the guys who is developing or reviewing this code. 
							 *  Even if it is time consuming or not reasonable, these kind of tryouts sometimes makes me excited. 
							 *  I think I may expect to find something great by trying something crazy :) 
							 *  
							 *  @mekya - June 12, 2018
							 *  
							 *  ---------------------------------------------------
							 *  
							 *  if (lastPTS[packetIndex] >= pkt.pts()) {
							 * 	   pkt.pts(lastPTS[packetIndex] + 1);
							 *  }
							 *  lastPTS[packetIndex] = pkt.pts();
							 *
							 ******************************************************/
							if (bufferTime > 0) 
							{
								AVPacket packet = getAVPacket();
								av_packet_ref(packet, pkt);
								bufferQueue.add(packet);

								AVPacket pktHead = bufferQueue.peek();
								/**
								 * BufferQueue may be polled in writer thread. 
								 * It's a very rare case to happen so that check if it's null
								 */
								if (pktHead != null) {
									lastPacketTime = av_rescale_q(pkt.pts(), inputFormatContext.streams(pkt.stream_index()).time_base(), avRationalTimeBaseMS);
									firstPacketTime = av_rescale_q(pktHead.pts(), inputFormatContext.streams(pktHead.stream_index()).time_base(), avRationalTimeBaseMS);
									bufferDuration = (lastPacketTime - firstPacketTime);

									if ( bufferDuration > bufferTime) {
										buffering = false;
									}

									bufferLogCounter++;
									if (bufferLogCounter % 100 == 0) {
										logger.info("Buffer status {}, buffer duration {}ms buffer time {}ms", buffering, bufferDuration, bufferTime);
										bufferLogCounter = 0;
									}
								}
							}
							else {
								muxAdaptor.writePacket(inputFormatContext.streams(pkt.stream_index()), pkt);
							}
							av_packet_unref(pkt);
							if (stopRequestReceived) {
								logger.warn("Stop request received, breaking the loop for {} ", stream.getStreamId());
								break;
							}
						}
						logger.info("Leaving the loop for {}", stream.getStreamId());

					}

				}
				else {
					logger.debug("Prepare for {} returned false", stream.getName());
				}

				setCameraError(result);
			} 
			catch (OutOfMemoryError | Exception e) {
				logger.error(ExceptionUtils.getStackTrace(e));
				exceptionInThread  = true;
			}


			if (packetWriterJobName != null) {
				logger.info("Removing packet writer job {}", packetWriterJobName);
				scheduler.removeScheduledJob(packetWriterJobName);
			}

			writeAllBufferedPackets();


			if (muxAdaptor != null) {
				logger.info("Writing trailer in Muxadaptor {}", stream.getStreamId());
				muxAdaptor.writeTrailer(inputFormatContext);
				muxAdaptor = null;
			}

			if (pkt != null) {
				av_packet_free(pkt);
			}

			if (inputFormatContext != null) {
				try {
					avformat_close_input(inputFormatContext);
				}
				catch (Exception e) {
					logger.info(e.getMessage());
				}
				inputFormatContext = null;
			}

			if(streamPublished) {
				getInstance().closeBroadcast(stream.getStreamId());
				streamPublished=false;
			}


			setThreadActive(false);
			if(!stopRequestReceived && restartStream) {
				thread = new WorkerThread();
				thread.start();
			}

			logger.debug("Leaving thread for {}", stream.getStreamUrl());


		}

		private void writeAllBufferedPackets() 
		{
			while (!bufferQueue.isEmpty()) {
				AVPacket pkt = bufferQueue.poll();
				muxAdaptor.writePacket(inputFormatContext.streams(pkt.stream_index()), pkt);
				av_packet_unref(pkt);
			}

			AVPacket pkt;
			while ((pkt = bufferQueue.poll()) != null) {
				pkt.close();
			}
		}

		public void setStopRequestReceived() {
			logger.warn("inside of setStopRequestReceived for {}", stream.getStreamId());
			stopRequestReceived = true;
		}

		public boolean isStopRequestReceived() {
			return stopRequestReceived;
		}

		@Override
		public void execute(ISchedulingService service) throws CloneNotSupportedException 
		{
			if (isJobRunning.compareAndSet(false, true)) 
			{
				if (!buffering) {
					AVPacket pkt = bufferQueue.poll();
					if (pkt != null) {
						muxAdaptor.writePacket(inputFormatContext.streams(pkt.stream_index()), pkt);
						av_packet_unref(pkt);
						availableBufferQueue.offer(pkt);
					}
					else {
						buffering = true;
					}
				}
				isJobRunning.compareAndSet(true, false);
			}
		}
	}

	public void startStream() {
		new Thread() {
			@Override
			public void run() {
				try {
					int i = 0;
					while (threadActive) {
						Thread.sleep(100);
						if (i % 50 == 0) {
							logger.info("waiting for thread to be finished for stream {}", stream.getStreamUrl());
							i = 0;
						}
						i++;
					}
					Thread.sleep(2000);
				} catch (InterruptedException e) {
					logger.error(e.getMessage());
					Thread.currentThread().interrupt();
				}

				exceptionInThread = false;
				thread = new WorkerThread();
				thread.start();
				logger.info("StartStream called, new thread is started for {}", stream.getStreamId());
			}
		}.start();

	}




	public AVPacket getAVPacket() {
		if (!availableBufferQueue.isEmpty()) {
			return availableBufferQueue.poll();
		}
		return new AVPacket();
	}

	/**
	 * If thread is alive and receiving packet with in the {@link PACKET_RECEIVED_INTERVAL_TIMEOUT} time
	 * mean it is running
	 * @return true if it is running and false it is not
	 */
	public boolean isStreamAlive() {
		return ((System.currentTimeMillis() - lastPacketReceivedTime) < PACKET_RECEIVED_INTERVAL_TIMEOUT);
	}

	public boolean isStopped() {
		return thread.isInterrupted();
	}

	public void stopStream() 
	{
		if (getThread() != null) {
			logger.warn("stop stream called for {}", stream.getStreamId());
			getThread().setStopRequestReceived();

		}else {
			logger.warn("stop stream is called and thread is null {}",  stream.getStreamId());
		}
	}

	public boolean isStopRequestReceived() {
		return getThread().isStopRequestReceived();
	}

	public WorkerThread getThread() {
		return thread;
	}

	public void setThread(WorkerThread thread) {
		this.thread = thread;
	}

	public Broadcast getStream() {
		return stream;
	}

	public void restart() {
		stopStream();
		new Thread() {
			@Override
			public void run() {
				try {
					while (threadActive) {
						Thread.sleep(100);
					}

					Thread.sleep(2000);
				} catch (InterruptedException e) {
					logger.error(e.getMessage());
					Thread.currentThread().interrupt();
				}
				startStream();
			}
		}.start();

	}
	/**
	 * Set timeout when establishing connection
	 * @param timeout in ms
	 */
	public void setConnectionTimeout(int timeout) {
		this.timeout = timeout * 1000;
	}

	public boolean isExceptionInThread() {
		return exceptionInThread;
	}

	public void setThreadActive(boolean threadActive) {
		this.threadActive = threadActive;
	}

	public boolean isThreadActive() {
		return threadActive;
	}
	public Result getCameraError() {
		return cameraError;
	}

	public void setCameraError(Result cameraError) {
		this.cameraError = cameraError;
	}
	public IScope getScope() {
		return scope;
	}

	public void setScope(IScope scope) {
		this.scope = scope;
	}

	public AntMediaApplicationAdapter getInstance() {
		if (appInstance == null) {
			appInstance = (AntMediaApplicationAdapter) scope.getContext().getApplicationContext().getBean("web.handler");
		}
		return appInstance;
	}

	public MuxAdaptor getMuxAdaptor() {
		return muxAdaptor;
	}

	public void setMuxAdaptor(MuxAdaptor muxAdaptor) {
		this.muxAdaptor = muxAdaptor;
	}

	public boolean isRestartStream() {
		return restartStream;
	}

	public void setRestartStream(boolean restartStream) {
		this.restartStream = restartStream;
	}

	public void setStream(Broadcast stream) {
		this.stream = stream;
	}

	public int getBufferTime() {
		return bufferTime;
	}

	public void setBufferTime(int bufferTime) {
		this.bufferTime = bufferTime;
	}

	private AppSettings getAppSettings() {
		if (appSettings == null) {
			appSettings = (AppSettings) scope.getContext().getApplicationContext().getBean(AppSettings.BEAN_NAME);
		}
		return appSettings;
	}


}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy