9

Buduję mały dowód koncepcji z Stripe i Ruby na Railsach 3.2. Do tej pory oglądałem Railscast, jak zaimplementować Stripe w aplikacji RoR i działa naprawdę dobrze.Gdzie i jak obchodzić się z wyjątkami Stripe?

Mam zbudowany mojej aplikacji, wykonując RailsCast #288 Billing with Stripe. Teraz moi użytkownicy mogą dodawać i edytować swoje karty kredytowe, a nawet rejestrować się na zajęciach i obciążać swoją kartę kredytową po jej zakończeniu.

Teraz już testuje z paskiem licznych test credit cards i chcę złapać jak najwięcej wyjątki, gdy podniesiona. Używam paskiem za example błędy w moim modelu Rejestracji jako pokaz tutaj:

class Registration < ActiveRecord::Base 

    belongs_to :user 
    belongs_to :session 

    attr_accessible :session_id, :user_id, :session, :user, :stripe_payment_id 
    validates :user_id, :uniqueness => {:scope => :session_id} 

    def save_with_payment(user, stripe_card_token) 
    if valid? 
     if user.stripe_customer_id.present? 
     charge = Stripe::Charge.create(
      :customer => user.stripe_customer_id, 
      :amount => self.session.price.to_i * 100, 
      :description => "Registration for #{self.session.name} (Id:#{self.session.id})", 
      :currency => 'cad' 
     ) 
     else 
     customer = Stripe::Customer.create(
      :email => user.email, 
      :card => stripe_card_token, 
      :description => user.name 
     ) 
     charge = Stripe::Charge.create(
      :customer => customer.id, 
      :amount => self.session.price.to_i * 100, 
      :description => "Registration for #{self.session.name} (Id:#{self.session.id})", 
      :currency => 'cad' 
     ) 
     user.update_attribute(:stripe_customer_id, customer.id) 
     end 
     self.stripe_payment_id = charge.id 
     save! 
    end 
    rescue Stripe::CardError => e 
    body = e.json_body 
    err = body[:error] 
    logger.debug "Status is: #{e.http_status}" 
    logger.debug "Type is: #{err[:type]}" 
    logger.debug "Code is: #{err[:code]}" 
    logger.debug "Param is: #{err[:param]}" 
    logger.debug "Message is: #{err[:message]}" 
    rescue Stripe::InvalidRequestError => e 
    # Invalid parameters were supplied to Stripe's API 
    rescue Stripe::AuthenticationError => e 
    # Authentication with Stripe's API failed 
    # (maybe you changed API keys recently) 
    rescue Stripe::APIConnectionError => e 
    # Network communication with Stripe failed 
    rescue Stripe::StripeError => e 
    # Display a very generic error to the user, and maybe send 
    # yourself an email 
    rescue => e 
    # Something else happened, completely unrelated to Stripe 
    end 
end 

mam tylko ratowanie od błędów już teraz, a nie naprawdę podejmowania działań po jednej podniesione i ostatecznie chciałbym zatrzymać bieżącą klasę rejestracja od zdarzenia i przekierowanie użytkownika z błędem flash.

Przeczytałem o rescure_from, ale nie jestem pewien, jaki jest najlepszy sposób na obsługę wszystkich możliwych błędów pasków. Wiem, że nie można przekierować z modelu, w jaki sposób eksperci sobie z tym poradzą?

Oto mój kontroler Rejestracja:

class Classroom::RegistrationsController < ApplicationController 
    before_filter :authenticate_user! 

    def new 
    if params[:session_id] 
     @session = Session.find(params[:session_id]) 
     @registration = Registration.new(user: current_user, session: @session) 
    else 
     flash[:error] = "Course session is required" 
    end 

    rescue ActiveRecord::RecordNotFound 
     render file: 'public/404', status: :not_found 

    end 

    def create 
    if params[:session_id] 
     @session = Session.find(params[:session_id]) 
     @registration = Registration.new(user: current_user, session: @session) 
     if @registration.save_with_payment(current_user, params[:stripe_card_token]) 
     flash[:notice] = "Course registration saved with success." 
     logger.debug "Course registration saved with success." 
     mixpanel.track 'Registered to a session', { :distinct_id => current_user.id, 
              :id => @session.id, 
              'Name' => @session.name, 
              'Description' => @session.description, 
              'Course' => @session.course.name 
     } 
     mixpanel.increment current_user.id, { :'Sessions Registered' => 1} 
     mixpanel.track_charge(current_user.id, @session.price.to_i) 
     else 
     flash[:error] = "There was a problem saving the registration." 
     logger.debug "There was a problem saving the registration." 
     end 
     redirect_to root_path 
    else 
     flash[:error] = "Session required." 
     redirect_to root_path 
    end 
    end 

end 

Dzięki za poświęcenie czasu, aby odpowiedzieć, mile widziane!

Francis

Odpowiedz

10

Czy myślałeś o umieszczenie faktycznie Stripe połączenia w niestandardowy walidator?

http://apidock.com/rails/ActiveModel/Validations/ClassMethods/validate

W ten sposób można dodać błędy do obiektu z czymś następującym

Logika to ty tylko chcesz zapisać udanych transakcji jako „transakcji” tak więc dlaczego nie wystarczy umieścić ładunek pasków w walidatorze.

validate :card_validation 

def card_validation 

    begin 
     charge = Stripe::Charge.create(
      :customer => user.stripe_customer_id, 
      :amount => self.session.price.to_i * 100, 
      :description => "Registration for #{self.session.name} (Id:#{self.session.id})", 
      :currency => 'cad' 
     ) 
     etc etc 
    rescue => e 
     errors.add(:credit_card, e.message) 
     #Then you might have a model to log the transaction error. 
     Error.create(charge, customer) 
    end 

end 

ten sposób można obsłużyć błędy jak każdy inny błędów będzie można uzyskać z wpisem nie oszczędności, zamiast dać pustą wiadomość o błędzie lub konieczności obsłużyć każdy ostatni błąd z paskiem.

class Classroom::RegistrationsController < ApplicationController 
    before_filter :authenticate_user! 

    def create 
    if params[:session_id] 
     @session = Session.find(params[:session_id]) 

     params[:registration][:user] = current_user 
     params[:registration][:session] = @session 
     params[:registration][:stripe_card_token] = params[:stripe_card_token] 

     @registration = Registration.new(params[:registration]) 
     respond_with(@registration) do |format| 
     if @registration.save 
      format.html {redirect_to root_path, :notice => "SOMETHING HERE TO TELL THEM SUC"} 
     else 
      format.html {render} 
     end 
     end 
    else 
     respond_with do |format| 
     format.html {redirect_to root_path, :error => "SOMETHING HERE TO TELL THEM GET SESSION"} 
     end 
    end 
    end 

end 
+0

Dzięki za twój wkład, nigdy wcześniej nie słyszałem o custom_validator (dopiero zaczynałem od szyn). Więc jeśli dobrze rozumiem, powinienem umieścić całą logikę rozliczeniową Stripe w metodzie w moim modelu rejestracyjnym? Co dzieje się z kontrolerem rejestracji? Jak mogę przekierować użytkownika z błędem Flasha? Niestety dla wielu pytań, wciąż mylić :) –

+1

Updated odpowiedzieć na jego 100% dokładny, ale generalnie idea – rovermicrover

+0

Myślę, że mam to, za każdym razem jest to model Rejestracja zapisywane w bazie danych, validate: card_validation zostanie wywołana (nieco jak przed filtrem) i zapisywać tylko prawidłowe transakcje/karty. Dzięki za Twój wkład !!! –