// Copyright 2000-2021 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license that can be found in the LICENSE file. package org.jetbrains.java.decompiler.struct; import org.jetbrains.java.decompiler.main.DecompilerContext; import org.jetbrains.java.decompiler.main.extern.IResultSaver; import org.jetbrains.java.decompiler.struct.lazy.LazyLoader; import org.jetbrains.java.decompiler.util.DataInputFullStream; import org.jetbrains.java.decompiler.util.InterpreterUtil; import java.io.File; import java.io.IOException; import java.util.Enumeration; import java.util.HashMap; import java.util.Map; import java.util.jar.JarFile; import java.util.zip.ZipEntry; import java.util.zip.ZipFile; public class StructContext { private final IResultSaver saver; private final IDecompiledData decompiledData; private final LazyLoader loader; private final Map units = new HashMap<>(); private final Map classes = new HashMap<>(); public StructContext(IResultSaver saver, IDecompiledData decompiledData, LazyLoader loader) { this.saver = saver; this.decompiledData = decompiledData; this.loader = loader; ContextUnit defaultUnit = new ContextUnit(ContextUnit.TYPE_FOLDER, null, "", true, saver, decompiledData); units.put("", defaultUnit); } public StructClass getClass(String name) { return classes.get(name); } public void reloadContext() throws IOException { for (ContextUnit unit : units.values()) { for (StructClass cl : unit.getClasses()) { classes.remove(cl.qualifiedName); } unit.reload(loader); // adjust global class collection for (StructClass cl : unit.getClasses()) { classes.put(cl.qualifiedName, cl); } } } public void saveContext() { for (ContextUnit unit : units.values()) { if (unit.isOwn()) { unit.save(); } } } public void addSpace(File file, boolean isOwn) { addSpace("", file, isOwn, 0); } private void addSpace(String path, File file, boolean isOwn, int level) { if (file.isDirectory()) { if (level == 1) path += file.getName(); else if (level > 1) path += "/" + file.getName(); File[] files = file.listFiles(); if (files != null) { for (int i = files.length - 1; i >= 0; i--) { addSpace(path, files[i], isOwn, level + 1); } } } else { String filename = file.getName(); boolean isArchive = false; try { if (filename.endsWith(".jar")) { isArchive = true; addArchive(path, file, ContextUnit.TYPE_JAR, isOwn); } else if (filename.endsWith(".zip")) { isArchive = true; addArchive(path, file, ContextUnit.TYPE_ZIP, isOwn); } } catch (IOException ex) { String message = "Corrupted archive file: " + file; DecompilerContext.getLogger().writeMessage(message, ex); } if (isArchive) { return; } ContextUnit unit = units.get(path); if (unit == null) { unit = new ContextUnit(ContextUnit.TYPE_FOLDER, null, path, isOwn, saver, decompiledData); units.put(path, unit); } if (filename.endsWith(".class")) { try (DataInputFullStream in = loader.getClassStream(file.getAbsolutePath(), null)) { StructClass cl = StructClass.create(in, isOwn, loader); classes.put(cl.qualifiedName, cl); unit.addClass(cl, filename); loader.addClassLink(cl.qualifiedName, new LazyLoader.Link(file.getAbsolutePath(), null)); } catch (IOException ex) { String message = "Corrupted class file: " + file; DecompilerContext.getLogger().writeMessage(message, ex); } } else { unit.addOtherEntry(file.getAbsolutePath(), filename); } } } private void addArchive(String path, File file, int type, boolean isOwn) throws IOException { try (ZipFile archive = type == ContextUnit.TYPE_JAR ? new JarFile(file) : new ZipFile(file)) { Enumeration entries = archive.entries(); while (entries.hasMoreElements()) { ZipEntry entry = entries.nextElement(); ContextUnit unit = units.get(path + "/" + file.getName()); if (unit == null) { unit = new ContextUnit(type, path, file.getName(), isOwn, saver, decompiledData); if (type == ContextUnit.TYPE_JAR) { unit.setManifest(((JarFile)archive).getManifest()); } units.put(path + "/" + file.getName(), unit); } String name = entry.getName(); if (!entry.isDirectory()) { if (name.endsWith(".class")) { byte[] bytes = InterpreterUtil.getBytes(archive, entry); StructClass cl = StructClass.create(new DataInputFullStream(bytes), isOwn, loader); classes.put(cl.qualifiedName, cl); unit.addClass(cl, name); loader.addClassLink(cl.qualifiedName, new LazyLoader.Link(file.getAbsolutePath(), name)); } else { unit.addOtherEntry(file.getAbsolutePath(), name); } } else { unit.addDirEntry(name); } } } } public Map getClasses() { return classes; } }