Files
ubicloud/spec/routes/api/project/location/load_balancer_spec.rb
mohi-kalantari ce8d3faf95 Move healthcheck logic from LoadBalancersVms to LoadBalancerVmPort allowing multiple ports
By moving healthcheck logic, we can have multiple ports per LB with
the right health check implementation.

LoadBalancersVms table was not dropped and was kept as it was since
it had its own usages and we needed the states stored in it for migration

LB and Firewall progs are changed to apply the right nft rules for each
port. The prog will iterate over the lb.ports and create the right rules for
each port
2025-03-25 11:02:41 +01:00

279 lines
10 KiB
Ruby
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
# frozen_string_literal: true
require_relative "../../spec_helper"
RSpec.describe Clover, "load-balancer" do
let(:user) { create_account }
let(:project) { project_with_default_policy(user) }
let(:lb) do
ps = Prog::Vnet::SubnetNexus.assemble(project.id, name: "subnet-1", location_id: Location[display_name: TEST_LOCATION].id)
dz = DnsZone.create_with_id(name: "test-dns-zone", project_id: project.id)
cert = Prog::Vnet::CertNexus.assemble("test-host-name", dz.id).subject
lb = Prog::Vnet::LoadBalancerNexus.assemble(ps.id, name: "lb-1", src_port: 80, dst_port: 8080).subject
lb.add_cert(cert)
lb
end
describe "unauthenticated" do
it "cannot perform authenticated operations" do
[
[:get, "/project/#{project.ubid}/load-balancer"],
[:post, "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/lb-1"],
[:delete, "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}"],
[:get, "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}"],
[:post, "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}/attach-vm", {vm_id: "vm-1"}],
[:post, "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}/detach-vm", {vm_id: "vm-1"}],
[:get, "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.ubid}"]
].each do |method, path, body|
send(method, path, body)
expect(last_response).to have_api_error(401, "Please login to continue")
end
end
end
describe "authenticated" do
before do
login_api(user.email)
lb_project = Project.create_with_id(name: "default")
allow(Config).to receive(:load_balancer_service_project_id).and_return(lb_project.id)
end
describe "list" do
it "empty" do
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer"
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["items"]).to eq([])
end
it "success single" do
lb
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer"
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["items"].length).to eq(1)
end
it "success multiple" do
lb
Prog::Vnet::LoadBalancerNexus.assemble(lb.private_subnet.id, name: "lb-2", src_port: 80, dst_port: 8080).subject
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer"
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["items"].length).to eq(2)
end
end
describe "id" do
it "success" do
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.ubid}"
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["name"]).to eq("lb-1")
end
it "not found" do
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/_invalid"
expect(last_response).to have_api_error(404, "Sorry, we couldnt find the resource youre looking for.")
end
end
describe "create" do
it "success" do
ps = Prog::Vnet::SubnetNexus.assemble(project.id, name: "subnet-1", location_id: Location[display_name: TEST_LOCATION].id).subject
post "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/lb1", {
private_subnet_id: ps.ubid,
stack: LoadBalancer::Stack::IPV4,
src_port: "80", dst_port: "8080",
health_check_endpoint: "/up", algorithm: "round_robin",
health_check_protocol: "http"
}.to_json
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["name"]).to eq("lb1")
end
it "invalid private_subnet_id" do
post "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/lb1", {
private_subnet_id: "invalid",
stack: LoadBalancer::Stack::IPV6,
src_port: "80", dst_port: "8080",
health_check_endpoint: "/up", algorithm: "round_robin",
health_check_protocol: "http"
}.to_json
expect(last_response).to have_api_error(400, "Validation failed for following fields: private_subnet_id")
end
end
describe "delete" do
it "success" do
delete "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}"
expect(last_response.status).to eq(204)
end
it "not found" do
delete "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/invalid_name"
expect(last_response.status).to eq(204)
end
end
describe "get" do
it "success" do
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}"
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["name"]).to eq("lb-1")
end
it "not found" do
get "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/invalid"
expect(last_response).to have_api_error(404, "Sorry, we couldnt find the resource youre looking for.")
end
end
describe "update" do
let(:vm) {
nic = Nic.create_with_id(name: "nic-1", private_subnet_id: lb.private_subnet.id, mac: "00:00:00:00:00:01", private_ipv4: "1.1.1.1", private_ipv6: "2001:db8::1")
vm = create_vm
nic.update(vm_id: vm.id)
vm.update(project_id: project.id)
vm
}
it "success" do
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}", {
src_port: "80", dst_port: "8080",
health_check_endpoint: "/up", algorithm: "round_robin", vms: []
}.to_json
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["name"]).to eq("lb-1")
end
it "not found" do
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/invalid", {
src_port: "80", dst_port: "8080",
health_check_endpoint: "/up", algorithm: "round_robin", vms: []
}.to_json
expect(last_response).to have_api_error(404, "Sorry, we couldnt find the resource youre looking for.")
end
it "missing required parameters" do
patch "/project/#{project.ubid}/location/#{lb.private_subnet.display_location}/load-balancer/#{lb.name}", {}.to_json
expect(last_response).to have_api_error(400, "Validation failed for following fields: body")
end
it "updates vms" do
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}", {
src_port: "80", dst_port: "8080", health_check_endpoint: "/up", algorithm: "round_robin", vms: [vm.ubid]
}.to_json
expect(last_response.status).to eq(200)
expect(JSON.parse(last_response.body)["vms"].length).to eq(1)
end
it "detaches vms" do
lb.add_vm(vm)
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}", {
src_port: "80", dst_port: "8080", health_check_endpoint: "/up", algorithm: "round_robin", vms: []
}.to_json
expect(last_response.status).to eq(200)
expect(lb.reload.vms.count).to eq(0)
expect(JSON.parse(last_response.body)["vms"]).to eq([])
end
it "invalid vm" do
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}", {
src_port: "80", dst_port: "80", health_check_endpoint: "/up", algorithm: "round_robin", vms: ["invalid"]
}.to_json
expect(last_response).to have_api_error(400, "Validation failed for following fields: vms")
end
it "vm already attached to a different load balancer" do
lb2 = Prog::Vnet::LoadBalancerNexus.assemble(lb.private_subnet.id, name: "lb-2", src_port: 80, dst_port: 8080).subject
dz = DnsZone.create_with_id(name: "test-dns-zone2", project_id: lb2.private_subnet.project_id)
cert = Prog::Vnet::CertNexus.assemble("test-host-name", dz.id).subject
lb2.add_cert(cert)
lb2.add_vm(vm)
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}", {
src_port: "80", dst_port: "8080", health_check_endpoint: "/up", algorithm: "round_robin", vms: [vm.ubid]
}.to_json
expect(last_response).to have_api_error(400)
end
it "vm already attached to the same load balancer" do
lb.add_vm(vm)
patch "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}", {
src_port: "80", dst_port: "8080", health_check_endpoint: "/up", algorithm: "round_robin", vms: [vm.ubid]
}.to_json
expect(last_response.status).to eq(200)
end
end
describe "attach-vm" do
let(:vm) {
nic = Nic.create_with_id(name: "nic-1", private_subnet_id: lb.private_subnet.id, mac: "00:00:00:00:00:01", private_ipv4: "1.1.1.1", private_ipv6: "2001:db8::1")
vm = create_vm
nic.update(vm_id: vm.id)
vm
}
it "success" do
vm.update(project_id: project.id)
post "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}/attach-vm", {vm_id: vm.ubid}.to_json
expect(last_response.status).to eq(200)
end
it "not existing vm" do
post "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}/attach-vm", {vm_id: "invalid"}.to_json
expect(last_response).to have_api_error(400, "Validation failed for following fields: vm_id")
end
end
describe "detach-vm" do
let(:vm) {
nic = Nic.create_with_id(name: "nic-1", private_subnet_id: lb.private_subnet.id, mac: "00:00:00:00:00:01", private_ipv4: "1.1.1.1", private_ipv6: "2001:db8::1")
vm = create_vm
nic.update(vm_id: vm.id)
vm
}
it "success" do
vm.update(project_id: project.id)
lb.add_vm(vm)
post "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}/detach-vm", {vm_id: vm.ubid}.to_json
expect(last_response.status).to eq(200)
end
it "not existing vm" do
post "/project/#{project.ubid}/location/#{TEST_LOCATION}/load-balancer/#{lb.name}/detach-vm", {vm_id: "invalid"}.to_json
expect(last_response).to have_api_error(400, "Validation failed for following fields: vm_id")
end
end
end
end