Commit 592ad96c authored by Jan Provaznik's avatar Jan Provaznik

Merge branch '321639-mlunoe-add-subscriptions-buy-minutes-route' into 'master'

Feat(subscriptions/buy_minutes): add route [RUN ALL RSPEC] [RUN AS-IF-FOSS]

See merge request gitlab-org/gitlab!54934
parents 3ea74509 9da85b33
---
name: new_route_ci_minutes_purchase
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/54934
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/322582
milestone: '13.10'
type: development
group: group::purchase
default_enabled: false
import mountSubscriptionsApplication from 'ee/subscriptions/buy_minutes';
mountSubscriptionsApplication();
<script>
import StepOrderApp from 'ee/vue_shared/components/step_order_app.vue';
export default {
components: {
StepOrderApp,
},
};
</script>
<template>
<step-order-app>
<template #checkout></template>
<template #order-summary></template>
</step-order-app>
</template>
import Vue from 'vue';
import App from './components/app.vue';
export default () => {
const el = document.getElementById('js-buy-minutes');
return new Vue({
el,
components: {
App,
},
render(createElement) {
return createElement(App);
},
});
};
<script>
import StepOrderApp from 'ee/vue_shared/components/step_order_app.vue';
import Checkout from './checkout.vue';
import OrderSummary from './order_summary.vue';
export default {
components: {
StepOrderApp,
Checkout,
OrderSummary,
},
};
</script>
<template>
<step-order-app>
<template #checkout>
<checkout />
</template>
<template #order-summary>
<order-summary />
</template>
</step-order-app>
</template>
import Vue from 'vue'; import Vue from 'vue';
import Checkout from './components/checkout.vue'; import App from './components/app.vue';
import OrderSummary from './components/order_summary.vue';
import createStore from './store'; import createStore from './store';
export default () => { export default () => {
const checkoutEl = document.getElementById('checkout'); const el = document.getElementById('js-new-subscription');
const summaryEl = document.getElementById('summary'); const store = createStore(el.dataset);
const store = createStore(checkoutEl.dataset);
// eslint-disable-next-line no-new
new Vue({
el: checkoutEl,
store,
render(createElement) {
return createElement(Checkout);
},
});
return new Vue({ return new Vue({
el: summaryEl, el,
store, store,
components: {
App,
},
render(createElement) { render(createElement) {
return createElement(OrderSummary); return createElement(App);
}, },
}); });
}; };
<template>
<div
class="row gl-flex-grow-1 gl-flex-direction-column flex-nowrap gl-lg-flex-direction-row flex-xl-row flex-lg-wrap flex-xl-wrap"
>
<div
class="checkout-pane gl-px-3 gl-align-items-center gl-bg-gray-10 col-lg-7 gl-display-flex gl-flex-direction-column gl-flex-grow-1"
>
<slot name="checkout"></slot>
</div>
<div
class="gl-pb-3 gl-px-3 px-lg-7 col-lg-5 gl-display-flex gl-flex-direction-row gl-justify-content-center"
>
<slot name="order-summary"></slot>
</div>
</div>
</template>
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
class SubscriptionsController < ApplicationController class SubscriptionsController < ApplicationController
layout 'checkout' layout 'checkout'
skip_before_action :authenticate_user!, only: :new skip_before_action :authenticate_user!, only: [:new, :buy_minutes]
feature_category :purchase feature_category :purchase
...@@ -24,10 +24,12 @@ class SubscriptionsController < ApplicationController ...@@ -24,10 +24,12 @@ class SubscriptionsController < ApplicationController
end end
def new def new
return if current_user redirect_unauthenticated_user('checkout')
end
store_location_for :user, request.fullpath def buy_minutes
redirect_to new_user_registration_path(redirect_from: 'checkout') render_404 unless Feature.enabled?(:new_route_ci_minutes_purchase, default_enabled: :yaml)
redirect_unauthenticated_user
end end
def payment_form def payment_form
...@@ -90,4 +92,11 @@ class SubscriptionsController < ApplicationController ...@@ -90,4 +92,11 @@ class SubscriptionsController < ApplicationController
def customer_portal_new_subscription_url def customer_portal_new_subscription_url
"#{EE::SUBSCRIPTIONS_URL}/subscriptions/new?plan_id=#{params[:plan_id]}&transaction=create_subscription" "#{EE::SUBSCRIPTIONS_URL}/subscriptions/new?plan_id=#{params[:plan_id]}&transaction=create_subscription"
end end
def redirect_unauthenticated_user(from = action_name)
return if current_user
store_location_for :user, request.fullpath
redirect_to new_user_registration_path(redirect_from: from)
end
end end
- page_title _('Buy CI Minutes')
#js-buy-minutes{ data: subscription_data }
- page_title _('Checkout') - page_title _('Checkout')
.row.flex-grow-1.flex-column.flex-nowrap.flex-lg-row.flex-xl-row.flex-lg-wrap.flex-xl-wrap #js-new-subscription{ data: subscription_data }
.checkout-pane.px-3.align-items-center.bg-gray-light.col-lg-7.d-flex.flex-column.flex-grow-1
#checkout{ data: subscription_data }
.pb-3.px-3.px-lg-7.col-lg-5.d-flex.flex-row.justify-content-center
#summary
# frozen_string_literal: true # frozen_string_literal: true
resource :subscriptions, only: [:new, :create] do resource :subscriptions, only: [:new, :create] do
get :buy_minutes
get :payment_form get :payment_form
get :payment_method get :payment_method
......
...@@ -5,22 +5,25 @@ require 'spec_helper' ...@@ -5,22 +5,25 @@ require 'spec_helper'
RSpec.describe SubscriptionsController do RSpec.describe SubscriptionsController do
let_it_be(:user) { create(:user) } let_it_be(:user) { create(:user) }
describe 'GET #new' do shared_examples 'unauthenticated subscription request' do |redirect_from|
subject { get :new, params: { plan_id: 'bronze_id' } }
context 'with unauthenticated user' do
it { is_expected.to have_gitlab_http_status(:redirect) } it { is_expected.to have_gitlab_http_status(:redirect) }
it { is_expected.to redirect_to new_user_registration_path(redirect_from: 'checkout') } it { is_expected.to redirect_to new_user_registration_path(redirect_from: redirect_from) }
it 'stores subscription URL for later' do it 'stores subscription URL for later' do
subject subject
expected_subscription_path = new_subscriptions_path(plan_id: 'bronze_id') expected_subscription_path = new_subscriptions_path(plan_id: 'bronze_id') if redirect_from == 'checkout'
expected_subscription_path = buy_minutes_subscriptions_path(plan_id: 'bronze_id') if redirect_from == 'buy_minutes'
expect(controller.stored_location_for(:user)).to eq(expected_subscription_path) expect(controller.stored_location_for(:user)).to eq(expected_subscription_path)
end end
end end
describe 'GET #new' do
subject { get :new, params: { plan_id: 'bronze_id' } }
it_behaves_like 'unauthenticated subscription request', 'checkout'
context 'with authenticated user' do context 'with authenticated user' do
before do before do
sign_in(user) sign_in(user)
...@@ -31,6 +34,30 @@ RSpec.describe SubscriptionsController do ...@@ -31,6 +34,30 @@ RSpec.describe SubscriptionsController do
end end
end end
describe 'GET #buy_minutes' do
subject { get :buy_minutes, params: { plan_id: 'bronze_id' } }
it_behaves_like 'unauthenticated subscription request', 'buy_minutes'
context 'with authenticated user' do
before do
sign_in(user)
end
it { is_expected.to render_template 'layouts/checkout' }
it { is_expected.to render_template :buy_minutes }
end
context 'with :new_route_ci_minutes_purchase disabled' do
before do
sign_in(user)
stub_feature_flags(new_route_ci_minutes_purchase: false)
end
it { is_expected.to have_gitlab_http_status(:not_found) }
end
end
describe 'GET #payment_form' do describe 'GET #payment_form' do
subject { get :payment_form, params: { id: 'cc' } } subject { get :payment_form, params: { id: 'cc' } }
......
# frozen_string_literal: true
RSpec.shared_examples_for 'subscription form data' do |js_selector|
before do
allow(view).to receive(:subscription_data).and_return(
setup_for_company: 'true',
full_name: 'First Last',
plan_data: '[{"id":"bronze_id","code":"bronze","price_per_year":48.0}]',
plan_id: 'bronze_id'
)
end
subject { render }
it { is_expected.to have_selector("#{js_selector}[data-setup-for-company='true']") }
it { is_expected.to have_selector("#{js_selector}[data-full-name='First Last']") }
it { is_expected.to have_selector("#{js_selector}[data-plan-data='[{\"id\":\"bronze_id\",\"code\":\"bronze\",\"price_per_year\":48.0}]']") }
it { is_expected.to have_selector("#{js_selector}[data-plan-id='bronze_id']") }
end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'subscriptions/buy_minutes' do
it_behaves_like 'subscription form data', '#js-buy-minutes'
end
...@@ -3,19 +3,5 @@ ...@@ -3,19 +3,5 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe 'subscriptions/new' do RSpec.describe 'subscriptions/new' do
before do it_behaves_like 'subscription form data', '#js-new-subscription'
allow(view).to receive(:subscription_data).and_return(
setup_for_company: 'true',
full_name: 'First Last',
plan_data: '[{"id":"bronze_id","code":"bronze","price_per_year":48.0}]',
plan_id: 'bronze_id'
)
end
subject { render }
it { is_expected.to have_selector("#checkout[data-setup-for-company='true']") }
it { is_expected.to have_selector("#checkout[data-full-name='First Last']") }
it { is_expected.to have_selector("#checkout[data-plan-data='[{\"id\":\"bronze_id\",\"code\":\"bronze\",\"price_per_year\":48.0}]']") }
it { is_expected.to have_selector("#checkout[data-plan-id='bronze_id']") }
end end
...@@ -5199,6 +5199,9 @@ msgstr "" ...@@ -5199,6 +5199,9 @@ msgstr ""
msgid "Busy" msgid "Busy"
msgstr "" msgstr ""
msgid "Buy CI Minutes"
msgstr ""
msgid "Buy License" msgid "Buy License"
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