入口是:bin/compile,該腳本和detect腳本很相似:須要一個構建目錄實例化buildpack對象,並調用其compile接口。java
注意:在這個腳本看似只有一個參數,但運行時實際須要第二個參數:應用緩存目錄,當下載JDK,web
compilempilecompile先調用component_detection,探測了對容器,JRE,framework的支持狀況,並依次調用JRE的編譯,每一個框架的編譯,和容器的編譯。redis
def compile 緩存
puts BUILDPACK_MESSAGE % @buildpack_version tomcat
container = component_detection('container', @containers, true).first session
fail 'No container can run this application' unless container app
component_detection('JRE', @jres, true).first.compile 框架
component_detection('framework', @frameworks, false).each(&:compile) #調用每個框架的編譯 less
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的源代碼,而是準備應用運行的環境。