Commit baa820af authored by Oswaldo Ferreira's avatar Oswaldo Ferreira Committed by Douglas Barbosa Alexandre

Add an Upgrade button to User's billings page

It adds an upgrade button to the user billings
page that redirects to the customers portal
upgrade checkout page.

If the user is not logged in at the portal he/she
will be automatically redirected after the login
process.

If the user is at the latest tier, the button
won't be presented and it'll fallback to the
existing plans table.
parent daa7e56f
.billing-plan-header { .billing-plan-header {
border-bottom: 0; border-bottom: 0;
padding-bottom: 0;
.billing-plan-logo svg { .billing-plan-logo svg {
height: 100px; height: 100px;
...@@ -18,10 +19,6 @@ ...@@ -18,10 +19,6 @@
} }
} }
.billing-plans-alert {
margin: 0;
}
.billing-plans { .billing-plans {
display: flex; display: flex;
flex-direction: row; flex-direction: row;
......
...@@ -13,10 +13,6 @@ module BillingPlansHelper ...@@ -13,10 +13,6 @@ module BillingPlansHelper
plan.purchase_link&.action == 'current_plan' plan.purchase_link&.action == 'current_plan'
end end
def has_plan_purchase_link?(plans_data)
plans_data.any? { |plan| plan.purchase_link&.href }
end
def plan_purchase_link(href, link_text) def plan_purchase_link(href, link_text)
if href if href
link_to link_text, href, class: 'btn btn-primary btn-inverted' link_to link_text, href, class: 'btn btn-primary btn-inverted'
......
...@@ -69,7 +69,7 @@ module EE ...@@ -69,7 +69,7 @@ module EE
validate :validate_plan_name validate :validate_plan_name
validate :validate_shared_runner_minutes_support validate :validate_shared_runner_minutes_support
delegate :trial?, :trial_ends_on, to: :gitlab_subscription, allow_nil: true delegate :trial?, :trial_ends_on, :upgradable?, to: :gitlab_subscription, allow_nil: true
before_create :sync_membership_lock_with_parent before_create :sync_membership_lock_with_parent
......
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
%h4 %h4
- plan_link = plan.about_page_href ? link_to(plan.code.titleize, plan.about_page_href) : plan.name - plan_link = plan.about_page_href ? link_to(plan.code.titleize, plan.about_page_href) : plan.name
- if namespace == current_user.namespace - if namespace == current_user.namespace
= s_("BillingPlans|@%{user_name} you are currently on the %{plan_link} plan.").html_safe % { user_name: current_user.username, plan_link: plan_link } = s_("BillingPlans|You are currently using the %{plan_link} plan.").html_safe % { plan_link: plan_link }
- else - else
= s_("BillingPlans|%{group_name} is currently using the %{plan_link} plan.").html_safe % { group_name: namespace.full_name, plan_link: plan_link } = s_("BillingPlans|%{group_name} is currently using the %{plan_link} plan.").html_safe % { group_name: namespace.full_name, plan_link: plan_link }
......
- current_plan = subscription_plan_info(plans_data, namespace.actual_plan_name) - current_plan = subscription_plan_info(plans_data, namespace.actual_plan_name)
- support_link = link_to s_("BillingPlans|Contact Support"), "https://support.gitlab.com"
- if current_plan - if current_plan
= render 'shared/billings/billing_plan_header', namespace: namespace, plan: current_plan = render 'shared/billings/billing_plan_header', namespace: namespace, plan: current_plan
- unless has_plan_purchase_link?(plans_data) - if namespace.upgradable?
.billing-plans-alert.card.prepend-top-10 .gl-p-4.center
.card-header.bg-warning.text-white = link_to s_("BillingPlan|Upgrade plan"), plan_upgrade_url(namespace, current_plan), class: "btn btn-success"
= s_("BillingPlans|Automatic downgrade and upgrade to some plans is currently not available.")
- customer_support_url = 'https://about.gitlab.com/sales/';
- customer_support_link = link_to s_("BillingPlans|Customer Support"), customer_support_url
= s_("BillingPlans|Please contact %{customer_support_link} in that case.").html_safe % { customer_support_link: customer_support_link }
.billing-plans - else
.billing-plans
- plans_data.each do |plan| - plans_data.each do |plan|
= render 'shared/billings/billing_plan', namespace: namespace, plan: plan = render 'shared/billings/billing_plan', namespace: namespace, plan: plan
- if namespace.actual_plan
.center
= s_("BillingPlans|If you would like to downgrade your plan please %{support_link}.").html_safe % { support_link: support_link }
---
title: Add Upgrade button to the User Billing page
merge_request: 15075
author:
type: added
require 'spec_helper' require 'spec_helper'
describe 'Billing plan pages', :feature do describe 'Billing plan pages', :feature do
include StubRequests
let(:user) { create(:user) } let(:user) { create(:user) }
let(:namespace) { user.namespace }
let(:free_plan) { create(:free_plan) }
let(:bronze_plan) { create(:bronze_plan) } let(:bronze_plan) { create(:bronze_plan) }
let(:gold_plan) { create(:gold_plan) }
let(:plans_data) do let(:plans_data) do
[ JSON.parse(File.read(Rails.root.join('ee/spec/fixtures/gitlab_com_plans.json'))).map do |data|
{ data.deep_symbolize_keys
name: "Free", end
price_per_month: 0, end
free: true,
code: "free", before do
price_per_year: 0, stub_full_request("https://customers.gitlab.com/gitlab_plans?plan=#{plan}")
purchase_link: { .to_return(status: 200, body: plans_data.to_json)
action: "downgrade", stub_application_setting(check_namespace_plan: true)
href: nil allow(Gitlab).to receive(:com?) { true }
}, gitlab_sign_in(user)
features: [] end
},
{ def external_upgrade_url(namespace, plan)
name: "Bronze", if Plan::PAID_HOSTED_PLANS.include?(plan.name)
price_per_month: 4, "#{EE::SUBSCRIPTIONS_URL}/gitlab/namespaces/#{namespace.id}/upgrade/#{plan.name}-external-id"
free: false, end
code: "bronze", end
price_per_year: 48,
purchase_link: { context 'users profile billing page' do
action: "current_plan", let(:page_path) { profile_billings_path }
href: nil
}, it_behaves_like 'billings gold trial callout'
features: []
}, context 'on free' do
{ let(:plan) { free_plan.name }
name: "Silver",
price_per_month: 19, let!(:subscription) do
free: false, create(:gitlab_subscription, namespace: namespace, hosted_plan: nil, seats: 15)
code: "silver", end
price_per_year: 228,
purchase_link: { before do
action: "upgrade", visit page_path
href: nil
},
features: []
},
{
name: "Gold",
price_per_month: 99,
free: false,
code: "gold",
price_per_year: 1188,
purchase_link: {
action: "upgrade",
href: nil
},
features: []
}
]
end end
shared_examples 'displays all plans and correct actions' do
it 'displays all plans' do it 'displays all plans' do
page.within('.billing-plans') do page.within('.billing-plans') do
panels = page.all('.card') panels = page.all('.card')
...@@ -86,71 +73,93 @@ describe 'Billing plan pages', :feature do ...@@ -86,71 +73,93 @@ describe 'Billing plan pages', :feature do
expect(action).to have_css('.disabled') expect(action).to have_css('.disabled')
when 'upgrade' when 'upgrade'
expect(action).to have_content('Upgrade') expect(action).to have_content('Upgrade')
expect(action).to have_css('.disabled') expect(action).not_to have_css('.disabled')
end end
end end
end end
end end
context 'on bronze' do
let(:plan) { bronze_plan.name }
let!(:subscription) do
create(:gitlab_subscription, namespace: namespace, hosted_plan: bronze_plan, seats: 15)
end
before do before do
expect(Gitlab::HTTP).to receive(:get).and_return(double(body: plans_data.to_json)) visit page_path
stub_application_setting(check_namespace_plan: true)
allow(Gitlab).to receive(:com?) { true }
gitlab_sign_in(user)
end end
context 'users profile billing page' do it 'displays header and actions' do
let(:page_path) { profile_billings_path } page.within('.billing-plan-header') do
expect(page).to have_content('You are currently using the Bronze plan.')
it_behaves_like 'billings gold trial callout' expect(page).to have_css('.billing-plan-logo img')
end
context 'on bronze' do page.within('.content') do
before do expect(page).to have_link('Upgrade plan', href: external_upgrade_url(namespace, bronze_plan))
allow_any_instance_of(EE::Namespace).to receive(:plan).and_return(bronze_plan) expect(page).to have_content('downgrade your plan')
expect(page).to have_link('Contact Support', href: 'https://support.gitlab.com')
end
end
end
visit page_path context 'on gold' do
let(:plan) { gold_plan.name }
let!(:subscription) do
create(:gitlab_subscription, namespace: namespace, hosted_plan: gold_plan, seats: 15)
end end
include_examples 'displays all plans and correct actions' before do
visit page_path
end
it 'displays plan header' do it 'displays header and actions' do
page.within('.billing-plan-header') do page.within('.billing-plan-header') do
expect(page).to have_content("@#{user.username} you are currently on the Bronze") expect(page).to have_content('You are currently using the Gold plan.')
expect(page).to have_css('.billing-plan-logo img') expect(page).to have_css('.billing-plan-logo img')
end end
page.within('.content') do
expect(page).not_to have_link('Upgrade plan')
expect(page).to have_content('downgrade your plan')
expect(page).to have_link('Contact Support', href: 'https://support.gitlab.com')
end
end end
end end
end end
context 'group billing page' do context 'group billing page' do
let(:group) { create(:group) } let(:namespace) { create(:group) }
let!(:group_member) { create(:group_member, :owner, group: group, user: user) } let!(:group_member) { create(:group_member, :owner, group: namespace, user: user) }
context 'top-most group' do context 'top-most group' do
let(:page_path) { group_billings_path(group) } let(:page_path) { group_billings_path(namespace) }
it_behaves_like 'billings gold trial callout' it_behaves_like 'billings gold trial callout'
context 'on bronze' do context 'on bronze' do
before do let(:plan) { bronze_plan.name }
expect_any_instance_of(EE::Group).to receive(:plan).at_least(:once).and_return(bronze_plan)
let!(:subscription) do
create(:gitlab_subscription, namespace: namespace, hosted_plan: bronze_plan, seats: 15)
end
before do
visit page_path visit page_path
end end
it 'displays plan header' do it 'displays plan header' do
page.within('.billing-plan-header') do page.within('.billing-plan-header') do
expect(page).to have_content("#{group.name} is currently using the Bronze plan") expect(page).to have_content("#{namespace.name} is currently using the Bronze plan")
expect(page).to have_css('.billing-plan-logo .identicon') expect(page).to have_css('.billing-plan-logo .identicon')
end end
end end
it 'does not display the billing plans table' do
expect(page).not_to have_css('.billing-plans')
end
it 'displays subscription table', :js do it 'displays subscription table', :js do
expect(page).to have_selector('.js-subscription-table') expect(page).to have_selector('.js-subscription-table')
end end
...@@ -168,10 +177,17 @@ describe 'Billing plan pages', :feature do ...@@ -168,10 +177,17 @@ describe 'Billing plan pages', :feature do
let(:subgroup2) { create(:group, parent: subgroup1) } let(:subgroup2) { create(:group, parent: subgroup1) }
let!(:subgroup2_member) { create(:group_member, :owner, group: subgroup2, user: user3) } let!(:subgroup2_member) { create(:group_member, :owner, group: subgroup2, user: user3) }
let(:page_path) { group_billings_path(subgroup2) } let(:page_path) { group_billings_path(subgroup2) }
let(:namespace) { group }
it_behaves_like 'billings gold trial callout' it_behaves_like 'billings gold trial callout'
context 'on bronze' do context 'on bronze' do
let(:plan) { bronze_plan.name }
let!(:subscription) do
create(:gitlab_subscription, namespace: namespace, hosted_plan: bronze_plan, seats: 15)
end
before do before do
visit page_path visit page_path
end end
...@@ -182,13 +198,13 @@ describe 'Billing plan pages', :feature do ...@@ -182,13 +198,13 @@ describe 'Billing plan pages', :feature do
expect(page).to have_css('.billing-plan-logo .identicon') expect(page).to have_css('.billing-plan-logo .identicon')
expect(page.find('.btn-success')).to have_content('Manage plan') expect(page.find('.btn-success')).to have_content('Manage plan')
end end
expect(page).not_to have_css('.billing-plans')
end end
end end
end end
context 'with unexpected JSON' do context 'with unexpected JSON' do
let(:plan) { 'free' }
let(:plans_data) do let(:plans_data) do
[ [
{ {
...@@ -214,15 +230,5 @@ describe 'Billing plan pages', :feature do ...@@ -214,15 +230,5 @@ describe 'Billing plan pages', :feature do
it 'renders no header for missing plan' do it 'renders no header for missing plan' do
expect(page).not_to have_css('.billing-plan-header') expect(page).not_to have_css('.billing-plan-header')
end end
it 'displays all plans' do
page.within('.billing-plans') do
panels = page.all('.card')
expect(panels.length).to eq(plans_data.length)
plans_data.each_with_index do |data, index|
expect(panels[index].find('.card-header')).to have_content(data[:name])
end
end
end
end end
end end
{
"free": {
"rows": [
{
"header": { "icon": "monitor", "title": "Usage" },
"columns": [
{
"id": "seatsInUse",
"label": "Seats currently in use",
"value": null,
"colClass": "number",
"popover": {
"content": "This is the number of seats you will be required to purchase if you update to a paid plan."
}
},
{
"id": "subscriptionStartDate",
"label": "Subscription start date",
"value": null,
"isDate": true
}
]
}
]
},
"trial": {
"rows": [
{
"header": { "icon": "monitor", "title": "Usage" },
"columns": [
{
"id": "seatsInUse",
"label": "Seats currently in use",
"value": null,
"colClass": "number",
"popover": { "content": "Usage count is performed once a day at 12:00 PM." }
},
{
"id": "subscriptionStartDate",
"label": "Trial start date",
"value": null,
"isDate": true
},
{ "id": "subscriptionEndDate", "label": "Trial end date", "value": null, "isDate": true }
]
}
]
},
"default": {
"rows": [
{
"header": { "icon": "monitor", "title": "Usage" },
"columns": [
{
"id": "seatsInSubscription",
"label": "Seats in subscription",
"value": 0,
"colClass": "number"
},
{
"id": "seatsInUse",
"label": "Seats currently in use",
"value": 1,
"colClass": "number",
"popover": { "content": "Usage count is performed once a day at 12:00 PM." }
},
{
"id": "maxSeatsUsed",
"label": "Max seats used",
"value": 0,
"colClass": "number",
"popover": {
"content": "This is the maximum number of users that have existed at the same time since this subscription started."
}
},
{
"id": "seatsOwed",
"label": "Seats owed",
"value": 0,
"colClass": "number",
"popover": {
"content": "GitLab allows you to continue using your subscription even if you exceed the number of seats you purchased. You will be required to pay for these seats upon renewal."
}
}
]
},
{
"header": { "icon": "calendar", "title": "Billing" },
"columns": [
{
"id": "subscriptionStartDate",
"label": "Subscription start date",
"value": "2019-07-31",
"isDate": true
},
{
"id": "subscriptionEndDate",
"label": "Subscription end date",
"value": null,
"isDate": true
},
{
"id": "lastInvoice",
"label": "Last invoice",
"value": null,
"isDate": true,
"popover": {
"content": "This is the last time the GitLab.com team was in contact with you to settle any outstanding balances."
},
"hideContent": true
},
{
"id": "nextInvoice",
"label": "Next invoice",
"value": null,
"isDate": true,
"popover": {
"content": "This is the next date when the GitLab.com team is scheduled to get in contact with you to settle any outstanding balances."
},
"hideContent": true
}
]
}
]
}
}
...@@ -75,10 +75,6 @@ describe('EE billings subscription module mutations', () => { ...@@ -75,10 +75,6 @@ describe('EE billings subscription module mutations', () => {
expect(getStateTableValues(tableKey)).toMatchSnapshot(); expect(getStateTableValues(tableKey)).toMatchSnapshot();
}); });
}); });
it('sets table values', () => {
expect(getStateTableValues('free')).toMatchSnapshot();
});
}); });
describe(types.RECEIVE_SUBSCRIPTION_ERROR, () => { describe(types.RECEIVE_SUBSCRIPTION_ERROR, () => {
......
...@@ -60,11 +60,13 @@ end ...@@ -60,11 +60,13 @@ end
shared_examples 'billings gold trial callout' do shared_examples 'billings gold trial callout' do
context 'on a free plan' do context 'on a free plan' do
let(:plan) { nil } let(:plan) { 'free' }
before do let!(:subscription) do
allow_any_instance_of(EE::Namespace).to receive(:plan).and_return(plan) create(:gitlab_subscription, namespace: namespace, hosted_plan: nil, seats: 15)
end
before do
visit page_path visit page_path
end end
...@@ -80,11 +82,13 @@ shared_examples 'billings gold trial callout' do ...@@ -80,11 +82,13 @@ shared_examples 'billings gold trial callout' do
where(case_names: ->(plan_type) {"like #{plan_type}"}, plan_type: [:bronze, :silver]) where(case_names: ->(plan_type) {"like #{plan_type}"}, plan_type: [:bronze, :silver])
with_them do with_them do
let(:plan) { plans[plan_type] } let(:plan) { plan_type }
before do let!(:subscription) do
allow_any_instance_of(EE::Namespace).to receive(:plan).and_return(plan) create(:gitlab_subscription, namespace: namespace, hosted_plan: plans[plan_type], seats: 15)
end
before do
visit page_path visit page_path
end end
...@@ -99,11 +103,13 @@ shared_examples 'billings gold trial callout' do ...@@ -99,11 +103,13 @@ shared_examples 'billings gold trial callout' do
end end
context 'on a gold plan' do context 'on a gold plan' do
set(:plan) { create(:gold_plan) } let(:plan) { gold_plan.name }
before do let!(:subscription) do
allow_any_instance_of(EE::Namespace).to receive(:plan).and_return(plan) create(:gitlab_subscription, namespace: namespace, hosted_plan: gold_plan, seats: 15)
end
before do
visit page_path visit page_path
end end
......
...@@ -2224,10 +2224,7 @@ msgstr "" ...@@ -2224,10 +2224,7 @@ msgstr ""
msgid "BillingPlans|%{group_name} is currently using the %{plan_link} plan." msgid "BillingPlans|%{group_name} is currently using the %{plan_link} plan."
msgstr "" msgstr ""
msgid "BillingPlans|@%{user_name} you are currently on the %{plan_link} plan." msgid "BillingPlans|Contact Support"
msgstr ""
msgid "BillingPlans|Automatic downgrade and upgrade to some plans is currently not available."
msgstr "" msgstr ""
msgid "BillingPlans|Contact Support" msgid "BillingPlans|Contact Support"
...@@ -2236,10 +2233,10 @@ msgstr "" ...@@ -2236,10 +2233,10 @@ msgstr ""
msgid "BillingPlans|Current plan" msgid "BillingPlans|Current plan"
msgstr "" msgstr ""
msgid "BillingPlans|Customer Support" msgid "BillingPlans|Downgrade"
msgstr "" msgstr ""
msgid "BillingPlans|Downgrade" msgid "BillingPlans|If you would like to downgrade your plan please %{support_link}."
msgstr "" msgstr ""
msgid "BillingPlans|If you would like to downgrade your plan please %{support_link}." msgid "BillingPlans|If you would like to downgrade your plan please %{support_link}."
...@@ -2254,7 +2251,7 @@ msgstr "" ...@@ -2254,7 +2251,7 @@ msgstr ""
msgid "BillingPlans|Manage plan" msgid "BillingPlans|Manage plan"
msgstr "" msgstr ""
msgid "BillingPlans|Please contact %{customer_support_link} in that case." msgid "BillingPlans|Pricing page"
msgstr "" msgstr ""
msgid "BillingPlans|Pricing page" msgid "BillingPlans|Pricing page"
...@@ -2272,6 +2269,9 @@ msgstr "" ...@@ -2272,6 +2269,9 @@ msgstr ""
msgid "BillingPlans|Upgrade" msgid "BillingPlans|Upgrade"
msgstr "" msgstr ""
msgid "BillingPlans|You are currently using the %{plan_link} plan."
msgstr ""
msgid "BillingPlans|Your GitLab.com trial expired on %{expiration_date}. %{learn_more_text}" msgid "BillingPlans|Your GitLab.com trial expired on %{expiration_date}. %{learn_more_text}"
msgstr "" msgstr ""
...@@ -2293,6 +2293,9 @@ msgstr "" ...@@ -2293,6 +2293,9 @@ msgstr ""
msgid "BillingPlans|per user" msgid "BillingPlans|per user"
msgstr "" msgstr ""
msgid "BillingPlan|Upgrade plan"
msgstr ""
msgid "Bitbucket Server Import" msgid "Bitbucket Server Import"
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