mirror of
https://github.com/GravitLauncher/Launcher
synced 2025-01-10 17:49:40 +03:00
[FEATURE][EXPERIMENTAL] Новое API трансформации классов при сборке
This commit is contained in:
parent
c075697316
commit
2119688609
4 changed files with 126 additions and 90 deletions
|
@ -29,6 +29,14 @@ public void pushFile(String filename, InputStream inputStream) throws IOExceptio
|
||||||
ZipEntry zip = IOHelper.newZipEntry(filename);
|
ZipEntry zip = IOHelper.newZipEntry(filename);
|
||||||
output.putNextEntry(zip);
|
output.putNextEntry(zip);
|
||||||
IOHelper.transfer(inputStream, output);
|
IOHelper.transfer(inputStream, output);
|
||||||
|
output.closeEntry();
|
||||||
|
fileList.add(filename);
|
||||||
|
}
|
||||||
|
public void pushBytes(String filename, byte[] bytes) throws IOException {
|
||||||
|
ZipEntry zip = IOHelper.newZipEntry(filename);
|
||||||
|
output.putNextEntry(zip);
|
||||||
|
output.write(bytes);
|
||||||
|
output.closeEntry();
|
||||||
fileList.add(filename);
|
fileList.add(filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -43,20 +43,6 @@ public Path process(Path inputFile) throws IOException {
|
||||||
public static void apply(Path inputFile, Path addFile, ZipOutputStream output, LaunchServer srv, Predicate<ZipEntry> excluder, boolean needFixes) throws IOException {
|
public static void apply(Path inputFile, Path addFile, ZipOutputStream output, LaunchServer srv, Predicate<ZipEntry> excluder, boolean needFixes) throws IOException {
|
||||||
try (ClassMetadataReader reader = new ClassMetadataReader()) {
|
try (ClassMetadataReader reader = new ClassMetadataReader()) {
|
||||||
reader.getCp().add(new JarFile(inputFile.toFile()));
|
reader.getCp().add(new JarFile(inputFile.toFile()));
|
||||||
List<JarFile> libs = srv.launcherBinary.coreLibs.stream().map(e -> {
|
|
||||||
try {
|
|
||||||
return new JarFile(e.toFile());
|
|
||||||
} catch (IOException e1) {
|
|
||||||
throw new RuntimeException(e1);
|
|
||||||
}
|
|
||||||
}).collect(Collectors.toList());
|
|
||||||
libs.addAll(srv.launcherBinary.addonLibs.stream().map(e -> {
|
|
||||||
try {
|
|
||||||
return new JarFile(e.toFile());
|
|
||||||
} catch (IOException e1) {
|
|
||||||
throw new RuntimeException(e1);
|
|
||||||
}
|
|
||||||
}).collect(Collectors.toList()));
|
|
||||||
try (ZipInputStream input = IOHelper.newZipInput(addFile)) {
|
try (ZipInputStream input = IOHelper.newZipInput(addFile)) {
|
||||||
ZipEntry e = input.getNextEntry();
|
ZipEntry e = input.getNextEntry();
|
||||||
while (e != null) {
|
while (e != null) {
|
||||||
|
|
|
@ -1,7 +1,10 @@
|
||||||
package pro.gravit.launchserver.binary.tasks;
|
package pro.gravit.launchserver.binary.tasks;
|
||||||
|
|
||||||
import org.objectweb.asm.ClassReader;
|
import org.objectweb.asm.ClassReader;
|
||||||
|
import org.objectweb.asm.ClassWriter;
|
||||||
|
import org.objectweb.asm.tree.AnnotationNode;
|
||||||
import org.objectweb.asm.tree.ClassNode;
|
import org.objectweb.asm.tree.ClassNode;
|
||||||
|
import org.objectweb.asm.tree.FieldNode;
|
||||||
import pro.gravit.launcher.AutogenConfig;
|
import pro.gravit.launcher.AutogenConfig;
|
||||||
import pro.gravit.launcher.Launcher;
|
import pro.gravit.launcher.Launcher;
|
||||||
import pro.gravit.launcher.LauncherConfig;
|
import pro.gravit.launcher.LauncherConfig;
|
||||||
|
@ -37,27 +40,80 @@
|
||||||
public class MainBuildTask implements LauncherBuildTask {
|
public class MainBuildTask implements LauncherBuildTask {
|
||||||
private final LaunchServer server;
|
private final LaunchServer server;
|
||||||
public final ClassMetadataReader reader;
|
public final ClassMetadataReader reader;
|
||||||
|
@FunctionalInterface
|
||||||
|
public interface Transformer {
|
||||||
|
byte[] transform(byte[] input, String classname, BuildContext context);
|
||||||
|
}
|
||||||
|
|
||||||
private final class RuntimeDirVisitor extends SimpleFileVisitor<Path> {
|
public interface ASMTransformer extends Transformer {
|
||||||
|
default byte[] transform(byte[] input, String classname, BuildContext context)
|
||||||
|
{
|
||||||
|
ClassReader reader = new ClassReader(input);
|
||||||
|
ClassNode cn = new ClassNode();
|
||||||
|
reader.accept(cn, 0);
|
||||||
|
transform(cn, classname, context);
|
||||||
|
ClassWriter writer = new ClassWriter(ClassWriter.COMPUTE_FRAMES | ClassWriter.COMPUTE_MAXS);
|
||||||
|
cn.accept(writer);
|
||||||
|
return writer.toByteArray();
|
||||||
|
}
|
||||||
|
void transform(ClassNode cn, String classname, BuildContext context);
|
||||||
|
}
|
||||||
|
public abstract static class ASMAnnotationFieldProcessor implements ASMTransformer
|
||||||
|
{
|
||||||
|
private final String desc;
|
||||||
|
|
||||||
|
protected ASMAnnotationFieldProcessor(String desc) {
|
||||||
|
this.desc = desc;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void transform(ClassNode cn, String classname, BuildContext context) {
|
||||||
|
for(FieldNode fn : cn.fields)
|
||||||
|
{
|
||||||
|
if(fn.invisibleAnnotations == null || fn.invisibleAnnotations.isEmpty()) continue;
|
||||||
|
AnnotationNode found = null;
|
||||||
|
for(AnnotationNode an : fn.invisibleAnnotations)
|
||||||
|
{
|
||||||
|
if(an == null) continue;
|
||||||
|
if(desc.equals(an.desc))
|
||||||
|
{
|
||||||
|
found = an;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(found != null)
|
||||||
|
{
|
||||||
|
transformField(found, fn, cn, classname, context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
abstract public void transformField(AnnotationNode an, FieldNode fn, ClassNode cn, String classname, BuildContext context);
|
||||||
|
}
|
||||||
|
|
||||||
|
private final static class RuntimeDirVisitor extends SimpleFileVisitor<Path> {
|
||||||
private final ZipOutputStream output;
|
private final ZipOutputStream output;
|
||||||
private final Map<String, byte[]> runtime;
|
private final Map<String, byte[]> hashs;
|
||||||
|
private final Path sourceDir;
|
||||||
|
private final String targetDir;
|
||||||
|
|
||||||
private RuntimeDirVisitor(ZipOutputStream output, Map<String, byte[]> runtime) {
|
private RuntimeDirVisitor(ZipOutputStream output, Map<String, byte[]> hashs, Path sourceDir, String targetDir) {
|
||||||
this.output = output;
|
this.output = output;
|
||||||
this.runtime = runtime;
|
this.hashs = hashs;
|
||||||
|
this.sourceDir = sourceDir;
|
||||||
|
this.targetDir = targetDir;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException {
|
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException {
|
||||||
String dirName = IOHelper.toString(server.launcherBinary.runtimeDir.relativize(dir));
|
String dirName = IOHelper.toString(sourceDir.relativize(dir));
|
||||||
output.putNextEntry(newEntry(dirName + '/'));
|
output.putNextEntry(newEntry(dirName + '/'));
|
||||||
return super.preVisitDirectory(dir, attrs);
|
return super.preVisitDirectory(dir, attrs);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
|
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
|
||||||
String fileName = IOHelper.toString(server.launcherBinary.runtimeDir.relativize(file));
|
String fileName = IOHelper.toString(sourceDir.relativize(file));
|
||||||
runtime.put(fileName, SecurityHelper.digest(SecurityHelper.DigestAlgorithm.MD5, file));
|
hashs.put(fileName, SecurityHelper.digest(SecurityHelper.DigestAlgorithm.MD5, file));
|
||||||
|
|
||||||
// Create zip entry and transfer contents
|
// Create zip entry and transfer contents
|
||||||
output.putNextEntry(newEntry(fileName));
|
output.putNextEntry(newEntry(fileName));
|
||||||
|
@ -66,45 +122,13 @@ public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IO
|
||||||
// Return result
|
// Return result
|
||||||
return super.visitFile(file, attrs);
|
return super.visitFile(file, attrs);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private final class GuardDirVisitor extends SimpleFileVisitor<Path> {
|
private ZipEntry newEntry(String fileName) {
|
||||||
private final ZipOutputStream output;
|
return newZipEntry( targetDir + IOHelper.CROSS_SEPARATOR + fileName);
|
||||||
private final Map<String, byte[]> guard;
|
|
||||||
|
|
||||||
private GuardDirVisitor(ZipOutputStream output, Map<String, byte[]> guard) {
|
|
||||||
this.output = output;
|
|
||||||
this.guard = guard;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException {
|
|
||||||
String dirName = IOHelper.toString(server.launcherBinary.guardDir.relativize(dir));
|
|
||||||
output.putNextEntry(newGuardEntry(dirName + '/'));
|
|
||||||
return super.preVisitDirectory(dir, attrs);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
|
|
||||||
String fileName = IOHelper.toString(server.launcherBinary.guardDir.relativize(file));
|
|
||||||
guard.put(fileName, SecurityHelper.digest(SecurityHelper.DigestAlgorithm.MD5, file));
|
|
||||||
|
|
||||||
// Create zip entry and transfer contents
|
|
||||||
output.putNextEntry(newGuardEntry(fileName));
|
|
||||||
IOHelper.transfer(file, output);
|
|
||||||
|
|
||||||
// Return result
|
|
||||||
return super.visitFile(file, attrs);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
public Set<String> blacklist = new HashSet<>();
|
||||||
private static ZipEntry newEntry(String fileName) {
|
public List<Transformer> transformers = new ArrayList<>();
|
||||||
return newZipEntry(Launcher.RUNTIME_DIR + IOHelper.CROSS_SEPARATOR + fileName);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static ZipEntry newGuardEntry(String fileName) {
|
|
||||||
return newZipEntry(Launcher.GUARD_DIR + IOHelper.CROSS_SEPARATOR + fileName);
|
|
||||||
}
|
|
||||||
|
|
||||||
public MainBuildTask(LaunchServer srv) {
|
public MainBuildTask(LaunchServer srv) {
|
||||||
server = srv;
|
server = srv;
|
||||||
|
@ -171,13 +195,13 @@ public Path process(Path inputJar) throws IOException {
|
||||||
LogHelper.error(e1);
|
LogHelper.error(e1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
String zPath = launcherConfigurator.getZipEntryPath();
|
context.pushBytes(launcherConfigurator.getZipEntryPath(), launcherConfigurator.getBytecode(reader));
|
||||||
String sPath = secureConfigurator.getZipEntryPath();
|
context.pushBytes(secureConfigurator.getZipEntryPath(), secureConfigurator.getBytecode(reader));
|
||||||
try (ZipInputStream input = new ZipInputStream(IOHelper.newInput(inputJar))) {
|
try (ZipInputStream input = new ZipInputStream(IOHelper.newInput(inputJar))) {
|
||||||
ZipEntry e = input.getNextEntry();
|
ZipEntry e = input.getNextEntry();
|
||||||
while (e != null) {
|
while (e != null) {
|
||||||
String filename = e.getName();
|
String filename = e.getName();
|
||||||
if (server.buildHookManager.isContainsBlacklist(filename) || e.isDirectory() || zPath.equals(filename) || sPath.equals(filename)) {
|
if (e.isDirectory() || blacklist.contains(filename) || context.fileList.contains(filename)) {
|
||||||
e = input.getNextEntry();
|
e = input.getNextEntry();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -191,12 +215,8 @@ public Path process(Path inputJar) throws IOException {
|
||||||
if (filename.endsWith(".class")) {
|
if (filename.endsWith(".class")) {
|
||||||
String classname = filename.replace('/', '.').substring(0,
|
String classname = filename.replace('/', '.').substring(0,
|
||||||
filename.length() - ".class".length());
|
filename.length() - ".class".length());
|
||||||
byte[] bytes;
|
byte[] bytes = IOHelper.read(input);
|
||||||
try (ByteArrayOutputStream outputStream = new ByteArrayOutputStream(2048)) {
|
bytes = transformClass(bytes, classname, context);
|
||||||
IOHelper.transfer(input, outputStream);
|
|
||||||
bytes = outputStream.toByteArray();
|
|
||||||
}
|
|
||||||
bytes = server.buildHookManager.classTransform(bytes, classname, this);
|
|
||||||
output.write(bytes);
|
output.write(bytes);
|
||||||
} else
|
} else
|
||||||
IOHelper.transfer(input, output);
|
IOHelper.transfer(input, output);
|
||||||
|
@ -204,17 +224,13 @@ public Path process(Path inputJar) throws IOException {
|
||||||
e = input.getNextEntry();
|
e = input.getNextEntry();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// write additional classes
|
|
||||||
for (Map.Entry<String, byte[]> ent : server.buildHookManager.getIncludeClass().entrySet()) {
|
|
||||||
output.putNextEntry(newZipEntry(JarHelper.getClassFile(ent.getKey())));
|
|
||||||
output.write(server.buildHookManager.classTransform(ent.getValue(), ent.getKey(), this));
|
|
||||||
}
|
|
||||||
// map for guard
|
// map for guard
|
||||||
Map<String, byte[]> runtime = new HashMap<>(256);
|
Map<String, byte[]> runtime = new HashMap<>(256);
|
||||||
if (server.buildHookManager.buildRuntime()) {
|
if (server.buildHookManager.buildRuntime()) {
|
||||||
// Write launcher guard dir
|
// Write launcher guard dir
|
||||||
IOHelper.walk(server.launcherBinary.runtimeDir, new RuntimeDirVisitor(output, runtime), false);
|
IOHelper.walk(server.launcherBinary.runtimeDir, new RuntimeDirVisitor(output, runtime, server.launcherBinary.runtimeDir, "runtime"), false);
|
||||||
IOHelper.walk(server.launcherBinary.guardDir, new GuardDirVisitor(output, runtime), false);
|
IOHelper.walk(server.launcherBinary.guardDir, new RuntimeDirVisitor(output, runtime, server.launcherBinary.guardDir, "guard"), false);
|
||||||
}
|
}
|
||||||
// Create launcher config file
|
// Create launcher config file
|
||||||
byte[] launcherConfigBytes;
|
byte[] launcherConfigBytes;
|
||||||
|
@ -229,18 +245,54 @@ public Path process(Path inputJar) throws IOException {
|
||||||
// Write launcher config file
|
// Write launcher config file
|
||||||
output.putNextEntry(newZipEntry(Launcher.CONFIG_FILE));
|
output.putNextEntry(newZipEntry(Launcher.CONFIG_FILE));
|
||||||
output.write(launcherConfigBytes);
|
output.write(launcherConfigBytes);
|
||||||
ZipEntry e = newZipEntry(zPath);
|
|
||||||
output.putNextEntry(e);
|
|
||||||
output.write(launcherConfigurator.getBytecode(reader));
|
|
||||||
|
|
||||||
e = newZipEntry(sPath);
|
|
||||||
output.putNextEntry(e);
|
|
||||||
output.write(secureConfigurator.getBytecode(reader));
|
|
||||||
}
|
}
|
||||||
reader.close();
|
reader.close();
|
||||||
return outputJar;
|
return outputJar;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public byte[] transformClass(byte[] bytes, String classname, BuildContext context)
|
||||||
|
{
|
||||||
|
byte[] result = bytes;
|
||||||
|
ClassReader cr = null;
|
||||||
|
ClassWriter writer = null;
|
||||||
|
ClassNode cn = null;
|
||||||
|
for(Transformer t : transformers)
|
||||||
|
{
|
||||||
|
if(t instanceof ASMTransformer)
|
||||||
|
{
|
||||||
|
ASMTransformer asmTransformer = (ASMTransformer) t;
|
||||||
|
if(cn == null)
|
||||||
|
{
|
||||||
|
cr = new ClassReader(result);
|
||||||
|
cn = new ClassNode();
|
||||||
|
cr.accept(cn, 0);
|
||||||
|
}
|
||||||
|
asmTransformer.transform(cn, classname, context);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
else if(cn != null)
|
||||||
|
{
|
||||||
|
writer = new ClassWriter(ClassWriter.COMPUTE_FRAMES | ClassWriter.COMPUTE_MAXS);
|
||||||
|
cn.accept(writer);
|
||||||
|
result = writer.toByteArray();
|
||||||
|
}
|
||||||
|
byte[] old_result = result;
|
||||||
|
result = t.transform(result, classname, context);
|
||||||
|
if(old_result != result)
|
||||||
|
{
|
||||||
|
cr = null;
|
||||||
|
cn = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(cn != null)
|
||||||
|
{
|
||||||
|
writer = new ClassWriter(ClassWriter.COMPUTE_FRAMES | ClassWriter.COMPUTE_MAXS);
|
||||||
|
cn.accept(writer);
|
||||||
|
result = writer.toByteArray();
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean allowDelete() {
|
public boolean allowDelete() {
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -27,13 +27,11 @@ public interface Transformer {
|
||||||
private final Set<Transformer> CLASS_TRANSFORMER;
|
private final Set<Transformer> CLASS_TRANSFORMER;
|
||||||
private final Set<String> CLASS_BLACKLIST;
|
private final Set<String> CLASS_BLACKLIST;
|
||||||
private final Set<String> MODULE_CLASS;
|
private final Set<String> MODULE_CLASS;
|
||||||
private final Map<String, byte[]> INCLUDE_CLASS;
|
|
||||||
|
|
||||||
public BuildHookManager() {
|
public BuildHookManager() {
|
||||||
HOOKS = new HashSet<>(4);
|
HOOKS = new HashSet<>(4);
|
||||||
CLASS_BLACKLIST = new HashSet<>(4);
|
CLASS_BLACKLIST = new HashSet<>(4);
|
||||||
MODULE_CLASS = new HashSet<>(4);
|
MODULE_CLASS = new HashSet<>(4);
|
||||||
INCLUDE_CLASS = new HashMap<>(4);
|
|
||||||
CLASS_TRANSFORMER = new HashSet<>(4);
|
CLASS_TRANSFORMER = new HashSet<>(4);
|
||||||
BUILDRUNTIME = true;
|
BUILDRUNTIME = true;
|
||||||
autoRegisterIgnoredClass(AutogenConfig.class.getName());
|
autoRegisterIgnoredClass(AutogenConfig.class.getName());
|
||||||
|
@ -56,14 +54,6 @@ public byte[] classTransform(byte[] clazz, String classname, MainBuildTask reade
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void registerIncludeClass(String classname, byte[] classdata) {
|
|
||||||
INCLUDE_CLASS.put(classname, classdata);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map<String, byte[]> getIncludeClass() {
|
|
||||||
return INCLUDE_CLASS;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isContainsBlacklist(String clazz) {
|
public boolean isContainsBlacklist(String clazz) {
|
||||||
for (String classB : CLASS_BLACKLIST) {
|
for (String classB : CLASS_BLACKLIST) {
|
||||||
if (clazz.startsWith(classB)) return true;
|
if (clazz.startsWith(classB)) return true;
|
||||||
|
|
Loading…
Reference in a new issue