Releases: devine-dl/pywidevine
Releases · devine-dl/pywidevine
v1.4.1
- Supported Serve API:
v1.4.0
tov1.4.2
Changed
- Rework
PSSH.overwrite_key_ids()
as an instance method now namedPSSH.set_key_ids()
. - Rework
PSSH.get_key_ids()
as a property method namedPSSH.key_ids
. This allows swift access to all the Key IDs of
the current PSSH object data. - Rework
PSSH.from_playready_pssh()
as an instance method now namedPSSH.playready_to_widevine()
that now converts
the current instances values directly. This allows you to more easily instance as any PSSH, then convert after wards
and only if wanted and when needed.
v1.4.0
- Supported Serve API:
v1.4.0
tov1.4.2
Added
- New PSSH boxes can now be manually crafted with
PSSH.new()
.- The box can be crafted from arbitrary init_data and/or key_ids.
- If only key_ids is supplied a new Widevine CENC Header will be created and the key IDs will be put into it.
- This allows you to make compliant v0 or v1 boxes with as little data as just a Key ID.
- PSSH boxes can now be exported as MP4 Box objects using pymp4 with
PSSH.dump()
. - PSSH boxes can now also be exported as Base64 strings with
PSSH.dumps()
. - License Keys can now be obtained from a Cdm session with a parsed license using
Cdm.get_keys()
.- This is the alternative to manually accessing the keys from the
Cdm._sessions
object. - It is also available on the Serve API through the new
/get_keys
endpoint.
- This is the alternative to manually accessing the keys from the
Changed
PSSH.get_as_box()
has been merged into the PSSH constructor, simplifying usage of the PSSH class.PSSH.from_playready_pssh()
is now a class method and returns as a PSSH object.- Only PSSH objects are now accepted by
Cdm.get_license_challenge()
.- You can no longer provide it anything else, that includes base64 or bytes form.
- You should first parse or make a new PSSH with the PSSH class, and then pass that object.
- This is to simplify typing and repetition across the codebase.
- Serve's
/challenge
endpoint has been changed to/get_license_challenge
, and/keys
to/parse_license
.- This is to be consistent with the method names of the underlying Cdm class.
- Serve now passes the license type value as-is (as a string) instead of parsing it to an integer.
- Serve now passes the key type value as-is (as a string) instead of parsing it to an integer.
- Serve no longer returns license keys in the response of the
/parse_license
endpoint.- Once parsed, the
/get_keys
endpoint should be used to retrieve keys.
- Once parsed, the
- Privatized the
Cdm._sessions
class instance variable even more toCdm.__sessions
.- If you still need something from it, while not advised, you can call it via
cdm._Cdm__sessions
.
- If you still need something from it, while not advised, you can call it via
Removed
PSSH.from_key_ids()
has been removed entirely, you should now usePSSH.new(key_ids=...)
instead.- Unnecessary parsing of the license message received by RemoteCdm is now skipped. Parsing should be done by the Serve
API as it will be able to actually decrypt and verify the message. - All uses of a local
Session
object has been removed fromRemoteCdm
. The session is now fully controlled by the
remote API and de-synchronization by external alteration or unexpected exceptions is no longer a possibility.
Fixed
- Correct the WidevinePsshData proto field name from
key_id
tokey_ids
in the PSSH class. - Handle
DecodeError
andSignatureMismatch
exceptions in the Serve/set_service_certificate
endpoint. - Handle
InvalidInitData
andInvalidLicenseType
exceptions in the Serve/get_license_challenge
endpoint. - Handle various exceptions in the Serve
/parse_license
endpoint. - Handle various client-side runtime errors in
RemoteCdm
with improved error handling.
v1.3.1
- Supported Serve API:
v1.3.0
tov1.3.1
Added
- String value support to the
device_type
parameter inCdm
s constructor.
Changed
- Serve no longer requires
force_privacy_mode
to be defined in the config file. It now assumes a default of false. - Serve now uses
pywidevine serve ...
instead of the full project url in the Server header. RemoteCdm
s Server version check is now case-insensitive.
Fixed
RemoteCdm
s Server version check now ignores other Server/Proxy names prepended or appended to the Server header.- For example, if reverse-proxied through Caddy it may have prepended "Caddy" to the Server header.
v1.3.0
- Supported Serve API:
v1.3.0
tov1.3.1
Added
- New Client for using the Serve API;
RemoteCdm
class. It has an identical interface as the originalCdm
class.- However, the constructor is different. Instead of passing a Widevine device object, you need to pass information
about the API like its host (including port if not on a reverse-proxy), and info about the device like its name and
security level. - Other than that, once the RemoteCdm object is created, you use it exactly the same. Magic!
- Any time there's a change or fix to
Cdm
in this update or any in the future, will also be done to RemoteCdm.
- However, the constructor is different. Instead of passing a Widevine device object, you need to pass information
- New Serve endpoint
/set_service_certificate
as an improved way of setting (or unsetting) the service certificate.
Changed
Cdm
s constructor now uses more direct values, so you don't have to use the Device class or.wvd
files.- To continue using
.wvd
files you must now useCdm.from_device()
instead.
- To continue using
- You can now unset the Service certificate by providing
None
to `Cdm.set_service_certificate().
Removed
- Serve's
/challenge
endpoint no longer accepts aservice_certificate
item in the JSON payload.- Instead, use the new
/set_service_certificate
endpoint before calling/challenge
. - You do not need to set it every time. Once per session is enough unless you now want to use a different certificate.
- Instead, use the new
v1.2.1
Added
- Support
SignedDrmCertificate
andSignedMessages
messages inCdm.encrypt_client_id()
. This is mainly as a
convenience for any scripts wanting to encrypt their Client ID with a service certificate manually. - All License Keys from Serve's
/keys
endpoint can now be received by providingALL
as the key type.- This adds support for systems needing more than two types of keys from the license, e.g., Netflix MSL.
- For faster response times it is best to still ask for only
CONTENT
keys if that's all you need.
- Serve now has a
/close
endpoint to close a session. All clients should close the session once they are finished
with it or the user will eventually hit a limit of 50 sessions per user and the server will hog memory til it
restarts. - Serve now verifies that all Devices in config actually exist before starting the server.
- Serve now responds with a
Server
header denoting that pywidevine serve is being used, and it's version.- This allows Clients to selectively support APIs based on version; verify the API as being supported.
Changed
- Lessened version pin on
lxml
from^4.9.1
to>=4.8.0
to support projects using pycaption. - Service Certificate is now saved in the session as a
SignedMessage
with aSignedDrmCertificate
instead of the raw
DrmCertificate
. TheSignedMessage
is unsigned as theSignedDrmCertificate
within it, is signed. This is so
anything inheriting or using the Cdm (e.g.,serve
) can verify the certificate down the chain and keep it signed. - Serve now constructs one Cdm object for each user+device combination so one user cannot fill or overuse the CDM
session limit. - All of Serve's endpoints now have a
/{device}
prefix. E.g., instead of/challenge/STREAMING
, it's now
/device_name/challenge/STREAMING
. This is to support the previous change.
Fixed
- Handle server crash when the session limit is reached in Serve's
/open
endpoint by returning a 400 error. - Serve now correctly updates (or rather now makes a new Cdm object) if a user switches from one Device to another.
- Previously it would reuse an existing Cdm object, but would forget to switch device if they changed.
- Note: It does still leave the previous Cdm with the older Device in memory.
- Handle IOError when parsing bytes as MP4 Box to allow arbitrary data to be made as new boxes in
PSSH.get_as_box()
.
v1.2.0
Added
- New CLI command
serve
that hosts a CDM API that can be externally accessed with authentication. This can be used to
access and/or share your CDM without exposing your Widevine device private key, or even it's identity by enforcing
Privacy Mode.- Requires installing with the
serve
extras, i.e.,pip install pywidevine[serve]
. - The default host of
127.0.0.1
blocks access outside your network, even if port-forwarded. Use
-h 0.0.0.0
to allow remote access. - Setup requires the use of a config file for configuring the CDM and authentication. An example config file named
serve.example.yml
in the project root folder has verbose documentation on available options.
- Requires installing with the
- Batch migration of WVD files by passing a folder as the path to the CLI command
migrate
. - Strict mode to
PSSH.get_as_box()
to raise an Exception if passed data is not already a box, as it has been improved
to create a new box if not detected as a box already.
Changed
- Elevated the Development Status Classifier from 4 (Beta) to 5 (Production/Stable).
- License messages passed to
Cdm.parse_license()
are now rejected if they are not ofLICENSE
type. - Service Certificates passed to
Cdm.set_service_certificate()
are now verified. This patches a trivial "exploit"
that allows an attacker to recover the plaintext Client ID from a license under Privacy Mode. See
https://gist.github.com/rlaphoenix/74acabdd7269a21845e18b621c5860ef. - Data passed to
PSSH.get_as_box()
now supports arbitrary and box data automatically as it tries to detect if it is a
valid box, otherwise makes a new box. - Renamed the
Cdm
constructor's parameterpssh
toinit_data
, as that's what the Cdm actually wants and uses,
whereas aPSSH
is anmp4
atom (aka box) containinginit_data
(a Widevine CENC Header). The full PSSH is never
kept nor ever used. It still accepts PSSH box data. - Service Certificate's Provider ID is now returned by
Cdm.set_service_certificate()
instead of the passed
certificate, of which they would already have. - The Cdm class now works more closely to the official CDM model. Instead of using one Cdm object per-request having to
provide device information each time,- You now initialize the Cdm with the Widevine device you wish to use and then open sessions with
Cdm.open()
. - You will receive a session ID that are then passed to other methods of the same Cdm object.
- The PSSH/init_data that used to be passed to the constructor is now passed to
Cdm.get_license_challenge()
. - This allows initializing one Cdm object with up to 50 sessions open at the same time.
Session limits seem to fluctuate between libraries and devices. 50 seems like a permissive value. - Once you are finished with DRM operations, discard all session (and key) data by calling
Cdm.close(session_id)
.
- You now initialize the Cdm with the Widevine device you wish to use and then open sessions with
- License Keys are no longer returned by
Cdm.parse_license()
and now must be obtained directly fromcdm._sessions
.- For example,
for key in cdm._sessions[session_id].keys: print(f"[{key.type}] {key.kid.hex}:{key.key.hex()}")
. - This is to detach the action of parsing a license as just for getting keys, as it isn't. It can be and should be
used for a lot more data like security requirements like HDCP, expiration, and more. - It is also to detour users from directly using the keys over the
Cdm.decrypt()
method.
- For example,
- Various std-lib exceptions have been replaced with custom exceptions under
pywidevine.exceptions
. - License responses can now only be parsed once by
Cdm.parse_license()
. Any further attempts will raise an
InvalidContext
exception.- This is as license context data is cleared once used to reduce data lingering in memory, otherwise the more license
requests you make without closing the session, the more and more memory is taken up. - Open multiple sessions in the same Cdm object if you need to request and parse multiple licenses on the same device.
- This is as license context data is cleared once used to reduce data lingering in memory, otherwise the more license
Removed
- Direct
DrmCertificate
s are no longer supported byCdm.set_service_certificate()
as they have no signature.
See the 3rd Change above. Provide either aSignedDrmCertificate
or aSignedMessage
containing a
SignedDrmCertificate
. ASignedMessage
containing aDrmCertificate
will also be rejected. PSSH.from_init_data()
, usePSSH.get_as_box()
.raw
parameter ofCdm
constructor, as well as CLI commands as it is now handled upstream by thePSSH
creation.
Fixed
- Detection of Widevine CENC Header data encoded as bytes in
PSSH.get_as_box()
. - Custom ValueError on missing contexts instead of the generic KeyError in
Cdm.parse_license()
. - Typing of
type_
parameter inCdm.get_license_challenge()
. - Value of
type_
parameter if is a string inCdm.get_license_challenge()
.
v1.1.1
Fixed
- The
-v/--vmp
parameter of thetest
CLI command is now optional.
v1.1.0
Added
- WVD (Widevine Device file) Version 2 bringing reduced file sizes by up to 30%~.
- New CLI command
create-device
to create.wvd
files (Widevine Device files) from RSA PEM/DER Private Keys and
Client ID blobs. You can also provide VMP (FileHashes) data which will be merged into the Client ID blob. - New CLI command
migrate
that usesDevice.migrate()
anddump()
to migrate older v1 Widevine Device files to v2. - New
Device
methodmigrate()
to load an older Widevine Device file format. It is recommended to then use the
dumps()
method to save it as a new v2 Widevine Device file, which can then be loaded normally. - Support
SignedDrmCertificate
andDrmCertificate
messages inCdm.set_service_certificate()
. Services can provide
the certificate as aSignedMessage
,SignedDrmCertificate
, or aDrmCertificate
. OnlySignedMessage
and
SignedDrmCertificate
are signed. - Privacy Mode can now be used in the
test
CLI command with the-p/--privacy
flag.
Changed
- Moved all
.wvd
Widevine Device file structures fromDevice
to a_Structures
class indevice.py
. The
_Structures
class can be imported and used directly, or viaDevice.structures
. - Moved the majority of Widevine Device file migration code from the CLI command
migrate
toDevice.migrate()
. The
CLI commandmigrate
now internally usesDevice.migrate()
. - Set Service Certificates are now stored as
DrmCertificate
s instead of aSignedMessage
as the signature and other
data in the message is unused and unneeded.
Removed
- Unused Widevine Device file flag
send_key_control_nonce
from v1 and v2 Structures as it was only used before initial
release, and isn't a necessary nor useful flag.
Fixed
- Correct the type argument name from
type
totype_
inDevice.dump()
.
Security
- Even though support for more kinds of Service Certificate Signatures were added, they are still unverified as the
signing public keyiswas Unknown.
v1.0.1
Changed
- Moved the License Type parameter from the
Cdm
constructor to it'sget_license_challenge()
method. - Every License request now uses a unique random value instead of the CDM Session ID.
- Only the Context Data of License requests are now stored in the Session instead of the full message.
- Session ID formula now uses a random 16-byte value for both Chrome and Android provisions.
Removed
- Unused and unnecessary
Cdm.raw
class instance variable.
Fixed
- Re-raise DecodeErrors instead of a new ValueError on DecodeErrors in
Cdm.set_service_certificate()
. - Creating a new License request no longer overwrites the context data of the previous challenge.
v1.0.0
Initial Release.
Security
- Service Certificate Signatures are unverified as the signing public key
iswas Unknown.