2024-05-04 15:35:32 +00:00
|
|
|
|
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-")){
|
|
|
|
|
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){
|
2024-06-19 00:16:02 +00:00
|
|
|
|
Console.Error.WriteLine("Private key missing");
|
2024-05-04 15:35:32 +00:00
|
|
|
|
canDecrypt = false;
|
|
|
|
|
} else if (identifierBlob.Length == 0){
|
2024-06-19 00:16:02 +00:00
|
|
|
|
Console.Error.WriteLine("Identifier blob missing");
|
2024-05-04 15:35:32 +00:00
|
|
|
|
canDecrypt = false;
|
|
|
|
|
}
|
|
|
|
|
} catch (Exception e){
|
2024-06-19 00:16:02 +00:00
|
|
|
|
Console.Error.WriteLine("Widevine: " + e);
|
2024-05-04 15:35:32 +00:00
|
|
|
|
canDecrypt = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-06-19 17:09:18 +00:00
|
|
|
|
|
2024-05-04 15:35:32 +00:00
|
|
|
|
public async Task<List<ContentKey>> getKeys(string? pssh, string licenseServer, Dictionary<string, string> authData){
|
|
|
|
|
if (pssh == null || !canDecrypt) return new List<ContentKey>();
|
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
try{
|
|
|
|
|
byte[] psshBuffer = Convert.FromBase64String(pssh);
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
Session ses = new Session(new ContentDecryptionModule{ identifierBlob = identifierBlob, privateKey = privateKey }, psshBuffer);
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
var playbackRequest2 = new HttpRequestMessage(HttpMethod.Post, licenseServer);
|
|
|
|
|
foreach (var keyValuePair in authData){
|
|
|
|
|
playbackRequest2.Headers.Add(keyValuePair.Key, keyValuePair.Value);
|
|
|
|
|
}
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
var licenceReq = ses.GetLicenseRequest();
|
|
|
|
|
playbackRequest2.Content = new ByteArrayContent(licenceReq);
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
var response = await HttpClientReq.Instance.SendHttpRequest(playbackRequest2);
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
if (!response.IsOk){
|
|
|
|
|
Console.Error.WriteLine("Failed to get Keys!");
|
|
|
|
|
return new List<ContentKey>();
|
|
|
|
|
}
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
LicenceReqResp resp = Helpers.Deserialize<LicenceReqResp>(response.ResponseContent, null) ?? new LicenceReqResp();
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
ses.ProvideLicense(Convert.FromBase64String(resp.license));
|
2024-05-04 15:35:32 +00:00
|
|
|
|
|
2024-06-19 17:09:18 +00:00
|
|
|
|
return ses.ContentKeys;
|
|
|
|
|
} catch (Exception e){
|
|
|
|
|
Console.Error.WriteLine(e);
|
|
|
|
|
return new List<ContentKey>();
|
|
|
|
|
}
|
2024-05-04 15:35:32 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class LicenceReqResp{
|
|
|
|
|
public string status{ get; set; }
|
|
|
|
|
public string license{ get; set; }
|
|
|
|
|
public string platform{ get; set; }
|
|
|
|
|
public string message_type{ get; set; }
|
|
|
|
|
}
|