mirror of
https://github.com/TheShadowEevee/libWiiSharp.git
synced 2025-01-11 15:38:51 -06:00
Add code to self-define a url
This commit is contained in:
parent
597516d426
commit
ea711aacfa
1 changed files with 375 additions and 2 deletions
377
NusClient.cs
377
NusClient.cs
|
@ -96,14 +96,34 @@ namespace libWiiSharp
|
||||||
PrivDownloadTitle(titleId, titleVersion, outputDir, storeTypes);
|
PrivDownloadTitle(titleId, titleVersion, outputDir, storeTypes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void DownloadTitle(
|
||||||
|
string titleId,
|
||||||
|
string titleVersion,
|
||||||
|
string outputDir,
|
||||||
|
string nusUrl,
|
||||||
|
params StoreType[] storeTypes)
|
||||||
|
{
|
||||||
|
if (titleId.Length != 16)
|
||||||
|
{
|
||||||
|
throw new Exception("Title ID must be 16 characters long!");
|
||||||
|
}
|
||||||
|
|
||||||
|
PrivDownloadTitle(titleId, titleVersion, outputDir, nusUrl, storeTypes);
|
||||||
|
}
|
||||||
|
|
||||||
public TMD DownloadTMD(string titleId, string titleVersion)
|
public TMD DownloadTMD(string titleId, string titleVersion)
|
||||||
{
|
{
|
||||||
return titleId.Length == 16 ? PrivDownloadTmd(titleId, titleVersion) : throw new Exception("Title ID must be 16 characters long!");
|
return titleId.Length == 16 ? PrivDownloadTmd(titleId, titleVersion) : throw new Exception("Title ID must be 16 characters long!");
|
||||||
}
|
}
|
||||||
|
|
||||||
public Ticket DownloadTicket(string titleId)
|
public TMD DownloadTMD(string titleId, string titleVersion, string nusUrl)
|
||||||
{
|
{
|
||||||
return titleId.Length == 16 ? PrivDownloadTicket(titleId) : throw new Exception("Title ID must be 16 characters long!");
|
return titleId.Length == 16 ? PrivDownloadTmd(titleId, titleVersion, nusUrl) : throw new Exception("Title ID must be 16 characters long!");
|
||||||
|
}
|
||||||
|
|
||||||
|
public Ticket DownloadTicket(string titleId, string nusUrl)
|
||||||
|
{
|
||||||
|
return titleId.Length == 16 ? PrivDownloadTicket(titleId, nusUrl) : throw new Exception("Title ID must be 16 characters long!");
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte[] DownloadSingleContent(string titleId, string titleVersion, string contentId)
|
public byte[] DownloadSingleContent(string titleId, string titleVersion, string contentId)
|
||||||
|
@ -141,6 +161,32 @@ namespace libWiiSharp
|
||||||
File.WriteAllBytes(savePath, bytes);
|
File.WriteAllBytes(savePath, bytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void DownloadSingleContent(
|
||||||
|
string titleId,
|
||||||
|
string titleVersion,
|
||||||
|
string contentId,
|
||||||
|
string savePath,
|
||||||
|
string nusUrl)
|
||||||
|
{
|
||||||
|
if (titleId.Length != 16)
|
||||||
|
{
|
||||||
|
throw new Exception("Title ID must be 16 characters long!");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Directory.Exists(Path.GetDirectoryName(savePath)))
|
||||||
|
{
|
||||||
|
Directory.CreateDirectory(Path.GetDirectoryName(savePath));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (File.Exists(savePath))
|
||||||
|
{
|
||||||
|
File.Delete(savePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] bytes = PrivDownloadSingleContent(titleId, titleVersion, contentId, nusUrl);
|
||||||
|
File.WriteAllBytes(savePath, bytes);
|
||||||
|
}
|
||||||
|
|
||||||
private byte[] PrivDownloadSingleContent(string titleId, string titleVersion, string contentId)
|
private byte[] PrivDownloadSingleContent(string titleId, string titleVersion, string contentId)
|
||||||
{
|
{
|
||||||
uint num = uint.Parse(contentId, NumberStyles.HexNumber);
|
uint num = uint.Parse(contentId, NumberStyles.HexNumber);
|
||||||
|
@ -208,6 +254,77 @@ namespace libWiiSharp
|
||||||
return array;
|
return array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private byte[] PrivDownloadSingleContent(string titleId, string titleVersion, string contentId, string nusUrl)
|
||||||
|
{
|
||||||
|
uint num = uint.Parse(contentId, NumberStyles.HexNumber);
|
||||||
|
contentId = num.ToString("x8");
|
||||||
|
FireDebug("Downloading Content (Content ID: {0}) of Title {1} v{2}...", contentId, titleId, string.IsNullOrEmpty(titleVersion) ? "[Latest]" : titleVersion);
|
||||||
|
FireDebug(" Checking for Internet connection...");
|
||||||
|
if (!PrivCheckInet())
|
||||||
|
{
|
||||||
|
FireDebug(" Connection not found...");
|
||||||
|
throw new Exception("You're not connected to the internet!");
|
||||||
|
}
|
||||||
|
FireProgress(0);
|
||||||
|
string str1 = "tmd" + (string.IsNullOrEmpty(titleVersion) ? string.Empty : "." + titleVersion);
|
||||||
|
string str2 = string.Format("{0}{1}/", nusUrl, titleId);
|
||||||
|
string empty = string.Empty;
|
||||||
|
int contentIndex = 0;
|
||||||
|
FireDebug(" Downloading TMD...");
|
||||||
|
byte[] tmdFile = wcNus.DownloadData(str2 + str1);
|
||||||
|
FireDebug(" Parsing TMD...");
|
||||||
|
TMD tmd = TMD.Load(tmdFile);
|
||||||
|
FireProgress(20);
|
||||||
|
FireDebug(" Looking for Content ID {0} in TMD...", (object)contentId);
|
||||||
|
bool flag = false;
|
||||||
|
for (int index = 0; index < tmd.Contents.Length; ++index)
|
||||||
|
{
|
||||||
|
if ((int)tmd.Contents[index].ContentID == (int)num)
|
||||||
|
{
|
||||||
|
FireDebug(" Content ID {0} found in TMD...", (object)contentId);
|
||||||
|
flag = true;
|
||||||
|
empty = tmd.Contents[index].ContentID.ToString("x8");
|
||||||
|
contentIndex = index;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!flag)
|
||||||
|
{
|
||||||
|
FireDebug(" Content ID {0} wasn't found in TMD...", (object)contentId);
|
||||||
|
throw new Exception("Content ID wasn't found in the TMD!");
|
||||||
|
}
|
||||||
|
if (!File.Exists("cetk") && !continueWithoutTicket)
|
||||||
|
{
|
||||||
|
FireDebug(" Downloading Ticket...");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
byte[] tikArray = wcNus.DownloadData(str2 + "cetk");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
FireDebug(" Downloading Ticket Failed...");
|
||||||
|
throw new Exception("CETK Doesn't Exist and Downloading Ticket Failed:\n" + ex.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
FireDebug("Parsing Ticket...");
|
||||||
|
Ticket tik = Ticket.Load("cetk");
|
||||||
|
FireProgress(40);
|
||||||
|
FireDebug(" Downloading Content... ({0} bytes)", (object)tmd.Contents[contentIndex].Size);
|
||||||
|
byte[] content = wcNus.DownloadData(str2 + empty);
|
||||||
|
FireProgress(80);
|
||||||
|
FireDebug(" Decrypting Content...");
|
||||||
|
byte[] array = PrivDecryptContent(content, contentIndex, tik, tmd);
|
||||||
|
Array.Resize<byte>(ref array, (int)tmd.Contents[contentIndex].Size);
|
||||||
|
if (!Shared.CompareByteArrays(SHA1.Create().ComputeHash(array), tmd.Contents[contentIndex].Hash))
|
||||||
|
{
|
||||||
|
FireDebug("/!\\ /!\\ /!\\ Hashes do not match /!\\ /!\\ /!\\");
|
||||||
|
throw new Exception("Hashes do not match!");
|
||||||
|
}
|
||||||
|
FireProgress(100);
|
||||||
|
FireDebug("Downloading Content (Content ID: {0}) of Title {1} v{2} Finished...", contentId, titleId, string.IsNullOrEmpty(titleVersion) ? "[Latest]" : titleVersion);
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
private Ticket PrivDownloadTicket(string titleId)
|
private Ticket PrivDownloadTicket(string titleId)
|
||||||
{
|
{
|
||||||
string nusUrl = PrivNUSUp();
|
string nusUrl = PrivNUSUp();
|
||||||
|
@ -218,6 +335,20 @@ namespace libWiiSharp
|
||||||
return Ticket.Load(tikArray);
|
return Ticket.Load(tikArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Ticket PrivDownloadTicket(string titleId, string nusUrl)
|
||||||
|
{
|
||||||
|
if (!PrivCheckInet())
|
||||||
|
{
|
||||||
|
FireDebug(" Connection not found...");
|
||||||
|
throw new Exception("You're not connected to the internet!");
|
||||||
|
}
|
||||||
|
|
||||||
|
string titleUrl = string.Format("{0}{1}/", nusUrl, titleId);
|
||||||
|
byte[] tikArray = wcNus.DownloadData(titleUrl + "cetk");
|
||||||
|
|
||||||
|
return Ticket.Load(tikArray);
|
||||||
|
}
|
||||||
|
|
||||||
private TMD PrivDownloadTmd(string titleId, string titleVersion)
|
private TMD PrivDownloadTmd(string titleId, string titleVersion)
|
||||||
{
|
{
|
||||||
string nusUrl = PrivNUSUp();
|
string nusUrl = PrivNUSUp();
|
||||||
|
@ -225,6 +356,17 @@ namespace libWiiSharp
|
||||||
return TMD.Load(wcNus.DownloadData(string.Format("{0}{1}/", nusUrl, titleId) + ("tmd" + (string.IsNullOrEmpty(titleVersion) ? string.Empty : "." + titleVersion))));
|
return TMD.Load(wcNus.DownloadData(string.Format("{0}{1}/", nusUrl, titleId) + ("tmd" + (string.IsNullOrEmpty(titleVersion) ? string.Empty : "." + titleVersion))));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private TMD PrivDownloadTmd(string titleId, string titleVersion, string nusUrl)
|
||||||
|
{
|
||||||
|
if (!PrivCheckInet())
|
||||||
|
{
|
||||||
|
FireDebug(" Connection not found...");
|
||||||
|
throw new Exception("You're not connected to the internet!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return TMD.Load(wcNus.DownloadData(string.Format("{0}{1}/", nusUrl, titleId) + ("tmd" + (string.IsNullOrEmpty(titleVersion) ? string.Empty : "." + titleVersion))));
|
||||||
|
}
|
||||||
|
|
||||||
private void PrivDownloadTitle(
|
private void PrivDownloadTitle(
|
||||||
string titleId,
|
string titleId,
|
||||||
string titleVersion,
|
string titleVersion,
|
||||||
|
@ -451,6 +593,237 @@ namespace libWiiSharp
|
||||||
FireProgress(100);
|
FireProgress(100);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void PrivDownloadTitle(
|
||||||
|
string titleId,
|
||||||
|
string titleVersion,
|
||||||
|
string outputDir,
|
||||||
|
string nusUrl,
|
||||||
|
StoreType[] storeTypes)
|
||||||
|
{
|
||||||
|
FireDebug("Downloading Title {0} v{1}...", titleId, string.IsNullOrEmpty(titleVersion) ? "[Latest]" : titleVersion);
|
||||||
|
if (storeTypes.Length < 1)
|
||||||
|
{
|
||||||
|
FireDebug(" No store types were defined...");
|
||||||
|
throw new Exception("You must at least define one store type!");
|
||||||
|
}
|
||||||
|
FireDebug(" Checking for Internet connection...");
|
||||||
|
if (!PrivCheckInet())
|
||||||
|
{
|
||||||
|
FireDebug(" Connection not found...");
|
||||||
|
throw new Exception("You're not connected to the internet!");
|
||||||
|
}
|
||||||
|
string str1 = string.Format("{0}{1}/", nusUrl, titleId);
|
||||||
|
bool flag1 = false;
|
||||||
|
bool flag2 = false;
|
||||||
|
bool flag3 = false;
|
||||||
|
FireProgress(0);
|
||||||
|
for (int index = 0; index < storeTypes.Length; ++index)
|
||||||
|
{
|
||||||
|
switch (storeTypes[index])
|
||||||
|
{
|
||||||
|
case StoreType.EncryptedContent:
|
||||||
|
FireDebug(" -> Storing Encrypted Content...");
|
||||||
|
flag1 = true;
|
||||||
|
break;
|
||||||
|
case StoreType.DecryptedContent:
|
||||||
|
FireDebug(" -> Storing Decrypted Content...");
|
||||||
|
flag2 = true;
|
||||||
|
break;
|
||||||
|
case StoreType.WAD:
|
||||||
|
FireDebug(" -> Storing WAD...");
|
||||||
|
flag3 = true;
|
||||||
|
break;
|
||||||
|
case StoreType.All:
|
||||||
|
FireDebug(" -> Storing Decrypted Content...");
|
||||||
|
FireDebug(" -> Storing Encrypted Content...");
|
||||||
|
FireDebug(" -> Storing WAD...");
|
||||||
|
flag2 = true;
|
||||||
|
flag1 = true;
|
||||||
|
flag3 = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (ContinueWithoutTicket == true)
|
||||||
|
{
|
||||||
|
flag2 = false;
|
||||||
|
flag1 = true;
|
||||||
|
flag3 = false;
|
||||||
|
}
|
||||||
|
if (outputDir[outputDir.Length - 1] != Path.DirectorySeparatorChar)
|
||||||
|
{
|
||||||
|
outputDir += Path.DirectorySeparatorChar.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Directory.Exists(outputDir))
|
||||||
|
{
|
||||||
|
Directory.CreateDirectory(outputDir);
|
||||||
|
}
|
||||||
|
|
||||||
|
string str2 = "tmd" + (string.IsNullOrEmpty(titleVersion) ? string.Empty : "." + titleVersion);
|
||||||
|
FireDebug(" Downloading TMD...");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
wcNus.DownloadFile(str1 + str2, outputDir + str2);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
FireDebug(" Downloading TMD Failed...");
|
||||||
|
throw new Exception("Downloading TMD Failed:\n" + ex.Message);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (!File.Exists(outputDir + "cetk"))
|
||||||
|
{
|
||||||
|
//Download cetk
|
||||||
|
FireDebug(" Downloading Ticket...");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
wcNus.DownloadFile(string.Format("{0}{1}/", nusUrl, titleId) + "cetk", outputDir + "cetk");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
if (!continueWithoutTicket || !flag1)
|
||||||
|
{
|
||||||
|
FireDebug(" Downloading Ticket Failed...");
|
||||||
|
throw new Exception("CETK Doesn't Exist and Downloading Ticket Failed:\n" + ex.Message);
|
||||||
|
}
|
||||||
|
|
||||||
|
flag2 = false;
|
||||||
|
flag3 = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FireProgress(10);
|
||||||
|
FireDebug(" Parsing TMD...");
|
||||||
|
TMD tmd = TMD.Load(outputDir + str2);
|
||||||
|
if (string.IsNullOrEmpty(titleVersion))
|
||||||
|
{
|
||||||
|
FireDebug(" -> Title Version: {0}", (object)tmd.TitleVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
FireDebug(" -> {0} Contents", (object)tmd.NumOfContents);
|
||||||
|
FireDebug(" Parsing Ticket...");
|
||||||
|
Ticket tik = null;
|
||||||
|
if (!continueWithoutTicket) { tik = Ticket.Load(outputDir + "cetk"); }
|
||||||
|
string[] strArray1 = new string[tmd.NumOfContents];
|
||||||
|
uint contentId;
|
||||||
|
for (int index1 = 0; index1 < tmd.NumOfContents; ++index1)
|
||||||
|
{
|
||||||
|
FireDebug(" Downloading Content #{0} of {1}... ({2} bytes)", index1 + 1, tmd.NumOfContents, tmd.Contents[index1].Size);
|
||||||
|
FireProgress((index1 + 1) * 60 / tmd.NumOfContents + 10);
|
||||||
|
if (useLocalFiles)
|
||||||
|
{
|
||||||
|
string str3 = outputDir;
|
||||||
|
contentId = tmd.Contents[index1].ContentID;
|
||||||
|
string str4 = contentId.ToString("x8");
|
||||||
|
if (File.Exists(str3 + str4))
|
||||||
|
{
|
||||||
|
FireDebug(" Using Local File, Skipping...");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
try
|
||||||
|
{
|
||||||
|
WebClient wcNus = this.wcNus;
|
||||||
|
string str3 = str1;
|
||||||
|
contentId = tmd.Contents[index1].ContentID;
|
||||||
|
string str4 = contentId.ToString("x8");
|
||||||
|
string address = str3 + str4;
|
||||||
|
string str5 = outputDir;
|
||||||
|
contentId = tmd.Contents[index1].ContentID;
|
||||||
|
string str6 = contentId.ToString("x8");
|
||||||
|
string fileName = str5 + str6;
|
||||||
|
wcNus.DownloadFile(address, fileName);
|
||||||
|
string[] strArray2 = strArray1;
|
||||||
|
int index2 = index1;
|
||||||
|
contentId = tmd.Contents[index1].ContentID;
|
||||||
|
string str7 = contentId.ToString("x8");
|
||||||
|
strArray2[index2] = str7;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
FireDebug(" Downloading Content #{0} of {1} failed...", index1 + 1, tmd.NumOfContents);
|
||||||
|
throw new Exception("Downloading Content Failed:\n" + ex.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (flag2 | flag3)
|
||||||
|
{
|
||||||
|
SHA1 shA1 = SHA1.Create();
|
||||||
|
for (int contentIndex = 0; contentIndex < tmd.NumOfContents; ++contentIndex)
|
||||||
|
{
|
||||||
|
FireDebug(" Decrypting Content #{0} of {1}...", contentIndex + 1, tmd.NumOfContents);
|
||||||
|
FireProgress((contentIndex + 1) * 20 / tmd.NumOfContents + 75);
|
||||||
|
string str3 = outputDir;
|
||||||
|
contentId = tmd.Contents[contentIndex].ContentID;
|
||||||
|
string str4 = contentId.ToString("x8");
|
||||||
|
byte[] array = PrivDecryptContent(File.ReadAllBytes(str3 + str4), contentIndex, tik, tmd);
|
||||||
|
Array.Resize<byte>(ref array, (int)tmd.Contents[contentIndex].Size);
|
||||||
|
if (!Shared.CompareByteArrays(shA1.ComputeHash(array), tmd.Contents[contentIndex].Hash))
|
||||||
|
{
|
||||||
|
FireDebug("/!\\ /!\\ /!\\ Hashes do not match /!\\ /!\\ /!\\");
|
||||||
|
throw new Exception(string.Format("Content #{0}: Hashes do not match!", contentIndex));
|
||||||
|
}
|
||||||
|
string str5 = outputDir;
|
||||||
|
contentId = tmd.Contents[contentIndex].ContentID;
|
||||||
|
string str6 = contentId.ToString("x8");
|
||||||
|
File.WriteAllBytes(str5 + str6 + ".app", array);
|
||||||
|
}
|
||||||
|
shA1.Clear();
|
||||||
|
}
|
||||||
|
if (flag3)
|
||||||
|
{
|
||||||
|
FireDebug(" Building Certificate Chain...");
|
||||||
|
CertificateChain cert = CertificateChain.FromTikTmd(outputDir + "cetk", outputDir + str2);
|
||||||
|
byte[][] contents = new byte[tmd.NumOfContents][];
|
||||||
|
for (int index1 = 0; index1 < tmd.NumOfContents; ++index1)
|
||||||
|
{
|
||||||
|
byte[][] numArray1 = contents;
|
||||||
|
int index2 = index1;
|
||||||
|
string str3 = outputDir;
|
||||||
|
contentId = tmd.Contents[index1].ContentID;
|
||||||
|
string str4 = contentId.ToString("x8");
|
||||||
|
byte[] numArray2 = File.ReadAllBytes(str3 + str4 + ".app");
|
||||||
|
numArray1[index2] = numArray2;
|
||||||
|
}
|
||||||
|
FireDebug(" Creating WAD...");
|
||||||
|
WAD.Create(cert, tik, tmd, contents).Save(outputDir + tmd.TitleID.ToString("x16") + "v" + tmd.TitleVersion.ToString() + ".wad");
|
||||||
|
}
|
||||||
|
if (!flag1)
|
||||||
|
{
|
||||||
|
FireDebug(" Deleting Encrypted Contents...");
|
||||||
|
for (int index = 0; index < strArray1.Length; ++index)
|
||||||
|
{
|
||||||
|
if (File.Exists(outputDir + strArray1[index]))
|
||||||
|
{
|
||||||
|
File.Delete(outputDir + strArray1[index]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (flag3 && !flag2)
|
||||||
|
{
|
||||||
|
FireDebug(" Deleting Decrypted Contents...");
|
||||||
|
for (int index = 0; index < strArray1.Length; ++index)
|
||||||
|
{
|
||||||
|
if (File.Exists(outputDir + strArray1[index] + ".app"))
|
||||||
|
{
|
||||||
|
File.Delete(outputDir + strArray1[index] + ".app");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!flag2 && !flag1)
|
||||||
|
{
|
||||||
|
FireDebug(" Deleting TMD and Ticket...");
|
||||||
|
File.Delete(outputDir + str2);
|
||||||
|
if (ContinueWithoutTicket == false)
|
||||||
|
{
|
||||||
|
File.Delete(outputDir + "cetk");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
FireDebug("Downloading Title {0} v{1} Finished...", titleId, string.IsNullOrEmpty(titleVersion) ? "[Latest]" : titleVersion);
|
||||||
|
FireProgress(100);
|
||||||
|
}
|
||||||
|
|
||||||
private byte[] PrivDecryptContent(byte[] content, int contentIndex, Ticket tik, TMD tmd)
|
private byte[] PrivDecryptContent(byte[] content, int contentIndex, Ticket tik, TMD tmd)
|
||||||
{
|
{
|
||||||
Array.Resize<byte>(ref content, Shared.AddPadding(content.Length, 16));
|
Array.Resize<byte>(ref content, Shared.AddPadding(content.Length, 16));
|
||||||
|
|
Loading…
Reference in a new issue