成都网站建设设计

将想法与焦点和您一起共享

如何进行源码分析Detect

本篇文章给大家分享的是有关如何进行源码分析Detect ,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

成都创新互联公司是一家集网站建设,宁河企业网站建设,宁河品牌网站建设,网站定制,宁河网站建设报价,网络营销,网络优化,宁河网站推广为一体的创新建站企业,帮助传统企业提升企业形象加强企业竞争力。可充分满足这一群体相比中小企业更为丰富、高端、多元的互联网需求。同时我们时刻保持专业、时尚、前沿,时刻以成就客户成长自我,坚持不断学习、思考、沉淀、净化自己,让我们为更多的企业打造出实用型网站。

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尚不支持。

以上就是如何进行源码分析Detect ,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注创新互联行业资讯频道。


当前名称:如何进行源码分析Detect
链接URL:http://chengdu.cdxwcx.cn/article/jhsopj.html