# # Copyright 2014 Chef Software, Inc. # # 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. # module Omnibus class Packager::DEB < Packager::Base id :deb setup do # Copy the full-stack installer into our scratch directory, accounting for # any excluded files. # # /opt/hamlet => /tmp/daj29013/opt/hamlet destination = File.join(staging_dir, project.install_dir) FileSyncer.sync(project.install_dir, destination, exclude: exclusions) # Copy over any user-specified extra package files. # # Files retain their relative paths inside the scratch directory, so # we need to grab the dirname of the file, create that directory, and # then copy the file into that directory. # # extra_package_file '/path/to/foo.txt' #=> /tmp/scratch/path/to/foo.txt project.extra_package_files.each do |file| parent = File.dirname(file) destination = File.join(staging_dir, parent) create_directory(destination) copy_file(file, destination) end # Create the Debain file directory create_directory(debian_dir) end build do # Render the Debian +control+ file write_control_file # Write the conffiles write_conffiles_file # Write the scripts write_scripts # Render the md5 sums write_md5_sums # Create the deb create_deb_file end # # @!group DSL methods # -------------------------------------------------- # # Set or return the vendor who made this package. # # @example # vendor "Seth Vargo " # # @param [String] val # the vendor who make this package # # @return [String] # the vendor who make this package # def vendor(val = NULL) if null?(val) @vendor || 'Omnibus ' else unless val.is_a?(String) raise InvalidValue.new(:vendor, 'be a String') end @vendor = val end end expose :vendor # # Set or return the license for this package. # # @example # license "Apache 2.0" # # @param [String] val # the license for this package # # @return [String] # the license for this package # def license(val = NULL) if null?(val) @license || 'unknown' else unless val.is_a?(String) raise InvalidValue.new(:license, 'be a String') end @license = val end end expose :license # # Set or return the priority for this package. # # @example # priority "extra" # # @param [String] val # the priority for this package # # @return [String] # the priority for this package # def priority(val = NULL) if null?(val) @priority || 'extra' else unless val.is_a?(String) raise InvalidValue.new(:priority, 'be a String') end @priority = val end end expose :priority # # Set or return the section for this package. # # @example # section "databases" # # @param [String] val # the section for this package # # @return [String] # the section for this package # def section(val = NULL) if null?(val) @section || 'misc' else unless val.is_a?(String) raise InvalidValue.new(:section, 'be a String') end @section = val end end expose :section # # @!endgroup # -------------------------------------------------- # # The name of the package to create. Note, this does **not** include the # extension. # def package_name "#{safe_base_package_name}_#{safe_version}-#{safe_build_iteration}_#{safe_architecture}.deb" end # # The path where Debian-specific files will live. # # @example # /var/.../chef-server_11.12.4/DEBIAN # # @return [String] # def debian_dir @debian_dir ||= File.join(staging_dir, 'DEBIAN') end # # Render a control file in +#{debian_dir}/control+ using the supplied ERB # template. # # @return [void] # def write_control_file render_template(resource_path('control.erb'), destination: File.join(debian_dir, 'control'), variables: { name: safe_base_package_name, version: safe_version, iteration: safe_build_iteration, vendor: vendor, license: license, architecture: safe_architecture, maintainer: project.maintainer, installed_size: package_size, homepage: project.homepage, description: project.description, priority: priority, section: section, conflicts: project.conflicts, replaces: project.replaces, dependencies: project.runtime_dependencies, } ) end # # Render the list of config files into the conffile. # # @return [void] # def write_conffiles_file return if project.config_files.empty? render_template(resource_path('conffiles.erb'), destination: File.join(debian_dir, 'conffiles'), variables: { config_files: project.config_files, } ) end # # Copy all scripts in {Project#package_scripts_path} to the control # directory of this repo. # # @return [void] # def write_scripts %w(preinst postinst prerm postrm).each do |script| path = File.join(project.package_scripts_path, script) if File.file?(path) log.debug(log_key) { "Adding script `#{script}' to `#{debian_dir}' from #{path}" } copy_file(path, debian_dir) log.debug(log_key) { "SCRIPT FILE: #{debian_dir}/#{script}" } FileUtils.chmod(0755, File.join(debian_dir, script)) end end end # # Generate a list of the md5 sums of every file in the package and write it # to +#{debian_dir}/control/md5sums+. # # @return [void] # def write_md5_sums path = "#{staging_dir}/**/*" hash = FileSyncer.glob(path).inject({}) do |hash, path| if File.file?(path) && !File.symlink?(path) relative_path = path.gsub("#{staging_dir}/", '') hash[relative_path] = digest(path, :md5) end hash end render_template(resource_path('md5sums.erb'), destination: File.join(debian_dir, 'md5sums'), variables: { md5sums: hash, } ) end # # Create the +.deb+ file, compressing at gzip level 9. The use of the # +fakeroot+ command is required so that the package is owned by # +root:root+, but the build user does not need to have sudo permissions. # # @return [void] # def create_deb_file log.info(log_key) { "Creating .deb file" } # Execute the build command Dir.chdir(Config.package_dir) do shellout!("fakeroot dpkg-deb -z9 -Zgzip -D --build #{staging_dir} #{package_name}") end end # # The size of this Debian package. This is dynamically calculated. # # @return [Fixnum] # def package_size @package_size ||= begin path = "#{project.install_dir}/**/*" total = FileSyncer.glob(path).inject(0) do |size, path| unless File.directory?(path) || File.symlink?(path) size += File.size(path) end size end # Per http://www.debian.org/doc/debian-policy/ch-controlfields.html, the # disk space is given as the integer value of the estimated installed # size in bytes, divided by 1024 and rounded up. total / 1024 end end # # Return the Debian-ready base package name, converting any invalid characters to # dashes (+-+). # # @return [String] # def safe_base_package_name if project.package_name =~ /\A[a-z0-9\.\+\-]+\z/ project.package_name.dup else converted = project.package_name.downcase.gsub(/[^a-z0-9\.\+\-]+/, '-') log.warn(log_key) do "The `name' component of Debian package names can only include " \ "lower case alphabetical characters (a-z), numbers (0-9), dots (.), " \ "plus signs (+), and dashes (-). Converting `#{project.package_name}' to " \ "`#{converted}'." end converted end end # # This is actually just the regular build_iteration, but it felt lonely # among all the other +safe_*+ methods. # # @return [String] # def safe_build_iteration project.build_iteration end # # Return the Debian-ready version, replacing all dashes (+-+) with tildes # (+~+) and converting any invalid characters to underscores (+_+). # # @return [String] # def safe_version version = project.build_version.dup if version =~ /\-/ converted = version.gsub('-', '~') log.warn(log_key) do "Dashes hold special significance in the Debian package versions. " \ "Versions that contain a dash and should be considered an earlier " \ "version (e.g. pre-releases) may actually be ordered as later " \ "(e.g. 12.0.0-rc.6 > 12.0.0). We'll work around this by replacing " \ "dashes (-) with tildes (~). Converting `#{project.build_version}' " \ "to `#{converted}'." end version = converted end if version =~ /\A[a-zA-Z0-9\.\+\:\~]+\z/ version else converted = version.gsub(/[^a-zA-Z0-9\.\+\:\~]+/, '_') log.warn(log_key) do "The `version' component of Debian package names can only include " \ "alphabetical characters (a-z, A-Z), numbers (0-9), dots (.), " \ "plus signs (+), dashes (-), tildes (~) and colons (:). Converting " \ "`#{project.build_version}' to `#{converted}'." end converted end end # # Debian does not follow the standards when naming 64-bit packages. # # @return [String] # def safe_architecture case Ohai['kernel']['machine'] when 'x86_64' 'amd64' when 'i686' 'i386' when /armv\dl/ if Ohai['platform'] == 'raspbian' || Ohai['platform'] == 'ubuntu' 'armhf' else Ohai['kernel']['machine'] end when 'aarch64' # Debian prefers amd64 on ARMv8/AArch64 (64bit ARM) platforms # see https://wiki.debian.org/Arm64Port 'arm64' when 'ppc64le' # Debian prefers to use ppc64el for little endian architecture name # where as others like gnutools/rhel use ppc64le( note the last 2 chars) # see http://linux.debian.ports.powerpc.narkive.com/8eeWSBtZ/switching-ppc64el-port-name-to-ppc64le 'ppc64el' #dpkg --print-architecture = ppc64el else Ohai['kernel']['machine'] end end end end