Commit 8cf4b506 authored by Shinya Maeda's avatar Shinya Maeda

Introduce Canary Object Model to check the canary status

This commit introduces the Canary Object Model
parent 64239221
...@@ -17,6 +17,7 @@ class DeploymentEntity < Grape::Entity ...@@ -17,6 +17,7 @@ class DeploymentEntity < Grape::Entity
end end
end end
expose :status
expose :created_at expose :created_at
expose :deployed_at expose :deployed_at
expose :tag expose :tag
......
---
name: canary_ingress_weight_control
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/43816
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/260295
type: development
group: group::progressive delivery
default_enabled: false
...@@ -14,8 +14,15 @@ module EE ...@@ -14,8 +14,15 @@ module EE
if result if result
deployments = read_deployments(environment.deployment_namespace) deployments = read_deployments(environment.deployment_namespace)
ingresses = if ::Feature.enabled?(:canary_ingress_weight_control, environment.project)
read_ingresses(environment.deployment_namespace)
else
[]
end
# extract_relevant_deployment_data avoids uploading all the deployment info into ReactiveCaching # extract_relevant_deployment_data avoids uploading all the deployment info into ReactiveCaching
result[:deployments] = extract_relevant_deployment_data(deployments) result[:deployments] = extract_relevant_deployment_data(deployments)
result[:ingresses] = extract_relevant_ingress_data(ingresses)
end end
result result
...@@ -26,8 +33,9 @@ module EE ...@@ -26,8 +33,9 @@ module EE
deployments = filter_by_project_environment(data[:deployments], project.full_path_slug, environment.slug) deployments = filter_by_project_environment(data[:deployments], project.full_path_slug, environment.slug)
pods = filter_by_project_environment(data[:pods], project.full_path_slug, environment.slug) pods = filter_by_project_environment(data[:pods], project.full_path_slug, environment.slug)
ingresses = data[:ingresses].presence || []
::Gitlab::Kubernetes::RolloutStatus.from_deployments(*deployments, pods_attrs: pods) ::Gitlab::Kubernetes::RolloutStatus.from_deployments(*deployments, pods_attrs: pods, ingresses: ingresses)
end end
private private
...@@ -38,6 +46,12 @@ module EE ...@@ -38,6 +46,12 @@ module EE
[] []
end end
def read_ingresses(namespace)
kubeclient.get_ingresses(namespace: namespace).as_json
rescue Kubeclient::ResourceNotFoundError
[]
end
def extract_relevant_deployment_data(deployments) def extract_relevant_deployment_data(deployments)
deployments.map do |deployment| deployments.map do |deployment|
{ {
...@@ -47,6 +61,14 @@ module EE ...@@ -47,6 +61,14 @@ module EE
} }
end end
end end
def extract_relevant_ingress_data(ingresses)
ingresses.map do |ingress|
{
'metadata' => ingress.fetch('metadata', {}).slice('name', 'labels', 'annotations')
}
end
end
end end
end end
end end
......
...@@ -13,4 +13,6 @@ class RolloutStatusEntity < Grape::Entity ...@@ -13,4 +13,6 @@ class RolloutStatusEntity < Grape::Entity
expose :instances, if: -> (rollout_status, _) { rollout_status.found? } expose :instances, if: -> (rollout_status, _) { rollout_status.found? }
expose :completion, if: -> (rollout_status, _) { rollout_status.found? } expose :completion, if: -> (rollout_status, _) { rollout_status.found? }
expose :complete?, as: :is_completed, if: -> (rollout_status, _) { rollout_status.found? } expose :complete?, as: :is_completed, if: -> (rollout_status, _) { rollout_status.found? }
expose :canary_ingress, using: RolloutStatuses::IngressEntity, expose_nil: false,
if: -> (rollout_status, _) { rollout_status.found? && rollout_status.canary_ingress_exists? }
end end
# frozen_string_literal: true
module RolloutStatuses
class IngressEntity < Grape::Entity
expose :canary_weight
end
end
# frozen_string_literal: true
module Gitlab
module Kubernetes
class Ingress
include Gitlab::Utils::StrongMemoize
# Canry Ingress Annotations https://kubernetes.github.io/ingress-nginx/user-guide/nginx-configuration/annotations/#canary
ANNOTATION_KEY_CANARY = 'nginx.ingress.kubernetes.io/canary'
ANNOTATION_KEY_CANARY_WEIGHT = 'nginx.ingress.kubernetes.io/canary-weight'
def initialize(attributes = {})
@attributes = attributes
end
def canary?
strong_memoize(:is_canary) do
annotations.any? do |key, value|
key == ANNOTATION_KEY_CANARY && value == 'true'
end
end
end
def canary_weight
return unless canary?
return unless annotations.key?(ANNOTATION_KEY_CANARY_WEIGHT)
annotations[ANNOTATION_KEY_CANARY_WEIGHT].to_i
end
private
def metadata
@attributes.fetch('metadata', {})
end
def annotations
metadata.fetch('annotations', {})
end
end
end
end
...@@ -8,7 +8,7 @@ module Gitlab ...@@ -8,7 +8,7 @@ module Gitlab
# other resources. The rollout status sums the Kubernetes deployments # other resources. The rollout status sums the Kubernetes deployments
# together. # together.
class RolloutStatus class RolloutStatus
attr_reader :deployments, :instances, :completion, :status attr_reader :deployments, :instances, :completion, :status, :canary_ingress
def complete? def complete?
completion == 100 completion == 100
...@@ -26,7 +26,11 @@ module Gitlab ...@@ -26,7 +26,11 @@ module Gitlab
@status == :found @status == :found
end end
def self.from_deployments(*deployments_attrs, pods_attrs: []) def canary_ingress_exists?
canary_ingress.present?
end
def self.from_deployments(*deployments_attrs, pods_attrs: [], ingresses: [])
return new([], status: :not_found) if deployments_attrs.empty? return new([], status: :not_found) if deployments_attrs.empty?
deployments = deployments_attrs.map do |attrs| deployments = deployments_attrs.map do |attrs|
...@@ -38,14 +42,16 @@ module Gitlab ...@@ -38,14 +42,16 @@ module Gitlab
::Gitlab::Kubernetes::Pod.new(attrs) ::Gitlab::Kubernetes::Pod.new(attrs)
end end
new(deployments, pods: pods) ingresses = ingresses.map { |ingress| ::Gitlab::Kubernetes::Ingress.new(ingress) }
new(deployments, pods: pods, ingresses: ingresses)
end end
def self.loading def self.loading
new([], status: :loading) new([], status: :loading)
end end
def initialize(deployments, pods: [], status: :found) def initialize(deployments, pods: [], ingresses: [], status: :found)
@status = status @status = status
@deployments = deployments @deployments = deployments
...@@ -55,6 +61,8 @@ module Gitlab ...@@ -55,6 +61,8 @@ module Gitlab
deployments.flat_map(&:instances) deployments.flat_map(&:instances)
end end
@canary_ingress = ingresses.find(&:canary?)
@completion = @completion =
if @instances.empty? if @instances.empty?
100 100
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::Kubernetes::Ingress do
include KubernetesHelpers
let(:ingress) { described_class.new(params) }
describe '#canary?' do
subject { ingress.canary? }
context 'with canary ingress parameters' do
let(:params) { canary_metadata }
it { is_expected.to be_truthy }
end
context 'with stable ingress parameters' do
let(:params) { stable_metadata }
it { is_expected.to be_falsey }
end
end
describe '#canary_weight' do
subject { ingress.canary_weight }
context 'with canary ingress parameters' do
let(:params) { canary_metadata }
it { is_expected.to eq(50) }
end
context 'with stable ingress parameters' do
let(:params) { stable_metadata }
it { is_expected.to be_nil }
end
end
def stable_metadata
kube_ingress(track: :stable)
end
def canary_metadata
kube_ingress(track: :canary)
end
end
...@@ -12,6 +12,8 @@ RSpec.describe Gitlab::Kubernetes::RolloutStatus do ...@@ -12,6 +12,8 @@ RSpec.describe Gitlab::Kubernetes::RolloutStatus do
create_pods(name: "one", count: 3, track: 'stable') + create_pods(name: "two", count: 3, track: "canary") create_pods(name: "one", count: 3, track: 'stable') + create_pods(name: "two", count: 3, track: "canary")
end end
let(:ingresses) { [] }
let(:specs_all_finished) do let(:specs_all_finished) do
[ [
kube_deployment(name: 'one'), kube_deployment(name: 'one'),
...@@ -26,7 +28,7 @@ RSpec.describe Gitlab::Kubernetes::RolloutStatus do ...@@ -26,7 +28,7 @@ RSpec.describe Gitlab::Kubernetes::RolloutStatus do
] ]
end end
subject(:rollout_status) { described_class.from_deployments(*specs, pods_attrs: pods) } subject(:rollout_status) { described_class.from_deployments(*specs, pods_attrs: pods, ingresses: ingresses) }
shared_examples 'rollout status' do shared_examples 'rollout status' do
describe '#deployments' do describe '#deployments' do
...@@ -223,6 +225,24 @@ RSpec.describe Gitlab::Kubernetes::RolloutStatus do ...@@ -223,6 +225,24 @@ RSpec.describe Gitlab::Kubernetes::RolloutStatus do
it { is_expected.to be_not_found } it { is_expected.to be_not_found }
end end
end end
describe '#canary_ingress_exists?' do
context 'when canary ingress exists' do
let(:ingresses) { [kube_ingress(track: :canary)] }
it 'returns true' do
expect(rollout_status.canary_ingress_exists?).to eq(true)
end
end
context 'when canary ingress does not exist' do
let(:ingresses) { [kube_ingress(track: :stable)] }
it 'returns false' do
expect(rollout_status.canary_ingress_exists?).to eq(false)
end
end
end
end end
context 'deploy_boards_dedupe_instances is disabled' do context 'deploy_boards_dedupe_instances is disabled' do
......
...@@ -27,11 +27,12 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -27,11 +27,12 @@ RSpec.describe Clusters::Platforms::Kubernetes do
describe '#rollout_status' do describe '#rollout_status' do
let(:deployments) { [] } let(:deployments) { [] }
let(:pods) { [] } let(:pods) { [] }
let(:ingresses) { [] }
let(:service) { create(:cluster_platform_kubernetes, :configured) } let(:service) { create(:cluster_platform_kubernetes, :configured) }
let!(:cluster) { create(:cluster, :project, enabled: true, platform_kubernetes: service) } let!(:cluster) { create(:cluster, :project, enabled: true, platform_kubernetes: service) }
let(:project) { cluster.project } let(:project) { cluster.project }
let(:environment) { build(:environment, project: project, name: "env", slug: "env-000000") } let(:environment) { build(:environment, project: project, name: "env", slug: "env-000000") }
let(:cache_data) { Hash(deployments: deployments, pods: pods) } let(:cache_data) { Hash(deployments: deployments, pods: pods, ingresses: ingresses) }
subject(:rollout_status) { service.rollout_status(environment, cache_data) } subject(:rollout_status) { service.rollout_status(environment, cache_data) }
...@@ -129,6 +130,15 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -129,6 +130,15 @@ RSpec.describe Clusters::Platforms::Kubernetes do
tooltip: "Not provided (Pending)", tooltip: "Not provided (Pending)",
track: "stable" }]) track: "stable" }])
end end
context 'with canary ingress' do
let(:ingresses) { [kube_ingress(track: :canary)] }
it 'has canary ingress' do
expect(rollout_status).to be_canary_ingress_exists
expect(rollout_status.canary_ingress.canary_weight).to eq(50)
end
end
end end
context 'with empty list of deployments' do context 'with empty list of deployments' do
...@@ -304,7 +314,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -304,7 +314,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do
let(:cluster) { create(:cluster, :project, platform_kubernetes: service) } let(:cluster) { create(:cluster, :project, platform_kubernetes: service) }
let(:service) { create(:cluster_platform_kubernetes, :configured) } let(:service) { create(:cluster_platform_kubernetes, :configured) }
let(:namespace) { 'project-namespace' } let(:namespace) { 'project-namespace' }
let(:environment) { instance_double(Environment, deployment_namespace: namespace) } let(:environment) { instance_double(Environment, deployment_namespace: namespace, project: cluster.project) }
let(:expected_pod_cached_data) do let(:expected_pod_cached_data) do
kube_pod.tap { |kp| kp['metadata'].delete('namespace') } kube_pod.tap { |kp| kp['metadata'].delete('namespace') }
end end
...@@ -315,10 +325,11 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -315,10 +325,11 @@ RSpec.describe Clusters::Platforms::Kubernetes do
before do before do
stub_kubeclient_pods(namespace) stub_kubeclient_pods(namespace)
stub_kubeclient_deployments(namespace) stub_kubeclient_deployments(namespace)
stub_kubeclient_ingresses(namespace)
end end
shared_examples 'successful deployment request' do shared_examples 'successful deployment request' do
it { is_expected.to include(pods: [expected_pod_cached_data], deployments: [kube_deployment]) } it { is_expected.to include(pods: [expected_pod_cached_data], deployments: [kube_deployment], ingresses: [kube_ingress]) }
end end
context 'on a project level cluster' do context 'on a project level cluster' do
...@@ -338,6 +349,16 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -338,6 +349,16 @@ RSpec.describe Clusters::Platforms::Kubernetes do
include_examples 'successful deployment request' include_examples 'successful deployment request'
end end
context 'when canary_ingress_weight_control feature flag is disabled' do
before do
stub_feature_flags(canary_ingress_weight_control: false)
end
it 'does not fetch ingress data from kubernetes' do
expect(subject[:ingresses]).to be_empty
end
end
end end
context 'when kubernetes responds with 500s' do context 'when kubernetes responds with 500s' do
...@@ -353,9 +374,10 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -353,9 +374,10 @@ RSpec.describe Clusters::Platforms::Kubernetes do
before do before do
stub_kubeclient_pods(namespace) stub_kubeclient_pods(namespace)
stub_kubeclient_deployments(namespace, status: 404) stub_kubeclient_deployments(namespace, status: 404)
stub_kubeclient_ingresses(namespace, status: 404)
end end
it { is_expected.to include(deployments: []) } it { is_expected.to include(deployments: [], ingresses: []) }
end end
end end
end end
...@@ -25,6 +25,18 @@ RSpec.describe RolloutStatusEntity do ...@@ -25,6 +25,18 @@ RSpec.describe RolloutStatusEntity do
it "exposes deployment data" do it "exposes deployment data" do
is_expected.to include(:instances, :completion, :is_completed) is_expected.to include(:instances, :completion, :is_completed)
end end
it 'does not expose canary ingress if it does not exist' do
is_expected.not_to include(:canary_ingress)
end
context 'when canary ingress exists' do
let(:rollout_status) { kube_deployment_rollout_status(ingresses: [kube_ingress(track: :canary)]) }
it 'expose canary ingress' do
is_expected.to include(:canary_ingress)
end
end
end end
context 'when kube deployment is empty' do context 'when kube deployment is empty' do
...@@ -35,7 +47,7 @@ RSpec.describe RolloutStatusEntity do ...@@ -35,7 +47,7 @@ RSpec.describe RolloutStatusEntity do
end end
it "does not expose deployment data" do it "does not expose deployment data" do
is_expected.not_to include(:instances, :completion, :is_completed) is_expected.not_to include(:instances, :completion, :is_completed, :canary_ingress)
end end
end end
end end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe RolloutStatuses::IngressEntity do
include KubernetesHelpers
let(:canary_ingress) { kube_ingress(track: :canary) }
let(:entity) do
described_class.new(canary_ingress, request: double)
end
subject { entity.as_json }
it 'exposes canary weight' do
is_expected.to include(:canary_weight)
end
end
...@@ -167,6 +167,21 @@ module Gitlab ...@@ -167,6 +167,21 @@ module Gitlab
end end
end end
# Ingresses resource is currently on the apis/extensions api group
# until Kubernetes 1.21. Kubernetest 1.22+ has ingresses resources in
# the networking.k8s.io/v1 api group.
#
# As we still support Kubernetes 1.12+, we will need to support both.
def get_ingresses(**args)
extensions_client.discover unless extensions_client.discovered
if extensions_client.respond_to?(:get_ingresses)
extensions_client.get_ingresses(**args)
else
networking_client.get_ingresses(**args)
end
end
def create_or_update_cluster_role_binding(resource) def create_or_update_cluster_role_binding(resource)
update_cluster_role_binding(resource) update_cluster_role_binding(resource)
end end
......
...@@ -347,6 +347,34 @@ RSpec.describe Gitlab::Kubernetes::KubeClient do ...@@ -347,6 +347,34 @@ RSpec.describe Gitlab::Kubernetes::KubeClient do
end end
end end
describe '#get_ingresses' do
let(:extensions_client) { client.extensions_client }
let(:networking_client) { client.networking_client }
include_examples 'redirection not allowed', 'get_ingresses'
include_examples 'dns rebinding not allowed', 'get_ingresses'
it 'delegates to the extensions client' do
expect(extensions_client).to receive(:get_ingresses)
client.get_ingresses
end
context 'extensions does not have deployments for Kubernetes 1.22+ clusters' do
before do
WebMock
.stub_request(:get, api_url + '/apis/extensions/v1beta1')
.to_return(kube_response(kube_1_22_extensions_v1beta1_discovery_body))
end
it 'delegates to the apps client' do
expect(networking_client).to receive(:get_ingresses)
client.get_ingresses
end
end
end
describe 'istio API group' do describe 'istio API group' do
let(:istio_client) { client.istio_client } let(:istio_client) { client.istio_client }
......
...@@ -412,7 +412,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -412,7 +412,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do
end end
let(:namespace) { "project-namespace" } let(:namespace) { "project-namespace" }
let(:environment) { instance_double(Environment, deployment_namespace: namespace) } let(:environment) { instance_double(Environment, deployment_namespace: namespace, project: service.cluster.project) }
subject { service.calculate_reactive_cache_for(environment) } subject { service.calculate_reactive_cache_for(environment) }
...@@ -428,6 +428,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -428,6 +428,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do
before do before do
stub_kubeclient_pods(namespace) stub_kubeclient_pods(namespace)
stub_kubeclient_deployments(namespace) stub_kubeclient_deployments(namespace)
stub_kubeclient_ingresses(namespace)
end end
it { is_expected.to include(pods: [expected_pod_cached_data]) } it { is_expected.to include(pods: [expected_pod_cached_data]) }
...@@ -437,6 +438,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -437,6 +438,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do
before do before do
stub_kubeclient_pods(namespace, status: 500) stub_kubeclient_pods(namespace, status: 500)
stub_kubeclient_deployments(namespace, status: 500) stub_kubeclient_deployments(namespace, status: 500)
stub_kubeclient_ingresses(namespace, status: 500)
end end
it { expect { subject }.to raise_error(Kubeclient::HttpError) } it { expect { subject }.to raise_error(Kubeclient::HttpError) }
...@@ -446,6 +448,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do ...@@ -446,6 +448,7 @@ RSpec.describe Clusters::Platforms::Kubernetes do
before do before do
stub_kubeclient_pods(namespace, status: 404) stub_kubeclient_pods(namespace, status: 404)
stub_kubeclient_deployments(namespace, status: 404) stub_kubeclient_deployments(namespace, status: 404)
stub_kubeclient_ingresses(namespace, status: 404)
end end
it { is_expected.to include(pods: []) } it { is_expected.to include(pods: []) }
......
...@@ -30,6 +30,10 @@ RSpec.describe DeploymentEntity do ...@@ -30,6 +30,10 @@ RSpec.describe DeploymentEntity do
expect(subject[:ref][:name]).to eq 'master' expect(subject[:ref][:name]).to eq 'master'
end end
it 'exposes status' do
expect(subject).to include(:status)
end
it 'exposes creation date' do it 'exposes creation date' do
expect(subject).to include(:created_at) expect(subject).to include(:created_at)
end end
......
...@@ -33,6 +33,10 @@ module KubernetesHelpers ...@@ -33,6 +33,10 @@ module KubernetesHelpers
kube_response(kube_deployments_body) kube_response(kube_deployments_body)
end end
def kube_ingresses_response
kube_response(kube_ingresses_body)
end
def stub_kubeclient_discover_base(api_url) def stub_kubeclient_discover_base(api_url)
WebMock.stub_request(:get, api_url + '/api/v1').to_return(kube_response(kube_v1_discovery_body)) WebMock.stub_request(:get, api_url + '/api/v1').to_return(kube_response(kube_v1_discovery_body))
WebMock WebMock
...@@ -63,6 +67,9 @@ module KubernetesHelpers ...@@ -63,6 +67,9 @@ module KubernetesHelpers
WebMock WebMock
.stub_request(:get, api_url + '/apis/serving.knative.dev/v1alpha1') .stub_request(:get, api_url + '/apis/serving.knative.dev/v1alpha1')
.to_return(kube_response(kube_v1alpha1_serving_knative_discovery_body)) .to_return(kube_response(kube_v1alpha1_serving_knative_discovery_body))
WebMock
.stub_request(:get, api_url + '/apis/networking.k8s.io/v1')
.to_return(kube_response(kube_v1_networking_discovery_body))
end end
def stub_kubeclient_discover_knative_not_found(api_url) def stub_kubeclient_discover_knative_not_found(api_url)
...@@ -148,6 +155,14 @@ module KubernetesHelpers ...@@ -148,6 +155,14 @@ module KubernetesHelpers
WebMock.stub_request(:get, deployments_url).to_return(response || kube_deployments_response) WebMock.stub_request(:get, deployments_url).to_return(response || kube_deployments_response)
end end
def stub_kubeclient_ingresses(namespace, status: nil)
stub_kubeclient_discover(service.api_url)
ingresses_url = service.api_url + "/apis/extensions/v1beta1/namespaces/#{namespace}/ingresses"
response = { status: status } if status
WebMock.stub_request(:get, ingresses_url).to_return(response || kube_ingresses_response)
end
def stub_kubeclient_knative_services(options = {}) def stub_kubeclient_knative_services(options = {})
namespace_path = options[:namespace].present? ? "namespaces/#{options[:namespace]}/" : "" namespace_path = options[:namespace].present? ? "namespaces/#{options[:namespace]}/" : ""
...@@ -304,6 +319,14 @@ module KubernetesHelpers ...@@ -304,6 +319,14 @@ module KubernetesHelpers
} }
end end
# From Kubernetes 1.22+ Ingresses are no longer served from apis/extensions
def kube_1_22_extensions_v1beta1_discovery_body
{
"kind" => "APIResourceList",
"resources" => []
}
end
def kube_knative_discovery_body def kube_knative_discovery_body
{ {
"kind" => "APIResourceList", "kind" => "APIResourceList",
...@@ -416,6 +439,17 @@ module KubernetesHelpers ...@@ -416,6 +439,17 @@ module KubernetesHelpers
} }
end end
def kube_v1_networking_discovery_body
{
"kind" => "APIResourceList",
"apiVersion" => "v1",
"groupVersion" => "networking.k8s.io/v1",
"resources" => [
{ "name" => "ingresses", "namespaced" => true, "kind" => "Ingress" }
]
}
end
def kube_istio_gateway_body(name, namespace) def kube_istio_gateway_body(name, namespace)
{ {
"apiVersion" => "networking.istio.io/v1alpha3", "apiVersion" => "networking.istio.io/v1alpha3",
...@@ -507,6 +541,13 @@ module KubernetesHelpers ...@@ -507,6 +541,13 @@ module KubernetesHelpers
} }
end end
def kube_ingresses_body
{
"kind" => "List",
"items" => [kube_ingress]
}
end
def kube_knative_pods_body(name, namespace) def kube_knative_pods_body(name, namespace)
{ {
"kind" => "PodList", "kind" => "PodList",
...@@ -548,6 +589,38 @@ module KubernetesHelpers ...@@ -548,6 +589,38 @@ module KubernetesHelpers
} }
end end
def kube_ingress(track: :stable)
additional_annotations =
if track == :canary
{
"nginx.ingress.kubernetes.io/canary" => "true",
"nginx.ingress.kubernetes.io/canary-by-header" => "canary",
"nginx.ingress.kubernetes.io/canary-weight" => "50"
}
else
{}
end
{
"metadata" => {
"name" => "production-auto-deploy",
"labels" => {
"app" => "production",
"app.kubernetes.io/managed-by" => "Helm",
"chart" => "auto-deploy-app-2.0.0-beta.2",
"heritage" => "Helm",
"release" => "production"
},
"annotations" => {
"kubernetes.io/ingress.class" => "nginx",
"kubernetes.io/tls-acme" => "true",
"meta.helm.sh/release-name" => "production",
"meta.helm.sh/release-namespace" => "awesome-app-1-production"
}.merge(additional_annotations)
}
}
end
# This is a partial response, it will have many more elements in reality but # This is a partial response, it will have many more elements in reality but
# these are the ones we care about at the moment # these are the ones we care about at the moment
def kube_node def kube_node
...@@ -862,8 +935,8 @@ module KubernetesHelpers ...@@ -862,8 +935,8 @@ module KubernetesHelpers
end end
end end
def kube_deployment_rollout_status def kube_deployment_rollout_status(ingresses: [])
::Gitlab::Kubernetes::RolloutStatus.from_deployments(kube_deployment) ::Gitlab::Kubernetes::RolloutStatus.from_deployments(kube_deployment, ingresses: ingresses)
end end
def empty_deployment_rollout_status def empty_deployment_rollout_status
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment