1
0
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:
Sylvain 2018-12-06 18:26:01 +01:00
parent b537571656
commit bef3118649
11 changed files with 319 additions and 256 deletions

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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)

View File

@ -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

View File

@ -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

View 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

View File

@ -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

View File

@ -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

View File

@ -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"

View File

@ -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