mirror of
https://github.com/Maschell/JNUSLib.git
synced 2024-11-25 01:16:55 +01:00
Update readme
This commit is contained in:
parent
95802c7f5d
commit
23f7066f0a
196
README.md
196
README.md
@ -24,85 +24,122 @@ For WUD files, following additional operations are possible:
|
|||||||
* Compressing into .wux file format (from .wud and splitted .wud)
|
* Compressing into .wux file format (from .wud and splitted .wud)
|
||||||
* Verifing (comparing) to different wud images.
|
* Verifing (comparing) to different wud images.
|
||||||
|
|
||||||
## How to use it
|
## Maven
|
||||||
At first you have to import the jnuslib.jar and the common key in the Settings class.
|
|
||||||
```
|
```
|
||||||
Settings.commonKey = Utils.StringToByteArray("12345678901234567890123456789012");
|
<repository>
|
||||||
|
<id>jitpack.io</id>
|
||||||
|
<url>https://jitpack.io</url>
|
||||||
|
</repository>
|
||||||
|
...
|
||||||
|
<!-- The core module -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>de.Maschell.JNUSLib</groupId>
|
||||||
|
<artifactId>sdcf4j-core</artifactId>
|
||||||
|
<version>%version OR commit%</version>
|
||||||
|
</dependency>
|
||||||
```
|
```
|
||||||
Then for each different source type, you need to use a an own NUSTitleLoader.
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Gettings NUSTitles
|
||||||
|
Gor each different source type, you need to use a an own NUSTitleLoader.
|
||||||
|
|
||||||
```
|
```
|
||||||
//REMOTE
|
byte[] commonKey = Utils.StringToByteArray("12345678901234567890123456789012");
|
||||||
|
```
|
||||||
|
#### Remote
|
||||||
|
```
|
||||||
|
// Loading a title with a public ticket
|
||||||
|
NUSTitle nusRemote = NUSTitleLoaderRemote.loadNUSTitle(0x0005000E12345678L, commonKey);
|
||||||
|
|
||||||
//Loading a title with a public ticket
|
// Loading a title with an own ticket (key, titleid)
|
||||||
NUSTitle nusRemote = NUSTitleLoaderRemote.loadNUSTitle(0x0005000E12345678L);
|
Ticket ticket = Ticket.createTicket(Utils.StringToByteArray("12345678901234567890123456789012"), 0x0005000E12345678L, commonKey);
|
||||||
|
NUSTitle nusRemoteWithTicket = NUSTitleLoaderRemote.loadNUSTitle(0x0005000E12345678L, commonKey);
|
||||||
//Loading a title with an own ticket (key, titleid)
|
```
|
||||||
Ticket ticket = Ticket.createTicket(Utils.StringToByteArray("12345678901234567890123456789012"), 0x0005000E12345678L);
|
|
||||||
NUSTitle nusRemoteWithTicket = NUSTitleLoaderRemote.loadNUSTitle(0x0005000E12345678L);
|
|
||||||
//LOCAL
|
|
||||||
|
|
||||||
//With ticket on disk
|
#### Local
|
||||||
NUSTitle nusLocal = NUSTitleLoaderLocal.loadNUSTitle("path-to-app-files");
|
```
|
||||||
|
// With ticket on disk
|
||||||
//Loading a title with an own ticket (key, titleid)
|
NUSTitle nusLocal = NUSTitleLoaderLocal.loadNUSTitle("path-to-app-files", commonKey);
|
||||||
Ticket ticket = Ticket.createTicket(Utils.StringToByteArray("12345678901234567890123456789012"), 0x0005000E12345678L);
|
|
||||||
NUSTitle nusRemoteWithTicket = NUSTitleLoaderLocal.loadNUSTitle("path-to-app-files");
|
|
||||||
|
|
||||||
//Loading a .woomy file
|
// Loading a title with an own ticket (key, titleid)
|
||||||
NUSTitle nusWoomy = NUSTitleLoaderWoomy.loadNUSTitle("testfile.woomy");
|
Ticket ticket = Ticket.createTicket(Utils.StringToByteArray("12345678901234567890123456789012"), 0x0005000E12345678L, commonKey);
|
||||||
|
NUSTitle nusLocalWithTicket = NUSTitleLoaderLocal.loadNUSTitle("path-to-app-files", ticket);
|
||||||
//WUD
|
|
||||||
//Loading a uncompressed WUD
|
// Loading a .woomy file
|
||||||
NUSTitle nusWUD = NUSTitleLoaderWUD.loadNUSTitle("game.wud");
|
NUSTitle nusWoomy = NUSTitleLoaderWoomy.loadNUSTitle("testfile.woomy");
|
||||||
//Loading a compressed WUD (WUX)
|
```
|
||||||
NUSTitle nusWUX = NUSTitleLoaderWUD.loadNUSTitle("game.wux");
|
#### WUD/WUX
|
||||||
//Loading a uncompressed splitted WUD (2gb parts)
|
```
|
||||||
NUSTitle nusWUDSplitted = NUSTitleLoaderWUD.loadNUSTitle("game_part1.wud");
|
// WUD
|
||||||
|
|
||||||
|
// Loading a uncompressed WUD
|
||||||
|
WUDInfo wiWUD = WUDLoader.load("game.wud"); // needs a game.key next to the .wud
|
||||||
|
// Loading a compressed WUD (WUX)
|
||||||
|
WUDInfo wiWUX = WUDLoader.load("game.wux"); // needs a game.key next to the .wux
|
||||||
|
// Loading a uncompressed splitted WUD (2gb parts)
|
||||||
|
WUDInfo wiWUDSplitted = WUDLoader.load("game_part1.wud"); // needs a game.key next to the .wud
|
||||||
|
|
||||||
|
// Loading providing the disc key
|
||||||
|
WUDInfo wiWUXWithDisc = WUDLoader.load("game.wux", Utils.StringToByteArray("12345678901234567890123456789012")); // needs a game.key next to the .wux
|
||||||
|
// Loading a wud with no titley key (kiosk)
|
||||||
|
WUDInfo wiKiosk = WUDLoader.loadDev("game.wux");
|
||||||
|
|
||||||
|
// Get NUSTitles from WUDInfo
|
||||||
|
List<NUSTitle> titlesFromWUD = WUDLoader.getGamePartionsAsNUSTitles(wudInfo, commonKey);
|
||||||
```
|
```
|
||||||
|
|
||||||
Once the title is loaded, you can use one of the services to extract and decrypt files.
|
Once the title is loaded, you can use one of the services to extract and decrypt files.
|
||||||
Here are some of the operations you can do. Look at the code for the full list of methods.
|
|
||||||
|
|
||||||
### Decryption:
|
### Decryption:
|
||||||
|
For the decryption you can use a FSTDataProvider in combinations with a FSTEntry. Example:
|
||||||
```
|
```
|
||||||
DecryptionService decrypt = DecryptionService.getInstance(title);
|
// Get a FSTDataProvider from NUSTitle
|
||||||
|
FSTDataProvider fstdataprovider = new FSTDataProviderNUSTitle(nustitle);
|
||||||
|
|
||||||
//Decrypt the whole FST into a folder called "DecryptedTitle"
|
// When loading from a WUD, you can get the data of all partitions via
|
||||||
decrypt.decryptAllFSTEntriesTo("DecryptedTitle");
|
List<FSTDataProvider> partitionsFromWUD = WUDLoader.getPartitonsAsFSTDataProvider(wudInfo, commonKey);
|
||||||
|
// the includes all non-nustitles like the SI or UP partitions.
|
||||||
|
|
||||||
//Decrypt the code folder into a folder called "code_folder"
|
FSTEntry fstRoot = fstdataprovider.getRoot();
|
||||||
decrypt.decryptFSTEntriesTo("/code/.*", code_folder);
|
|
||||||
|
|
||||||
//Decrypt all .js files into a folder called "js_files"
|
FSTEntry appxml = FSTUtils.getFSTEntriesByRegEx(fstdataprovider.getRoot(), ".*app.xml").get(0); // get all .rpx files
|
||||||
decrypt.decryptFSTEntriesTo(".*.js", js_files);
|
|
||||||
|
|
||||||
|
// Get data as byte array
|
||||||
|
byte[] appxmlData = fstdataprovider.readFile(appxml);
|
||||||
|
// Get 1024 bytes from entry appxml from offset 0
|
||||||
|
byte[] appxmlChunk = fstdataprovider.readFile(appxml, 0, 1024);
|
||||||
|
|
||||||
//Decrypt all .app files into a folder called "decrypted_contents"
|
// Get data as input stream
|
||||||
decrypt.decryptAllPlainContents("decrypted_contents");
|
InputStream appxmlStream = fstdataprovider.readFileAsStream(appxml);
|
||||||
|
// Get 1024 bytes from entry appxml from offset 0 as input stream
|
||||||
|
InputStream appxmlStream = fstdataprovider.readFileAsStream(appxml, 0, 1024);
|
||||||
|
|
||||||
|
// Save data to output stream
|
||||||
//Use decrypting inputstream. Data will be only loaded/decrypted on demand.
|
FileOutputStream appxmlOut = new FileOutputStream(new File(appxml.getFilename()));
|
||||||
|
if (fstdataprovider.readFileToStream(appxmlOut, appxml)) {
|
||||||
//Display the app.xml as hex dump
|
System.out.println("Okay.");
|
||||||
FSTEntry appXMLEntry = title.getFSTEntryByFullPath("code/app.xml");
|
|
||||||
decrypt.getDecryptedOutputAsInputStream(appXMLEntry);
|
|
||||||
//Lets just print the app.xml as hex data
|
|
||||||
int BUFFER_SIZE = 0x40;
|
|
||||||
byte[] buffer = new byte[BUFFER_SIZE];
|
|
||||||
int i = 0;
|
|
||||||
while(in.read(buffer) > 0){
|
|
||||||
System.out.println(String.format("0x%04X: ", (i++ * BUFFER_SIZE)) + Utils.ByteArrayToString(buffer));
|
|
||||||
}
|
}
|
||||||
in.close();
|
```
|
||||||
[...]
|
|
||||||
|
Some wrapper functions can be found in the DecryptionService:
|
||||||
|
```
|
||||||
|
FSTDataProvider fstdataprovider = new FSTDataProviderNUSTitle(nustitle);
|
||||||
|
DecryptionService decrypt = DecryptionService.getInstance(fstdataprovider);
|
||||||
|
|
||||||
|
// Decrypt the whole FST into a folder called "DecryptedTitle" and skip existing
|
||||||
|
decrypt.decryptAllFSTEntriesTo("DecryptedTitle", true);
|
||||||
|
|
||||||
|
// Decrypt the code folder into a folder called "code_folder" and skip existing
|
||||||
|
decrypt.decryptFSTEntriesTo("/code/.*", "code_folder", true);
|
||||||
```
|
```
|
||||||
|
|
||||||
### Extraction:
|
### Extraction:
|
||||||
```
|
```
|
||||||
//Get the Service for the NUSTitle
|
//Get the Service for the NUSTitle
|
||||||
ExtractionService extract = ExtractionService.getInstance(title);
|
ExtractionService extract = ExtractionService.getInstance(nusTitle);
|
||||||
|
|
||||||
//Saving all .app/.h3/tmd/tik/cert files into the folder "encryptedFiles"
|
//Saving all .app/.h3/tmd/tik/cert files into the folder "encryptedFiles"
|
||||||
extract.extractAll("encryptedFiles");
|
extract.extractAll("encryptedFiles");
|
||||||
@ -114,35 +151,42 @@ extract.extractAllEncrpytedContentFileHashes("contentHashes");
|
|||||||
extract.extractAllEncryptedContentFilesWithoutHashesTo("contents");
|
extract.extractAllEncryptedContentFilesWithoutHashesTo("contents");
|
||||||
|
|
||||||
//Save tmd, cert and ticket
|
//Save tmd, cert and ticket
|
||||||
extract.extractTMDTo(output);
|
extract.extractTMDTo("output");
|
||||||
extract.extractTickeTo(output);
|
extract.extractTicketTo("output");
|
||||||
extract.extractCertTo(output);
|
extract.extractCertTo("output");
|
||||||
|
|
||||||
[...]
|
|
||||||
```
|
```
|
||||||
### WUD Services
|
### WUD Services
|
||||||
Example for compressing and verifing into .wux files.
|
Example for compressing and verifing .wux files.
|
||||||
|
|
||||||
```
|
```
|
||||||
WUDImage imageUncompressed = new WUDImage(new File("game_part1.wud")); //Splitted and not splitted .wud possible here
|
WUDImage imageUncompressed = new WUDImage(new File("game_part1.wud")); // Splitted and not splitted .wud possible here
|
||||||
|
|
||||||
WUDService.compressWUDToWUX(imageUncompressed, "compressedImage","game.wux");
|
Optional<File> compressedWUD = WUDService.compressWUDToWUX(imageUncompressed, "compressedImage", "game.wux", false);
|
||||||
|
|
||||||
WUDImage imageCompressed = new WUDImage(new File("compressedImage" + File.separator + "game.wux"));
|
if (compressedWUD.isPresent()) {
|
||||||
|
WUDImage imageCompressed = new WUDImage(compressedWUD.get());
|
||||||
//Verify compression
|
|
||||||
WUDService.compareWUDImage(imageUncompressed, imageCompressed);
|
// Verify compression
|
||||||
|
if (WUDService.compareWUDImage(imageUncompressed, imageCompressed)) {
|
||||||
|
System.out.println("Both images are the same");
|
||||||
|
} else {
|
||||||
|
System.err.println("The images are different");
|
||||||
|
}
|
||||||
|
|
||||||
|
//Turn it back into .wud
|
||||||
|
WUDService.decompressWUX(imageCompressed, "newdecompressed", "test.wud", false);
|
||||||
|
} else {
|
||||||
|
System.err.println("Failed to compress wud");
|
||||||
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
### Cleanup:
|
### Cleanup:
|
||||||
Call the method cleanup() for a NUSTitle to cleanup/close all opened ressources.
|
Call the method cleanup() for a NUSTitle to cleanup/close all opened ressources.
|
||||||
|
|
||||||
# Credits
|
# Credits
|
||||||
Maschell
|
Maschell for creating the lib
|
||||||
|
Crediar for [CDecrypt](https://github.com/crediar/cdecrypt)
|
||||||
Thanks to:
|
All people who have contributed to [vgmtoolbox](https://sourceforge.net/projects/vgmtoolbox/)
|
||||||
Crediar for CDecrypt (https://github.com/crediar/cdecrypt)
|
Exzap for the [.wux file format](https://gbatemp.net/threads/wii-u-image-wud-compression-tool.397901/)
|
||||||
All people who have contributed to vgmtoolbox (https://sourceforge.net/projects/vgmtoolbox/)
|
FIX94 for [wudump](https://gbatemp.net/threads/wudump-dump-raw-images-from-a-wiiu-game-disc.451736/)
|
||||||
Exzap for the .wux file format (https://gbatemp.net/threads/wii-u-image-wud-compression-tool.397901/)
|
The creators of lombok for [lombok](https://projectlombok.org/index.html)
|
||||||
FIX94 for wudump (https://gbatemp.net/threads/wudump-dump-raw-images-from-a-wiiu-game-disc.451736/)
|
|
||||||
The creators of lombok for lombok https://projectlombok.org/index.html
|
|
||||||
|
Loading…
Reference in New Issue
Block a user