277 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			277 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| require 'lib/executors/server_executor'
 | |
| require 'providers/aws/server_category_provider_aws'
 | |
| require 'workers/delete_expired_server_worker'
 | |
| 
 | |
| RSpec.describe Devops::Executor::ServerExecutor, type: :executor, stubbed_logger: true do
 | |
|   let(:project) { build(:project) }
 | |
|   let(:environment) { build(:environment, id: 'foo') }
 | |
|   let(:category) { build(:category, id: 'name') }
 | |
|   let(:server) { build(:server, project: project.id, environment: 'foo', category: 'name') }
 | |
|   let(:output) { File.open(File::NULL, "w") }
 | |
|   let(:provider) { double('Provider instance') }
 | |
|   let(:current_user) { 'user' }
 | |
|   let(:executor) { described_class.new(server, output, current_user) }
 | |
|   let(:account_instance) { double('account instance', ssh_key: 'provider_key') }
 | |
|   let(:provider_instance) {
 | |
|     instance_double(Devops::Model::ServerCategoryProviderAws,
 | |
|       name: 'aws',
 | |
|       account: 'acc',
 | |
|       account_instance: account_instance,
 | |
|       image: build(:image),
 | |
|       create_server: true,
 | |
|       waiting_server: true,
 | |
|       flavor: 'flavor',
 | |
|       subnet: [],
 | |
|       security_groups: []
 | |
|     )
 | |
|   }
 | |
| 
 | |
| 
 | |
|   before do
 | |
|     allow(Devops::Model::Project).to receive(:find_with_environment) { project }
 | |
|     allow(Provider).to receive(:get_connector) { provider_instance }
 | |
|     allow(Provider).to receive(:create_category_provider) { provider_instance }
 | |
|     allow(project).to receive(:environment) { environment }
 | |
|     allow(environment).to receive(:get_category) { category }
 | |
|     executor.job_task = build(:task)
 | |
|   end
 | |
| 
 | |
|   describe '#initialize' do
 | |
|     it 'sets server, project, environment, out instance variables' do
 | |
|       expect(executor.server).to eq server
 | |
|       expect(executor.environment).to eq environment
 | |
|       expect(executor).to have_instance_variable_value(:project, project)
 | |
|       expect(executor).to have_instance_variable_value(:out, output)
 | |
|     end
 | |
| 
 | |
|     it 'defines :flush method on @out if it is absent' do
 | |
|       out = Class.new.new
 | |
|       expect(out).not_to respond_to(:flush)
 | |
|       described_class.new(server, out, current_user)
 | |
|       expect(out).to respond_to(:flush)
 | |
|     end
 | |
| 
 | |
|     it 'sets current_user from options' do
 | |
|       executor = described_class.new(server, '', current_user)
 | |
|       expect(executor.current_user).to eq current_user
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#task=' do
 | |
|     it 'sets task instance variable' do
 | |
|       executor.job_task= 'foo'
 | |
|       expect(executor).to have_instance_variable_value(:job_task, 'foo')
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '#project=' do
 | |
|     it 'sets project instance variable' do
 | |
|       executor.project= 'foo'
 | |
|       expect(executor).to have_instance_variable_value(:project, 'foo')
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#create_server_object' do
 | |
|     it 'builds Server object' do
 | |
|       server = executor.create_server_object({})
 | |
|       expect(server).to be_a(Devops::Model::Server)
 | |
|       expect(server.project).to eq 'my_project'
 | |
|       expect(server.environment).to eq 'foo'
 | |
|       expect(server.created_by).to eq current_user
 | |
|     end
 | |
|     it 'sets private_ip from options' do
 | |
|       server = executor.create_server_object('private_ip' => 'ip')
 | |
|       expect(server.private_ip).to eq 'ip'
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#create_server', clean_db_after_example: true do
 | |
|     let(:image) { double('Image instance', remote_user: 'remote_user') }
 | |
|     let(:run_list) { %w(role[asd]) }
 | |
|     let(:create_server_options) { {
 | |
|       'run_list' => run_list,
 | |
|       'name' => 'node_name',
 | |
|       'cm_name' => 'node_name',
 | |
|       'ssh_key' => 'key',
 | |
|       'without_bootstrap' => nil
 | |
|     } }
 | |
|     let(:create_server) { executor.create_server(create_server_options) }
 | |
| 
 | |
|     before do
 | |
|       allow(provider).to receive(:create_server) { true }
 | |
|       allow(image).to receive(:bootstrap_template) { 'template' }
 | |
|       allow(Devops::Model::Image).to receive_message_chain('where.first') { build(:image) }
 | |
|     end
 | |
| 
 | |
|     it 'builds server model from given options' do
 | |
|       create_server
 | |
|       expect(executor.server.created_by).to eq current_user
 | |
|       expect(executor.server.ssh_key).to eq 'key'
 | |
|       expect(executor.server.run_list).to eq run_list
 | |
|     end
 | |
| 
 | |
|     it 'sets run list to an empty array by default' do
 | |
|       create_server_options.delete('run_list')
 | |
|       create_server
 | |
|       expect(executor.server.run_list).to eq []
 | |
|     end
 | |
| 
 | |
|     it 'sets key to default provider ssh key by default' do
 | |
|       create_server_options.delete('ssh_key')
 | |
|       create_server
 | |
|       expect(executor.server.ssh_key).to eq 'provider_key'
 | |
|     end
 | |
| 
 | |
|     it 'runs hooks' do
 | |
|       expect(executor).to receive(:run_hook).with(:before_create).ordered
 | |
|       expect(executor).to receive(:run_hook).with(:after_create).ordered
 | |
|       create_server
 | |
|     end
 | |
| 
 | |
|     it 'creates server in cloud' do
 | |
|       built_server = build(:server)
 | |
|       allow(executor).to receive(:create_server_object) { built_server }
 | |
|       expect(provider_instance).to receive(:create_server).with(built_server, output)
 | |
|       create_server
 | |
|     end
 | |
| 
 | |
|     it 'inserts built server into mongo' do
 | |
|       built_server = build(:server)
 | |
|       allow(executor).to receive(:create_server_object) { built_server }
 | |
|       expect(built_server).to receive(:save).at_least(1).times
 | |
|       create_server
 | |
|     end
 | |
| 
 | |
|     it 'schedules expiration for server' do
 | |
|       environment.expires = '2m'
 | |
|       allow(DeleteExpiredServerWorker).to receive(:perform_in)
 | |
|       expect(DeleteExpiredServerWorker).to receive(:perform_in).with(120, hash_including(:server_id))
 | |
|       create_server
 | |
|     end
 | |
| 
 | |
|     it "doesn't schedule expiration if environment.expires is nil" do
 | |
|       environment.expires = nil
 | |
|       expect(DeleteExpiredServerWorker).not_to receive(:perform_in)
 | |
|       create_server
 | |
|     end
 | |
| 
 | |
|     context "when project is sandbox" do
 | |
|       pending 'it sets deployers to owner and passed deployers' do
 | |
|         allow(project).to receive(:is_sandbox?) { true }
 | |
|         create_server_options['project_info'] = {'deployers' =>  %w(user1 user2)}
 | |
|         expect(executor).to receive(:two_phase_bootstrap).with(hash_including(deployers: %w(me user1 user2)))
 | |
|         create_server
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context "if creating server in cloud wasn't successful" do
 | |
|       it 'raises an error' do
 | |
|         allow(provider_instance).to receive(:create_server) { false }
 | |
|         expect{create_server}.to raise_error(Devops::Exception::CreationError)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#bootstrap_server', clean_db_after_example: true do
 | |
|     let(:image) { double('Key instance', path: 'path') }
 | |
|     let(:bootstrap) { executor.bootstrap_server({}, {}) }
 | |
| 
 | |
|     before do
 | |
|       allow(executor).to receive(:sleep)
 | |
|       allow(provider).to receive(:create_default_chef_node_name).and_return('chef_node')
 | |
|       allow(category).to receive(:cm_tool) { double('cm_tool', bootstrap_instance: nil) }
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#unbootstrap' do
 | |
|     it 'delegates to cm_tool' do
 | |
|       unbootstrap_result = double('unbootstrap result')
 | |
|       allow(category).to receive(:cm_tool) {
 | |
|         double('cm_tool', delete_instance: unbootstrap_result)
 | |
|       }
 | |
|       expect(executor.unbootstrap_server).to eq unbootstrap_result
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#rollback' do
 | |
|     before do
 | |
|       allow(server).to receive_message_chain('provider_instance.delete_server')
 | |
|       allow(category).to receive_message_chain('cm_tool.delete_instance')
 | |
|     end
 | |
| 
 | |
|     it "does nothing if server.id is nil" do
 | |
|       server.id = nil
 | |
|       expect(category).not_to receive(:cm_tool)
 | |
|       expect(server).not_to receive(:provider_instance)
 | |
|       executor.roll_back
 | |
|     end
 | |
| 
 | |
|     it 'deletes node from chef server and instance from cloud' do
 | |
|       expect(category).to receive_message_chain('cm_tool.delete_instance')
 | |
|       expect(server).to receive_message_chain('provider_instance.delete_server')
 | |
|       executor.roll_back
 | |
|     end
 | |
| 
 | |
|     it "doesn't raise if deleting server in cloud raises an error" do
 | |
|       allow(server).to receive_message_chain('provider_instance.delete_server') { raise }
 | |
|       expect { executor.roll_back }.not_to raise_error
 | |
|     end
 | |
|   end
 | |
| 
 | |
| 
 | |
|   describe '#add_run_list_to_deploy_info' do
 | |
|     it "doesn't change deploy info if it already includes run list" do
 | |
|       deploy_info = {'run_list' => %w(foo)}
 | |
|       expect {
 | |
|         executor.add_run_list_to_deploy_info(output, deploy_info)
 | |
|       }.not_to change { deploy_info }
 | |
|     end
 | |
| 
 | |
|     it 'computes and adds run_list to deploy_info' do
 | |
|       deploy_info = {}
 | |
|       allow(executor).to receive(:compute_run_list) { %w(foo) }
 | |
|       expect(executor).to receive(:compute_run_list)
 | |
|       executor.add_run_list_to_deploy_info(output, deploy_info)
 | |
|       expect(deploy_info['run_list']).to eq %w(foo)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   describe '#compute_run_list' do
 | |
|     before do
 | |
|       allow(category).to receive_message_chain('provider_instance.run_list') { %w(a) }
 | |
|       project.run_list = %w(b)
 | |
|       environment.run_list = %w(c)
 | |
|       server.run_list = %w(d)
 | |
|     end
 | |
| 
 | |
|     pending "returns array with run list merged from provider's, project's, env's and server's run lists" do
 | |
|       expect(executor.compute_run_list).to be_an(Array).and contain_exactly(*%w(a b c d))
 | |
|     end
 | |
| 
 | |
|     pending "includes stack's run list if stack is set", stubbed_connector: true do
 | |
|       server.stack = 'stack'
 | |
|       allow(stubbed_connector).to receive(:stack) { instance_double(Devops::Model::StackEc2, run_list: %w(e)) }
 | |
|       expect(executor.compute_run_list).to be_an(Array).and contain_exactly(*%w(a b c d e))
 | |
|     end
 | |
| 
 | |
|     pending "doesn't contain nils" do
 | |
|       server.run_list = nil
 | |
|       server.stack = 'stack'
 | |
|       allow(stubbed_connector).to receive(:stack) { instance_double(Devops::Model::StackEc2, run_list: nil) }
 | |
|       expect(executor.compute_run_list).to be_an(Array).and contain_exactly(*%w(a b c))
 | |
|     end
 | |
| 
 | |
|     pending 'returns uniq elements' do
 | |
|       project.run_list = %w(a)
 | |
|       environment.run_list = %w(a)
 | |
|       expect(executor.compute_run_list).to be_an(Array).and contain_exactly(*%w(a d))
 | |
|     end
 | |
|   end
 | |
| end
 | 
