-
Notifications
You must be signed in to change notification settings - Fork 1.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Basic ClearKey support #2268
Basic ClearKey support #2268
Conversation
Great contribution @CSilivestru. Thanks for the detailed explanation. |
I am happy to see movement on Clear Key support improvement! As promised over email, I will try to review from a perspective of interoperability. I would first request that the PR description include some basic guidelines on how this new feature is meant to be used. It is not clear what the new feature even is - is the expectation that by specifying the The test content shown in the PR description does not conform to DASH-IF IOP guidelines. Therefore it is difficult to evaluate this PR from an interoperability standpoint. I would recommend using some well-known conforming content to test instead. On a cursory review, I note at least the following interoperability issues with this test content:
I would encourage you to test with already existing content that is known to be conforming to IOP, such as https://github.com/Axinom/dash-test-vectors. This content has the appropriate Clear Key signaling and will likely be far more useful to you as a benchmark than creating custom test content. This content is also in the builtin streams menu of the sample player, located here in the menu: When I try to play this video as-is from the player in your branch, I do not get any DRM activity and no keys are acquired, so I assume the implementation does not work with this Clear Key content for whatever reason. When I try to add the
|
@sandersaares thank you so much for the detailed review and questions! This is great and I really appreciate the time you took to write that all out. The purpose of this PR is for someone to be able to produce encrypted dash content with a simple key and to provide that key back to the player for decryption. I wanted to avoid the use of an actual CDM or any requests out to a server to fetch the key. My understanding is that ClearKey should be able to be used without the need of a fetch if the key is provided to the player. The manifest was automatically created by You're correct that I did not test this using the vectors provided and I'm sorry about that. I believe that part of the problem here might be how I am actually producing the content using However, going to the ClearKey vector example you provided, I noticed that the ContentProtection nodes containing the ClearKey KID attribute are using the I did see that there are ContentProtection nodes that reference I was under the impression that this is technically all I would need (in terms of structure). This plays in Shaka as is currently.
I am able to play the vector you pointed to on my branch if I simply provide the keys you posted and change the schemeUri to be It looks like if I do that, then the player properly identifies the key system as ClearKey now and executes the rest of my code that takes that KeyID and uses the supplied key that is provided to the player upon load :). My main purpose for going down this path was because I produced encrypted dash content with |
I will try to explain some of the effects you are seeing by describing the general principles and then seeing how they apply here. The goal of key-related interactions in the player is to activate a CDM (whether Clear Key CDM or some other) that can use content keys to decrypt content. To activate a CDM, it must be provided initialization data, which is a CDM-specific data structure with freeform contents. A CDM may then request the player to mediate a license request if one is required in order to obtain the keys. Alternatively, a license request may not be necessary (if the keys are already provided in the init data or have been stored previously in another location accessible to the CDM). The player needs to know what initialization data to provide to a CDM in order to activate it. There are three ways that initialization data can be obtained:
Note that there is a possibility that initialization data for the same CDM is available using several of the above approaches. In that case, I would expect more specific data to override any more general data (synthesis is overridden by content is overridden by app configuration). As browsers can contain any number of CDM implementations, the player needs to make a choice - which one does it use? The simple answer is: it uses whatever CDM it has initialization data for! I believe most players just pick the first one that matches without much futher thought, although there can definitely be situations where some smart logic is needed. In the situation of interest for this PR, we can reasonably assume that no other CDM besides Clear Key can be used because the player has no initialization data for them. Now let's explore how the player should activate the Clear Key CDM.
With Clear Key, we also have the additional consideration that EME does not define any required format of initialization data that must be supported by browsers. To support different browsers, the player may need to be able to use different initialization data. In practice, most EME implementations support the "keyids" format. To ensure interoperability in this regard, DASH-IF IOP requires that the player synthesize the initialization data on the fly, as only this way can be browser-specific init data needs be considered. But if the player is meant to synthesize the initialization data on its own initiative, how is it to know that it is appropriate to do so? It needs some signal that Clear Key is meant to be used. For this, DASH-IF IOP specifies a ContentProtection element with the systemID Alternatively, the app configuration can always be used to instruct the player on correct behavior, overriding any other inputs. This mechanism should be usable to activate Clear Key even if not specified in the manifest. Now we know how the CDM should be activated but wait - where do the keys come from? Initialization data does not contain keys, just the key IDs. EME defines a license request protocol for acquiring content keys for use with Clear Key. When the Clear Key CDM signals a license request, it is the duty of the player to fulfill it. Now, the player could just forward this license request to a license server. For this it needs to know the license server URL (either from the ContentProtection element defined by DASH-IF IOP or the configuration defined by the app). Alternatively, since Clear Key requires no protection of keys, the player could just synthesize the license response directly. This, of course, assumes the player already knows the content keys - which could just be embedded into the app configuration (as, of course, they do not need to be protected). And there you go - once the license request gets a response, Clear Key CDM has the keys and will decrypt content. The above workflow is how Clear Key is supposed to work. Now, I will try to directly address some of the other points you raised above.
A worthy goal, indeed, though I would also consider Clear Key an "actual CDM" - there is really not that big of a difference there. I agree with your interpretation about key management - if the player knows the content keys, it can generate the license response without contacting any external service.
The mp4protection element just means "this content uses Common Encryption and the key ID is written here in the default_KID attribute". It does not participate in CDM selection logic, though it does provide the authoritative key ID that the player should use.
That GUID is the system ID of the W3C Generic PSSH and it should have no relevance on CDM selection. It is basically an equivalent of mp4protection - all it does it say "This content is encrypted using Common Encryption". I do know that there are players that misinterpret this GUID to mean "the player should select Clear Key" but that is a mistake from a DASH-IF IOP viewpoint. Possibly Shaka Player makes this mistake - I have not checked but from what you say it sounds like it does.
Well, the purpose of DASH-IF IOP is to provide a single interoperable way to do things. Unfortunately, many player/packager/etc manufacturers invent their own ways to do things, which means that we end up in "it works only with X but not Y" situations. The more we follow IOP, the more interoperable we can all be. Let's consider what the minimum scenario would be to get Clear Key CDM activated and working with some keys:
And that's it. If those two things are done, the player should activate the Clear Key CDM, synthesizing it initialization data and generating the license response. I think ensuring that dash.js can do the second step here would be a good scope/goal for this PR. Not sure how exactly it currently matches. But I would emphasize that the app configuration and data in the manifest are just alternative ways of signaling the CDM activation and should not both be required at the same time. The alternative approach, without player configuration by the app, would be:
As a third option, the content could signal Clear Key but not the license server URL. The app would then need to specify the license server URL in the player configuration (or just provide the keys inline). |
@@ -33,7 +33,7 @@ import KeyPair from '../vo/KeyPair'; | |||
import ClearKeyKeySet from '../vo/ClearKeyKeySet'; | |||
import CommonEncryption from '../CommonEncryption'; | |||
|
|||
const uuid = '1077efec-c0b2-4d02-ace3-3c1e52e2fb4b'; | |||
const uuid = 'e2719d58-a985-b3c9-781a-b030af78d30e'; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Correct me if I'm wrong, but wouldn't this be a breaking change for people relying on the existing, non-DASH-IF uuid?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes you're correct in this regard.
I believe this is to follow the IOP spec regarding ClearKey which states that the 1077...
id shall not be used to indicate ClearKey usage.
I'm not entirely sure what the correct approach is here in terms of what we want to see going forward. Allowing both is technically against the IOP spec but I agree that it may be the correct implementation for this PR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would perhaps suggest accepting both but emitting a warning in the log about it being IOP-non-conforming content when the W3C Common PSSH systemID is used.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think both should be acceptable as Sander suggests, possibly with a warning. Given the existing architecture, this probably requires a new KeySystem implementation so there are two for clearkey - one for DASH-IF, one for W3C. In time the W3C implementation could possibly be deprecated, but I think removal would be inappropriate at this stage.
Again, thank you thank you thank you for your patience in this endeavour. Your explanation packed a lot and was very helpful! I made a small change in my latest commit to change the uuid in the ClearKey module to match that of the IOP spec you linked to as well as always treating a matched key system as supported if it matched one that is in the list of all supported key systems. I have also changed my manifest to the following. After your comments and reading the spec, I understand that the
This plays as expected on my branch. In addition, the test vector linked earlier for ClearKey encrypted data also plays on my branch. |
I modified my changes to add a new KeySystem for support of the W3C Common UUID. Initially I was thinking of adding the barebones to that module and actually using the other ClearKey System for all of the function calls but I opted to do a copy because I thought it might be more readable in the end. I also added the correct protection data to the sources for the single key ClearKey vectors -- I have not tested multiple keys but I'm not entirely sure where to find the decryptions keys for those test streams. I'd be happy to test against those and add them in to complete the ClearKey samples :). Other than that I tested this branch with a manifest using only the one |
I confirm that in my quick testing with your branch the videos I tried seemed to play without issues using clear key.
The Axinom test vector keys are listed at https://github.com/axinom/dash-test-vectors, although in a slightly different format so there is some annoying conversion needed to make them the base64url format used by this particular configuration block. Do I understand it right that this implementation expects the keys to be provided in the configuration block? That is, Clear Key license requests are not implemented - only pre-provided keys? I did not see license requests discussed, so I assume not but thought I would ask just in case (I am not very familiar with the codebase, so I imagine I could easily miss it). |
Sounds good I will add in that test data as well then :). And yes this PR is only focused on providing the keys in the initialization data to the player and does not add support for ClearKey license requests from an external server. |
I've added the keys to the protData for the multi-clearkey sources. The media refuses to play in Chrome for the multi key sources but with debugging it looks like the video codes is not currently supported on those. The multi key audio source does play in Chrome and all but the 1080p multi-key source plays in Firefox. I'm guessing this would point to a browser support issue and not a clearkey decryption issue? |
There are known issues with multi-key playback in dash.js (e.g. #2280), so failure to play that content might be normal. I think it is unlikely that this PR has broken any of that, at least. |
@sandersaares @epiclabsDASH would this mean that this PR is in a good state to be scheduled for merge and release? |
I would perhaps also add "See DASH-IF IOP v4.1 section 7.6.2.4" to the emitted warning if non-IOP-conforming content is used, just to make it easier for users to understand why they are being told something is wrong. Otherwise, LGTM from an interoperability perspective. |
@@ -417,6 +419,12 @@ function ProtectionController(config) { | |||
for (let i = 0; i < pendingNeedKeyData.length; i++) { | |||
for (ksIdx = 0; ksIdx < pendingNeedKeyData[i].length; ksIdx++) { | |||
if (keySystem === pendingNeedKeyData[i][ksIdx].ks) { | |||
if (pendingNeedKeyData[i][ksIdx].initData === null && protData.hasOwnProperty('clearkeys')) { | |||
var initData = { kids: Object.keys(protData.clearkeys) }; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Small fix. To avoid possible future issues related with variables scope, could you please declare initData using const instead of var?
@@ -0,0 +1,114 @@ | |||
/** |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could you please add unit tests for this class? That helps us a lot to detect possible regression in future releases.
|
||
// Generate initial key request | ||
session.generateRequest('cenc', initData).then(function () { | ||
session.generateRequest(ks.systemString === 'org.w3.clearkey' ? 'keyids' : 'cenc', initData).then(function () { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think makes sense declaring a new Constants file, focused on EME stuff, where we declare system strings of the different keySystems so we have just one place to define them all. I am afraid of having these strings (ex: "org.w3.clearkey") in different places.
@CSilivestru, in the meantime and given this is your first contribution (fix me if I am wrong but I didn't find you in the list of contributors), could you please sign the Contribution License Agreement and send it to me? You can find it here: http://dashif.org/wp-content/uploads/2015/04/DASH-IF-Feedback-Agreement-3-7-2014.pdf Many thanks for your contribution |
Right added a change related with management of key strings constants and a bit of reformat. PR looks good to me. There is some pending work related with ClearKey working with license requests but that can be managed in a different PR. |
@CSilivestru, PR ready to be merged. I just need you to sign the CLA and send it to me ([email protected]) to proceed. You can find the CLA here: http://dashif.org/wp-content/uploads/2015/04/DASH-IF-Feedback-Agreement-3-7-2014.pdf Thanks |
2a3a1c1
to
ddbf84c
Compare
Looks like everything is ready to go (I rebased to avoid a merge commit in this PR) :) |
Ooops, seems there is something wrong after the rebase. Some changes has dissapeared (the ones related with KEY constants management) and there are new files added that shouldn't be part of development branch (dist files). By the way, ClearKey is not working now for sample streams. |
Whoa yeah something very wrong happened. Ok I will get this cleared up asap. Sorry about that. My guess would be the streams are failing due to this as well but I will re-test to make sure. |
No problem, take your time. |
Switch ClearKey module to use correct uuid according to the latest IOP documentation ('e2719d58-a985-b3c9-781a-b030af78d30e'). In addition, always add the key system to the list of supported if we match based on the current list of supported key systems. The requirement of always having a pssh box is not a requirement for every CDM and if the initData is non-conforming to a particular key system, creating the session will fail later. We don't need the clearkey specific condition when determining if a system is supported -- it should be enough that it's in the list.
Log a warning if their are using that uuid since it's not part of the IOP spec. Eventually the plan would be to deprecate this and remove the newly created KeySystem.
ddbf84c
to
a13e329
Compare
My bad -- force of habit that I pulled down the I have re-stested the vectors and the ones that worked before are still working now. The multi period test streams don't work but that is due to a video codec problem. |
Hello!
I was experiencing the similar issues as in #2249
Why am I Submitting this?
After quite a bit of digging, I discovered that the ClearKey
ContentProtection
information in my manifest was being ignored since they did not havepssh
entry. This is not the case with the Shaka player so I am submitting this as a fix. My understanding is that ClearKeys only need the key id specified and the actual key used should simply be provided to the player for use at the time of playback.What did I do?
In the case of a ClearKey key system, I would still push that as a supported KeySystem. Since, at that time, I do not yet have the actual decryption key, I need to set
initData
tonull
so that I can properly set it later with my key. I'm not sure if this is the best way of doing it but it seems to follow the correct event handlers since ClearKey doesn't need to communicate with a CDMI'm extremely new to everything involved in dash and DRM systems in general so please please let me know if I'm way off base here with my proposed solution.
My Testing
I tested this solution by encrypting a video file with the following very simple
crypt.xml
file:The command I used was:
This would produce the following manifest (after I switched the generic
xmlns
to the properkeyids
one:After all this, I'm currently using the related
video-dash-contrib
plugin and initializing it like so:This all works on my current branch when loading all libraries locally. On the
development
branch, the video buffers forever and no DRM events are logged.