Не удалось сохранить значения карты после изменения версии в Rails 4.2

У меня есть страница регистрации пользователя, где я буду вводить личные данные вместе с информацией о кредитной карте. Аккаунты пользователя has_many и аккаунт принадлежат пользователю. Когда пользователь создается, учетная запись создается, а информация и адрес кредитной карты сохраняются в таблице подписки. После создания аккаунта он собирается использовать метод store_card в модели subscription.rb. Оно работает нормально, когда я запускаю приложение в Rails 3.2.13, и когда я перенес его на Rails 4.2.6, оно не хранит информацию о подписке и не отображает никаких ошибок. Я обновил гем activemerchant до 1.47.0 и изменил код в методе load_billing в контроллере ниже:

@creditcard = ActiveMerchant::Billing::CreditCard.new(params[:account].blank? ? nil : params[:account][:creditcard])

в

@creditcard = ActiveMerchant::Billing::CreditCard.new(params[:account].blank? ? {} : params[:account][:creditcard])

и так же @address тоже.

Нет конкретного отношения, но только эта строка добавлена ​​в account.rb:

has_subscription :user_limit => Proc.new {|a| a.users.count }

Когда я попытался запустить с консоли:

u = Account.find(1)
u.subscription 

он генерирует подписные столбцы.

class AccountsController < ApplicationController

  before_filter :build_account, :only => [:new, :create]
  before_filter :build_user, :only => [:new, :create]
  before_filter :load_billing, :only => [:new, :create, :billing]
  def create
    @address.first_name = @creditcard.first_name
    @address.last_name = @creditcard.last_name
    @account.address = @address
    @account.creditcard = @creditcard
    if @account.new_record?
      if @account.save
        flash[:notice] = 'Account was created.'
        bypass_sign_in(@user)
        redirect_to session[:previous_url] || user_reports_path(@user)
      else
        render :action => 'new'
      end
    else
      @user.account_id = @account.id
      if @user.save
        flash[:notice] = 'User was created.'
        bypass_sign_in(@user)
        redirect_to session[:previous_url] || user_reports_path(@user)
      else
        render :action => 'new'
      end
    end
  end

  def billing
    @user = current_user
    @account = Account.find(params[:id])
    if request.put?
      @address.first_name = @creditcard.first_name
      @address.last_name = @creditcard.last_name
      puts @address.first_name
      if @creditcard.valid? & @address.valid?
       if @subscription.store_card(@creditcard, :billing_address => @address.to_activemerchant, :ip => request.remote_ip)
         flash[:notice] = "Your billing information has been updated."
         redirect_to settings_path(@user)
       end
      end
    end
  end
  protected

  def resource
    @account ||= current_account
  end

  def build_account
    @account = params[:account_name].blank? ? Account.new : Account.find_by_name(params[:account_name])
  end

  def build_user
    @account.user = @user = User.new(params[:account].blank? ? nil : params[:account][:user])
  end

  def load_billing
    @creditcard = ActiveMerchant::Billing::CreditCard.new(params[:account].blank? ? {} : params[:account][:creditcard])
    @address = SubscriptionAddress.new(params[:account].blank? ? {} : params[:account][:address])
  end

end

subscription.rb:

class Subscription < ActiveRecord::Base

  attr_accessor :creditcard, :address
  def store_card(creditcard, gw_options = {})
    # Clear out payment info if switching to CC from PayPal
    destroy_gateway_record(paypal) if paypal?

    @response = if billing_id.blank?
      gateway.store(creditcard, gw_options)
    else
      gateway.update(billing_id, creditcard, gw_options)
    end

    if @response.success?
      if active_card = @response.params['active_card']
        # Stripe token-based response
        self.card_number = "XXXX-XXXX-XXXX-#{active_card['last4']}"
        self.card_expiration = "%02d-%d" % [active_card['exp_month'], active_card['exp_year']]
      else
        self.card_number = creditcard.display_number
        self.card_expiration = "%02d-%d" % [creditcard.expiry_date.month, creditcard.expiry_date.year]
      end
      set_billing
    else
      errors.add(:base, @response.message)
      false
    end
  end

  def card_storage
      self.store_card(@creditcard, :billing_address => @address.to_activemerchant) if @creditcard && @address && card_number.blank?
    end

  def set_billing
      self.billing_id = @response.token
    end

end

account.rb:

class Account < ActiveRecord::Base

validate :valid_subscription?, :on => :create

  def valid_subscription?
    puts "valid_subscription**************************"
    return if errors.any? # Don't bother with a subscription if there are errors already
    self.build_subscription(:plan => @plan, :next_renewal_at => @plan_start, :creditcard => @creditcard, :address => @address)
    if !subscription.valid?
      errors.add(:base, "Error with payment: #{subscription.errors.full_messages.to_sentence}")
      return false
    end
  end
end

Пожалуйста, помогите, нужно ли мне использовать сильные параметры и каким-либо другим способом?

1 ответ

Было бы хорошо увидеть Account модель, но, исходя из вашего кода, похоже, что вы никогда не сохраняете данные кредитной карты. метод store_card просто присваивает значения, но никогда не сохраняет их в базе данных. Самое простое решение будет просто позвонить self.save в вашем store_card метод.

 if active_card = @response.params['active_card']
    # Stripe token-based response
    self.card_number = "XXXX-XXXX-XXXX-#{active_card['last4']}"
    self.card_expiration = "%02d-%d" % [active_card['exp_month'], active_card['exp_year']]
  else
    self.card_number = creditcard.display_number
    self.card_expiration = "%02d-%d" % [creditcard.expiry_date.month, creditcard.expiry_date.year]
  end
  self.save
Другие вопросы по тегам