CF运行应用的基本过程是将用户发布的应用程序包解压开,而后将本身的全部buildpack拿来,按照指定顺序与程序包进行匹配,直到找到第一个可以运行这些代码的buildpack,而后将buildpack也解开,与这些应用代码打成一个包(即droplet),在按照指定的运行环境参数生成容器,将droplet扔进去,按照buildpack指定的启动命令,启动应用。在上面的过程当中,buildpack实现了三步功能:
第一步,detect:检查当前应用程序包是否可以用本buildpack支持运行,好比,java buildpack发现WEB-INF路径就认为本身可以运行它。
第二步,compile:将应用程序包与buildpack包水乳交融一下,好比将java程序包放到tomcat的应用目录下,而后替换某些参数,好比将当前dea里的随机端口赋予这个tomcat实例。
第三部,release:将droplet启动,好比运行tomcat的startup.sh。
任何一个buildpack都有一个bin路径,放着三个指定名字(detect、compile、release)的脚本(任何dea的os能执行的脚本均可以),而后具体的实现逻辑就从这里触发了。下面将以java buildpack为例,经过三个实际需求,介绍buildpack的开发和使用。java
国际软件都使用格林尼治时间做为系统中的默认时间,因此我们写出的日志或者使用程序代码获取的系统时间(其实都是dea的系统时间)都是+0000的,而咱们须要+0800的,固然,能够从代码里自行处理。可是个人用户们不爽,他们说我这程序还要本地测视呢,太乱了,你得给我处理。
tomcat里的默认编码应该都是utf8,但是,在某些场景下,好比tomcat上运行的rest服务,其报文中包含中文,是没法被正常接收的。CF的java buildpack里包含的就是默认的tomcat,因此为了解决这个问题,可使用下面的命令进行设置,可是毕竟不大爽,仍是直接改掉buildpack比较友好。linux
cf set-env appname CATALINA_OPTS "$CATALINA_OPTS -Dfile.encoding=UTF-8"
java buildpack是ruby写的,因此若是不是原则性的上的变更,大能够经过将压缩包打开,修改ruby代码,而后再压缩上的办法。不过,winrar彷佛有些莫名其妙的问题,若是是解压开再压缩会有问题,可是若是直接将修改后的同名文件拖动到winrar的窗口中进行替换,就能够工做。为了达到时区和编码这两个需求,须要作的文件修改以下:git
在java-buildpack-offline-v#.#.#\lib\java_buildpack\component\java_opts.rb
中增长两个方法:github
# @return [JavaOpts] +self+-Duser.timezone def add_timezone(value) self << "-Duser.timezone=#{value}" self end # @return [JavaOpts] +self+-Dfile.encoding def add_fileencode(value) self << "-Dfile.encoding=#{value}" self end
在java-buildpack-offline-v#.#.#\lib\java_buildpack\jre\open_jdk_like.rb中,修改release方法,增长add_timezone和add_fileencode调用express
@droplet.java_opts .add_system_property('java.io.tmpdir', '$TMPDIR') .add_option('-XX:OnOutOfMemoryError', killjava) .add_timezone('GMT+08') .add_fileencode('UTF-8') .concat memory
修改后,将这两个文件替换到原来的离线buildpack包里,而后将其发布到CF中,查看当前buildpack状况:apache
cf buidlpacks
而后,建立新的buildpack,其中最后一个参数决定了cf查看buildpack符合应用的顺序,须要根据实际状况调整。windows
cf create-buildpack java_buildpack_with_gmt0800_offline d:\somedir\java-buildpack-with-gmt0800-offline-v2.4.zip 1
若是是更新的话,使用cf update-buildpack
。centos
有些应用,尤为是作产品的公司作的,会对tomcat进行必定的剪裁,这样的应用迁移到cf上时,很是痛苦。简单的办法就是,本身作个buildpack,提供的只是jre。tomcat
假定这个产品的标识是LiveBOS,在程序包的第一层路径下就有一个LiveBOS文件夹:ruby
在java-buildpack-with-livebos-offline-v#.#\lib\java_buildpack\container\里添加一个libe_bos.rb:
# Encoding: utf-8 # Cloud Foundry Java Buildpack # Copyright 2013 the original author or authors. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. require 'java_buildpack/container' require 'java_buildpack/container/dist_zip_like' require 'java_buildpack/util/dash_case' module JavaBuildpack module Container # Encapsulates the detect, compile, and release functionality for +LiveBOS+ style applications. class LiveBOS < JavaBuildpack::Container::DistZipLike # Creates an instance # # @param [Hash] context a collection of utilities used the component def initialize(context) super(context) end # (see JavaBuildpack::Component::BaseComponent#compile) def compile startup.chmod 0755 end # (see JavaBuildpack::Component::BaseComponent#release) def release @droplet.java_opts.add_system_property 'http.port', '$PORT' [ @droplet.java_home.as_env_var, @droplet.java_opts.as_env_var, qualify_path(catalina, @droplet.root), 'run' ].flatten.compact.join(' ') end protected # (see JavaBuildpack::Container::DistZipLike#id) def id LiveBOS.to_s.dash_case end # (see JavaBuildpack::Container::DistZipLike#supports?) def supports? livebos? && catalina.exist? end private def startup candidates = (root + 'bin/startup.sh').glob candidates.size == 1 ? candidates.first : nil end def catalina candidates = (root + 'bin/catalina.sh').glob candidates.size == 1 ? candidates.first : nil end def livebos? (root + 'LiveBos').exist? end end end end
在java-buildpack-with-livebos-offline-v2.4\config\components.yml的containers下面添加:
- "JavaBuildpack::Container::LiveBOS"
后面就是打包发布了。
有些应用只能使用特定版本的jre,若是这jre比较新还好办,可是若是是旧的,好比1.6,那如今cf中能找到的buildpack就全不支持了。另外,也有可能应用要求使用oracle jre(cf中默认是open jre)。
开始以前,建议你们不要在windows上尝试这个编译过程,基本没有成功的可能,随便找个能上网能运行ruby的linux吧。
修改java-buildpack-offline-v2.4\config\components.yml,将不使用的jre注释掉:
- "JavaBuildpack::Jre::OpenJdkJRE" # - "JavaBuildpack::Jre::OracleJRE"
修改java-buildpack-offline-v2.4\config\oracle_jre.yml,将下面这行里的内容更新为一个本地tomcat的地址:
repository_root: http://localhost:8080/myapp
在本地的myapp下放一个index.yml文件,其中放上相似下列内容的oracle jre下载地址,若是oracle网站上已经找不到了,就也放在本地tomcat上吧。
--- 1.7.0_01: http://download.run.pivotal.io/openjdk/centos6/x86_64/openjdk-1.7.0_01.tar.gz ...... 1.8.0_M6: http://download.run.pivotal.io/openjdk/centos6/x86_64/openjdk-1.8.0_M6.tar.gz
而后,就能够编译打包了。
bundle install bundle exec rake package OFFLINE=true ... Creating build/java-buildpack-offline-cfd6b17.zip
最后,固然仍是发布到cf上。