Feat:动态代码编译引擎成熟!

This commit is contained in:
wangyu 2021-01-04 23:12:16 +08:00
parent 392b1ebc09
commit 4bfc72dcea
13 changed files with 305 additions and 40 deletions

View File

@ -59,10 +59,6 @@
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
</dependency>
<dependency>
<groupId>com.itranswarp</groupId>
<artifactId>compiler</artifactId>
</dependency>
<dependency>
<groupId>org.reflections</groupId>
<artifactId>reflections</artifactId>

View File

@ -3,6 +3,7 @@ package com.flyfish.framework.compiler;
import com.flyfish.framework.compiler.support.DelegateJavaCompiler;
import com.flyfish.framework.compiler.support.SimpleJavaCompiler;
import java.io.Closeable;
import java.io.IOException;
/**
@ -11,10 +12,11 @@ import java.io.IOException;
*
* @author wangyu
*/
public interface DynamicJavaCompiler {
public interface DynamicJavaCompiler extends Closeable {
/**
* 代理的java编译器使用库文件
*
* @return 结果
*/
static DynamicJavaCompiler delegate() {

View File

@ -0,0 +1,82 @@
package com.flyfish.framework.compiler.core;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileObject;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import java.io.Closeable;
import java.io.IOException;
import java.util.Collections;
import java.util.Map;
/**
* In-memory compile Java source code as String.
*
* @author michael
*/
public class JavaStringCompiler implements Closeable {
private final JavaCompiler compiler;
private final MemoryJavaFileManager manager;
private final MemoryClassLoader classLoader;
public JavaStringCompiler() {
this.compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager stdManager = compiler.getStandardFileManager(null, null, null);
this.manager = new MemoryJavaFileManager(stdManager);
this.classLoader = new MemoryClassLoader(manager.getClassBytes());
}
/**
* Compile a Java source file in memory.
*
* @param fileName Java file name, e.g. "Test.java"
* @param source The source code as String.
* @return The compiled results as Map that contains class name as key,
* class binary as value.
* @throws IOException If compile error.
*/
public Map<String, byte[]> compile(String fileName, String source) {
JavaFileObject javaFileObject = manager.makeStringSource(fileName, source);
JavaCompiler.CompilationTask task = compiler.getTask(null, manager, null, null, null, Collections.singletonList(javaFileObject));
Boolean result = task.call();
if (result == null || !result) {
throw new RuntimeException("Compilation failed.");
}
return manager.getClassBytes();
}
/**
* Load class from compiled classes.
*
* @param name Full class name.
* @return The Class instance.
* @throws ClassNotFoundException If class not found.
*/
public Class<?> loadClass(String name) throws ClassNotFoundException {
return classLoader.loadClass(name);
}
/**
* get the classLoader
*
* @return 结果
*/
public ClassLoader getClassLoader() {
return classLoader;
}
@Override
public void close() {
try {
if (null != manager) {
manager.close();
}
if (null != classLoader) {
classLoader.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}

View File

@ -0,0 +1,32 @@
package com.flyfish.framework.compiler.core;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Map;
/**
* Load class from byte[] which is compiled in memory.
*
* @author michael
*/
class MemoryClassLoader extends URLClassLoader {
// class name to class bytes:
private final Map<String, byte[]> classBytes;
public MemoryClassLoader(Map<String, byte[]> classBytes) {
super(new URL[0], MemoryClassLoader.class.getClassLoader());
this.classBytes = classBytes;
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
byte[] buf = classBytes.get(name);
if (buf == null) {
return super.findClass(name);
}
classBytes.remove(name);
return defineClass(name, buf, 0, buf.length);
}
}

View File

@ -0,0 +1,91 @@
package com.flyfish.framework.compiler.core;
import javax.tools.*;
import java.io.ByteArrayOutputStream;
import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URI;
import java.nio.CharBuffer;
import java.util.HashMap;
import java.util.Map;
/**
* In-memory java file manager.
*
* @author michael
*/
class MemoryJavaFileManager extends ForwardingJavaFileManager<JavaFileManager> {
// compiled classes in bytes:
final Map<String, byte[]> classBytes = new HashMap<>();
MemoryJavaFileManager(JavaFileManager fileManager) {
super(fileManager);
}
public Map<String, byte[]> getClassBytes() {
return this.classBytes;
}
@Override
public void flush() {
}
@Override
public void close() throws IOException {
classBytes.clear();
}
@Override
public JavaFileObject getJavaFileForOutput(JavaFileManager.Location location, String className, JavaFileObject.Kind kind,
FileObject sibling) throws IOException {
if (kind == JavaFileObject.Kind.CLASS) {
return new MemoryOutputJavaFileObject(className);
} else {
return super.getJavaFileForOutput(location, className, kind, sibling);
}
}
JavaFileObject makeStringSource(String name, String code) {
return new MemoryInputJavaFileObject(name, code);
}
static class MemoryInputJavaFileObject extends SimpleJavaFileObject {
final String code;
MemoryInputJavaFileObject(String name, String code) {
super(URI.create("string:///" + name), Kind.SOURCE);
this.code = code;
}
@Override
public CharBuffer getCharContent(boolean ignoreEncodingErrors) {
return CharBuffer.wrap(code);
}
}
class MemoryOutputJavaFileObject extends SimpleJavaFileObject {
final String name;
MemoryOutputJavaFileObject(String name) {
super(URI.create("string:///" + name), Kind.CLASS);
this.name = name;
}
@Override
public OutputStream openOutputStream() {
return new FilterOutputStream(new ByteArrayOutputStream()) {
@Override
public void close() throws IOException {
out.close();
ByteArrayOutputStream bos = (ByteArrayOutputStream) out;
classBytes.put(name, bos.toByteArray());
}
};
}
}
}

View File

@ -1,7 +1,7 @@
package com.flyfish.framework.compiler.support;
import com.flyfish.framework.compiler.DynamicJavaCompiler;
import com.itranswarp.compiler.JavaStringCompiler;
import com.flyfish.framework.compiler.core.JavaStringCompiler;
import org.apache.commons.lang3.StringUtils;
import org.springframework.util.StreamUtils;
@ -19,7 +19,6 @@ public class DelegateJavaCompiler implements DynamicJavaCompiler {
// 第三方编译器
private final JavaStringCompiler compiler = new JavaStringCompiler();
private DelegateJavaCompiler() {
}
@ -27,7 +26,9 @@ public class DelegateJavaCompiler implements DynamicJavaCompiler {
return SingletonHolder.INSTANCE;
}
public static ClassLoader classLoader() {
return getInstance().compiler.getClassLoader();
}
/**
* 编译java源码通过字符串
@ -37,22 +38,26 @@ public class DelegateJavaCompiler implements DynamicJavaCompiler {
* @return 结果
*/
@Override
public Class<?> compile(String name, String source) throws IOException, ClassNotFoundException {
public Class<?> compile(String name, String source) throws ClassNotFoundException {
Map<String, byte[]> byteMap = compiler.compile(name, source);
String className = StringUtils.substringBeforeLast(name, ".");
for (String key : byteMap.keySet()) {
if (key.contains(className)) {
return compiler.loadClass(key, byteMap);
return compiler.loadClass(key);
}
}
throw new RuntimeException("【java编译器】源码中没有可以编译的类");
}
@Override
public void close() {
compiler.close();
}
private static class SingletonHolder {
private static final DelegateJavaCompiler INSTANCE = new DelegateJavaCompiler();
}

View File

@ -63,6 +63,11 @@ public class SimpleJavaCompiler implements DynamicJavaCompiler {
throw new RuntimeException("【java编译器】尝试编译源代码出现异常");
}
@Override
public void close() throws IOException {
}
private static class SingletonHolder {
private static final SimpleJavaCompiler INSTANCE = new SimpleJavaCompiler();

View File

@ -32,6 +32,7 @@ public class CustomRepositoryRegistrar {
// 构建器
this.builder = new CustomRepositoryBuilder(registry, extension,
configurationSource, resourceLoader, environment);
}
/**
@ -50,6 +51,7 @@ public class CustomRepositoryRegistrar {
extension.postProcess(definitionBuilder, configurationSource);
extension.postProcess(definitionBuilder, (AnnotationRepositoryConfigurationSource) configurationSource);
AbstractBeanDefinition beanDefinition = definitionBuilder.getBeanDefinition();
beanDefinition.setPrimary(configuration.isPrimary());
String beanName = StringUtils.uncapitalize(clazz.getSimpleName());

View File

@ -1,10 +1,52 @@
package com.flyfish.framework.beans.resolver;
import com.flyfish.framework.compiler.DynamicJavaCompiler;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.annotation.Lazy;
import org.springframework.web.reactive.result.method.AbstractHandlerMethodMapping;
import org.springframework.web.reactive.result.method.annotation.RequestMappingHandlerMapping;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
/**
* 动态Bean生成解析器
*
* @author wangyu
*/
public class DynamicRestBeanResolver {
@Slf4j
@Lazy(false)
public class DynamicRestBeanResolver implements InitializingBean, DisposableBean {
private static final List<String> controllers = new ArrayList<>();
private final Method handlerRegister;
private final RequestMappingHandlerMapping mapping;
public DynamicRestBeanResolver(RequestMappingHandlerMapping mapping) throws NoSuchMethodException {
this.mapping = mapping;
this.handlerRegister = AbstractHandlerMethodMapping.class.getDeclaredMethod("detectHandlerMethods", Object.class);
}
public static void addController(String controller) {
controllers.add(controller);
}
@Override
public void destroy() throws Exception {
log.info("正在销毁rest自动配置器");
DynamicJavaCompiler.delegate().close();
}
@Override
public void afterPropertiesSet() throws Exception {
log.info("正在注册controller...");
for (String bean : controllers) {
log.info("注册controller{}", bean);
handlerRegister.setAccessible(true);
handlerRegister.invoke(mapping, bean);
}
}
}

View File

@ -0,0 +1,21 @@
package com.flyfish.framework.boot;
import com.flyfish.framework.compiler.support.DelegateJavaCompiler;
import org.springframework.boot.SpringApplication;
import org.springframework.context.ConfigurableApplicationContext;
/**
* 启动类包装
*
* @author wangyu
* 实现了自定义的类加载器
*/
public class FlyfishAppRunner {
public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {
Thread.currentThread().setContextClassLoader(DelegateJavaCompiler.classLoader());
return SpringApplication.run(new Class[]{primarySource}, args);
}
}

View File

@ -1,10 +1,14 @@
package com.flyfish.framework.config;
import com.flyfish.framework.beans.resolver.DynamicRestBeanResolver;
import org.springframework.boot.autoconfigure.web.ServerProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.web.reactive.result.method.annotation.RequestMappingHandlerMapping;
import org.springframework.web.server.WebFilter;
import java.util.ArrayList;
/**
* 默认初始化的Bean配置
*
@ -33,4 +37,11 @@ public class BeanConfig {
};
}
@Bean
public DynamicRestBeanResolver dynamicRestBeanResolver(RequestMappingHandlerMapping requestMappingHandlerMapping)
throws NoSuchMethodException {
return new DynamicRestBeanResolver(requestMappingHandlerMapping);
}
}

View File

@ -3,6 +3,7 @@ package com.flyfish.framework.config;
import com.flyfish.framework.beans.annotations.EnableRestBeanDetect;
import com.flyfish.framework.beans.annotations.RestBean;
import com.flyfish.framework.beans.repository.CustomRepositoryRegistrar;
import com.flyfish.framework.beans.resolver.DynamicRestBeanResolver;
import com.flyfish.framework.compiler.DynamicJavaCompiler;
import com.flyfish.framework.compiler.support.JavaSource;
import com.flyfish.framework.compiler.template.TemplateCompiler;
@ -11,7 +12,6 @@ import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.reflections.Reflections;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
@ -22,13 +22,8 @@ import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;
import org.springframework.core.env.Environment;
import org.springframework.core.io.ResourceLoader;
import org.springframework.core.type.AnnotationMetadata;
import org.springframework.web.reactive.result.method.AbstractHandlerMethodMapping;
import org.springframework.web.reactive.result.method.annotation.RequestMappingHandlerMapping;
import javax.annotation.Resource;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function;
@ -41,24 +36,19 @@ import java.util.stream.Stream;
* @author wangyu
*/
@Slf4j
public class RestBeanAutoConfigure implements ImportBeanDefinitionRegistrar, ResourceLoaderAware, EnvironmentAware, InitializingBean {
public class RestBeanAutoConfigure implements ImportBeanDefinitionRegistrar, ResourceLoaderAware, EnvironmentAware {
// 预编译模板
private final Map<String, Function<JavaSource, String>> templates = Stream.of("Repository", "Service", "Controller")
.collect(Collectors.toMap(key -> key, filename -> TemplateCompiler.compile("/templates/" + filename + ".tpl")));
private final Map<String, Function<RestBean, Class<?>>> superClasses;
private final Method handlerRegister;
private final List<String> controllers = new ArrayList<>();
@Resource
private RequestMappingHandlerMapping requestMappingHandlerMapping;
// 资源加载器
private ResourceLoader resourceLoader;
// 环境变量
private Environment environment;
public RestBeanAutoConfigure() throws NoSuchMethodException {
this.handlerRegister = AbstractHandlerMethodMapping.class.getDeclaredMethod("detectHandlerMethods", Object.class);
public RestBeanAutoConfigure() {
superClasses = new HashMap<>();
superClasses.put("Controller", RestBean::controllerClass);
superClasses.put("Repository", RestBean::repoClass);
@ -138,7 +128,7 @@ public class RestBeanAutoConfigure implements ImportBeanDefinitionRegistrar, Res
registry.registerBeanDefinition(beanName, builder.getBeanDefinition());
// 如果是controller注册
if (clazz.getSimpleName().endsWith("Controller")) {
controllers.add(beanName);
DynamicRestBeanResolver.addController(beanName);
}
}
@ -160,15 +150,6 @@ public class RestBeanAutoConfigure implements ImportBeanDefinitionRegistrar, Res
}
}
@Override
public void afterPropertiesSet() throws Exception {
for (String bean : controllers) {
log.info("注册controller{}", bean);
handlerRegister.setAccessible(true);
handlerRegister.invoke(requestMappingHandlerMapping, bean);
}
}
@Override
public void setResourceLoader(ResourceLoader resourceLoader) {
this.resourceLoader = resourceLoader;

View File

@ -84,11 +84,6 @@
<artifactId>commons-io</artifactId>
<version>2.6</version>
</dependency>
<dependency>
<groupId>com.itranswarp</groupId>
<artifactId>compiler</artifactId>
<version>1.0</version>
</dependency>
<dependency>
<groupId>org.reflections</groupId>
<artifactId>reflections</artifactId>