@basmilius/apple-raop 0.9.2 → 0.9.4
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.
- package/dist/index.d.mts +1 -2
- package/dist/index.mjs +1 -2
- package/package.json +4 -4
- package/dist/index.d.mts.map +0 -1
- package/dist/index.mjs.map +0 -1
package/dist/index.d.mts
CHANGED
|
@@ -185,5 +185,4 @@ declare class RaopClient extends EventEmitter<EventMap> {
|
|
|
185
185
|
static discover(deviceId: string, timingServer: TimingServer): Promise<RaopClient>;
|
|
186
186
|
}
|
|
187
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
|
|
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 };
|
package/dist/index.mjs
CHANGED
|
@@ -927,5 +927,4 @@ function createStreamContext() {
|
|
|
927
927
|
}
|
|
928
928
|
|
|
929
929
|
//#endregion
|
|
930
|
-
export { AudioPacketHeader, ControlClient, EncryptionType, MetadataType, PacketFifo, RaopClient, RtspClient, Statistics, StreamClient, SyncPacket, decodeRetransmitRequest, getAudioProperties, getEncryptionTypes, getMetadataTypes, pctToDbfs };
|
|
931
|
-
//# sourceMappingURL=index.mjs.map
|
|
930
|
+
export { AudioPacketHeader, ControlClient, EncryptionType, MetadataType, PacketFifo, RaopClient, RtspClient, Statistics, StreamClient, SyncPacket, decodeRetransmitRequest, getAudioProperties, getEncryptionTypes, getMetadataTypes, pctToDbfs };
|
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.9.
|
|
4
|
+
"version": "0.9.4",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"author": {
|
|
@@ -46,9 +46,9 @@
|
|
|
46
46
|
}
|
|
47
47
|
},
|
|
48
48
|
"dependencies": {
|
|
49
|
-
"@basmilius/apple-common": "0.9.
|
|
50
|
-
"@basmilius/apple-encoding": "0.9.
|
|
51
|
-
"@basmilius/apple-encryption": "0.9.
|
|
49
|
+
"@basmilius/apple-common": "0.9.4",
|
|
50
|
+
"@basmilius/apple-encoding": "0.9.4",
|
|
51
|
+
"@basmilius/apple-encryption": "0.9.4"
|
|
52
52
|
},
|
|
53
53
|
"devDependencies": {
|
|
54
54
|
"@types/bun": "^1.3.9",
|
package/dist/index.d.mts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
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.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
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":"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"}
|