Crunchy-Downloader/CRD/Utils/DRM/Widevine.cs
Elwador 5b33d2336c Add - Added **retry** for license key requests
Add - Added **Sonarr season/episode numbers** to the history view
Add - Added option to **match episodes to Sonarr episodes** to correct mismatches
Add - Added **button to rematch all Sonarr episodes**
Add - Added an **optional secondary endpoint**
Chg - Changed **changelog heading sizes**
Chg - Changed Sonarr matching to only process **new or unmatched episodes**
Chg - Changed logic to **request audio license keys only when needed**
Fix - Fixed **Sonarr series manual matching dialog**
2025-06-17 18:56:24 +02:00

160 lines
No EOL
5.6 KiB
C#

using System;
using System.Collections.Generic;
using System.IO;
using System.IO.IsolatedStorage;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using CRD.Utils.Files;
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)){
foreach (var file in Directory.EnumerateFiles(CfgManager.PathWIDEVINE_DIR)){
var fileInfo = new FileInfo(file);
if (fileInfo.Length >= 1024 * 8 || fileInfo.Attributes.HasFlag(FileAttributes.Directory))
continue;
string fileContents = File.ReadAllText(file, Encoding.UTF8);
if (IsPrivateKey(fileContents)){
privateKey = File.ReadAllBytes(file);
} else if (IsWidevineIdentifierBlob(fileContents)){
identifierBlob = File.ReadAllBytes(file);
}
}
}
if (privateKey?.Length > 0 && identifierBlob?.Length > 0){
canDecrypt = true;
} else{
canDecrypt = false;
if (privateKey == null || privateKey.Length == 0){
Console.Error.WriteLine("Private key missing");
}
if (identifierBlob == null || identifierBlob.Length == 0){
Console.Error.WriteLine("Identifier blob missing");
}
}
} catch (IOException ioEx){
Console.Error.WriteLine("I/O error accessing Widevine files: " + ioEx);
canDecrypt = false;
} catch (UnauthorizedAccessException uaEx){
Console.Error.WriteLine("Permission error accessing Widevine files: " + uaEx);
canDecrypt = false;
} catch (Exception ex){
Console.Error.WriteLine("Unexpected Widevine error: " + ex);
canDecrypt = false;
}
Console.WriteLine($"CDM available: {canDecrypt}");
}
private bool IsPrivateKey(string content){
return content.Contains("-BEGIN RSA PRIVATE KEY-", StringComparison.Ordinal) ||
content.Contains("-BEGIN PRIVATE KEY-", StringComparison.Ordinal);
}
private bool IsWidevineIdentifierBlob(string content){
return content.Contains("widevine_cdm_version", StringComparison.Ordinal);
}
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();
var content = new ByteArrayContent(licenceReq);
content.Headers.ContentType = new System.Net.Http.Headers.MediaTypeHeaderValue("application/octet-stream");
playbackRequest2.Content = content;
// var response = await HttpClientReq.Instance.SendHttpRequest(playbackRequest2);
var response = (IsOk: false, ResponseContent: "", error: "");
for (var attempt = 0; attempt < 3 + 1; attempt++){
using (var request = Helpers.CloneHttpRequestMessage(playbackRequest2)){
response = await HttpClientReq.Instance.SendHttpRequest(request);
if (response.IsOk){
break;
}
if (response.error.Contains("System.Net.Sockets.SocketException (10054)")){
Console.Error.WriteLine($"Key Request Attempt {attempt + 1} failed.");
if (attempt == 3)
break;
await Task.Delay(1000);
} else{
break;
}
}
}
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; }
}