Bug 1876865 - Add media.peerconnection.sdp.quirk.duplicate_fingerprint.allowlist. r=bwc,webidl,smaug

Differential Revision: https://phabricator.services.mozilla.com/D217016
This commit is contained in:
Jan-Ivar Bruaroey 2024-07-23 22:39:26 +00:00
Родитель c4707fa06b
Коммит d978eb6a0b
6 изменённых файлов: 148 добавлений и 53 удалений

Просмотреть файл

@ -1029,10 +1029,33 @@ export class RTCPeerConnection {
});
}
_maybeDuplicateFingerprints(sdp) {
if (this._pc.duplicateFingerprintQuirk) {
// hack to put a=fingerprint under all m= lines
const lines = sdp.split("\n");
const fingerprint = lines.find(line => line.startsWith("a=fingerprint"));
if (fingerprint) {
for (let i = 0; i < lines.length; i++) {
if (lines[i].startsWith("m=")) {
let j = i + 1;
while (j < lines.length && lines[j].startsWith("c=")) {
j++;
}
lines.splice(j, 0, fingerprint);
}
}
sdp = lines.join("\n");
}
}
return sdp;
}
_createOffer(options) {
this._checkClosed();
this._ensureTransceiversForOfferToReceive(options);
return this._chain(() => this._createAnOffer(options));
return this._chain(async () =>
Cu.cloneInto(await this._createAnOffer(options), this._win)
);
}
async _createAnOffer(options = {}) {
@ -1057,11 +1080,12 @@ export class RTCPeerConnection {
this._onCreateOfferFailure = reject;
this._pc.createOffer(options);
});
sdp = this._maybeDuplicateFingerprints(sdp);
if (haveAssertion) {
await haveAssertion;
sdp = this._localIdp.addIdentityAttribute(sdp);
}
return Cu.cloneInto({ type: "offer", sdp }, this._win);
return { type: "offer", sdp };
}
createAnswer(optionsOrOnSucc, onErr) {
@ -1074,7 +1098,9 @@ export class RTCPeerConnection {
_createAnswer() {
this._checkClosed();
return this._chain(() => this._createAnAnswer());
return this._chain(async () =>
Cu.cloneInto(await this._createAnAnswer(), this._win)
);
}
async _createAnAnswer() {
@ -1095,11 +1121,12 @@ export class RTCPeerConnection {
this._onCreateAnswerFailure = reject;
this._pc.createAnswer();
});
sdp = this._maybeDuplicateFingerprints(sdp);
if (haveAssertion) {
await haveAssertion;
sdp = this._localIdp.addIdentityAttribute(sdp);
}
return Cu.cloneInto({ type: "answer", sdp }, this._win);
return { type: "answer", sdp };
}
async _getPermission() {
@ -1175,9 +1202,9 @@ export class RTCPeerConnection {
}
if (!sdp) {
if (type == "offer") {
await this._createAnOffer();
sdp = (await this._createAnOffer()).sdp;
} else if (type == "answer") {
await this._createAnAnswer();
sdp = (await this._createAnAnswer()).sdp;
}
} else {
this._sanityCheckSdp(sdp);

Просмотреть файл

@ -398,6 +398,9 @@ PeerConnectionImpl::PeerConnectionImpl(const GlobalObject* aGlobal)
mRtxIsAllowed = !HostnameInPref(
"media.peerconnection.video.use_rtx.blocklist", mHostname);
mDuplicateFingerprintQuirk = HostnameInPref(
"media.peerconnection.sdp.quirk.duplicate_fingerprint.allowlist",
mHostname);
}
}

Просмотреть файл

@ -346,6 +346,8 @@ class PeerConnectionImpl final
PrincipalPrivacy::Private;
}
bool DuplicateFingerprintQuirk() { return mDuplicateFingerprintQuirk; }
NS_IMETHODIMP GetFingerprint(char** fingerprint);
void GetFingerprint(nsAString& fingerprint) {
char* tmp;
@ -869,6 +871,8 @@ class PeerConnectionImpl final
// See Bug 1642419, this can be removed when all sites are working with RTX.
bool mRtxIsAllowed = true;
bool mDuplicateFingerprintQuirk = false;
nsTArray<RefPtr<Operation>> mOperations;
bool mChainingOperation = false;
bool mUpdateNegotiationNeededFlagOnEmptyChain = false;

Просмотреть файл

@ -19,59 +19,117 @@
is(d1.sdp, d2.sdp, msg + " — and same sdp");
}
runNetworkTest(async () => {
await SpecialPowers.pushPrefEnv({
set: [['media.peerconnection.description.legacy.enabled', true]]
});
const addIce = (pc, can, failed) => can && pc.addIceCandidate(can).catch(failed);
const pc1 = new RTCPeerConnection();
const pc2 = new RTCPeerConnection();
const tests = [
async function checkLegacyDescriptions() {
await withPrefs([["media.peerconnection.description.legacy.enabled", true]],
async () => {
const pc1 = new RTCPeerConnection();
const pc2 = new RTCPeerConnection();
const add = (pc, can, failed) => can && pc.addIceCandidate(can).catch(failed);
pc1.onicecandidate = e => add(pc2, e.candidate, generateErrorCallback());
pc2.onicecandidate = e => add(pc1, e.candidate, generateErrorCallback());
pc1.onicecandidate = e => addIce(pc2, e.candidate, generateErrorCallback());
pc2.onicecandidate = e => addIce(pc1, e.candidate, generateErrorCallback());
const v1 = createMediaElement('video', 'v1');
const v2 = createMediaElement('video', 'v2');
const v1 = createMediaElement('video', 'v1');
const v2 = createMediaElement('video', 'v2');
const stream = await navigator.mediaDevices.getUserMedia({
video: true, audio: true
});
v1.srcObject = stream;
for (const track of stream.getTracks()) {
pc1.addTrack(track, stream);
const stream = await navigator.mediaDevices.getUserMedia({
video: true, audio: true
});
v1.srcObject = stream;
for (const track of stream.getTracks()) {
pc1.addTrack(track, stream);
}
await pc1.setLocalDescription();
is(pc1.currentLocalDescription, null, "pc1 currentLocalDescription is null");
ok(pc1.pendingLocalDescription, "pc1 pendingLocalDescription is set");
ok(pc1.localDescription, "pc1 localDescription is set");
isSimilarDescription(pc1.pendingLocalDescription, pc1.pendingLocalDescription, "pendingLocalDescription");
isSimilarDescription(pc1.localDescription, pc1.localDescription, "localDescription");
isSimilarDescription(pc1.localDescription, pc1.pendingLocalDescription, "local and pending");
await pc2.setRemoteDescription(pc1.localDescription);
is(pc2.currentRemoteDescription, null, "pc2 currentRemoteDescription is null");
ok(pc2.pendingRemoteDescription, "pc2 pendingRemoteDescription is set");
ok(pc2.remoteDescription, "pc2 remoteDescription is set");
isSimilarDescription(pc2.pendingRemoteDescription, pc2.pendingRemoteDescription, "pendingRemoteDescription");
isSimilarDescription(pc2.remoteDescription, pc2.remoteDescription, "remoteDescription");
isSimilarDescription(pc2.remoteDescription, pc2.pendingRemoteDescription, "remote and pending");
await pc2.setLocalDescription();
ok(pc2.currentLocalDescription, "pc2 currentLocalDescription is set");
is(pc2.pendingLocalDescription, null, "pc2 pendingLocalDescription is null");
ok(pc2.localDescription, "pc2 localDescription is set");
isSimilarDescription(pc2.currentLocalDescription, pc2.currentLocalDescription, "currentLocalDescription");
isSimilarDescription(pc2.localDescription, pc2.localDescription, "localDescription");
isSimilarDescription(pc2.localDescription, pc2.currentLocalDescription, "local and current");
await pc1.setRemoteDescription(pc2.localDescription);
ok(pc1.currentRemoteDescription, "pc1 currentRemoteDescription is set");
is(pc1.pendingRemoteDescription, null, "pc1 pendingRemoteDescription is null");
ok(pc1.remoteDescription, "pc1 remoteDescription is set");
isSimilarDescription(pc1.currentRemoteDescription, pc1.currentRemoteDescription, "currentRemoteDescription");
isSimilarDescription(pc1.remoteDescription, pc1.remoteDescription, "remoteDescription");
isSimilarDescription(pc1.remoteDescription, pc1.currentRemoteDescription, "remote and current");
}
);
},
async function checkDuplicateFingerprintQuirk() {
await withPrefs([["media.peerconnection.sdp.quirk.duplicate_fingerprint.allowlist", "example.com"]],
async () => {
const pc1 = new RTCPeerConnection();
const pc2 = new RTCPeerConnection();
pc1.onicecandidate = e => addIce(pc2, e.candidate, generateErrorCallback());
pc2.onicecandidate = e => addIce(pc1, e.candidate, generateErrorCallback());
const stream = await navigator.mediaDevices.getUserMedia({
video: true, audio: true
});
v1.srcObject = stream;
for (const track of stream.getTracks()) {
pc1.addTrack(track, stream);
}
const fingerprints = ({sdp}) => sdp.match(/a=fingerprint/g).length;
// with implicit offer and answer
await pc1.setLocalDescription();
is(pc1.pendingLocalDescription, pc1.localDescription);
is(fingerprints(pc1.localDescription), 3, "implicit pc1 fingerprints");
await pc2.setRemoteDescription(pc1.localDescription);
await pc2.setLocalDescription();
is(pc2.currentLocalDescription, pc2.localDescription);
is(fingerprints(pc2.localDescription), 3, "implicit pc2 fingerprints");
await pc1.setRemoteDescription(pc2.localDescription);
// with explicit offer and answer
const offer = await pc1.createOffer();
is(fingerprints(offer), 3, "offer fingerprints");
await pc1.setLocalDescription(offer);
is(pc1.pendingLocalDescription, pc1.localDescription);
is(fingerprints(pc1.localDescription), 3, "explicit pc1 fingerprints");
await pc2.setRemoteDescription(pc1.localDescription);
const answer = pc2.createAnswer();
is(fingerprints(offer), 3, "answer fingerprints");
await pc2.setLocalDescription(answer);
is(pc2.currentLocalDescription, pc2.localDescription);
is(fingerprints(pc2.localDescription), 3, "explicit pc2 fingerprints");
await pc1.setRemoteDescription(pc2.localDescription);
}
);
}
await pc1.setLocalDescription();
is(pc1.currentLocalDescription, null, "pc1 currentLocalDescription is null");
ok(pc1.pendingLocalDescription, "pc1 pendingLocalDescription is set");
ok(pc1.localDescription, "pc1 localDescription is set");
isSimilarDescription(pc1.pendingLocalDescription, pc1.pendingLocalDescription, "pendingLocalDescription");
isSimilarDescription(pc1.localDescription, pc1.localDescription, "localDescription");
isSimilarDescription(pc1.localDescription, pc1.pendingLocalDescription, "local and pending");
];
await pc2.setRemoteDescription(pc1.localDescription);
is(pc2.currentRemoteDescription, null, "pc2 currentRemoteDescription is null");
ok(pc2.pendingRemoteDescription, "pc2 pendingRemoteDescription is set");
ok(pc2.remoteDescription, "pc2 remoteDescription is set");
isSimilarDescription(pc2.pendingRemoteDescription, pc2.pendingRemoteDescription, "pendingRemoteDescription");
isSimilarDescription(pc2.remoteDescription, pc2.remoteDescription, "remoteDescription");
isSimilarDescription(pc2.remoteDescription, pc2.pendingRemoteDescription, "remote and pending");
await pc2.setLocalDescription();
ok(pc2.currentLocalDescription, "pc2 currentLocalDescription is set");
is(pc2.pendingLocalDescription, null, "pc2 pendingLocalDescription is null");
ok(pc2.localDescription, "pc2 localDescription is set");
isSimilarDescription(pc2.currentLocalDescription, pc2.currentLocalDescription, "currentLocalDescription");
isSimilarDescription(pc2.localDescription, pc2.localDescription, "localDescription");
isSimilarDescription(pc2.localDescription, pc2.currentLocalDescription, "local and current");
await pc1.setRemoteDescription(pc2.localDescription);
ok(pc1.currentRemoteDescription, "pc1 currentRemoteDescription is set");
is(pc1.pendingRemoteDescription, null, "pc1 pendingRemoteDescription is null");
ok(pc1.remoteDescription, "pc1 remoteDescription is set");
isSimilarDescription(pc1.currentRemoteDescription, pc1.currentRemoteDescription, "currentRemoteDescription");
isSimilarDescription(pc1.remoteDescription, pc1.remoteDescription, "remoteDescription");
isSimilarDescription(pc1.remoteDescription, pc1.currentRemoteDescription, "remote and current");
runNetworkTest(async () => {
for (const test of tests) {
info(`Running test: ${test.name}`);
await test();
info(`Done running test: ${test.name}`);
}
});
</script>
</pre>

Просмотреть файл

@ -117,6 +117,8 @@ interface PeerConnectionImpl {
attribute DOMString peerIdentity;
readonly attribute boolean privacyRequested;
readonly attribute boolean duplicateFingerprintQuirk;
readonly attribute RTCSctpTransport? sctp;
/* Data channels */

Просмотреть файл

@ -240,6 +240,7 @@ pref("media.videocontrols.keyboard-tab-to-all-controls", true);
pref("media.navigator.video.use_transport_cc", true);
pref("media.peerconnection.video.use_rtx", true);
pref("media.peerconnection.video.use_rtx.blocklist", "doxy.me,*.doxy.me");
pref("media.peerconnection.sdp.quirk.duplicate_fingerprint.allowlist", "");
pref("media.navigator.video.use_tmmbr", false);
pref("media.navigator.audio.use_fec", true);
pref("media.navigator.video.offer_rtcp_rsize", true);