@basmilius/apple-raop 0.7.2 → 0.8.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,189 @@
1
+ import { Socket } from "node:dgram";
2
+ import { EventEmitter } from "node:events";
3
+ import { AudioSource, Connection, Context, DiscoveryResult, TimingServer } from "@basmilius/apple-common";
4
+
5
+ //#region src/types.d.ts
6
+ type MediaMetadata = {
7
+ readonly title: string;
8
+ readonly artist: string;
9
+ readonly album: string;
10
+ readonly duration: number;
11
+ readonly artwork?: Buffer;
12
+ };
13
+ type PlaybackInfo = {
14
+ readonly metadata: MediaMetadata;
15
+ readonly position: number;
16
+ };
17
+ type StreamContext = {
18
+ sampleRate: number;
19
+ channels: number;
20
+ bytesPerChannel: number;
21
+ rtpseq: number;
22
+ rtptime: number;
23
+ headTs: number;
24
+ latency: number;
25
+ serverPort: number;
26
+ controlPort: number;
27
+ rtspSession: string;
28
+ volume: number;
29
+ position: number;
30
+ packetSize: number;
31
+ frameSize: number;
32
+ paddingSent: number;
33
+ reset(): void;
34
+ };
35
+ interface StreamProtocol {
36
+ setup(timingPort: number, controlPort: number): Promise<void>;
37
+ startFeedback(): Promise<void>;
38
+ sendAudioPacket(transport: Socket, header: Buffer, audio: Buffer): Promise<[number, Buffer]>;
39
+ teardown(): void;
40
+ }
41
+ interface Settings {
42
+ protocols: {
43
+ raop: {
44
+ controlPort: number;
45
+ timingPort: number;
46
+ };
47
+ };
48
+ }
49
+ declare enum EncryptionType {
50
+ Unknown = 0,
51
+ Unencrypted = 1,
52
+ MFiSAP = 2
53
+ }
54
+ declare enum MetadataType {
55
+ NotSupported = 0,
56
+ Text = 1,
57
+ Artwork = 2,
58
+ Progress = 4
59
+ }
60
+ interface RaopListener {
61
+ playing(playbackInfo: PlaybackInfo): void;
62
+ stopped(): void;
63
+ }
64
+ //#endregion
65
+ //#region src/packets.d.ts
66
+ declare class PacketFifo {
67
+ #private;
68
+ constructor(maxSize: number);
69
+ get(seqno: number): Buffer | undefined;
70
+ set(seqno: number, packet: Buffer): void;
71
+ has(seqno: number): boolean;
72
+ clear(): void;
73
+ }
74
+ declare const AudioPacketHeader: {
75
+ encode(header: number, payloadType: number, seqno: number, timestamp: number, ssrc: number): Buffer;
76
+ };
77
+ declare const SyncPacket: {
78
+ encode(header: number, payloadType: number, seqno: number, rtpTimestamp: number, ntpSec: number, ntpFrac: number, rtpTimestampNow: number): Buffer;
79
+ };
80
+ type RetransmitRequest = {
81
+ readonly lostSeqno: number;
82
+ readonly lostPackets: number;
83
+ };
84
+ declare function decodeRetransmitRequest(data: Buffer): RetransmitRequest;
85
+ //#endregion
86
+ //#region src/utils.d.ts
87
+ declare function pctToDbfs(volume: number): number;
88
+ declare function getEncryptionTypes(properties: Map<string, string>): EncryptionType;
89
+ declare function getMetadataTypes(properties: Map<string, string>): MetadataType;
90
+ declare function getAudioProperties(properties: Map<string, string>): [number, number, number];
91
+ //#endregion
92
+ //#region src/controlClient.d.ts
93
+ declare class ControlClient extends EventEmitter {
94
+ #private;
95
+ constructor(context: StreamContext, packetBacklog: PacketFifo);
96
+ get port(): number;
97
+ bind(localIp: string, port: number): Promise<void>;
98
+ close(): void;
99
+ start(remoteAddr: string): void;
100
+ stop(): void;
101
+ }
102
+ //#endregion
103
+ //#region src/rtspClient.d.ts
104
+ declare class RtspClient extends Connection<{}> {
105
+ #private;
106
+ get activeRemoteId(): string;
107
+ get dacpId(): string;
108
+ get rtspSessionId(): string;
109
+ get sessionId(): number;
110
+ get uri(): string;
111
+ get connection(): {
112
+ localIp: string;
113
+ remoteIp: string;
114
+ };
115
+ constructor(context: Context, address: string, port: number);
116
+ info(): Promise<Record<string, unknown>>;
117
+ authSetup(): Promise<void>;
118
+ announce(bytesPerChannel: number, channels: number, sampleRate: number, password?: string): Promise<Response>;
119
+ setup(headers?: Record<string, string>, body?: Buffer | string | Record<string, unknown>): Promise<Response>;
120
+ record(headers?: Record<string, string>): Promise<void>;
121
+ flush(options: {
122
+ headers: Record<string, string>;
123
+ }): Promise<void>;
124
+ setParameter(name: string, value: string): Promise<void>;
125
+ setMetadata(session: string, rtpseq: number, rtptime: number, metadata: MediaMetadata): Promise<void>;
126
+ setArtwork(session: string, rtpseq: number, rtptime: number, artwork: Buffer): Promise<void>;
127
+ feedback(allowError?: boolean): Promise<Response>;
128
+ teardown(session: string): Promise<void>;
129
+ }
130
+ //#endregion
131
+ //#region src/statistics.d.ts
132
+ declare class Statistics {
133
+ readonly sampleRate: number;
134
+ readonly startTimeNs: bigint;
135
+ intervalTime: number;
136
+ totalFrames: number;
137
+ intervalFrames: number;
138
+ constructor(sampleRate: number);
139
+ get expectedFrameCount(): number;
140
+ get framesBehind(): number;
141
+ get intervalCompleted(): boolean;
142
+ tick(sentFrames: number): void;
143
+ newInterval(): [number, number];
144
+ }
145
+ //#endregion
146
+ //#region src/streamClient.d.ts
147
+ type EventMap$1 = {
148
+ readonly playing: [playbackInfo: PlaybackInfo];
149
+ readonly stopped: [];
150
+ };
151
+ declare class StreamClient extends EventEmitter<EventMap$1> {
152
+ #private;
153
+ get info(): Record<string, unknown>;
154
+ get playbackInfo(): PlaybackInfo;
155
+ constructor(context: Context, rtsp: RtspClient, streamContext: StreamContext, protocol: StreamProtocol, settings: Settings, timingServer: TimingServer);
156
+ close(): void;
157
+ initialize(properties: Map<string, string>): Promise<void>;
158
+ stop(): void;
159
+ setVolume(volume: number): Promise<void>;
160
+ sendAudio(source: AudioSource, metadata?: MediaMetadata, volume?: number): Promise<void>;
161
+ }
162
+ //#endregion
163
+ //#region src/raop.d.ts
164
+ type EventMap = {
165
+ readonly playing: [playbackInfo: PlaybackInfo];
166
+ readonly stopped: [];
167
+ };
168
+ type StreamOptions = {
169
+ readonly metadata?: MediaMetadata;
170
+ readonly volume?: number;
171
+ };
172
+ declare class RaopClient extends EventEmitter<EventMap> {
173
+ #private;
174
+ get context(): Context;
175
+ get deviceId(): string;
176
+ get address(): string;
177
+ get modelName(): string;
178
+ get info(): Record<string, unknown>;
179
+ private constructor();
180
+ stream(source: AudioSource, options?: StreamOptions): Promise<void>;
181
+ stop(): void;
182
+ setVolume(volume: number): Promise<void>;
183
+ close(): Promise<void>;
184
+ static create(discoveryResult: DiscoveryResult, timingServer: TimingServer): Promise<RaopClient>;
185
+ static discover(deviceId: string, timingServer: TimingServer): Promise<RaopClient>;
186
+ }
187
+ //#endregion
188
+ export { AudioPacketHeader, ControlClient, EncryptionType, MediaMetadata, MetadataType, PacketFifo, PlaybackInfo, RaopClient, RaopListener, RetransmitRequest, RtspClient, Settings, Statistics, StreamClient, StreamContext, type StreamOptions, StreamProtocol, SyncPacket, decodeRetransmitRequest, getAudioProperties, getEncryptionTypes, getMetadataTypes, pctToDbfs };
189
+ //# sourceMappingURL=index.d.mts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.mts","names":[],"sources":["../src/types.ts","../src/packets.ts","../src/utils.ts","../src/controlClient.ts","../src/rtspClient.ts","../src/statistics.ts","../src/streamClient.ts","../src/raop.ts"],"mappings":"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"}
package/dist/index.mjs ADDED
@@ -0,0 +1,5 @@
1
+ import{createSocket as e}from"node:dgram";import{EventEmitter as t}from"node:events";import{DAAP as n,NTP as r,Plist as i,RTSP as a}from"@basmilius/apple-encoding";import{createHash as o}from"node:crypto";import{Connection as s,Context as c,Discovery as l,HTTP_TIMEOUT as u,generateActiveRemoteId as d,generateDacpId as f,generateSessionId as p,waitFor as m}from"@basmilius/apple-common";let h=function(e){return e[e.Unknown=0]=`Unknown`,e[e.Unencrypted=1]=`Unencrypted`,e[e.MFiSAP=2]=`MFiSAP`,e}({}),g=function(e){return e[e.NotSupported=0]=`NotSupported`,e[e.Text=1]=`Text`,e[e.Artwork=2]=`Artwork`,e[e.Progress=4]=`Progress`,e}({});var _=class{#e;#t=new Map;#n=[];constructor(e){this.#e=e}get(e){return this.#t.get(e)}set(e,t){if(!this.#t.has(e))for(this.#t.set(e,t),this.#n.push(e);this.#n.length>this.#e;){let e=this.#n.shift();e!==void 0&&this.#t.delete(e)}}has(e){return this.#t.has(e)}clear(){this.#t.clear(),this.#n.length=0}};const v={encode(e,t,n,r,i){let a=Buffer.allocUnsafe(12);return a.writeUInt8(e,0),a.writeUInt8(t,1),a.writeUInt16BE(n,2),a.writeUInt32BE(r,4),a.writeUInt32BE(i,8),a}},y={encode(e,t,n,r,i,a,o){let s=Buffer.allocUnsafe(20);return s.writeUInt8(e,0),s.writeUInt8(t,1),s.writeUInt16BE(n,2),s.writeUInt32BE(r,4),s.writeUInt32BE(i,8),s.writeUInt32BE(a,12),s.writeUInt32BE(o,16),s}};function b(e){return{lostSeqno:e.readUInt16BE(4),lostPackets:e.readUInt16BE(6)}}function x(e){return e<=0?-144:e>=100?0:20*Math.log10(e/100)}function S(e){let t=e.get(`et`);if(!t)return h.Unknown;let n=h.Unknown;for(let e of t.split(`,`)){let t=parseInt(e.trim(),10);t===0&&(n|=h.Unencrypted),t===1&&(n|=h.MFiSAP)}return n}function C(e){let t=e.get(`md`);if(!t)return g.NotSupported;let n=g.NotSupported;for(let e of t.split(`,`)){let t=parseInt(e.trim(),10);t===0&&(n|=g.Text),t===1&&(n|=g.Artwork),t===2&&(n|=g.Progress)}return n}function w(e){return[parseInt(e.get(`sr`)??`44100`,10),parseInt(e.get(`ch`)??`2`,10),parseInt(e.get(`ss`)??`16`,10)/8]}function T(e,t){let n=Math.floor(e/t),r=e%t*4294967295/t;return BigInt(n)<<32n|BigInt(Math.floor(r))}var E=class extends t{#e;#t;#n;#r;#i;#a;constructor(e,t){super(),this.#t=e,this.#n=t}get port(){return this.#a??0}async bind(t,n){return new Promise((r,i)=>{this.#e=e(`udp4`),this.#e.on(`error`,e=>{console.error(`Control connection error:`,e),i(e)}),this.#e.on(`message`,(e,t)=>{this.#s(e,t)}),this.#e.on(`listening`,()=>{this.#a=this.#e.address().port,r()}),this.#e.bind(n,t)})}close(){this.stop(),this.#e&&=(this.#e.close(),void 0)}start(e){if(this.#r)throw Error(`Already running`);this.#i=new AbortController,this.#o(e,this.#t.controlPort)}stop(){this.#i&&=(this.#i.abort(),void 0),this.#r&&=(clearInterval(this.#r),void 0)}#o(e,t){let n=!0,i=()=>{if(!this.#e)return;let i=T(this.#t.headTs,this.#t.sampleRate),[a,o]=r.parts(i),s=y.encode(n?144:128,212,7,this.#t.headTs-this.#t.latency,a,o,this.#t.headTs);n=!1,this.#e.send(s,t,e)};i(),this.#r=setInterval(i,1e3)}#s(e,t){(e[1]&127)==85?this.#c(b(e),t):console.debug(`Received unhandled control data from`,t,e)}#c(e,t){for(let n=0;n<e.lostPackets;n++){let r=e.lostSeqno+n;if(this.#n.has(r)){let e=this.#n.get(r),n=e.subarray(2,4),i=Buffer.concat([Buffer.from([128,214]),n,e]);this.#e&&this.#e.send(i,t.port,t.address)}else console.debug(`Packet ${r} not in backlog`)}}};const D=Buffer.from([1]),O=Buffer.from([89,2,237,233,13,78,242,189,76,182,138,99,48,3,130,7,169,77,189,80,216,170,70,91,93,140,1,42,12,126,29,78]);function k(e,t,n){let r=o(`md5`).update(`${n.username}:${n.realm}:${n.password}`).digest(`hex`),i=o(`md5`).update(`${e}:${t}`).digest(`hex`),a=o(`md5`).update(`${r}:${n.nonce}:${i}`).digest(`hex`);return`Digest username="${n.username}", realm="${n.realm}", nonce="${n.nonce}", uri="${t}", response="${a}"`}function A(){return Math.floor(Math.random()*4294967295)}function j(e){return[`v=0`,`o=iTunes ${e.sessionId} 0 IN IP4 ${e.localIp}`,`s=iTunes`,`c=IN IP4 ${e.remoteIp}`,`t=0 0`,`m=audio 0 RTP/AVP 96`,`a=rtpmap:96 L16/${e.sampleRate}/${e.channels}`,`a=fmtp:96 352 0 ${e.bitsPerChannel} 40 10 14 ${e.channels} 255 0 0 ${e.sampleRate}`].join(`\r
2
+ `)+`\r
3
+ `}var M=class extends s{get activeRemoteId(){return this.#e}get dacpId(){return this.#t}get rtspSessionId(){return this.#n}get sessionId(){return this.#r}get uri(){return`rtsp://${this.connection.localIp}/${this.#r}`}get connection(){return{localIp:this.#i,remoteIp:this.address}}#e;#t;#n;#r;#i=`0.0.0.0`;#a=Buffer.alloc(0);#o=0;#s;#c=new Map;constructor(e,t,n){super(e,t,n),this.#e=d(),this.#t=f(),this.#n=p(),this.#r=A(),this.on(`close`,this.#d.bind(this)),this.on(`data`,this.#f.bind(this)),this.on(`error`,this.#p.bind(this)),this.on(`timeout`,this.#m.bind(this)),this.on(`connect`,this.#u.bind(this))}async info(){try{let e=await this.#l(`GET`,`/info`,{allowError:!0});if(e.ok){let t=Buffer.from(await e.arrayBuffer());if(t.length>0)try{return i.parse(t.buffer)}catch{return{}}}return{}}catch{return{}}}async authSetup(){let e=Buffer.concat([D,O]);await this.#l(`POST`,`/auth-setup`,{contentType:`application/octet-stream`,body:e,protocol:`HTTP/1.1`})}async announce(e,t,n,r){let i=j({sessionId:this.#r,localIp:this.connection.localIp,remoteIp:this.connection.remoteIp,bitsPerChannel:8*e,channels:t,sampleRate:n}),a=await this.#l(`ANNOUNCE`,void 0,{contentType:`application/sdp`,body:i,allowError:!!r});if(a.status===401&&r){let e=a.headers.get(`www-authenticate`);if(e){let t=e.split(`"`);t.length>=5&&(this.#s={username:`pyatv`,realm:t[1],password:r,nonce:t[3]},a=await this.#l(`ANNOUNCE`,void 0,{contentType:`application/sdp`,body:i}))}}return a}async setup(e,t){return await this.#l(`SETUP`,void 0,{headers:e,body:t})}async record(e){await this.#l(`RECORD`,void 0,{headers:e})}async flush(e){await this.#l(`FLUSH`,void 0,{headers:e.headers})}async setParameter(e,t){await this.#l(`SET_PARAMETER`,void 0,{contentType:`text/parameters`,body:`${e}: ${t}`})}async setMetadata(e,t,r,i){let a=n.encodeTrackMetadata({title:i.title,artist:i.artist,album:i.album,duration:i.duration});await this.#l(`SET_PARAMETER`,void 0,{contentType:`application/x-dmap-tagged`,headers:{Session:e,"RTP-Info":`seq=${t};rtptime=${r}`},body:a})}async setArtwork(e,t,n,r){let i=`image/jpeg`;r[0]===137&&r[1]===80&&(i=`image/png`),await this.#l(`SET_PARAMETER`,void 0,{contentType:i,headers:{Session:e,"RTP-Info":`seq=${t};rtptime=${n}`},body:r})}async feedback(e=!1){return await this.#l(`POST`,`/feedback`,{allowError:e})}async teardown(e){await this.#l(`TEARDOWN`,void 0,{headers:{Session:e}})}async#l(e,t,n={}){let{contentType:r,headers:a={},allowError:o=!1,protocol:s=`RTSP/1.0`,timeout:c=u}=n,{body:l}=n,d=this.#o++,f=t??this.uri,p={CSeq:d,"DACP-ID":this.#t,"Active-Remote":this.#e,"Client-Instance":this.#t,"User-Agent":`AirPlay/550.10`};this.#s&&(p.Authorization=k(e,f,this.#s)),Object.assign(p,a),l&&typeof l==`object`&&!Buffer.isBuffer(l)?(p[`Content-Type`]=`application/x-apple-binary-plist`,l=Buffer.from(i.serialize(l))):r&&(p[`Content-Type`]=r);let m;l?(m=typeof l==`string`?Buffer.from(l):l,p[`Content-Length`]=m.length):p[`Content-Length`]=0;let h=[`${e} ${f} ${s}`,...Object.entries(p).map(([e,t])=>`${e}: ${t}`),``,``].join(`\r
4
+ `),g=m?Buffer.concat([Buffer.from(h),m]):Buffer.from(h);return this.context.logger.net(`[rtsp]`,e,f,`cseq=${d}`),new Promise((e,t)=>{this.#c.set(d,{resolve:e,reject:t});let n=setTimeout(()=>{this.#c.delete(d),t(Error(`No response to CSeq ${d} (${f})`))},c);this.write(g);let r=e;this.#c.set(d,{resolve:e=>{clearTimeout(n),!o&&!e.ok?t(Error(`RTSP error: ${e.status} ${e.statusText}`)):r(e)},reject:e=>{clearTimeout(n),t(e)}})})}#u(){this.#i=`0.0.0.0`}#d(){this.#a=Buffer.alloc(0);for(let[e,{reject:t}]of this.#c)t(Error(`Connection closed`)),this.#c.delete(e);this.context.logger.net(`[rtsp]`,`#onClose()`)}#f(e){try{for(this.#a=Buffer.concat([this.#a,e]);this.#a.byteLength>0;){let e=a.makeResponse(this.#a);if(e===null)return;this.#a=this.#a.subarray(e.responseLength);let t=e.response.headers.get(`CSeq`),n=t?parseInt(t,10):-1;if(this.#c.has(n)){let{resolve:t}=this.#c.get(n);this.#c.delete(n),t(e.response)}else this.context.logger.warn(`[rtsp]`,`Unexpected response for CSeq ${n}`)}}catch(e){this.context.logger.error(`[rtsp]`,`#onData()`,e),this.emit(`error`,e)}}#p(e){for(let[t,{reject:n}]of this.#c)n(e),this.#c.delete(t);this.context.logger.error(`[rtsp]`,`#onError()`,e)}#m(){let e=Error(`Connection timed out`);for(let[t,{reject:n}]of this.#c)n(e),this.#c.delete(t);this.context.logger.net(`[rtsp]`,`#onTimeout()`)}},N=class{sampleRate;startTimeNs;intervalTime;totalFrames=0;intervalFrames=0;constructor(e){this.sampleRate=e,this.startTimeNs=process.hrtime.bigint(),this.intervalTime=performance.now()}get expectedFrameCount(){let e=Number(process.hrtime.bigint()-this.startTimeNs);return Math.floor(e/(1e9/this.sampleRate))}get framesBehind(){return this.expectedFrameCount-this.totalFrames}get intervalCompleted(){return this.intervalFrames>=this.sampleRate}tick(e){this.totalFrames+=e,this.intervalFrames+=e}newInterval(){let e=performance.now(),t=(e-this.intervalTime)/1e3;this.intervalTime=e;let n=this.intervalFrames;return this.intervalFrames=0,[t,n]}};const P={title:`Streaming with apple-raop`,artist:`apple-raop`,album:`AirPlay`,duration:0},F={title:``,artist:``,album:``,duration:0},I=h.Unencrypted|h.MFiSAP;var L=class extends t{get info(){return this.#f}get playbackInfo(){return{metadata:this.#v(this.#d)?P:this.#d,position:this.#r.position}}get#e(){let e=this.#p.get(`am`)??``;return(this.#l&h.MFiSAP)!==0&&e.startsWith(`AirPort`)}#t;#n;#r;#i;#a;#o;#s;#c;#l=h.Unknown;#u=g.NotSupported;#d=F;#f={};#p=new Map;#m=!1;constructor(e,t,n,r,i,a){super(),this.#t=e,this.#n=t,this.#r=n,this.#a=r,this.#i=i,this.#o=new _(1e3),this.#s=a}close(){this.#a.teardown(),this.#c?.close()}async initialize(e){this.#p=e,this.#l=S(e),this.#u=C(e),this.#t.logger.info(`Initializing RTSP with encryption=${this.#l}, metadata=${this.#u}`);let t=this.#l&I;(!t||t===h.Unknown)&&this.#t.logger.debug(`No supported encryption type, continuing anyway`),this.#y(e),this.#c=new E(this.#r,this.#o),await this.#c.bind(this.#n.connection.localIp,this.#i.protocols.raop.controlPort),this.#t.logger.debug(`Local ports: control=${this.#c.port}, timing=${this.#s.port}`);let n=await this.#n.info();Object.assign(this.#f,n),this.#t.logger.debug(`Updated info parameters to:`,this.#f),this.#e&&await this.#n.authSetup(),await this.#a.setup(this.#s.port,this.#c.port)}stop(){this.#t.logger.debug(`Stopping audio playback`),this.#m=!1}async setVolume(e){await this.#n.setParameter(`volume`,String(e)),this.#r.volume=e}async sendAudio(t,n=F,r){if(!this.#c)throw Error(`Not initialized`);this.#r.reset();let i;try{if(i=e(`udp4`),await new Promise(e=>{i.connect(this.#r.serverPort,this.#n.connection.remoteIp,e)}),this.#c.start(this.#n.connection.remoteIp),(this.#u&g.Progress)!==0){let e=this.#r.rtptime,n=this.#r.rtptime,r=e+t.duration*this.#r.sampleRate;await this.#n.setParameter(`progress`,`${e}/${n}/${r}`)}this.#d=n,(this.#u&g.Text)!==0&&(this.#t.logger.debug(`Playing with metadata:`,this.playbackInfo.metadata),await this.#n.setMetadata(this.#r.rtspSession,this.#r.rtpseq,this.#r.rtptime,this.playbackInfo.metadata)),(this.#u&g.Artwork)!==0&&n.artwork&&(this.#t.logger.debug(`Sending ${n.artwork.length} bytes artwork`),await this.#n.setArtwork(this.#r.rtspSession,this.#r.rtpseq,this.#r.rtptime,n.artwork)),await this.#a.startFeedback(),this.emit(`playing`,this.playbackInfo),await this.#n.record({Range:`npt=0-`,Session:this.#r.rtspSession,"RTP-Info":`seq=${this.#r.rtpseq};rtptime=${this.#r.rtptime}`}),await this.#n.flush({headers:{Range:`npt=0-`,Session:this.#r.rtspSession,"RTP-Info":`seq=${this.#r.rtpseq};rtptime=${this.#r.rtptime}`}}),r!==void 0&&await this.setVolume(x(r)),await this.#h(t,i)}catch(e){throw this.#t.logger.error(`An error occurred during streaming.`,e),Error(`An error occurred during streaming: ${e}`)}finally{this.#o.clear(),i&&(await this.#n.teardown(this.#r.rtspSession),i.close()),this.#a.teardown(),this.close(),this.emit(`stopped`)}}async#h(e,t){let n=new N(this.#r.sampleRate),r=performance.now(),i=null,a=0;for(this.#m=!0;this.#m;){let o=this.#r.rtpseq-1,s=await this.#g(e,n.totalFrames===0,t);if(s===0)break;n.tick(s);let c=n.framesBehind;if(c>=352){let r=Math.min(Math.floor(c/352),3);this.#t.logger.debug(`Compensating with ${r} packets (${c} frames behind)`);let[i,a]=await this.#_(e,t,r);if(n.tick(i),!a)break}if(n.intervalCompleted){let[e,t]=n.newInterval();this.#t.logger.debug(`Sent ${t} frames in ${e.toFixed(3)}s (current frames: ${n.totalFrames}, expected: ${n.expectedFrameCount})`)}let l=n.totalFrames/this.#r.sampleRate,u=(performance.now()-r)/1e3,d=l-u;d>0?(a=0,await m(d*1e3)):(i===o-1&&a++,a>=5?this.#t.logger.warn(`Too slow to keep up for seqno ${o} (${l.toFixed(3)} vs ${u.toFixed(3)} => ${d.toFixed(3)})`):this.#t.logger.debug(`Too slow to keep up for seqno ${o} (${l.toFixed(3)} vs ${u.toFixed(3)} => ${d.toFixed(3)})`),i=o)}let o=Number(process.hrtime.bigint()-n.startTimeNs);this.#t.logger.debug(`Audio finished sending in ${(o/1e9).toFixed(3)}s`)}async#g(e,t,n){if(this.#r.paddingSent>=this.#r.latency)return 0;let r=await e.readFrames(352);if(!r)r=Buffer.alloc(this.#r.packetSize),this.#r.paddingSent+=Math.floor(r.length/this.#r.frameSize);else if(r.length!==this.#r.packetSize){let e=Buffer.alloc(this.#r.packetSize);r.copy(e),r=e}let i=v.encode(128,t?224:96,this.#r.rtpseq,this.#r.headTs,this.#n.sessionId),[a,o]=await this.#a.sendAudioPacket(n,i,r);return this.#o.set(a,o),this.#r.rtpseq=(this.#r.rtpseq+1)%2**16,this.#r.headTs+=Math.floor(r.length/this.#r.frameSize),Math.floor(r.length/this.#r.frameSize)}async#_(e,t,n){let r=0;for(let i=0;i<n;i++){let n=await this.#g(e,!1,t);if(r+=n,n===0)return[r,!1]}return[r,!0]}#v(e){return e.title===``&&e.artist===``&&e.album===``&&e.duration===0}#y(e){let[t,n,r]=w(e);this.#r.sampleRate=t,this.#r.channels=n,this.#r.bytesPerChannel=r,this.#t.logger.debug(`Update play settings to ${t}/${n}/${r*8}bit`)}};const R=44100;var z=class e extends t{get context(){return this.#e}get deviceId(){return this.#r.id}get address(){return this.#r.address}get modelName(){return this.#r.modelName}get info(){return this.#n.info}#e;#t;#n;#r;constructor(e,t,n,r){super(),this.#e=e,this.#t=t,this.#n=n,this.#r=r,this.#n.on(`playing`,e=>this.emit(`playing`,e)),this.#n.on(`stopped`,()=>this.emit(`stopped`))}async stream(e,t={}){await e.start();try{await this.#n.sendAudio(e,t.metadata,t.volume)}finally{await e.stop()}}stop(){this.#n.stop()}async setVolume(e){await this.#n.setVolume(e)}async close(){this.#n.close(),await this.#t.disconnect()}static async create(t,n){let r=new c(t.id),i=new M(r,t.address,t.service.port);await i.connect();let a=V();a.rtspSession=i.rtspSessionId;let o=new L(r,i,a,new B(i,a),{protocols:{raop:{controlPort:0,timingPort:0}}},n),s=new Map(Object.entries(t.txt));return await o.initialize(s),new e(r,i,o,t)}static async discover(t,n){let r=await l.raop().findUntil(t);return e.create(r,n)}},B=class{#e;#t;#n;constructor(e,t){this.#e=e,this.#t=t}async setup(e,t){await this.#e.announce(this.#t.bytesPerChannel,this.#t.channels,this.#t.sampleRate);let n=[`RTP/AVP/UDP`,`unicast`,`interleaved=0-1`,`mode=record`,`control_port=${t}`,`timing_port=${e}`].filter(Boolean).join(`;`),r=(await this.#e.setup({Transport:n})).headers.get(`Transport`);if(!r)return;let i=r.match(/server_port=(\d+)/);i&&(this.#t.serverPort=parseInt(i[1],10));let a=r.match(/control_port=(\d+)/);a&&(this.#t.controlPort=parseInt(a[1],10))}async startFeedback(){this.#n=setInterval(async()=>{try{await this.#e.feedback(!0)}catch{}},2e3)}async sendAudioPacket(e,t,n){let r=Buffer.concat([t,n]),i=t.readUInt16BE(2);return await new Promise((t,n)=>{e.send(r,e=>e?n(e):t())}),[i,r]}teardown(){this.#n&&=(clearInterval(this.#n),void 0)}};function V(){return{sampleRate:R,channels:2,bytesPerChannel:2,rtpseq:Math.floor(Math.random()*65536),rtptime:Math.floor(Math.random()*4294967295),headTs:0,latency:Math.floor(R*2),serverPort:0,controlPort:0,rtspSession:``,volume:-20,position:0,packetSize:352*2*2,frameSize:4,paddingSent:0,reset(){this.rtpseq=Math.floor(Math.random()*65536),this.rtptime=Math.floor(Math.random()*4294967295),this.headTs=this.rtptime,this.paddingSent=0,this.position=0}}}export{v as AudioPacketHeader,E as ControlClient,h as EncryptionType,g as MetadataType,_ as PacketFifo,z as RaopClient,M as RtspClient,N as Statistics,L as StreamClient,y as SyncPacket,b as decodeRetransmitRequest,w as getAudioProperties,S as getEncryptionTypes,C as getMetadataTypes,x as pctToDbfs};
5
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.mjs","names":["#maxSize","#packets","#order","#context","#packetBacklog","#localPort","#transport","#onMessage","#syncTask","#abortController","#startSyncTask","#retransmitLostPackets","FRAMES_PER_PACKET","#activeRemoteId","#dacpId","#rtspSessionId","#sessionId","#localIp","#onClose","#onData","#onError","#onTimeout","#onConnect","#exchange","#digestInfo","#cseq","#requests","#buffer","FRAMES_PER_PACKET","#info","#isMetadataEmpty","#metadata","#streamContext","#requiresAuthSetup","#properties","#encryptionTypes","#context","#rtsp","#settings","#protocol","#packetBacklog","#timingServer","#controlClient","#metadataTypes","#updateOutputProperties","#isPlaying","#streamData","#sendPacket","FRAMES_PER_PACKET","#sendNumberOfPackets","#context","#discoveryResult","#streamClient","#rtsp","#streamContext","#feedbackInterval"],"sources":["../src/types.ts","../src/packets.ts","../src/utils.ts","../src/controlClient.ts","../src/rtspClient.ts","../src/statistics.ts","../src/const.ts","../src/streamClient.ts","../src/raop.ts"],"sourcesContent":["import type { Socket as UdpSocket } from 'node:dgram';\n\nexport type MediaMetadata = {\n readonly title: string;\n readonly artist: string;\n readonly album: string;\n readonly duration: number;\n readonly artwork?: Buffer;\n}\n\nexport type PlaybackInfo = {\n readonly metadata: MediaMetadata;\n readonly position: number;\n}\n\nexport type StreamContext = {\n sampleRate: number;\n channels: number;\n bytesPerChannel: number;\n rtpseq: number;\n rtptime: number;\n headTs: number;\n latency: number;\n serverPort: number;\n controlPort: number;\n rtspSession: string;\n volume: number;\n position: number;\n packetSize: number;\n frameSize: number;\n paddingSent: number;\n\n reset(): void;\n}\n\nexport interface StreamProtocol {\n setup(timingPort: number, controlPort: number): Promise<void>;\n startFeedback(): Promise<void>;\n sendAudioPacket(transport: UdpSocket, header: Buffer, audio: Buffer): Promise<[number, Buffer]>;\n teardown(): void;\n}\n\nexport interface Settings {\n protocols: {\n raop: {\n controlPort: number;\n timingPort: number;\n };\n };\n}\n\nexport enum EncryptionType {\n Unknown = 0,\n Unencrypted = 1 << 0,\n MFiSAP = 1 << 1\n}\n\nexport enum MetadataType {\n NotSupported = 0,\n Text = 1 << 0,\n Artwork = 1 << 1,\n Progress = 1 << 2\n}\n\nexport interface RaopListener {\n playing(playbackInfo: PlaybackInfo): void;\n stopped(): void;\n}\n","export class PacketFifo {\n readonly #maxSize: number;\n readonly #packets: Map<number, Buffer> = new Map();\n readonly #order: number[] = [];\n\n constructor(maxSize: number) {\n this.#maxSize = maxSize;\n }\n\n get(seqno: number): Buffer | undefined {\n return this.#packets.get(seqno);\n }\n\n set(seqno: number, packet: Buffer): void {\n if (this.#packets.has(seqno)) {\n return;\n }\n\n this.#packets.set(seqno, packet);\n this.#order.push(seqno);\n\n while (this.#order.length > this.#maxSize) {\n const oldest = this.#order.shift();\n if (oldest !== undefined) {\n this.#packets.delete(oldest);\n }\n }\n }\n\n has(seqno: number): boolean {\n return this.#packets.has(seqno);\n }\n\n clear(): void {\n this.#packets.clear();\n this.#order.length = 0;\n }\n}\n\nexport const AudioPacketHeader = {\n encode(\n header: number,\n payloadType: number,\n seqno: number,\n timestamp: number,\n ssrc: number\n ): Buffer {\n const packet = Buffer.allocUnsafe(12);\n packet.writeUInt8(header, 0);\n packet.writeUInt8(payloadType, 1);\n packet.writeUInt16BE(seqno, 2);\n packet.writeUInt32BE(timestamp, 4);\n packet.writeUInt32BE(ssrc, 8);\n return packet;\n }\n};\n\nexport const SyncPacket = {\n encode(\n header: number,\n payloadType: number,\n seqno: number,\n rtpTimestamp: number,\n ntpSec: number,\n ntpFrac: number,\n rtpTimestampNow: number\n ): Buffer {\n const packet = Buffer.allocUnsafe(20);\n packet.writeUInt8(header, 0);\n packet.writeUInt8(payloadType, 1);\n packet.writeUInt16BE(seqno, 2);\n packet.writeUInt32BE(rtpTimestamp, 4);\n packet.writeUInt32BE(ntpSec, 8);\n packet.writeUInt32BE(ntpFrac, 12);\n packet.writeUInt32BE(rtpTimestampNow, 16);\n return packet;\n }\n};\n\nexport type RetransmitRequest = {\n readonly lostSeqno: number;\n readonly lostPackets: number;\n}\n\nexport function decodeRetransmitRequest(data: Buffer): RetransmitRequest {\n return {\n lostSeqno: data.readUInt16BE(4),\n lostPackets: data.readUInt16BE(6)\n };\n}\n","import { EncryptionType, MetadataType } from './types';\n\nexport function pctToDbfs(volume: number): number {\n if (volume <= 0) return -144;\n if (volume >= 100) return 0;\n return 20 * Math.log10(volume / 100);\n}\n\nexport function getEncryptionTypes(properties: Map<string, string>): EncryptionType {\n const et = properties.get('et');\n if (!et) return EncryptionType.Unknown;\n\n let types = EncryptionType.Unknown;\n for (const t of et.split(',')) {\n const num = parseInt(t.trim(), 10);\n if (num === 0) types |= EncryptionType.Unencrypted;\n if (num === 1) types |= EncryptionType.MFiSAP;\n }\n return types;\n}\n\nexport function getMetadataTypes(properties: Map<string, string>): MetadataType {\n const md = properties.get('md');\n if (!md) return MetadataType.NotSupported;\n\n let types = MetadataType.NotSupported;\n for (const t of md.split(',')) {\n const num = parseInt(t.trim(), 10);\n if (num === 0) types |= MetadataType.Text;\n if (num === 1) types |= MetadataType.Artwork;\n if (num === 2) types |= MetadataType.Progress;\n }\n return types;\n}\n\nexport function getAudioProperties(properties: Map<string, string>): [number, number, number] {\n const sr = parseInt(properties.get('sr') ?? '44100', 10);\n const ch = parseInt(properties.get('ch') ?? '2', 10);\n const ss = parseInt(properties.get('ss') ?? '16', 10);\n return [sr, ch, ss / 8];\n}\n","import { createSocket, type Socket as UdpSocket } from 'node:dgram';\nimport { EventEmitter } from 'node:events';\nimport { NTP } from '@basmilius/apple-encoding';\nimport { decodeRetransmitRequest, PacketFifo, SyncPacket } from './packets';\nimport type { StreamContext } from './types';\n\nfunction ntpFromTs(timestamp: number, sampleRate: number): bigint {\n const seconds = Math.floor(timestamp / sampleRate);\n const fraction = ((timestamp % sampleRate) * 0xFFFFFFFF) / sampleRate;\n\n return (BigInt(seconds) << 32n) | BigInt(Math.floor(fraction));\n}\n\nexport default class ControlClient extends EventEmitter {\n #transport?: UdpSocket;\n #context: StreamContext;\n #packetBacklog: PacketFifo;\n #syncTask?: NodeJS.Timeout;\n #abortController?: AbortController;\n #localPort?: number;\n\n constructor(context: StreamContext, packetBacklog: PacketFifo) {\n super();\n this.#context = context;\n this.#packetBacklog = packetBacklog;\n }\n\n get port(): number {\n return this.#localPort ?? 0;\n }\n\n async bind(localIp: string, port: number): Promise<void> {\n return new Promise((resolve, reject) => {\n this.#transport = createSocket('udp4');\n\n this.#transport.on('error', (err) => {\n console.error('Control connection error:', err);\n reject(err);\n });\n\n this.#transport.on('message', (data, rinfo) => {\n this.#onMessage(data, rinfo);\n });\n\n this.#transport.on('listening', () => {\n const address = this.#transport!.address();\n this.#localPort = address.port;\n resolve();\n });\n\n this.#transport.bind(port, localIp);\n });\n }\n\n close(): void {\n this.stop();\n\n if (this.#transport) {\n this.#transport.close();\n this.#transport = undefined;\n }\n }\n\n start(remoteAddr: string): void {\n if (this.#syncTask) {\n throw new Error('Already running');\n }\n\n this.#abortController = new AbortController();\n this.#startSyncTask(remoteAddr, this.#context.controlPort);\n }\n\n stop(): void {\n if (this.#abortController) {\n this.#abortController.abort();\n this.#abortController = undefined;\n }\n\n if (this.#syncTask) {\n clearInterval(this.#syncTask);\n this.#syncTask = undefined;\n }\n }\n\n #startSyncTask(addr: string, port: number): void {\n let firstPacket = true;\n\n const sendSync = () => {\n if (!this.#transport) return;\n\n const currentTime = ntpFromTs(this.#context.headTs, this.#context.sampleRate);\n const [currentSec, currentFrac] = NTP.parts(currentTime);\n\n const packet = SyncPacket.encode(\n firstPacket ? 0x90 : 0x80,\n 0xD4,\n 0x0007,\n this.#context.headTs - this.#context.latency,\n currentSec,\n currentFrac,\n this.#context.headTs\n );\n\n firstPacket = false;\n this.#transport.send(packet, port, addr);\n };\n\n sendSync();\n this.#syncTask = setInterval(sendSync, 1000);\n }\n\n #onMessage(data: Buffer, rinfo: { address: string; port: number }): void {\n const actualType = data[1] & 0x7F;\n\n if (actualType === 0x55) {\n this.#retransmitLostPackets(decodeRetransmitRequest(data), rinfo);\n } else {\n console.debug('Received unhandled control data from', rinfo, data);\n }\n }\n\n #retransmitLostPackets(request: { lostSeqno: number; lostPackets: number }, addr: { address: string; port: number }): void {\n for (let i = 0; i < request.lostPackets; i++) {\n const seqno = request.lostSeqno + i;\n if (this.#packetBacklog.has(seqno)) {\n const packet = this.#packetBacklog.get(seqno)!;\n const originalSeqno = packet.subarray(2, 4);\n const resp = Buffer.concat([Buffer.from([0x80, 0xD6]), originalSeqno, packet]);\n\n if (this.#transport) {\n this.#transport.send(resp, addr.port, addr.address);\n }\n } else {\n console.debug(`Packet ${seqno} not in backlog`);\n }\n }\n }\n}\n","import { createHash } from 'node:crypto';\nimport { Connection, type Context, generateActiveRemoteId, generateDacpId, generateSessionId, HTTP_TIMEOUT } from '@basmilius/apple-common';\nimport { DAAP, Plist, RTSP } from '@basmilius/apple-encoding';\nimport type { MediaMetadata } from './types';\n\nconst USER_AGENT = 'AirPlay/550.10';\nconst FRAMES_PER_PACKET = 352;\n\n// Used to signal that traffic is to be unencrypted\nconst AUTH_SETUP_UNENCRYPTED = Buffer.from([0x01]);\n\n// Static Curve25519 public key for auth-setup (from owntone-server)\nconst CURVE25519_PUB_KEY = Buffer.from([\n 0x59, 0x02, 0xed, 0xe9, 0x0d, 0x4e, 0xf2, 0xbd,\n 0x4c, 0xb6, 0x8a, 0x63, 0x30, 0x03, 0x82, 0x07,\n 0xa9, 0x4d, 0xbd, 0x50, 0xd8, 0xaa, 0x46, 0x5b,\n 0x5d, 0x8c, 0x01, 0x2a, 0x0c, 0x7e, 0x1d, 0x4e\n]);\n\ntype AnnouncePayloadOptions = {\n readonly sessionId: number;\n readonly localIp: string;\n readonly remoteIp: string;\n readonly bitsPerChannel: number;\n readonly channels: number;\n readonly sampleRate: number;\n};\n\ntype DigestInfo = {\n readonly username: string;\n readonly realm: string;\n readonly password: string;\n readonly nonce: string;\n}\n\nfunction getDigestPayload(method: string, uri: string, info: DigestInfo): string {\n const ha1 = createHash('md5')\n .update(`${info.username}:${info.realm}:${info.password}`)\n .digest('hex');\n\n const ha2 = createHash('md5')\n .update(`${method}:${uri}`)\n .digest('hex');\n\n const response = createHash('md5')\n .update(`${ha1}:${info.nonce}:${ha2}`)\n .digest('hex');\n\n return `Digest username=\"${info.username}\", realm=\"${info.realm}\", nonce=\"${info.nonce}\", uri=\"${uri}\", response=\"${response}\"`;\n}\n\nfunction generateRandomSessionId(): number {\n return Math.floor(Math.random() * 0xFFFFFFFF);\n}\n\nfunction buildAnnouncePayload(options: AnnouncePayloadOptions): string {\n return [\n 'v=0',\n `o=iTunes ${options.sessionId} 0 IN IP4 ${options.localIp}`,\n 's=iTunes',\n `c=IN IP4 ${options.remoteIp}`,\n 't=0 0',\n 'm=audio 0 RTP/AVP 96',\n `a=rtpmap:96 L16/${options.sampleRate}/${options.channels}`,\n `a=fmtp:96 ${FRAMES_PER_PACKET} 0 ${options.bitsPerChannel} 40 10 14 ${options.channels} 255 0 0 ${options.sampleRate}`\n ].join('\\r\\n') + '\\r\\n';\n}\n\nexport default class RtspClient extends Connection<{}> {\n get activeRemoteId(): string {\n return this.#activeRemoteId;\n }\n\n get dacpId(): string {\n return this.#dacpId;\n }\n\n get rtspSessionId(): string {\n return this.#rtspSessionId;\n }\n\n get sessionId(): number {\n return this.#sessionId;\n }\n\n get uri(): string {\n return `rtsp://${this.connection.localIp}/${this.#sessionId}`;\n }\n\n get connection(): { localIp: string; remoteIp: string } {\n return {\n localIp: this.#localIp,\n remoteIp: this.address\n };\n }\n\n readonly #activeRemoteId: string;\n readonly #dacpId: string;\n readonly #rtspSessionId: string;\n readonly #sessionId: number;\n #localIp: string = '0.0.0.0';\n #buffer: Buffer = Buffer.alloc(0);\n #cseq: number = 0;\n #digestInfo?: DigestInfo;\n #requests: Map<number, {\n resolve: (response: Response) => void;\n reject: (error: Error) => void;\n }> = new Map();\n\n constructor(context: Context, address: string, port: number) {\n super(context, address, port);\n\n this.#activeRemoteId = generateActiveRemoteId();\n this.#dacpId = generateDacpId();\n this.#rtspSessionId = generateSessionId();\n this.#sessionId = generateRandomSessionId();\n\n this.on('close', this.#onClose.bind(this));\n this.on('data', this.#onData.bind(this));\n this.on('error', this.#onError.bind(this));\n this.on('timeout', this.#onTimeout.bind(this));\n this.on('connect', this.#onConnect.bind(this));\n }\n\n async info(): Promise<Record<string, unknown>> {\n try {\n const response = await this.#exchange('GET', '/info', {\n allowError: true\n });\n\n if (response.ok) {\n const buffer = Buffer.from(await response.arrayBuffer());\n if (buffer.length > 0) {\n try {\n return Plist.parse(buffer.buffer) as Record<string, unknown>;\n } catch {\n return {};\n }\n }\n }\n\n return {};\n } catch {\n return {};\n }\n }\n\n async authSetup(): Promise<void> {\n const body = Buffer.concat([AUTH_SETUP_UNENCRYPTED, CURVE25519_PUB_KEY]);\n\n await this.#exchange('POST', '/auth-setup', {\n contentType: 'application/octet-stream',\n body,\n protocol: 'HTTP/1.1'\n });\n }\n\n async announce(bytesPerChannel: number, channels: number, sampleRate: number, password?: string): Promise<Response> {\n const body = buildAnnouncePayload({\n sessionId: this.#sessionId,\n localIp: this.connection.localIp,\n remoteIp: this.connection.remoteIp,\n bitsPerChannel: 8 * bytesPerChannel,\n channels,\n sampleRate\n });\n\n let response = await this.#exchange('ANNOUNCE', undefined, {\n contentType: 'application/sdp',\n body,\n allowError: !!password\n });\n\n // Handle password authentication\n if (response.status === 401 && password) {\n const wwwAuthenticate = response.headers.get('www-authenticate');\n\n if (wwwAuthenticate) {\n const parts = wwwAuthenticate.split('\"');\n\n if (parts.length >= 5) {\n this.#digestInfo = {\n username: 'pyatv',\n realm: parts[1],\n password,\n nonce: parts[3]\n };\n\n response = await this.#exchange('ANNOUNCE', undefined, {\n contentType: 'application/sdp',\n body\n });\n }\n }\n }\n\n return response;\n }\n\n async setup(headers?: Record<string, string>, body?: Buffer | string | Record<string, unknown>): Promise<Response> {\n return await this.#exchange('SETUP', undefined, {headers, body});\n }\n\n async record(headers?: Record<string, string>): Promise<void> {\n await this.#exchange('RECORD', undefined, {headers});\n }\n\n async flush(options: { headers: Record<string, string> }): Promise<void> {\n await this.#exchange('FLUSH', undefined, {headers: options.headers});\n }\n\n async setParameter(name: string, value: string): Promise<void> {\n await this.#exchange('SET_PARAMETER', undefined, {\n contentType: 'text/parameters',\n body: `${name}: ${value}`\n });\n }\n\n async setMetadata(session: string, rtpseq: number, rtptime: number, metadata: MediaMetadata): Promise<void> {\n const daapData = DAAP.encodeTrackMetadata({\n title: metadata.title,\n artist: metadata.artist,\n album: metadata.album,\n duration: metadata.duration\n });\n\n await this.#exchange('SET_PARAMETER', undefined, {\n contentType: 'application/x-dmap-tagged',\n headers: {\n 'Session': session,\n 'RTP-Info': `seq=${rtpseq};rtptime=${rtptime}`\n },\n body: daapData\n });\n }\n\n async setArtwork(session: string, rtpseq: number, rtptime: number, artwork: Buffer): Promise<void> {\n let contentType = 'image/jpeg';\n if (artwork[0] === 0x89 && artwork[1] === 0x50) {\n contentType = 'image/png';\n }\n\n await this.#exchange('SET_PARAMETER', undefined, {\n contentType,\n headers: {\n 'Session': session,\n 'RTP-Info': `seq=${rtpseq};rtptime=${rtptime}`\n },\n body: artwork\n });\n }\n\n async feedback(allowError: boolean = false): Promise<Response> {\n return await this.#exchange('POST', '/feedback', {allowError});\n }\n\n async teardown(session: string): Promise<void> {\n await this.#exchange('TEARDOWN', undefined, {\n headers: {'Session': session}\n });\n }\n\n async #exchange(\n method: RTSP.Method,\n uri?: string,\n options: {\n contentType?: string;\n headers?: Record<string, string>;\n body?: Buffer | string | Record<string, unknown>;\n allowError?: boolean;\n protocol?: 'RTSP/1.0' | 'HTTP/1.1';\n timeout?: number;\n } = {}\n ): Promise<Response> {\n const {\n contentType,\n headers: extraHeaders = {},\n allowError = false,\n protocol = 'RTSP/1.0',\n timeout = HTTP_TIMEOUT\n } = options;\n let {body} = options;\n\n const cseq = this.#cseq++;\n const targetUri = uri ?? this.uri;\n\n const headers: Record<string, string | number> = {\n 'CSeq': cseq,\n 'DACP-ID': this.#dacpId,\n 'Active-Remote': this.#activeRemoteId,\n 'Client-Instance': this.#dacpId,\n 'User-Agent': USER_AGENT\n };\n\n if (this.#digestInfo) {\n headers['Authorization'] = getDigestPayload(method, targetUri, this.#digestInfo);\n }\n\n Object.assign(headers, extraHeaders);\n\n if (body && typeof body === 'object' && !Buffer.isBuffer(body)) {\n headers['Content-Type'] = 'application/x-apple-binary-plist';\n body = Buffer.from(Plist.serialize(body as {}));\n } else if (contentType) {\n headers['Content-Type'] = contentType;\n }\n\n let bodyBuffer: Buffer | undefined;\n if (body) {\n bodyBuffer = typeof body === 'string' ? Buffer.from(body) : body as Buffer;\n headers['Content-Length'] = bodyBuffer.length;\n } else {\n headers['Content-Length'] = 0;\n }\n\n const headerLines = [\n `${method} ${targetUri} ${protocol}`,\n ...Object.entries(headers).map(([k, v]) => `${k}: ${v}`),\n '',\n ''\n ].join('\\r\\n');\n\n const data = bodyBuffer\n ? Buffer.concat([Buffer.from(headerLines), bodyBuffer])\n : Buffer.from(headerLines);\n\n this.context.logger.net('[rtsp]', method, targetUri, `cseq=${cseq}`);\n\n return new Promise((resolve, reject) => {\n this.#requests.set(cseq, {resolve, reject});\n\n const timer = setTimeout(() => {\n this.#requests.delete(cseq);\n reject(new Error(`No response to CSeq ${cseq} (${targetUri})`));\n }, timeout);\n\n this.write(data);\n\n const originalResolve = resolve;\n\n this.#requests.set(cseq, {\n resolve: (response) => {\n clearTimeout(timer);\n if (!allowError && !response.ok) {\n reject(new Error(`RTSP error: ${response.status} ${response.statusText}`));\n } else {\n originalResolve(response);\n }\n },\n reject: (error) => {\n clearTimeout(timer);\n reject(error);\n }\n });\n });\n }\n\n #onConnect(): void {\n this.#localIp = '0.0.0.0';\n }\n\n #onClose(): void {\n this.#buffer = Buffer.alloc(0);\n\n for (const [cseq, {reject}] of this.#requests) {\n reject(new Error('Connection closed'));\n this.#requests.delete(cseq);\n }\n\n this.context.logger.net('[rtsp]', '#onClose()');\n }\n\n #onData(data: Buffer): void {\n try {\n this.#buffer = Buffer.concat([this.#buffer, data]);\n\n while (this.#buffer.byteLength > 0) {\n const result = RTSP.makeResponse(this.#buffer);\n\n if (result === null) {\n return;\n }\n\n this.#buffer = this.#buffer.subarray(result.responseLength);\n\n const cseqHeader = result.response.headers.get('CSeq');\n const cseq = cseqHeader ? parseInt(cseqHeader, 10) : -1;\n\n if (this.#requests.has(cseq)) {\n const {resolve} = this.#requests.get(cseq)!;\n this.#requests.delete(cseq);\n resolve(result.response);\n } else {\n this.context.logger.warn('[rtsp]', `Unexpected response for CSeq ${cseq}`);\n }\n }\n } catch (err) {\n this.context.logger.error('[rtsp]', '#onData()', err);\n this.emit('error', err as Error);\n }\n }\n\n #onError(err: Error): void {\n for (const [cseq, {reject}] of this.#requests) {\n reject(err);\n this.#requests.delete(cseq);\n }\n\n this.context.logger.error('[rtsp]', '#onError()', err);\n }\n\n #onTimeout(): void {\n const err = new Error('Connection timed out');\n\n for (const [cseq, {reject}] of this.#requests) {\n reject(err);\n this.#requests.delete(cseq);\n }\n\n this.context.logger.net('[rtsp]', '#onTimeout()');\n }\n}\n","export default class Statistics {\n readonly sampleRate: number;\n readonly startTimeNs: bigint;\n intervalTime: number;\n totalFrames: number = 0;\n intervalFrames: number = 0;\n\n constructor(sampleRate: number) {\n this.sampleRate = sampleRate;\n this.startTimeNs = process.hrtime.bigint();\n this.intervalTime = performance.now();\n }\n\n get expectedFrameCount(): number {\n const elapsedNs = Number(process.hrtime.bigint() - this.startTimeNs);\n\n return Math.floor(elapsedNs / (1e9 / this.sampleRate));\n }\n\n get framesBehind(): number {\n return this.expectedFrameCount - this.totalFrames;\n }\n\n get intervalCompleted(): boolean {\n return this.intervalFrames >= this.sampleRate;\n }\n\n tick(sentFrames: number): void {\n this.totalFrames += sentFrames;\n this.intervalFrames += sentFrames;\n }\n\n newInterval(): [number, number] {\n const endTime = performance.now();\n const diff = (endTime - this.intervalTime) / 1000;\n this.intervalTime = endTime;\n\n const frames = this.intervalFrames;\n this.intervalFrames = 0;\n\n return [diff, frames];\n }\n}\n","import { EncryptionType, type MediaMetadata } from './types';\n\nexport const MAX_PACKETS_COMPENSATE = 3;\nexport const PACKET_BACKLOG_SIZE = 1000;\nexport const SLOW_WARNING_THRESHOLD = 5;\nexport const FRAMES_PER_PACKET = 352;\n\nexport const MISSING_METADATA: MediaMetadata = {\n title: 'Streaming with apple-raop',\n artist: 'apple-raop',\n album: 'AirPlay',\n duration: 0\n};\n\nexport const EMPTY_METADATA: MediaMetadata = {\n title: '',\n artist: '',\n album: '',\n duration: 0\n};\n\nexport const SUPPORTED_ENCRYPTIONS = EncryptionType.Unencrypted | EncryptionType.MFiSAP;\n","import { createSocket, type Socket as UdpSocket } from 'node:dgram';\nimport { EventEmitter } from 'node:events';\nimport { type AudioSource, type Context, type TimingServer, waitFor } from '@basmilius/apple-common';\nimport { EMPTY_METADATA, FRAMES_PER_PACKET, MAX_PACKETS_COMPENSATE, MISSING_METADATA, PACKET_BACKLOG_SIZE, SLOW_WARNING_THRESHOLD, SUPPORTED_ENCRYPTIONS } from './const';\nimport { AudioPacketHeader, PacketFifo } from './packets';\nimport { EncryptionType, type MediaMetadata, MetadataType, type PlaybackInfo, type Settings, type StreamContext, type StreamProtocol } from './types';\nimport { getAudioProperties, getEncryptionTypes, getMetadataTypes, pctToDbfs } from './utils';\nimport ControlClient from './controlClient';\nimport Statistics from './statistics';\nimport RtspClient from './rtspClient';\n\nexport type EventMap = {\n readonly playing: [playbackInfo: PlaybackInfo];\n readonly stopped: [];\n};\n\nexport default class StreamClient extends EventEmitter<EventMap> {\n get info(): Record<string, unknown> {\n return this.#info;\n }\n\n get playbackInfo(): PlaybackInfo {\n return {\n metadata: this.#isMetadataEmpty(this.#metadata) ? MISSING_METADATA : this.#metadata,\n position: this.#streamContext.position\n };\n }\n\n get #requiresAuthSetup(): boolean {\n const modelName = this.#properties.get('am') ?? '';\n\n return (\n (this.#encryptionTypes & EncryptionType.MFiSAP) !== 0\n && modelName.startsWith('AirPort')\n );\n }\n\n readonly #context: Context;\n readonly #rtsp: RtspClient;\n readonly #streamContext: StreamContext;\n readonly #settings: Settings;\n readonly #protocol: StreamProtocol;\n readonly #packetBacklog: PacketFifo;\n readonly #timingServer: TimingServer;\n\n #controlClient?: ControlClient;\n #encryptionTypes: EncryptionType = EncryptionType.Unknown;\n #metadataTypes: MetadataType = MetadataType.NotSupported;\n #metadata: MediaMetadata = EMPTY_METADATA;\n #info: Record<string, unknown> = {};\n #properties: Map<string, string> = new Map();\n #isPlaying: boolean = false;\n\n constructor(context: Context, rtsp: RtspClient, streamContext: StreamContext, protocol: StreamProtocol, settings: Settings, timingServer: TimingServer) {\n super();\n\n this.#context = context;\n this.#rtsp = rtsp;\n this.#streamContext = streamContext;\n this.#protocol = protocol;\n this.#settings = settings;\n this.#packetBacklog = new PacketFifo(PACKET_BACKLOG_SIZE);\n this.#timingServer = timingServer;\n }\n\n close(): void {\n this.#protocol.teardown();\n this.#controlClient?.close();\n }\n\n async initialize(properties: Map<string, string>): Promise<void> {\n this.#properties = properties;\n this.#encryptionTypes = getEncryptionTypes(properties);\n this.#metadataTypes = getMetadataTypes(properties);\n\n this.#context.logger.info(`Initializing RTSP with encryption=${this.#encryptionTypes}, metadata=${this.#metadataTypes}`);\n\n const intersection = this.#encryptionTypes & SUPPORTED_ENCRYPTIONS;\n if (!intersection || intersection === EncryptionType.Unknown) {\n this.#context.logger.debug('No supported encryption type, continuing anyway');\n }\n\n this.#updateOutputProperties(properties);\n\n this.#controlClient = new ControlClient(this.#streamContext, this.#packetBacklog);\n await this.#controlClient.bind(\n this.#rtsp.connection.localIp,\n this.#settings.protocols.raop.controlPort\n );\n\n this.#context.logger.debug(`Local ports: control=${this.#controlClient.port}, timing=${this.#timingServer.port}`);\n\n const info = await this.#rtsp.info();\n Object.assign(this.#info, info);\n this.#context.logger.debug('Updated info parameters to:', this.#info);\n\n if (this.#requiresAuthSetup) {\n await this.#rtsp.authSetup();\n }\n\n await this.#protocol.setup(this.#timingServer.port, this.#controlClient.port);\n }\n\n stop(): void {\n this.#context.logger.debug('Stopping audio playback');\n this.#isPlaying = false;\n }\n\n async setVolume(volume: number): Promise<void> {\n await this.#rtsp.setParameter('volume', String(volume));\n this.#streamContext.volume = volume;\n }\n\n async sendAudio(source: AudioSource, metadata: MediaMetadata = EMPTY_METADATA, volume?: number): Promise<void> {\n if (!this.#controlClient) {\n throw new Error('Not initialized');\n }\n\n this.#streamContext.reset();\n\n let transport: UdpSocket | undefined;\n\n try {\n transport = createSocket('udp4');\n await new Promise<void>((resolve) => {\n transport!.connect(this.#streamContext.serverPort, this.#rtsp.connection.remoteIp, resolve);\n });\n\n this.#controlClient.start(this.#rtsp.connection.remoteIp);\n\n if ((this.#metadataTypes & MetadataType.Progress) !== 0) {\n const start = this.#streamContext.rtptime;\n const now = this.#streamContext.rtptime;\n const end = start + source.duration * this.#streamContext.sampleRate;\n await this.#rtsp.setParameter('progress', `${start}/${now}/${end}`);\n }\n\n this.#metadata = metadata;\n\n if ((this.#metadataTypes & MetadataType.Text) !== 0) {\n this.#context.logger.debug('Playing with metadata:', this.playbackInfo.metadata);\n await this.#rtsp.setMetadata(\n this.#streamContext.rtspSession,\n this.#streamContext.rtpseq,\n this.#streamContext.rtptime,\n this.playbackInfo.metadata\n );\n }\n\n if ((this.#metadataTypes & MetadataType.Artwork) !== 0 && metadata.artwork) {\n this.#context.logger.debug(`Sending ${metadata.artwork.length} bytes artwork`);\n\n await this.#rtsp.setArtwork(\n this.#streamContext.rtspSession,\n this.#streamContext.rtpseq,\n this.#streamContext.rtptime,\n metadata.artwork\n );\n }\n\n await this.#protocol.startFeedback();\n\n this.emit('playing', this.playbackInfo);\n\n await this.#rtsp.record({\n 'Range': 'npt=0-',\n 'Session': this.#streamContext.rtspSession,\n 'RTP-Info': `seq=${this.#streamContext.rtpseq};rtptime=${this.#streamContext.rtptime}`\n });\n\n await this.#rtsp.flush({\n headers: {\n 'Range': 'npt=0-',\n 'Session': this.#streamContext.rtspSession,\n 'RTP-Info': `seq=${this.#streamContext.rtpseq};rtptime=${this.#streamContext.rtptime}`\n }\n });\n\n if (volume !== undefined) {\n await this.setVolume(pctToDbfs(volume));\n }\n\n await this.#streamData(source, transport);\n } catch (err) {\n this.#context.logger.error('An error occurred during streaming.', err);\n throw new Error(`An error occurred during streaming: ${err}`);\n } finally {\n this.#packetBacklog.clear();\n\n if (transport) {\n await this.#rtsp.teardown(this.#streamContext.rtspSession);\n transport.close();\n }\n\n this.#protocol.teardown();\n this.close();\n\n this.emit('stopped');\n }\n }\n\n async #streamData(source: AudioSource, transport: UdpSocket): Promise<void> {\n const stats = new Statistics(this.#streamContext.sampleRate);\n\n const initialTime = performance.now();\n let prevSlowSeqno: number | null = null;\n let numberSlowSeqno = 0;\n\n this.#isPlaying = true;\n\n while (this.#isPlaying) {\n const currentSeqno = this.#streamContext.rtpseq - 1;\n const numSent = await this.#sendPacket(source, stats.totalFrames === 0, transport);\n\n if (numSent === 0) {\n break;\n }\n\n stats.tick(numSent);\n const framesBehind = stats.framesBehind;\n\n if (framesBehind >= FRAMES_PER_PACKET) {\n const maxPackets = Math.min(\n Math.floor(framesBehind / FRAMES_PER_PACKET),\n MAX_PACKETS_COMPENSATE\n );\n\n this.#context.logger.debug(\n `Compensating with ${maxPackets} packets (${framesBehind} frames behind)`\n );\n\n const [sentFrames, hasMorePackets] = await this.#sendNumberOfPackets(\n source,\n transport,\n maxPackets\n );\n stats.tick(sentFrames);\n\n if (!hasMorePackets) {\n break;\n }\n }\n\n if (stats.intervalCompleted) {\n const [intervalTime, intervalFrames] = stats.newInterval();\n this.#context.logger.debug(\n `Sent ${intervalFrames} frames in ${intervalTime.toFixed(3)}s (current frames: ${stats.totalFrames}, expected: ${stats.expectedFrameCount})`\n );\n }\n\n const absTimeStream = stats.totalFrames / this.#streamContext.sampleRate;\n const relToStart = (performance.now() - initialTime) / 1000;\n const diff = absTimeStream - relToStart;\n\n if (diff > 0) {\n numberSlowSeqno = 0;\n await waitFor(diff * 1000);\n } else {\n if (prevSlowSeqno === currentSeqno - 1) {\n numberSlowSeqno++;\n }\n\n if (numberSlowSeqno >= SLOW_WARNING_THRESHOLD) {\n this.#context.logger.warn(`Too slow to keep up for seqno ${currentSeqno} (${absTimeStream.toFixed(3)} vs ${relToStart.toFixed(3)} => ${diff.toFixed(3)})`);\n } else {\n this.#context.logger.debug(`Too slow to keep up for seqno ${currentSeqno} (${absTimeStream.toFixed(3)} vs ${relToStart.toFixed(3)} => ${diff.toFixed(3)})`);\n }\n\n prevSlowSeqno = currentSeqno;\n }\n }\n\n const elapsedNs = Number(process.hrtime.bigint() - stats.startTimeNs);\n this.#context.logger.debug(`Audio finished sending in ${(elapsedNs / 1e9).toFixed(3)}s`);\n }\n\n async #sendPacket(source: AudioSource, firstPacket: boolean, transport: UdpSocket): Promise<number> {\n if (this.#streamContext.paddingSent >= this.#streamContext.latency) {\n return 0;\n }\n\n let frames = await source.readFrames(FRAMES_PER_PACKET);\n\n if (!frames) {\n frames = Buffer.alloc(this.#streamContext.packetSize);\n this.#streamContext.paddingSent += Math.floor(frames.length / this.#streamContext.frameSize);\n } else if (frames.length !== this.#streamContext.packetSize) {\n const padded = Buffer.alloc(this.#streamContext.packetSize);\n frames.copy(padded);\n frames = padded;\n }\n\n const header = AudioPacketHeader.encode(\n 0x80,\n firstPacket ? 0xE0 : 0x60,\n this.#streamContext.rtpseq,\n this.#streamContext.headTs,\n this.#rtsp.sessionId\n );\n\n const [rtpseq, packet] = await this.#protocol.sendAudioPacket(transport, header, frames);\n this.#packetBacklog.set(rtpseq, packet);\n\n this.#streamContext.rtpseq = (this.#streamContext.rtpseq + 1) % (2 ** 16);\n this.#streamContext.headTs += Math.floor(frames.length / this.#streamContext.frameSize);\n\n return Math.floor(frames.length / this.#streamContext.frameSize);\n }\n\n async #sendNumberOfPackets(source: AudioSource, transport: UdpSocket, count: number): Promise<[number, boolean]> {\n let totalFrames = 0;\n\n for (let i = 0; i < count; i++) {\n const sent = await this.#sendPacket(source, false, transport);\n totalFrames += sent;\n\n if (sent === 0) {\n return [totalFrames, false];\n }\n }\n\n return [totalFrames, true];\n }\n\n #isMetadataEmpty(metadata: MediaMetadata): boolean {\n return metadata.title === ''\n && metadata.artist === ''\n && metadata.album === ''\n && metadata.duration === 0;\n }\n\n #updateOutputProperties(properties: Map<string, string>): void {\n const [sampleRate, channels, bytesPerChannel] = getAudioProperties(properties);\n\n this.#streamContext.sampleRate = sampleRate;\n this.#streamContext.channels = channels;\n this.#streamContext.bytesPerChannel = bytesPerChannel;\n\n this.#context.logger.debug(`Update play settings to ${sampleRate}/${channels}/${bytesPerChannel * 8}bit`);\n }\n}\n","import { EventEmitter } from 'node:events';\nimport type { Socket as UdpSocket } from 'node:dgram';\nimport { type AudioSource, Context, Discovery, type DiscoveryResult, TimingServer } from '@basmilius/apple-common';\nimport type { MediaMetadata, PlaybackInfo, Settings, StreamContext, StreamProtocol } from './types';\nimport RtspClient from './rtspClient';\nimport StreamClient from './streamClient';\n\nconst SAMPLE_RATE = 44100;\nconst CHANNELS = 2;\nconst BYTES_PER_CHANNEL = 2;\nconst FRAMES_PER_PACKET = 352;\n\nexport type EventMap = {\n readonly playing: [playbackInfo: PlaybackInfo];\n readonly stopped: [];\n};\n\nexport type StreamOptions = {\n readonly metadata?: MediaMetadata;\n readonly volume?: number;\n}\n\nexport class RaopClient extends EventEmitter<EventMap> {\n get context(): Context {\n return this.#context;\n }\n\n get deviceId(): string {\n return this.#discoveryResult.id;\n }\n\n get address(): string {\n return this.#discoveryResult.address;\n }\n\n get modelName(): string {\n return this.#discoveryResult.modelName;\n }\n\n get info(): Record<string, unknown> {\n return this.#streamClient.info;\n }\n\n readonly #context: Context;\n readonly #rtsp: RtspClient;\n readonly #streamClient: StreamClient;\n readonly #discoveryResult: DiscoveryResult;\n\n private constructor(context: Context, rtsp: RtspClient, streamClient: StreamClient, discoveryResult: DiscoveryResult) {\n super();\n\n this.#context = context;\n this.#rtsp = rtsp;\n this.#streamClient = streamClient;\n this.#discoveryResult = discoveryResult;\n\n this.#streamClient.on('playing', info => this.emit('playing', info));\n this.#streamClient.on('stopped', () => this.emit('stopped'));\n }\n\n async stream(source: AudioSource, options: StreamOptions = {}): Promise<void> {\n await source.start();\n\n try {\n await this.#streamClient.sendAudio(\n source,\n options.metadata,\n options.volume\n );\n } finally {\n await source.stop();\n }\n }\n\n stop(): void {\n this.#streamClient.stop();\n }\n\n async setVolume(volume: number): Promise<void> {\n await this.#streamClient.setVolume(volume);\n }\n\n async close(): Promise<void> {\n this.#streamClient.close();\n await this.#rtsp.disconnect();\n }\n\n static async create(discoveryResult: DiscoveryResult, timingServer: TimingServer): Promise<RaopClient> {\n const context = new Context(discoveryResult.id);\n const rtsp = new RtspClient(context, discoveryResult.address, discoveryResult.service.port);\n\n await rtsp.connect();\n\n const streamContext = createStreamContext();\n streamContext.rtspSession = rtsp.rtspSessionId;\n\n const protocol = new RaopStreamProtocol(rtsp, streamContext);\n\n const settings: Settings = {\n protocols: {\n raop: {\n controlPort: 0,\n timingPort: 0\n }\n }\n };\n\n const streamClient = new StreamClient(context, rtsp, streamContext, protocol, settings, timingServer);\n\n const properties = new Map<string, string>(Object.entries(discoveryResult.txt));\n await streamClient.initialize(properties);\n\n return new RaopClient(context, rtsp, streamClient, discoveryResult);\n }\n\n static async discover(deviceId: string, timingServer: TimingServer): Promise<RaopClient> {\n const discovery = Discovery.raop();\n const result = await discovery.findUntil(deviceId);\n\n return RaopClient.create(result, timingServer);\n }\n}\n\nclass RaopStreamProtocol implements StreamProtocol {\n readonly #rtsp: RtspClient;\n readonly #streamContext: StreamContext;\n #feedbackInterval?: NodeJS.Timeout;\n\n constructor(rtsp: RtspClient, streamContext: StreamContext) {\n this.#rtsp = rtsp;\n this.#streamContext = streamContext;\n }\n\n async setup(timingPort: number, controlPort: number): Promise<void> {\n await this.#rtsp.announce(\n this.#streamContext.bytesPerChannel,\n this.#streamContext.channels,\n this.#streamContext.sampleRate\n );\n\n const transport = [\n 'RTP/AVP/UDP',\n 'unicast',\n 'interleaved=0-1',\n 'mode=record',\n `control_port=${controlPort}`,\n `timing_port=${timingPort}`\n ].filter(Boolean).join(';');\n\n const response = await this.#rtsp.setup({\n 'Transport': transport\n });\n\n const transportHeader = response.headers.get('Transport');\n\n if (!transportHeader) {\n return;\n }\n\n const serverPortMatch = transportHeader.match(/server_port=(\\d+)/);\n\n if (serverPortMatch) {\n this.#streamContext.serverPort = parseInt(serverPortMatch[1], 10);\n }\n\n const controlPortMatch = transportHeader.match(/control_port=(\\d+)/);\n\n if (controlPortMatch) {\n this.#streamContext.controlPort = parseInt(controlPortMatch[1], 10);\n }\n }\n\n async startFeedback(): Promise<void> {\n this.#feedbackInterval = setInterval(async () => {\n try {\n await this.#rtsp.feedback(true);\n } catch {\n }\n }, 2000);\n }\n\n async sendAudioPacket(transport: UdpSocket, header: Buffer, audio: Buffer): Promise<[number, Buffer]> {\n const packet = Buffer.concat([header, audio]);\n const seqno = header.readUInt16BE(2);\n\n await new Promise<void>((resolve, reject) => {\n transport.send(packet, (err) => err ? reject(err) : resolve());\n });\n\n return [seqno, packet];\n }\n\n teardown(): void {\n if (!this.#feedbackInterval) {\n return;\n }\n\n clearInterval(this.#feedbackInterval);\n\n this.#feedbackInterval = undefined;\n }\n}\n\nfunction createStreamContext(): StreamContext {\n return {\n sampleRate: SAMPLE_RATE,\n channels: CHANNELS,\n bytesPerChannel: BYTES_PER_CHANNEL,\n rtpseq: Math.floor(Math.random() * 65536),\n rtptime: Math.floor(Math.random() * 0xFFFFFFFF),\n headTs: 0,\n latency: Math.floor(SAMPLE_RATE * 2),\n serverPort: 0,\n controlPort: 0,\n rtspSession: '',\n volume: -20,\n position: 0,\n packetSize: FRAMES_PER_PACKET * CHANNELS * BYTES_PER_CHANNEL,\n frameSize: CHANNELS * BYTES_PER_CHANNEL,\n paddingSent: 0,\n\n reset() {\n this.rtpseq = Math.floor(Math.random() * 65536);\n this.rtptime = Math.floor(Math.random() * 0xFFFFFFFF);\n this.headTs = this.rtptime;\n this.paddingSent = 0;\n this.position = 0;\n }\n };\n}\n"],"mappings":"oYAmDA,IAAY,EAAL,SAAA,EAAA,OACH,GAAA,EAAA,QAAA,GAAA,UACA,EAAA,EAAA,YAAA,GAAA,cACA,EAAA,EAAA,OAAA,GAAA,eACH,CAEW,EAAL,SAAA,EAAA,OACH,GAAA,EAAA,aAAA,GAAA,eACA,EAAA,EAAA,KAAA,GAAA,OACA,EAAA,EAAA,QAAA,GAAA,UACA,EAAA,EAAA,SAAA,GAAA,iBACH,CC9DD,IAAa,EAAb,KAAwB,CACpB,GACA,GAAyC,IAAI,IAC7C,GAA4B,EAAE,CAE9B,YAAY,EAAiB,CACzB,MAAA,EAAgB,EAGpB,IAAI,EAAmC,CACnC,OAAO,MAAA,EAAc,IAAI,EAAM,CAGnC,IAAI,EAAe,EAAsB,CACjC,UAAA,EAAc,IAAI,EAAM,CAO5B,IAHA,MAAA,EAAc,IAAI,EAAO,EAAO,CAChC,MAAA,EAAY,KAAK,EAAM,CAEhB,MAAA,EAAY,OAAS,MAAA,GAAe,CACvC,IAAM,EAAS,MAAA,EAAY,OAAO,CAC9B,IAAW,IAAA,IACX,MAAA,EAAc,OAAO,EAAO,EAKxC,IAAI,EAAwB,CACxB,OAAO,MAAA,EAAc,IAAI,EAAM,CAGnC,OAAc,CACV,MAAA,EAAc,OAAO,CACrB,MAAA,EAAY,OAAS,IAI7B,MAAa,EAAoB,CAC7B,OACI,EACA,EACA,EACA,EACA,EACM,CACN,IAAM,EAAS,OAAO,YAAY,GAAG,CAMrC,OALA,EAAO,WAAW,EAAQ,EAAE,CAC5B,EAAO,WAAW,EAAa,EAAE,CACjC,EAAO,cAAc,EAAO,EAAE,CAC9B,EAAO,cAAc,EAAW,EAAE,CAClC,EAAO,cAAc,EAAM,EAAE,CACtB,GAEd,CAEY,EAAa,CACtB,OACI,EACA,EACA,EACA,EACA,EACA,EACA,EACM,CACN,IAAM,EAAS,OAAO,YAAY,GAAG,CAQrC,OAPA,EAAO,WAAW,EAAQ,EAAE,CAC5B,EAAO,WAAW,EAAa,EAAE,CACjC,EAAO,cAAc,EAAO,EAAE,CAC9B,EAAO,cAAc,EAAc,EAAE,CACrC,EAAO,cAAc,EAAQ,EAAE,CAC/B,EAAO,cAAc,EAAS,GAAG,CACjC,EAAO,cAAc,EAAiB,GAAG,CAClC,GAEd,CAOD,SAAgB,EAAwB,EAAiC,CACrE,MAAO,CACH,UAAW,EAAK,aAAa,EAAE,CAC/B,YAAa,EAAK,aAAa,EAAE,CACpC,CCtFL,SAAgB,EAAU,EAAwB,CAG9C,OAFI,GAAU,EAAU,KACpB,GAAU,IAAY,EACnB,GAAK,KAAK,MAAM,EAAS,IAAI,CAGxC,SAAgB,EAAmB,EAAiD,CAChF,IAAM,EAAK,EAAW,IAAI,KAAK,CAC/B,GAAI,CAAC,EAAI,OAAO,EAAe,QAE/B,IAAI,EAAQ,EAAe,QAC3B,IAAK,IAAM,KAAK,EAAG,MAAM,IAAI,CAAE,CAC3B,IAAM,EAAM,SAAS,EAAE,MAAM,CAAE,GAAG,CAC9B,IAAQ,IAAG,GAAS,EAAe,aACnC,IAAQ,IAAG,GAAS,EAAe,QAE3C,OAAO,EAGX,SAAgB,EAAiB,EAA+C,CAC5E,IAAM,EAAK,EAAW,IAAI,KAAK,CAC/B,GAAI,CAAC,EAAI,OAAO,EAAa,aAE7B,IAAI,EAAQ,EAAa,aACzB,IAAK,IAAM,KAAK,EAAG,MAAM,IAAI,CAAE,CAC3B,IAAM,EAAM,SAAS,EAAE,MAAM,CAAE,GAAG,CAC9B,IAAQ,IAAG,GAAS,EAAa,MACjC,IAAQ,IAAG,GAAS,EAAa,SACjC,IAAQ,IAAG,GAAS,EAAa,UAEzC,OAAO,EAGX,SAAgB,EAAmB,EAA2D,CAI1F,MAAO,CAHI,SAAS,EAAW,IAAI,KAAK,EAAI,QAAS,GAAG,CAC7C,SAAS,EAAW,IAAI,KAAK,EAAI,IAAK,GAAG,CACzC,SAAS,EAAW,IAAI,KAAK,EAAI,KAAM,GAAG,CAChC,EAAE,CCjC3B,SAAS,EAAU,EAAmB,EAA4B,CAC9D,IAAM,EAAU,KAAK,MAAM,EAAY,EAAW,CAC5C,EAAa,EAAY,EAAc,WAAc,EAE3D,OAAQ,OAAO,EAAQ,EAAI,IAAO,OAAO,KAAK,MAAM,EAAS,CAAC,CAGlE,IAAqB,EAArB,cAA2C,CAAa,CACpD,GACA,GACA,GACA,GACA,GACA,GAEA,YAAY,EAAwB,EAA2B,CAC3D,OAAO,CACP,MAAA,EAAgB,EAChB,MAAA,EAAsB,EAG1B,IAAI,MAAe,CACf,OAAO,MAAA,GAAmB,EAG9B,MAAM,KAAK,EAAiB,EAA6B,CACrD,OAAO,IAAI,SAAS,EAAS,IAAW,CACpC,MAAA,EAAkB,EAAa,OAAO,CAEtC,MAAA,EAAgB,GAAG,QAAU,GAAQ,CACjC,QAAQ,MAAM,4BAA6B,EAAI,CAC/C,EAAO,EAAI,EACb,CAEF,MAAA,EAAgB,GAAG,WAAY,EAAM,IAAU,CAC3C,MAAA,EAAgB,EAAM,EAAM,EAC9B,CAEF,MAAA,EAAgB,GAAG,gBAAmB,CAElC,MAAA,EADgB,MAAA,EAAiB,SAAS,CAChB,KAC1B,GAAS,EACX,CAEF,MAAA,EAAgB,KAAK,EAAM,EAAQ,EACrC,CAGN,OAAc,CACV,KAAK,MAAM,CAEX,AAEI,MAAA,KADA,MAAA,EAAgB,OAAO,CACL,IAAA,IAI1B,MAAM,EAA0B,CAC5B,GAAI,MAAA,EACA,MAAU,MAAM,kBAAkB,CAGtC,MAAA,EAAwB,IAAI,gBAC5B,MAAA,EAAoB,EAAY,MAAA,EAAc,YAAY,CAG9D,MAAa,CACT,AAEI,MAAA,KADA,MAAA,EAAsB,OAAO,CACL,IAAA,IAG5B,AAEI,MAAA,KADA,cAAc,MAAA,EAAe,CACZ,IAAA,IAIzB,GAAe,EAAc,EAAoB,CAC7C,IAAI,EAAc,GAEZ,MAAiB,CACnB,GAAI,CAAC,MAAA,EAAiB,OAEtB,IAAM,EAAc,EAAU,MAAA,EAAc,OAAQ,MAAA,EAAc,WAAW,CACvE,CAAC,EAAY,GAAe,EAAI,MAAM,EAAY,CAElD,EAAS,EAAW,OACtB,EAAc,IAAO,IACrB,IACA,EACA,MAAA,EAAc,OAAS,MAAA,EAAc,QACrC,EACA,EACA,MAAA,EAAc,OACjB,CAED,EAAc,GACd,MAAA,EAAgB,KAAK,EAAQ,EAAM,EAAK,EAG5C,GAAU,CACV,MAAA,EAAiB,YAAY,EAAU,IAAK,CAGhD,GAAW,EAAc,EAAgD,EAClD,EAAK,GAAK,MAEV,GACf,MAAA,EAA4B,EAAwB,EAAK,CAAE,EAAM,CAEjE,QAAQ,MAAM,uCAAwC,EAAO,EAAK,CAI1E,GAAuB,EAAqD,EAA+C,CACvH,IAAK,IAAI,EAAI,EAAG,EAAI,EAAQ,YAAa,IAAK,CAC1C,IAAM,EAAQ,EAAQ,UAAY,EAClC,GAAI,MAAA,EAAoB,IAAI,EAAM,CAAE,CAChC,IAAM,EAAS,MAAA,EAAoB,IAAI,EAAM,CACvC,EAAgB,EAAO,SAAS,EAAG,EAAE,CACrC,EAAO,OAAO,OAAO,CAAC,OAAO,KAAK,CAAC,IAAM,IAAK,CAAC,CAAE,EAAe,EAAO,CAAC,CAE1E,MAAA,GACA,MAAA,EAAgB,KAAK,EAAM,EAAK,KAAM,EAAK,QAAQ,MAGvD,QAAQ,MAAM,UAAU,EAAM,iBAAiB,IChI/D,MAIM,EAAyB,OAAO,KAAK,CAAC,EAAK,CAAC,CAG5C,EAAqB,OAAO,KAAK,CACnC,GAAM,EAAM,IAAM,IAAM,GAAM,GAAM,IAAM,IAC1C,GAAM,IAAM,IAAM,GAAM,GAAM,EAAM,IAAM,EAC1C,IAAM,GAAM,IAAM,GAAM,IAAM,IAAM,GAAM,GAC1C,GAAM,IAAM,EAAM,GAAM,GAAM,IAAM,GAAM,GAC7C,CAAC,CAkBF,SAAS,EAAiB,EAAgB,EAAa,EAA0B,CAC7E,IAAM,EAAM,EAAW,MAAM,CACxB,OAAO,GAAG,EAAK,SAAS,GAAG,EAAK,MAAM,GAAG,EAAK,WAAW,CACzD,OAAO,MAAM,CAEZ,EAAM,EAAW,MAAM,CACxB,OAAO,GAAG,EAAO,GAAG,IAAM,CAC1B,OAAO,MAAM,CAEZ,EAAW,EAAW,MAAM,CAC7B,OAAO,GAAG,EAAI,GAAG,EAAK,MAAM,GAAG,IAAM,CACrC,OAAO,MAAM,CAElB,MAAO,oBAAoB,EAAK,SAAS,YAAY,EAAK,MAAM,YAAY,EAAK,MAAM,UAAU,EAAI,eAAe,EAAS,GAGjI,SAAS,GAAkC,CACvC,OAAO,KAAK,MAAM,KAAK,QAAQ,CAAG,WAAW,CAGjD,SAAS,EAAqB,EAAyC,CACnE,MAAO,CACH,MACA,YAAY,EAAQ,UAAU,YAAY,EAAQ,UAClD,WACA,YAAY,EAAQ,WACpB,QACA,uBACA,mBAAmB,EAAQ,WAAW,GAAG,EAAQ,WACjD,mBAAoC,EAAQ,eAAe,YAAY,EAAQ,SAAS,WAAW,EAAQ,aAC9G,CAAC,KAAK;EAAO,CAAG;EAGrB,IAAqB,EAArB,cAAwC,CAAe,CACnD,IAAI,gBAAyB,CACzB,OAAO,MAAA,EAGX,IAAI,QAAiB,CACjB,OAAO,MAAA,EAGX,IAAI,eAAwB,CACxB,OAAO,MAAA,EAGX,IAAI,WAAoB,CACpB,OAAO,MAAA,EAGX,IAAI,KAAc,CACd,MAAO,UAAU,KAAK,WAAW,QAAQ,GAAG,MAAA,IAGhD,IAAI,YAAoD,CACpD,MAAO,CACH,QAAS,MAAA,EACT,SAAU,KAAK,QAClB,CAGL,GACA,GACA,GACA,GACA,GAAmB,UACnB,GAAkB,OAAO,MAAM,EAAE,CACjC,GAAgB,EAChB,GACA,GAGK,IAAI,IAET,YAAY,EAAkB,EAAiB,EAAc,CACzD,MAAM,EAAS,EAAS,EAAK,CAE7B,MAAA,EAAuB,GAAwB,CAC/C,MAAA,EAAe,GAAgB,CAC/B,MAAA,EAAsB,GAAmB,CACzC,MAAA,EAAkB,GAAyB,CAE3C,KAAK,GAAG,QAAS,MAAA,EAAc,KAAK,KAAK,CAAC,CAC1C,KAAK,GAAG,OAAQ,MAAA,EAAa,KAAK,KAAK,CAAC,CACxC,KAAK,GAAG,QAAS,MAAA,EAAc,KAAK,KAAK,CAAC,CAC1C,KAAK,GAAG,UAAW,MAAA,EAAgB,KAAK,KAAK,CAAC,CAC9C,KAAK,GAAG,UAAW,MAAA,EAAgB,KAAK,KAAK,CAAC,CAGlD,MAAM,MAAyC,CAC3C,GAAI,CACA,IAAM,EAAW,MAAM,MAAA,EAAe,MAAO,QAAS,CAClD,WAAY,GACf,CAAC,CAEF,GAAI,EAAS,GAAI,CACb,IAAM,EAAS,OAAO,KAAK,MAAM,EAAS,aAAa,CAAC,CACxD,GAAI,EAAO,OAAS,EAChB,GAAI,CACA,OAAO,EAAM,MAAM,EAAO,OAAO,MAC7B,CACJ,MAAO,EAAE,EAKrB,MAAO,EAAE,MACL,CACJ,MAAO,EAAE,EAIjB,MAAM,WAA2B,CAC7B,IAAM,EAAO,OAAO,OAAO,CAAC,EAAwB,EAAmB,CAAC,CAExE,MAAM,MAAA,EAAe,OAAQ,cAAe,CACxC,YAAa,2BACb,OACA,SAAU,WACb,CAAC,CAGN,MAAM,SAAS,EAAyB,EAAkB,EAAoB,EAAsC,CAChH,IAAM,EAAO,EAAqB,CAC9B,UAAW,MAAA,EACX,QAAS,KAAK,WAAW,QACzB,SAAU,KAAK,WAAW,SAC1B,eAAgB,EAAI,EACpB,WACA,aACH,CAAC,CAEE,EAAW,MAAM,MAAA,EAAe,WAAY,IAAA,GAAW,CACvD,YAAa,kBACb,OACA,WAAY,CAAC,CAAC,EACjB,CAAC,CAGF,GAAI,EAAS,SAAW,KAAO,EAAU,CACrC,IAAM,EAAkB,EAAS,QAAQ,IAAI,mBAAmB,CAEhE,GAAI,EAAiB,CACjB,IAAM,EAAQ,EAAgB,MAAM,IAAI,CAEpC,EAAM,QAAU,IAChB,MAAA,EAAmB,CACf,SAAU,QACV,MAAO,EAAM,GACb,WACA,MAAO,EAAM,GAChB,CAED,EAAW,MAAM,MAAA,EAAe,WAAY,IAAA,GAAW,CACnD,YAAa,kBACb,OACH,CAAC,GAKd,OAAO,EAGX,MAAM,MAAM,EAAkC,EAAqE,CAC/G,OAAO,MAAM,MAAA,EAAe,QAAS,IAAA,GAAW,CAAC,UAAS,OAAK,CAAC,CAGpE,MAAM,OAAO,EAAiD,CAC1D,MAAM,MAAA,EAAe,SAAU,IAAA,GAAW,CAAC,UAAQ,CAAC,CAGxD,MAAM,MAAM,EAA6D,CACrE,MAAM,MAAA,EAAe,QAAS,IAAA,GAAW,CAAC,QAAS,EAAQ,QAAQ,CAAC,CAGxE,MAAM,aAAa,EAAc,EAA8B,CAC3D,MAAM,MAAA,EAAe,gBAAiB,IAAA,GAAW,CAC7C,YAAa,kBACb,KAAM,GAAG,EAAK,IAAI,IACrB,CAAC,CAGN,MAAM,YAAY,EAAiB,EAAgB,EAAiB,EAAwC,CACxG,IAAM,EAAW,EAAK,oBAAoB,CACtC,MAAO,EAAS,MAChB,OAAQ,EAAS,OACjB,MAAO,EAAS,MAChB,SAAU,EAAS,SACtB,CAAC,CAEF,MAAM,MAAA,EAAe,gBAAiB,IAAA,GAAW,CAC7C,YAAa,4BACb,QAAS,CACL,QAAW,EACX,WAAY,OAAO,EAAO,WAAW,IACxC,CACD,KAAM,EACT,CAAC,CAGN,MAAM,WAAW,EAAiB,EAAgB,EAAiB,EAAgC,CAC/F,IAAI,EAAc,aACd,EAAQ,KAAO,KAAQ,EAAQ,KAAO,KACtC,EAAc,aAGlB,MAAM,MAAA,EAAe,gBAAiB,IAAA,GAAW,CAC7C,cACA,QAAS,CACL,QAAW,EACX,WAAY,OAAO,EAAO,WAAW,IACxC,CACD,KAAM,EACT,CAAC,CAGN,MAAM,SAAS,EAAsB,GAA0B,CAC3D,OAAO,MAAM,MAAA,EAAe,OAAQ,YAAa,CAAC,aAAW,CAAC,CAGlE,MAAM,SAAS,EAAgC,CAC3C,MAAM,MAAA,EAAe,WAAY,IAAA,GAAW,CACxC,QAAS,CAAC,QAAW,EAAQ,CAChC,CAAC,CAGN,MAAA,EACI,EACA,EACA,EAOI,EAAE,CACW,CACjB,GAAM,CACF,cACA,QAAS,EAAe,EAAE,CAC1B,aAAa,GACb,WAAW,WACX,UAAU,GACV,EACA,CAAC,QAAQ,EAEP,EAAO,MAAA,IACP,EAAY,GAAO,KAAK,IAExB,EAA2C,CAC7C,KAAQ,EACR,UAAW,MAAA,EACX,gBAAiB,MAAA,EACjB,kBAAmB,MAAA,EACnB,aAAc,iBACjB,CAEG,MAAA,IACA,EAAQ,cAAmB,EAAiB,EAAQ,EAAW,MAAA,EAAiB,EAGpF,OAAO,OAAO,EAAS,EAAa,CAEhC,GAAQ,OAAO,GAAS,UAAY,CAAC,OAAO,SAAS,EAAK,EAC1D,EAAQ,gBAAkB,mCAC1B,EAAO,OAAO,KAAK,EAAM,UAAU,EAAW,CAAC,EACxC,IACP,EAAQ,gBAAkB,GAG9B,IAAI,EACA,GACA,EAAa,OAAO,GAAS,SAAW,OAAO,KAAK,EAAK,CAAG,EAC5D,EAAQ,kBAAoB,EAAW,QAEvC,EAAQ,kBAAoB,EAGhC,IAAM,EAAc,CAChB,GAAG,EAAO,GAAG,EAAU,GAAG,IAC1B,GAAG,OAAO,QAAQ,EAAQ,CAAC,KAAK,CAAC,EAAG,KAAO,GAAG,EAAE,IAAI,IAAI,CACxD,GACA,GACH,CAAC,KAAK;EAAO,CAER,EAAO,EACP,OAAO,OAAO,CAAC,OAAO,KAAK,EAAY,CAAE,EAAW,CAAC,CACrD,OAAO,KAAK,EAAY,CAI9B,OAFA,KAAK,QAAQ,OAAO,IAAI,SAAU,EAAQ,EAAW,QAAQ,IAAO,CAE7D,IAAI,SAAS,EAAS,IAAW,CACpC,MAAA,EAAe,IAAI,EAAM,CAAC,UAAS,SAAO,CAAC,CAE3C,IAAM,EAAQ,eAAiB,CAC3B,MAAA,EAAe,OAAO,EAAK,CAC3B,EAAW,MAAM,uBAAuB,EAAK,IAAI,EAAU,GAAG,CAAC,EAChE,EAAQ,CAEX,KAAK,MAAM,EAAK,CAEhB,IAAM,EAAkB,EAExB,MAAA,EAAe,IAAI,EAAM,CACrB,QAAU,GAAa,CACnB,aAAa,EAAM,CACf,CAAC,GAAc,CAAC,EAAS,GACzB,EAAW,MAAM,eAAe,EAAS,OAAO,GAAG,EAAS,aAAa,CAAC,CAE1E,EAAgB,EAAS,EAGjC,OAAS,GAAU,CACf,aAAa,EAAM,CACnB,EAAO,EAAM,EAEpB,CAAC,EACJ,CAGN,IAAmB,CACf,MAAA,EAAgB,UAGpB,IAAiB,CACb,MAAA,EAAe,OAAO,MAAM,EAAE,CAE9B,IAAK,GAAM,CAAC,EAAM,CAAC,aAAY,MAAA,EAC3B,EAAW,MAAM,oBAAoB,CAAC,CACtC,MAAA,EAAe,OAAO,EAAK,CAG/B,KAAK,QAAQ,OAAO,IAAI,SAAU,aAAa,CAGnD,GAAQ,EAAoB,CACxB,GAAI,CAGA,IAFA,MAAA,EAAe,OAAO,OAAO,CAAC,MAAA,EAAc,EAAK,CAAC,CAE3C,MAAA,EAAa,WAAa,GAAG,CAChC,IAAM,EAAS,EAAK,aAAa,MAAA,EAAa,CAE9C,GAAI,IAAW,KACX,OAGJ,MAAA,EAAe,MAAA,EAAa,SAAS,EAAO,eAAe,CAE3D,IAAM,EAAa,EAAO,SAAS,QAAQ,IAAI,OAAO,CAChD,EAAO,EAAa,SAAS,EAAY,GAAG,CAAG,GAErD,GAAI,MAAA,EAAe,IAAI,EAAK,CAAE,CAC1B,GAAM,CAAC,WAAW,MAAA,EAAe,IAAI,EAAK,CAC1C,MAAA,EAAe,OAAO,EAAK,CAC3B,EAAQ,EAAO,SAAS,MAExB,KAAK,QAAQ,OAAO,KAAK,SAAU,gCAAgC,IAAO,QAG7E,EAAK,CACV,KAAK,QAAQ,OAAO,MAAM,SAAU,YAAa,EAAI,CACrD,KAAK,KAAK,QAAS,EAAa,EAIxC,GAAS,EAAkB,CACvB,IAAK,GAAM,CAAC,EAAM,CAAC,aAAY,MAAA,EAC3B,EAAO,EAAI,CACX,MAAA,EAAe,OAAO,EAAK,CAG/B,KAAK,QAAQ,OAAO,MAAM,SAAU,aAAc,EAAI,CAG1D,IAAmB,CACf,IAAM,EAAU,MAAM,uBAAuB,CAE7C,IAAK,GAAM,CAAC,EAAM,CAAC,aAAY,MAAA,EAC3B,EAAO,EAAI,CACX,MAAA,EAAe,OAAO,EAAK,CAG/B,KAAK,QAAQ,OAAO,IAAI,SAAU,eAAe,GCnapC,EAArB,KAAgC,CAC5B,WACA,YACA,aACA,YAAsB,EACtB,eAAyB,EAEzB,YAAY,EAAoB,CAC5B,KAAK,WAAa,EAClB,KAAK,YAAc,QAAQ,OAAO,QAAQ,CAC1C,KAAK,aAAe,YAAY,KAAK,CAGzC,IAAI,oBAA6B,CAC7B,IAAM,EAAY,OAAO,QAAQ,OAAO,QAAQ,CAAG,KAAK,YAAY,CAEpE,OAAO,KAAK,MAAM,GAAa,IAAM,KAAK,YAAY,CAG1D,IAAI,cAAuB,CACvB,OAAO,KAAK,mBAAqB,KAAK,YAG1C,IAAI,mBAA6B,CAC7B,OAAO,KAAK,gBAAkB,KAAK,WAGvC,KAAK,EAA0B,CAC3B,KAAK,aAAe,EACpB,KAAK,gBAAkB,EAG3B,aAAgC,CAC5B,IAAM,EAAU,YAAY,KAAK,CAC3B,GAAQ,EAAU,KAAK,cAAgB,IAC7C,KAAK,aAAe,EAEpB,IAAM,EAAS,KAAK,eAGpB,MAFA,MAAK,eAAiB,EAEf,CAAC,EAAM,EAAO,GCtC7B,MAKa,EAAkC,CAC3C,MAAO,4BACP,OAAQ,aACR,MAAO,UACP,SAAU,EACb,CAEY,EAAgC,CACzC,MAAO,GACP,OAAQ,GACR,MAAO,GACP,SAAU,EACb,CAEY,EAAwB,EAAe,YAAc,EAAe,OCLjF,IAAqB,EAArB,cAA0C,CAAuB,CAC7D,IAAI,MAAgC,CAChC,OAAO,MAAA,EAGX,IAAI,cAA6B,CAC7B,MAAO,CACH,SAAU,MAAA,EAAsB,MAAA,EAAe,CAAG,EAAmB,MAAA,EACrE,SAAU,MAAA,EAAoB,SACjC,CAGL,IAAA,GAAkC,CAC9B,IAAM,EAAY,MAAA,EAAiB,IAAI,KAAK,EAAI,GAEhD,OACK,MAAA,EAAwB,EAAe,UAAY,GACjD,EAAU,WAAW,UAAU,CAI1C,GACA,GACA,GACA,GACA,GACA,GACA,GAEA,GACA,GAAmC,EAAe,QAClD,GAA+B,EAAa,aAC5C,GAA2B,EAC3B,GAAiC,EAAE,CACnC,GAAmC,IAAI,IACvC,GAAsB,GAEtB,YAAY,EAAkB,EAAkB,EAA8B,EAA0B,EAAoB,EAA4B,CACpJ,OAAO,CAEP,MAAA,EAAgB,EAChB,MAAA,EAAa,EACb,MAAA,EAAsB,EACtB,MAAA,EAAiB,EACjB,MAAA,EAAiB,EACjB,MAAA,EAAsB,IAAI,EAAW,IAAoB,CACzD,MAAA,EAAqB,EAGzB,OAAc,CACV,MAAA,EAAe,UAAU,CACzB,MAAA,GAAqB,OAAO,CAGhC,MAAM,WAAW,EAAgD,CAC7D,MAAA,EAAmB,EACnB,MAAA,EAAwB,EAAmB,EAAW,CACtD,MAAA,EAAsB,EAAiB,EAAW,CAElD,MAAA,EAAc,OAAO,KAAK,qCAAqC,MAAA,EAAsB,aAAa,MAAA,IAAsB,CAExH,IAAM,EAAe,MAAA,EAAwB,GACzC,CAAC,GAAgB,IAAiB,EAAe,UACjD,MAAA,EAAc,OAAO,MAAM,kDAAkD,CAGjF,MAAA,EAA6B,EAAW,CAExC,MAAA,EAAsB,IAAI,EAAc,MAAA,EAAqB,MAAA,EAAoB,CACjF,MAAM,MAAA,EAAoB,KACtB,MAAA,EAAW,WAAW,QACtB,MAAA,EAAe,UAAU,KAAK,YACjC,CAED,MAAA,EAAc,OAAO,MAAM,wBAAwB,MAAA,EAAoB,KAAK,WAAW,MAAA,EAAmB,OAAO,CAEjH,IAAM,EAAO,MAAM,MAAA,EAAW,MAAM,CACpC,OAAO,OAAO,MAAA,EAAY,EAAK,CAC/B,MAAA,EAAc,OAAO,MAAM,8BAA+B,MAAA,EAAW,CAEjE,MAAA,GACA,MAAM,MAAA,EAAW,WAAW,CAGhC,MAAM,MAAA,EAAe,MAAM,MAAA,EAAmB,KAAM,MAAA,EAAoB,KAAK,CAGjF,MAAa,CACT,MAAA,EAAc,OAAO,MAAM,0BAA0B,CACrD,MAAA,EAAkB,GAGtB,MAAM,UAAU,EAA+B,CAC3C,MAAM,MAAA,EAAW,aAAa,SAAU,OAAO,EAAO,CAAC,CACvD,MAAA,EAAoB,OAAS,EAGjC,MAAM,UAAU,EAAqB,EAA0B,EAAgB,EAAgC,CAC3G,GAAI,CAAC,MAAA,EACD,MAAU,MAAM,kBAAkB,CAGtC,MAAA,EAAoB,OAAO,CAE3B,IAAI,EAEJ,GAAI,CAQA,GAPA,EAAY,EAAa,OAAO,CAChC,MAAM,IAAI,QAAe,GAAY,CACjC,EAAW,QAAQ,MAAA,EAAoB,WAAY,MAAA,EAAW,WAAW,SAAU,EAAQ,EAC7F,CAEF,MAAA,EAAoB,MAAM,MAAA,EAAW,WAAW,SAAS,EAEpD,MAAA,EAAsB,EAAa,YAAc,EAAG,CACrD,IAAM,EAAQ,MAAA,EAAoB,QAC5B,EAAM,MAAA,EAAoB,QAC1B,EAAM,EAAQ,EAAO,SAAW,MAAA,EAAoB,WAC1D,MAAM,MAAA,EAAW,aAAa,WAAY,GAAG,EAAM,GAAG,EAAI,GAAG,IAAM,CAGvE,MAAA,EAAiB,GAEZ,MAAA,EAAsB,EAAa,QAAU,IAC9C,MAAA,EAAc,OAAO,MAAM,yBAA0B,KAAK,aAAa,SAAS,CAChF,MAAM,MAAA,EAAW,YACb,MAAA,EAAoB,YACpB,MAAA,EAAoB,OACpB,MAAA,EAAoB,QACpB,KAAK,aAAa,SACrB,GAGA,MAAA,EAAsB,EAAa,WAAa,GAAK,EAAS,UAC/D,MAAA,EAAc,OAAO,MAAM,WAAW,EAAS,QAAQ,OAAO,gBAAgB,CAE9E,MAAM,MAAA,EAAW,WACb,MAAA,EAAoB,YACpB,MAAA,EAAoB,OACpB,MAAA,EAAoB,QACpB,EAAS,QACZ,EAGL,MAAM,MAAA,EAAe,eAAe,CAEpC,KAAK,KAAK,UAAW,KAAK,aAAa,CAEvC,MAAM,MAAA,EAAW,OAAO,CACpB,MAAS,SACT,QAAW,MAAA,EAAoB,YAC/B,WAAY,OAAO,MAAA,EAAoB,OAAO,WAAW,MAAA,EAAoB,UAChF,CAAC,CAEF,MAAM,MAAA,EAAW,MAAM,CACnB,QAAS,CACL,MAAS,SACT,QAAW,MAAA,EAAoB,YAC/B,WAAY,OAAO,MAAA,EAAoB,OAAO,WAAW,MAAA,EAAoB,UAChF,CACJ,CAAC,CAEE,IAAW,IAAA,IACX,MAAM,KAAK,UAAU,EAAU,EAAO,CAAC,CAG3C,MAAM,MAAA,EAAiB,EAAQ,EAAU,OACpC,EAAK,CAEV,MADA,MAAA,EAAc,OAAO,MAAM,sCAAuC,EAAI,CAC5D,MAAM,uCAAuC,IAAM,QACvD,CACN,MAAA,EAAoB,OAAO,CAEvB,IACA,MAAM,MAAA,EAAW,SAAS,MAAA,EAAoB,YAAY,CAC1D,EAAU,OAAO,EAGrB,MAAA,EAAe,UAAU,CACzB,KAAK,OAAO,CAEZ,KAAK,KAAK,UAAU,EAI5B,MAAA,EAAkB,EAAqB,EAAqC,CACxE,IAAM,EAAQ,IAAI,EAAW,MAAA,EAAoB,WAAW,CAEtD,EAAc,YAAY,KAAK,CACjC,EAA+B,KAC/B,EAAkB,EAItB,IAFA,MAAA,EAAkB,GAEX,MAAA,GAAiB,CACpB,IAAM,EAAe,MAAA,EAAoB,OAAS,EAC5C,EAAU,MAAM,MAAA,EAAiB,EAAQ,EAAM,cAAgB,EAAG,EAAU,CAElF,GAAI,IAAY,EACZ,MAGJ,EAAM,KAAK,EAAQ,CACnB,IAAM,EAAe,EAAM,aAE3B,GAAI,GAAgBgD,IAAmB,CACnC,IAAM,EAAa,KAAK,IACpB,KAAK,MAAM,EAAeA,IAAkB,CAC5C,EACH,CAED,MAAA,EAAc,OAAO,MACjB,qBAAqB,EAAW,YAAY,EAAa,iBAC5D,CAED,GAAM,CAAC,EAAY,GAAkB,MAAM,MAAA,EACvC,EACA,EACA,EACH,CAGD,GAFA,EAAM,KAAK,EAAW,CAElB,CAAC,EACD,MAIR,GAAI,EAAM,kBAAmB,CACzB,GAAM,CAAC,EAAc,GAAkB,EAAM,aAAa,CAC1D,MAAA,EAAc,OAAO,MACjB,QAAQ,EAAe,aAAa,EAAa,QAAQ,EAAE,CAAC,qBAAqB,EAAM,YAAY,cAAc,EAAM,mBAAmB,GAC7I,CAGL,IAAM,EAAgB,EAAM,YAAc,MAAA,EAAoB,WACxD,GAAc,YAAY,KAAK,CAAG,GAAe,IACjD,EAAO,EAAgB,EAEzB,EAAO,GACP,EAAkB,EAClB,MAAM,EAAQ,EAAO,IAAK,GAEtB,IAAkB,EAAe,GACjC,IAGA,GAAmB,EACnB,MAAA,EAAc,OAAO,KAAK,iCAAiC,EAAa,IAAI,EAAc,QAAQ,EAAE,CAAC,MAAM,EAAW,QAAQ,EAAE,CAAC,MAAM,EAAK,QAAQ,EAAE,CAAC,GAAG,CAE1J,MAAA,EAAc,OAAO,MAAM,iCAAiC,EAAa,IAAI,EAAc,QAAQ,EAAE,CAAC,MAAM,EAAW,QAAQ,EAAE,CAAC,MAAM,EAAK,QAAQ,EAAE,CAAC,GAAG,CAG/J,EAAgB,GAIxB,IAAM,EAAY,OAAO,QAAQ,OAAO,QAAQ,CAAG,EAAM,YAAY,CACrE,MAAA,EAAc,OAAO,MAAM,8BAA8B,EAAY,KAAK,QAAQ,EAAE,CAAC,GAAG,CAG5F,MAAA,EAAkB,EAAqB,EAAsB,EAAuC,CAChG,GAAI,MAAA,EAAoB,aAAe,MAAA,EAAoB,QACvD,MAAO,GAGX,IAAI,EAAS,MAAM,EAAO,WAAWA,IAAkB,CAEvD,GAAI,CAAC,EACD,EAAS,OAAO,MAAM,MAAA,EAAoB,WAAW,CACrD,MAAA,EAAoB,aAAe,KAAK,MAAM,EAAO,OAAS,MAAA,EAAoB,UAAU,SACrF,EAAO,SAAW,MAAA,EAAoB,WAAY,CACzD,IAAM,EAAS,OAAO,MAAM,MAAA,EAAoB,WAAW,CAC3D,EAAO,KAAK,EAAO,CACnB,EAAS,EAGb,IAAM,EAAS,EAAkB,OAC7B,IACA,EAAc,IAAO,GACrB,MAAA,EAAoB,OACpB,MAAA,EAAoB,OACpB,MAAA,EAAW,UACd,CAEK,CAAC,EAAQ,GAAU,MAAM,MAAA,EAAe,gBAAgB,EAAW,EAAQ,EAAO,CAMxF,OALA,MAAA,EAAoB,IAAI,EAAQ,EAAO,CAEvC,MAAA,EAAoB,QAAU,MAAA,EAAoB,OAAS,GAAM,GAAK,GACtE,MAAA,EAAoB,QAAU,KAAK,MAAM,EAAO,OAAS,MAAA,EAAoB,UAAU,CAEhF,KAAK,MAAM,EAAO,OAAS,MAAA,EAAoB,UAAU,CAGpE,MAAA,EAA2B,EAAqB,EAAsB,EAA2C,CAC7G,IAAI,EAAc,EAElB,IAAK,IAAI,EAAI,EAAG,EAAI,EAAO,IAAK,CAC5B,IAAM,EAAO,MAAM,MAAA,EAAiB,EAAQ,GAAO,EAAU,CAG7D,GAFA,GAAe,EAEX,IAAS,EACT,MAAO,CAAC,EAAa,GAAM,CAInC,MAAO,CAAC,EAAa,GAAK,CAG9B,GAAiB,EAAkC,CAC/C,OAAO,EAAS,QAAU,IACnB,EAAS,SAAW,IACpB,EAAS,QAAU,IACnB,EAAS,WAAa,EAGjC,GAAwB,EAAuC,CAC3D,GAAM,CAAC,EAAY,EAAU,GAAmB,EAAmB,EAAW,CAE9E,MAAA,EAAoB,WAAa,EACjC,MAAA,EAAoB,SAAW,EAC/B,MAAA,EAAoB,gBAAkB,EAEtC,MAAA,EAAc,OAAO,MAAM,2BAA2B,EAAW,GAAG,EAAS,GAAG,EAAkB,EAAE,KAAK,GC3UjH,MAAM,EAAc,MAepB,IAAa,EAAb,MAAa,UAAmB,CAAuB,CACnD,IAAI,SAAmB,CACnB,OAAO,MAAA,EAGX,IAAI,UAAmB,CACnB,OAAO,MAAA,EAAsB,GAGjC,IAAI,SAAkB,CAClB,OAAO,MAAA,EAAsB,QAGjC,IAAI,WAAoB,CACpB,OAAO,MAAA,EAAsB,UAGjC,IAAI,MAAgC,CAChC,OAAO,MAAA,EAAmB,KAG9B,GACA,GACA,GACA,GAEA,YAAoB,EAAkB,EAAkB,EAA4B,EAAkC,CAClH,OAAO,CAEP,MAAA,EAAgB,EAChB,MAAA,EAAa,EACb,MAAA,EAAqB,EACrB,MAAA,EAAwB,EAExB,MAAA,EAAmB,GAAG,UAAW,GAAQ,KAAK,KAAK,UAAW,EAAK,CAAC,CACpE,MAAA,EAAmB,GAAG,cAAiB,KAAK,KAAK,UAAU,CAAC,CAGhE,MAAM,OAAO,EAAqB,EAAyB,EAAE,CAAiB,CAC1E,MAAM,EAAO,OAAO,CAEpB,GAAI,CACA,MAAM,MAAA,EAAmB,UACrB,EACA,EAAQ,SACR,EAAQ,OACX,QACK,CACN,MAAM,EAAO,MAAM,EAI3B,MAAa,CACT,MAAA,EAAmB,MAAM,CAG7B,MAAM,UAAU,EAA+B,CAC3C,MAAM,MAAA,EAAmB,UAAU,EAAO,CAG9C,MAAM,OAAuB,CACzB,MAAA,EAAmB,OAAO,CAC1B,MAAM,MAAA,EAAW,YAAY,CAGjC,aAAa,OAAO,EAAkC,EAAiD,CACnG,IAAM,EAAU,IAAI,EAAQ,EAAgB,GAAG,CACzC,EAAO,IAAI,EAAW,EAAS,EAAgB,QAAS,EAAgB,QAAQ,KAAK,CAE3F,MAAM,EAAK,SAAS,CAEpB,IAAM,EAAgB,GAAqB,CAC3C,EAAc,YAAc,EAAK,cAajC,IAAM,EAAe,IAAI,EAAa,EAAS,EAAM,EAXpC,IAAI,EAAmB,EAAM,EAAc,CAEjC,CACvB,UAAW,CACP,KAAM,CACF,YAAa,EACb,WAAY,EACf,CACJ,CACJ,CAEuF,EAAa,CAE/F,EAAa,IAAI,IAAoB,OAAO,QAAQ,EAAgB,IAAI,CAAC,CAG/E,OAFA,MAAM,EAAa,WAAW,EAAW,CAElC,IAAI,EAAW,EAAS,EAAM,EAAc,EAAgB,CAGvE,aAAa,SAAS,EAAkB,EAAiD,CAErF,IAAM,EAAS,MADG,EAAU,MAAM,CACH,UAAU,EAAS,CAElD,OAAO,EAAW,OAAO,EAAQ,EAAa,GAIhD,EAAN,KAAmD,CAC/C,GACA,GACA,GAEA,YAAY,EAAkB,EAA8B,CACxD,MAAA,EAAa,EACb,MAAA,EAAsB,EAG1B,MAAM,MAAM,EAAoB,EAAoC,CAChE,MAAM,MAAA,EAAW,SACb,MAAA,EAAoB,gBACpB,MAAA,EAAoB,SACpB,MAAA,EAAoB,WACvB,CAED,IAAM,EAAY,CACd,cACA,UACA,kBACA,cACA,gBAAgB,IAChB,eAAe,IAClB,CAAC,OAAO,QAAQ,CAAC,KAAK,IAAI,CAMrB,GAJW,MAAM,MAAA,EAAW,MAAM,CACpC,UAAa,EAChB,CAAC,EAE+B,QAAQ,IAAI,YAAY,CAEzD,GAAI,CAAC,EACD,OAGJ,IAAM,EAAkB,EAAgB,MAAM,oBAAoB,CAE9D,IACA,MAAA,EAAoB,WAAa,SAAS,EAAgB,GAAI,GAAG,EAGrE,IAAM,EAAmB,EAAgB,MAAM,qBAAqB,CAEhE,IACA,MAAA,EAAoB,YAAc,SAAS,EAAiB,GAAI,GAAG,EAI3E,MAAM,eAA+B,CACjC,MAAA,EAAyB,YAAY,SAAY,CAC7C,GAAI,CACA,MAAM,MAAA,EAAW,SAAS,GAAK,MAC3B,IAET,IAAK,CAGZ,MAAM,gBAAgB,EAAsB,EAAgB,EAA0C,CAClG,IAAM,EAAS,OAAO,OAAO,CAAC,EAAQ,EAAM,CAAC,CACvC,EAAQ,EAAO,aAAa,EAAE,CAMpC,OAJA,MAAM,IAAI,SAAe,EAAS,IAAW,CACzC,EAAU,KAAK,EAAS,GAAQ,EAAM,EAAO,EAAI,CAAG,GAAS,CAAC,EAChE,CAEK,CAAC,EAAO,EAAO,CAG1B,UAAiB,CACR,AAML,MAAA,KAFA,cAAc,MAAA,EAAuB,CAEZ,IAAA,MAIjC,SAAS,GAAqC,CAC1C,MAAO,CACH,WAAY,EACZ,SAAU,EACV,gBAAiB,EACjB,OAAQ,KAAK,MAAM,KAAK,QAAQ,CAAG,MAAM,CACzC,QAAS,KAAK,MAAM,KAAK,QAAQ,CAAG,WAAW,CAC/C,OAAQ,EACR,QAAS,KAAK,MAAM,EAAc,EAAE,CACpC,WAAY,EACZ,YAAa,EACb,YAAa,GACb,OAAQ,IACR,SAAU,EACV,WAAY,IAAoB,EAAW,EAC3C,UAAW,EACX,YAAa,EAEb,OAAQ,CACJ,KAAK,OAAS,KAAK,MAAM,KAAK,QAAQ,CAAG,MAAM,CAC/C,KAAK,QAAU,KAAK,MAAM,KAAK,QAAQ,CAAG,WAAW,CACrD,KAAK,OAAS,KAAK,QACnB,KAAK,YAAc,EACnB,KAAK,SAAW,GAEvB"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@basmilius/apple-raop",
3
3
  "description": "Implementation of Apple's RAOP protocol in Node.js.",
4
- "version": "0.7.2",
4
+ "version": "0.8.1",
5
5
  "type": "module",
6
6
  "license": "MIT",
7
7
  "author": {
@@ -26,27 +26,27 @@
26
26
  "provenance": false
27
27
  },
28
28
  "scripts": {
29
- "build": "tsgo && bun -b build.ts",
29
+ "build": "tsgo --noEmit && tsdown",
30
+ "dev": "tsdown --watch",
30
31
  "watch:test": "bun --watch test.ts"
31
32
  },
32
- "main": "./dist/index.js",
33
- "types": "./dist/index.d.ts",
34
- "typings": "./dist/index.d.ts",
33
+ "main": "./dist/index.mjs",
34
+ "types": "./dist/index.d.mts",
35
+ "typings": "./dist/index.d.mts",
35
36
  "sideEffects": false,
36
37
  "exports": {
37
38
  ".": {
38
- "types": "./dist/index.d.ts",
39
- "default": "./dist/index.js"
39
+ "types": "./dist/index.d.mts",
40
+ "default": "./dist/index.mjs"
40
41
  }
41
42
  },
42
43
  "dependencies": {
43
- "@basmilius/apple-airplay": "0.7.2",
44
- "@basmilius/apple-common": "0.7.2",
45
- "@basmilius/apple-encoding": "0.7.2",
46
- "@basmilius/apple-encryption": "0.7.2"
44
+ "@basmilius/apple-common": "0.8.1",
45
+ "@basmilius/apple-encoding": "0.8.1",
46
+ "@basmilius/apple-encryption": "0.8.1"
47
47
  },
48
48
  "devDependencies": {
49
- "@basmilius/tools": "^2.25.0",
50
- "@types/bun": "^1.3.9"
49
+ "@types/bun": "^1.3.9",
50
+ "tsdown": "^0.21.0-beta.2"
51
51
  }
52
52
  }
package/dist/const.d.ts DELETED
@@ -1,8 +0,0 @@
1
- import { type MediaMetadata } from "./types";
2
- export declare const MAX_PACKETS_COMPENSATE = 3;
3
- export declare const PACKET_BACKLOG_SIZE = 1e3;
4
- export declare const SLOW_WARNING_THRESHOLD = 5;
5
- export declare const FRAMES_PER_PACKET = 352;
6
- export declare const MISSING_METADATA: MediaMetadata;
7
- export declare const EMPTY_METADATA: MediaMetadata;
8
- export declare const SUPPORTED_ENCRYPTIONS: unknown;
@@ -1,12 +0,0 @@
1
- import { EventEmitter } from "node:events";
2
- import { PacketFifo } from "./packets";
3
- import type { StreamContext } from "./types";
4
- export default class ControlClient extends EventEmitter {
5
- #private;
6
- constructor(context: StreamContext, packetBacklog: PacketFifo);
7
- get port(): number;
8
- bind(localIp: string, port: number): Promise<void>;
9
- close(): void;
10
- start(remoteAddr: string): void;
11
- stop(): void;
12
- }
package/dist/index.d.ts DELETED
@@ -1,8 +0,0 @@
1
- export * from "./types";
2
- export * from "./packets";
3
- export * from "./utils";
4
- export { default as ControlClient } from "./controlClient";
5
- export { default as RtspClient } from "./rtspClient";
6
- export { default as Statistics } from "./statistics";
7
- export { default as StreamClient } from "./streamClient";
8
- export { RaopClient, type StreamOptions } from "./raop";