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

org.jitsi.examples.AVReceive2 Maven / Gradle / Ivy

/*
 * Copyright @ 2015 Atlassian Pty Ltd
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.jitsi.examples;

import java.io.*;
import java.net.*;
import java.util.*;

import org.jitsi.service.libjitsi.*;
import org.jitsi.service.neomedia.*;
import org.jitsi.service.neomedia.device.*;
import org.jitsi.service.neomedia.format.*;
import org.jitsi.utils.*;

/**
 * Implements an example application in the fashion of JMF's AVReceive2 example
 * which demonstrates the use of the libjitsi library for the purposes
 * of receiving audio and video via RTP means.
 *
 * @author Lyubomir Marinov
 */
public class AVReceive2
{
    /**
     * The port which is the target of the transmission i.e. on which the media
     * is to be received.
     *
     * @see #LOCAL_PORT_BASE_ARG_NAME
     */
    private int localPortBase;

    /**
     * The MediaStream instances initialized by this instance indexed
     * by their respective MediaType ordinal.
     */
    private MediaStream[] mediaStreams;

    /**
     * The InetAddress of the host which is the target of the receipt
     * i.e. from which the media is to be received.
     *
     * @see #REMOTE_HOST_ARG_NAME
     */
    private InetAddress remoteAddr;

    /**
     * The port which is the target of the receipt i.e. from which the media is
     * to be received.
     *
     * @see #REMOTE_PORT_BASE_ARG_NAME
     */
    private int remotePortBase;

    /**
     * Initializes a new AVReceive2 instance which is to receive audio
     * and video from a specific host and a specific port.
     *
     * @param localPortBase the port on which the audio and video are to be
     * received
     * @param remoteHost the name of the host from which the media is
     * transmitted
     * @param remotePortBase the port from which the media is transmitted
     * @throws Exception if any error arises during the parsing of the specified
     * localPortBase, remoteHost and remotePortBase
     */
    private AVReceive2(
            String localPortBase,
            String remoteHost, String remotePortBase)
        throws Exception
    {
        this.localPortBase
            = (localPortBase == null)
                ? -1
                : Integer.parseInt(localPortBase);
        this.remoteAddr = InetAddress.getByName(remoteHost);
        this.remotePortBase = Integer.parseInt(remotePortBase);
    }

    /**
     * Initializes the receipt of audio and video.
     *
     * @return true if this instance has been successfully initialized
     * to receive audio and video
     * @throws Exception if anything goes wrong while initializing this instance
     * for the receipt of audio and video
     */
    private boolean initialize()
        throws Exception
    {
        /*
         * Prepare for the start of the transmission i.e. initialize the
         * MediaStream instances.
         */
        MediaType[] mediaTypes = MediaType.values();
        MediaService mediaService = LibJitsi.getMediaService();
        int localPort = localPortBase;
        int remotePort = remotePortBase;

        mediaStreams = new MediaStream[mediaTypes.length];
        for (MediaType mediaType : mediaTypes)
        {
            /*
             * The default MediaDevice (for a specific MediaType) is configured
             * (by the user of the application via some sort of UI) into the
             * ConfigurationService. If there is no ConfigurationService
             * instance known to LibJitsi, the first available MediaDevice of
             * the specified MediaType will be chosen by MediaService.
             */
            MediaDevice device
                = mediaService.getDefaultDevice(mediaType, MediaUseCase.CALL);
            MediaStream mediaStream = mediaService.createMediaStream(device);

            // direction
            /*
             * The AVTransmit2 example sends only and the AVReceive2 receives
             * only. In a call, the MediaStream's direction will most commonly
             * be set to SENDRECV.
             */
            mediaStream.setDirection(MediaDirection.RECVONLY);

            // format
            String encoding;
            double clockRate;
            /*
             * The AVTransmit2 and AVReceive2 examples use the H.264 video
             * codec. Its RTP transmission has no static RTP payload type number
             * assigned.
             */
            byte dynamicRTPPayloadType;

            switch (device.getMediaType())
            {
            case AUDIO:
                encoding = "PCMU";
                clockRate = 8000;
                /* PCMU has a static RTP payload type number assigned. */
                dynamicRTPPayloadType = -1;
                break;
            case VIDEO:
                encoding = "H264";
                clockRate = MediaFormatFactory.CLOCK_RATE_NOT_SPECIFIED;
                /*
                 * The dymanic RTP payload type numbers are usually negotiated
                 * in the signaling functionality.
                 */
                dynamicRTPPayloadType = 99;
                break;
            default:
                encoding = null;
                clockRate = MediaFormatFactory.CLOCK_RATE_NOT_SPECIFIED;
                dynamicRTPPayloadType = -1;
            }

            if (encoding != null)
            {
                MediaFormat format
                    = mediaService.getFormatFactory().createMediaFormat(
                            encoding,
                            clockRate);

                /*
                 * The MediaFormat instances which do not have a static RTP
                 * payload type number association must be explicitly assigned
                 * a dynamic RTP payload type number.
                 */
                if (dynamicRTPPayloadType != -1)
                {
                    mediaStream.addDynamicRTPPayloadType(
                            dynamicRTPPayloadType,
                            format);
                }

                mediaStream.setFormat(format);
            }

            // connector
            StreamConnector connector;

            if (localPortBase == -1)
            {
                connector = new DefaultStreamConnector();
            }
            else
            {
                int localRTPPort = localPort++;
                int localRTCPPort = localPort++;

                connector
                    = new DefaultStreamConnector(
                            new DatagramSocket(localRTPPort),
                            new DatagramSocket(localRTCPPort));
            }
            mediaStream.setConnector(connector);

            // target
            /*
             * The AVTransmit2 and AVReceive2 examples follow the common
             * practice that the RTCP port is right after the RTP port.
             */
            int remoteRTPPort = remotePort++;
            int remoteRTCPPort = remotePort++;

            mediaStream.setTarget(
                    new MediaStreamTarget(
                            new InetSocketAddress(remoteAddr, remoteRTPPort),
                            new InetSocketAddress(remoteAddr, remoteRTCPPort)));

            // name
            /*
             * The name is completely optional and it is not being used by the
             * MediaStream implementation at this time, it is just remembered so
             * that it can be retrieved via MediaStream#getName(). It may be
             * integrated with the signaling functionality if necessary.
             */
            mediaStream.setName(mediaType.toString());

            mediaStreams[mediaType.ordinal()] = mediaStream;
        }

        /*
         * Do start the transmission i.e. start the initialized MediaStream
         * instances.
         */
        for (MediaStream mediaStream : mediaStreams)
            if (mediaStream != null)
                mediaStream.start();

        return true;
    }

    /**
     * Close the MediaStreams.
     */
    private void close()
    {
        if (mediaStreams != null)
        {
            for (int i = 0; i < mediaStreams.length; i++)
            {
                MediaStream mediaStream = mediaStreams[i];

                if (mediaStream != null)
                {
                    try
                    {
                        mediaStream.stop();
                    }
                    finally
                    {
                        mediaStream.close();
                        mediaStreams[i] = null;
                    }
                }
            }

            mediaStreams = null;
        }
    }

    /**
     * The name of the command-line argument which specifies the port on which
     * the media is to be received. The command-line argument value will be used
     * as the port to receive the audio RTP on, the next port after it will be
     * used to receive the audio RTCP on. Respectively, the subsequent ports
     * ports will be used to transmit the video RTP and RTCP on."
     */
    private static final String LOCAL_PORT_BASE_ARG_NAME
        = "--local-port-base=";

    /**
     * The name of the command-line argument which specifies the name of the
     * host from which the media is to be received.
     */
    private static final String REMOTE_HOST_ARG_NAME = "--remote-host=";

    /**
     * The name of the command-line argument which specifies the port from which
     * the media is to be received. The command-line argument value will be
     * used as the port to receive the audio RTP from, the next port after it
     * will be to receive the audio RTCP from. Respectively, the subsequent
     * ports will be used to receive the video RTP and RTCP from."
     */
    private static final String REMOTE_PORT_BASE_ARG_NAME
        = "--remote-port-base=";

    /**
     * The list of command-line arguments accepted as valid by the
     * AVReceive2 application along with their human-readable usage
     * descriptions.
     */
    private static final String[][] ARGS
        = {
            {
                LOCAL_PORT_BASE_ARG_NAME,
                "The port on which media is to be received. The specified value"
                    + " will be used as the port to receive the audio RTP on,"
                    + " the next port after it will be used to receive the"
                    + " audio RTCP on. Respectively, the subsequent ports will"
                    + " be used to receive the video RTP and RTCP on."
            },
            {
                REMOTE_HOST_ARG_NAME,
                "The name of the host from which the media is to be received."
            },
            {
                REMOTE_PORT_BASE_ARG_NAME,
                "The port from which media is to be received. The specified"
                    + " vaue will be used as the port to receive the audio RTP"
                    + " from, the next port after it will be used to receive"
                    + " the audio RTCP from. Respectively, the subsequent ports"
                    + " will be used to receive the video RTP and RTCP from."
            }
        };

    public static void main(String[] args)
        throws Exception
    {
        // We need three parameters to do the transmission. For example,
        // ant run-example -Drun.example.name=AVReceive2 -Drun.example.arg.line="--local-port-base=10000 --remote-host=129.130.131.132 --remote-port-base=5000"
        if (args.length < 3)
        {
            prUsage();
        }
        else
        {
            Map argMap = AVTransmit2.parseCommandLineArgs(args);

            LibJitsi.start();
            try
            {
                AVReceive2 avReceive
                    = new AVReceive2(
                            argMap.get(LOCAL_PORT_BASE_ARG_NAME),
                            argMap.get(REMOTE_HOST_ARG_NAME),
                            argMap.get(REMOTE_PORT_BASE_ARG_NAME));

                if (avReceive.initialize())
                {
                    try
                    {
                        /*
                         * Wait for the media to be received and played back.
                         * AVTransmit2 transmits for 1 minute so AVReceive2
                         * waits for 2 minutes to allow AVTransmit2 to start the
                         * tranmission with a bit of a delay (if necessary).
                         */
                        long then = System.currentTimeMillis();
                        long waitingPeriod = 2 * 60000;

                        try
                        {
                            while ((System.currentTimeMillis() - then)
                                    < waitingPeriod)
                                Thread.sleep(1000);
                        }
                        catch (InterruptedException ie)
                        {
                        }
                    }
                    finally
                    {
                        avReceive.close();
                    }

                    System.err.println("Exiting AVReceive2");
                }
                else
                {
                    System.err.println("Failed to initialize the sessions.");
                }
            }
            finally
            {
                LibJitsi.stop();
            }
        }
    }

    /**
     * Outputs human-readable description about the usage of the
     * AVReceive2 application and the command-line arguments it
     * accepts as valid.
     */
    private static void prUsage()
    {
        PrintStream err = System.err;

        err.println("Usage: " + AVReceive2.class.getName() + " ");
        err.println("Valid args:");
        for (String[] arg : ARGS)
            err.println("  " + arg[0] + " " + arg[1]);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy