diff --git a/brut.apktool/apktool-cli/src/main/java/brut/apktool/Main.java b/brut.apktool/apktool-cli/src/main/java/brut/apktool/Main.java index 208db76067..b67ae53d47 100644 --- a/brut.apktool/apktool-cli/src/main/java/brut/apktool/Main.java +++ b/brut.apktool/apktool-cli/src/main/java/brut/apktool/Main.java @@ -29,8 +29,7 @@ import brut.util.OSDetection; import org.apache.commons.cli.*; -import java.io.File; -import java.io.IOException; +import java.io.*; import java.util.logging.*; /** diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/AaptInvoker.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/AaptInvoker.java index 3720829fbb..4bd891d987 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/AaptInvoker.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/AaptInvoker.java @@ -22,13 +22,8 @@ import brut.util.AaptManager; import brut.util.OS; -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.util.ArrayList; -import java.util.List; -import java.util.Objects; +import java.io.*; +import java.util.*; import java.util.logging.Logger; public class AaptInvoker { @@ -81,8 +76,7 @@ private File createDoNotCompressExtensionsFile(ApkInfo apkInfo) throws AndrolibE } private void invokeAapt2(File apkFile, File manifest, File resDir, File rawDir, File assetDir, File[] include, - List cmd, boolean customAapt) - throws AndrolibException { + List cmd, boolean customAapt) throws AndrolibException { List compileCommand = new ArrayList<>(cmd); File resourcesZip = null; @@ -249,8 +243,7 @@ private void invokeAapt2(File apkFile, File manifest, File resDir, File rawDir, } private void invokeAapt1(File apkFile, File manifest, File resDir, File rawDir, File assetDir, File[] include, - List cmd, boolean customAapt) - throws AndrolibException { + List cmd, boolean customAapt) throws AndrolibException { cmd.add("p"); @@ -365,7 +358,7 @@ private void invokeAapt1(File apkFile, File manifest, File resDir, File rawDir, } public void invokeAapt(File apkFile, File manifest, File resDir, File rawDir, File assetDir, File[] include) - throws AndrolibException { + throws AndrolibException { String aaptPath = mConfig.aaptPath; boolean customAapt = !aaptPath.isEmpty(); diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkBuilder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkBuilder.java index aee8ccb8ff..a7c0c16450 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkBuilder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkBuilder.java @@ -52,9 +52,9 @@ public class ApkBuilder { private final static Logger LOGGER = Logger.getLogger(ApkBuilder.class.getName()); private final Config mConfig; + private final ExtFile mApkDir; private ApkInfo mApkInfo; private int mMinSdkVersion = 0; - private final ExtFile mApkDir; private final static String APK_DIRNAME = "build/apk"; private final static String UNK_DIRNAME = "unknown"; @@ -87,7 +87,7 @@ public void build(File outFile) throws BrutException { } if (outFile == null) { - String outFileName = mApkInfo.getApkFileName(); + String outFileName = mApkInfo.apkFileName; outFile = new File(mApkDir, "dist" + File.separator + (outFileName == null ? "out.apk" : outFileName)); } @@ -96,17 +96,17 @@ public void build(File outFile) throws BrutException { File manifest = new File(mApkDir, "AndroidManifest.xml"); File manifestOriginal = new File(mApkDir, "AndroidManifest.xml.orig"); - buildSources(mApkDir); - buildNonDefaultSources(mApkDir); - buildManifestFile(mApkDir, manifest, manifestOriginal); - buildResources(mApkDir, mApkInfo.usesFramework); - buildLibs(mApkDir); - buildCopyOriginalFiles(mApkDir); - buildApk(mApkDir, outFile); + buildSources(); + buildNonDefaultSources(); + buildManifestFile(manifest, manifestOriginal); + buildResources(); + buildLibs(); + buildCopyOriginalFiles(); + buildApk(outFile); // we must go after the Apk is built, and copy the files in via Zip // this is because Aapt won't add files it doesn't know (ex unknown files) - buildUnknownFiles(mApkDir, outFile, mApkInfo); + buildUnknownFiles(outFile); // we copied the AndroidManifest.xml to AndroidManifest.xml.orig so we can edit it // lets restore the unedited one, to not change the original @@ -122,11 +122,9 @@ public void build(File outFile) throws BrutException { LOGGER.info("Built apk into: " + outFile.getPath()); } - private void buildManifestFile(File appDir, File manifest, File manifestOriginal) - throws AndrolibException { - + private void buildManifestFile(File manifest, File manifestOriginal) throws AndrolibException { // If we decoded in "raw", we cannot patch AndroidManifest - if (new File(appDir, "resources.arsc").exists()) { + if (new File(mApkDir, "resources.arsc").exists()) { return; } if (manifest.isFile() && manifest.exists()) { @@ -143,36 +141,34 @@ private void buildManifestFile(File appDir, File manifest, File manifestOriginal } } - private void buildSources(File appDir) - throws AndrolibException { - if (!buildSourcesRaw(appDir, "classes.dex") && !buildSourcesSmali(appDir, "smali", "classes.dex")) { + private void buildSources() throws AndrolibException { + if (!buildSourcesRaw("classes.dex") && !buildSourcesSmali("smali", "classes.dex")) { LOGGER.warning("Could not find sources"); } } - private void buildNonDefaultSources(ExtFile appDir) - throws AndrolibException { + private void buildNonDefaultSources() throws AndrolibException { try { // loop through any smali_ directories for multi-dex apks - Map dirs = appDir.getDirectory().getDirs(); + Map dirs = mApkDir.getDirectory().getDirs(); for (Map.Entry directory : dirs.entrySet()) { String name = directory.getKey(); if (name.startsWith("smali_")) { String filename = name.substring(name.indexOf("_") + 1) + ".dex"; - if (!buildSourcesRaw(appDir, filename) && !buildSourcesSmali(appDir, name, filename)) { + if (!buildSourcesRaw(filename) && !buildSourcesSmali(name, filename)) { LOGGER.warning("Could not find sources"); } } } // loop through any classes#.dex files for multi-dex apks - File[] dexFiles = appDir.listFiles(); + File[] dexFiles = mApkDir.listFiles(); if (dexFiles != null) { for (File dex : dexFiles) { // skip classes.dex because we have handled it in buildSources() - if (dex.getName().endsWith(".dex") && ! dex.getName().equalsIgnoreCase("classes.dex")) { - buildSourcesRaw(appDir, dex.getName()); + if (dex.getName().endsWith(".dex") && !dex.getName().equalsIgnoreCase("classes.dex")) { + buildSourcesRaw(dex.getName()); } } } @@ -181,15 +177,14 @@ private void buildNonDefaultSources(ExtFile appDir) } } - private boolean buildSourcesRaw(File appDir, String filename) - throws AndrolibException { - File working = new File(appDir, filename); + private boolean buildSourcesRaw(String filename) throws AndrolibException { + File working = new File(mApkDir, filename); if (!working.exists()) { return false; } - File stored = new File(appDir, APK_DIRNAME + "/" + filename); + File stored = new File(mApkDir, APK_DIRNAME + "/" + filename); if (mConfig.forceBuildAll || isModified(working, stored)) { - LOGGER.info("Copying " + appDir.toString() + " " + filename + " file..."); + LOGGER.info("Copying " + mApkDir.toString() + " " + filename + " file..."); try { BrutIO.copyAndClose(Files.newInputStream(working.toPath()), Files.newOutputStream(stored.toPath())); return true; @@ -200,14 +195,13 @@ private boolean buildSourcesRaw(File appDir, String filename) return true; } - private boolean buildSourcesSmali(File appDir, String folder, String filename) - throws AndrolibException { - ExtFile smaliDir = new ExtFile(appDir, folder); + private boolean buildSourcesSmali(String folder, String filename) throws AndrolibException { + ExtFile smaliDir = new ExtFile(mApkDir, folder); if (!smaliDir.exists()) { return false; } - File dex = new File(appDir, APK_DIRNAME + "/" + filename); - if (! mConfig.forceBuildAll) { + File dex = new File(mApkDir, APK_DIRNAME + "/" + filename); + if (!mConfig.forceBuildAll) { LOGGER.info("Checking whether sources has changed..."); } if (mConfig.forceBuildAll || isModified(smaliDir, dex)) { @@ -219,28 +213,25 @@ private boolean buildSourcesSmali(File appDir, String folder, String filename) return true; } - private void buildResources(ExtFile appDir, UsesFramework usesFramework) - throws BrutException { - if (!buildResourcesRaw(appDir) && !buildResourcesFull(appDir, usesFramework) - && !buildManifest(appDir, usesFramework)) { + private void buildResources() throws BrutException { + if (!buildResourcesRaw() && !buildResourcesFull() && !buildManifest()) { LOGGER.warning("Could not find resources"); } } - private boolean buildResourcesRaw(ExtFile appDir) - throws AndrolibException { + private boolean buildResourcesRaw() throws AndrolibException { try { - if (!new File(appDir, "resources.arsc").exists()) { + if (!new File(mApkDir, "resources.arsc").exists()) { return false; } - File apkDir = new File(appDir, APK_DIRNAME); - if (! mConfig.forceBuildAll) { + File apkDir = new File(mApkDir, APK_DIRNAME); + if (!mConfig.forceBuildAll) { LOGGER.info("Checking whether resources has changed..."); } - if (mConfig.forceBuildAll || isModified(newFiles(APK_RESOURCES_FILENAMES, appDir), + if (mConfig.forceBuildAll || isModified(newFiles(APK_RESOURCES_FILENAMES, mApkDir), newFiles(APK_RESOURCES_FILENAMES, apkDir))) { LOGGER.info("Copying raw resources..."); - appDir.getDirectory().copyToDir(apkDir, APK_RESOURCES_FILENAMES); + mApkDir.getDirectory().copyToDir(apkDir, APK_RESOURCES_FILENAMES); } return true; } catch (DirectoryException ex) { @@ -248,46 +239,45 @@ private boolean buildResourcesRaw(ExtFile appDir) } } - private boolean buildResourcesFull(File appDir, UsesFramework usesFramework) - throws AndrolibException { + private boolean buildResourcesFull() throws AndrolibException { try { - if (!new File(appDir, "res").exists()) { + if (!new File(mApkDir, "res").exists()) { return false; } - if (! mConfig.forceBuildAll) { + if (!mConfig.forceBuildAll) { LOGGER.info("Checking whether resources has changed..."); } - File apkDir = new File(appDir, APK_DIRNAME); + File apkDir = new File(mApkDir, APK_DIRNAME); File resourceFile = new File(apkDir.getParent(), "resources.zip"); - if (mConfig.forceBuildAll || isModified(newFiles(APP_RESOURCES_FILENAMES, appDir), + if (mConfig.forceBuildAll || isModified(newFiles(APP_RESOURCES_FILENAMES, mApkDir), newFiles(APK_RESOURCES_FILENAMES, apkDir)) || (mConfig.isAapt2() && !isFile(resourceFile))) { LOGGER.info("Building resources..."); if (mConfig.debugMode) { if (mConfig.isAapt2()) { LOGGER.info("Using aapt2 - setting 'debuggable' attribute to 'true' in AndroidManifest.xml"); - ResXmlPatcher.setApplicationDebugTagTrue(new File(appDir, "AndroidManifest.xml")); + ResXmlPatcher.setApplicationDebugTagTrue(new File(mApkDir, "AndroidManifest.xml")); } else { - ResXmlPatcher.removeApplicationDebugTag(new File(appDir, "AndroidManifest.xml")); + ResXmlPatcher.removeApplicationDebugTag(new File(mApkDir, "AndroidManifest.xml")); } } if (mConfig.netSecConf) { - ApkInfo meta = ApkInfo.load(new ExtFile(appDir)); + ApkInfo meta = ApkInfo.load(new ExtFile(mApkDir)); if (meta.getSdkInfo() != null && meta.getSdkInfo().get("targetSdkVersion") != null) { if (Integer.parseInt(meta.getSdkInfo().get("targetSdkVersion")) < ResConfigFlags.SDK_NOUGAT) { LOGGER.warning("Target SDK version is lower than 24! Network Security Configuration might be ignored!"); } } - File netSecConfOrig = new File(appDir, "res/xml/network_security_config.xml"); + File netSecConfOrig = new File(mApkDir, "res/xml/network_security_config.xml"); if (netSecConfOrig.exists()) { LOGGER.info("Replacing existing network_security_config.xml!"); //noinspection ResultOfMethodCallIgnored netSecConfOrig.delete(); } ResXmlPatcher.modNetworkSecurityConfig(netSecConfOrig); - ResXmlPatcher.setNetworkSecurityConfig(new File(appDir, "AndroidManifest.xml")); + ResXmlPatcher.setNetworkSecurityConfig(new File(mApkDir, "AndroidManifest.xml")); LOGGER.info("Added permissive network security config in manifest"); } @@ -297,14 +287,13 @@ private boolean buildResourcesFull(File appDir, UsesFramework usesFramework) //noinspection ResultOfMethodCallIgnored resourceFile.delete(); - File ninePatch = new File(appDir, "9patch"); + File ninePatch = new File(mApkDir, "9patch"); if (!ninePatch.exists()) { ninePatch = null; } AaptInvoker invoker = new AaptInvoker(mConfig, mApkInfo); - invoker.invokeAapt(apkFile, new File(appDir, - "AndroidManifest.xml"), new File(appDir, "res"), - ninePatch, null, parseUsesFramework(usesFramework)); + invoker.invokeAapt(apkFile, new File(mApkDir, "AndroidManifest.xml"), + new File(mApkDir, "res"), ninePatch, null, getIncludeFiles()); ExtFile tmpExtFile = new ExtFile(apkFile); Directory tmpDir = tmpExtFile.getDirectory(); @@ -332,31 +321,29 @@ private boolean buildResourcesFull(File appDir, UsesFramework usesFramework) } } - private boolean buildManifestRaw(ExtFile appDir) - throws AndrolibException { + private boolean buildManifestRaw() throws AndrolibException { try { - File apkDir = new File(appDir, APK_DIRNAME); + File apkDir = new File(mApkDir, APK_DIRNAME); LOGGER.info("Copying raw AndroidManifest.xml..."); - appDir.getDirectory().copyToDir(apkDir, APK_MANIFEST_FILENAMES); + mApkDir.getDirectory().copyToDir(apkDir, APK_MANIFEST_FILENAMES); return true; } catch (DirectoryException ex) { throw new AndrolibException(ex); } } - private boolean buildManifest(ExtFile appDir, UsesFramework usesFramework) - throws BrutException { + private boolean buildManifest() throws BrutException { try { - if (!new File(appDir, "AndroidManifest.xml").exists()) { + if (!new File(mApkDir, "AndroidManifest.xml").exists()) { return false; } - if (! mConfig.forceBuildAll) { + if (!mConfig.forceBuildAll) { LOGGER.info("Checking whether resources has changed..."); } - File apkDir = new File(appDir, APK_DIRNAME); + File apkDir = new File(mApkDir, APK_DIRNAME); - if (mConfig.forceBuildAll || isModified(newFiles(APK_MANIFEST_FILENAMES, appDir), + if (mConfig.forceBuildAll || isModified(newFiles(APK_MANIFEST_FILENAMES, mApkDir), newFiles(APK_MANIFEST_FILENAMES, apkDir))) { LOGGER.info("Building AndroidManifest.xml..."); @@ -364,15 +351,14 @@ private boolean buildManifest(ExtFile appDir, UsesFramework usesFramework) //noinspection ResultOfMethodCallIgnored apkFile.delete(); - File ninePatch = new File(appDir, "9patch"); + File ninePatch = new File(mApkDir, "9patch"); if (!ninePatch.exists()) { ninePatch = null; } AaptInvoker invoker = new AaptInvoker(mConfig, mApkInfo); - invoker.invokeAapt(apkFile, new File(appDir, - "AndroidManifest.xml"), null, ninePatch, null, - parseUsesFramework(usesFramework)); + invoker.invokeAapt(apkFile, new File(mApkDir, "AndroidManifest.xml"), + null, ninePatch, null, getIncludeFiles()); Directory tmpDir = new ExtFile(apkFile).getDirectory(); tmpDir.copyToDir(apkDir, APK_MANIFEST_FILENAMES); @@ -385,25 +371,25 @@ private boolean buildManifest(ExtFile appDir, UsesFramework usesFramework) throw new AndrolibException(ex); } catch (AndrolibException ex) { LOGGER.warning("Parse AndroidManifest.xml failed, treat it as raw file."); - return buildManifestRaw(appDir); + return buildManifestRaw(); } } - private void buildLibs(File appDir) throws AndrolibException { - buildLibrary(appDir, "lib"); - buildLibrary(appDir, "libs"); - buildLibrary(appDir, "kotlin"); - buildLibrary(appDir, "META-INF/services"); + private void buildLibs() throws AndrolibException { + buildLibrary("lib"); + buildLibrary("libs"); + buildLibrary("kotlin"); + buildLibrary("META-INF/services"); } - private void buildLibrary(File appDir, String folder) throws AndrolibException { - File working = new File(appDir, folder); + private void buildLibrary(String folder) throws AndrolibException { + File working = new File(mApkDir, folder); - if (! working.exists()) { + if (!working.exists()) { return; } - File stored = new File(appDir, APK_DIRNAME + "/" + folder); + File stored = new File(mApkDir, APK_DIRNAME + "/" + folder); if (mConfig.forceBuildAll || isModified(working, stored)) { LOGGER.info("Copying libs... (/" + folder + ")"); try { @@ -415,25 +401,24 @@ private void buildLibrary(File appDir, String folder) throws AndrolibException { } } - private void buildCopyOriginalFiles(File appDir) - throws AndrolibException { + private void buildCopyOriginalFiles() throws AndrolibException { if (mConfig.copyOriginalFiles) { - File originalDir = new File(appDir, "original"); + File originalDir = new File(mApkDir, "original"); if (originalDir.exists()) { try { LOGGER.info("Copy original files..."); Directory in = (new ExtFile(originalDir)).getDirectory(); if (in.containsFile("AndroidManifest.xml")) { LOGGER.info("Copy AndroidManifest.xml..."); - in.copyToDir(new File(appDir, APK_DIRNAME), "AndroidManifest.xml"); + in.copyToDir(new File(mApkDir, APK_DIRNAME), "AndroidManifest.xml"); } if (in.containsFile("stamp-cert-sha256")) { LOGGER.info("Copy stamp-cert-sha256..."); - in.copyToDir(new File(appDir, APK_DIRNAME), "stamp-cert-sha256"); + in.copyToDir(new File(mApkDir, APK_DIRNAME), "stamp-cert-sha256"); } if (in.containsDir("META-INF")) { LOGGER.info("Copy META-INF..."); - in.copyToDir(new File(appDir, APK_DIRNAME), "META-INF"); + in.copyToDir(new File(mApkDir, APK_DIRNAME), "META-INF"); } } catch (DirectoryException ex) { throw new AndrolibException(ex); @@ -442,12 +427,11 @@ private void buildCopyOriginalFiles(File appDir) } } - private void buildUnknownFiles(File appDir, File outFile, ApkInfo meta) - throws AndrolibException { - if (meta.unknownFiles != null) { + private void buildUnknownFiles(File outFile) throws AndrolibException { + if (mApkInfo.unknownFiles != null) { LOGGER.info("Copying unknown files/dir..."); - Map files = meta.unknownFiles; + Map files = mApkInfo.unknownFiles; File tempFile = new File(outFile.getParent(), outFile.getName() + ".apktool_temp"); boolean renamed = outFile.renameTo(tempFile); if (!renamed) { @@ -455,11 +439,11 @@ private void buildUnknownFiles(File appDir, File outFile, ApkInfo meta) } try ( - ZipFile inputFile = new ZipFile(tempFile); - ZipOutputStream actualOutput = new ZipOutputStream(Files.newOutputStream(outFile.toPath())) + ZipFile inputFile = new ZipFile(tempFile); + ZipOutputStream actualOutput = new ZipOutputStream(Files.newOutputStream(outFile.toPath())) ) { copyExistingFiles(inputFile, actualOutput); - copyUnknownFiles(appDir, actualOutput, files); + copyUnknownFiles(actualOutput, files); } catch (IOException | BrutException ex) { throw new AndrolibException(ex); } @@ -481,7 +465,7 @@ private void copyExistingFiles(ZipFile inputFile, ZipOutputStream outputFile) th outputFile.putNextEntry(entry); // No need to create directory entries in the final apk - if (! entry.isDirectory()) { + if (!entry.isDirectory()) { BrutIO.copy(inputFile, outputFile, entry); } @@ -489,9 +473,9 @@ private void copyExistingFiles(ZipFile inputFile, ZipOutputStream outputFile) th } } - private void copyUnknownFiles(File appDir, ZipOutputStream outputFile, Map files) + private void copyUnknownFiles(ZipOutputStream outputFile, Map files) throws BrutException, IOException { - File unknownFileDir = new File(appDir, UNK_DIRNAME); + File unknownFileDir = new File(mApkDir, UNK_DIRNAME); // loop through unknown files for (Map.Entry unknownFileInfo : files.entrySet()) { @@ -529,7 +513,7 @@ private void copyUnknownFiles(File appDir, ZipOutputStream outputFile, Map BrutIO .recursiveModifiedTime(stored); + return !stored.exists() || BrutIO.recursiveModifiedTime(working) > BrutIO .recursiveModifiedTime(stored); } private boolean isFile(File working) { @@ -604,17 +586,15 @@ private File[] newFiles(String[] names, File dir) { return files; } - public boolean detectWhetherAppIsFramework(File appDir) - throws AndrolibException { - File publicXml = new File(appDir, "res/values/public.xml"); - if (! publicXml.exists()) { + public boolean detectWhetherAppIsFramework() throws AndrolibException { + File publicXml = new File(mApkDir, "res/values/public.xml"); + if (!publicXml.exists()) { return false; } Iterator it; try { - it = IOUtils.lineIterator(new FileReader(new File(appDir, - "res/values/public.xml"))); + it = IOUtils.lineIterator(new FileReader(new File(mApkDir, "res/values/public.xml"))); } catch (FileNotFoundException ex) { throw new AndrolibException( "Could not detect whether app is framework one", ex); diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkDecoder.java index 1d9417de68..8019aa8ff7 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/ApkDecoder.java @@ -32,8 +32,7 @@ import com.android.tools.smali.dexlib2.iface.DexFile; import org.apache.commons.io.FilenameUtils; -import java.io.File; -import java.io.IOException; +import java.io.*; import java.util.*; import java.util.logging.Logger; import java.util.regex.Pattern; @@ -44,6 +43,7 @@ public class ApkDecoder { private final Config mConfig; private final ExtFile mApkFile; protected final ResUnknownFiles mResUnknownFiles; + private ApkInfo mApkInfo; private int mMinSdkVersion = 0; private final static String SMALI_DIRNAME = "smali"; @@ -59,22 +59,22 @@ public class ApkDecoder { "jpg|jpeg|png|gif|wav|mp2|mp3|ogg|aac|mpg|mpeg|mid|midi|smf|jet|rtttl|imy|xmf|mp4|" + "m4a|m4v|3gp|3gpp|3g2|3gpp2|amr|awb|wma|wmv|webm|webp|mkv)$"); + public ApkDecoder(File apkFile) { + this(Config.getDefaultConfig(), new ExtFile(apkFile)); + } + public ApkDecoder(ExtFile apkFile) { this(Config.getDefaultConfig(), apkFile); } + public ApkDecoder(Config config, File apkFile) { + this(config, new ExtFile(apkFile)); + } + public ApkDecoder(Config config, ExtFile apkFile) { mConfig = config; - mResUnknownFiles = new ResUnknownFiles(); mApkFile = apkFile; - } - - public ApkDecoder(File apkFile) { - this(new ExtFile(apkFile)); - } - - public ApkDecoder(Config config, File apkFile) { - this(config, new ExtFile(apkFile)); + mResUnknownFiles = new ResUnknownFiles(); } public ApkInfo decode(File outDir) throws AndrolibException, IOException, DirectoryException { @@ -96,8 +96,11 @@ public ApkInfo decode(File outDir) throws AndrolibException, IOException, Direct LOGGER.info("Using Apktool " + ApktoolProperties.getVersion() + " on " + mApkFile.getName()); - ResourcesDecoder resourcesDecoder = new ResourcesDecoder(mConfig, mApkFile); - if (hasResources()) { + mApkInfo = new ApkInfo(mApkFile); + + ResourcesDecoder resourcesDecoder = new ResourcesDecoder(mConfig, mApkFile, mApkInfo); + + if (mApkInfo.hasResources()) { switch (mConfig.decodeResources) { case Config.DECODE_RESOURCES_NONE: copyResourcesRaw(outDir); @@ -108,7 +111,7 @@ public ApkInfo decode(File outDir) throws AndrolibException, IOException, Direct } } - if (hasManifest()) { + if (mApkInfo.hasManifest()) { if (mConfig.decodeResources == Config.DECODE_RESOURCES_FULL || mConfig.forceDecodeManifest == Config.FORCE_DECODE_MANIFEST_FULL) { resourcesDecoder.decodeManifest(outDir); @@ -119,7 +122,7 @@ public ApkInfo decode(File outDir) throws AndrolibException, IOException, Direct } resourcesDecoder.updateApkInfo(outDir); - if (hasSources()) { + if (mApkInfo.hasSources()) { switch (mConfig.decodeSources) { case Config.DECODE_SOURCES_NONE: copySourcesRaw(outDir, "classes.dex"); @@ -131,12 +134,12 @@ public ApkInfo decode(File outDir) throws AndrolibException, IOException, Direct } } - if (hasMultipleSources()) { + if (mApkInfo.hasMultipleSources()) { // foreach unknown dex file in root, lets disassemble it Set files = mApkFile.getDirectory().getFiles(true); for (String file : files) { if (file.endsWith(".dex")) { - if (! file.equalsIgnoreCase("classes.dex")) { + if (!file.equalsIgnoreCase("classes.dex")) { switch(mConfig.decodeSources) { case Config.DECODE_SOURCES_NONE: copySourcesRaw(outDir, file); @@ -158,19 +161,17 @@ public ApkInfo decode(File outDir) throws AndrolibException, IOException, Direct } // In case we have no resources. We should store the minSdk we pulled from the source opcode api level - ApkInfo apkInfo = resourcesDecoder.getApkInfo(); - if (!hasResources() && mMinSdkVersion > 0) { - apkInfo.setSdkInfoField("minSdkVersion", Integer.toString(mMinSdkVersion)); + if (!mApkInfo.hasResources() && mMinSdkVersion > 0) { + mApkInfo.setSdkInfoField("minSdkVersion", Integer.toString(mMinSdkVersion)); } copyRawFiles(outDir); - copyUnknownFiles(apkInfo, outDir); - List mUncompressedFiles = new ArrayList<>(); - recordUncompressedFiles(apkInfo, resourcesDecoder.getResFileMapping(), mUncompressedFiles); + copyUnknownFiles(outDir); + recordUncompressedFiles(resourcesDecoder.getResFileMapping()); copyOriginalFiles(outDir); - writeApkInfo(apkInfo, outDir); + writeApkInfo(outDir); - return apkInfo; + return mApkInfo; } finally { try { mApkFile.close(); @@ -178,53 +179,11 @@ public ApkInfo decode(File outDir) throws AndrolibException, IOException, Direct } } - private boolean hasManifest() throws AndrolibException { - try { - return mApkFile.getDirectory().containsFile("AndroidManifest.xml"); - } catch (DirectoryException ex) { - throw new AndrolibException(ex); - } - } - - private boolean hasResources() throws AndrolibException { - try { - return mApkFile.getDirectory().containsFile("resources.arsc"); - } catch (DirectoryException ex) { - throw new AndrolibException(ex); - } - } - - private boolean hasSources() throws AndrolibException { - try { - return mApkFile.getDirectory().containsFile("classes.dex"); - } catch (DirectoryException ex) { - throw new AndrolibException(ex); - } - } - - private boolean hasMultipleSources() throws AndrolibException { - try { - Set files = mApkFile.getDirectory().getFiles(false); - for (String file : files) { - if (file.endsWith(".dex")) { - if (! file.equalsIgnoreCase("classes.dex")) { - return true; - } - } - } - - return false; - } catch (DirectoryException ex) { - throw new AndrolibException(ex); - } + private void writeApkInfo(File outDir) throws AndrolibException { + mApkInfo.save(new File(outDir, "apktool.yml")); } - private void writeApkInfo(ApkInfo apkInfo, File outDir) throws AndrolibException { - apkInfo.save(new File(outDir, "apktool.yml")); - } - - private void copyManifestRaw(File outDir) - throws AndrolibException { + private void copyManifestRaw(File outDir) throws AndrolibException { try { LOGGER.info("Copying raw manifest..."); mApkFile.getDirectory().copyToDir(outDir, APK_MANIFEST_FILENAMES); @@ -233,8 +192,7 @@ private void copyManifestRaw(File outDir) } } - private void copyResourcesRaw(File outDir) - throws AndrolibException { + private void copyResourcesRaw(File outDir) throws AndrolibException { try { LOGGER.info("Copying raw resources..."); mApkFile.getDirectory().copyToDir(outDir, APK_RESOURCES_FILENAMES); @@ -243,8 +201,7 @@ private void copyResourcesRaw(File outDir) } } - private void copySourcesRaw(File outDir, String filename) - throws AndrolibException { + private void copySourcesRaw(File outDir, String filename) throws AndrolibException { try { LOGGER.info("Copying raw " + filename + " file..."); mApkFile.getDirectory().copyToDir(outDir, filename); @@ -253,8 +210,7 @@ private void copySourcesRaw(File outDir, String filename) } } - private void decodeSourcesSmali(File outDir, String filename) - throws AndrolibException { + private void decodeSourcesSmali(File outDir, String filename) throws AndrolibException { try { File smaliDir; if (filename.equalsIgnoreCase("classes.dex")) { @@ -277,8 +233,7 @@ private void decodeSourcesSmali(File outDir, String filename) } } - private void copyRawFiles(File outDir) - throws AndrolibException { + private void copyRawFiles(File outDir) throws AndrolibException { LOGGER.info("Copying assets and libs..."); try { Directory in = mApkFile.getDirectory(); @@ -311,8 +266,7 @@ private boolean isAPKFileNames(String file) { return false; } - private void copyUnknownFiles(ApkInfo apkInfo, File outDir) - throws AndrolibException { + private void copyUnknownFiles(File outDir) throws AndrolibException { LOGGER.info("Copying unknown files..."); File unknownOut = new File(outDir, UNK_DIRNAME); try { @@ -331,14 +285,13 @@ private void copyUnknownFiles(ApkInfo apkInfo, File outDir) } } // update apk info - apkInfo.unknownFiles = mResUnknownFiles.getUnknownFiles(); + mApkInfo.unknownFiles = mResUnknownFiles.getUnknownFiles(); } catch (DirectoryException ex) { throw new AndrolibException(ex); } } - private void copyOriginalFiles(File outDir) - throws AndrolibException { + private void copyOriginalFiles(File outDir) throws AndrolibException { LOGGER.info("Copying original files..."); File originalDir = new File(outDir, "original"); if (!originalDir.exists()) { @@ -370,11 +323,9 @@ private void copyOriginalFiles(File outDir) } } - private void recordUncompressedFiles(ApkInfo apkInfo, - Map resFileMapping, - List uncompressedFilesOrExts) - throws AndrolibException { + private void recordUncompressedFiles(Map resFileMapping) throws AndrolibException { try { + List uncompressedFilesOrExts = new ArrayList<>(); Directory unk = mApkFile.getDirectory(); Set files = unk.getFiles(true); @@ -398,9 +349,8 @@ private void recordUncompressedFiles(ApkInfo apkInfo, } // update apk info if (!uncompressedFilesOrExts.isEmpty()) { - apkInfo.doNotCompress = uncompressedFilesOrExts; + mApkInfo.doNotCompress = uncompressedFilesOrExts; } - } catch (DirectoryException ex) { throw new AndrolibException(ex); } diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/apk/ApkInfo.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/apk/ApkInfo.java index 0f1b82d072..ed966b3161 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/apk/ApkInfo.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/apk/ApkInfo.java @@ -20,18 +20,17 @@ import brut.androlib.exceptions.AndrolibException; import brut.androlib.res.data.ResConfigFlags; import brut.directory.DirectoryException; +import brut.directory.ExtFile; import brut.directory.FileDirectory; import java.io.*; -import java.util.ArrayList; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; +import java.util.*; public class ApkInfo implements YamlSerializable { - public String version; + private transient ExtFile mApkFile; - private String apkFileName; + public String version; + public String apkFileName; public boolean isFrameworkApk; public UsesFramework usesFramework; private Map sdkInfo = new LinkedHashMap<>(); @@ -47,7 +46,78 @@ public class ApkInfo implements YamlSerializable { public boolean compressionType; public ApkInfo() { + this(null); + } + + public ApkInfo(ExtFile apkFile) { this.version = ApktoolProperties.getVersion(); + if (apkFile != null) { + setApkFile(apkFile); + } + } + + public ExtFile getApkFile() { + return mApkFile; + } + + public void setApkFile(ExtFile apkFile) { + mApkFile = apkFile; + if (this.apkFileName == null) { + this.apkFileName = apkFile.getName(); + } + } + + public boolean hasManifest() throws AndrolibException { + if (mApkFile == null) { + return false; + } + try { + return mApkFile.getDirectory().containsFile("AndroidManifest.xml"); + } catch (DirectoryException ex) { + throw new AndrolibException(ex); + } + } + + public boolean hasResources() throws AndrolibException { + if (mApkFile == null) { + return false; + } + try { + return mApkFile.getDirectory().containsFile("resources.arsc"); + } catch (DirectoryException ex) { + throw new AndrolibException(ex); + } + } + + public boolean hasSources() throws AndrolibException { + if (mApkFile == null) { + return false; + } + try { + return mApkFile.getDirectory().containsFile("classes.dex"); + } catch (DirectoryException ex) { + throw new AndrolibException(ex); + } + } + + public boolean hasMultipleSources() throws AndrolibException { + if (mApkFile == null) { + return false; + } + try { + Set files = mApkFile.getDirectory().getFiles(false); + for (String file : files) { + if (file.endsWith(".dex")) { + if (!file.equalsIgnoreCase("classes.dex")) { + return true; + } + } + } + + return false; + } catch (DirectoryException ex) { + throw new AndrolibException(ex); + } } public String checkTargetSdkVersionBounds() { @@ -61,14 +131,6 @@ public String checkTargetSdkVersionBounds() { return Integer.toString(target); } - public String getApkFileName() { - return apkFileName; - } - - public void setApkFileName(String apkFileName) { - this.apkFileName = apkFileName; - } - public Map getSdkInfo() { return sdkInfo; } @@ -134,9 +196,7 @@ private int mapSdkShorthandToVersion(String sdkVersion) { } public void save(File file) throws AndrolibException { - try ( - YamlWriter writer = new YamlWriter(new FileOutputStream(file)); - ) { + try (YamlWriter writer = new YamlWriter(new FileOutputStream(file))) { write(writer); } catch (FileNotFoundException e) { throw new AndrolibException("File not found"); @@ -153,10 +213,10 @@ public static ApkInfo load(InputStream is) throws AndrolibException { } public static ApkInfo load(File appDir) throws AndrolibException { - try( - InputStream in = new FileDirectory(appDir).getFileInput("apktool.yml"); - ) { - return ApkInfo.load(in); + try (InputStream in = new FileDirectory(appDir).getFileInput("apktool.yml")) { + ApkInfo apkInfo = ApkInfo.load(in); + apkInfo.setApkFile(new ExtFile(appDir)); + return apkInfo; } catch (DirectoryException | IOException ex) { throw new AndrolibException(ex); } diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/Framework.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/Framework.java index 37379af2af..96d83faba7 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/Framework.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/Framework.java @@ -49,8 +49,7 @@ public void installFramework(File frameFile) throws AndrolibException { installFramework(frameFile, config.frameworkTag); } - public void installFramework(File frameFile, String tag) - throws AndrolibException { + public void installFramework(File frameFile, String tag) throws AndrolibException { InputStream in = null; ZipOutputStream out = null; try { @@ -198,8 +197,7 @@ public File getFrameworkDirectory() throws AndrolibException { return dir; } - public File getFrameworkApk(int id, String frameTag) - throws AndrolibException { + public File getFrameworkApk(int id, String frameTag) throws AndrolibException { File dir = getFrameworkDirectory(); File apk; @@ -216,8 +214,10 @@ public File getFrameworkApk(int id, String frameTag) } if (id == 1) { - try (InputStream in = getAndroidFrameworkResourcesAsStream(); - OutputStream out = Files.newOutputStream(apk.toPath())) { + try ( + InputStream in = getAndroidFrameworkResourcesAsStream(); + OutputStream out = Files.newOutputStream(apk.toPath()) + ) { IOUtils.copy(in, out); return apk; } catch (IOException ex) { diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/ResourcesDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/ResourcesDecoder.java index 77a8efb1c3..517c725631 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/ResourcesDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/ResourcesDecoder.java @@ -31,10 +31,7 @@ import brut.directory.FileDirectory; import org.xmlpull.v1.XmlSerializer; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; +import java.io.*; import java.util.*; import java.util.logging.Logger; @@ -43,93 +40,51 @@ public class ResourcesDecoder { private final Config mConfig; private final ExtFile mApkFile; - private final ResTable mResTable; private final ApkInfo mApkInfo; + private final ResTable mResTable; private final Map mResFileMapping = new HashMap<>(); private final static String[] IGNORED_PACKAGES = new String[] { "android", "com.htc", "com.lge", "com.lge.internal", "yi", "flyme", "air.com.adobe.appentry", "FFFFFFFFFFFFFFFFFFFFFF" }; - public ResourcesDecoder(Config config, ExtFile apkFile) { + public ResourcesDecoder(Config config, ExtFile apkFile, ApkInfo apkInfo) { mConfig = config; mApkFile = apkFile; - mApkInfo = new ApkInfo(); - mApkInfo.setApkFileName(apkFile.getName()); + mApkInfo = apkInfo; mResTable = new ResTable(mConfig, mApkInfo); } - public boolean hasManifest() throws AndrolibException { - try { - return mApkFile.getDirectory().containsFile("AndroidManifest.xml"); - } catch (DirectoryException ex) { - throw new AndrolibException(ex); - } - } - - public boolean hasResources() throws AndrolibException { - try { - return mApkFile.getDirectory().containsFile("resources.arsc"); - } catch (DirectoryException ex) { - throw new AndrolibException(ex); - } - } - public ResTable getResTable() throws AndrolibException { - if (! (hasManifest() || hasResources())) { + if (!mApkInfo.hasManifest() && !mApkInfo.hasResources()) { throw new AndrolibException( "Apk doesn't contain either AndroidManifest.xml file or resources.arsc file"); } return mResTable; } - public ApkInfo getApkInfo() { - return mApkInfo; - } - public Map getResFileMapping() { return mResFileMapping; } - public void decodeManifest(File outDir) throws AndrolibException { - if (hasManifest()) { - decodeManifest(getResTable(), mApkFile, outDir); - if (hasResources()) { - if (!mConfig.analysisMode) { - // Remove versionName / versionCode (aapt API 16) - // - // check for a mismatch between resources.arsc package and the package listed in AndroidManifest - // also remove the android::versionCode / versionName from manifest for rebuild - // this is a required change to prevent aapt warning about conflicting versions - // it will be passed as a parameter to aapt like "--min-sdk-version" via apktool.yml - adjustPackageManifest(getResTable(), outDir.getAbsolutePath() + File.separator + "AndroidManifest.xml"); - - ResXmlPatcher.removeManifestVersions(new File( - outDir.getAbsolutePath() + File.separator + "AndroidManifest.xml")); - - // update apk info - mApkInfo.packageInfo.forcedPackageId = String.valueOf(mResTable.getPackageId()); - } - } - } - } - - public void updateApkInfo(File outDir) throws AndrolibException { - mResTable.initApkInfo(mApkInfo, outDir); + public void loadMainPkg() throws AndrolibException { + mResTable.loadMainPkg(mApkFile); } - private void decodeManifest(ResTable resTable, ExtFile apkFile, File outDir) - throws AndrolibException { + public void decodeManifest(File outDir) throws AndrolibException { + if (!mApkInfo.hasManifest()) { + return; + } - AXmlResourceParser axmlParser = new AndroidManifestResourceParser(resTable); + AXmlResourceParser axmlParser = new AndroidManifestResourceParser(mResTable); XmlPullStreamDecoder fileDecoder = new XmlPullStreamDecoder(axmlParser, getResXmlSerializer()); Directory inApk, out; try { - inApk = apkFile.getDirectory(); + inApk = mApkFile.getDirectory(); out = new FileDirectory(outDir); - if (hasResources()) { + if (mApkInfo.hasResources()) { LOGGER.info("Decoding AndroidManifest.xml with resources..."); } else { LOGGER.info("Decoding AndroidManifest.xml with only framework resources..."); @@ -141,18 +96,38 @@ private void decodeManifest(ResTable resTable, ExtFile apkFile, File outDir) } catch (DirectoryException ex) { throw new AndrolibException(ex); } + + if (mApkInfo.hasResources()) { + if (!mConfig.analysisMode) { + // Remove versionName / versionCode (aapt API 16) + // + // check for a mismatch between resources.arsc package and the package listed in AndroidManifest + // also remove the android::versionCode / versionName from manifest for rebuild + // this is a required change to prevent aapt warning about conflicting versions + // it will be passed as a parameter to aapt like "--min-sdk-version" via apktool.yml + adjustPackageManifest(outDir.getAbsolutePath() + File.separator + "AndroidManifest.xml"); + + ResXmlPatcher.removeManifestVersions(new File( + outDir.getAbsolutePath() + File.separator + "AndroidManifest.xml")); + + // update apk info + mApkInfo.packageInfo.forcedPackageId = String.valueOf(mResTable.getPackageId()); + } + } } - private void adjustPackageManifest(ResTable resTable, String filePath) - throws AndrolibException { + public void updateApkInfo(File outDir) throws AndrolibException { + mResTable.initApkInfo(mApkInfo, outDir); + } + private void adjustPackageManifest(String filePath) throws AndrolibException { // compare resources.arsc package name to the one present in AndroidManifest - ResPackage resPackage = resTable.getCurrentResPackage(); + ResPackage resPackage = mResTable.getCurrentResPackage(); String pkgOriginal = resPackage.getName(); - String pkgRenamed = resTable.getPackageRenamed(); + String pkgRenamed = mResTable.getPackageRenamed(); - resTable.setPackageId(resPackage.getId()); - resTable.setPackageOriginal(pkgOriginal); + mResTable.setPackageId(resPackage.getId()); + mResTable.setPackageOriginal(pkgOriginal); // 1) Check if pkgOriginal is null (empty resources.arsc) // 2) Check if pkgRenamed is null @@ -167,35 +142,18 @@ private void adjustPackageManifest(ResTable resTable, String filePath) } } - private ExtMXSerializer getResXmlSerializer() { - ExtMXSerializer serial = new ExtMXSerializer(); - serial.setProperty(ExtXmlSerializer.PROPERTY_SERIALIZER_INDENTATION, " "); - serial.setProperty(ExtXmlSerializer.PROPERTY_SERIALIZER_LINE_SEPARATOR, System.getProperty("line.separator")); - serial.setProperty(ExtXmlSerializer.PROPERTY_DEFAULT_ENCODING, "utf-8"); - serial.setDisabledAttrEscape(true); - return serial; - } - - public void loadMainPkg() throws AndrolibException { - mResTable.loadMainPkg(mApkFile); - } - - public ResTable decodeResources(File outDir) throws AndrolibException { - if (hasResources()) { - loadMainPkg(); - decodeResources(getResTable(), mApkFile, outDir); + public void decodeResources(File outDir) throws AndrolibException { + if (!mApkInfo.hasResources()) { + return; } - return mResTable; - } - private void decodeResources(ResTable resTable, ExtFile apkFile, File outDir) - throws AndrolibException { + mResTable.loadMainPkg(mApkFile); ResStreamDecoderContainer decoders = new ResStreamDecoderContainer(); decoders.setDecoder("raw", new ResRawStreamDecoder()); decoders.setDecoder("9patch", new Res9patchStreamDecoder()); - AXmlResourceParser axmlParser = new AXmlResourceParser(resTable); + AXmlResourceParser axmlParser = new AXmlResourceParser(mResTable); decoders.setDecoder("xml", new XmlPullStreamDecoder(axmlParser, getResXmlSerializer())); ResFileDecoder fileDecoder = new ResFileDecoder(decoders); @@ -203,14 +161,14 @@ private void decodeResources(ResTable resTable, ExtFile apkFile, File outDir) try { out = new FileDirectory(outDir); - in = apkFile.getDirectory(); + in = mApkFile.getDirectory(); out = out.createDir("res"); } catch (DirectoryException ex) { throw new AndrolibException(ex); } ExtMXSerializer xmlSerializer = getResXmlSerializer(); - for (ResPackage pkg : resTable.listMainPackages()) { + for (ResPackage pkg : mResTable.listMainPackages()) { LOGGER.info("Decoding file-resources..."); for (ResResource res : pkg.listFiles()) { @@ -230,6 +188,15 @@ private void decodeResources(ResTable resTable, ExtFile apkFile, File outDir) } } + private ExtMXSerializer getResXmlSerializer() { + ExtMXSerializer serial = new ExtMXSerializer(); + serial.setProperty(ExtXmlSerializer.PROPERTY_SERIALIZER_INDENTATION, " "); + serial.setProperty(ExtXmlSerializer.PROPERTY_SERIALIZER_LINE_SEPARATOR, System.getProperty("line.separator")); + serial.setProperty(ExtXmlSerializer.PROPERTY_DEFAULT_ENCODING, "utf-8"); + serial.setDisabledAttrEscape(true); + return serial; + } + private void generateValuesFile(ResValuesFile valuesFile, Directory out, ExtXmlSerializer serial) throws AndrolibException { try { diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/ResTable.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/ResTable.java index 09939fa175..3ec0f23a23 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/ResTable.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/ResTable.java @@ -57,6 +57,10 @@ public ResTable() { this(Config.getDefaultConfig(), new ApkInfo()); } + public ResTable(ExtFile apkFile) { + this(Config.getDefaultConfig(), new ApkInfo(apkFile)); + } + public ResTable(Config config, ApkInfo apkInfo) { mConfig = config; mApkInfo = apkInfo; @@ -145,8 +149,7 @@ public void loadMainPkg(ExtFile apkFile) throws AndrolibException { mMainPkgLoaded = true; } - private ResPackage loadFrameworkPkg(int id) - throws AndrolibException { + private ResPackage loadFrameworkPkg(int id) throws AndrolibException { Framework framework = new Framework(mConfig); File frameworkApk = framework.getFrameworkApk(id, mConfig.frameworkTag); @@ -168,8 +171,7 @@ private ResPackage loadFrameworkPkg(int id) return pkg; } - private ResPackage[] loadResPackagesFromApk(ExtFile apkFile, boolean keepBrokenResources) - throws AndrolibException { + private ResPackage[] loadResPackagesFromApk(ExtFile apkFile, boolean keepBrokenResources) throws AndrolibException { try { Directory dir = apkFile.getDirectory(); try (BufferedInputStream bfi = new BufferedInputStream(dir.getFileInput("resources.arsc"))) { diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/value/ResAttr.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/value/ResAttr.java index 09530f5391..f47b224314 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/value/ResAttr.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/data/value/ResAttr.java @@ -40,8 +40,7 @@ public String convertToResXmlFormat(ResScalarValue value) throws AndrolibExcepti } @Override - public void serializeToResValuesXml(XmlSerializer serializer, ResResource res) - throws IOException, AndrolibException { + public void serializeToResValuesXml(XmlSerializer serializer, ResResource res) throws IOException, AndrolibException { String type = getTypeAsString(); serializer.startTag(null, "attr"); @@ -106,7 +105,7 @@ public static ResAttr factory(ResReferenceValue parent, throw new AndrolibException("Could not decode attr value"); } - protected void serializeBody(XmlSerializer serializer, ResResource res) throws AndrolibException, IOException { + protected void serializeBody(XmlSerializer serializer, ResResource res) throws IOException, AndrolibException { } protected String getTypeAsString() { diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ARSCDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ARSCDecoder.java index 285d6be5ff..605d82ef6f 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ARSCDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ARSCDecoder.java @@ -19,19 +19,14 @@ import android.util.TypedValue; import brut.androlib.exceptions.AndrolibException; import brut.androlib.res.data.*; +import brut.androlib.res.data.arsc.*; import brut.androlib.res.data.value.*; -import brut.androlib.res.data.arsc.ARSCData; -import brut.androlib.res.data.arsc.ARSCHeader; -import brut.androlib.res.data.arsc.EntryData; -import brut.androlib.res.data.arsc.FlagsOffset; import brut.util.Duo; import brut.util.ExtDataInput; import com.google.common.io.LittleEndianDataInputStream; import org.apache.commons.io.input.CountingInputStream; -import java.io.DataInput; -import java.io.IOException; -import java.io.InputStream; +import java.io.*; import java.math.BigInteger; import java.util.*; import java.util.logging.Logger; diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/AXmlResourceParser.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/AXmlResourceParser.java index 66d798dfb1..59b63a0187 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/AXmlResourceParser.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/AXmlResourceParser.java @@ -25,13 +25,11 @@ import brut.androlib.res.data.axml.NamespaceStack; import brut.androlib.res.xml.ResXmlEncoders; import brut.util.ExtDataInput; -import org.apache.commons.io.input.CountingInputStream; import com.google.common.io.LittleEndianDataInputStream; +import org.apache.commons.io.input.CountingInputStream; import org.xmlpull.v1.XmlPullParserException; -import java.io.DataInput; -import java.io.IOException; -import java.io.InputStream; -import java.io.Reader; + +import java.io.*; import java.util.logging.Level; import java.util.logging.Logger; diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/Res9patchStreamDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/Res9patchStreamDecoder.java index 5dfad02ee0..1750ba7855 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/Res9patchStreamDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/Res9patchStreamDecoder.java @@ -27,11 +27,7 @@ import java.awt.image.BufferedImage; import java.awt.image.Raster; import java.awt.image.WritableRaster; -import java.io.ByteArrayInputStream; -import java.io.DataInput; -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; +import java.io.*; public class Res9patchStreamDecoder implements ResStreamDecoder { @Override diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResAttrDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResAttrDecoder.java index 20d5066b3b..7a12b6370a 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResAttrDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResAttrDecoder.java @@ -31,8 +31,7 @@ public ResAttrDecoder(ResTable resTable) { mResTable = resTable; } - public String decode(int type, int value, String rawValue, int attrResId) - throws AndrolibException { + public String decode(int type, int value, String rawValue, int attrResId) throws AndrolibException { ResScalarValue resValue = mResTable.getCurrentResPackage().getValueFactory().factory(type, value, rawValue); String decoded = null; @@ -47,9 +46,7 @@ public String decode(int type, int value, String rawValue, int attrResId) return decoded != null ? decoded : resValue.encodeAsResXmlAttr(); } - public String decodeFromResourceId(int attrResId) - throws AndrolibException { - + public String decodeFromResourceId(int attrResId) throws AndrolibException { if (attrResId != 0) { try { ResResSpec resResSpec = mResTable.getResSpec(attrResId); diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResFileDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResFileDecoder.java index a6111889ee..4632de5c34 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResFileDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/ResFileDecoder.java @@ -139,8 +139,8 @@ public void decode(ResResource res, Directory inDir, Directory outDir, Map mDecoders = new HashMap<>(); diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/XmlPullStreamDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/XmlPullStreamDecoder.java index b8bd8256cb..08cf89232c 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/XmlPullStreamDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/decoder/XmlPullStreamDecoder.java @@ -28,9 +28,7 @@ import org.xmlpull.v1.wrapper.XmlSerializerWrapper; import org.xmlpull.v1.wrapper.classic.StaticXmlSerializerWrapper; -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; +import java.io.*; public class XmlPullStreamDecoder implements ResStreamDecoder { public XmlPullStreamDecoder(XmlPullParser parser, diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/util/ExtMXSerializer.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/util/ExtMXSerializer.java index d087ccedac..7114dae4ee 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/util/ExtMXSerializer.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/util/ExtMXSerializer.java @@ -18,9 +18,7 @@ import org.xmlpull.renamed.MXSerializer; -import java.io.IOException; -import java.io.OutputStream; -import java.io.Writer; +import java.io.*; public class ExtMXSerializer extends MXSerializer implements ExtXmlSerializer { @Override diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/xml/ResXmlPatcher.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/xml/ResXmlPatcher.java index a7043767d7..53cdf52cc8 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/xml/ResXmlPatcher.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/res/xml/ResXmlPatcher.java @@ -29,9 +29,7 @@ import javax.xml.transform.dom.DOMSource; import javax.xml.transform.stream.StreamResult; import javax.xml.xpath.*; -import java.io.File; -import java.io.FileInputStream; -import java.io.IOException; +import java.io.*; import java.util.logging.Logger; public final class ResXmlPatcher { @@ -135,7 +133,7 @@ public static void setNetworkSecurityConfig(File file) { * @throws ParserConfigurationException XML nodes could be written */ public static void modNetworkSecurityConfig(File file) - throws ParserConfigurationException, TransformerException, IOException, SAXException { + throws ParserConfigurationException, TransformerException, IOException, SAXException { DocumentBuilderFactory documentFactory = DocumentBuilderFactory.newInstance(); DocumentBuilder documentBuilder = documentFactory.newDocumentBuilder(); diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliBuilder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliBuilder.java index 143f3b7489..6880c1557a 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliBuilder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliBuilder.java @@ -25,9 +25,7 @@ import com.android.tools.smali.dexlib2.writer.builder.DexBuilder; import com.android.tools.smali.dexlib2.writer.io.FileDataStore; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; +import java.io.*; import java.nio.file.Files; import java.util.logging.Logger; diff --git a/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliDecoder.java b/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliDecoder.java index 2ee83ac7b7..688025b693 100644 --- a/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliDecoder.java +++ b/brut.apktool/apktool-lib/src/main/java/brut/androlib/src/SmaliDecoder.java @@ -27,8 +27,7 @@ import com.android.tools.smali.dexlib2.iface.DexFile; import com.android.tools.smali.dexlib2.iface.MultiDexContainer; -import java.io.File; -import java.io.IOException; +import java.io.*; public class SmaliDecoder { diff --git a/brut.apktool/apktool-lib/src/main/java/org/xmlpull/renamed/MXSerializer.java b/brut.apktool/apktool-lib/src/main/java/org/xmlpull/renamed/MXSerializer.java index 773a47fa3d..62fed7e5c5 100644 --- a/brut.apktool/apktool-lib/src/main/java/org/xmlpull/renamed/MXSerializer.java +++ b/brut.apktool/apktool-lib/src/main/java/org/xmlpull/renamed/MXSerializer.java @@ -18,10 +18,7 @@ import org.xmlpull.v1.XmlSerializer; -import java.io.IOException; -import java.io.OutputStream; -import java.io.OutputStreamWriter; -import java.io.Writer; +import java.io.*; import java.util.Objects; /** diff --git a/brut.apktool/apktool-lib/src/test/java/brut/androlib/aapt2/NonStandardPkgIdTest.java b/brut.apktool/apktool-lib/src/test/java/brut/androlib/aapt2/NonStandardPkgIdTest.java index 7568a646b2..659c42b1bd 100644 --- a/brut.apktool/apktool-lib/src/test/java/brut/androlib/aapt2/NonStandardPkgIdTest.java +++ b/brut.apktool/apktool-lib/src/test/java/brut/androlib/aapt2/NonStandardPkgIdTest.java @@ -17,6 +17,7 @@ package brut.androlib.aapt2; import brut.androlib.*; +import brut.androlib.apk.ApkInfo; import brut.androlib.exceptions.AndrolibException; import brut.androlib.res.ResourcesDecoder; import brut.androlib.res.data.ResTable; @@ -27,8 +28,6 @@ import org.junit.BeforeClass; import org.junit.Test; -import java.io.File; - import static org.junit.Assert.*; public class NonStandardPkgIdTest extends BaseTest { @@ -50,16 +49,18 @@ public static void beforeClass() throws Exception { config.verbose = true; LOGGER.info("Building pkgid8.apk..."); - File testApk = new File(sTmpDir, "pkgid8.apk"); + ExtFile testApk = new ExtFile(sTmpDir, "pkgid8.apk"); new ApkBuilder(config, sTestOrigDir).build(testApk); LOGGER.info("Decoding pkgid8.apk..."); - ResourcesDecoder resourcesDecoder = new ResourcesDecoder( - Config.getDefaultConfig(), new ExtFile(testApk)); + ApkInfo testInfo = new ApkInfo(testApk); + ResourcesDecoder resourcesDecoder = new ResourcesDecoder(Config.getDefaultConfig(), testApk, testInfo); sTestNewDir.mkdirs(); - mResTable = resourcesDecoder.decodeResources(sTestNewDir); + resourcesDecoder.decodeResources(sTestNewDir); resourcesDecoder.decodeManifest(sTestNewDir); + + mResTable = resourcesDecoder.getResTable(); } @AfterClass diff --git a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoReaderTest.java b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoReaderTest.java index ff2c9ad9e3..4e3b1c2ebd 100644 --- a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoReaderTest.java +++ b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoReaderTest.java @@ -24,7 +24,7 @@ public class ApkInfoReaderTest { private void checkStandard(ApkInfo apkInfo) { - assertEquals("standard.apk", apkInfo.getApkFileName()); + assertEquals("standard.apk", apkInfo.apkFileName); assertFalse(apkInfo.resourcesAreCompressed); assertEquals(1, apkInfo.doNotCompress.size()); assertEquals("resources.arsc", apkInfo.doNotCompress.iterator().next()); @@ -85,7 +85,7 @@ public void testUnknownFiles() throws AndrolibException { ApkInfo apkInfo = ApkInfo.load( this.getClass().getResourceAsStream("/apk/unknown_files.yml")); assertEquals("2.0.0", apkInfo.version); - assertEquals("testapp.apk", apkInfo.getApkFileName()); + assertEquals("testapp.apk", apkInfo.apkFileName); assertFalse(apkInfo.isFrameworkApk); assertNotNull(apkInfo.usesFramework); assertEquals(1, apkInfo.usesFramework.ids.size()); @@ -118,7 +118,7 @@ public void testUlist_with_indent() throws AndrolibException { ApkInfo apkInfo = ApkInfo.load( this.getClass().getResourceAsStream("/apk/list_with_indent.yml")); assertEquals("2.8.0", apkInfo.version); - assertEquals("basic.apk", apkInfo.getApkFileName()); + assertEquals("basic.apk", apkInfo.apkFileName); assertFalse(apkInfo.isFrameworkApk); assertNotNull(apkInfo.usesFramework); assertEquals(1, apkInfo.usesFramework.ids.size()); diff --git a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoSerializationTest.java b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoSerializationTest.java index 9c9b9739c2..295493a06a 100644 --- a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoSerializationTest.java +++ b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ApkInfoSerializationTest.java @@ -39,9 +39,7 @@ public void checkApkInfoSerialization() throws IOException, AndrolibException { File savedApkInfo = folder.newFile( "saved.yml" ); control.save(savedApkInfo); - try ( - FileInputStream fis = new FileInputStream(savedApkInfo); - ) { + try (FileInputStream fis = new FileInputStream(savedApkInfo)) { ApkInfo saved = ApkInfo.load(fis); check(saved); } @@ -49,7 +47,7 @@ public void checkApkInfoSerialization() throws IOException, AndrolibException { private void check(ApkInfo apkInfo) { assertEquals("2.0.0", apkInfo.version); - assertEquals("testapp.apk", apkInfo.getApkFileName()); + assertEquals("testapp.apk", apkInfo.apkFileName); assertFalse(apkInfo.isFrameworkApk); assertNotNull(apkInfo.usesFramework); assertEquals(1, apkInfo.usesFramework.ids.size()); diff --git a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ConsistentPropertyTest.java b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ConsistentPropertyTest.java index cca483bb73..bada5313e7 100644 --- a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ConsistentPropertyTest.java +++ b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/ConsistentPropertyTest.java @@ -29,7 +29,7 @@ public void testAssertingAllKnownApkInfoProperties() throws AndrolibException { this.getClass().getResourceAsStream("/apk/basic.yml")); assertEquals("2.8.0", apkInfo.version); - assertEquals("basic.apk", apkInfo.getApkFileName()); + assertEquals("basic.apk", apkInfo.apkFileName); assertFalse(apkInfo.isFrameworkApk); assertEquals(1, apkInfo.usesFramework.ids.size()); assertEquals("tag", apkInfo.usesFramework.tag); diff --git a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/DoNotCompressHieroglyphTest.java b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/DoNotCompressHieroglyphTest.java index 18251394a7..9c9fbd3c50 100644 --- a/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/DoNotCompressHieroglyphTest.java +++ b/brut.apktool/apktool-lib/src/test/java/brut/androlib/apk/DoNotCompressHieroglyphTest.java @@ -28,7 +28,7 @@ public void testHieroglyph() throws AndrolibException { ApkInfo apkInfo = ApkInfo.load( this.getClass().getResourceAsStream("/apk/donotcompress_with_hieroglyph.yml")); assertEquals("2.0.0", apkInfo.version); - assertEquals("testapp.apk", apkInfo.getApkFileName()); + assertEquals("testapp.apk", apkInfo.apkFileName); assertEquals(2, apkInfo.doNotCompress.size()); assertEquals("assets/AllAssetBundles/Andriod/tx_1001_冰原1", apkInfo.doNotCompress.get(0)); assertEquals("assets/AllAssetBundles/Andriod/tx_1001_冰原1.manifest", apkInfo.doNotCompress.get(1)); diff --git a/brut.apktool/apktool-lib/src/test/java/brut/androlib/decode/DecodeArrayTest.java b/brut.apktool/apktool-lib/src/test/java/brut/androlib/decode/DecodeArrayTest.java index 68b1104f62..941e6dd653 100644 --- a/brut.apktool/apktool-lib/src/test/java/brut/androlib/decode/DecodeArrayTest.java +++ b/brut.apktool/apktool-lib/src/test/java/brut/androlib/decode/DecodeArrayTest.java @@ -20,6 +20,7 @@ import brut.androlib.BaseTest; import brut.androlib.Config; import brut.androlib.TestUtils; +import brut.androlib.apk.ApkInfo; import brut.androlib.res.ResourcesDecoder; import brut.androlib.res.data.ResTable; import brut.androlib.res.data.value.ResArrayValue; @@ -51,11 +52,10 @@ public static void afterClass() throws BrutException { @Test public void decodeStringArray() throws BrutException { - String apk = "issue1994.apk"; - //ApkDecoder apkDecoder = new ApkDecoder(new File(sTmpDir + File.separator + apk)); - ResourcesDecoder resourcesDecoder = new ResourcesDecoder( - Config.getDefaultConfig(), - new ExtFile(sTmpDir + File.separator + apk)); + ExtFile apkFile = new ExtFile(sTmpDir, "issue1994.apk"); + ApkInfo apkInfo = new ApkInfo(apkFile); + //ApkDecoder apkDecoder = new ApkDecoder(apkFile); + ResourcesDecoder resourcesDecoder = new ResourcesDecoder(Config.getDefaultConfig(), apkFile, apkInfo); resourcesDecoder.loadMainPkg(); ResTable resTable = resourcesDecoder.getResTable(); @@ -66,10 +66,9 @@ public void decodeStringArray() throws BrutException { @Test public void decodeArray() throws BrutException { - String apk = "issue1994.apk"; - ResourcesDecoder resourcesDecoder = new ResourcesDecoder( - Config.getDefaultConfig(), - new ExtFile(sTmpDir + File.separator + apk)); + ExtFile apkFile = new ExtFile(sTmpDir, "issue1994.apk"); + ApkInfo apkInfo = new ApkInfo(apkFile); + ResourcesDecoder resourcesDecoder = new ResourcesDecoder(Config.getDefaultConfig(), apkFile, apkInfo); resourcesDecoder.loadMainPkg(); ResTable resTable = resourcesDecoder.getResTable(); diff --git a/brut.j.dir/src/main/java/brut/directory/AbstractDirectory.java b/brut.j.dir/src/main/java/brut/directory/AbstractDirectory.java index a242b6f412..ef7ddf50ad 100644 --- a/brut.j.dir/src/main/java/brut/directory/AbstractDirectory.java +++ b/brut.j.dir/src/main/java/brut/directory/AbstractDirectory.java @@ -262,12 +262,9 @@ private ParsedPath parsePath(String path) { protected abstract void loadFiles(); protected abstract void loadDirs(); - protected abstract InputStream getFileInputLocal(String name) - throws DirectoryException; - protected abstract OutputStream getFileOutputLocal(String name) - throws DirectoryException; - protected abstract AbstractDirectory createDirLocal(String name) - throws DirectoryException; + protected abstract InputStream getFileInputLocal(String name) throws DirectoryException; + protected abstract OutputStream getFileOutputLocal(String name) throws DirectoryException; + protected abstract AbstractDirectory createDirLocal(String name) throws DirectoryException; protected abstract void removeFileLocal(String name); diff --git a/brut.j.dir/src/main/java/brut/directory/Directory.java b/brut.j.dir/src/main/java/brut/directory/Directory.java index 86c848a63d..c5ece15837 100644 --- a/brut.j.dir/src/main/java/brut/directory/Directory.java +++ b/brut.j.dir/src/main/java/brut/directory/Directory.java @@ -48,28 +48,21 @@ public interface Directory { void copyToDir(Directory out) throws DirectoryException; - void copyToDir(Directory out, String[] fileNames) - throws DirectoryException; + void copyToDir(Directory out, String[] fileNames) throws DirectoryException; - void copyToDir(Directory out, String fileName) - throws DirectoryException; + void copyToDir(Directory out, String fileName) throws DirectoryException; void copyToDir(File out) throws DirectoryException; - void copyToDir(File out, String[] fileNames) - throws DirectoryException; + void copyToDir(File out, String[] fileNames) throws DirectoryException; - void copyToDir(File out, String fileName) - throws DirectoryException; + void copyToDir(File out, String fileName) throws DirectoryException; - long getSize(String fileName) - throws DirectoryException; + long getSize(String fileName) throws DirectoryException; - long getCompressedSize(String fileName) - throws DirectoryException; + long getCompressedSize(String fileName) throws DirectoryException; - int getCompressionLevel(String fileName) - throws DirectoryException; + int getCompressionLevel(String fileName) throws DirectoryException; void close() throws IOException; diff --git a/brut.j.util/src/main/java/brut/util/BrutIO.java b/brut.j.util/src/main/java/brut/util/BrutIO.java index 7643263842..ede9a85330 100644 --- a/brut.j.util/src/main/java/brut/util/BrutIO.java +++ b/brut.j.util/src/main/java/brut/util/BrutIO.java @@ -105,17 +105,13 @@ public static String normalizePath(String path) { } public static void copy(File inputFile, ZipOutputStream outputFile) throws IOException { - try ( - FileInputStream fis = new FileInputStream(inputFile) - ) { + try (FileInputStream fis = new FileInputStream(inputFile)) { IOUtils.copy(fis, outputFile); } } public static void copy(ZipFile inputFile, ZipOutputStream outputFile, ZipEntry entry) throws IOException { - try ( - InputStream is = inputFile.getInputStream(entry) - ) { + try (InputStream is = inputFile.getInputStream(entry)) { IOUtils.copy(is, outputFile); } }