Compile编译用户的应用。
入口是:bin/compile,该脚本和detect脚本很类似:需要一个构建目录实例化buildpack对象,并调用其compile接口。
注意:在这个脚本看似只有一个参数,但运行时实际需要第二个参数:应用缓存目录,当下载JDK,
compile先调用component_detection,探测了对容器,JRE,framework的支持情况,并依次调用JRE的编译,每个框架的编译,和容器的编译。
class="ruby" name="code">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编译调用的是JavaBuildpack::Jre::OpenJdkJRE的compile,而JavaBuildpack::Jre::OpenJdkJRE又是继承自OpenJDKLike,因此追溯到OpenJDKLike的compile
def compile download_tar @droplet.copy_resources end
?可以看到编译就干了两件事:
下载Jdk的包,拷贝resources,即:拷贝resources/open_jdk_jre下面的文件
容器的编译调用了JavaBuildpack::Container::Tomcat的compile,方法是定义在其父类:JavaBuildpack::Component::ModularComponent
def compile @sub_components.each(&:compile) end
?即调用其子组件的编译,子组件包括:TomcatInstance,TomcatLifecycleSupport,TomcatLoggingSupport,TomcatAccessLoggingSupport,TomcatRedisStore,TomcatInsightSupport。
该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
?
该方法只是下载了tomcat_lifecycle_support的jar包
def compile download_jar(jar_name, tomcat_lib) end
?TomcatLoggingSupport,TomcatAccessLoggingSupport的compile方法都只是下载了相应的jar包
先检查了是否需要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的源代码,而是准备应用运行的环境。