Commit bca413ab authored by Paul Slaughter's avatar Paul Slaughter

Merge branch...

Merge branch '12155-update-pipelines-minutes-expiry-banner-to-an-alert-component-type' into 'master'

Resolve "Update Pipelines Minutes expiry banner to Alert Component"

Closes #12156 and #12155

See merge request gitlab-org/gitlab-ee!14786
parents 38cd0daf f651043d
...@@ -83,6 +83,11 @@ export default { ...@@ -83,6 +83,11 @@ export default {
type: String, type: String,
required: true, required: true,
}, },
subscriptionsMoreMinutesUrl: {
type: String,
required: false,
default: null,
},
}, },
computed: { computed: {
...mapState([ ...mapState([
...@@ -265,6 +270,7 @@ export default { ...@@ -265,6 +270,7 @@ export default {
:quota-limit="job.runners.quota.limit" :quota-limit="job.runners.quota.limit"
:runners-path="runnerHelpUrl" :runners-path="runnerHelpUrl"
:project-path="projectPath" :project-path="projectPath"
:subscriptions-more-minutes-url="subscriptionsMoreMinutesUrl"
/> />
<environments-block <environments-block
......
...@@ -15,6 +15,7 @@ export default () => { ...@@ -15,6 +15,7 @@ export default () => {
runnerHelpUrl, runnerHelpUrl,
runnerSettingsUrl, runnerSettingsUrl,
variablesSettingsUrl, variablesSettingsUrl,
subscriptionsMoreMinutesUrl,
endpoint, endpoint,
pagePath, pagePath,
logState, logState,
...@@ -28,6 +29,7 @@ export default () => { ...@@ -28,6 +29,7 @@ export default () => {
runnerHelpUrl, runnerHelpUrl,
runnerSettingsUrl, runnerSettingsUrl,
variablesSettingsUrl, variablesSettingsUrl,
subscriptionsMoreMinutesUrl,
endpoint, endpoint,
pagePath, pagePath,
logState, logState,
......
...@@ -73,13 +73,6 @@ export default class Project { ...@@ -73,13 +73,6 @@ export default class Project {
.remove(); .remove();
return e.preventDefault(); return e.preventDefault();
}); });
$('.hide-shared-runner-limit-message').on('click', function(e) {
var $alert = $(this).parents('.shared-runner-quota-message');
var scope = $alert.data('scope');
Cookies.set('hide_shared_runner_quota_message', 'false', { path: scope });
$alert.remove();
e.preventDefault();
});
$('.hide-auto-devops-implicitly-enabled-banner').on('click', function(e) { $('.hide-auto-devops-implicitly-enabled-banner').on('click', function(e) {
const projectId = $(this).data('project-id'); const projectId = $(this).data('project-id');
const cookieKey = `hide_auto_devops_implicitly_enabled_banner_${projectId}`; const cookieKey = `hide_auto_devops_implicitly_enabled_banner_${projectId}`;
......
# frozen_string_literal: true
module JobsHelper
def jobs_data
{
"endpoint" => project_job_path(@project, @build, format: :json),
"project_path" => @project.full_path,
"deployment_help_url" => help_page_path('user/project/clusters/index.html', anchor: 'troubleshooting-failed-deployment-jobs'),
"runner_help_url" => help_page_path('ci/runners/README.html', anchor: 'setting-maximum-job-timeout-for-a-runner'),
"runner_settings_url" => project_runners_path(@build.project, anchor: 'js-runners-settings'),
"variables_settings_url" => project_variables_path(@build.project, anchor: 'js-cicd-variables-settings'),
"page_path" => project_job_path(@project, @build),
"build_status" => @build.status,
"build_stage" => @build.stage,
"log_state" => '',
"build_options" => javascript_build_options
}
end
end
...@@ -5,10 +5,4 @@ ...@@ -5,10 +5,4 @@
- content_for :page_specific_javascripts do - content_for :page_specific_javascripts do
= stylesheet_link_tag 'page_bundles/xterm' = stylesheet_link_tag 'page_bundles/xterm'
#js-job-vue-app{ data: { endpoint: project_job_path(@project, @build, format: :json), project_path: @project.full_path, #js-job-vue-app{ data: jobs_data }
deployment_help_url: help_page_path('user/project/clusters/index.html', anchor: 'troubleshooting-failed-deployment-jobs'),
runner_help_url: help_page_path('ci/runners/README.html', anchor: 'setting-maximum-job-timeout-for-a-runner'),
runner_settings_url: project_runners_path(@build.project, anchor: 'js-runners-settings'),
variables_settings_url: project_variables_path(@build.project, anchor: 'js-cicd-variables-settings'),
page_path: project_job_path(@project, @build), build_status: @build.status, build_stage: @build.stage, log_state: '',
build_options: javascript_build_options } }
<script> <script>
/* eslint-disable @gitlab/vue-i18n/no-bare-strings */ import { GlButton } from '@gitlab/ui';
import { s__, sprintf } from '~/locale';
export default { export default {
components: {
GlButton,
},
props: { props: {
quotaUsed: { quotaUsed: {
type: Number, type: Number,
...@@ -15,19 +20,48 @@ export default { ...@@ -15,19 +20,48 @@ export default {
required: false, required: false,
default: null, default: null,
}, },
projectPath: {
type: String,
required: true,
},
subscriptionsMoreMinutesUrl: {
type: String,
required: false,
default: null,
},
},
computed: {
isExpired() {
return this.artifact.expired;
},
runnersWarningMessage() {
return sprintf(
s__(
'Runners|You have used %{quotaUsed} out of %{quotaLimit} of your shared Runners pipeline minutes.',
),
{ quotaUsed: this.quotaUsed, quotaLimit: this.quotaLimit },
);
},
}, },
}; };
</script> </script>
<template> <template>
<div class="bs-callout bs-callout-warning"> <div class="bs-callout bs-callout-danger">
<p> <p>
{{ s__('Runners|You have used all your shared Runners pipeline minutes.') }} {{ runnersWarningMessage }}
{{ quotaUsed }} of {{ quotaLimit }}
<template v-if="runnersPath"> <template v-if="runnersPath">
{{ __('For more information, go to the ') }} {{ __('For more information, go to the ') }}
<a :href="runnersPath"> {{ __('Runners page.') }} </a> <a :href="runnersPath">{{ __('Runners page.') }}</a>
</template> </template>
</p> </p>
<gl-button
v-if="subscriptionsMoreMinutesUrl"
variant="danger"
:href="subscriptionsMoreMinutesUrl"
class="btn-inverted"
>
{{ __('Purchase more minutes') }}
</gl-button>
</div> </div>
</template> </template>
# frozen_string_literal: true
module EE
module JobsHelper
extend ::Gitlab::Utils::Override
override :jobs_data
def jobs_data
super.merge({
"subscriptions_more_minutes_url" => ::EE::SUBSCRIPTIONS_MORE_MINUTES_URL
})
end
end
end
::JobsHelper.prepend_if_ee('::EE::JobsHelper')
...@@ -9,20 +9,12 @@ module EE ...@@ -9,20 +9,12 @@ module EE
if ::Gitlab.com? && can?(current_user, :admin_project, project) if ::Gitlab.com? && can?(current_user, :admin_project, project)
message += " #{purchase_shared_runner_minutes_link}" message += " #{purchase_shared_runner_minutes_link}"
elsif namespace.shared_runners_minutes_used? elsif namespace.shared_runners_minutes_used?
message += s_('Pipelines|Pipelines will not run anymore on shared Runners.') message += " #{s_('Pipelines|Pipelines will not run anymore on shared Runners.')}"
end end
message.html_safe message.html_safe
end end
def ci_usage_warning_class(namespace)
if EE::Namespace::CI_USAGE_ALERT_LEVELS.min == namespace.last_ci_minutes_usage_notification_level
'alert-danger'
else
'alert-warning'
end
end
private private
def purchase_shared_runner_minutes_link def purchase_shared_runner_minutes_link
......
...@@ -5,9 +5,9 @@ ...@@ -5,9 +5,9 @@
- can_see_status = project.nil? || can?(current_user, :create_pipeline, project) - can_see_status = project.nil? || can?(current_user, :create_pipeline, project)
- ci_warning_message = ci_usage_warning_message(namespace, project) - ci_warning_message = ci_usage_warning_message(namespace, project)
- if cookies[:hide_shared_runner_quota_message].blank? && has_limit && can_see_status && ci_warning_message.present? - if has_limit && can_see_status && ci_warning_message.present?
.shared-runner-quota-message.alert.d-none.d-sm-block{ class: ci_usage_warning_class(namespace), data: { scope: scope } } %div{ class: ["pt-2", (classes if defined? classes)] }
= ci_warning_message .bs-callout.shared-runner-quota-message.d-none.d-sm-block.bs-callout-danger{ data: { scope: scope } }
%p
.float-right = ci_warning_message
= link_to 'Remind later', '#', class: 'hide-shared-runner-limit-message alert-link' = link_to _('Purchase more minutes'), ::EE::SUBSCRIPTIONS_MORE_MINUTES_URL, class: "btn btn-danger btn-inverted"
= content_for :flash_message do = content_for :flash_message do
= render 'shared/shared_runners_minutes_limit', project: @project %div{ class: container_class }
= render 'shared/shared_runners_minutes_limit', project: @project
---
title: Update Pipelines Minutes expiry banner to Alert Component
merge_request: 14786
author:
type: other
...@@ -3,4 +3,5 @@ ...@@ -3,4 +3,5 @@
module EE module EE
SUBSCRIPTIONS_URL = 'https://customers.gitlab.com'.freeze SUBSCRIPTIONS_URL = 'https://customers.gitlab.com'.freeze
SUBSCRIPTIONS_PLANS_URL = "#{SUBSCRIPTIONS_URL}/plans".freeze SUBSCRIPTIONS_PLANS_URL = "#{SUBSCRIPTIONS_URL}/plans".freeze
SUBSCRIPTIONS_MORE_MINUTES_URL = "#{SUBSCRIPTIONS_URL}/buy_pipeline_minutes".freeze
end end
...@@ -22,7 +22,7 @@ describe 'Jobs', :clean_gitlab_redis_shared_state do ...@@ -22,7 +22,7 @@ describe 'Jobs', :clean_gitlab_redis_shared_state do
visit project_job_path(project, job) visit project_job_path(project, job)
wait_for_requests wait_for_requests
expect(page).to have_content('You have used all your shared Runners pipeline minutes.') expect(page).to have_content('You have used 1000 out of 500 of your shared Runners pipeline minutes.')
end end
end end
...@@ -34,7 +34,7 @@ describe 'Jobs', :clean_gitlab_redis_shared_state do ...@@ -34,7 +34,7 @@ describe 'Jobs', :clean_gitlab_redis_shared_state do
visit project_job_path(project, job) visit project_job_path(project, job)
wait_for_requests wait_for_requests
expect(page).not_to have_content('You have used all your shared Runners pipeline minutes.') expect(page).not_to have_content('You have used 1000 out of 500 of your shared Runners pipeline minutes.')
end end
end end
end end
......
import Vue from 'vue';
import component from 'ee/jobs/components/shared_runner_limit_block.vue'; import component from 'ee/jobs/components/shared_runner_limit_block.vue';
import mountComponent from 'spec/helpers/vue_mount_component_helper'; import { createLocalVue, shallowMount } from '@vue/test-utils';
import { GlButton } from '@gitlab/ui';
import { trimText } from 'spec/helpers/text_helper'; import { trimText } from 'spec/helpers/text_helper';
const localVue = createLocalVue();
describe('Shared Runner Limit Block', () => { describe('Shared Runner Limit Block', () => {
const Component = Vue.extend(component); let wrapper;
let vm;
const Component = localVue.extend(component);
const runnersPath = 'root/project/runners';
const projectPath = 'h5bp/html5-boilerplate';
const subscriptionsMoreMinutesUrl = 'https://customers.gitlab.com/buy_pipeline_minutes';
const factory = (options = {}) => {
wrapper = shallowMount(Component, {
localVue,
sync: false,
...options,
});
};
afterEach(() => { afterEach(() => {
vm.$destroy(); wrapper.destroy();
wrapper = null;
}); });
describe('quota information', () => { describe('quota information', () => {
it('renders provided quota limit and used quota', () => { beforeEach(() => {
vm = mountComponent(Component, { factory({
quotaUsed: 1000, propsData: {
quotaLimit: 4000, quotaUsed: 1000,
runnersPath: 'root/project/runners', quotaLimit: 4000,
runnersPath,
projectPath,
subscriptionsMoreMinutesUrl,
},
}); });
});
expect(vm.$el.textContent).toContain( it('renders provided quota limit and used quota', () => {
'You have used all your shared Runners pipeline minutes.', expect(wrapper.text()).toContain(
'You have used 1000 out of 4000 of your shared Runners pipeline minutes',
); );
});
it('renders call to action gl-button with the right href', () => {
const glButton = wrapper.find(GlButton);
expect(vm.$el.textContent).toContain('1000 of 4000'); expect(glButton.isVisible()).toBe(true);
expect(glButton.attributes('variant')).toBe('danger');
expect(glButton.attributes('href')).toBe(subscriptionsMoreMinutesUrl);
}); });
}); });
describe('with runnersPath', () => { describe('with runnersPath', () => {
it('renders runner link', () => { it('renders runner link', () => {
vm = mountComponent(Component, { factory({
quotaUsed: 1000, propsData: {
quotaLimit: 4000, quotaUsed: 1000,
runnersPath: 'root/project/runners', quotaLimit: 4000,
projectPath,
runnersPath,
subscriptionsMoreMinutesUrl,
},
}); });
expect(trimText(vm.$el.textContent)).toContain( expect(trimText(wrapper.text())).toContain('For more information, go to the Runners page.');
'For more information, go to the Runners page.',
);
}); });
}); });
describe('without runnersPath', () => { describe('without runnersPath', () => {
it('does not renbder runner link', () => { it('does not render runner link', () => {
vm = mountComponent(Component, { factory({
quotaUsed: 1000, propsData: {
quotaLimit: 4000, quotaUsed: 1000,
quotaLimit: 4000,
projectPath,
subscriptionsMoreMinutesUrl,
},
}); });
expect(trimText(vm.$el.textContent)).not.toContain( expect(trimText(wrapper.element.textContent)).not.toContain(
'For more information, go to the Runners page.', 'For more information, go to the Runners page.',
); );
}); });
......
...@@ -10918,9 +10918,6 @@ msgstr "" ...@@ -10918,9 +10918,6 @@ msgstr ""
msgid "Pipelines|Loading Pipelines" msgid "Pipelines|Loading Pipelines"
msgstr "" msgstr ""
msgid "Pipelines|Pipelines will not run anymore on shared Runners."
msgstr ""
msgid "Pipelines|Project cache successfully reset." msgid "Pipelines|Project cache successfully reset."
msgstr "" msgstr ""
...@@ -12307,6 +12304,9 @@ msgstr "" ...@@ -12307,6 +12304,9 @@ msgstr ""
msgid "Pull" msgid "Pull"
msgstr "" msgstr ""
msgid "Purchase more minutes"
msgstr ""
msgid "Push" msgid "Push"
msgstr "" msgstr ""
...@@ -13098,7 +13098,7 @@ msgstr "" ...@@ -13098,7 +13098,7 @@ msgstr ""
msgid "Runners page." msgid "Runners page."
msgstr "" msgstr ""
msgid "Runners|You have used all your shared Runners pipeline minutes." msgid "Runners|You have used %{quotaUsed} out of %{quotaLimit} of your shared Runners pipeline minutes."
msgstr "" msgstr ""
msgid "Running" msgid "Running"
......
...@@ -25,6 +25,7 @@ describe('Job App ', () => { ...@@ -25,6 +25,7 @@ describe('Job App ', () => {
terminalPath: 'jobs/123/terminal', terminalPath: 'jobs/123/terminal',
pagePath: `${gl.TEST_HOST}jobs/123`, pagePath: `${gl.TEST_HOST}jobs/123`,
projectPath: 'user-name/project-name', projectPath: 'user-name/project-name',
subscriptionsMoreMinutesUrl: 'https://customers.gitlab.com/buy_pipeline_minutes',
logState: logState:
'eyJvZmZzZXQiOjE3NDUxLCJuX29wZW5fdGFncyI6MCwiZmdfY29sb3IiOm51bGwsImJnX2NvbG9yIjpudWxsLCJzdHlsZV9tYXNrIjowfQ%3D%3D', 'eyJvZmZzZXQiOjE3NDUxLCJuX29wZW5fdGFncyI6MCwiZmdfY29sb3IiOm51bGwsImJnX2NvbG9yIjpudWxsLCJzdHlsZV9tYXNrIjowfQ%3D%3D',
}; };
......
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