![]() System : Linux absol.cf 5.4.0-198-generic #218-Ubuntu SMP Fri Sep 27 20:18:53 UTC 2024 x86_64 User : www-data ( 33) PHP Version : 7.4.33 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare, Directory : /proc/thread-self/root/usr/local/lib/node_modules/mediasoup/src/ |
Upload File : |
import { v4 as uuidv4 } from 'uuid'; import { Logger } from './Logger'; import { EnhancedEventEmitter } from './EnhancedEventEmitter'; import * as ortc from './ortc'; import { Transport, TransportListenIp, TransportTuple, TransportTraceEventData, SctpState } from './Transport'; import { Consumer, ConsumerOptions } from './Consumer'; import { SctpParameters, NumSctpStreams } from './SctpParameters'; import { SrtpParameters } from './SrtpParameters'; export type PipeTransportOptions = { /** * Listening IP address. */ listenIp: TransportListenIp | string; /** * Create a SCTP association. Default false. */ enableSctp?: boolean; /** * SCTP streams number. */ numSctpStreams?: NumSctpStreams; /** * Maximum allowed size for SCTP messages sent by DataProducers. * Default 268435456. */ maxSctpMessageSize?: number; /** * Maximum SCTP send buffer used by DataConsumers. * Default 268435456. */ sctpSendBufferSize?: number; /** * Enable RTX and NACK for RTP retransmission. Useful if both Routers are * located in different hosts and there is packet lost in the link. For this * to work, both PipeTransports must enable this setting. Default false. */ enableRtx?: boolean; /** * Enable SRTP. Useful to protect the RTP and RTCP traffic if both Routers * are located in different hosts. For this to work, connect() must be called * with remote SRTP parameters. Default false. */ enableSrtp?: boolean; /** * Custom application data. */ appData?: any; } export type PipeTransportStat = { // Common to all Transports. type: string; transportId: string; timestamp: number; sctpState?: SctpState; bytesReceived: number; recvBitrate: number; bytesSent: number; sendBitrate: number; rtpBytesReceived: number; rtpRecvBitrate: number; rtpBytesSent: number; rtpSendBitrate: number; rtxBytesReceived: number; rtxRecvBitrate: number; rtxBytesSent: number; rtxSendBitrate: number; probationBytesSent: number; probationSendBitrate: number; availableOutgoingBitrate?: number; availableIncomingBitrate?: number; maxIncomingBitrate?: number; // PipeTransport specific. tuple: TransportTuple; } const logger = new Logger('PipeTransport'); export class PipeTransport extends Transport { // PipeTransport data. protected readonly _data: { tuple: TransportTuple; sctpParameters?: SctpParameters; sctpState?: SctpState; rtx: boolean; srtpParameters?: SrtpParameters; }; /** * @private * @emits sctpstatechange - (sctpState: SctpState) * @emits trace - (trace: TransportTraceEventData) */ constructor(params: any) { super(params); logger.debug('constructor()'); const { data } = params; this._data = { tuple : data.tuple, sctpParameters : data.sctpParameters, sctpState : data.sctpState, rtx : data.rtx, srtpParameters : data.srtpParameters }; this._handleWorkerNotifications(); } /** * Transport tuple. */ get tuple(): TransportTuple { return this._data.tuple; } /** * SCTP parameters. */ get sctpParameters(): SctpParameters | undefined { return this._data.sctpParameters; } /** * SCTP state. */ get sctpState(): SctpState | undefined { return this._data.sctpState; } /** * SRTP parameters. */ get srtpParameters(): SrtpParameters | undefined { return this._data.srtpParameters; } /** * Observer. * * @override * @emits close * @emits newproducer - (producer: Producer) * @emits newconsumer - (consumer: Consumer) * @emits newdataproducer - (dataProducer: DataProducer) * @emits newdataconsumer - (dataConsumer: DataConsumer) * @emits sctpstatechange - (sctpState: SctpState) * @emits trace - (trace: TransportTraceEventData) */ get observer(): EnhancedEventEmitter { return this._observer; } /** * Close the PipeTransport. * * @override */ close(): void { if (this._closed) return; if (this._data.sctpState) this._data.sctpState = 'closed'; super.close(); } /** * Router was closed. * * @private * @override */ routerClosed(): void { if (this._closed) return; if (this._data.sctpState) this._data.sctpState = 'closed'; super.routerClosed(); } /** * Get PipeTransport stats. * * @override */ async getStats(): Promise<PipeTransportStat[]> { logger.debug('getStats()'); return this._channel.request('transport.getStats', this._internal); } /** * Provide the PipeTransport remote parameters. * * @override */ async connect( { ip, port, srtpParameters }: { ip: string; port: number; srtpParameters?: SrtpParameters; } ): Promise<void> { logger.debug('connect()'); const reqData = { ip, port, srtpParameters }; const data = await this._channel.request('transport.connect', this._internal, reqData); // Update data. this._data.tuple = data.tuple; } /** * Create a pipe Consumer. * * @override */ async consume({ producerId, appData = {} }: ConsumerOptions): Promise<Consumer> { logger.debug('consume()'); if (!producerId || typeof producerId !== 'string') throw new TypeError('missing producerId'); else if (appData && typeof appData !== 'object') throw new TypeError('if given, appData must be an object'); const producer = this._getProducerById(producerId); if (!producer) throw Error(`Producer with id "${producerId}" not found`); // This may throw. const rtpParameters = ortc.getPipeConsumerRtpParameters( producer.consumableRtpParameters, this._data.rtx); const internal = { ...this._internal, consumerId: uuidv4(), producerId }; const reqData = { kind : producer.kind, rtpParameters, type : 'pipe', consumableRtpEncodings : producer.consumableRtpParameters.encodings }; const status = await this._channel.request('transport.consume', internal, reqData); const data = { kind: producer.kind, rtpParameters, type: 'pipe' }; const consumer = new Consumer( { internal, data, channel : this._channel, payloadChannel : this._payloadChannel, appData, paused : status.paused, producerPaused : status.producerPaused }); this._consumers.set(consumer.id, consumer); consumer.on('@close', () => this._consumers.delete(consumer.id)); consumer.on('@producerclose', () => this._consumers.delete(consumer.id)); // Emit observer event. this._observer.safeEmit('newconsumer', consumer); return consumer; } private _handleWorkerNotifications(): void { this._channel.on(this._internal.transportId, (event: string, data?: any) => { switch (event) { case 'sctpstatechange': { const sctpState = data.sctpState as SctpState; this._data.sctpState = sctpState; this.safeEmit('sctpstatechange', sctpState); // Emit observer event. this._observer.safeEmit('sctpstatechange', sctpState); break; } case 'trace': { const trace = data as TransportTraceEventData; this.safeEmit('trace', trace); // Emit observer event. this._observer.safeEmit('trace', trace); break; } default: { logger.error('ignoring unknown event "%s"', event); } } }); } }