2020-12-28 22:28:44 -06:00
// Decompiled with JetBrains decompiler
// Type: libWiiSharp.NusClient
// Assembly: libWiiSharp, Version=0.4.0.0, Culture=neutral, PublicKeyToken=null
// MVID: FBF36F3D-B5D6-481F-B5F5-1BD3C19E13B2
// Assembly location: C:\Users\theso\Downloads\NCPatcher\pack\libWiiSharp.dll
using System ;
using System.ComponentModel ;
using System.Globalization ;
using System.IO ;
using System.Net ;
using System.Security.Cryptography ;
namespace libWiiSharp
{
2021-02-06 22:53:40 -06:00
public class NusClient : IDisposable
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
private const string nusUrl = "http://nus.cdn.shop.wii.com/ccs/download/" ;
private readonly WebClient wcNus = new WebClient ( ) ;
private bool useLocalFiles ;
private bool continueWithoutTicket ;
private bool isDisposed ;
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public bool UseLocalFiles
{
get = > useLocalFiles ;
set = > useLocalFiles = value ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public bool ContinueWithoutTicket
{
get = > continueWithoutTicket ;
set = > continueWithoutTicket = value ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public event EventHandler < ProgressChangedEventArgs > Progress ;
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public event EventHandler < MessageEventArgs > Debug ;
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
~ NusClient ( ) = > Dispose ( false ) ;
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public void Dispose ( )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
Dispose ( true ) ;
GC . SuppressFinalize ( this ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
protected virtual void Dispose ( bool disposing )
{
if ( disposing & & ! isDisposed )
{
wcNus . Dispose ( ) ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
isDisposed = true ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public void DownloadTitle (
string titleId ,
string titleVersion ,
string outputDir ,
params StoreType [ ] storeTypes )
{
if ( titleId . Length ! = 16 )
{
throw new Exception ( "Title ID must be 16 characters long!" ) ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
PrivDownloadTitle ( titleId , titleVersion , outputDir , storeTypes ) ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
public TMD DownloadTMD ( string titleId , string titleVersion )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
return titleId . Length = = 16 ? DownloadTmd ( titleId , titleVersion ) : throw new Exception ( "Title ID must be 16 characters long!" ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
public Ticket DownloadTicket ( string titleId )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
return titleId . Length = = 16 ? PrivDownloadTicket ( titleId ) : throw new Exception ( "Title ID must be 16 characters long!" ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
public byte [ ] DownloadSingleContent ( string titleId , string titleVersion , string contentId )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
if ( titleId . Length ! = 16 )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
throw new Exception ( "Title ID must be 16 characters long!" ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
return PrivDownloadSingleContent ( titleId , titleVersion , contentId ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
public void DownloadSingleContent (
string titleId ,
string titleVersion ,
string contentId ,
string savePath )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
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 ( System . IO . File . Exists ( savePath ) )
{
System . IO . File . Delete ( savePath ) ;
}
byte [ ] bytes = PrivDownloadSingleContent ( titleId , titleVersion , contentId ) ;
System . IO . File . WriteAllBytes ( savePath , bytes ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private byte [ ] PrivDownloadSingleContent ( string titleId , string titleVersion , string contentId )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
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 ( ! CheckInet ( ) )
{
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}/" , "http://nus.cdn.shop.wii.com/ccs/download/" , 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" ) )
{
FireDebug ( " Downloading Ticket..." ) ;
//byte[] tikArray = wcNus.DownloadData(str2 + "cetk");
Console . WriteLine ( "Downloading" ) ;
}
Console . WriteLine ( "Continuing" ) ;
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 = DecryptContent ( 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 ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private Ticket PrivDownloadTicket ( string titleId )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
if ( ! CheckInet ( ) )
{
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 ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private TMD DownloadTmd ( string titleId , string titleVersion )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
if ( ! CheckInet ( ) )
{
throw new Exception ( "You're not connected to the internet!" ) ;
}
return TMD . Load ( wcNus . DownloadData ( string . Format ( "{0}{1}/" , "http://nus.cdn.shop.wii.com/ccs/download/" , titleId ) + ( "tmd" + ( string . IsNullOrEmpty ( titleVersion ) ? string . Empty : "." + titleVersion ) ) ) ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private void PrivDownloadTitle (
string titleId ,
string titleVersion ,
string outputDir ,
StoreType [ ] storeTypes )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
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!" ) ;
}
string str1 = string . Format ( "{0}{1}/" , "http://nus.cdn.shop.wii.com/ccs/download/" , 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 ;
}
}
FireDebug ( " Checking for Internet connection..." ) ;
if ( ! CheckInet ( ) )
{
FireDebug ( " Connection not found..." ) ;
throw new Exception ( "You're not connected to the internet!" ) ;
}
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 = 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 ( System . IO . 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 = DecryptContent ( System . IO . 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" ) ;
System . IO . 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 = System . IO . 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 ( System . IO . File . Exists ( outputDir + strArray1 [ index ] ) )
{
System . IO . File . Delete ( outputDir + strArray1 [ index ] ) ;
}
}
}
if ( flag3 & & ! flag2 )
{
FireDebug ( " Deleting Decrypted Contents..." ) ;
for ( int index = 0 ; index < strArray1 . Length ; + + index )
{
if ( System . IO . File . Exists ( outputDir + strArray1 [ index ] + ".app" ) )
{
System . IO . File . Delete ( outputDir + strArray1 [ index ] + ".app" ) ;
}
}
}
if ( ! flag2 & & ! flag1 )
{
FireDebug ( " Deleting TMD and Ticket..." ) ;
System . IO . File . Delete ( outputDir + str2 ) ;
System . IO . File . Delete ( outputDir + "cetk" ) ;
}
FireDebug ( "Downloading Title {0} v{1} Finished..." , titleId , string . IsNullOrEmpty ( titleVersion ) ? "[Latest]" : titleVersion ) ;
FireProgress ( 100 ) ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private byte [ ] DecryptContent ( byte [ ] content , int contentIndex , Ticket tik , TMD tmd )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
Array . Resize < byte > ( ref content , Shared . AddPadding ( content . Length , 16 ) ) ;
byte [ ] titleKey = tik . TitleKey ;
byte [ ] numArray = new byte [ 16 ] ;
byte [ ] bytes = BitConverter . GetBytes ( tmd . Contents [ contentIndex ] . Index ) ;
numArray [ 0 ] = bytes [ 1 ] ;
numArray [ 1 ] = bytes [ 0 ] ;
RijndaelManaged rijndaelManaged = new RijndaelManaged
{
Mode = CipherMode . CBC ,
Padding = PaddingMode . None ,
KeySize = 128 ,
BlockSize = 128 ,
Key = titleKey ,
IV = numArray
} ;
ICryptoTransform decryptor = rijndaelManaged . CreateDecryptor ( ) ;
MemoryStream memoryStream = new MemoryStream ( content ) ;
CryptoStream cryptoStream = new CryptoStream ( memoryStream , decryptor , CryptoStreamMode . Read ) ;
byte [ ] buffer = new byte [ content . Length ] ;
cryptoStream . Read ( buffer , 0 , buffer . Length ) ;
cryptoStream . Dispose ( ) ;
memoryStream . Dispose ( ) ;
return buffer ;
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private bool CheckInet ( )
2020-12-28 22:28:44 -06:00
{
2021-02-06 22:53:40 -06:00
try
{
Dns . GetHostEntry ( "www.google.com" ) ;
return true ;
}
catch
{
return false ;
}
2020-12-28 22:28:44 -06:00
}
2021-02-06 22:53:40 -06:00
private void FireDebug ( string debugMessage , params object [ ] args )
{
EventHandler < MessageEventArgs > debug = Debug ;
if ( debug = = null )
{
return ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
debug ( new object ( ) , new MessageEventArgs ( string . Format ( debugMessage , args ) ) ) ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
private void FireProgress ( int progressPercentage )
{
EventHandler < ProgressChangedEventArgs > progress = Progress ;
if ( progress = = null )
{
return ;
}
2020-12-28 22:28:44 -06:00
2021-02-06 22:53:40 -06:00
progress ( new object ( ) , new ProgressChangedEventArgs ( progressPercentage , string . Empty ) ) ;
}
2020-12-28 22:28:44 -06:00
}
}