java-buildpack源码分析之Detect
2015-10-27 00:00
411 查看
摘要: java-buildpack源码分析之Detect
由该文件可以知道该buidpack支持的容器,JRE和框架。当然你可以自己添加。
detect的入口是bin/detect,该脚本非常简单,调用JavaBuildpack的with_buildpack静态方法创建实例,然后在代码块里调用JavaBuildpack.detect方法。最后将detect的结果组成一个字符串输出,如果探测不对,则什么也不输出。
接下来看JavaBuildpack::Buildpack类,在buildpack.rb中。
with_buildpack方法是在class << self中的,因此它是一个静态方法,在看方法定义:
detect方法
detect方法分别对应用使用的容器,JRE,框架探测
detection方法
调用每个组件的detect方法,这些其实是子组件(如:容器大项中的tomcat子项),并将结果集合返回。
具体项目的detect方法
这里重点关注下tomcat的detect方法,tomcat.rb
tomcat的类定义:
class Tomcat < JavaBuildpack::Component::ModularComponent
初始化方法
可以知道Tomcat是继承自ModularComponent,detect方法是在ModularComponent中定义的,
tomcat的support?方法
再看子项目有哪些?
TomcatInstance的detect
先从配置文件config/tomcat.yml中找到tomcat的版本及下载路径
然后从repository_root(默认是:https://download.run.pivotal.io/) 下载tomcat/index.yml,该文件包含了所有版本的tomcat的下载路径,找到具体的版本,如:8.0+指,8.0以上的版本,找到了8.0.14。
需要说明的是:
tomcat支持之后,TomcatInstance是默认支持的,因为tomcat instance就是基本的tomcat包, TomcatLifecycleSupport,TomcatLoggingSupport,TomcatAccessLoggingSupport也是默认支持的。
如果需要会话共享,复制,设置了service环境变量,TomcatRedisStore则是支持的。
而TomcatInsightSupport尚不支持。
Detect
该buildpack的探测的内容包含:容器,JRE,框架。具体内容在components.yml中可以看到:# Configuration for components to use in the buildpack --- containers: - "JavaBuildpack::Container::DistZip" - "JavaBuildpack::Container::Groovy" - "JavaBuildpack::Container::JavaMain" - "JavaBuildpack::Container::PlayFramework" - "JavaBuildpack::Container::Ratpack" - "JavaBuildpack::Container::SpringBoot" - "JavaBuildpack::Container::SpringBootCLI" - "JavaBuildpack::Container::Tomcat" # In order to use Oracle JREs instead of OpenJDK, you must comment out the OpenJDK line and uncomment the Oracle line. # Please see the documentation for more detail. jres: - "JavaBuildpack::Jre::OpenJdkJRE" # - "JavaBuildpack::Jre::OracleJRE" frameworks: - "JavaBuildpack::Framework::AppDynamicsAgent" - "JavaBuildpack::Framework::JavaOpts" - "JavaBuildpack::Framework::MariaDbJDBC" - "JavaBuildpack::Framework::NewRelicAgent" - "JavaBuildpack::Framework::PlayFrameworkAutoReconfiguration" - "JavaBuildpack::Framework::PlayFrameworkJPAPlugin" - "JavaBuildpack::Framework::PostgresqlJDBC" - "JavaBuildpack::Framework::SpringAutoReconfiguration" - "JavaBuildpack::Framework::SpringInsight"
由该文件可以知道该buidpack支持的容器,JRE和框架。当然你可以自己添加。
detect的入口是bin/detect,该脚本非常简单,调用JavaBuildpack的with_buildpack静态方法创建实例,然后在代码块里调用JavaBuildpack.detect方法。最后将detect的结果组成一个字符串输出,如果探测不对,则什么也不输出。
components = JavaBuildpack::Buildpack.with_buildpack(build_dir, 'Detect failed with exception %s') do |buildpack| # 这里with_buildpack创建了一个JavaBuildpack::Buildpack的实例,查看buildpack.rb可以看到 buildpack.detect end.compact
接下来看JavaBuildpack::Buildpack类,在buildpack.rb中。
with_buildpack方法是在class << self中的,因此它是一个静态方法,在看方法定义:
def with_buildpack(app_dir, message) app_dir = Pathname.new(File.expand_path(app_dir)) # app_dir变成一个Pathname实例 application = Component::Application.new(app_dir) # 创建application实例 Logging::LoggerFactory.instance.setup app_dir # 初始化日志 yield new(app_dir, application) if block_given? # **创建一个对象,并且调用外部给的代码块** rescue => e handle_error(e, message) end特别注意其中注释加**的那句,语法比较绕
detect方法
detect方法分别对应用使用的容器,JRE,框架探测
def detect tags = tag_detection('container', @containers, true) tags.concat tag_detection('JRE', @jres, true) unless tags.empty? # 如果不为空则连接tags tags.concat tag_detection('framework', @frameworks, false) unless tags.empty? tags << "java-buildpack=#{@buildpack_version.to_s false}" unless tags.empty? # 加上buildpack版本 tags = tags.flatten.compact.sort @logger.debug { "Detection Tags: #{tags}" } tags end
detection方法
调用每个组件的detect方法,这些其实是子组件(如:容器大项中的tomcat子项),并将结果集合返回。
def detection(type, components, unique) detected = [] tags = [] components.each do |component| result = component.detect next unless result # 如果结果不为空则跳出循环 detected << component tags << result end fail "Application can be run by more than one #{type}: #{names detected}" if unique && detected.size > 1 [detected, tags] end
具体项目的detect方法
这里重点关注下tomcat的detect方法,tomcat.rb
tomcat的类定义:
class Tomcat < JavaBuildpack::Component::ModularComponent
初始化方法
def initialize(context, &version_validator) super(context, &version_validator) @sub_components = supports? ? sub_components(context) : [] # tomcat支持吗?,如果支持,初始化子项目,注意:这里两个问号,前一个是方法support?后一个是问号表达式 end
可以知道Tomcat是继承自ModularComponent,detect方法是在ModularComponent中定义的,
def detect supports? ? @sub_components.map(&:detect).flatten.compact : nil # 如果支持探测子项目 end
tomcat的support?方法
def supports? web_inf? && !JavaBuildpack::Util::JavaMainUtils.main_class(@application) # WEB-INF目录存在,且不存在Main方法 end private def web_inf? (@application.root + 'WEB-INF').exist? # 其实检查的是,WEB-INF目录是否存在 end
再看子项目有哪些?
def sub_components(context) [ TomcatInstance.new(sub_configuration_context(context, 'tomcat')), TomcatLifecycleSupport.new(sub_configuration_context(context, 'lifecycle_support')), TomcatLoggingSupport.new(sub_configuration_context(context, 'logging_support')), TomcatAccessLoggingSupport.new(sub_configuration_context(context, 'access_logging_support')), TomcatRedisStore.new(sub_configuration_context(context, 'redis_store')), TomcatInsightSupport.new(context) ] end
TomcatInstance的detect
先从配置文件config/tomcat.yml中找到tomcat的版本及下载路径
tomcat: version: 8.0.+ repository_root: "{default.repository.root}/tomcat"
然后从repository_root(默认是:https://download.run.pivotal.io/) 下载tomcat/index.yml,该文件包含了所有版本的tomcat的下载路径,找到具体的版本,如:8.0+指,8.0以上的版本,找到了8.0.14。
需要说明的是:
tomcat支持之后,TomcatInstance是默认支持的,因为tomcat instance就是基本的tomcat包, TomcatLifecycleSupport,TomcatLoggingSupport,TomcatAccessLoggingSupport也是默认支持的。
如果需要会话共享,复制,设置了service环境变量,TomcatRedisStore则是支持的。
而TomcatInsightSupport尚不支持。
相关文章推荐
- PHP strtotime函数用法、实现原理和源码分析
- jQuery 源码分析笔记(3) Deferred机制
- jQuery 源码分析笔记(5) jQuery.support
- php自动获取字符串编码函数mb_detect_encoding
- Cocos2d-x学习笔记之Hello World源码分析
- PHP源码分析之变量的存储过程分解
- opencv-python学习一--人脸检测
- Glusterfs3.3.1DHT(hash分布)源代码分析
- 开源中国 OsChina Android 客户端源码分析(1)启动界面 app_start
- 【原创】OpenStack Swift源码分析(二)ring文件的生成
- zg手册 之 python2.7.7源码分析(1)-- python中的对象
- Struts2请求处理流程及源码分析
- libevent源码分析之 tail queue
- Android Touch事件分发过程
- WinCE 6.0启动过程源码分析
- Geekos源码分析专题:键盘处理
- 剖析easyui tree源码(设计原理)来学习easyui
- Linux内核0.11版 lib/Malloc.c 文件分析
- 从子线程不能直接新建一个Handler对象来剖析android的Handler机制