165 lines
9.1 KiB
Ruby
165 lines
9.1 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
RSpec.describe Prog::Aws::Nic do
|
|
subject(:nx) {
|
|
described_class.new(st)
|
|
}
|
|
|
|
let(:st) {
|
|
Strand.create(prog: "Aws::Nic", stack: [{"subject_id" => nic.id}], label: "create_network_interface")
|
|
}
|
|
|
|
let(:nic) {
|
|
prj = Project.create(name: "test-prj")
|
|
loc = Location.create(name: "us-west-2", provider: "aws", project_id: prj.id, display_name: "aws-us-west-2", ui_name: "AWS US East 1", visible: true)
|
|
LocationCredential.create_with_id(loc.id, access_key: "test-access-key", secret_key: "test-secret-key")
|
|
ps = Prog::Vnet::SubnetNexus.assemble(prj.id, name: "test-ps", location_id: loc.id).subject
|
|
PrivateSubnetAwsResource.create_with_id(ps.id, subnet_id: "subnet-0123456789abcdefg", security_group_id: "sg-0123456789abcdefg")
|
|
nic = Prog::Vnet::NicNexus.assemble(ps.id, name: "test-nic").subject
|
|
NicAwsResource.create_with_id(nic.id)
|
|
nic
|
|
}
|
|
|
|
let(:client) {
|
|
Aws::EC2::Client.new(stub_responses: true)
|
|
}
|
|
|
|
before do
|
|
allow(nx).to receive(:nic).and_return(nic)
|
|
allow(Aws::EC2::Client).to receive(:new).with(access_key_id: "test-access-key", secret_access_key: "test-secret-key", region: "us-west-2").and_return(client)
|
|
end
|
|
|
|
describe "#create_network_interface" do
|
|
it "creates a network interface" do
|
|
client.stub_responses(:create_network_interface, network_interface: {network_interface_id: "eni-0123456789abcdefg", ipv_6_addresses: []})
|
|
expect(client).to receive(:create_network_interface).with({subnet_id: "subnet-0123456789abcdefg", private_ip_address: nic.private_ipv4.network.to_s, ipv_6_prefix_count: 1, groups: ["sg-0123456789abcdefg"], tag_specifications: Util.aws_tag_specifications("network-interface", nic.name), client_token: nic.id}).and_call_original
|
|
expect { nx.create_network_interface }.to hop("assign_ipv6_address")
|
|
end
|
|
end
|
|
|
|
describe "#assign_ipv6_address" do
|
|
it "assigns an IPv6 address" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [{ipv_6_addresses: []}])
|
|
expect(nic.nic_aws_resource).to receive(:network_interface_id).and_return("eni-0123456789abcdefg").at_least(:once)
|
|
client.stub_responses(:assign_ipv_6_addresses)
|
|
expect(client).to receive(:assign_ipv_6_addresses).with({network_interface_id: "eni-0123456789abcdefg", ipv_6_address_count: 1}).and_call_original
|
|
expect { nx.assign_ipv6_address }.to hop("wait_network_interface_created")
|
|
end
|
|
|
|
it "skips assigning IPv6 addresses if already assigned" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [{ipv_6_addresses: [{ipv_6_address: "2a01:4f8:173:1ed3::1"}]}])
|
|
expect(client).not_to receive(:assign_ipv_6_addresses)
|
|
expect { nx.assign_ipv6_address }.to hop("wait_network_interface_created")
|
|
end
|
|
end
|
|
|
|
describe "#wait_network_interface_created" do
|
|
it "checks if network interface is available, if not naps" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [{status: "pending"}])
|
|
expect(nic.nic_aws_resource).to receive(:network_interface_id).and_return("eni-0123456789abcdefg").at_least(:once)
|
|
expect(client).to receive(:describe_network_interfaces).with({filters: [{name: "network-interface-id", values: ["eni-0123456789abcdefg"]}, {name: "tag:Ubicloud", values: ["true"]}]}).and_call_original
|
|
expect { nx.wait_network_interface_created }.to nap(1)
|
|
end
|
|
|
|
it "checks if network interface is available, if it is, it allocates an elastic ip and associates it with the network interface" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [{status: "available"}])
|
|
client.stub_responses(:allocate_address, allocation_id: "eip-0123456789abcdefg")
|
|
expect(nic.nic_aws_resource).to receive(:network_interface_id).and_return("eni-0123456789abcdefg").at_least(:once)
|
|
expect(client).to receive(:describe_network_interfaces).with({filters: [{name: "network-interface-id", values: ["eni-0123456789abcdefg"]}, {name: "tag:Ubicloud", values: ["true"]}]}).and_call_original
|
|
|
|
expect { nx.wait_network_interface_created }.to hop("allocate_eip")
|
|
end
|
|
end
|
|
|
|
describe "#allocate_eip" do
|
|
it "allocates an elastic ip" do
|
|
client.stub_responses(:describe_addresses, addresses: [])
|
|
client.stub_responses(:allocate_address, allocation_id: "eip-0123456789abcdefg")
|
|
expect(client).to receive(:allocate_address).and_call_original
|
|
expect(nic.nic_aws_resource).to receive(:update).with(eip_allocation_id: "eip-0123456789abcdefg")
|
|
expect { nx.allocate_eip }.to hop("attach_eip_network_interface")
|
|
end
|
|
|
|
it "reuses an existing elastic ip if available" do
|
|
client.stub_responses(:describe_addresses, addresses: [{allocation_id: "eip-0123456789abcdefg"}])
|
|
expect(client).not_to receive(:allocate_address)
|
|
expect(nic.nic_aws_resource).to receive(:update).with(eip_allocation_id: "eip-0123456789abcdefg")
|
|
expect { nx.allocate_eip }.to hop("attach_eip_network_interface")
|
|
end
|
|
end
|
|
|
|
describe "#attach_eip_network_interface" do
|
|
it "associates the elastic ip with the network interface" do
|
|
expect(nic.nic_aws_resource).to receive(:eip_allocation_id).and_return("eip-0123456789abcdefg").at_least(:once)
|
|
client.stub_responses(:describe_addresses, addresses: [{allocation_id: "eip-0123456789abcdefg", network_interface_id: nil}])
|
|
client.stub_responses(:associate_address)
|
|
expect(client).to receive(:associate_address).with({allocation_id: "eip-0123456789abcdefg", network_interface_id: nic.nic_aws_resource.network_interface_id}).and_call_original
|
|
expect { nx.attach_eip_network_interface }.to exit({"msg" => "nic created"})
|
|
end
|
|
|
|
it "skips association if elastic ip is already associated" do
|
|
client.stub_responses(:describe_addresses, addresses: [{allocation_id: "eip-0123456789abcdefg", network_interface_id: "eni-existing"}])
|
|
expect(client).not_to receive(:associate_address)
|
|
expect { nx.attach_eip_network_interface }.to exit({"msg" => "nic created"})
|
|
end
|
|
end
|
|
|
|
describe "#destroy" do
|
|
it "naps if the network interface is in use" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [{status: "in-use"}])
|
|
expect { nx.destroy }.to nap(5)
|
|
end
|
|
|
|
it "deletes the network interface" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [{status: "available"}])
|
|
client.stub_responses(:delete_network_interface)
|
|
expect(nic.nic_aws_resource).to receive(:network_interface_id).and_return("eni-0123456789abcdefg").at_least(:once)
|
|
expect(client).to receive(:delete_network_interface).with({network_interface_id: "eni-0123456789abcdefg"}).and_call_original
|
|
expect { nx.destroy }.to hop("release_eip")
|
|
end
|
|
|
|
it "doesn't nap if network_interfaces are empty" do
|
|
client.stub_responses(:describe_network_interfaces, network_interfaces: [])
|
|
client.stub_responses(:delete_network_interface, Aws::EC2::Errors::InvalidNetworkInterfaceIDNotFound.new(nil, "The network interface 'eni-0123456789abcdefg' does not exist."))
|
|
expect(nic.nic_aws_resource).to receive(:network_interface_id).and_return("eni-0123456789abcdefg").at_least(:once)
|
|
expect { nx.destroy }.to hop("release_eip")
|
|
end
|
|
|
|
it "hops to release_eip if the network interface is not found" do
|
|
client.stub_responses(:delete_network_interface, Aws::EC2::Errors::InvalidNetworkInterfaceIDNotFound.new(nil, "The network interface 'eni-0123456789abcdefg' does not exist."))
|
|
expect(nic.nic_aws_resource).to receive(:network_interface_id).and_return("eni-0123456789abcdefg").at_least(:once)
|
|
expect { nx.destroy }.to hop("release_eip")
|
|
end
|
|
end
|
|
|
|
describe "#release_eip" do
|
|
it "releases the elastic ip" do
|
|
expect(nic.nic_aws_resource).to receive(:eip_allocation_id).and_return("eip-0123456789abcdefg").at_least(:once)
|
|
client.stub_responses(:release_address)
|
|
expect(client).to receive(:release_address).with({allocation_id: "eip-0123456789abcdefg"}).and_call_original
|
|
expect { nx.release_eip }.to exit({"msg" => "nic destroyed"})
|
|
end
|
|
|
|
it "gracefully continues if the nic is not found" do
|
|
expect(nic.nic_aws_resource).to receive(:eip_allocation_id).and_return(nil).at_least(:once)
|
|
expect { nx.release_eip }.to exit({"msg" => "nic destroyed"})
|
|
end
|
|
|
|
it "gracefully continues if the nic_aws_resource is not found" do
|
|
expect(nic).to receive(:nic_aws_resource).and_return(nil).at_least(:once)
|
|
expect { nx.release_eip }.to exit({"msg" => "nic destroyed"})
|
|
end
|
|
|
|
it "continues if the eip_allocation_id is found" do
|
|
expect(nic.nic_aws_resource).to receive(:eip_allocation_id).and_return("eip-0123456789abcdefg").at_least(:once)
|
|
expect { nx.release_eip }.to exit({"msg" => "nic destroyed"})
|
|
end
|
|
|
|
it "pops if the address is already released" do
|
|
client.stub_responses(:describe_addresses, addresses: [{allocation_id: "eip-0123456789abcdefg"}])
|
|
client.stub_responses(:release_address, Aws::EC2::Errors::InvalidAllocationIDNotFound.new(nil, "The address 'eip-0123456789abcdefg' does not exist."))
|
|
expect { nx.release_eip }.to exit({"msg" => "nic destroyed"})
|
|
end
|
|
end
|
|
end
|