okova
v0.12.8
Published
Advanced DRM inspection toolkit
Maintainers
Readme
okova
Okova is a set of tools (JavaScript library, command-line utility and browser extension) for diagnosing, researching, and pentesting DRM systems used in playable media content.
Okova is still in the early stages of development, so until version 1.0 is released, performance may be unstable and major changes may be made
Features
- Logging details from EME events in DevTools console
- Network-independent interception via browser extension, so it doesn't matter if license request has one-time tokens or a custom request/response body format
- Remote instance to manage sessions via REST API
- Custom CDM client support: create *.wvd or *.prd from raw client files and import them into browser extension
- Runtime agnostic core: works in Node.js, Bun, Deno, browsers and more
- Encrypted Media Extensions API compatibility via
requestMediaKeySystemAccess()method
Browser Extension
Installing Chrome extension
Developer Mode needs to be enabled in chrome://extensions/ page
- Download archive from latest release
- Go to
chrome://extensions/page - Ensure Developer Mode enabled and then drag and drop downloaded zip file to this page
Installing Firefox extension
- Download archive from latest release
- Go to
about:debugging#/runtime/this-firefoxpage - Click
Load Temporary Add-onbutton and choose downloaded zip file
Temporary add-on is not persistent and will be removed after browser restart
Command-line tool
Installation
Command-line tool installation requires pre-installed JavaScript runtime (e.g. Node.js).
npm install -g okovaUsage
See help for all possible arguments and options:
okova --help
Convert DRM client files ./drm-files/device_client_id_blob and ./drm-files/device_private_key to single WVD file:
okova client pack ./drm-files ./unknown_android-sdk-built-for-x86.wvdClient packed: /Users/.../unknown_android-sdk-built-for-x86.wvdShow DRM client info:
okova client info ./unknown_android-sdk-built-for-x86.wvdapplication_name: org.chromium.webview_shell
company_name: unknown
model_name: Android SDK built for x86
architecture_name: x86
device_name: generic_x86
product_name: sdk_phone_x86
build_info: Android/sdk_phone_x86/generic_x86:10/RSR1.410600.002.B3/1792159:userdebug/dev-keys
widevine_cdm_version: 16.0.0
oem_crypto_security_patch_level: 0
oem_crypto_build_information: OEMCrypto Level3 Code 8162 May 9 2018 14:01:12JavaScript library
Library installation requires pre-installed JavaScript runtime (e.g. Node.js).
Installation
npm install okovaUsage
See examples for more.
Obtain a Widevine license for Bitmovin's video:
import { readFile } from 'node:fs/promises';
import { fromBase64, Widevine, WidevineDeviceCredentials } from 'okova';
async function main() {
// Prepare init data (PSSH)
const initData = fromBase64(
'AAAAW3Bzc2gAAAAA7e+LqXnWSs6jyCfc1R0h7QAAADsIARIQ62dqu8s0Xpa7z2FmMPGj2hoNd2lkZXZpbmVfdGVzdCIQZmtqM2xqYVNkZmFsa3IzaioCSEQyAA==',
).toBuffer();
// Load device credentials
const deviceCredentials = await WidevineDeviceCredentials.from({
wvd: await readFile('client.wvd'),
});
const widevine = new Widevine({ deviceCredentials });
const session = widevine.createSession();
// Handle generated license challenge (or other session messages like individualization request)
session.onmessage = async (event) => {
const { message } = event.detail;
// Send license request
const licenseUrl = 'https://cwip-shaka-proxy.appspot.com/no_auth';
const response = await fetch(licenseUrl, { body: message, method: 'POST' })
.then((r) => r.arrayBuffer())
.then((buffer) => new Uint8Array(buffer));
// Update session with license response
await session.update(response);
};
// Generate license challenge
await session.generateRequest(initData);
// Wait for keys
const keys = await session.waitForKeys();
for (const [keyId, key] of keys) {
console.log(`${keyId}:${key}`);
}
await session.close(); // Close session to delete of any license(s) and key(s) that have not been explicitly stored.
await session.remove(); // Destroy the license(s) and/or key(s) associated with the session whether they are in memory, persistent store or both.
}Disclaimer
- This project does not condone piracy or any action against the terms of the DRM systems.
- All efforts in this project have been the result of Reverse-Engineering, Publicly available research, and Trial & Error.
- Do not use this program to decrypt or access any content for which you do not have the legal rights or explicit permission.
- Unauthorized decryption or distribution of copyrighted materials is a violation of applicable laws and intellectual property rights.
- This tool must not be used for any illegal activities, including but not limited to piracy, circumventing digital rights management (DRM), or unauthorized access to protected content.
- The developers, contributors, and maintainers of this program are not responsible for any misuse or illegal activities performed using this software.
- By using this program, you agree to comply with all applicable laws and regulations governing digital rights and copyright protections.
