mirror of
https://github.com/LaCasemate/fab-manager.git
synced 2025-01-18 07:52:23 +01:00
[ongoing] refactor user.subscriptions to save history
TODO: - refactor subscription.save_with_payment (stripe) - offer free days - extend the subscription - renew a subscription - buy subscription + reservation
This commit is contained in:
parent
b537571656
commit
bef3118649
@ -1,7 +1,7 @@
|
|||||||
class API::SubscriptionsController < API::ApiController
|
class API::SubscriptionsController < API::ApiController
|
||||||
include FablabConfiguration
|
include FablabConfiguration
|
||||||
|
|
||||||
before_action :set_subscription, only: [:show, :edit, :update, :destroy]
|
before_action :set_subscription, only: %i[show edit update destroy]
|
||||||
before_action :authenticate_user!
|
before_action :authenticate_user!
|
||||||
|
|
||||||
def show
|
def show
|
||||||
@ -12,15 +12,12 @@ class API::SubscriptionsController < API::ApiController
|
|||||||
if fablab_plans_deactivated?
|
if fablab_plans_deactivated?
|
||||||
head 403
|
head 403
|
||||||
else
|
else
|
||||||
if current_user.is_admin?
|
method = current_user.is_admin? ? :local : :stripe
|
||||||
@subscription = Subscription.find_or_initialize_by(user_id: subscription_params[:user_id])
|
user_id = current_user.is_admin? ? subscription_params[:user_id] : current_user.id
|
||||||
@subscription.attributes = subscription_params
|
|
||||||
is_subscribe = @subscription.save_with_local_payment(true, coupon_params[:coupon_code])
|
@subscription = Subscription.new(subscription_params)
|
||||||
else
|
is_subscribe = Subscribe.new(method, user_id).pay_and_save(@subscription, coupon_params[:coupon_code], true)
|
||||||
@subscription = Subscription.find_or_initialize_by(user_id: current_user.id)
|
|
||||||
@subscription.attributes = subscription_params
|
|
||||||
is_subscribe = @subscription.save_with_payment(true, coupon_params[:coupon_code])
|
|
||||||
end
|
|
||||||
if is_subscribe
|
if is_subscribe
|
||||||
render :show, status: :created, location: @subscription
|
render :show, status: :created, location: @subscription
|
||||||
else
|
else
|
||||||
@ -44,31 +41,30 @@ class API::SubscriptionsController < API::ApiController
|
|||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
# Use callbacks to share common setup or constraints between actions.
|
|
||||||
def set_subscription
|
|
||||||
@subscription = Subscription.find(params[:id])
|
|
||||||
end
|
|
||||||
|
|
||||||
# Never trust parameters from the scary internet, only allow the white list through.
|
# Use callbacks to share common setup or constraints between actions.
|
||||||
def subscription_params
|
def set_subscription
|
||||||
params.require(:subscription).permit(:plan_id, :user_id, :card_token)
|
@subscription = Subscription.find(params[:id])
|
||||||
end
|
end
|
||||||
|
|
||||||
def coupon_params
|
# Never trust parameters from the scary internet, only allow the white list through.
|
||||||
params.permit(:coupon_code)
|
def subscription_params
|
||||||
end
|
params.require(:subscription).permit(:plan_id, :user_id, :card_token)
|
||||||
|
end
|
||||||
|
|
||||||
def subscription_update_params
|
def coupon_params
|
||||||
params.require(:subscription).permit(:expired_at)
|
params.permit(:coupon_code)
|
||||||
end
|
end
|
||||||
|
|
||||||
# TODO refactor subscriptions logic and move this in model/validator
|
def subscription_update_params
|
||||||
def valid_card_token?(token)
|
params.require(:subscription).permit(:expired_at)
|
||||||
begin
|
end
|
||||||
Stripe::Token.retrieve(token)
|
|
||||||
rescue Stripe::InvalidRequestError => e
|
# TODO refactor subscriptions logic and move this in model/validator
|
||||||
@subscription.errors[:card_token] << e.message
|
def valid_card_token?(token)
|
||||||
false
|
Stripe::Token.retrieve(token)
|
||||||
end
|
rescue Stripe::InvalidRequestError => e
|
||||||
end
|
@subscription.errors[:card_token] << e.message
|
||||||
|
false
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
@ -169,7 +169,7 @@ class Invoice < ActiveRecord::Base
|
|||||||
|
|
||||||
def is_subscription_invoice?
|
def is_subscription_invoice?
|
||||||
invoice_items.each do |ii|
|
invoice_items.each do |ii|
|
||||||
return true if ii.subscription and !ii.subscription.is_expired?
|
return true if ii.subscription and !ii.subscription.expired?
|
||||||
end
|
end
|
||||||
false
|
false
|
||||||
end
|
end
|
||||||
|
@ -2,7 +2,7 @@ class NotificationType
|
|||||||
include NotifyWith::NotificationType
|
include NotifyWith::NotificationType
|
||||||
|
|
||||||
# DANGER: dont remove a notification type!!!
|
# DANGER: dont remove a notification type!!!
|
||||||
notification_type_names %w(
|
notification_type_names %w[
|
||||||
notify_admin_when_project_published
|
notify_admin_when_project_published
|
||||||
notify_project_collaborator_to_valid
|
notify_project_collaborator_to_valid
|
||||||
notify_project_author_when_collaborator_valid
|
notify_project_author_when_collaborator_valid
|
||||||
@ -42,5 +42,5 @@ class NotificationType
|
|||||||
notify_admin_export_complete
|
notify_admin_export_complete
|
||||||
notify_member_about_coupon
|
notify_member_about_coupon
|
||||||
notify_member_reservation_reminder
|
notify_member_reservation_reminder
|
||||||
)
|
]
|
||||||
end
|
end
|
||||||
|
@ -13,163 +13,150 @@ class Subscription < ActiveRecord::Base
|
|||||||
attr_accessor :card_token
|
attr_accessor :card_token
|
||||||
|
|
||||||
# creation
|
# creation
|
||||||
after_save :notify_member_subscribed_plan, if: :is_new?
|
after_save :notify_member_subscribed_plan, if: :new?
|
||||||
after_save :notify_admin_subscribed_plan, if: :is_new?
|
after_save :notify_admin_subscribed_plan, if: :new?
|
||||||
after_save :notify_partner_subscribed_plan, if: :of_partner_plan?
|
after_save :notify_partner_subscribed_plan, if: :of_partner_plan?
|
||||||
|
|
||||||
# Stripe subscription payment
|
# Stripe subscription payment
|
||||||
# @params [invoice] if true then subscription pay itself, dont pay with reservation
|
# @params [invoice] if true then subscription pay itself, dont pay with reservation
|
||||||
# if false then subscription pay with reservation
|
# if false then subscription pay with reservation
|
||||||
def save_with_payment(invoice = true, coupon_code = nil)
|
def save_with_payment(invoice = true, coupon_code = nil)
|
||||||
if valid?
|
return unless valid?
|
||||||
begin
|
|
||||||
customer = Stripe::Customer.retrieve(user.stp_customer_id)
|
|
||||||
invoice_items = []
|
|
||||||
|
|
||||||
unless coupon_code.nil?
|
begin
|
||||||
@coupon = Coupon.find_by(code: coupon_code)
|
customer = Stripe::Customer.retrieve(user.stp_customer_id)
|
||||||
if not @coupon.nil? and @coupon.status(user.id) == 'active'
|
invoice_items = []
|
||||||
total = plan.amount
|
|
||||||
|
|
||||||
discount = 0
|
unless coupon_code.nil?
|
||||||
if @coupon.type == 'percent_off'
|
@coupon = Coupon.find_by(code: coupon_code)
|
||||||
discount = (total * @coupon.percent_off / 100).to_i
|
raise InvalidCouponError if @coupon.nil? || @coupon.status(user.id) != 'active'
|
||||||
elsif @coupon.type == 'amount_off'
|
|
||||||
discount = @coupon.amount_off
|
|
||||||
else
|
|
||||||
raise InvalidCouponError
|
|
||||||
end
|
|
||||||
|
|
||||||
invoice_items << Stripe::InvoiceItem.create(
|
total = plan.amount
|
||||||
customer: user.stp_customer_id,
|
|
||||||
amount: -discount,
|
discount = 0
|
||||||
currency: Rails.application.secrets.stripe_currency,
|
if @coupon.type == 'percent_off'
|
||||||
description: "coupon #{@coupon.code} - subscription"
|
discount = (total * @coupon.percent_off / 100).to_i
|
||||||
)
|
elsif @coupon.type == 'amount_off'
|
||||||
else
|
discount = @coupon.amount_off
|
||||||
raise InvalidCouponError
|
else
|
||||||
end
|
raise InvalidCouponError
|
||||||
end
|
end
|
||||||
|
|
||||||
# only add a wallet invoice item if pay subscription
|
invoice_items << Stripe::InvoiceItem.create(
|
||||||
# dont add if pay subscription + reservation
|
customer: user.stp_customer_id,
|
||||||
if invoice
|
amount: -discount,
|
||||||
@wallet_amount_debit = get_wallet_amount_debit
|
currency: Rails.application.secrets.stripe_currency,
|
||||||
if @wallet_amount_debit != 0
|
description: "coupon #{@coupon.code} - subscription"
|
||||||
invoice_items << Stripe::InvoiceItem.create(
|
)
|
||||||
customer: user.stp_customer_id,
|
|
||||||
amount: -@wallet_amount_debit,
|
|
||||||
currency: Rails.application.secrets.stripe_currency,
|
|
||||||
description: "wallet -#{@wallet_amount_debit / 100.0}"
|
|
||||||
)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
new_subscription = customer.subscriptions.create(plan: plan.stp_plan_id, source: card_token)
|
|
||||||
# very important to set expired_at to nil that can allow method is_new? to return true
|
|
||||||
# for send the notification
|
|
||||||
# TODO: Refactoring
|
|
||||||
update_column(:expired_at, nil) unless new_record?
|
|
||||||
self.stp_subscription_id = new_subscription.id
|
|
||||||
self.canceled_at = nil
|
|
||||||
self.expired_at = Time.at(new_subscription.current_period_end)
|
|
||||||
save!
|
|
||||||
|
|
||||||
UsersCredits::Manager.new(user: self.user).reset_credits if expired_date_changed
|
|
||||||
|
|
||||||
# generate invoice
|
|
||||||
stp_invoice = Stripe::Invoice.all(customer: user.stp_customer_id, limit: 1).data.first
|
|
||||||
if invoice
|
|
||||||
invoc = generate_invoice(stp_invoice.id, coupon_code)
|
|
||||||
# debit wallet
|
|
||||||
wallet_transaction = debit_user_wallet
|
|
||||||
if wallet_transaction
|
|
||||||
invoc.wallet_amount = @wallet_amount_debit
|
|
||||||
invoc.wallet_transaction_id = wallet_transaction.id
|
|
||||||
end
|
|
||||||
invoc.save
|
|
||||||
end
|
|
||||||
# cancel subscription after create
|
|
||||||
cancel
|
|
||||||
return true
|
|
||||||
rescue Stripe::CardError => card_error
|
|
||||||
clear_wallet_and_goupon_invoice_items(invoice_items)
|
|
||||||
logger.error card_error
|
|
||||||
errors[:card] << card_error.message
|
|
||||||
return false
|
|
||||||
rescue Stripe::InvalidRequestError => e
|
|
||||||
clear_wallet_and_goupon_invoice_items(invoice_items)
|
|
||||||
# Invalid parameters were supplied to Stripe's API
|
|
||||||
logger.error e
|
|
||||||
errors[:payment] << e.message
|
|
||||||
return false
|
|
||||||
rescue Stripe::AuthenticationError => e
|
|
||||||
clear_wallet_and_goupon_invoice_items(invoice_items)
|
|
||||||
# Authentication with Stripe's API failed
|
|
||||||
# (maybe you changed API keys recently)
|
|
||||||
logger.error e
|
|
||||||
errors[:payment] << e.message
|
|
||||||
return false
|
|
||||||
rescue Stripe::APIConnectionError => e
|
|
||||||
clear_wallet_and_goupon_invoice_items(invoice_items)
|
|
||||||
# Network communication with Stripe failed
|
|
||||||
logger.error e
|
|
||||||
errors[:payment] << e.message
|
|
||||||
return false
|
|
||||||
rescue Stripe::StripeError => e
|
|
||||||
clear_wallet_and_goupon_invoice_items(invoice_items)
|
|
||||||
# Display a very generic error to the user, and maybe send
|
|
||||||
# yourself an email
|
|
||||||
logger.error e
|
|
||||||
errors[:payment] << e.message
|
|
||||||
return false
|
|
||||||
rescue => e
|
|
||||||
clear_wallet_and_goupon_invoice_items(invoice_items)
|
|
||||||
# Something else happened, completely unrelated to Stripe
|
|
||||||
logger.error e
|
|
||||||
errors[:payment] << e.message
|
|
||||||
return false
|
|
||||||
end
|
end
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
# @params [invoice] if true then subscription pay itself, dont pay with reservation
|
# only add a wallet invoice item if pay subscription
|
||||||
# if false then subscription pay with reservation
|
# dont add if pay subscription + reservation
|
||||||
def save_with_local_payment(invoice = true, coupon_code = nil)
|
if invoice
|
||||||
if valid?
|
@wallet_amount_debit = get_wallet_amount_debit
|
||||||
|
if @wallet_amount_debit != 0
|
||||||
|
invoice_items << Stripe::InvoiceItem.create(
|
||||||
|
customer: user.stp_customer_id,
|
||||||
|
amount: -@wallet_amount_debit,
|
||||||
|
currency: Rails.application.secrets.stripe_currency,
|
||||||
|
description: "wallet -#{@wallet_amount_debit / 100.0}"
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
new_subscription = customer.subscriptions.create(plan: plan.stp_plan_id, source: card_token)
|
||||||
# very important to set expired_at to nil that can allow method is_new? to return true
|
# very important to set expired_at to nil that can allow method is_new? to return true
|
||||||
# for send the notification
|
# for send the notification
|
||||||
# TODO: Refactoring
|
# TODO: Refactoring
|
||||||
update_column(:expired_at, nil) unless new_record?
|
update_column(:expired_at, nil) unless new_record?
|
||||||
self.stp_subscription_id = nil
|
self.stp_subscription_id = new_subscription.id
|
||||||
self.canceled_at = nil
|
self.canceled_at = nil
|
||||||
set_expired_at
|
self.expired_at = Time.at(new_subscription.current_period_end)
|
||||||
if save
|
save!
|
||||||
UsersCredits::Manager.new(user: self.user).reset_credits if expired_date_changed
|
|
||||||
if invoice
|
|
||||||
@wallet_amount_debit = get_wallet_amount_debit
|
|
||||||
|
|
||||||
# debit wallet
|
UsersCredits::Manager.new(user: self.user).reset_credits if expired_date_changed
|
||||||
wallet_transaction = debit_user_wallet
|
|
||||||
|
|
||||||
if !self.user.invoicing_disabled?
|
# generate invoice
|
||||||
invoc = generate_invoice(nil, coupon_code)
|
stp_invoice = Stripe::Invoice.all(customer: user.stp_customer_id, limit: 1).data.first
|
||||||
if wallet_transaction
|
if invoice
|
||||||
invoc.wallet_amount = @wallet_amount_debit
|
invoc = generate_invoice(stp_invoice.id, coupon_code)
|
||||||
invoc.wallet_transaction_id = wallet_transaction.id
|
# debit wallet
|
||||||
end
|
wallet_transaction = debit_user_wallet
|
||||||
invoc.save
|
if wallet_transaction
|
||||||
end
|
invoc.wallet_amount = @wallet_amount_debit
|
||||||
|
invoc.wallet_transaction_id = wallet_transaction.id
|
||||||
end
|
end
|
||||||
return true
|
invoc.save
|
||||||
else
|
|
||||||
return false
|
|
||||||
end
|
end
|
||||||
else
|
# cancel subscription after create
|
||||||
|
cancel
|
||||||
|
return true
|
||||||
|
rescue Stripe::CardError => card_error
|
||||||
|
clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
logger.error card_error
|
||||||
|
errors[:card] << card_error.message
|
||||||
|
return false
|
||||||
|
rescue Stripe::InvalidRequestError => e
|
||||||
|
clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
# Invalid parameters were supplied to Stripe's API
|
||||||
|
logger.error e
|
||||||
|
errors[:payment] << e.message
|
||||||
|
return false
|
||||||
|
rescue Stripe::AuthenticationError => e
|
||||||
|
clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
# Authentication with Stripe's API failed
|
||||||
|
# (maybe you changed API keys recently)
|
||||||
|
logger.error e
|
||||||
|
errors[:payment] << e.message
|
||||||
|
return false
|
||||||
|
rescue Stripe::APIConnectionError => e
|
||||||
|
clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
# Network communication with Stripe failed
|
||||||
|
logger.error e
|
||||||
|
errors[:payment] << e.message
|
||||||
|
return false
|
||||||
|
rescue Stripe::StripeError => e
|
||||||
|
clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
# Display a very generic error to the user, and maybe send
|
||||||
|
# yourself an email
|
||||||
|
logger.error e
|
||||||
|
errors[:payment] << e.message
|
||||||
|
return false
|
||||||
|
rescue => e
|
||||||
|
clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
# Something else happened, completely unrelated to Stripe
|
||||||
|
logger.error e
|
||||||
|
errors[:payment] << e.message
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @params [invoice] if true then only the subscription is payed, without reservation
|
||||||
|
# if false then the subscription is payed with reservation
|
||||||
|
def save_with_local_payment(invoice = true, coupon_code = nil)
|
||||||
|
return false unless valid?
|
||||||
|
|
||||||
|
return false unless save
|
||||||
|
|
||||||
|
UsersCredits::Manager.new(user: user).reset_credits if expired_date_changed
|
||||||
|
if invoice
|
||||||
|
@wallet_amount_debit = get_wallet_amount_debit
|
||||||
|
|
||||||
|
# debit wallet
|
||||||
|
wallet_transaction = debit_user_wallet
|
||||||
|
|
||||||
|
unless user.invoicing_disabled?
|
||||||
|
invoc = generate_invoice(nil, coupon_code)
|
||||||
|
if wallet_transaction
|
||||||
|
invoc.wallet_amount = @wallet_amount_debit
|
||||||
|
invoc.wallet_transaction_id = wallet_transaction.id
|
||||||
|
end
|
||||||
|
invoc.save
|
||||||
|
end
|
||||||
|
end
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
def generate_invoice(stp_invoice_id = nil, coupon_code = nil)
|
def generate_invoice(stp_invoice_id = nil, coupon_code = nil)
|
||||||
coupon_id = nil
|
coupon_id = nil
|
||||||
total = plan.amount
|
total = plan.amount
|
||||||
@ -200,11 +187,11 @@ class Subscription < ActiveRecord::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def cancel
|
def cancel
|
||||||
if stp_subscription_id.present?
|
return unless stp_subscription_id.present?
|
||||||
stp_subscription = stripe_subscription
|
|
||||||
stp_subscription.delete(at_period_end: true)
|
stp_subscription = stripe_subscription
|
||||||
update_columns(canceled_at: Time.now)
|
stp_subscription.delete(at_period_end: true)
|
||||||
end
|
update_columns(canceled_at: Time.now)
|
||||||
end
|
end
|
||||||
|
|
||||||
def stripe_subscription
|
def stripe_subscription
|
||||||
@ -212,7 +199,7 @@ class Subscription < ActiveRecord::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def expire(time)
|
def expire(time)
|
||||||
if !is_expired?
|
if !expired?
|
||||||
update_columns(expired_at: time, canceled_at: time)
|
update_columns(expired_at: time, canceled_at: time)
|
||||||
notify_admin_subscription_canceled
|
notify_admin_subscription_canceled
|
||||||
notify_member_subscription_canceled
|
notify_member_subscription_canceled
|
||||||
@ -222,7 +209,7 @@ class Subscription < ActiveRecord::Base
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def is_expired?
|
def expired?
|
||||||
expired_at <= Time.now
|
expired_at <= Time.now
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -231,14 +218,15 @@ class Subscription < ActiveRecord::Base
|
|||||||
|
|
||||||
self.expired_at = expired_at
|
self.expired_at = expired_at
|
||||||
if save
|
if save
|
||||||
UsersCredits::Manager.new(user: self.user).reset_credits if !free_days
|
UsersCredits::Manager.new(user: user).reset_credits unless free_days
|
||||||
notify_subscription_extended(free_days)
|
notify_subscription_extended(free_days)
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
return false
|
false
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def notify_member_subscribed_plan
|
def notify_member_subscribed_plan
|
||||||
NotificationCenter.call type: 'notify_member_subscribed_plan',
|
NotificationCenter.call type: 'notify_member_subscribed_plan',
|
||||||
receiver: user,
|
receiver: user,
|
||||||
@ -289,8 +277,9 @@ class Subscription < ActiveRecord::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def expired_date_changed
|
def expired_date_changed
|
||||||
p_value = self.previous_changes[:expired_at][0]
|
p_value = previous_changes[:expired_at][0]
|
||||||
return true if p_value.nil?
|
return true if p_value.nil?
|
||||||
|
|
||||||
p_value.to_date != expired_at.to_date and expired_at > p_value
|
p_value.to_date != expired_at.to_date and expired_at > p_value
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -298,7 +287,7 @@ class Subscription < ActiveRecord::Base
|
|||||||
# !expired_at_was.nil? and expired_at_changed?
|
# !expired_at_was.nil? and expired_at_changed?
|
||||||
# end
|
# end
|
||||||
|
|
||||||
def is_new?
|
def new?
|
||||||
expired_at_was.nil?
|
expired_at_was.nil?
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -308,18 +297,16 @@ class Subscription < ActiveRecord::Base
|
|||||||
|
|
||||||
def get_wallet_amount_debit
|
def get_wallet_amount_debit
|
||||||
total = plan.amount
|
total = plan.amount
|
||||||
if @coupon
|
total = CouponService.new.apply(total, @coupon, user.id) if @coupon
|
||||||
total = CouponService.new.apply(total, @coupon, user.id)
|
|
||||||
end
|
|
||||||
wallet_amount = (user.wallet.amount * 100).to_i
|
wallet_amount = (user.wallet.amount * 100).to_i
|
||||||
return wallet_amount >= total ? total : wallet_amount
|
wallet_amount >= total ? total : wallet_amount
|
||||||
end
|
end
|
||||||
|
|
||||||
def debit_user_wallet
|
def debit_user_wallet
|
||||||
if @wallet_amount_debit.present? and @wallet_amount_debit != 0
|
return unless @wallet_amount_debit.present? || @wallet_amount_debit.zero?
|
||||||
amount = @wallet_amount_debit / 100.0
|
|
||||||
return WalletService.new(user: user, wallet: user.wallet).debit(amount, self)
|
amount = @wallet_amount_debit / 100.0
|
||||||
end
|
WalletService.new(user: user, wallet: user.wallet).debit(amount, self)
|
||||||
end
|
end
|
||||||
|
|
||||||
def clear_wallet_and_goupon_invoice_items(invoice_items)
|
def clear_wallet_and_goupon_invoice_items(invoice_items)
|
||||||
|
@ -220,7 +220,7 @@ class User < ActiveRecord::Base
|
|||||||
|
|
||||||
## used to allow the migration of existing users between authentication providers
|
## used to allow the migration of existing users between authentication providers
|
||||||
def generate_auth_migration_token
|
def generate_auth_migration_token
|
||||||
update_attributes(:auth_token => Devise.friendly_token)
|
update_attributes(auth_token: Devise.friendly_token)
|
||||||
end
|
end
|
||||||
|
|
||||||
## link the current user to the given provider (omniauth attributes hash)
|
## link the current user to the given provider (omniauth attributes hash)
|
||||||
@ -241,7 +241,7 @@ class User < ActiveRecord::Base
|
|||||||
## Merge the provided User's SSO details into the current user and drop the provided user to ensure the unity
|
## Merge the provided User's SSO details into the current user and drop the provided user to ensure the unity
|
||||||
## @param sso_user {User} the provided user will be DELETED after the merge was successful
|
## @param sso_user {User} the provided user will be DELETED after the merge was successful
|
||||||
def merge_from_sso(sso_user)
|
def merge_from_sso(sso_user)
|
||||||
# update the attibutes to link the account to the sso account
|
# update the attributes to link the account to the sso account
|
||||||
self.provider = sso_user.provider
|
self.provider = sso_user.provider
|
||||||
self.uid = sso_user.uid
|
self.uid = sso_user.uid
|
||||||
|
|
||||||
@ -264,7 +264,7 @@ class User < ActiveRecord::Base
|
|||||||
set_data_from_sso_mapping(field, value) unless field == 'user.email' && value.end_with?('-duplicate')
|
set_data_from_sso_mapping(field, value) unless field == 'user.email' && value.end_with?('-duplicate')
|
||||||
end
|
end
|
||||||
|
|
||||||
# run the account transfert in an SQL transaction to ensure data integrity
|
# run the account transfer in an SQL transaction to ensure data integrity
|
||||||
User.transaction do
|
User.transaction do
|
||||||
# remove the temporary account
|
# remove the temporary account
|
||||||
sso_user.destroy
|
sso_user.destroy
|
||||||
@ -319,7 +319,7 @@ class User < ActiveRecord::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def notify_admin_when_user_is_created
|
def notify_admin_when_user_is_created
|
||||||
if need_completion? and not provider.nil?
|
if need_completion? && !provider.nil?
|
||||||
NotificationCenter.call type: 'notify_admin_when_user_is_imported',
|
NotificationCenter.call type: 'notify_admin_when_user_is_imported',
|
||||||
receiver: User.admins,
|
receiver: User.admins,
|
||||||
attached_object: self
|
attached_object: self
|
||||||
|
@ -2,7 +2,8 @@ class SubscriptionExtensionAfterReservation
|
|||||||
attr_accessor :user, :reservation
|
attr_accessor :user, :reservation
|
||||||
|
|
||||||
def initialize(reservation)
|
def initialize(reservation)
|
||||||
@user, @reservation = reservation.user, reservation
|
@user = reservation.user
|
||||||
|
@reservation = reservation
|
||||||
end
|
end
|
||||||
|
|
||||||
def extend_subscription_if_eligible
|
def extend_subscription_if_eligible
|
||||||
@ -13,9 +14,10 @@ class SubscriptionExtensionAfterReservation
|
|||||||
return false unless reservation.reservable_type == 'Training'
|
return false unless reservation.reservable_type == 'Training'
|
||||||
return false if user.reservations.where(reservable_type: 'Training').count != 1
|
return false if user.reservations.where(reservable_type: 'Training').count != 1
|
||||||
return false unless user.subscription
|
return false unless user.subscription
|
||||||
return false if user.subscription.is_expired?
|
return false if user.subscription.expired?
|
||||||
return false unless user.subscribed_plan.is_rolling
|
return false unless user.subscribed_plan.is_rolling
|
||||||
return true
|
|
||||||
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
def extend_subscription
|
def extend_subscription
|
||||||
|
17
app/services/subscriptions/subscribe.rb
Normal file
17
app/services/subscriptions/subscribe.rb
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
class Subscribe
|
||||||
|
attr_accessor :payment_method, :user_id
|
||||||
|
|
||||||
|
def initialize(payment_method, user_id)
|
||||||
|
@payment_method = payment_method
|
||||||
|
@user_id = user_id
|
||||||
|
end
|
||||||
|
|
||||||
|
def pay_and_save(subscription, coupon, invoice)
|
||||||
|
subscription.user_id = user_id
|
||||||
|
if payment_method == :local
|
||||||
|
subscription.save_with_local_payment(invoice, coupon)
|
||||||
|
elsif payment_method == :stripe
|
||||||
|
subscription.save_with_payment(invoice, coupon)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
@ -1,7 +1,7 @@
|
|||||||
require 'forwardable'
|
require 'forwardable'
|
||||||
|
|
||||||
module UsersCredits
|
module UsersCredits
|
||||||
class AlreadyUpdatedError < StandardError;end;
|
class AlreadyUpdatedError < StandardError; end
|
||||||
|
|
||||||
class Manager
|
class Manager
|
||||||
extend Forwardable
|
extend Forwardable
|
||||||
@ -31,7 +31,8 @@ module UsersCredits
|
|||||||
end
|
end
|
||||||
|
|
||||||
module Managers
|
module Managers
|
||||||
class User # that class is responsible for reseting users_credits of a user
|
# that class is responsible for resetting users_credits of a user
|
||||||
|
class User
|
||||||
attr_reader :user
|
attr_reader :user
|
||||||
|
|
||||||
def initialize(user)
|
def initialize(user)
|
||||||
@ -46,7 +47,8 @@ module UsersCredits
|
|||||||
class Reservation
|
class Reservation
|
||||||
attr_reader :reservation
|
attr_reader :reservation
|
||||||
|
|
||||||
def initialize(reservation, plan) # a plan can be passed to do a simulation (if user didn't have a subscription YET)
|
# a plan can be passed to do a simulation (if user didn't have a subscription YET)
|
||||||
|
def initialize(reservation, plan)
|
||||||
@reservation = reservation
|
@reservation = reservation
|
||||||
@already_updated = false
|
@already_updated = false
|
||||||
@plan = plan
|
@plan = plan
|
||||||
@ -71,8 +73,10 @@ module UsersCredits
|
|||||||
private_constant :Reservation
|
private_constant :Reservation
|
||||||
|
|
||||||
|
|
||||||
class Machine < Reservation # that class is responsible for knowing how to update users_credit of a given user for a given reservation
|
# that class is responsible for knowing how to update users_credit of a given user for a given reservation
|
||||||
def will_use_credits? # to known if a credit will be used in the context of the given reservation
|
class Machine < Reservation
|
||||||
|
# to known if a credit will be used in the context of the given reservation
|
||||||
|
def will_use_credits?
|
||||||
_will_use_credits?[0]
|
_will_use_credits?[0]
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -97,28 +101,30 @@ module UsersCredits
|
|||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def _will_use_credits?
|
|
||||||
return false, 0 unless plan
|
|
||||||
|
|
||||||
if machine_credit = plan.machine_credits.find_by(creditable_id: reservation.reservable_id)
|
def _will_use_credits?
|
||||||
users_credit = user.users_credits.find_by(credit_id: machine_credit.id)
|
return false, 0 unless plan
|
||||||
already_used_hours = users_credit ? users_credit.hours_used : 0
|
|
||||||
|
|
||||||
remaining_hours = machine_credit.hours - already_used_hours
|
if machine_credit = plan.machine_credits.find_by(creditable_id: reservation.reservable_id)
|
||||||
|
users_credit = user.users_credits.find_by(credit_id: machine_credit.id)
|
||||||
|
already_used_hours = users_credit ? users_credit.hours_used : 0
|
||||||
|
|
||||||
free_hours_count = [remaining_hours, reservation.slots.size].min
|
remaining_hours = machine_credit.hours - already_used_hours
|
||||||
|
|
||||||
if free_hours_count > 0
|
free_hours_count = [remaining_hours, reservation.slots.size].min
|
||||||
return true, free_hours_count, machine_credit
|
|
||||||
else
|
if free_hours_count.positive?
|
||||||
return false, free_hours_count, machine_credit
|
return true, free_hours_count, machine_credit
|
||||||
end
|
else
|
||||||
|
return false, free_hours_count, machine_credit
|
||||||
end
|
end
|
||||||
return false, 0
|
|
||||||
end
|
end
|
||||||
|
return false, 0
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class Training < Reservation # same as class Machine but for Training reservation
|
# same as class Machine but for Training reservation
|
||||||
|
class Training < Reservation
|
||||||
def will_use_credits?
|
def will_use_credits?
|
||||||
_will_use_credits?[0]
|
_will_use_credits?[0]
|
||||||
end
|
end
|
||||||
@ -132,18 +138,19 @@ module UsersCredits
|
|||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def _will_use_credits?
|
|
||||||
return false, nil unless plan
|
|
||||||
|
|
||||||
# if there is a training_credit defined for this plan and this training
|
def _will_use_credits?
|
||||||
if training_credit = plan.training_credits.find_by(creditable_id: reservation.reservable_id)
|
return false, nil unless plan
|
||||||
# if user has not used all the plan credits
|
|
||||||
if user.training_credits.where(plan: plan).count < plan.training_credit_nb
|
# if there is a training_credit defined for this plan and this training
|
||||||
return true, training_credit
|
if training_credit = plan.training_credits.find_by(creditable_id: reservation.reservable_id)
|
||||||
end
|
# if user has not used all the plan credits
|
||||||
|
if user.training_credits.where(plan: plan).count < plan.training_credit_nb
|
||||||
|
return true, training_credit
|
||||||
end
|
end
|
||||||
return false, nil
|
|
||||||
end
|
end
|
||||||
|
return false, nil
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class Event < Reservation
|
class Event < Reservation
|
||||||
@ -151,12 +158,12 @@ module UsersCredits
|
|||||||
false
|
false
|
||||||
end
|
end
|
||||||
|
|
||||||
def update_credits
|
def update_credits; end
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
class Space < Reservation
|
class Space < Reservation
|
||||||
def will_use_credits? # to known if a credit will be used in the context of the given reservation
|
# to known if a credit will be used in the context of the given reservation
|
||||||
|
def will_use_credits?
|
||||||
_will_use_credits?[0]
|
_will_use_credits?[0]
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -168,19 +175,20 @@ module UsersCredits
|
|||||||
super
|
super
|
||||||
|
|
||||||
will_use_credits, free_hours_count, space_credit = _will_use_credits?
|
will_use_credits, free_hours_count, space_credit = _will_use_credits?
|
||||||
if will_use_credits
|
return unless will_use_credits
|
||||||
users_credit = user.users_credits.find_or_initialize_by(credit_id: space_credit.id)
|
|
||||||
|
|
||||||
if users_credit.new_record?
|
users_credit = user.users_credits.find_or_initialize_by(credit_id: space_credit.id)
|
||||||
users_credit.hours_used = free_hours_count
|
|
||||||
else
|
if users_credit.new_record?
|
||||||
users_credit.hours_used += free_hours_count
|
users_credit.hours_used = free_hours_count
|
||||||
end
|
else
|
||||||
users_credit.save!
|
users_credit.hours_used += free_hours_count
|
||||||
end
|
end
|
||||||
|
users_credit.save!
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def _will_use_credits?
|
def _will_use_credits?
|
||||||
return false, 0 unless plan
|
return false, 0 unless plan
|
||||||
|
|
||||||
@ -192,7 +200,7 @@ module UsersCredits
|
|||||||
|
|
||||||
free_hours_count = [remaining_hours, reservation.slots.size].min
|
free_hours_count = [remaining_hours, reservation.slots.size].min
|
||||||
|
|
||||||
if free_hours_count > 0
|
if free_hours_count.positive?
|
||||||
return true, free_hours_count, space_credit
|
return true, free_hours_count, space_credit
|
||||||
else
|
else
|
||||||
return false, free_hours_count, space_credit
|
return false, free_hours_count, space_credit
|
||||||
|
@ -10,13 +10,13 @@ class Subscriptions::RenewAsAdminTest < ActionDispatch::IntegrationTest
|
|||||||
user = User.find_by(username: 'kdumas')
|
user = User.find_by(username: 'kdumas')
|
||||||
plan = Plan.find_by(base_name: 'Mensuel tarif réduit')
|
plan = Plan.find_by(base_name: 'Mensuel tarif réduit')
|
||||||
|
|
||||||
VCR.use_cassette("subscriptions_admin_renew_success") do
|
VCR.use_cassette('subscriptions_admin_renew_success') do
|
||||||
post '/api/subscriptions',
|
post '/api/subscriptions',
|
||||||
{
|
{
|
||||||
subscription: {
|
subscription: {
|
||||||
plan_id: plan.id,
|
plan_id: plan.id,
|
||||||
user_id: user.id
|
user_id: user.id
|
||||||
}
|
}
|
||||||
}.to_json, default_headers
|
}.to_json, default_headers
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -39,10 +39,16 @@ class Subscriptions::RenewAsAdminTest < ActionDispatch::IntegrationTest
|
|||||||
|
|
||||||
# Check that the user benefit from prices of his plan
|
# Check that the user benefit from prices of his plan
|
||||||
printer = Machine.find_by(slug: 'imprimante-3d')
|
printer = Machine.find_by(slug: 'imprimante-3d')
|
||||||
assert_equal 10, (printer.prices.find_by(group_id: user.group_id, plan_id: user.subscription.plan_id).amount / 100), 'machine hourly price does not match'
|
assert_equal 10,
|
||||||
|
(printer.prices.find_by(group_id: user.group_id, plan_id: user.subscription.plan_id).amount / 100),
|
||||||
|
'machine hourly price does not match'
|
||||||
|
|
||||||
# Check notification was sent to the user
|
# Check notification was sent to the user
|
||||||
notification = Notification.find_by(notification_type_id: NotificationType.find_by_name('notify_member_subscribed_plan'), attached_object_type: 'Subscription', attached_object_id: subscription[:id])
|
notification = Notification.find_by(
|
||||||
|
notification_type_id: NotificationType.find_by_name('notify_member_subscribed_plan'),
|
||||||
|
attached_object_type: 'Subscription',
|
||||||
|
attached_object_id: subscription[:id]
|
||||||
|
)
|
||||||
assert_not_nil notification, 'user notification was not created'
|
assert_not_nil notification, 'user notification was not created'
|
||||||
assert_equal user.id, notification.receiver_id, 'wrong user notified'
|
assert_equal user.id, notification.receiver_id, 'wrong user notified'
|
||||||
|
|
||||||
@ -50,7 +56,41 @@ class Subscriptions::RenewAsAdminTest < ActionDispatch::IntegrationTest
|
|||||||
invoice = Invoice.find_by(invoiced_type: 'Subscription', invoiced_id: subscription[:id])
|
invoice = Invoice.find_by(invoiced_type: 'Subscription', invoiced_id: subscription[:id])
|
||||||
assert_invoice_pdf invoice
|
assert_invoice_pdf invoice
|
||||||
assert_equal plan.amount, invoice.total, 'Invoice total price does not match the bought subscription'
|
assert_equal plan.amount, invoice.total, 'Invoice total price does not match the bought subscription'
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
test 'admin successfully offer free days' do
|
||||||
|
user = User.find_by(username: 'pdurand')
|
||||||
|
subscription = user.subscription
|
||||||
|
new_date = (1.month.from_now - 4.days).utc
|
||||||
|
|
||||||
|
VCR.use_cassette('subscriptions_admin_offer_free_days') do
|
||||||
|
put "/api/subscriptions/#{subscription.id}",
|
||||||
|
{
|
||||||
|
subscription: {
|
||||||
|
expired_at: new_date.strftime('%Y-%m-%d %H:%M:%S.%9N Z'),
|
||||||
|
free: true
|
||||||
|
}
|
||||||
|
}.to_json, default_headers
|
||||||
|
end
|
||||||
|
|
||||||
|
# Check response format & status
|
||||||
|
assert_equal 200, response.status, response.body
|
||||||
|
assert_equal Mime::JSON, response.content_type
|
||||||
|
|
||||||
|
# Check that the subscribed plan was not altered
|
||||||
|
res_subscription = json_response(response.body)
|
||||||
|
assert_equal subscription.id, res_subscription[:id], 'subscription id has changed'
|
||||||
|
assert_equal subscription.plan_id, res_subscription[:plan_id], 'subscribed plan does not match'
|
||||||
|
assert_dates_equal new_date, res_subscription[:expired_at], 'subscription end date was not updated'
|
||||||
|
|
||||||
|
# Check notification was sent to the user
|
||||||
|
notification = Notification.find_by(
|
||||||
|
notification_type_id: NotificationType.find_by_name('notify_member_subscription_extended'),
|
||||||
|
attached_object_type: 'Subscription',
|
||||||
|
attached_object_id: subscription[:id]
|
||||||
|
)
|
||||||
|
assert_not_nil notification, 'user notification was not created'
|
||||||
|
assert_equal user.id, notification.receiver_id, 'wrong user notified'
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
@ -9,7 +9,7 @@ class Subscriptions::RenewAsUserTest < ActionDispatch::IntegrationTest
|
|||||||
test 'user successfully renew a subscription after it has ended' do
|
test 'user successfully renew a subscription after it has ended' do
|
||||||
plan = Plan.find_by(group_id: @user.group.id, type: 'Plan', base_name: 'Mensuel')
|
plan = Plan.find_by(group_id: @user.group.id, type: 'Plan', base_name: 'Mensuel')
|
||||||
|
|
||||||
VCR.use_cassette("subscriptions_user_renew_success", :erb => true) do
|
VCR.use_cassette('subscriptions_user_renew_success', erb: true) do
|
||||||
post '/api/subscriptions',
|
post '/api/subscriptions',
|
||||||
{
|
{
|
||||||
subscription: {
|
subscription: {
|
||||||
@ -21,7 +21,7 @@ class Subscriptions::RenewAsUserTest < ActionDispatch::IntegrationTest
|
|||||||
end
|
end
|
||||||
|
|
||||||
# Check response format & status
|
# Check response format & status
|
||||||
assert_equal 201, response.status, "API does not return the expected status."+response.body
|
assert_equal 201, response.status, "API does not return the expected status. #{response.body}"
|
||||||
assert_equal Mime::JSON, response.content_type
|
assert_equal Mime::JSON, response.content_type
|
||||||
|
|
||||||
# Check the correct plan was subscribed
|
# Check the correct plan was subscribed
|
||||||
@ -30,16 +30,24 @@ class Subscriptions::RenewAsUserTest < ActionDispatch::IntegrationTest
|
|||||||
|
|
||||||
# Check that the user has the correct subscription
|
# Check that the user has the correct subscription
|
||||||
assert_not_nil @user.subscription, "user's subscription was not found"
|
assert_not_nil @user.subscription, "user's subscription was not found"
|
||||||
assert (@user.subscription.expired_at > DateTime.now), "user's subscription expiration was not updated ... VCR cassettes may be outdated, please check the gitlab wiki"
|
assert (@user.subscription.expired_at > DateTime.now),
|
||||||
assert_in_delta 5, (DateTime.now.to_i - @user.subscription.updated_at.to_i), 10, "user's subscription was not updated recently"
|
"user's subscription expiration was not updated ... VCR cassettes may be outdated, please check the gitlab wiki"
|
||||||
|
assert_in_delta 5,
|
||||||
|
(DateTime.now.to_i - @user.subscription.updated_at.to_i),
|
||||||
|
10,
|
||||||
|
"user's subscription was not updated recently"
|
||||||
|
|
||||||
# Check that the credits were reset correctly
|
# Check that the credits were reset correctly
|
||||||
assert_empty @user.users_credits, 'credits were not reset'
|
assert_empty @user.users_credits, 'credits were not reset'
|
||||||
|
|
||||||
# Check notifications were sent for every admins
|
# Check notifications were sent for every admins
|
||||||
notifications = Notification.where(notification_type_id: NotificationType.find_by_name('notify_admin_subscribed_plan'), attached_object_type: 'Subscription', attached_object_id: subscription[:id])
|
notifications = Notification.where(
|
||||||
|
notification_type_id: NotificationType.find_by_name('notify_admin_subscribed_plan'),
|
||||||
|
attached_object_type: 'Subscription',
|
||||||
|
attached_object_id: subscription[:id]
|
||||||
|
)
|
||||||
assert_not_empty notifications, 'no notifications were created'
|
assert_not_empty notifications, 'no notifications were created'
|
||||||
notified_users_ids = notifications.map {|n| n.receiver_id }
|
notified_users_ids = notifications.map(&:receiver_id)
|
||||||
User.admins.each do |adm|
|
User.admins.each do |adm|
|
||||||
assert_includes notified_users_ids, adm.id, "Admin #{adm.id} was not notified"
|
assert_includes notified_users_ids, adm.id, "Admin #{adm.id} was not notified"
|
||||||
end
|
end
|
||||||
@ -57,23 +65,23 @@ class Subscriptions::RenewAsUserTest < ActionDispatch::IntegrationTest
|
|||||||
|
|
||||||
previous_expiration = @user.subscription.expired_at.to_i
|
previous_expiration = @user.subscription.expired_at.to_i
|
||||||
|
|
||||||
VCR.use_cassette("subscriptions_user_renew_failed") do
|
VCR.use_cassette('subscriptions_user_renew_failed') do
|
||||||
post '/api/subscriptions',
|
post '/api/subscriptions',
|
||||||
{
|
{
|
||||||
subscription: {
|
subscription: {
|
||||||
plan_id: plan.id,
|
plan_id: plan.id,
|
||||||
user_id: @user.id,
|
user_id: @user.id,
|
||||||
card_token: 'invalid_card_token'
|
card_token: 'invalid_card_token'
|
||||||
}
|
}
|
||||||
}.to_json, default_headers
|
}.to_json, default_headers
|
||||||
end
|
end
|
||||||
|
|
||||||
# Check response format & status
|
# Check response format & status
|
||||||
assert_equal 422, response.status, "API does not return the expected status."+response.body
|
assert_equal 422, response.status, "API does not return the expected status. #{response.body}"
|
||||||
assert_equal Mime::JSON, response.content_type
|
assert_equal Mime::JSON, response.content_type
|
||||||
|
|
||||||
# Check the error was handled
|
# Check the error was handled
|
||||||
assert_match /No such token/, response.body
|
assert_match /No such token/, response.body
|
||||||
|
|
||||||
# Check that the user's subscription has not changed
|
# Check that the user's subscription has not changed
|
||||||
assert_equal previous_expiration, @user.subscription.expired_at.to_i, "user's subscription has changed"
|
assert_equal previous_expiration, @user.subscription.expired_at.to_i, "user's subscription has changed"
|
||||||
|
@ -120,6 +120,11 @@ class ActiveSupport::TestCase
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def assert_dates_equal(expected, actual, msg = nil)
|
||||||
|
assert_not_nil actual, msg
|
||||||
|
assert_equal expected.to_date, actual.to_date, msg
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
# Parse a line of text read from a PDF file and return the price included inside
|
# Parse a line of text read from a PDF file and return the price included inside
|
||||||
|
Loading…
x
Reference in New Issue
Block a user