Source code

Revision control

Copy as Markdown

Other Tools

Test Info: Warnings

<!doctype html>
<meta charset=utf-8>
<title>payload type handling (assuming rtcp-mux)</title>
<script src="/resources/testharness.js"></script>
<script src="/resources/testharnessreport.js"></script>
<script src="../third_party/sdp/sdp.js"></script>
<script>
'use strict';
function createOfferSdp(extmaps) {
let sdp = `v=0
o=- 0 3 IN IP4 127.0.0.1
s=-
t=0 0
a=fingerprint:sha-256 A7:24:72:CA:6E:02:55:39:BA:66:DF:6E:CC:4C:D8:B0:1A:BF:1A:56:65:7D:F4:03:AD:7E:77:43:2A:29:EC:93
a=ice-ufrag:6HHHdzzeIhkE0CKj
a=ice-pwd:XYDGVpfvklQIEnZ6YnyLsAew
`;
sdp += 'a=group:BUNDLE ' + ['audio', 'video'].filter(kind => extmaps[kind]).join(' ') + '\r\n';
if (extmaps.audio) {
sdp += `m=audio 9 RTP/SAVPF 111
c=IN IP4 0.0.0.0
a=rtcp-mux
a=sendonly
a=mid:audio
a=rtpmap:111 opus/48000/2
a=setup:actpass
` + extmaps.audio.map(ext => SDPUtils.writeExtmap(ext)).join('');
}
if (extmaps.video) {
sdp += `m=video 9 RTP/SAVPF 112
c=IN IP4 0.0.0.0
a=rtcp-mux
a=sendonly
a=mid:video
a=rtpmap:112 VP8/90000
a=setup:actpass
` + extmaps.video.map(ext => SDPUtils.writeExtmap(ext)).join('');
}
return sdp;
}
[
{
audio: [{id: 1, uri: 'urn:ietf:params:rtp-hdrext:sdes:mid'}],
video: [{id: 1, uri: 'urn:ietf:params:rtp-hdrext:sdes:mid'}],
description: 'MID',
},
{
audio: [{id: 1, uri: 'urn:ietf:params:rtp-hdrext:ssrc-audio-level'}],
description: 'Audio level',
},
{
video: [{id: 1, uri: 'urn:3gpp:video-orientation'}],
description: 'Video orientation',
}
].forEach(testcase => {
promise_test(async t => {
const pc = new RTCPeerConnection();
t.add_cleanup(() => pc.close());
await pc.setRemoteDescription({type: 'offer', sdp: createOfferSdp(testcase)});
const answer = await pc.createAnswer();
const sections = SDPUtils.splitSections(answer.sdp);
sections.shift();
sections.forEach(section => {
const rtpParameters = SDPUtils.parseRtpParameters(section);
assert_equals(rtpParameters.headerExtensions.length, 1);
assert_equals(rtpParameters.headerExtensions[0].id, testcase[SDPUtils.getKind(section)][0].id);
assert_equals(rtpParameters.headerExtensions[0].uri, testcase[SDPUtils.getKind(section)][0].uri);
});
}, testcase.description + ' header extension is supported.');
});
promise_test(async t => {
const pc = new RTCPeerConnection();
t.add_cleanup(() => pc.close());
pc.addTransceiver('video');
const offer = await pc.createOffer();
const section = SDPUtils.splitSections(offer.sdp)[1];
const extensions = SDPUtils.matchPrefix(section, 'a=extmap:')
.map(line => SDPUtils.parseExtmap(line));
const extension_not_mid = extensions.find(e => e.uri !== 'urn:ietf:params:rtp-hdrext:sdes:mid');
await pc.setRemoteDescription({type :'offer', sdp: offer.sdp.replace(extension_not_mid.uri, 'bogus')});
await pc.setLocalDescription();
const answer_section = SDPUtils.splitSections(pc.localDescription.sdp)[1];
const answer_extensions = SDPUtils.matchPrefix(answer_section, 'a=extmap:')
.map(line => SDPUtils.parseExtmap(line));
assert_equals(answer_extensions.length, extensions.length - 1);
assert_false(!!extensions.find(e => e.uri === 'bogus'));
for (const answer_extension of answer_extensions) {
assert_true(!!extensions.find(e => e.uri === answer_extension.uri));
}
}, 'Negotiates the subset of supported extensions offered');
promise_test(async t => {
const pc = new RTCPeerConnection();
t.add_cleanup(() => pc.close());
// Some implementations may refuse 15 as invalid id because of
// which only applies to one-byte extensions with ids 0-14.
const sdp = createOfferSdp({audio: [{
id: 15, uri: 'urn:ietf:params:rtp-hdrext:sdes:mid',
}]});
await pc.setRemoteDescription({type: 'offer', sdp});
}, 'Supports header extensions with id=15');
promise_test(async t => {
const pc = new RTCPeerConnection();
t.add_cleanup(() => pc.close());
const sdp = createOfferSdp({audio: [{
id: 16, uri: 'urn:ietf:params:rtp-hdrext:sdes:mid',
}, {
id: 17, uri: 'urn:ietf:params:rtp-hdrext:ssrc-audio-level',
}]});
await pc.setRemoteDescription({type: 'offer', sdp});
await pc.setLocalDescription();
const answer_section = SDPUtils.splitSections(pc.localDescription.sdp)[1];
const answer_extensions = SDPUtils.matchPrefix(answer_section, 'a=extmap:')
.map(line => SDPUtils.parseExtmap(line));
assert_equals(answer_extensions.length, 2);
assert_true(!!answer_extensions.find(e => e.uri === 'urn:ietf:params:rtp-hdrext:sdes:mid'));
assert_true(!!answer_extensions.find(e => e.uri === 'urn:ietf:params:rtp-hdrext:ssrc-audio-level'));
}, 'Supports two-byte header extensions');
</script>