Consolida o trabalho desta branch de abril/2026 em um bloco pronto pra testar em staging antes do merge pra main. ## Correções de memória semântica - ExtractionService: Princípio Zero + Regra de Ouro (ação consumada vs intenção). - Cenário Daniela_Reservas: Passo 0 de classificação (consulta/intenção/fora). ## Roleta da Sorte (end-to-end) - Schema Supabase + 7 RPCs atômicas (server-side, idempotentes). - Services: Offer, Redeem, WeeklyReport. - Jobs: OfferRouletteJob (hook em ConfirmationService após Pix pago), NotifyRevealed + Scheduler de fallback. - Tool manual GenerateRoletaLinkTool + endpoint público /roleta/notify. - Dashboard /captain/roleta com Resgate + Relatório + anomaly detection. ## Cenário Reclamacoes_Ouvidoria - Triagem P1-P4, framework LAST, Three-level listening, Self-check. - Sem compensação material, detecção de cliente frustrado eleva prioridade. ## Analytics - Funil de conversão /captain/funnel: 5 etapas via regex, zero LLM. - Detector de churn via ChurnOutreach* (cron dias úteis 10h-17h BRT). ## Trabalho pré-existente incluído - Captain Executive Reports (ceo_digest, mattermost_delivery). - get_reserva_preco_tool, Lifecycle ajustes, Reservations UI polimentos. ## Outros - .gitignore: patterns pra credenciais. - Migrations de scenarios idempotentes. - i18n completa pt_BR+en pra roleta/funnel. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
62 lines
1.7 KiB
Ruby
62 lines
1.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
# Cron de fallback (a cada 5 min): enfileira NotifyRevealedJob pra qualquer
|
|
# draw que foi revelado há >60s e ainda não foi notificado. Cobre cenários onde
|
|
# o front não conseguiu chamar o endpoint (browser fechado, rede caiu).
|
|
class Captain::Roleta::NotifyRevealedSchedulerJob < ApplicationJob
|
|
DEFAULT_SCHEMA = 'reserva_hotel'
|
|
|
|
queue_as :scheduled_jobs
|
|
|
|
def perform
|
|
pending = fetch_pending_tokens
|
|
return if pending.blank?
|
|
|
|
Rails.logger.info "[NotifyRevealedScheduler] enfileirando #{pending.size} drawn(s)"
|
|
pending.each { |row| Captain::Roleta::NotifyRevealedJob.perform_later(row['token']) }
|
|
end
|
|
|
|
private
|
|
|
|
def fetch_pending_tokens
|
|
body = {
|
|
:select => 'token',
|
|
:status => 'eq.revealed',
|
|
:notified_at => 'is.null',
|
|
'revealed_at' => "lt.#{1.minute.ago.iso8601}",
|
|
:limit => 100
|
|
}
|
|
supabase_get('roulette_draws', body)
|
|
rescue StandardError => e
|
|
Rails.logger.warn("[NotifyRevealedScheduler] falha: #{e.class} - #{e.message}")
|
|
[]
|
|
end
|
|
|
|
def supabase_get(table, query)
|
|
url = "#{supabase_url}/rest/v1/#{table}"
|
|
response = Faraday.get(url, query) do |req|
|
|
req.headers['apikey'] = supabase_key
|
|
req.headers['Authorization'] = "Bearer #{supabase_key}"
|
|
req.headers['Accept-Profile'] = supabase_schema
|
|
req.headers['Accept'] = 'application/json'
|
|
end
|
|
return [] unless response.success?
|
|
|
|
JSON.parse(response.body)
|
|
rescue JSON::ParserError
|
|
[]
|
|
end
|
|
|
|
def supabase_url
|
|
ENV.fetch('RESERVA_1001_SUPABASE_URL').chomp('/')
|
|
end
|
|
|
|
def supabase_key
|
|
ENV.fetch('RESERVA_1001_SUPABASE_ANON_KEY')
|
|
end
|
|
|
|
def supabase_schema
|
|
ENV.fetch('RESERVA_1001_SUPABASE_SCHEMA', DEFAULT_SCHEMA)
|
|
end
|
|
end
|