require 'tempfile' module Gitlab module QA module Component class InternetTunnel include Scenario::Actable DOCKER_IMAGE = 'dylangriffith/ssh'.freeze DOCKER_IMAGE_TAG = 'latest'.freeze attr_writer :gitlab_hostname, :name attr_accessor :network def initialize @docker = Docker::Engine.new @volumes = {} key_dir = ENV['CI_PROJECT_DIR'] || Dir.tmpdir @ssh_key = Tempfile.new('tunnel-ssh-private-key', key_dir) @ssh_key.write(ENV.fetch('TUNNEL_SSH_PRIVATE_KEY')) @ssh_key.close File.chmod(0o600, @ssh_key.path) @volumes['/root/.ssh/id_rsa'] = @ssh_key.path end def instance raise ArgumentError, 'Please provide a block!' unless block_given? prepare start yield self ensure teardown end def url "https://#{subdomain}.#{tunnel_server_hostname}" end private def name @name ||= "ssh-tunnel-#{SecureRandom.hex(4)}" end def prepare @docker.pull(DOCKER_IMAGE, DOCKER_IMAGE_TAG) return if @docker.network_exists?(network) @docker.network_create(network) end def tunnel_server_hostname ENV.fetch("TUNNEL_SERVER_HOSTNAME") end def subdomain @subdomain ||= rand(20_000..30_000) end def start raise "Must set gitlab_hostname" unless @gitlab_hostname @docker.run(DOCKER_IMAGE, DOCKER_IMAGE_TAG, "-o StrictHostKeyChecking=no -N -R #{subdomain}:#{@gitlab_hostname}:80 #{ENV.fetch('TUNNEL_SSH_USER')}@#{tunnel_server_hostname}") do |command| command << '-d ' command << "--name #{name}" command << "--net #{network}" @volumes.to_h.each do |to, from| command.volume(from, to, 'Z') end end end def restart @docker.restart(name) end def teardown raise 'Invalid instance name!' unless name @docker.stop(name) @docker.remove(name) @ssh_key.unlink end end end end end