inspec/test/docker_run.rb

160 lines
4 KiB
Ruby
Raw Normal View History

require 'docker'
require 'yaml'
require 'concurrent'
class DockerRunner
def initialize(conf_path = nil)
@conf_path = conf_path ||
ENV['config']
docker_run_concurrency = (ENV['N'] || 5).to_i
if @conf_path.nil?
fail "You must provide a configuration file with docker boxes"
end
unless File.file?(@conf_path)
fail "Can't find configuration in #{@conf_path}"
end
@conf = YAML.load_file(@conf_path)
if @conf.nil? or @conf.empty?
fail "Can't read configuration in #{@conf_path}"
end
if @conf['images'].nil?
fail "You must configure test images in your #{@conf_path}"
end
@images = docker_images_by_tag
@image_pull_tickets = Concurrent::Semaphore.new(2)
@docker_run_tickets = Concurrent::Semaphore.new(docker_run_concurrency)
end
def run_all(&block)
fail 'You must provide a block for run_all' unless block_given?
promises = @conf['images'].map do |id|
run_on_target(id, &block)
end
# wait for all tests to be finished
sleep(0.1) until promises.all?(&:fulfilled?)
# return resulting values
promises.map(&:value)
end
def run_on_target(name, &block)
pr = Concurrent::Promise.new {
begin
container = start_container(name)
res = block.call(name, container)
# special rescue block to handle not implemented error
rescue NotImplementedError => err
stop_container(container)
raise err.message + "\n" + err.backtrace.join("\n")
rescue StandardError => err
stop_container(container)
raise err.message + "\n" + err.backtrace.join("\n")
end
# always stop the container
stop_container(container)
res
}.execute
# failure handling
pr.rescue do |err|
msg = "\033[31;1m#{err.message}\033[0m"
puts msg
msg + "\n" + err.backtrace.join("\n")
end
end
def provision_image(image, prov, files)
return image if prov['script'].nil?
path = File.join(File.dirname(@conf_path), prov['script'])
unless File.file?(path)
puts "Can't find script file #{path}"
return image
end
puts " script #{path}"
dst = "/bootstrap#{files.length}.sh"
files.push(dst)
image.insert_local('localPath' => path, 'outputPath' => dst)
end
def bootstrap_image(name, image)
files = []
provisions = Array(@conf['provision'])
puts "--> provision docker #{name}" unless provisions.empty?
provisions.each do |prov|
image = provision_image(image, prov, files)
end
[image, files]
end
def start_container(name, version = nil)
unless name.include?(':')
version ||= 'latest'
name = "#{name}:#{version}"
end
puts "--> schedule docker #{name}"
image = @images[name]
if image.nil?
puts "\033[35;1m--> pull docker images #{name} "\
"(this may take a while)\033[0m"
@image_pull_tickets.acquire(1)
puts "... start pull image #{name}"
image = Docker::Image.create('fromImage' => name)
@image_pull_tickets.release(1)
unless image.nil?
puts "\033[35;1m--> pull docker images finished for #{name}\033[0m"
end
end
fail "Can't find nor pull docker image #{name}" if image.nil?
image, scripts = bootstrap_image(name, image)
@docker_run_tickets.acquire(1)
puts "--> start docker #{name}"
container = Docker::Container.create(
'Cmd' => %w{sleep 3600},
'Image' => image.id,
'OpenStdin' => true,
)
container.start
scripts.each do |script|
container.exec(%w{chmod +x}.push(script))
container.exec(%w{sh -c}.push(script))
end
container
end
def stop_container(container)
@docker_run_tickets.release(1)
puts "--> killrm docker #{container.id}"
container.kill
container.delete(force: true)
end
private
# get all docker image tags
def docker_images_by_tag
images = {}
Docker::Image.all.map do |img|
Array(img.info['RepoTags']).each do |tag|
images[tag] = img
end
end
images
end
end