Commit 0c753217 authored by Robert Hunt's avatar Robert Hunt Committed by Mayra Cabrera

Set up new edit compliance frameworks page

- Added new endpoint to controller and routes
- Updated HAML helper to add the framework ID if it is set
- Updated HAML helper to pass edit path
- Updated listing to pass the new edit path to the list item
- Updated list item to generate correct edit path and render edit icon
- Updated translations
parent 67d91db4
...@@ -7,6 +7,7 @@ import { s__ } from '~/locale'; ...@@ -7,6 +7,7 @@ import { s__ } from '~/locale';
import { DANGER, INFO } from '../constants'; import { DANGER, INFO } from '../constants';
import getComplianceFrameworkQuery from '../graphql/queries/get_compliance_framework.query.graphql'; import getComplianceFrameworkQuery from '../graphql/queries/get_compliance_framework.query.graphql';
import { injectIdIntoEditPath } from '../utils';
import DeleteModal from './delete_modal.vue'; import DeleteModal from './delete_modal.vue';
import EmptyState from './list_empty_state.vue'; import EmptyState from './list_empty_state.vue';
import ListItem from './list_item.vue'; import ListItem from './list_item.vue';
...@@ -27,6 +28,10 @@ export default { ...@@ -27,6 +28,10 @@ export default {
type: String, type: String,
required: true, required: true,
}, },
editFrameworkPath: {
type: String,
required: true,
},
emptyStateSvgPath: { emptyStateSvgPath: {
type: String, type: String,
required: true, required: true,
...@@ -56,10 +61,15 @@ export default { ...@@ -56,10 +61,15 @@ export default {
update(data) { update(data) {
const nodes = data.namespace?.complianceFrameworks?.nodes; const nodes = data.namespace?.complianceFrameworks?.nodes;
return ( return (
nodes?.map((framework) => ({ nodes?.map((framework) => {
...framework, const parsedId = getIdFromGraphQLId(framework.id);
parsedId: getIdFromGraphQLId(framework.id),
})) || [] return {
...framework,
parsedId,
editPath: injectIdIntoEditPath(this.editFrameworkPath, parsedId),
};
}) || []
); );
}, },
error(error) { error(error) {
......
<script> <script>
import { GlLabel, GlButton, GlTooltipDirective } from '@gitlab/ui'; import { GlLabel, GlButton, GlTooltipDirective } from '@gitlab/ui';
import { s__ } from '~/locale'; import { DELETE_BUTTON_LABEL, EDIT_BUTTON_LABEL } from '../constants';
export default { export default {
directives: { directives: {
...@@ -26,17 +26,18 @@ export default { ...@@ -26,17 +26,18 @@ export default {
}, },
}, },
i18n: { i18n: {
deleteFramework: s__('ComplianceFrameworks|Delete framework'), editFramework: EDIT_BUTTON_LABEL,
deleteFramework: DELETE_BUTTON_LABEL,
}, },
}; };
</script> </script>
<template> <template>
<div <div
class="gl-display-flex gl-align-items-center gl-justify-content-space-between gl-inset-border-1-gray-100 gl-px-5 gl-p-6 gl-mb-4" class="gl-display-flex gl-align-items-center gl-justify-content-space-between gl-inset-border-1-gray-100 gl-px-5 gl-p-6 gl-mb-4 gl-rounded-base"
> >
<div class="gl-w-quarter gl-mr-3 gl-flex-shrink-0"> <div class="gl-w-quarter gl-mr-3 gl-flex-shrink-0">
<gl-label <gl-label
target="#" :target="framework.editPath"
:background-color="framework.color" :background-color="framework.color"
:title="framework.name" :title="framework.name"
:scoped="isScoped" :scoped="isScoped"
...@@ -46,9 +47,20 @@ export default { ...@@ -46,9 +47,20 @@ export default {
<p class="gl-w-full gl-m-0!" data-testid="compliance-framework-description"> <p class="gl-w-full gl-m-0!" data-testid="compliance-framework-description">
{{ framework.description }} {{ framework.description }}
</p> </p>
<div> <div class="gl-display-flex">
<gl-button
v-gl-tooltip="$options.i18n.editFramework"
:loading="loading"
:disabled="loading"
:aria-label="$options.i18n.editFramework"
:href="framework.editPath"
data-testid="compliance-framework-edit-button"
icon="pencil"
category="tertiary"
/>
<gl-button <gl-button
v-gl-tooltip="$options.i18n.deleteFramework" v-gl-tooltip="$options.i18n.deleteFramework"
class="gl-ml-3"
:loading="loading" :loading="loading"
:disabled="loading" :disabled="loading"
:aria-label="$options.i18n.deleteFramework" :aria-label="$options.i18n.deleteFramework"
......
...@@ -4,12 +4,16 @@ export const DANGER = 'danger'; ...@@ -4,12 +4,16 @@ export const DANGER = 'danger';
export const INFO = 'info'; export const INFO = 'info';
export const FETCH_ERROR = s__( export const FETCH_ERROR = s__(
'ComplianceFrameworks|Error fetching compliance frameworks data. Please refresh the page', 'ComplianceFrameworks|Error fetching compliance frameworks data. Please refresh the page or try a different framework',
); );
export const SAVE_ERROR = s__( export const SAVE_ERROR = s__(
'ComplianceFrameworks|Unable to save this compliance framework. Please try again', 'ComplianceFrameworks|Unable to save this compliance framework. Please try again',
); );
export const EDIT_BUTTON_LABEL = s__('ComplianceFrameworks|Edit framework');
export const DELETE_BUTTON_LABEL = s__('ComplianceFrameworks|Delete framework');
export const EDIT_PATH_ID_FORMAT = /\/id\//;
// Check that it matches the format [FILE].y(a)ml@[GROUP]/[PROJECT] // Check that it matches the format [FILE].y(a)ml@[GROUP]/[PROJECT]
export const PIPELINE_CONFIGURATION_PATH_FORMAT = /^([^@]*\.ya?ml)@([^/]*)\/(.*)$/; export const PIPELINE_CONFIGURATION_PATH_FORMAT = /^([^@]*\.ya?ml)@([^/]*)\/(.*)$/;
...@@ -15,7 +15,7 @@ const createComplianceFrameworksListApp = (el) => { ...@@ -15,7 +15,7 @@ const createComplianceFrameworksListApp = (el) => {
return false; return false;
} }
const { addFrameworkPath, emptyStateSvgPath, groupPath } = el.dataset; const { addFrameworkPath, editFrameworkPath, emptyStateSvgPath, groupPath } = el.dataset;
return new Vue({ return new Vue({
el, el,
...@@ -24,6 +24,7 @@ const createComplianceFrameworksListApp = (el) => { ...@@ -24,6 +24,7 @@ const createComplianceFrameworksListApp = (el) => {
return createElement(Form, { return createElement(Form, {
props: { props: {
addFrameworkPath, addFrameworkPath,
editFrameworkPath,
emptyStateSvgPath, emptyStateSvgPath,
groupPath, groupPath,
}, },
......
import Api from '~/api'; import Api from '~/api';
import httpStatus from '~/lib/utils/http_status'; import httpStatus from '~/lib/utils/http_status';
import { PIPELINE_CONFIGURATION_PATH_FORMAT } from './constants'; import { EDIT_PATH_ID_FORMAT, PIPELINE_CONFIGURATION_PATH_FORMAT } from './constants';
const isNumeric = (value) => {
return !Number.isNaN(parseInt(value, 10));
};
export const injectIdIntoEditPath = (path, id) => {
if (!path.match(EDIT_PATH_ID_FORMAT) || !isNumeric(id)) {
return '';
}
return path.replace(EDIT_PATH_ID_FORMAT, `/${id}/`);
};
export const initialiseFormData = () => ({ export const initialiseFormData = () => ({
name: null, name: null,
......
import { createComplianceFrameworksFormApp } from 'ee/groups/settings/compliance_frameworks/init_form';
createComplianceFrameworksFormApp(document.getElementById('js-compliance-frameworks-form'));
...@@ -11,6 +11,9 @@ class Groups::ComplianceFrameworksController < Groups::ApplicationController ...@@ -11,6 +11,9 @@ class Groups::ComplianceFrameworksController < Groups::ApplicationController
def new def new
end end
def edit
end
protected protected
def check_group_compliance_frameworks_available! def check_group_compliance_frameworks_available!
......
...@@ -3,31 +3,33 @@ ...@@ -3,31 +3,33 @@
module ComplianceManagement module ComplianceManagement
module ComplianceFramework module ComplianceFramework
module GroupSettingsHelper module GroupSettingsHelper
def show_compliance_frameworks? def show_compliance_frameworks?(group)
can?(current_user, :admin_compliance_framework, @group) can?(current_user, :admin_compliance_framework, group)
end end
def compliance_frameworks_list_data def compliance_frameworks_list_data(group)
{ {
empty_state_svg_path: image_path('illustrations/welcome/ee_trial.svg'), empty_state_svg_path: image_path('illustrations/welcome/ee_trial.svg'),
group_path: @group.full_path, group_path: group.full_path,
add_framework_path: new_group_compliance_framework_path(@group) add_framework_path: new_group_compliance_framework_path(group),
edit_framework_path: edit_group_compliance_framework_path(group, :id)
} }
end end
def compliance_frameworks_new_form_data def compliance_frameworks_form_data(group, framework_id = nil)
{ {
group_path: @group.full_path, framework_id: framework_id,
group_edit_path: edit_group_path(@group, anchor: 'js-compliance-frameworks-settings'), group_path: group.full_path,
group_edit_path: edit_group_path(group, anchor: 'js-compliance-frameworks-settings'),
graphql_field_name: ComplianceManagement::Framework.name, graphql_field_name: ComplianceManagement::Framework.name,
pipeline_configuration_full_path_enabled: pipeline_configuration_full_path_enabled?.to_s pipeline_configuration_full_path_enabled: pipeline_configuration_full_path_enabled?(group).to_s
} }
end end
private private
def pipeline_configuration_full_path_enabled? def pipeline_configuration_full_path_enabled?(group)
can?(current_user, :admin_compliance_pipeline_configuration, @group) can?(current_user, :admin_compliance_pipeline_configuration, group)
end end
end end
end end
......
- expanded = expanded_by_default? - expanded = expanded_by_default?
- if show_compliance_frameworks? - if show_compliance_frameworks?(@group)
%section.settings.no-animate#js-compliance-frameworks-settings{ class: ('expanded' if expanded) } %section.settings.no-animate#js-compliance-frameworks-settings{ class: ('expanded' if expanded) }
.settings-header .settings-header
%h4 %h4
...@@ -10,4 +10,4 @@ ...@@ -10,4 +10,4 @@
%p %p
= s_('GroupSettings|Configure frameworks to apply enforceable rules to projects.') = s_('GroupSettings|Configure frameworks to apply enforceable rules to projects.')
.settings-content .settings-content
#js-compliance-frameworks-list{ data: compliance_frameworks_list_data } #js-compliance-frameworks-list{ data: compliance_frameworks_list_data(@group) }
- add_to_breadcrumbs _('General Settings'), edit_group_path(@group)
- title = s_('ComplianceFramework|Edit Compliance Framework')
- page_title title
%h3.page-title= title
#js-compliance-frameworks-form{ data: compliance_frameworks_form_data(@group, params[:id]) }
...@@ -4,4 +4,4 @@ ...@@ -4,4 +4,4 @@
%h3.page-title= title %h3.page-title= title
#js-compliance-frameworks-form{ data: compliance_frameworks_new_form_data } #js-compliance-frameworks-form{ data: compliance_frameworks_form_data(@group) }
...@@ -11,7 +11,7 @@ constraints(::Constraints::GroupUrlConstrainer.new) do ...@@ -11,7 +11,7 @@ constraints(::Constraints::GroupUrlConstrainer.new) do
patch :override, on: :member patch :override, on: :member
end end
resources :compliance_frameworks, only: [:new] resources :compliance_frameworks, only: [:new, :edit]
get '/analytics', to: redirect('groups/%{group_id}/-/analytics/value_stream_analytics') get '/analytics', to: redirect('groups/%{group_id}/-/analytics/value_stream_analytics')
resource :contribution_analytics, only: [:show] resource :contribution_analytics, only: [:show]
......
...@@ -2,33 +2,59 @@ import { GlLabel } from '@gitlab/ui'; ...@@ -2,33 +2,59 @@ import { GlLabel } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import ListItem from 'ee/groups/settings/compliance_frameworks/components/list_item.vue'; import ListItem from 'ee/groups/settings/compliance_frameworks/components/list_item.vue';
import {
DELETE_BUTTON_LABEL,
EDIT_BUTTON_LABEL,
} from 'ee/groups/settings/compliance_frameworks/constants';
import { createMockDirective, getBinding } from 'helpers/vue_mock_directive'; import { createMockDirective, getBinding } from 'helpers/vue_mock_directive';
import { extendedWrapper } from 'helpers/vue_test_utils_helper';
describe('ListItem', () => { describe('ListItem', () => {
let wrapper; let wrapper;
const framework = { name: 'framework', description: 'a framework', color: '#112233' }; const framework = {
const findLabel = () => wrapper.find(GlLabel); parsedId: 1,
const findDescription = () => wrapper.find('[data-testid="compliance-framework-description"]'); name: 'framework',
const findDeleteButton = () => wrapper.find('[data-testid="compliance-framework-delete-button"]'); description: 'a framework',
color: '#112233',
editPath: 'group/framework/1/edit',
};
const findLabel = () => wrapper.findComponent(GlLabel);
const findDescription = () => wrapper.findByTestId('compliance-framework-description');
const findEditButton = () => wrapper.findByTestId('compliance-framework-edit-button');
const findDeleteButton = () => wrapper.findByTestId('compliance-framework-delete-button');
const createComponent = (props = {}) => { const createComponent = (props = {}) => {
wrapper = shallowMount(ListItem, { wrapper = extendedWrapper(
propsData: { shallowMount(ListItem, {
framework, propsData: {
loading: false, framework,
...props, loading: false,
}, ...props,
directives: { },
GlTooltip: createMockDirective(), directives: {
}, GlTooltip: createMockDirective(),
}); },
}),
);
}; };
afterEach(() => { afterEach(() => {
wrapper.destroy(); wrapper.destroy();
}); });
const displaysTheButton = (button, icon, ariaLabel) => {
expect(button.props('icon')).toBe(icon);
expect(button.props('disabled')).toBe(false);
expect(button.props('loading')).toBe(false);
expect(button.attributes('aria-label')).toBe(ariaLabel);
};
const disablesTheButton = (button) => {
expect(button.props('disabled')).toBe(true);
expect(button.props('loading')).toBe(true);
};
it('displays the description defined by the framework', () => { it('displays the description defined by the framework', () => {
createComponent(); createComponent();
...@@ -39,6 +65,7 @@ describe('ListItem', () => { ...@@ -39,6 +65,7 @@ describe('ListItem', () => {
createComponent(); createComponent();
expect(findLabel().props('title')).toBe('framework'); expect(findLabel().props('title')).toBe('framework');
expect(findLabel().props('target')).toBe(framework.editPath);
expect(findLabel().props('scoped')).toBe(false); expect(findLabel().props('scoped')).toBe(false);
}); });
...@@ -46,20 +73,27 @@ describe('ListItem', () => { ...@@ -46,20 +73,27 @@ describe('ListItem', () => {
createComponent({ framework: { ...framework, name: 'scoped::framework' } }); createComponent({ framework: { ...framework, name: 'scoped::framework' } });
expect(findLabel().props('title')).toBe('scoped::framework'); expect(findLabel().props('title')).toBe('scoped::framework');
expect(findLabel().props('target')).toBe(framework.editPath);
expect(findLabel().props('scoped')).toBe(true); expect(findLabel().props('scoped')).toBe(true);
expect(findLabel().props('disabled')).toBe(false); expect(findLabel().props('disabled')).toBe(false);
}); });
it('displays the edit button', () => {
createComponent();
const button = findEditButton();
displaysTheButton(button, 'pencil', EDIT_BUTTON_LABEL);
expect(button.attributes('href')).toBe('group/framework/1/edit');
});
it('displays a delete button', () => { it('displays a delete button', () => {
createComponent(); createComponent();
const button = findDeleteButton(); const button = findDeleteButton();
const tooltip = getBinding(button.element, 'gl-tooltip'); const tooltip = getBinding(button.element, 'gl-tooltip');
expect(button.props('icon')).toBe('remove'); displaysTheButton(button, 'remove', DELETE_BUTTON_LABEL);
expect(button.props('disabled')).toBe(false);
expect(button.props('loading')).toBe(false);
expect(button.attributes('aria-label')).toBe('Delete framework');
expect(tooltip.value).toBe('Delete framework'); expect(tooltip.value).toBe('Delete framework');
}); });
...@@ -81,8 +115,11 @@ describe('ListItem', () => { ...@@ -81,8 +115,11 @@ describe('ListItem', () => {
}); });
it('disables the delete button and shows loading', () => { it('disables the delete button and shows loading', () => {
expect(findDeleteButton().props('disabled')).toBe(true); disablesTheButton(findDeleteButton());
expect(findDeleteButton().props('loading')).toBe(true); });
it('disables the edit button and shows loading', () => {
disablesTheButton(findEditButton());
}); });
}); });
}); });
...@@ -27,14 +27,14 @@ describe('List', () => { ...@@ -27,14 +27,14 @@ describe('List', () => {
const fetchLoading = jest.fn().mockResolvedValue(new Promise(() => {})); const fetchLoading = jest.fn().mockResolvedValue(new Promise(() => {}));
const fetchWithErrors = jest.fn().mockRejectedValue(sentryError); const fetchWithErrors = jest.fn().mockRejectedValue(sentryError);
const findAlert = () => wrapper.find(GlAlert); const findAlert = () => wrapper.findComponent(GlAlert);
const findDeleteModal = () => wrapper.findComponent(DeleteModal); const findDeleteModal = () => wrapper.findComponent(DeleteModal);
const findLoadingIcon = () => wrapper.find(GlLoadingIcon); const findLoadingIcon = () => wrapper.findComponent(GlLoadingIcon);
const findEmptyState = () => wrapper.find(EmptyState); const findEmptyState = () => wrapper.findComponent(EmptyState);
const findTabs = () => wrapper.findAll(GlTab); const findTabs = () => wrapper.findAllComponents(GlTab);
const findAddBtn = () => wrapper.find(GlButton); const findAddBtn = () => wrapper.findComponent(GlButton);
const findTabsContainer = () => wrapper.find(GlTabs); const findTabsContainer = () => wrapper.findComponent(GlTabs);
const findListItems = () => wrapper.findAll(ListItem); const findListItems = () => wrapper.findAllComponents(ListItem);
function createMockApolloProvider(resolverMock) { function createMockApolloProvider(resolverMock) {
localVue.use(VueApollo); localVue.use(VueApollo);
...@@ -50,6 +50,7 @@ describe('List', () => { ...@@ -50,6 +50,7 @@ describe('List', () => {
apolloProvider: createMockApolloProvider(resolverMock), apolloProvider: createMockApolloProvider(resolverMock),
propsData: { propsData: {
addFrameworkPath: 'group/framework/new', addFrameworkPath: 'group/framework/new',
editFrameworkPath: 'group/framework/id/edit',
emptyStateSvgPath: 'dir/image.svg', emptyStateSvgPath: 'dir/image.svg',
groupPath: 'group-1', groupPath: 'group-1',
}, },
...@@ -170,7 +171,7 @@ describe('List', () => { ...@@ -170,7 +171,7 @@ describe('List', () => {
expect(findListItems()).toHaveLength(2); expect(findListItems()).toHaveLength(2);
findListItems().wrappers.forEach((item) => findListItems().wrappers.forEach((item) =>
expect(item.props()).toEqual( expect(item.props()).toStrictEqual(
expect.objectContaining({ expect.objectContaining({
framework: { framework: {
id: expect.stringContaining('gid://gitlab/ComplianceManagement::Framework/'), id: expect.stringContaining('gid://gitlab/ComplianceManagement::Framework/'),
...@@ -181,6 +182,7 @@ describe('List', () => { ...@@ -181,6 +182,7 @@ describe('List', () => {
PIPELINE_CONFIGURATION_PATH_FORMAT, PIPELINE_CONFIGURATION_PATH_FORMAT,
), ),
color: expect.stringMatching(/^#([0-9A-F]{3}){1,2}$/i), color: expect.stringMatching(/^#([0-9A-F]{3}){1,2}$/i),
editPath: expect.stringMatching(/^group\/framework\/[0-9+]\/edit$/i),
}, },
loading: false, loading: false,
}), }),
......
...@@ -16,6 +16,20 @@ describe('Utils', () => { ...@@ -16,6 +16,20 @@ describe('Utils', () => {
mock.restore(); mock.restore();
}); });
describe('injectIdIntoEditPath', () => {
it.each`
path | id | output
${'group/framework/abc/edit'} | ${1} | ${''}
${'group/framework/id/edit'} | ${undefined} | ${''}
${'group/framework/id/edit'} | ${null} | ${''}
${'group/framework/id/edit'} | ${'abc'} | ${''}
${'group/framework/id/edit'} | ${'1'} | ${'group/framework/1/edit'}
${'group/framework/id/edit'} | ${1} | ${'group/framework/1/edit'}
`('should return $output when $path and $id are given', ({ path, id, output }) => {
expect(Utils.injectIdIntoEditPath(path, id)).toStrictEqual(output);
});
});
describe('initialiseFormData', () => { describe('initialiseFormData', () => {
it('returns the initial form data object', () => { it('returns the initial form data object', () => {
expect(Utils.initialiseFormData()).toStrictEqual({ expect(Utils.initialiseFormData()).toStrictEqual({
......
...@@ -7,12 +7,11 @@ RSpec.describe ComplianceManagement::ComplianceFramework::GroupSettingsHelper do ...@@ -7,12 +7,11 @@ RSpec.describe ComplianceManagement::ComplianceFramework::GroupSettingsHelper do
let_it_be(:current_user) { build(:admin) } let_it_be(:current_user) { build(:admin) }
before do before do
assign(:group, group)
allow(helper).to receive(:current_user) { current_user } allow(helper).to receive(:current_user) { current_user }
end end
describe '#show_compliance_frameworks?' do describe '#show_compliance_frameworks?' do
subject { helper.show_compliance_frameworks? } subject { helper.show_compliance_frameworks?(group) }
context 'the user has permission' do context 'the user has permission' do
before do before do
...@@ -33,30 +32,48 @@ RSpec.describe ComplianceManagement::ComplianceFramework::GroupSettingsHelper do ...@@ -33,30 +32,48 @@ RSpec.describe ComplianceManagement::ComplianceFramework::GroupSettingsHelper do
describe '#compliance_frameworks_list_data' do describe '#compliance_frameworks_list_data' do
it 'returns the correct data' do it 'returns the correct data' do
expect(helper.compliance_frameworks_list_data).to contain_exactly( expect(helper.compliance_frameworks_list_data(group)).to contain_exactly(
[:empty_state_svg_path, ActionController::Base.helpers.image_path('illustrations/welcome/ee_trial.svg')], [:empty_state_svg_path, ActionController::Base.helpers.image_path('illustrations/welcome/ee_trial.svg')],
[:group_path, group.full_path], [:group_path, group.full_path],
[:add_framework_path, new_group_compliance_framework_path(group)] [:add_framework_path, new_group_compliance_framework_path(group)],
[:edit_framework_path, edit_group_compliance_framework_path(group, :id)]
) )
end end
end end
describe '#compliance_frameworks_new_form_data' do describe '#compliance_frameworks_form_data' do
subject { helper.compliance_frameworks_new_form_data } let(:framework_id) { nil }
subject { helper.compliance_frameworks_form_data(group, framework_id) }
shared_examples 'returns the correct data' do |pipeline_configuration_enabled| shared_examples 'returns the correct data' do |pipeline_configuration_enabled|
before do before do
allow(helper).to receive(:can?).with(current_user, :admin_compliance_pipeline_configuration, group).and_return(pipeline_configuration_enabled) allow(helper).to receive(:can?).with(current_user, :admin_compliance_pipeline_configuration, group).and_return(pipeline_configuration_enabled)
end end
it { it 'does not contain a framework ID' do
is_expected.to contain_exactly( is_expected.to contain_exactly(
[:framework_id, nil],
[:group_path, group.full_path], [:group_path, group.full_path],
[:group_edit_path, edit_group_path(group, anchor: 'js-compliance-frameworks-settings')], [:group_edit_path, edit_group_path(group, anchor: 'js-compliance-frameworks-settings')],
[:graphql_field_name, ComplianceManagement::Framework.name], [:graphql_field_name, ComplianceManagement::Framework.name],
[:pipeline_configuration_full_path_enabled, pipeline_configuration_enabled.to_s] [:pipeline_configuration_full_path_enabled, pipeline_configuration_enabled.to_s]
) )
} end
context 'with a framework ID' do
let(:framework_id) { 12345 }
it {
is_expected.to contain_exactly(
[:framework_id, framework_id],
[:group_path, group.full_path],
[:group_edit_path, edit_group_path(group, anchor: 'js-compliance-frameworks-settings')],
[:graphql_field_name, ComplianceManagement::Framework.name],
[:pipeline_configuration_full_path_enabled, pipeline_configuration_enabled.to_s]
)
}
end
end end
context 'the user has pipeline configuration permission' do context 'the user has pipeline configuration permission' do
......
...@@ -5,6 +5,7 @@ require 'spec_helper' ...@@ -5,6 +5,7 @@ require 'spec_helper'
RSpec.describe 'group compliance frameworks' do RSpec.describe 'group compliance frameworks' do
let_it_be(:group) { create(:group) } let_it_be(:group) { create(:group) }
let_it_be(:user) { create(:user) } let_it_be(:user) { create(:user) }
let_it_be(:framework) { create(:compliance_framework, namespace: group, name: 'Framework') }
before do before do
login_as(user) login_as(user)
...@@ -23,6 +24,14 @@ RSpec.describe 'group compliance frameworks' do ...@@ -23,6 +24,14 @@ RSpec.describe 'group compliance frameworks' do
expect(response).to have_gitlab_http_status(:not_found) expect(response).to have_gitlab_http_status(:not_found)
end end
end end
describe 'GET /groups/:group/-/compliance_frameworks/:id/edit' do
it 'returns 404 not found' do
get edit_group_compliance_framework_path(group, framework)
expect(response).to have_gitlab_http_status(:not_found)
end
end
end end
context 'when compliance frameworks feature is enabled' do context 'when compliance frameworks feature is enabled' do
...@@ -47,5 +56,22 @@ RSpec.describe 'group compliance frameworks' do ...@@ -47,5 +56,22 @@ RSpec.describe 'group compliance frameworks' do
end end
end end
end end
describe 'GET /groups/:group/-/compliance_frameworks/:id/edit' do
it 'renders template' do
group.add_owner(user)
get edit_group_compliance_framework_path(group, framework)
expect(response).to render_template 'groups/compliance_frameworks/edit'
end
context 'with unauthorized user' do
it 'returns 404 not found' do
get edit_group_compliance_framework_path(group, framework)
expect(response).to have_gitlab_http_status(:not_found)
end
end
end
end end
end end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'groups/compliance_frameworks/edit.html.haml' do
let_it_be(:group) { build(:group) }
let_it_be(:user) { build(:user) }
before do
assign(:group, group)
allow(view).to receive(:current_user).and_return(user)
allow(user).to receive(:can?).with(:admin_compliance_pipeline_configuration, group).and_return(true)
allow(view).to receive(:params).and_return(id: 1)
end
it 'shows the compliance frameworks form', :aggregate_failures do
render
expect(rendered).to have_content('Edit Compliance Framework')
expect(rendered).to have_css('#js-compliance-frameworks-form')
expect(rendered).to have_css('[data-framework-id="1"]')
end
end
...@@ -7687,12 +7687,18 @@ msgstr "" ...@@ -7687,12 +7687,18 @@ msgstr ""
msgid "ComplianceFrameworks|Delete framework" msgid "ComplianceFrameworks|Delete framework"
msgstr "" msgstr ""
msgid "ComplianceFrameworks|Edit framework"
msgstr ""
msgid "ComplianceFrameworks|Error deleting the compliance framework. Please try again" msgid "ComplianceFrameworks|Error deleting the compliance framework. Please try again"
msgstr "" msgstr ""
msgid "ComplianceFrameworks|Error fetching compliance frameworks data. Please refresh the page" msgid "ComplianceFrameworks|Error fetching compliance frameworks data. Please refresh the page"
msgstr "" msgstr ""
msgid "ComplianceFrameworks|Error fetching compliance frameworks data. Please refresh the page or try a different framework"
msgstr ""
msgid "ComplianceFrameworks|Invalid format: it should follow the format [PATH].y(a)ml@[GROUP]/[PROJECT]" msgid "ComplianceFrameworks|Invalid format: it should follow the format [PATH].y(a)ml@[GROUP]/[PROJECT]"
msgstr "" msgstr ""
...@@ -7717,6 +7723,9 @@ msgstr "" ...@@ -7717,6 +7723,9 @@ msgstr ""
msgid "ComplianceFrameworks|e.g. include-gitlab.ci.yml@group-name/project-name" msgid "ComplianceFrameworks|e.g. include-gitlab.ci.yml@group-name/project-name"
msgstr "" msgstr ""
msgid "ComplianceFramework|Edit Compliance Framework"
msgstr ""
msgid "ComplianceFramework|GDPR" msgid "ComplianceFramework|GDPR"
msgstr "" msgstr ""
......
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