Commit 08095221 authored by Kamil Trzciński's avatar Kamil Trzciński

Merge branch '34758-extend-can-create-cluster-logic-ee' into 'master'

EE port of Allow user to add cluster when there are ancestor clusters

See merge request gitlab-org/gitlab-ee!8706
parents ad0f45fe 6f4e3ebd
# frozen_string_literal: true
module ClusterableActions
private
# Overridden on EE module
def multiple_clusters_available?
false
end
def clusterable_has_clusters?
!subject.clusters.empty?
end
end
# frozen_string_literal: true
class GroupPolicy < BasePolicy
include ClusterableActions
desc "Group is public"
with_options scope: :subject, score: 0
condition(:public_group) { @subject.public? }
......@@ -27,6 +29,9 @@ class GroupPolicy < BasePolicy
GroupProjectsFinder.new(group: @subject, current_user: @user, options: { include_subgroups: true }).execute.any?
end
condition(:has_clusters, scope: :subject) { clusterable_has_clusters? }
condition(:can_have_multiple_clusters) { multiple_clusters_available? }
with_options scope: :subject, score: 0
condition(:request_access_enabled) { @subject.request_access_enabled }
......@@ -45,7 +50,7 @@ class GroupPolicy < BasePolicy
enable :read_label
end
rule { admin } .enable :read_group
rule { admin }.enable :read_group
rule { has_projects }.policy do
enable :read_group
......@@ -67,6 +72,7 @@ class GroupPolicy < BasePolicy
enable :admin_pipeline
enable :admin_build
enable :read_cluster
enable :add_cluster
enable :create_cluster
enable :update_cluster
enable :admin_cluster
......@@ -106,6 +112,8 @@ class GroupPolicy < BasePolicy
rule { owner & (~share_with_group_locked | ~has_parent | ~parent_share_with_group_locked | can_change_parent_share_with_group_lock) }.enable :change_share_with_group_lock
rule { ~can_have_multiple_clusters & has_clusters }.prevent :add_cluster
def access_level
return GroupMember::NO_ACCESS if @user.nil?
......
......@@ -2,6 +2,8 @@
class ProjectPolicy < BasePolicy
extend ClassMethods
include ClusterableActions
READONLY_FEATURES_WHEN_ARCHIVED = %i[
issue
list
......@@ -102,6 +104,9 @@ class ProjectPolicy < BasePolicy
@subject.feature_available?(:merge_requests, @user)
end
condition(:has_clusters, scope: :subject) { clusterable_has_clusters? }
condition(:can_have_multiple_clusters) { multiple_clusters_available? }
features = %w[
merge_requests
issues
......@@ -256,6 +261,7 @@ class ProjectPolicy < BasePolicy
enable :read_pages
enable :update_pages
enable :read_cluster
enable :add_cluster
enable :create_cluster
enable :update_cluster
enable :admin_cluster
......@@ -380,6 +386,8 @@ class ProjectPolicy < BasePolicy
(can?(:read_project_for_iids) & merge_requests_visible_to_user) | can?(:read_merge_request)
end.enable :read_merge_request_iid
rule { ~can_have_multiple_clusters & has_clusters }.prevent :add_cluster
private
def team_member?
......
......@@ -12,6 +12,10 @@ class ClusterablePresenter < Gitlab::View::Presenter::Delegated
.fabricate!
end
def can_add_cluster?
can?(current_user, :add_cluster, clusterable)
end
def can_create_cluster?
can?(current_user, :create_cluster, clusterable)
end
......
-# This partial is overridden in EE
.nav-controls
- if clusterable.can_create_cluster? && clusterable.clusters.empty?
- if clusterable.can_add_cluster?
= link_to s_('ClusterIntegration|Add Kubernetes cluster'), clusterable.new_path, class: 'btn btn-success js-add-cluster'
- else
%span.btn.btn-add-cluster.disabled.js-add-cluster
......
......@@ -9,6 +9,6 @@
= clusterable.empty_state_help_text
= clusterable.learn_more_link
- if clusterable.can_create_cluster?
- if clusterable.can_add_cluster?
.text-center
= link_to s_('ClusterIntegration|Add Kubernetes cluster'), clusterable.new_path, class: 'btn btn-success'
---
title: Allow user to add Kubernetes cluster for clusterable when there are ancestor clusters
merge_request: 23569
author:
type: other
# frozen_string_literal: true
module EE
module ClusterableActions
extend ::Gitlab::Utils::Override
private
override :multiple_clusters_available?
def multiple_clusters_available?
subject.feature_available?(:multiple_clusters)
end
end
end
......@@ -82,3 +82,5 @@ module EE
end
end
end
EE::GroupPolicy.include(EE::ClusterableActions)
......@@ -212,3 +212,5 @@ module EE
end
end
end
EE::ProjectPolicy.include(EE::ClusterableActions)
.nav-controls
- if clusterable.feature_available?(:multiple_clusters)
= link_to s_("ClusterIntegration|Add Kubernetes cluster"), clusterable.new_path, class: "btn btn-success btn-add-cluster has-tooltip js-add-cluster"
- else
%span.btn.btn-add-cluster.disabled.js-add-cluster
= s_("ClusterIntegration|Add Kubernetes cluster")
......@@ -464,4 +464,15 @@ describe GroupPolicy do
it { is_expected.to be_disallowed(:read_group_security_dashboard) }
end
end
it_behaves_like 'ee clusterable policies' do
let(:clusterable) { create(:group) }
let(:cluster) do
create(:cluster,
:provided_by_gcp,
:group,
groups: [clusterable])
end
end
end
......@@ -697,4 +697,15 @@ describe ProjectPolicy do
it { is_expected.to be_disallowed(:create_web_ide_terminal) }
end
end
it_behaves_like 'ee clusterable policies' do
let(:clusterable) { create(:project, :repository) }
let(:cluster) do
create(:cluster,
:provided_by_gcp,
:project,
projects: [clusterable])
end
end
end
# frozen_string_literal: true
require 'spec_helper'
shared_examples 'ee clusterable policies' do
describe '#can_add_cluster?' do
let(:current_user) { create(:user) }
subject { described_class.new(current_user, clusterable) }
before do
clusterable.add_maintainer(current_user)
end
context 'when multiple_clusters feature is available' do
before do
stub_licensed_features(multiple_clusters: true)
end
context 'when clusterable has clusters' do
before do
cluster
end
it { is_expected.to be_allowed(:add_cluster) }
end
context 'when clusterable does not have clusters' do
it { is_expected.to be_allowed(:add_cluster) }
end
end
context 'when multiple_clusters feature is not available' do
before do
stub_licensed_features(multiple_clusters: false)
end
context 'when clusterable has clusters' do
before do
cluster
end
it { is_expected.to be_disallowed(:add_cluster) }
end
context 'when clusterable does not have clusters' do
it { is_expected.to be_allowed(:add_cluster) }
end
end
end
end
......@@ -26,7 +26,8 @@ describe GroupPolicy do
:read_cluster,
:create_cluster,
:update_cluster,
:admin_cluster
:admin_cluster,
:add_cluster
]
end
......@@ -484,4 +485,14 @@ describe GroupPolicy do
it { expect_disallowed(:change_share_with_group_lock) }
end
end
it_behaves_like 'clusterable policies' do
let(:clusterable) { create(:group) }
let(:cluster) do
create(:cluster,
:provided_by_gcp,
:group,
groups: [clusterable])
end
end
end
......@@ -48,7 +48,7 @@ describe ProjectPolicy do
update_deployment admin_project_snippet
admin_project_member admin_note admin_wiki admin_project
admin_commit_status admin_build admin_container_image
admin_pipeline admin_environment admin_deployment
admin_pipeline admin_environment admin_deployment add_cluster
]
end
......@@ -519,4 +519,14 @@ describe ProjectPolicy do
expect_disallowed(*maintainer_abilities)
end
end
it_behaves_like 'clusterable policies' do
let(:clusterable) { create(:project, :repository) }
let(:cluster) do
create(:cluster,
:provided_by_gcp,
:project,
projects: [clusterable])
end
end
end
......@@ -14,4 +14,68 @@ describe ClusterablePresenter do
expect(subject).to be_kind_of(ProjectClusterablePresenter)
end
end
shared_examples 'appropriate member permissions' do
context 'with a developer' do
before do
clusterable.add_developer(user)
end
it { is_expected.to be_falsy }
end
context 'with a maintainer' do
before do
clusterable.add_maintainer(user)
end
it { is_expected.to be_truthy }
end
end
describe '#can_create_cluster?' do
let(:user) { create(:user) }
subject { described_class.new(clusterable).can_create_cluster? }
before do
allow(clusterable).to receive(:current_user).and_return(user)
end
context 'when clusterable is a group' do
let(:clusterable) { create(:group) }
it_behaves_like 'appropriate member permissions'
end
context 'when clusterable is a project' do
let(:clusterable) { create(:project, :repository) }
it_behaves_like 'appropriate member permissions'
end
end
describe '#can_add_cluster?' do
let(:user) { create(:user) }
subject { described_class.new(clusterable).can_add_cluster? }
before do
clusterable.add_maintainer(user)
allow(clusterable).to receive(:current_user).and_return(user)
end
context 'when clusterable is a group' do
let(:clusterable) { create(:group) }
it_behaves_like 'appropriate member permissions'
end
context 'when clusterable is a project' do
let(:clusterable) { create(:project, :repository) }
it_behaves_like 'appropriate member permissions'
end
end
end
# frozen_string_literal: true
require 'spec_helper'
shared_examples 'clusterable policies' do
describe '#add_cluster?' do
let(:current_user) { create(:user) }
subject { described_class.new(current_user, clusterable) }
context 'with a developer' do
before do
clusterable.add_developer(current_user)
end
it { expect_disallowed(:add_cluster) }
end
context 'with a maintainer' do
before do
clusterable.add_maintainer(current_user)
end
context 'with no clusters' do
it { expect_allowed(:add_cluster) }
end
context 'with an existing cluster' do
before do
cluster
end
it { expect_disallowed(:add_cluster) }
end
end
end
end
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