Files
aperonight/app/models/ticket.rb
kbe 3c1e17c2af feat(payouts): implement promoter earnings viewing, request flow, and admin Stripe processing with webhooks
Add model methods for accurate net calculations (€0.50 + 1.5% fees), eligibility, refund handling
Update promoter/payouts controller for index (pending events), create (eligibility checks)
Integrate admin processing via Stripe::Transfer, webhook for status sync
Enhance views: index pending cards, events/show preview/form
Add comprehensive tests (models, controllers, service, integration); run migrations
2025-09-17 02:07:52 +02:00

87 lines
2.1 KiB
Ruby
Executable File

class Ticket < ApplicationRecord
# === Associations ===
belongs_to :order
belongs_to :ticket_type
has_one :event, through: :ticket_type
has_one :user, through: :order
# === Validations ===
validates :qr_code, presence: true, uniqueness: true
validates :order_id, presence: true
validates :ticket_type_id, presence: true
validates :price_cents, presence: true, numericality: { greater_than_or_equal_to: 0 }
validates :status, presence: true, inclusion: { in: %w[draft active used expired refunded] }
validates :first_name, presence: true
validates :last_name, presence: true
# === Scopes ===
scope :draft, -> { where(status: "draft") }
scope :active, -> { where(status: "active") }
scope :expired_drafts, -> { joins(:order).where(status: "draft").where("orders.expires_at < ?", Time.current) }
before_validation :set_price_from_ticket_type, on: :create
before_validation :generate_qr_code, on: :create
after_update :recalculate_earning_if_refunded, if: :saved_change_to_status?
# Generate PDF ticket
def to_pdf
TicketPdfGenerator.new(self).generate
end
# Price in euros (formatted)
def price_euros
price_cents / 100.0
end
# Delegate payment methods to order
def can_retry_payment?
order.can_retry_payment?
end
def expired?
order.expired?
end
def expiring_soon?
order.expiring_soon?
end
# Mark ticket as expired if it's past expiry time
def expire_if_overdue!
order.expire_if_overdue!
end
private
def set_price_from_ticket_type
return unless ticket_type
self.price_cents = ticket_type.price_cents
end
def generate_qr_code
return if qr_code.present?
loop do
self.qr_code = SecureRandom.uuid
break unless Ticket.exists?(qr_code: qr_code)
end
rescue => e
Rails.logger.error "Failed to generate QR code for ticket: #{e.message}"
# Generate a simple fallback QR code
self.qr_code = "#{id || 'temp'}-#{Time.current.to_i}-#{SecureRandom.hex(4)}"
end
def draft?
status == "draft"
end
private
def recalculate_earning_if_refunded
if status == "refunded"
order.earning&.recalculate!
end
end
end