Compile编译用户的应用。
bin/compile
入口是:bin/compile,该脚本和detect脚本很类似:需要一个构建目录实例化buildpack对象,并调用其compile接口。
注意:在这个脚本看似只有一个参数,但运行时实际需要第二个参数:应用缓存目录,当下载JDK,
compile方法
compile先调用component_detection,探测了对容器,JRE,framework的支持情况,并依次调用JRE的编译,每个框架的编译,和容器的编译。
def compile puts BUILDPACK_MESSAGE % @buildpack_version container = component_detection('container', @containers, true).first fail 'No container can run this application' unless container component_detection('JRE', @jres, true).first.compile component_detection('framework', @frameworks, false).each(&:compile) #调用每一个框架的编译 container.compile end
component_detection返回的是component,如JRE的component_detection返回的是JavaBuildpack::Jre::OpenJdkJRE。
JRE的编译
JRE编译调用的是JavaBuildpack::Jre::OpenJdkJRE的compile,而JavaBuildpack::Jre::OpenJdkJRE又是继承自OpenJDKLike,因此追溯到OpenJDKLike的compile
def compile download_tar @droplet.copy_resources end
可以看到编译就干了两件事:
下载Jdk的包,拷贝resources,即:拷贝resources/open_jdk_jre下面的文件
Tomcat的编译
容器的编译调用了JavaBuildpack::Container::Tomcat的compile,方法是定义在其父类:JavaBuildpack::Component::ModularComponent
def compile @sub_components.each(&:compile) end
即调用其子组件的编译,子组件包括:TomcatInstance,TomcatLifecycleSupport,TomcatLoggingSupport,TomcatAccessLoggingSupport,TomcatRedisStore,TomcatInsightSupport。
TomcatInstance的compile方法
该compile方法完成了三件事:
1. 下载tomcat的包;
2. 在tomcat的webapps/WEB-INF/ROOT中链接用户应用根目录;
3. 链接jar包到WEB-INF/lib
def compile download(@version, @uri) { |file| expand file } link_to(@application.root.children, root) @droplet.additional_libraries << tomcat_datasource_jar if tomcat_datasource_jar.exist? # 追加数组的意思 @droplet.additional_libraries.link_to web_inf_lib end
TomcatLifecycleSupport的compile方法
该方法只是下载了tomcat_lifecycle_support的jar包
def compile download_jar(jar_name, tomcat_lib) end
TomcatLoggingSupport,TomcatAccessLoggingSupport的compile方法都只是下载了相应的jar包
TomcatRedisStore的compile方法
先检查了是否需要redis做session共享中间件,接着下载了redis_store.jar包,并修改tomcat的conf/context.xml配置
def compile return unless supports? download_jar(jar_name, tomcat_lib) mutate_context end
总结
从源码来看,compile并非是编译java的源代码,而是准备应用运行的环境。