Crunchy-Downloader/CRD/Utils/DRM/Widevine.cs
Elwador 5d94025fcc Add - Added encoding option to muxing settings
Add - Added Custom encoding presets
Add - Added Skip Muxing to muxing settings
Add - Added Dubs to file name settings
Add - IP check in settings to check if VPN is being used
Add - Dubs to "Add Downloads" Tab
Add - Series folder link to history series if it finds the folder
Add - Added command line arguments
Add - Added proxy settings to the Settings tab (changes require a restart to take effect)
Add - Added option to set "Sign" subs forced flag
Add - Added option to set "CC" subs "hearing-impaired" flag
Add - Added encoding presets editing
Add - Added CC subtitles font option to the settings
Add - Added available dubs to history episodes

Chg - Defaults to system accent color when no color is selected in the settings
Chg - Audio only mux to only copy and not encode
Chg - Update dialog
Chg - Light mode color adjustments
Chg - Http Connection change to detect proxy (Clash)
Chg - Settings filename description
Chg - Changed FPS on encoding presets to 24fps
Chg - Adjusted encoding to allow h264_nvenc & hevc_nvenc
Chg - Moved sync timing folders from the Windows temp folder to the application root's temp folder
Chg - The temp folder will now be deleted automatically when empty

Fix - Locale not correctly applied to Urls in the "Add Downloads" Tab
Fix - Locale not correctly applied to Search in the "Add Downloads" Tab
Fix - Scrolling issue in settings
Fix - Fix crash when removing streaming tokens (TOO_MANY_ACTIVE_STREAMS)
Fix - Search didn't reset correctly
Fix - Clash proxy didn't work
Fix - Chapters were always taken from the original version (mainly JP)
Fix - Connection issue
Fix - Fixed an issue where proxy settings were only available when history was enabled
Fix - Fixed scrolling issues with certain series in the "Add Downloads" tab
Fix - Fixed an issue where History Series appeared incomplete after being added then deleted and re-added
Fix - Fixed a crash related to sync timing
2024-09-30 20:08:37 +02:00

118 lines
No EOL
3.9 KiB
C#

using System;
using System.Collections.Generic;
using System.IO;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using Newtonsoft.Json;
namespace CRD.Utils.DRM;
public class Widevine{
private byte[] privateKey = new byte[0];
private byte[] identifierBlob = new byte[0];
public bool canDecrypt = false;
#region Singelton
private static Widevine? instance;
private static readonly object padlock = new object();
public static Widevine Instance{
get{
if (instance == null){
lock (padlock){
if (instance == null){
instance = new Widevine();
}
}
}
return instance;
}
}
#endregion
public Widevine(){
try{
if (Directory.Exists(CfgManager.PathWIDEVINE_DIR)){
var files = Directory.GetFiles(CfgManager.PathWIDEVINE_DIR);
foreach (var file in files){
var fileInfo = new FileInfo(file);
if (fileInfo.Length < 1024 * 8 && !fileInfo.Attributes.HasFlag(FileAttributes.Directory)){
string fileContents = File.ReadAllText(file, Encoding.UTF8);
if (fileContents.Contains("-BEGIN RSA PRIVATE KEY-") || fileContents.Contains("-BEGIN PRIVATE KEY-")){
privateKey = File.ReadAllBytes(file);
}
if (fileContents.Contains("widevine_cdm_version")){
identifierBlob = File.ReadAllBytes(file);
}
}
}
}
if (privateKey.Length != 0 && identifierBlob.Length != 0){
canDecrypt = true;
} else if (privateKey.Length == 0){
Console.Error.WriteLine("Private key missing");
canDecrypt = false;
} else if (identifierBlob.Length == 0){
Console.Error.WriteLine("Identifier blob missing");
canDecrypt = false;
}
} catch (Exception e){
Console.Error.WriteLine("Widevine: " + e);
canDecrypt = false;
}
}
public async Task<List<ContentKey>> getKeys(string? pssh, string licenseServer, Dictionary<string, string> authData){
if (pssh == null || !canDecrypt){
Console.Error.WriteLine("Missing pssh or cdm files");
return new List<ContentKey>();
}
try{
byte[] psshBuffer = Convert.FromBase64String(pssh);
Session ses = new Session(new ContentDecryptionModule{ identifierBlob = identifierBlob, privateKey = privateKey }, psshBuffer);
var playbackRequest2 = new HttpRequestMessage(HttpMethod.Post, licenseServer);
foreach (var keyValuePair in authData){
playbackRequest2.Headers.Add(keyValuePair.Key, keyValuePair.Value);
}
var licenceReq = ses.GetLicenseRequest();
playbackRequest2.Content = new ByteArrayContent(licenceReq);
var response = await HttpClientReq.Instance.SendHttpRequest(playbackRequest2);
if (!response.IsOk){
Console.Error.WriteLine("Failed to get Keys!");
return new List<ContentKey>();
}
LicenceReqResp resp = Helpers.Deserialize<LicenceReqResp>(response.ResponseContent, null) ?? new LicenceReqResp();
ses.ProvideLicense(Convert.FromBase64String(resp.license));
return ses.ContentKeys;
} catch (Exception e){
Console.Error.WriteLine(e);
return new List<ContentKey>();
}
}
}
public class LicenceReqResp{
public string status{ get; set; }
public string license{ get; set; }
public string platform{ get; set; }
public string message_type{ get; set; }
}