mirror of
https://github.com/Maschell/JNUSLib.git
synced 2024-11-26 18:04:17 +01:00
Try to parallize decryption if possible
This commit is contained in:
parent
a46011c6ea
commit
8fffcb1343
@ -17,6 +17,7 @@
|
|||||||
package de.mas.wiiu.jnus;
|
package de.mas.wiiu.jnus;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
import java.io.FileOutputStream;
|
import java.io.FileOutputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
@ -27,8 +28,11 @@ import java.util.Arrays;
|
|||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.Callable;
|
import java.util.concurrent.CompletableFuture;
|
||||||
import java.util.concurrent.ForkJoinPool;
|
import java.util.concurrent.CompletionException;
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
|
||||||
|
import javax.management.RuntimeErrorException;
|
||||||
|
|
||||||
import de.mas.wiiu.jnus.entities.TMD;
|
import de.mas.wiiu.jnus.entities.TMD;
|
||||||
import de.mas.wiiu.jnus.entities.Ticket;
|
import de.mas.wiiu.jnus.entities.Ticket;
|
||||||
@ -36,11 +40,14 @@ import de.mas.wiiu.jnus.entities.content.Content;
|
|||||||
import de.mas.wiiu.jnus.entities.fst.FSTEntry;
|
import de.mas.wiiu.jnus.entities.fst.FSTEntry;
|
||||||
import de.mas.wiiu.jnus.implementations.NUSDataProvider;
|
import de.mas.wiiu.jnus.implementations.NUSDataProvider;
|
||||||
import de.mas.wiiu.jnus.utils.CheckSumWrongException;
|
import de.mas.wiiu.jnus.utils.CheckSumWrongException;
|
||||||
|
import de.mas.wiiu.jnus.utils.FileUtils;
|
||||||
import de.mas.wiiu.jnus.utils.HashUtil;
|
import de.mas.wiiu.jnus.utils.HashUtil;
|
||||||
|
import de.mas.wiiu.jnus.utils.Parallelizable;
|
||||||
import de.mas.wiiu.jnus.utils.StreamUtils;
|
import de.mas.wiiu.jnus.utils.StreamUtils;
|
||||||
import de.mas.wiiu.jnus.utils.Utils;
|
import de.mas.wiiu.jnus.utils.Utils;
|
||||||
import de.mas.wiiu.jnus.utils.cryptography.NUSDecryption;
|
import de.mas.wiiu.jnus.utils.cryptography.NUSDecryption;
|
||||||
import lombok.Getter;
|
import lombok.Getter;
|
||||||
|
import lombok.val;
|
||||||
import lombok.extern.java.Log;
|
import lombok.extern.java.Log;
|
||||||
|
|
||||||
@Log
|
@Log
|
||||||
@ -48,6 +55,8 @@ public final class DecryptionService {
|
|||||||
private static Map<NUSTitle, DecryptionService> instances = new HashMap<>();
|
private static Map<NUSTitle, DecryptionService> instances = new HashMap<>();
|
||||||
@Getter private final NUSTitle NUSTitle;
|
@Getter private final NUSTitle NUSTitle;
|
||||||
|
|
||||||
|
private boolean parallelizable = false;
|
||||||
|
|
||||||
public static DecryptionService getInstance(NUSTitle nustitle) {
|
public static DecryptionService getInstance(NUSTitle nustitle) {
|
||||||
if (!instances.containsKey(nustitle)) {
|
if (!instances.containsKey(nustitle)) {
|
||||||
instances.put(nustitle, new DecryptionService(nustitle));
|
instances.put(nustitle, new DecryptionService(nustitle));
|
||||||
@ -56,6 +65,9 @@ public final class DecryptionService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private DecryptionService(NUSTitle nustitle) {
|
private DecryptionService(NUSTitle nustitle) {
|
||||||
|
if (nustitle.getDataProvider() instanceof Parallelizable) {
|
||||||
|
parallelizable = true;
|
||||||
|
}
|
||||||
this.NUSTitle = nustitle;
|
this.NUSTitle = nustitle;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -63,70 +75,83 @@ public final class DecryptionService {
|
|||||||
return getNUSTitle().getTicket();
|
return getNUSTitle().getTicket();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptFSTEntryTo(boolean useFullPath, FSTEntry entry, String outputPath, boolean skipExistingFile) throws IOException, CheckSumWrongException {
|
public void decryptFSTEntryToSync(boolean useFullPath, FSTEntry entry, String outputPath, boolean skipExistingFile) {
|
||||||
if (entry.isNotInPackage() || entry.getContent() == null) {
|
try {
|
||||||
return;
|
decryptFSTEntryToAsync(useFullPath, entry, outputPath, skipExistingFile).get();
|
||||||
|
} catch (InterruptedException | ExecutionException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// log.info("Decrypting " + entry.getFilename());
|
public CompletableFuture<Void> decryptFSTEntryToAsync(boolean useFullPath, FSTEntry entry, String outputPath, boolean skipExistingFile) {
|
||||||
|
return CompletableFuture.runAsync(() -> {
|
||||||
String targetFilePath = new StringBuilder().append(outputPath).append("/").append(entry.getFilename()).toString();
|
try {
|
||||||
String fullPath = new StringBuilder().append(outputPath).toString();
|
if (entry.isNotInPackage() || entry.getContent() == null) {
|
||||||
|
|
||||||
if (useFullPath) {
|
|
||||||
targetFilePath = new StringBuilder().append(outputPath).append(entry.getFullPath()).toString();
|
|
||||||
fullPath = new StringBuilder().append(outputPath).append(entry.getPath()).toString();
|
|
||||||
if (entry.isDir()) { // If the entry is a directory. Create it and return.
|
|
||||||
Utils.createDir(targetFilePath);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else if (entry.isDir()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!Utils.createDir(fullPath)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
File target = new File(targetFilePath);
|
|
||||||
|
|
||||||
if (skipExistingFile) {
|
|
||||||
File targetFile = new File(targetFilePath);
|
|
||||||
if (targetFile.exists()) {
|
|
||||||
if (entry.isDir()) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (targetFile.length() == entry.getFileSize()) {
|
|
||||||
Content c = entry.getContent();
|
log.fine("Decrypting " + entry.getFilename());
|
||||||
if (c.isHashed()) {
|
|
||||||
log.info("File already exists: " + entry.getFilename());
|
String targetFilePath = new StringBuilder().append(outputPath).append("/").append(entry.getFilename()).toString();
|
||||||
|
String fullPath = new StringBuilder().append(outputPath).toString();
|
||||||
|
|
||||||
|
if (useFullPath) {
|
||||||
|
targetFilePath = new StringBuilder().append(outputPath).append(entry.getFullPath()).toString();
|
||||||
|
fullPath = new StringBuilder().append(outputPath).append(entry.getPath()).toString();
|
||||||
|
if (entry.isDir()) { // If the entry is a directory. Create it and return.
|
||||||
|
Utils.createDir(targetFilePath);
|
||||||
return;
|
return;
|
||||||
} else {
|
}
|
||||||
if (Arrays.equals(HashUtil.hashSHA1(target, (int) c.getDecryptedFileSize()), c.getSHA2Hash())) {
|
} else if (entry.isDir()) {
|
||||||
log.info("File already exists: " + entry.getFilename());
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Utils.createDir(fullPath)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
File target = new File(targetFilePath);
|
||||||
|
|
||||||
|
if (skipExistingFile) {
|
||||||
|
File targetFile = new File(targetFilePath);
|
||||||
|
if (targetFile.exists()) {
|
||||||
|
if (entry.isDir()) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
if (targetFile.length() == entry.getFileSize()) {
|
||||||
|
Content c = entry.getContent();
|
||||||
|
if (c.isHashed()) {
|
||||||
|
log.info("File already exists: " + entry.getFilename());
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
if (Arrays.equals(HashUtil.hashSHA1(target, (int) c.getDecryptedFileSize()), c.getSHA2Hash())) {
|
||||||
|
log.info("File already exists: " + entry.getFilename());
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
log.info("File already exists with the same filesize, but the hash doesn't match: " + entry.getFilename());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
log.info("File already exists with the same filesize, but the hash doesn't match: " + entry.getFilename());
|
log.info("File already exists but the filesize doesn't match: " + entry.getFilename());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
|
||||||
log.info("File already exists but the filesize doesn't match: " + entry.getFilename());
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
FileOutputStream outputStream = new FileOutputStream(new File(targetFilePath));
|
// to avoid having fragmented files.
|
||||||
try {
|
FileUtils.FileAsOutputStreamWrapper(new File(targetFilePath), entry.getFileSize(), newOutputStream -> {
|
||||||
decryptFSTEntryToStream(entry, outputStream);
|
try {
|
||||||
} catch (CheckSumWrongException e) {
|
decryptFSTEntryToStream(entry, newOutputStream);
|
||||||
if (entry.getFilename().endsWith(".xml") && Utils.checkXML(new File(targetFilePath))) {
|
} catch (CheckSumWrongException e) {
|
||||||
log.info("Hash doesn't match, but it's an XML file and it looks okay.");
|
log.info("Hash doesn't match!");
|
||||||
} else {
|
// Wrapp it into a IOException
|
||||||
log.info("Hash doesn't match!");
|
throw new IOException(e);
|
||||||
throw e;
|
}
|
||||||
|
});
|
||||||
|
} catch (Exception ex) {
|
||||||
|
throw new CompletionException(ex);
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptFSTEntryToStream(FSTEntry entry, OutputStream outputStream) throws IOException, CheckSumWrongException {
|
public void decryptFSTEntryToStream(FSTEntry entry, OutputStream outputStream) throws IOException, CheckSumWrongException {
|
||||||
@ -147,17 +172,11 @@ public final class DecryptionService {
|
|||||||
try {
|
try {
|
||||||
decryptFSTEntryFromStreams(in, outputStream, fileSize, fileOffset, c);
|
decryptFSTEntryFromStreams(in, outputStream, fileSize, fileOffset, c);
|
||||||
} catch (CheckSumWrongException e) {
|
} catch (CheckSumWrongException e) {
|
||||||
log.info("Hash doesn't match");
|
if (entry.getContent().isUNKNWNFlag1Set()) {
|
||||||
if (entry.getFilename().endsWith(".xml")) {
|
log.info("Hash doesn't match. But file is optional. Don't worry.");
|
||||||
if (outputStream instanceof PipedOutputStream) {
|
|
||||||
log.info("Hash doesn't match. Please check the data for " + entry.getFullPath());
|
|
||||||
} else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
} else if (entry.getContent().isUNKNWNFlag1Set()) {
|
|
||||||
log.info("But file is optional. Don't worry.");
|
|
||||||
} else {
|
} else {
|
||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("Hash doesn't match").append(System.lineSeparator());
|
||||||
sb.append("Detailed info:").append(System.lineSeparator());
|
sb.append("Detailed info:").append(System.lineSeparator());
|
||||||
sb.append(entry).append(System.lineSeparator());
|
sb.append(entry).append(System.lineSeparator());
|
||||||
sb.append(entry.getContent()).append(System.lineSeparator());
|
sb.append(entry.getContent()).append(System.lineSeparator());
|
||||||
@ -200,33 +219,51 @@ public final class DecryptionService {
|
|||||||
StreamUtils.saveInputStreamToOutputStreamWithHash(inputStream, outputStream, size, content.getSHA2Hash(), encryptedFileSize);
|
StreamUtils.saveInputStreamToOutputStreamWithHash(inputStream, outputStream, size, content.getSHA2Hash(), encryptedFileSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
inputStream.close();
|
synchronized (inputStream) {
|
||||||
outputStream.close();
|
inputStream.close();
|
||||||
|
}
|
||||||
|
synchronized (outputStream) {
|
||||||
|
outputStream.close();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptContentTo(Content content, String outPath, boolean skipExistingFile) throws IOException, CheckSumWrongException {
|
public void decryptContentToSync(Content content, String outPath, boolean skipExistingFile) {
|
||||||
String targetFilePath = outPath + File.separator + content.getFilenameDecrypted();
|
try {
|
||||||
if (skipExistingFile) {
|
decryptContentToAsync(content, outPath, skipExistingFile).get();
|
||||||
File targetFile = new File(targetFilePath);
|
} catch (InterruptedException | ExecutionException e) {
|
||||||
if (targetFile.exists()) {
|
throw new RuntimeException(e);
|
||||||
if (targetFile.length() == content.getDecryptedFileSize()) {
|
}
|
||||||
log.info("File already exists : " + content.getFilenameDecrypted());
|
}
|
||||||
return;
|
|
||||||
} else {
|
public CompletableFuture<Void> decryptContentToAsync(Content content, String outPath, boolean skipExistingFile) {
|
||||||
log.info("File already exists but the filesize doesn't match: " + content.getFilenameDecrypted());
|
return CompletableFuture.runAsync(() -> {
|
||||||
|
try {
|
||||||
|
String targetFilePath = outPath + File.separator + content.getFilenameDecrypted();
|
||||||
|
if (skipExistingFile) {
|
||||||
|
File targetFile = new File(targetFilePath);
|
||||||
|
if (targetFile.exists()) {
|
||||||
|
if (targetFile.length() == content.getDecryptedFileSize()) {
|
||||||
|
log.info("File already exists : " + content.getFilenameDecrypted());
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
log.info("File already exists but the filesize doesn't match: " + content.getFilenameDecrypted());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!Utils.createDir(outPath)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
log.info("Decrypting Content " + String.format("%08X", content.getID()));
|
||||||
|
|
||||||
|
FileOutputStream outputStream = new FileOutputStream(new File(targetFilePath));
|
||||||
|
|
||||||
|
decryptContentToStream(content, outputStream);
|
||||||
|
} catch (Exception ex) {
|
||||||
|
throw new CompletionException(ex);
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
|
|
||||||
if (!Utils.createDir(outPath)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
log.info("Decrypting Content " + String.format("%08X", content.getID()));
|
|
||||||
|
|
||||||
FileOutputStream outputStream = new FileOutputStream(new File(targetFilePath));
|
|
||||||
|
|
||||||
decryptContentToStream(content, outputStream);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptContentToStream(Content content, OutputStream outputStream) throws IOException, CheckSumWrongException {
|
public void decryptContentToStream(Content content, OutputStream outputStream) throws IOException, CheckSumWrongException {
|
||||||
@ -240,18 +277,17 @@ public final class DecryptionService {
|
|||||||
decryptContentFromStream(inputStream, outputStream, content);
|
decryptContentFromStream(inputStream, outputStream, content);
|
||||||
}
|
}
|
||||||
|
|
||||||
public PipedInputStreamWithException getDecryptedOutputAsInputStream(FSTEntry fstEntry) throws IOException {
|
public InputStreamWithException getDecryptedOutputAsInputStream(FSTEntry fstEntry) throws IOException {
|
||||||
PipedInputStreamWithException in = new PipedInputStreamWithException();
|
PipedInputStreamWithException in = new PipedInputStreamWithException();
|
||||||
PipedOutputStream out = new PipedOutputStream(in);
|
PipedOutputStream out = new PipedOutputStream(in);
|
||||||
|
|
||||||
new Thread(() -> {
|
new Thread(() -> {
|
||||||
try { // Throwing it in both cases is EXTREMLY important. Otherwise it'll end in a deadlock
|
try {
|
||||||
decryptFSTEntryToStream(fstEntry, out);
|
decryptFSTEntryToStream(fstEntry, out);
|
||||||
in.throwException(null);
|
in.throwException(null);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
in.throwException(e);
|
in.throwException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
}).start();
|
}).start();
|
||||||
|
|
||||||
return in;
|
return in;
|
||||||
@ -262,7 +298,8 @@ public final class DecryptionService {
|
|||||||
PipedOutputStream out = new PipedOutputStream(in);
|
PipedOutputStream out = new PipedOutputStream(in);
|
||||||
|
|
||||||
new Thread(() -> {
|
new Thread(() -> {
|
||||||
try {// Throwing it in both cases is EXTREMLY important. Otherwise it'll end in a deadlock
|
try {// Throwing it in both cases is EXTREMLY important. Otherwise it'll end in a
|
||||||
|
// deadlock
|
||||||
decryptContentToStream(content, out);
|
decryptContentToStream(content, out);
|
||||||
in.throwException(null);
|
in.throwException(null);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
@ -280,6 +317,7 @@ public final class DecryptionService {
|
|||||||
FSTEntry entry = getNUSTitle().getFSTEntryByFullPath(entryFullPath);
|
FSTEntry entry = getNUSTitle().getFSTEntryByFullPath(entryFullPath);
|
||||||
if (entry == null) {
|
if (entry == null) {
|
||||||
log.info("File not found");
|
log.info("File not found");
|
||||||
|
throw new FileNotFoundException("File not found");
|
||||||
}
|
}
|
||||||
|
|
||||||
decryptFSTEntryToStream(entry, outputStream);
|
decryptFSTEntryToStream(entry, outputStream);
|
||||||
@ -302,31 +340,27 @@ public final class DecryptionService {
|
|||||||
|
|
||||||
public void decryptFSTEntryTo(boolean fullPath, String entryFullPath, String outputFolder, boolean skipExistingFiles)
|
public void decryptFSTEntryTo(boolean fullPath, String entryFullPath, String outputFolder, boolean skipExistingFiles)
|
||||||
throws IOException, CheckSumWrongException {
|
throws IOException, CheckSumWrongException {
|
||||||
|
|
||||||
FSTEntry entry = getNUSTitle().getFSTEntryByFullPath(entryFullPath);
|
FSTEntry entry = getNUSTitle().getFSTEntryByFullPath(entryFullPath);
|
||||||
if (entry == null) {
|
if (entry == null) {
|
||||||
log.info("File not found");
|
log.info("File not found");
|
||||||
return;
|
CompletableFuture.completedFuture(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
decryptFSTEntryTo(fullPath, entry, outputFolder, skipExistingFiles);
|
decryptFSTEntryToSync(fullPath, entry, outputFolder, skipExistingFiles);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptFSTEntryTo(FSTEntry entry, String outputFolder) throws IOException, CheckSumWrongException {
|
public void decryptFSTEntryTo(FSTEntry entry, String outputFolder) throws IOException, CheckSumWrongException {
|
||||||
decryptFSTEntryTo(false, entry, outputFolder);
|
decryptFSTEntryTo(false, entry, outputFolder);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptFSTEntryTo(boolean fullPath, FSTEntry entry, String outputFolder) throws IOException, CheckSumWrongException {
|
|
||||||
decryptFSTEntryTo(fullPath, entry, outputFolder, getNUSTitle().isSkipExistingFiles());
|
|
||||||
}
|
|
||||||
|
|
||||||
public void decryptFSTEntryTo(FSTEntry entry, String outputFolder, boolean skipExistingFiles) throws IOException, CheckSumWrongException {
|
public void decryptFSTEntryTo(FSTEntry entry, String outputFolder, boolean skipExistingFiles) throws IOException, CheckSumWrongException {
|
||||||
decryptFSTEntryTo(false, entry, outputFolder, getNUSTitle().isSkipExistingFiles());
|
decryptFSTEntryToSync(false, entry, outputFolder, getNUSTitle().isSkipExistingFiles());
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
public void decryptFSTEntryTo(boolean fullPath, FSTEntry entry, String outputFolder) throws IOException, CheckSumWrongException {
|
||||||
* public void decryptFSTEntryTo(boolean fullPath, FSTEntry entry,String outputFolder, boolean skipExistingFiles) throws IOException{
|
decryptFSTEntryToSync(fullPath, entry, outputFolder, getNUSTitle().isSkipExistingFiles());
|
||||||
* decryptFSTEntry(fullPath,entry,outputFolder,skipExistingFiles); }
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
// Decrypt list of FSTEntry to Files
|
// Decrypt list of FSTEntry to Files
|
||||||
@ -350,10 +384,25 @@ public final class DecryptionService {
|
|||||||
decryptFSTEntryListTo(true, list, outputFolder);
|
decryptFSTEntryListTo(true, list, outputFolder);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public CompletableFuture<Void> decryptFSTEntryListToAsync(boolean fullPath, List<FSTEntry> list, String outputFolder)
|
||||||
|
throws IOException, CheckSumWrongException {
|
||||||
|
return CompletableFuture.allOf(list.stream().map(entry -> decryptFSTEntryToAsync(fullPath, entry, outputFolder, getNUSTitle().isSkipExistingFiles()))
|
||||||
|
.toArray(CompletableFuture[]::new));
|
||||||
|
}
|
||||||
|
|
||||||
public void decryptFSTEntryListTo(boolean fullPath, List<FSTEntry> list, String outputFolder) throws IOException, CheckSumWrongException {
|
public void decryptFSTEntryListTo(boolean fullPath, List<FSTEntry> list, String outputFolder) throws IOException, CheckSumWrongException {
|
||||||
for (FSTEntry entry : list) {
|
if (parallelizable && Settings.ALLOW_PARALLELISATION) {
|
||||||
decryptFSTEntryTo(fullPath, entry, outputFolder, getNUSTitle().isSkipExistingFiles());
|
try {
|
||||||
|
decryptFSTEntryListToAsync(fullPath, list, outputFolder).get();
|
||||||
|
} catch (InterruptedException | ExecutionException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (val entry : list) {
|
||||||
|
decryptFSTEntryToSync(fullPath, entry, outputFolder, getNUSTitle().isSkipExistingFiles());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||||
@ -372,9 +421,25 @@ public final class DecryptionService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void decryptPlainContents(List<Content> list, String outputFolder) throws IOException, CheckSumWrongException {
|
public void decryptPlainContents(List<Content> list, String outputFolder) throws IOException, CheckSumWrongException {
|
||||||
for (Content c : list) {
|
|
||||||
decryptContentTo(c, outputFolder, getNUSTitle().isSkipExistingFiles());
|
if (parallelizable && Settings.ALLOW_PARALLELISATION) {
|
||||||
|
try {
|
||||||
|
decryptPlainContentsAsync(list, outputFolder).get();
|
||||||
|
} catch (InterruptedException | ExecutionException e) {
|
||||||
|
// wrap it.
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (val c : list) {
|
||||||
|
decryptContentToSync(c, outputFolder, getNUSTitle().isSkipExistingFiles());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public CompletableFuture<Void> decryptPlainContentsAsync(List<Content> list, String outputFolder) throws IOException, CheckSumWrongException {
|
||||||
|
return CompletableFuture
|
||||||
|
.allOf(list.stream().map(c -> decryptContentToAsync(c, outputFolder, getNUSTitle().isSkipExistingFiles())).toArray(CompletableFuture[]::new));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void decryptAllPlainContents(String outputFolder) throws IOException, CheckSumWrongException {
|
public void decryptAllPlainContents(String outputFolder) throws IOException, CheckSumWrongException {
|
||||||
|
Loading…
Reference in New Issue
Block a user