|
|
|
// Copyright 2000-2017 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<String, ContextUnit> units = new HashMap<>();
|
|
|
|
private final Map<String, StructClass> 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 = new StructClass(in, isOwn, loader);
|
|
|
|
classes.put(cl.qualifiedName, cl);
|
|
|
|
unit.addClass(cl, filename);
|
|
|
|
loader.addClassLink(cl.qualifiedName, new LazyLoader.Link(LazyLoader.Link.CLASS, 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 {
|
|
|
|
//noinspection IOResourceOpenedButNotSafelyClosed
|
|
|
|
try (ZipFile archive = type == ContextUnit.TYPE_JAR ? new JarFile(file) : new ZipFile(file)) {
|
|
|
|
Enumeration<? extends ZipEntry> 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 = new StructClass(bytes, isOwn, loader);
|
|
|
|
classes.put(cl.qualifiedName, cl);
|
|
|
|
unit.addClass(cl, name);
|
|
|
|
loader.addClassLink(cl.qualifiedName, new LazyLoader.Link(LazyLoader.Link.ENTRY, file.getAbsolutePath(), name));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
unit.addOtherEntry(file.getAbsolutePath(), name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
unit.addDirEntry(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Map<String, StructClass> getClasses() {
|
|
|
|
return classes;
|
|
|
|
}
|
|
|
|
}
|