Commit 2a87ae2e authored by GitLab Bot's avatar GitLab Bot

Add latest changes from gitlab-org/gitlab@master

parent eb0d9e20
<script> <script>
import { GlButton, GlFormGroup, GlFormInput, GlLink } from '@gitlab/ui'; import { GlButton, GlFormGroup, GlFormInput, GlFormCheckbox, GlLink } from '@gitlab/ui';
import Icon from '~/vue_shared/components/icon.vue'; import Icon from '~/vue_shared/components/icon.vue';
import { mapState, mapActions } from 'vuex'; import { mapState, mapActions } from 'vuex';
export default { export default {
components: { components: {
GlButton, GlButton,
GlFormCheckbox,
GlFormGroup, GlFormGroup,
GlFormInput, GlFormInput,
GlLink, GlLink,
...@@ -15,7 +16,15 @@ export default { ...@@ -15,7 +16,15 @@ export default {
return { placeholderUrl: 'https://my-url.grafana.net/my-dashboard' }; return { placeholderUrl: 'https://my-url.grafana.net/my-dashboard' };
}, },
computed: { computed: {
...mapState(['operationsSettingsEndpoint', 'grafanaToken', 'grafanaUrl']), ...mapState(['operationsSettingsEndpoint', 'grafanaToken', 'grafanaUrl', 'grafanaEnabled']),
integrationEnabled: {
get() {
return this.grafanaEnabled;
},
set(grafanaEnabled) {
this.setGrafanaEnabled(grafanaEnabled);
},
},
localGrafanaToken: { localGrafanaToken: {
get() { get() {
return this.grafanaToken; return this.grafanaToken;
...@@ -34,7 +43,12 @@ export default { ...@@ -34,7 +43,12 @@ export default {
}, },
}, },
methods: { methods: {
...mapActions(['setGrafanaUrl', 'setGrafanaToken', 'updateGrafanaIntegration']), ...mapActions([
'setGrafanaUrl',
'setGrafanaToken',
'setGrafanaEnabled',
'updateGrafanaIntegration',
]),
}, },
}; };
</script> </script>
...@@ -52,6 +66,13 @@ export default { ...@@ -52,6 +66,13 @@ export default {
</div> </div>
<div class="settings-content"> <div class="settings-content">
<form> <form>
<gl-form-checkbox
id="grafana-integration-enabled"
v-model="integrationEnabled"
class="mb-4"
>
{{ s__('GrafanaIntegration|Active') }}
</gl-form-checkbox>
<gl-form-group <gl-form-group
:label="s__('GrafanaIntegration|Grafana URL')" :label="s__('GrafanaIntegration|Grafana URL')"
label-for="grafana-url" label-for="grafana-url"
......
...@@ -4,7 +4,6 @@ import GrafanaIntegration from './components/grafana_integration.vue'; ...@@ -4,7 +4,6 @@ import GrafanaIntegration from './components/grafana_integration.vue';
export default () => { export default () => {
const el = document.querySelector('.js-grafana-integration'); const el = document.querySelector('.js-grafana-integration');
return new Vue({ return new Vue({
el, el,
store: store(el.dataset), store: store(el.dataset),
......
...@@ -9,6 +9,9 @@ export const setGrafanaUrl = ({ commit }, url) => commit(mutationTypes.SET_GRAFA ...@@ -9,6 +9,9 @@ export const setGrafanaUrl = ({ commit }, url) => commit(mutationTypes.SET_GRAFA
export const setGrafanaToken = ({ commit }, token) => export const setGrafanaToken = ({ commit }, token) =>
commit(mutationTypes.SET_GRAFANA_TOKEN, token); commit(mutationTypes.SET_GRAFANA_TOKEN, token);
export const setGrafanaEnabled = ({ commit }, enabled) =>
commit(mutationTypes.SET_GRAFANA_ENABLED, enabled);
export const updateGrafanaIntegration = ({ state, dispatch }) => export const updateGrafanaIntegration = ({ state, dispatch }) =>
axios axios
.patch(state.operationsSettingsEndpoint, { .patch(state.operationsSettingsEndpoint, {
...@@ -16,6 +19,7 @@ export const updateGrafanaIntegration = ({ state, dispatch }) => ...@@ -16,6 +19,7 @@ export const updateGrafanaIntegration = ({ state, dispatch }) =>
grafana_integration_attributes: { grafana_integration_attributes: {
grafana_url: state.grafanaUrl, grafana_url: state.grafanaUrl,
token: state.grafanaToken, token: state.grafanaToken,
enabled: state.grafanaEnabled,
}, },
}, },
}) })
......
export const SET_GRAFANA_URL = 'SET_GRAFANA_URL'; export const SET_GRAFANA_URL = 'SET_GRAFANA_URL';
export const SET_GRAFANA_TOKEN = 'SET_GRAFANA_TOKEN'; export const SET_GRAFANA_TOKEN = 'SET_GRAFANA_TOKEN';
export const SET_GRAFANA_ENABLED = 'SET_GRAFANA_ENABLED';
...@@ -7,4 +7,7 @@ export default { ...@@ -7,4 +7,7 @@ export default {
[types.SET_GRAFANA_TOKEN](state, token) { [types.SET_GRAFANA_TOKEN](state, token) {
state.grafanaToken = token; state.grafanaToken = token;
}, },
[types.SET_GRAFANA_ENABLED](state, enabled) {
state.grafanaEnabled = enabled;
},
}; };
import { parseBoolean } from '~/lib/utils/common_utils';
export default (initialState = {}) => ({ export default (initialState = {}) => ({
operationsSettingsEndpoint: initialState.operationsSettingsEndpoint, operationsSettingsEndpoint: initialState.operationsSettingsEndpoint,
grafanaToken: initialState.grafanaIntegrationToken || '', grafanaToken: initialState.grafanaIntegrationToken || '',
grafanaUrl: initialState.grafanaIntegrationUrl || '', grafanaUrl: initialState.grafanaIntegrationUrl || '',
grafanaEnabled: parseBoolean(initialState.grafanaIntegrationEnabled) || false,
}); });
...@@ -50,6 +50,6 @@ export default { ...@@ -50,6 +50,6 @@ export default {
}" }"
class="report-block-list-icon" class="report-block-list-icon"
> >
<icon :name="iconName" :size="statusIconSize" /> <icon :name="iconName" :size="statusIconSize" :data-qa-selector="`status_${status}_icon`" />
</div> </div>
</template> </template>
...@@ -46,6 +46,7 @@ export default { ...@@ -46,6 +46,7 @@ export default {
<li <li
:class="{ 'is-dismissed': issue.isDismissed }" :class="{ 'is-dismissed': issue.isDismissed }"
class="report-block-list-issue align-items-center" class="report-block-list-issue align-items-center"
data-qa-selector="report_item_row"
> >
<issue-status-icon <issue-status-icon
v-if="showReportSectionStatusIcon" v-if="showReportSectionStatusIcon"
......
...@@ -27,3 +27,4 @@ ...@@ -27,3 +27,4 @@
.border-style-solid { border-style: solid; } .border-style-solid { border-style: solid; }
.border-color-blue-300 { border-color: $blue-300; } .border-color-blue-300 { border-color: $blue-300; }
.border-color-default { border-color: $border-color; } .border-color-default { border-color: $border-color; }
.box-shadow-default { box-shadow: 0 2px 4px 0 $black-transparent; }
...@@ -10,7 +10,7 @@ class PersonalSnippetPolicy < BasePolicy ...@@ -10,7 +10,7 @@ class PersonalSnippetPolicy < BasePolicy
enable :create_note enable :create_note
end end
rule { is_author }.policy do rule { is_author | admin }.policy do
enable :read_personal_snippet enable :read_personal_snippet
enable :update_personal_snippet enable :update_personal_snippet
enable :destroy_personal_snippet enable :destroy_personal_snippet
......
.js-grafana-integration{ data: { operations_settings_endpoint: project_settings_operations_path(@project), .js-grafana-integration{ data: { operations_settings_endpoint: project_settings_operations_path(@project),
grafana_integration: { url: grafana_integration_url, token: grafana_integration_token } } } grafana_integration: { url: grafana_integration_url, token: grafana_integration_token, enabled: grafana_integration_enabled?.to_s } } }
...@@ -32,7 +32,7 @@ ...@@ -32,7 +32,7 @@
- explore_groups_button_label = _('Explore groups') - explore_groups_button_label = _('Explore groups')
- explore_groups_button_link = explore_groups_path - explore_groups_button_link = explore_groups_path
.js-projects-list-holder .js-projects-list-holder{ data: { qa_selector: 'projects_list' } }
- if any_projects?(projects) - if any_projects?(projects)
- load_pipeline_status(projects) if pipeline_status - load_pipeline_status(projects) if pipeline_status
%ul.projects-list{ class: css_classes } %ul.projects-list{ class: css_classes }
......
---
title: Allow admins to administer personal snippets
merge_request: 19693
author: Oren Kanner
type: fixed
---
title: Upgrade design/copy for issue weights locked feature
merge_request: 17352
author:
type: changed
---
title: Add grafana integration active status checkbox
merge_request: 19255
author:
type: added
...@@ -73,7 +73,7 @@ With the purpose of being [respectful of others' time](https://about.gitlab.com/ ...@@ -73,7 +73,7 @@ With the purpose of being [respectful of others' time](https://about.gitlab.com/
- Before assigning to a maintainer, assign to a reviewer. - Before assigning to a maintainer, assign to a reviewer.
- If you assigned a merge request, or pinged someone directly, keep in mind that we work in different timezones and asynchronously, so be patient. Unless the merge request is urgent (like fixing a broken master), please don't DM or reassign the merge request before waiting for a 24-hour window. - If you assigned a merge request, or pinged someone directly, keep in mind that we work in different timezones and asynchronously, so be patient. Unless the merge request is urgent (like fixing a broken master), please don't DM or reassign the merge request before waiting for a 24-hour window.
- If you have a question regarding your merge request/issue, make it on the merge request/issue. When we DM each other, we no longer have a SSOT and [no one else is able to contribute](https://about.gitlab.com/handbook/values/#public-by-default). - If you have a question regarding your merge request/issue, make it on the merge request/issue. When we DM each other, we no longer have a SSOT and [no one else is able to contribute](https://about.gitlab.com/handbook/values/#public-by-default).
- When you have a big WIP merge request with many changes, you're adivsed to get the review started before adding/removing significant code. Make sure it is assigned well before the release cut-off, as the reviewer(s)/maintainer(s) would always prioritize reviewing finished MRs before WIP ones. - When you have a big WIP merge request with many changes, you're advised to get the review started before adding/removing significant code. Make sure it is assigned well before the release cut-off, as the reviewer(s)/maintainer(s) would always prioritize reviewing finished MRs before WIP ones.
- Make sure to remove the WIP title before the last round of review. - Make sure to remove the WIP title before the last round of review.
### Share your work early ### Share your work early
......
...@@ -26,6 +26,23 @@ SAML SSO for GitLab.com groups does not sync users between providers without usi ...@@ -26,6 +26,23 @@ SAML SSO for GitLab.com groups does not sync users between providers without usi
![Issuer and callback for configuring SAML identity provider with GitLab.com](img/group_saml_configuration_information.png) ![Issuer and callback for configuring SAML identity provider with GitLab.com](img/group_saml_configuration_information.png)
### NameID
GitLab.com uses the SAML NameID to identify users. The NameID element:
- Is a required field in the SAML response.
- Must be unique to each user.
- Must be a persistent value that will never change, such as a randomly generated unique user ID.
- Is case sensitive. The NameID must match exactly on subsequent login attempts, so should not rely on user input that could change between upper and lower case.
- Should not be an email address or username. We strongly recommend against these as it is hard to guarantee they will never change, for example when a person's name changes. Email addresses are also case-insensitive, which can result in users being unable to sign in.
CAUTION: **Warning:**
Once users have signed into GitLab using the SSO SAML setup, changing the `NameID` will break the configuration and potentially lock users out of the GitLab group.
#### NameID Format
We recommend setting the NameID format to `Persistent` unless using a field (such as email) that requires a different format.
### SSO enforcement ### SSO enforcement
SSO enforcement was: SSO enforcement was:
...@@ -58,21 +75,12 @@ Since use of the group managed account requires the use of SSO, users of group m ...@@ -58,21 +75,12 @@ Since use of the group managed account requires the use of SSO, users of group m
- The user will be unable to access the group (their credentials will no longer work on the identity provider when prompted to SSO). - The user will be unable to access the group (their credentials will no longer work on the identity provider when prompted to SSO).
- Contributions in the group (e.g. issues, merge requests) will remain intact. - Contributions in the group (e.g. issues, merge requests) will remain intact.
### NameID #### Assertions
GitLab.com uses the SAML NameID to identify users. The NameID element:
- Is a required field in the SAML response.
- Must be unique to each user.
- Must be a persistent value that will never change, such as a randomly generated unique user ID.
- Is case sensitive. The NameID must match exactly on subsequent login attempts, so should not rely on user input that could change between upper and lower case.
We strongly recommend against using Email as the NameID as it is hard to guarantee it will never change, for example when a person's name changes. Similarly usernames should be avoided if possible.
### Assertions When using Group Manged Accounts, the following user details need to be passed to GitLab as SAML Assertions in order for us to be able to create a user:
| Field | Supported keys | | Field | Supported keys |
|-------|----------------| |-----------------|----------------|
| Email (required)| `email`, `mail` | | Email (required)| `email`, `mail` |
| Full Name | `name` | | Full Name | `name` |
| First Name | `first_name`, `firstname`, `firstName` | | First Name | `first_name`, `firstname`, `firstName` |
......
...@@ -66,8 +66,13 @@ You can then test the connection by clicking on **Test Connection**. If the conn ...@@ -66,8 +66,13 @@ You can then test the connection by clicking on **Test Connection**. If the conn
1. Click **Delete** next to the `mail` mapping. 1. Click **Delete** next to the `mail` mapping.
1. Map `userPrincipalName` to `emails[type eq "work"].value` and change it's **Matching precedence** to `2`. 1. Map `userPrincipalName` to `emails[type eq "work"].value` and change it's **Matching precedence** to `2`.
1. Map `mailNickname` to `userName`. 1. Map `mailNickname` to `userName`.
1. Create a new mapping by clicking **Add New Mapping** then set **Source attribute** to `objectId`, **Target attribute** to `id`, **Match objects using this attribute** to `Yes`, and **Matching precedence** to `1`. 1. Determine how GitLab will uniquely identify users.
1. Create a new mapping by clicking **Add New Mapping** then set **Source attribute** to `objectId`, and **Target attribute** to `externalId`.
- Use `objectId` unless users already have SAML linked for your group.
- If you already have users with SAML linked then use the `Name ID` value from the [SAML configuration](#azure). Using a different value will likely cause duplicate users and prevent users from accessing the GitLab group.
1. Create a new mapping by clicking **Add New Mapping** then set **Source attribute** to the unique identifier determined above, **Target attribute** to `id`, **Match objects using this attribute** to `Yes`, and **Matching precedence** to `1`.
1. Create a new mapping by clicking **Add New Mapping** then set **Source attribute** to the unique identifier determined above, and **Target attribute** to `externalId`.
1. Click the `userPrincipalName` mapping and change **Match objects using this attribute** to `No`. 1. Click the `userPrincipalName` mapping and change **Match objects using this attribute** to `No`.
Save your changes and you should have the following configuration: Save your changes and you should have the following configuration:
...@@ -99,6 +104,9 @@ You can then test the connection by clicking on **Test Connection**. If the conn ...@@ -99,6 +104,9 @@ You can then test the connection by clicking on **Test Connection**. If the conn
Once enabled, the synchronization details and any errors will appear on the Once enabled, the synchronization details and any errors will appear on the
bottom of the **Provisioning** screen, together with a link to the audit logs. bottom of the **Provisioning** screen, together with a link to the audit logs.
CAUTION: **Warning:**
Once synchronized, changing the field mapped to `id` and `externalId` will likely cause provisioning errors, duplicate users, and prevent existing users from accessing the GitLab group.
## Troubleshooting ## Troubleshooting
### Testing Azure connection: invalid credentials ### Testing Azure connection: invalid credentials
......
...@@ -33,6 +33,10 @@ to be enabled: ...@@ -33,6 +33,10 @@ to be enabled:
project level, navigate to your project's **Settings > General**, expand **Visibility, project features, permissions** project level, navigate to your project's **Settings > General**, expand **Visibility, project features, permissions**
and enable **Git Large File Storage**. and enable **Git Large File Storage**.
Design Management requires that projects are using
[hashed storage](../../../administration/repository_storage_types.html#hashed-storage)
(the default storage type since v10.0).
## Limitations ## Limitations
- Files uploaded must have a file extension of either `png`, `jpg`, `jpeg`, `gif`, `bmp`, `tiff` or `ico`. - Files uploaded must have a file extension of either `png`, `jpg`, `jpeg`, `gif`, `bmp`, `tiff` or `ico`.
......
...@@ -2917,9 +2917,6 @@ msgstr "" ...@@ -2917,9 +2917,6 @@ msgstr ""
msgid "Certificate (PEM)" msgid "Certificate (PEM)"
msgstr "" msgstr ""
msgid "Change Weight"
msgstr ""
msgid "Change assignee" msgid "Change assignee"
msgstr "" msgstr ""
...@@ -8307,6 +8304,9 @@ msgstr "" ...@@ -8307,6 +8304,9 @@ msgstr ""
msgid "GrafanaIntegration|API Token" msgid "GrafanaIntegration|API Token"
msgstr "" msgstr ""
msgid "GrafanaIntegration|Active"
msgstr ""
msgid "GrafanaIntegration|Embed Grafana charts in GitLab issues." msgid "GrafanaIntegration|Embed Grafana charts in GitLab issues."
msgstr "" msgstr ""
...@@ -13483,12 +13483,24 @@ msgstr "" ...@@ -13483,12 +13483,24 @@ msgstr ""
msgid "Promotions|Epics let you manage your portfolio of projects more efficiently and with less effort by tracking groups of issues that share a theme, across projects and milestones." msgid "Promotions|Epics let you manage your portfolio of projects more efficiently and with less effort by tracking groups of issues that share a theme, across projects and milestones."
msgstr "" msgstr ""
msgid "Promotions|Learn more"
msgstr ""
msgid "Promotions|See the other features in the %{subscription_link_start}bronze plan%{subscriptions_link_end}"
msgstr ""
msgid "Promotions|This feature is locked." msgid "Promotions|This feature is locked."
msgstr "" msgstr ""
msgid "Promotions|Upgrade plan" msgid "Promotions|Upgrade plan"
msgstr "" msgstr ""
msgid "Promotions|Weighting your issue"
msgstr ""
msgid "Promotions|When you have a lot of issues, it can be hard to get an overview. By adding a weight to your issues, you can get a better idea of the effort, cost, required time, or value of each, and so better manage them."
msgstr ""
msgid "Prompt users to upload SSH keys" msgid "Prompt users to upload SSH keys"
msgstr "" msgstr ""
...@@ -18227,9 +18239,6 @@ msgstr "" ...@@ -18227,9 +18239,6 @@ msgstr ""
msgid "Upgrade your plan to activate Group Webhooks." msgid "Upgrade your plan to activate Group Webhooks."
msgstr "" msgstr ""
msgid "Upgrade your plan to activate Issue weight."
msgstr ""
msgid "Upgrade your plan to improve Issue boards." msgid "Upgrade your plan to improve Issue boards."
msgstr "" msgstr ""
......
...@@ -18,6 +18,10 @@ module QA ...@@ -18,6 +18,10 @@ module QA
'/' '/'
end end
def clear_project_filter
fill_element(:project_filter_form, "")
end
private private
def filter_by_name(name) def filter_by_name(name)
......
...@@ -35,3 +35,5 @@ module QA ...@@ -35,3 +35,5 @@ module QA
end end
end end
end end
QA::Page::Project::Settings::CICD.prepend_if_ee('QA::EE::Page::Project::Settings::CICD')
...@@ -35,6 +35,15 @@ exports[`grafana integration component default state to match the default snapsh ...@@ -35,6 +35,15 @@ exports[`grafana integration component default state to match the default snapsh
class="settings-content" class="settings-content"
> >
<form> <form>
<glformcheckbox-stub
class="mb-4"
id="grafana-integration-enabled"
>
Active
</glformcheckbox-stub>
<glformgroup-stub <glformgroup-stub
description="Enter the base URL of the Grafana instance." description="Enter the base URL of the Grafana instance."
label="Grafana URL" label="Grafana URL"
......
...@@ -86,6 +86,7 @@ describe('grafana integration component', () => { ...@@ -86,6 +86,7 @@ describe('grafana integration component', () => {
grafana_integration_attributes: { grafana_integration_attributes: {
grafana_url: grafanaIntegrationUrl, grafana_url: grafanaIntegrationUrl,
token: grafanaIntegrationToken, token: grafanaIntegrationToken,
enabled: false,
}, },
}, },
}, },
......
...@@ -25,4 +25,11 @@ describe('grafana integration mutations', () => { ...@@ -25,4 +25,11 @@ describe('grafana integration mutations', () => {
expect(localState.grafanaToken).toBe(mockToken); expect(localState.grafanaToken).toBe(mockToken);
}); });
}); });
describe('SET_GRAFANA_ENABLED', () => {
it('updates grafanaEnabled for integration', () => {
mutations.SET_GRAFANA_ENABLED(localState, true);
expect(localState.grafanaEnabled).toBe(true);
});
});
}); });
...@@ -20,6 +20,19 @@ describe PersonalSnippetPolicy do ...@@ -20,6 +20,19 @@ describe PersonalSnippetPolicy do
described_class.new(user, snippet) described_class.new(user, snippet)
end end
shared_examples 'admin access' do
context 'admin user' do
subject { permissions(admin_user) }
it do
is_expected.to be_allowed(:read_personal_snippet)
is_expected.to be_allowed(:create_note)
is_expected.to be_allowed(:award_emoji)
is_expected.to be_allowed(*author_permissions)
end
end
end
context 'public snippet' do context 'public snippet' do
let(:snippet) { create(:personal_snippet, :public) } let(:snippet) { create(:personal_snippet, :public) }
...@@ -55,6 +68,8 @@ describe PersonalSnippetPolicy do ...@@ -55,6 +68,8 @@ describe PersonalSnippetPolicy do
is_expected.to be_allowed(*author_permissions) is_expected.to be_allowed(*author_permissions)
end end
end end
it_behaves_like 'admin access'
end end
context 'internal snippet' do context 'internal snippet' do
...@@ -103,6 +118,8 @@ describe PersonalSnippetPolicy do ...@@ -103,6 +118,8 @@ describe PersonalSnippetPolicy do
is_expected.to be_allowed(*author_permissions) is_expected.to be_allowed(*author_permissions)
end end
end end
it_behaves_like 'admin access'
end end
context 'private snippet' do context 'private snippet' do
...@@ -130,17 +147,6 @@ describe PersonalSnippetPolicy do ...@@ -130,17 +147,6 @@ describe PersonalSnippetPolicy do
end end
end end
context 'admin user' do
subject { permissions(admin_user) }
it do
is_expected.to be_allowed(:read_personal_snippet)
is_expected.to be_disallowed(:create_note)
is_expected.to be_disallowed(:award_emoji)
is_expected.to be_disallowed(*author_permissions)
end
end
context 'external user' do context 'external user' do
subject { permissions(external_user) } subject { permissions(external_user) }
...@@ -162,5 +168,7 @@ describe PersonalSnippetPolicy do ...@@ -162,5 +168,7 @@ describe PersonalSnippetPolicy do
is_expected.to be_allowed(*author_permissions) is_expected.to be_allowed(*author_permissions)
end end
end end
it_behaves_like 'admin access'
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