@lox-audioserver/node-airplay-sender 0.4.0
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/README.md +93 -0
- package/dist/core/ap2_test.d.ts +1 -0
- package/dist/core/ap2_test.js +8 -0
- package/dist/core/atv.d.ts +16 -0
- package/dist/core/atv.js +215 -0
- package/dist/core/atvAuthenticator.d.ts +30 -0
- package/dist/core/atvAuthenticator.js +134 -0
- package/dist/core/audioOut.d.ts +43 -0
- package/dist/core/audioOut.js +220 -0
- package/dist/core/deviceAirtunes.d.ts +76 -0
- package/dist/core/deviceAirtunes.js +536 -0
- package/dist/core/devices.d.ts +50 -0
- package/dist/core/devices.js +221 -0
- package/dist/core/index.d.ts +56 -0
- package/dist/core/index.js +144 -0
- package/dist/core/rtsp.d.ts +12 -0
- package/dist/core/rtsp.js +1678 -0
- package/dist/core/srp.d.ts +14 -0
- package/dist/core/srp.js +128 -0
- package/dist/core/udpServers.d.ts +44 -0
- package/dist/core/udpServers.js +244 -0
- package/dist/esm/core/ap2_test.js +8 -0
- package/dist/esm/core/atv.js +215 -0
- package/dist/esm/core/atvAuthenticator.js +134 -0
- package/dist/esm/core/audioOut.js +220 -0
- package/dist/esm/core/deviceAirtunes.js +536 -0
- package/dist/esm/core/devices.js +221 -0
- package/dist/esm/core/index.js +144 -0
- package/dist/esm/core/rtsp.js +1678 -0
- package/dist/esm/core/srp.js +128 -0
- package/dist/esm/core/udpServers.js +244 -0
- package/dist/esm/homekit/credentials.js +109 -0
- package/dist/esm/homekit/encryption.js +82 -0
- package/dist/esm/homekit/number.js +47 -0
- package/dist/esm/homekit/tlv.js +97 -0
- package/dist/esm/index.js +310 -0
- package/dist/esm/package.json +1 -0
- package/dist/esm/utils/alac.js +62 -0
- package/dist/esm/utils/alacEncoder.js +34 -0
- package/dist/esm/utils/circularBuffer.js +132 -0
- package/dist/esm/utils/config.js +49 -0
- package/dist/esm/utils/http.js +148 -0
- package/dist/esm/utils/ntp.js +56 -0
- package/dist/esm/utils/numUtil.js +17 -0
- package/dist/esm/utils/packetPool.js +52 -0
- package/dist/esm/utils/util.js +9 -0
- package/dist/homekit/credentials.d.ts +31 -0
- package/dist/homekit/credentials.js +109 -0
- package/dist/homekit/encryption.d.ts +12 -0
- package/dist/homekit/encryption.js +82 -0
- package/dist/homekit/number.d.ts +7 -0
- package/dist/homekit/number.js +47 -0
- package/dist/homekit/tlv.d.ts +25 -0
- package/dist/homekit/tlv.js +97 -0
- package/dist/index.d.ts +121 -0
- package/dist/index.js +310 -0
- package/dist/utils/alac.d.ts +9 -0
- package/dist/utils/alac.js +62 -0
- package/dist/utils/alacEncoder.d.ts +14 -0
- package/dist/utils/alacEncoder.js +34 -0
- package/dist/utils/circularBuffer.d.ts +32 -0
- package/dist/utils/circularBuffer.js +132 -0
- package/dist/utils/config.d.ts +42 -0
- package/dist/utils/config.js +49 -0
- package/dist/utils/http.d.ts +19 -0
- package/dist/utils/http.js +148 -0
- package/dist/utils/ntp.d.ts +21 -0
- package/dist/utils/ntp.js +56 -0
- package/dist/utils/numUtil.d.ts +5 -0
- package/dist/utils/numUtil.js +17 -0
- package/dist/utils/packetPool.d.ts +25 -0
- package/dist/utils/packetPool.js +52 -0
- package/dist/utils/util.d.ts +2 -0
- package/dist/utils/util.js +9 -0
- package/package.json +71 -0
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const elliptic = require('elliptic');
|
|
40
|
+
const crypto_1 = __importDefault(require("crypto"));
|
|
41
|
+
const ed = __importStar(require("@noble/ed25519"));
|
|
42
|
+
const util_1 = require("../utils/util");
|
|
43
|
+
// ...
|
|
44
|
+
// Note: All functions expect parameters to be hex strings.
|
|
45
|
+
function pair_setup_aes_key(K) {
|
|
46
|
+
return crypto_1.default.createHash('sha512')
|
|
47
|
+
.update('Pair-Setup-AES-Key')
|
|
48
|
+
.update((0, util_1.hexString2ArrayBuffer)(K))
|
|
49
|
+
.digest('hex')
|
|
50
|
+
.substring(0, 32);
|
|
51
|
+
}
|
|
52
|
+
function pair_setup_aes_iv(K) {
|
|
53
|
+
let ab = crypto_1.default.createHash('sha512')
|
|
54
|
+
.update('Pair-Setup-AES-IV')
|
|
55
|
+
.update((0, util_1.hexString2ArrayBuffer)(K))
|
|
56
|
+
.digest();
|
|
57
|
+
ab = ab.slice(0, 16);
|
|
58
|
+
ab[ab.length - 1] += 0x01;
|
|
59
|
+
return (0, util_1.buf2hex)(ab);
|
|
60
|
+
}
|
|
61
|
+
function pair_verify_aes_key(shared) {
|
|
62
|
+
return (0, util_1.buf2hex)(crypto_1.default.createHash('sha512')
|
|
63
|
+
.update('Pair-Verify-AES-Key')
|
|
64
|
+
.update((0, util_1.hexString2ArrayBuffer)(shared))
|
|
65
|
+
.digest()
|
|
66
|
+
.slice(0, 16));
|
|
67
|
+
}
|
|
68
|
+
function pair_verify_aes_iv(shared) {
|
|
69
|
+
return (0, util_1.buf2hex)(crypto_1.default.createHash('sha512')
|
|
70
|
+
.update('Pair-Verify-AES-IV')
|
|
71
|
+
.update((0, util_1.hexString2ArrayBuffer)(shared))
|
|
72
|
+
.digest()
|
|
73
|
+
.slice(0, 16));
|
|
74
|
+
}
|
|
75
|
+
// ...
|
|
76
|
+
// Public.
|
|
77
|
+
function a_pub(a) {
|
|
78
|
+
return elliptic.utils.toHex(new elliptic.eddsa('ed25519').keyFromSecret(a).getPublic());
|
|
79
|
+
}
|
|
80
|
+
function confirm(a, K) {
|
|
81
|
+
const key = pair_setup_aes_key(K);
|
|
82
|
+
const iv = pair_setup_aes_iv(K);
|
|
83
|
+
const cipher = crypto_1.default.createCipheriv('aes-128-gcm', (0, util_1.hexString2ArrayBuffer)(key), (0, util_1.hexString2ArrayBuffer)(iv));
|
|
84
|
+
const encrypted = Buffer.concat([
|
|
85
|
+
cipher.update((0, util_1.hexString2ArrayBuffer)(a_pub(a))),
|
|
86
|
+
cipher.final(),
|
|
87
|
+
]);
|
|
88
|
+
return {
|
|
89
|
+
epk: encrypted.toString('hex'),
|
|
90
|
+
authTag: (0, util_1.buf2hex)(cipher.getAuthTag()),
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
function verifier(a) {
|
|
94
|
+
const privateKey = Buffer.from(ed.utils.randomPrivateKey());
|
|
95
|
+
const publicKey = Buffer.from(ed.curve25519.scalarMultBase(privateKey));
|
|
96
|
+
const v_pri = (0, util_1.buf2hex)(privateKey);
|
|
97
|
+
const v_pub = (0, util_1.buf2hex)(publicKey);
|
|
98
|
+
const header = Buffer.from([0x01, 0x00, 0x00, 0x00]);
|
|
99
|
+
const a_pub_buf = Buffer.from(a_pub(a), 'hex');
|
|
100
|
+
return {
|
|
101
|
+
verifierBody: Buffer.concat([header, publicKey, a_pub_buf], header.byteLength + publicKey.byteLength + a_pub_buf.byteLength),
|
|
102
|
+
v_pri,
|
|
103
|
+
v_pub
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
function shared(v_pri, atv_pub) {
|
|
107
|
+
return (0, util_1.buf2hex)(Buffer.from(ed.curve25519.scalarMult((0, util_1.hexString2ArrayBuffer)(v_pri), (0, util_1.hexString2ArrayBuffer)(atv_pub))));
|
|
108
|
+
}
|
|
109
|
+
function signed(a, v_pub, atv_pub) {
|
|
110
|
+
const key = new elliptic.eddsa('ed25519').keyFromSecret(a);
|
|
111
|
+
return key.sign(v_pub + atv_pub).toHex();
|
|
112
|
+
}
|
|
113
|
+
function signature(shared, atv_data, signed) {
|
|
114
|
+
const cipher = crypto_1.default.createCipheriv('aes-128-ctr', (0, util_1.hexString2ArrayBuffer)(pair_verify_aes_key(shared)), (0, util_1.hexString2ArrayBuffer)(pair_verify_aes_iv(shared)));
|
|
115
|
+
// discard the result of encrypting atv_data.
|
|
116
|
+
cipher.update((0, util_1.hexString2ArrayBuffer)(atv_data));
|
|
117
|
+
const encrypted = Buffer.concat([
|
|
118
|
+
cipher.update(Buffer.from(signed, 'hex')),
|
|
119
|
+
cipher.final(),
|
|
120
|
+
]);
|
|
121
|
+
return encrypted.toString('hex');
|
|
122
|
+
}
|
|
123
|
+
exports.default = {
|
|
124
|
+
pair_setup_aes_key,
|
|
125
|
+
pair_setup_aes_iv,
|
|
126
|
+
pair_verify_aes_key,
|
|
127
|
+
pair_verify_aes_iv,
|
|
128
|
+
a_pub,
|
|
129
|
+
confirm,
|
|
130
|
+
verifier,
|
|
131
|
+
shared,
|
|
132
|
+
signed,
|
|
133
|
+
signature,
|
|
134
|
+
};
|
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const node_events_1 = require("node:events");
|
|
40
|
+
const node_perf_hooks_1 = require("node:perf_hooks");
|
|
41
|
+
const config_1 = __importDefault(require("../utils/config"));
|
|
42
|
+
const numUtil_1 = require("../utils/numUtil");
|
|
43
|
+
const ntp_1 = __importStar(require("../utils/ntp"));
|
|
44
|
+
const SEQ_NUM_WRAP = Math.pow(2, 16);
|
|
45
|
+
const FRAC_PER_SEC = 0x1_0000_0000;
|
|
46
|
+
/**
|
|
47
|
+
* Generates RTP timestamps and sequence, pulling PCM/ALAC packets from a circular buffer.
|
|
48
|
+
* Emits `packet` events for devices and sync requests (`need_sync`) at intervals.
|
|
49
|
+
*/
|
|
50
|
+
class AudioOut extends node_events_1.EventEmitter {
|
|
51
|
+
lastSeq = -1;
|
|
52
|
+
lastWireSeq = 0;
|
|
53
|
+
hasAirTunes = false;
|
|
54
|
+
rtpTimeRef = 0;
|
|
55
|
+
startTimeMs;
|
|
56
|
+
startTimeNtp;
|
|
57
|
+
latencyFrames = 0;
|
|
58
|
+
latencyApplied = false;
|
|
59
|
+
seqOffset = 0;
|
|
60
|
+
tsOffset = 0;
|
|
61
|
+
syncOffsetFrames = 0;
|
|
62
|
+
deviceMagic = 0;
|
|
63
|
+
packetCount = 0;
|
|
64
|
+
octetCount = 0;
|
|
65
|
+
frameDurationMs = (config_1.default.frames_per_packet / config_1.default.sampling_rate) * 1000;
|
|
66
|
+
muteUntilMs = 0;
|
|
67
|
+
/**
|
|
68
|
+
* Begin pulling from the buffer and emitting packets at the configured cadence.
|
|
69
|
+
* @param devices Device manager for sync events.
|
|
70
|
+
* @param circularBuffer PCM/ALAC buffer.
|
|
71
|
+
* @param startTimeMs Optional unix ms to align playback.
|
|
72
|
+
* @param startTimeNtp Optional NTP uint64 (sec<<32|frac) to align playback.
|
|
73
|
+
*/
|
|
74
|
+
init(devices, circularBuffer, startTimeMs, startTimeNtp, deviceMagic, underrunMuteMs) {
|
|
75
|
+
this.startTimeMs =
|
|
76
|
+
typeof startTimeMs === 'number' && Number.isFinite(startTimeMs)
|
|
77
|
+
? startTimeMs
|
|
78
|
+
: undefined;
|
|
79
|
+
this.startTimeNtp = startTimeNtp;
|
|
80
|
+
this.seqOffset = Math.floor(Math.random() * SEQ_NUM_WRAP);
|
|
81
|
+
this.tsOffset = Math.floor(Math.random() * 0xffffffff);
|
|
82
|
+
this.syncOffsetFrames = this.tsOffset / config_1.default.frames_per_packet;
|
|
83
|
+
this.deviceMagic = typeof deviceMagic === 'number' ? deviceMagic : config_1.default.device_magic;
|
|
84
|
+
if (typeof underrunMuteMs === 'number') {
|
|
85
|
+
config_1.default.underrun_mute_ms = underrunMuteMs;
|
|
86
|
+
}
|
|
87
|
+
circularBuffer.on('underrun', () => this.handleUnderrun());
|
|
88
|
+
const monoNow = node_perf_hooks_1.performance.now();
|
|
89
|
+
const wallToMonoOffset = Date.now() - monoNow;
|
|
90
|
+
if (typeof this.startTimeNtp === 'bigint' || typeof this.startTimeNtp === 'number') {
|
|
91
|
+
const { sec, frac } = (0, ntp_1.parseNtpTimestamp)(this.startTimeNtp);
|
|
92
|
+
const unixMs = (sec - config_1.default.ntp_epoch) * 1000 + Math.floor((frac * 1000) / FRAC_PER_SEC);
|
|
93
|
+
const nowMs = config_1.default.use_monotonic_clock ? monoNow : Date.now();
|
|
94
|
+
const delta = unixMs - nowMs;
|
|
95
|
+
// If target is too close/past, clamp to now.
|
|
96
|
+
this.rtpTimeRef = delta > -config_1.default.stream_latency ? unixMs - wallToMonoOffset : nowMs;
|
|
97
|
+
}
|
|
98
|
+
else if (this.startTimeMs !== undefined) {
|
|
99
|
+
const nowMs = config_1.default.use_monotonic_clock ? monoNow : Date.now();
|
|
100
|
+
const delta = this.startTimeMs - nowMs;
|
|
101
|
+
this.rtpTimeRef = delta > -config_1.default.stream_latency ? this.startTimeMs - wallToMonoOffset : nowMs;
|
|
102
|
+
}
|
|
103
|
+
else if (config_1.default.use_monotonic_clock) {
|
|
104
|
+
this.rtpTimeRef = monoNow;
|
|
105
|
+
}
|
|
106
|
+
else {
|
|
107
|
+
this.rtpTimeRef = Date.now();
|
|
108
|
+
}
|
|
109
|
+
devices.on('airtunes_devices', (hasAirTunes) => {
|
|
110
|
+
this.hasAirTunes = hasAirTunes;
|
|
111
|
+
});
|
|
112
|
+
devices.on('need_sync', () => {
|
|
113
|
+
this.emit('need_sync', { seq: this.lastWireSeq, tsOffsetFrames: this.syncOffsetFrames });
|
|
114
|
+
});
|
|
115
|
+
const syncEvery = config_1.default.sync_period && config_1.default.sync_period > 0
|
|
116
|
+
? config_1.default.sync_period
|
|
117
|
+
: Math.max(1, Math.round(config_1.default.sampling_rate / config_1.default.frames_per_packet));
|
|
118
|
+
const sendPacket = (seq) => {
|
|
119
|
+
const wireSeq = (this.seqOffset + seq) % SEQ_NUM_WRAP;
|
|
120
|
+
const packet = circularBuffer.readPacket();
|
|
121
|
+
packet.seq = wireSeq;
|
|
122
|
+
packet.timestamp = (0, numUtil_1.low32)(this.tsOffset + wireSeq * config_1.default.frames_per_packet + 2 * config_1.default.sampling_rate);
|
|
123
|
+
this.packetCount += 1;
|
|
124
|
+
this.octetCount += packet.pcm.length;
|
|
125
|
+
if (this.hasAirTunes && seq % syncEvery === 0) {
|
|
126
|
+
this.emit('need_sync', {
|
|
127
|
+
seq: wireSeq,
|
|
128
|
+
tsOffsetFrames: this.syncOffsetFrames,
|
|
129
|
+
rtcp: {
|
|
130
|
+
rtpTimestamp: packet.timestamp,
|
|
131
|
+
ntp: ntp_1.default.timestamp(),
|
|
132
|
+
packetCount: this.packetCount,
|
|
133
|
+
octetCount: this.octetCount,
|
|
134
|
+
xr: { ntp: ntp_1.default.timestamp() },
|
|
135
|
+
},
|
|
136
|
+
});
|
|
137
|
+
const nowMs = config_1.default.use_monotonic_clock ? node_perf_hooks_1.performance.now() : Date.now();
|
|
138
|
+
const expectedTimeMs = this.rtpTimeRef +
|
|
139
|
+
(((seq + this.syncOffsetFrames) * config_1.default.frames_per_packet) / config_1.default.sampling_rate) * 1000;
|
|
140
|
+
const deltaMs = nowMs - expectedTimeMs;
|
|
141
|
+
this.emit('metrics', { type: 'sync', seq: wireSeq, deltaMs, latencyFrames: this.latencyFrames });
|
|
142
|
+
}
|
|
143
|
+
this.emit('packet', packet);
|
|
144
|
+
if (this.muteUntilMs > 0) {
|
|
145
|
+
const nowMsSend = config_1.default.use_monotonic_clock ? node_perf_hooks_1.performance.now() : Date.now();
|
|
146
|
+
if (nowMsSend < this.muteUntilMs) {
|
|
147
|
+
packet.pcm.fill(0);
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
this.muteUntilMs = 0;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
packet.release();
|
|
154
|
+
this.lastWireSeq = wireSeq;
|
|
155
|
+
};
|
|
156
|
+
const syncAudio = () => {
|
|
157
|
+
const nowMs = config_1.default.use_monotonic_clock ? node_perf_hooks_1.performance.now() : Date.now();
|
|
158
|
+
const elapsed = nowMs - this.rtpTimeRef;
|
|
159
|
+
if (elapsed < 0) {
|
|
160
|
+
setTimeout(syncAudio, Math.min(config_1.default.stream_latency, Math.abs(elapsed)));
|
|
161
|
+
return;
|
|
162
|
+
}
|
|
163
|
+
let currentSeq = Math.floor((elapsed * config_1.default.sampling_rate) / (config_1.default.frames_per_packet * 1000));
|
|
164
|
+
// If we're lagging behind significantly, jump forward to avoid long hitches.
|
|
165
|
+
if (config_1.default.jump_forward_enabled) {
|
|
166
|
+
const expectedTimeMs = this.rtpTimeRef + currentSeq * this.frameDurationMs;
|
|
167
|
+
const deltaMs = nowMs - expectedTimeMs;
|
|
168
|
+
if (deltaMs > config_1.default.jump_forward_threshold_ms) {
|
|
169
|
+
const jumpSeq = Math.ceil((config_1.default.jump_forward_lead_ms * config_1.default.sampling_rate) /
|
|
170
|
+
(config_1.default.frames_per_packet * 1000));
|
|
171
|
+
const newSeq = currentSeq + jumpSeq;
|
|
172
|
+
this.rtpTimeRef = nowMs - newSeq * this.frameDurationMs;
|
|
173
|
+
this.lastSeq = newSeq - 1;
|
|
174
|
+
currentSeq = newSeq;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
for (let i = this.lastSeq + 1; i <= currentSeq; i += 1) {
|
|
178
|
+
sendPacket(i);
|
|
179
|
+
}
|
|
180
|
+
this.lastSeq = currentSeq;
|
|
181
|
+
setTimeout(syncAudio, config_1.default.stream_latency);
|
|
182
|
+
};
|
|
183
|
+
// If a future start is scheduled, defer until then; otherwise start immediately.
|
|
184
|
+
const nowMs = config_1.default.use_monotonic_clock ? node_perf_hooks_1.performance.now() : Date.now();
|
|
185
|
+
const delayMs = Math.max(0, this.rtpTimeRef - nowMs);
|
|
186
|
+
if (delayMs > 0) {
|
|
187
|
+
setTimeout(syncAudio, delayMs);
|
|
188
|
+
}
|
|
189
|
+
else {
|
|
190
|
+
syncAudio();
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Apply latency (in audio frames) when aligning start time.
|
|
195
|
+
*/
|
|
196
|
+
setLatencyFrames(latencyFrames) {
|
|
197
|
+
if (!Number.isFinite(latencyFrames) || latencyFrames <= 0) {
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
this.latencyFrames = latencyFrames;
|
|
201
|
+
if ((this.startTimeMs === undefined && this.startTimeNtp === undefined) || this.latencyApplied) {
|
|
202
|
+
return;
|
|
203
|
+
}
|
|
204
|
+
const latencyMs = (this.latencyFrames / config_1.default.sampling_rate) * 1000;
|
|
205
|
+
this.rtpTimeRef -= latencyMs;
|
|
206
|
+
this.latencyApplied = true;
|
|
207
|
+
}
|
|
208
|
+
handleUnderrun() {
|
|
209
|
+
const nowMs = config_1.default.use_monotonic_clock ? node_perf_hooks_1.performance.now() : Date.now();
|
|
210
|
+
const currentSeq = Math.max(0, this.lastSeq);
|
|
211
|
+
const currentFrames = currentSeq * config_1.default.frames_per_packet;
|
|
212
|
+
const offsetMs = (currentFrames / config_1.default.sampling_rate) * 1000;
|
|
213
|
+
this.rtpTimeRef = nowMs - offsetMs;
|
|
214
|
+
if (config_1.default.underrun_mute_ms > 0) {
|
|
215
|
+
this.muteUntilMs = nowMs + config_1.default.underrun_mute_ms;
|
|
216
|
+
}
|
|
217
|
+
this.emit('underrun');
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
exports.default = AudioOut;
|