feat: improve seo urls?

This commit is contained in:
kbe
2025-09-06 01:44:48 +02:00
parent fa99a167a5
commit 5105964b39
22 changed files with 702 additions and 367 deletions

View File

@@ -1,11 +1,11 @@
# Handle order management and checkout process
# Handle order management and checkout process with SEO-friendly URLs
#
# This controller manages the order lifecycle from checkout to payment completion
# Orders group multiple tickets together for better transaction management
class OrdersController < ApplicationController
before_action :authenticate_user!
before_action :set_order, only: [ :show, :checkout, :retry_payment, :increment_payment_attempt ]
before_action :set_event, only: [ :new, :create ]
before_action :set_event_from_seo_params, only: [:new, :create, :checkout]
before_action :set_order_from_id, only: [:show, :retry_payment, :increment_payment_attempt]
# Display new order form with name collection
#
@@ -15,7 +15,7 @@ class OrdersController < ApplicationController
@cart_data = params[:cart_data] || session[:pending_cart] || {}
if @cart_data.empty?
redirect_to event_path(@event.slug, @event), alert: "Veuillez d'abord sélectionner vos billets sur la page de l'événement"
redirect_to seo_event_path(@event), alert: "Veuillez d'abord sélectionner vos billets sur la page de l'événement"
return
end
@@ -47,7 +47,7 @@ class OrdersController < ApplicationController
@cart_data = params[:cart_data] || session[:pending_cart] || {}
if @cart_data.empty?
redirect_to event_path(@event.slug, @event), alert: "Aucun billet sélectionné"
redirect_to seo_event_path(@event), alert: "Aucun billet sélectionné"
return
end
@@ -87,32 +87,44 @@ class OrdersController < ApplicationController
if success
session[:draft_order_id] = @order.id
session.delete(:pending_cart)
redirect_to checkout_order_path(@order)
year = @event.start_time.year
month = format("%02d", @event.start_time.month)
redirect_to event_checkout_path(year: year, month: month, slug: @event.slug)
else
redirect_to event_order_new_path(@event.slug, @event.id)
year = @event.start_time.year
month = format("%02d", @event.start_time.month)
redirect_to book_event_tickets_path(year: year, month: month, slug: @event.slug)
end
rescue => e
error_message = e.message.present? ? e.message : "Erreur inconnue"
flash[:alert] = "Une erreur est survenue: #{error_message}"
redirect_to event_order_new_path(@event.slug, @event.id)
year = @event.start_time.year
month = format("%02d", @event.start_time.month)
redirect_to book_event_tickets_path(year: year, month: month, slug: @event.slug)
end
# Display order summary
#
#
def show
@tickets = @order.tickets.includes(:ticket_type)
end
# Display payment page for an order
# Display payment page for an order (SEO-friendly checkout URL)
#
# Display a summary of all tickets in the order and permit user
# to proceed to payment via Stripe
def checkout
# Find order from session or create one
@order = current_user.orders.find_by(id: session[:draft_order_id], event: @event, status: "draft")
unless @order
redirect_to seo_event_path(@event), alert: "Aucune commande en attente trouvée"
return
end
# Handle expired orders
if @order.expired?
@order.expire_if_overdue!
return redirect_to event_path(@order.event.slug, @order.event),
return redirect_to seo_event_path(@event),
alert: "Votre commande a expiré. Veuillez recommencer."
end
@@ -141,117 +153,41 @@ class OrdersController < ApplicationController
# Allow users to retry payment for failed/cancelled payments
def retry_payment
unless @order.can_retry_payment?
redirect_to event_path(@order.event.slug, @order.event),
redirect_to seo_event_path(@order.event),
alert: "Cette commande ne peut plus être payée"
return
end
redirect_to checkout_order_path(@order)
end
# Handle successful payment
def payment_success
session_id = params[:session_id]
# Check if Stripe is properly configured
stripe_configured = Rails.application.config.stripe[:secret_key].present?
Rails.logger.debug "Payment success - Stripe configured: #{stripe_configured}"
unless stripe_configured
redirect_to root_path, alert: "Le système de paiement n'est pas correctement configuré. Veuillez contacter l'administrateur."
return
end
begin
stripe_session = Stripe::Checkout::Session.retrieve(session_id)
if stripe_session.payment_status == "paid"
# Get order_id from session metadata
order_id = stripe_session.metadata["order_id"]
unless order_id.present?
redirect_to dashboard_path, alert: "Informations de commande manquantes"
return
end
# Find and update the order
@order = current_user.orders.includes(tickets: :ticket_type).find(order_id)
@order.mark_as_paid!
# Schedule Stripe invoice generation in background
# This creates accounting records without blocking the payment success flow
begin
StripeInvoiceGenerationJob.perform_later(@order.id)
Rails.logger.info "Scheduled Stripe invoice generation for order #{@order.id}"
rescue => e
Rails.logger.error "Failed to schedule invoice generation for order #{@order.id}: #{e.message}"
# Don't fail the payment process due to job scheduling issues
end
# Send confirmation emails
@order.tickets.each do |ticket|
begin
TicketMailer.purchase_confirmation(ticket).deliver_now
rescue => e
Rails.logger.error "Failed to send confirmation email for ticket #{ticket.id}: #{e.message}"
# Don't fail the entire payment process due to email/PDF generation issues
end
end
# Clear session data
session.delete(:pending_cart)
session.delete(:ticket_names)
session.delete(:draft_order_id)
render "payment_success"
else
redirect_to dashboard_path, alert: "Le paiement n'a pas été complété avec succès"
end
rescue Stripe::StripeError => e
error_message = e.message.present? ? e.message : "Erreur Stripe inconnue"
redirect_to dashboard_path, alert: "Erreur lors du traitement de votre confirmation de paiement : #{error_message}"
rescue => e
error_message = e.message.present? ? e.message : "Erreur inconnue"
Rails.logger.error "Payment success error: #{e.class} - #{error_message}"
redirect_to dashboard_path, alert: "Une erreur inattendue s'est produite : #{error_message}"
end
end
# Handle payment failure/cancellation
def payment_cancel
order_id = params[:order_id] || session[:draft_order_id]
if order_id.present?
order = current_user.orders.find_by(id: order_id, status: "draft")
if order&.can_retry_payment?
redirect_to checkout_order_path(order),
alert: "Le paiement a été annulé. Vous pouvez réessayer."
else
session.delete(:draft_order_id)
redirect_to root_path, alert: "Le paiement a été annulé et votre commande a expiré."
end
else
redirect_to root_path, alert: "Le paiement a été annulé"
end
year = @order.event.start_time.year
month = format("%02d", @order.event.start_time.month)
redirect_to event_checkout_path(year: year, month: month, slug: @order.event.slug)
end
private
def set_order
@order = current_user.orders.includes(:tickets, :event).find(params[:id])
def set_event_from_seo_params
year = params[:year].to_i
month = params[:month].to_i
start_of_month = Date.new(year, month, 1).beginning_of_month
end_of_month = start_of_month.end_of_month
@event = Event.includes(:ticket_types)
.where(slug: params[:slug])
.where(start_time: start_of_month..end_of_month)
.first
return redirect_to events_path, alert: "Événement non trouvé" unless @event
end
def set_order_from_id
@order = current_user.orders.includes(:tickets, :event).find(params[:order_id])
@event = @order.event
rescue ActiveRecord::RecordNotFound
redirect_to root_path, alert: "Commande non trouvée"
end
def set_event
@event = Event.includes(:ticket_types).find(params[:id])
rescue ActiveRecord::RecordNotFound
redirect_to events_path, alert: "Événement non trouvé"
end
def order_params
params.permit(tickets_attributes: [ :ticket_type_id, :first_name, :last_name ])
params.permit(tickets_attributes: [:ticket_type_id, :first_name, :last_name])
end
def create_stripe_session
@@ -270,15 +206,23 @@ class OrdersController < ApplicationController
end
Stripe::Checkout::Session.create(
payment_method_types: [ "card" ],
payment_method_types: ["card"],
line_items: line_items,
mode: "payment",
success_url: order_payment_success_url + "?session_id={CHECKOUT_SESSION_ID}",
cancel_url: order_payment_cancel_url,
success_url: booking_payment_success_url + "?session_id={CHECKOUT_SESSION_ID}",
cancel_url: booking_payment_cancelled_url + "?order_id=#{@order.id}",
metadata: {
order_id: @order.id,
user_id: current_user.id
}
)
end
end
# Generate SEO-friendly path for an event
def seo_event_path(event)
year = event.start_time.year
month = format("%02d", event.start_time.month)
event_path(year: year, month: month, slug: event.slug)
end
helper_method :seo_event_path
end