mirror of
https://github.com/marcogll/AnchorOS.git
synced 2026-03-15 21:24:35 +00:00
feat: Implementar sistema de kiosko, enrollment e integración Telegram
## Sistema de Kiosko ✅ - Nuevo rol 'kiosk' en enum user_role - Tabla kiosks con autenticación por API key (64 caracteres) - Funciones SQL: generate_kiosk_api_key(), is_kiosk(), get_available_resources_with_priority() - API Routes: authenticate, bookings (GET/POST), confirm, resources/available, walkin - Componentes UI: BookingConfirmation, WalkInFlow, ResourceAssignment - Página kiosko: /kiosk/[locationId]/page.tsx ## Sistema de Enrollment ✅ - API routes para administración: /api/admin/users, /api/admin/kiosks, /api/admin/locations - Frontend enrollment: /admin/enrollment con autenticación por ADMIN_KEY - Creación de staff (admin, manager, staff, artist) con Supabase Auth - Creación de kiosks con generación automática de API key - Componentes UI: card, button, input, label, select, tabs ## Actualización de Recursos ✅ - Reemplazo de recursos con códigos estándarizados - Estructura por location: 3 mkup, 1 lshs, 4 pedi, 4 mani - Migración de limpieza: elimina duplicados - Total: 12 recursos por location ## Integración Telegram y Scoring ✅ - Campos agregados a staff: telegram_id, email, gmail, google_account, telegram_chat_id - Sistema de scoring: performance_score, total_bookings_completed, total_guarantees_count - Tablas: telegram_notifications, telegram_groups, telegram_bots - Funciones: update_staff_performance_score(), get_top_performers(), get_performance_summary() - Triggers automáticos: notificaciones al crear/confirmar/completar booking - Cálculo de score: base 50 +10 por booking +5 por garantía +1 por $100 ## Actualización de Tipos ✅ - UserRole: agregado 'kiosk' - CustomerTier: agregado 'black', 'VIP' - Nuevas interfaces: Kiosk ## Documentación ✅ - KIOSK_SYSTEM.md: Documentación completa del sistema - KIOSK_IMPLEMENTATION.md: Guía rápida - ENROLLMENT_SYSTEM.md: Sistema de enrollment - RESOURCES_UPDATE.md: Actualización de recursos - PROJECT_UPDATE_JAN_2026.md: Resumen de proyecto ## Componentes UI (7) - button.tsx, card.tsx, input.tsx, label.tsx, select.tsx, tabs.tsx ## Migraciones SQL (4) - 20260116000000_add_kiosk_system.sql - 20260116010000_update_resources.sql - 20260116020000_cleanup_and_fix_resources.sql - 20260116030000_telegram_integration.sql ## Métricas - ~7,500 líneas de código - 32 archivos creados/modificados - 7 componentes UI - 10 API routes - 4 migraciones SQL
This commit is contained in:
660
supabase/migrations/20260116030000_telegram_integration.sql
Normal file
660
supabase/migrations/20260116030000_telegram_integration.sql
Normal file
@@ -0,0 +1,660 @@
|
||||
-- ============================================
|
||||
-- TELEGRAM INTEGRATION Y SCORING SYSTEM
|
||||
-- Agrega campos de Telegram y sistema de métricas
|
||||
-- ============================================
|
||||
|
||||
-- ============================================
|
||||
-- AGREGAR CAMPOS DE TELEGRAM A STAFF
|
||||
-- ============================================
|
||||
|
||||
DO $$
|
||||
BEGIN
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'telegram_id'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN telegram_id BIGINT;
|
||||
|
||||
CREATE INDEX idx_staff_telegram ON staff(telegram_id);
|
||||
|
||||
RAISE NOTICE 'Added telegram_id to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'email'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN email VARCHAR(255);
|
||||
|
||||
CREATE INDEX idx_staff_email ON staff(email);
|
||||
|
||||
RAISE NOTICE 'Added email to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'gmail'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN gmail VARCHAR(255);
|
||||
|
||||
CREATE INDEX idx_staff_gmail ON staff(gmail);
|
||||
|
||||
RAISE NOTICE 'Added gmail to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'google_account'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN google_account VARCHAR(255);
|
||||
|
||||
CREATE INDEX idx_staff_google ON staff(google_account);
|
||||
|
||||
RAISE NOTICE 'Added google_account to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'telegram_chat_id'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN telegram_chat_id BIGINT;
|
||||
|
||||
CREATE INDEX idx_staff_telegram_chat ON staff(telegram_chat_id);
|
||||
|
||||
RAISE NOTICE 'Added telegram_chat_id to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'telegram_notifications_enabled'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN telegram_notifications_enabled BOOLEAN DEFAULT true;
|
||||
|
||||
RAISE NOTICE 'Added telegram_notifications_enabled to staff';
|
||||
END IF;
|
||||
|
||||
END
|
||||
$$;
|
||||
|
||||
-- ============================================
|
||||
-- AGREGAR CAMPOS DE SCORING A STAFF
|
||||
-- ============================================
|
||||
|
||||
DO $$
|
||||
BEGIN
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'total_bookings_completed'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN total_bookings_completed INTEGER DEFAULT 0;
|
||||
|
||||
RAISE NOTICE 'Added total_bookings_completed to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'total_guarantees_count'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN total_guarantees_count INTEGER DEFAULT 0;
|
||||
|
||||
RAISE NOTICE 'Added total_guarantees_count to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'total_guarantees_amount'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN total_guarantees_amount DECIMAL(10,2) DEFAULT 0.00;
|
||||
|
||||
RAISE NOTICE 'Added total_guarantees_amount to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'performance_score'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN performance_score DECIMAL(5,2) DEFAULT 0.00;
|
||||
|
||||
RAISE NOTICE 'Added performance_score to staff';
|
||||
END IF;
|
||||
|
||||
IF NOT EXISTS (
|
||||
SELECT 1 FROM information_schema.columns
|
||||
WHERE table_name = 'staff'
|
||||
AND column_name = 'last_performance_update'
|
||||
) THEN
|
||||
ALTER TABLE staff ADD COLUMN last_performance_update TIMESTAMPTZ;
|
||||
|
||||
RAISE NOTICE 'Added last_performance_update to staff';
|
||||
END IF;
|
||||
END
|
||||
$$;
|
||||
|
||||
-- ============================================
|
||||
-- CREAR TABLA TELEGRAM_NOTIFICATIONS
|
||||
-- ============================================
|
||||
|
||||
CREATE TABLE IF NOT EXISTS telegram_notifications (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
recipient_type VARCHAR(50) NOT NULL, -- 'staff', 'group', 'all'
|
||||
recipient_id UUID, -- ID del staff si recipient_type = 'staff'
|
||||
telegram_chat_id BIGINT NOT NULL, -- Chat ID de Telegram
|
||||
message_type VARCHAR(50) NOT NULL, -- 'booking_created', 'booking_confirmed', 'booking_completed', 'guarantee_processed'
|
||||
message_content TEXT NOT NULL,
|
||||
booking_id UUID, -- Referencia opcional al booking
|
||||
status VARCHAR(20) DEFAULT 'pending', -- 'pending', 'sent', 'failed'
|
||||
error_message TEXT,
|
||||
sent_at TIMESTAMPTZ,
|
||||
created_at TIMESTAMPTZ DEFAULT NOW()
|
||||
);
|
||||
|
||||
-- ============================================
|
||||
-- CREAR TABLA TELEGRAM_GROUPS
|
||||
-- ============================================
|
||||
|
||||
CREATE TABLE IF NOT EXISTS telegram_groups (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
location_id UUID NOT NULL REFERENCES locations(id) ON DELETE CASCADE,
|
||||
group_name VARCHAR(100) NOT NULL,
|
||||
telegram_chat_id BIGINT UNIQUE NOT NULL,
|
||||
group_type VARCHAR(50) NOT NULL, -- 'general', 'artists', 'management', 'alerts'
|
||||
notifications_enabled BOOLEAN DEFAULT true,
|
||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
||||
updated_at TIMESTAMPTZ DEFAULT NOW()
|
||||
);
|
||||
|
||||
-- ============================================
|
||||
-- CREAR TABLA TELEGRAM_BOTS
|
||||
-- ============================================
|
||||
|
||||
CREATE TABLE IF NOT EXISTS telegram_bots (
|
||||
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
bot_name VARCHAR(100) NOT NULL UNIQUE,
|
||||
bot_token VARCHAR(255) NOT NULL UNIQUE,
|
||||
bot_username VARCHAR(100) NOT NULL,
|
||||
is_active BOOLEAN DEFAULT true,
|
||||
created_at TIMESTAMPTZ DEFAULT NOW(),
|
||||
updated_at TIMESTAMPTZ DEFAULT NOW()
|
||||
);
|
||||
|
||||
-- ============================================
|
||||
-- ÍNDICES PARA TABLAS TELEGRAM
|
||||
-- ============================================
|
||||
|
||||
CREATE INDEX IF NOT EXISTS idx_telegram_notifications_recipient ON telegram_notifications(recipient_type, recipient_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_telegram_notifications_status ON telegram_notifications(status, created_at);
|
||||
CREATE INDEX IF NOT EXISTS idx_telegram_notifications_booking ON telegram_notifications(booking_id);
|
||||
|
||||
CREATE INDEX IF NOT EXISTS idx_telegram_groups_location ON telegram_groups(location_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_telegram_groups_type ON telegram_groups(group_type);
|
||||
|
||||
-- ============================================
|
||||
-- FUNCIÓN: ACTUALIZAR SCORING DE STAFF
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION update_staff_performance_score(p_staff_id UUID)
|
||||
RETURNS JSONB AS $$
|
||||
DECLARE
|
||||
v_staff_record RECORD;
|
||||
v_completed_bookings INTEGER;
|
||||
v_guarantees_count INTEGER;
|
||||
v_guarantees_amount DECIMAL(10,2);
|
||||
v_performance_score DECIMAL(5,2);
|
||||
BEGIN
|
||||
-- Obtener datos actuales del staff
|
||||
SELECT * INTO v_staff_record
|
||||
FROM staff
|
||||
WHERE id = p_staff_id;
|
||||
|
||||
-- Contar bookings completados en el último mes
|
||||
SELECT COUNT(*) INTO v_completed_bookings
|
||||
FROM bookings
|
||||
WHERE staff_id = p_staff_id
|
||||
AND status = 'completed'
|
||||
AND created_at >= NOW() - INTERVAL '30 days';
|
||||
|
||||
-- Contar garantías procesadas (simulamos por bookings de servicios con garantía)
|
||||
-- En un sistema real, esto vendría de una tabla de garantías
|
||||
SELECT
|
||||
COUNT(*) INTO v_guarantees_count,
|
||||
COALESCE(SUM(b.total_amount * 0.1), 0) INTO v_guarantees_amount
|
||||
FROM bookings b
|
||||
JOIN services s ON s.id = b.service_id
|
||||
WHERE b.staff_id = p_staff_id
|
||||
AND b.status = 'completed'
|
||||
AND b.created_at >= NOW() - INTERVAL '30 days'
|
||||
AND (s.name ILIKE '%garant%' OR s.description ILIKE '%garant%');
|
||||
|
||||
-- Calcular score de desempeño (base 100)
|
||||
-- +10 por cada booking completado
|
||||
-- +5 por cada garantía procesada
|
||||
-- +1 por cada $100 en garantías
|
||||
v_performance_score := 50.00 +
|
||||
(v_completed_bookings * 10.00) +
|
||||
(v_guarantees_count * 5.00) +
|
||||
(v_guarantees_amount / 100.00 * 1.00);
|
||||
|
||||
-- Limitar score entre 0 y 100
|
||||
v_performance_score := LEAST(v_performance_score, 100.00);
|
||||
v_performance_score := GREATEST(v_performance_score, 0.00);
|
||||
|
||||
-- Actualizar staff
|
||||
UPDATE staff SET
|
||||
total_bookings_completed = v_completed_bookings,
|
||||
total_guarantees_count = v_guarantees_count,
|
||||
total_guarantees_amount = v_guarantees_amount,
|
||||
performance_score = v_performance_score,
|
||||
last_performance_update = NOW()
|
||||
WHERE id = p_staff_id;
|
||||
|
||||
RETURN jsonb_build_object(
|
||||
'staff_id', p_staff_id,
|
||||
'completed_bookings', v_completed_bookings,
|
||||
'guarantees_count', v_guarantees_count,
|
||||
'guarantees_amount', v_guarantees_amount,
|
||||
'performance_score', v_performance_score
|
||||
);
|
||||
END;
|
||||
$$ LANGUAGE plpgsql SECURITY DEFINER;
|
||||
|
||||
-- ============================================
|
||||
-- TRIGGER: ACTUALIZAR SCORING AL COMPLETAR BOOKING
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION trigger_update_staff_performance()
|
||||
RETURNS TRIGGER AS $$
|
||||
BEGIN
|
||||
IF NEW.status = 'completed' AND (OLD.status IS NULL OR OLD.status != 'completed') THEN
|
||||
PERFORM update_staff_performance_score(NEW.staff_id);
|
||||
|
||||
IF NEW.secondary_artist_id IS NOT NULL THEN
|
||||
PERFORM update_staff_performance_score(NEW.secondary_artist_id);
|
||||
END IF;
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
DROP TRIGGER IF EXISTS update_performance_on_booking_complete ON bookings;
|
||||
CREATE TRIGGER update_performance_on_booking_complete
|
||||
AFTER UPDATE OF status ON bookings
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION trigger_update_staff_performance();
|
||||
|
||||
-- ============================================
|
||||
-- FUNCIÓN: ENVIAR NOTIFICACIÓN TELEGRAM
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION create_telegram_notification(
|
||||
p_recipient_type VARCHAR(50),
|
||||
p_recipient_id UUID,
|
||||
p_telegram_chat_id BIGINT,
|
||||
p_message_type VARCHAR(50),
|
||||
p_message_content TEXT,
|
||||
p_booking_id UUID DEFAULT NULL
|
||||
)
|
||||
RETURNS UUID AS $$
|
||||
DECLARE
|
||||
v_notification_id UUID;
|
||||
BEGIN
|
||||
INSERT INTO telegram_notifications (
|
||||
recipient_type,
|
||||
recipient_id,
|
||||
telegram_chat_id,
|
||||
message_type,
|
||||
message_content,
|
||||
booking_id,
|
||||
status
|
||||
)
|
||||
VALUES (
|
||||
p_recipient_type,
|
||||
p_recipient_id,
|
||||
p_telegram_chat_id,
|
||||
p_message_type,
|
||||
p_message_content,
|
||||
p_booking_id,
|
||||
'pending'
|
||||
)
|
||||
RETURNING id INTO v_notification_id;
|
||||
|
||||
RETURN v_notification_id;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql SECURITY DEFINER;
|
||||
|
||||
-- ============================================
|
||||
-- TRIGGER: NOTIFICAR CREACIÓN DE BOOKING
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION notify_booking_created()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
v_staff_telegram_id BIGINT;
|
||||
v_message TEXT;
|
||||
BEGIN
|
||||
-- Solo notificar si el staff tiene telegram configurado
|
||||
SELECT telegram_chat_id INTO v_staff_telegram_id
|
||||
FROM staff
|
||||
WHERE id = NEW.staff_id
|
||||
AND telegram_chat_id IS NOT NULL
|
||||
AND telegram_notifications_enabled = true;
|
||||
|
||||
IF v_staff_telegram_id IS NOT NULL THEN
|
||||
v_message := format('📅 NUEVA CITA ASIGNADA!%sCliente: %s%sServicio: %s%sHora: %s',
|
||||
E'\n',
|
||||
COALESCE((SELECT display_name FROM customers WHERE id = NEW.customer_id), 'Cliente'),
|
||||
E'\n',
|
||||
(SELECT name FROM services WHERE id = NEW.service_id),
|
||||
E'\n',
|
||||
to_char(NEW.start_time_utc, 'DD/MM/YYYY HH24:MI')
|
||||
);
|
||||
|
||||
PERFORM create_telegram_notification(
|
||||
'staff',
|
||||
NEW.staff_id,
|
||||
v_staff_telegram_id,
|
||||
'booking_created',
|
||||
v_message,
|
||||
NEW.id
|
||||
);
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
DROP TRIGGER IF EXISTS notify_booking_created_trigger ON bookings;
|
||||
CREATE TRIGGER notify_booking_created_trigger
|
||||
AFTER INSERT ON bookings
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION notify_booking_created();
|
||||
|
||||
-- ============================================
|
||||
-- TRIGGER: NOTIFICAR CONFIRMACIÓN DE BOOKING
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION notify_booking_confirmed()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
v_staff_telegram_id BIGINT;
|
||||
v_message TEXT;
|
||||
BEGIN
|
||||
IF NEW.status = 'confirmed' AND OLD.status = 'pending' THEN
|
||||
SELECT telegram_chat_id INTO v_staff_telegram_id
|
||||
FROM staff
|
||||
WHERE id = NEW.staff_id
|
||||
AND telegram_chat_id IS NOT NULL
|
||||
AND telegram_notifications_enabled = true;
|
||||
|
||||
IF v_staff_telegram_id IS NOT NULL THEN
|
||||
v_message := format('✅ CITA CONFIRMADA!%sCódigo: %s%sCliente: %s%sHora: %s',
|
||||
E'\n',
|
||||
NEW.short_id,
|
||||
E'\n',
|
||||
COALESCE((SELECT display_name FROM customers WHERE id = NEW.customer_id), 'Cliente'),
|
||||
E'\n',
|
||||
to_char(NEW.start_time_utc, 'DD/MM/YYYY HH24:MI')
|
||||
);
|
||||
|
||||
PERFORM create_telegram_notification(
|
||||
'staff',
|
||||
NEW.staff_id,
|
||||
v_staff_telegram_id,
|
||||
'booking_confirmed',
|
||||
v_message,
|
||||
NEW.id
|
||||
);
|
||||
END IF;
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
DROP TRIGGER IF EXISTS notify_booking_confirmed_trigger ON bookings;
|
||||
CREATE TRIGGER notify_booking_confirmed_trigger
|
||||
AFTER UPDATE OF status ON bookings
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION notify_booking_confirmed();
|
||||
|
||||
-- ============================================
|
||||
-- TRIGGER: NOTIFICAR COMPLETADO DE BOOKING
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION notify_booking_completed()
|
||||
RETURNS TRIGGER AS $$
|
||||
DECLARE
|
||||
v_staff_telegram_id BIGINT;
|
||||
v_message TEXT;
|
||||
v_score_info JSONB;
|
||||
BEGIN
|
||||
IF NEW.status = 'completed' AND (OLD.status IS NULL OR OLD.status != 'completed') THEN
|
||||
SELECT telegram_chat_id INTO v_staff_telegram_id
|
||||
FROM staff
|
||||
WHERE id = NEW.staff_id
|
||||
AND telegram_chat_id IS NOT NULL
|
||||
AND telegram_notifications_enabled = true;
|
||||
|
||||
IF v_staff_telegram_id IS NOT NULL THEN
|
||||
v_message := format('💅 CITA COMPLETADA!%sCódigo: %s%sCliente: %s%sServicio: %s%sTotal: $%s',
|
||||
E'\n',
|
||||
NEW.short_id,
|
||||
E'\n',
|
||||
COALESCE((SELECT display_name FROM customers WHERE id = NEW.customer_id), 'Cliente'),
|
||||
E'\n',
|
||||
(SELECT name FROM services WHERE id = NEW.service_id),
|
||||
E'\n',
|
||||
NEW.total_amount
|
||||
);
|
||||
|
||||
PERFORM create_telegram_notification(
|
||||
'staff',
|
||||
NEW.staff_id,
|
||||
v_staff_telegram_id,
|
||||
'booking_completed',
|
||||
v_message,
|
||||
NEW.id
|
||||
);
|
||||
|
||||
-- Enviar actualización de score
|
||||
v_score_info := update_staff_performance_score(NEW.staff_id);
|
||||
|
||||
-- Mensaje con score
|
||||
IF v_score_info IS NOT NULL THEN
|
||||
v_message := format('📊 TU SCORE ACTUALIZADO!%sBookings completados: %d%sGarantías procesadas: %d ($%.2f)%sScore de desempeño: %.2f%s📈 ¡Sigue así!',
|
||||
E'\n',
|
||||
v_score_info->>'completed_bookings',
|
||||
E'\n',
|
||||
v_score_info->>'guarantees_count',
|
||||
(v_score_info->>'guarantees_amount')::DECIMAL(10,2),
|
||||
E'\n',
|
||||
(v_score_info->>'performance_score')::DECIMAL(5,2),
|
||||
E'\n'
|
||||
);
|
||||
|
||||
PERFORM create_telegram_notification(
|
||||
'staff',
|
||||
NEW.staff_id,
|
||||
v_staff_telegram_id,
|
||||
'performance_update',
|
||||
v_message,
|
||||
NEW.id
|
||||
);
|
||||
END IF;
|
||||
END IF;
|
||||
END IF;
|
||||
|
||||
RETURN NEW;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
DROP TRIGGER IF EXISTS notify_booking_completed_trigger ON bookings;
|
||||
CREATE TRIGGER notify_booking_completed_trigger
|
||||
AFTER UPDATE OF status ON bookings
|
||||
FOR EACH ROW
|
||||
EXECUTE FUNCTION notify_booking_completed();
|
||||
|
||||
-- ============================================
|
||||
-- FUNCIÓN: ENVIAR NOTIFICACIÓN A GRUPO TELEGRAM
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION notify_telegram_group(
|
||||
p_group_id UUID,
|
||||
p_message_type VARCHAR(50),
|
||||
p_message_content TEXT,
|
||||
p_booking_id UUID DEFAULT NULL
|
||||
)
|
||||
RETURNS UUID AS $$
|
||||
DECLARE
|
||||
v_group_record RECORD;
|
||||
v_notification_id UUID;
|
||||
BEGIN
|
||||
SELECT * INTO v_group_record
|
||||
FROM telegram_groups
|
||||
WHERE id = p_group_id
|
||||
AND notifications_enabled = true;
|
||||
|
||||
IF v_group_record.id IS NULL THEN
|
||||
RAISE EXCEPTION 'Telegram group not found or notifications disabled';
|
||||
END IF;
|
||||
|
||||
v_notification_id := create_telegram_notification(
|
||||
'group',
|
||||
NULL,
|
||||
v_group_record.telegram_chat_id,
|
||||
p_message_type,
|
||||
p_message_content,
|
||||
p_booking_id
|
||||
);
|
||||
|
||||
RETURN v_notification_id;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql SECURITY DEFINER;
|
||||
|
||||
-- ============================================
|
||||
-- FUNCIÓN: OBTENER STAFF TOP POR SCORE
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION get_top_performers(p_location_id UUID, p_limit INTEGER DEFAULT 10)
|
||||
RETURNS TABLE (
|
||||
staff_id UUID,
|
||||
display_name VARCHAR,
|
||||
role VARCHAR,
|
||||
performance_score DECIMAL(5,2),
|
||||
total_bookings_completed INTEGER,
|
||||
total_guarantees_count INTEGER,
|
||||
total_guarantees_amount DECIMAL(10,2),
|
||||
last_performance_update TIMESTAMPTZ
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
s.id,
|
||||
s.display_name,
|
||||
s.role,
|
||||
s.performance_score,
|
||||
s.total_bookings_completed,
|
||||
s.total_guarantees_count,
|
||||
s.total_guarantees_amount,
|
||||
s.last_performance_update
|
||||
FROM staff s
|
||||
WHERE s.location_id = p_location_id
|
||||
AND s.is_active = true
|
||||
AND s.role IN ('artist', 'staff', 'manager')
|
||||
ORDER BY s.performance_score DESC NULLS LAST
|
||||
LIMIT p_limit;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql SECURITY DEFINER;
|
||||
|
||||
-- ============================================
|
||||
-- FUNCIÓN: OBTENER RESUMEN DE SCORES
|
||||
-- ============================================
|
||||
|
||||
CREATE OR REPLACE FUNCTION get_performance_summary(p_location_id UUID)
|
||||
RETURNS JSONB AS $$
|
||||
DECLARE
|
||||
v_summary JSONB;
|
||||
BEGIN
|
||||
SELECT jsonb_build_object(
|
||||
'top_performers', jsonb_agg(
|
||||
jsonb_build_object(
|
||||
'staff_id', id,
|
||||
'display_name', display_name,
|
||||
'score', performance_score,
|
||||
'bookings', total_bookings_completed,
|
||||
'guarantees', total_guarantees_count,
|
||||
'guarantees_amount', total_guarantees_amount
|
||||
)
|
||||
),
|
||||
'average_score', AVG(performance_score),
|
||||
'total_bookings', SUM(total_bookings_completed),
|
||||
'total_guarantees', SUM(total_guarantees_count),
|
||||
'total_guarantees_amount', SUM(total_guarantees_amount),
|
||||
'location_id', p_location_id
|
||||
) INTO v_summary
|
||||
FROM staff
|
||||
WHERE location_id = p_location_id
|
||||
AND is_active = true
|
||||
AND role IN ('artist', 'staff', 'manager');
|
||||
|
||||
RETURN v_summary;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql SECURITY DEFINER;
|
||||
|
||||
-- ============================================
|
||||
-- VERIFICACIÓN
|
||||
-- ============================================
|
||||
|
||||
DO $$
|
||||
BEGIN
|
||||
RAISE NOTICE '==========================================';
|
||||
RAISE NOTICE 'TELEGRAM INTEGRATION COMPLETED';
|
||||
RAISE NOTICE '==========================================';
|
||||
RAISE NOTICE '✅ Campos agregados a staff:';
|
||||
RAISE NOTICE ' - telegram_id';
|
||||
RAISE NOTICE ' - email';
|
||||
RAISE NOTICE ' - gmail';
|
||||
RAISE NOTICE ' - google_account';
|
||||
RAISE NOTICE ' - telegram_chat_id';
|
||||
RAISE NOTICE ' - telegram_notifications_enabled';
|
||||
RAISE NOTICE ' - total_bookings_completed';
|
||||
RAISE NOTICE ' - total_guarantees_count';
|
||||
RAISE NOTICE ' - total_guarantees_amount';
|
||||
RAISE NOTICE ' - performance_score';
|
||||
RAISE NOTICE ' - last_performance_update';
|
||||
RAISE NOTICE '==========================================';
|
||||
RAISE NOTICE '✅ Nuevas tablas creadas:';
|
||||
RAISE NOTICE ' - telegram_notifications';
|
||||
RAISE NOTICE ' - telegram_groups';
|
||||
RAISE NOTICE ' - telegram_bots';
|
||||
RAISE NOTICE '==========================================';
|
||||
RAISE NOTICE '✅ Funciones de scoring creadas:';
|
||||
RAISE NOTICE ' - update_staff_performance_score()';
|
||||
RAISE NOTICE ' - get_top_performers()';
|
||||
RAISE NOTICE ' - get_performance_summary()';
|
||||
RAISE NOTICE '==========================================';
|
||||
RAISE NOTICE '✅ Triggers automáticos:';
|
||||
RAISE NOTICE ' - Notificar al crear booking';
|
||||
RAISE NOTICE ' - Notificar al confirmar booking';
|
||||
RAISE NOTICE ' - Notificar al completar booking';
|
||||
RAISE NOTICE ' - Actualizar score al completar booking';
|
||||
RAISE NOTICE '==========================================';
|
||||
RAISE NOTICE 'PRÓXIMOS PASOS:';
|
||||
RAISE NOTICE '1. Crear bot de Telegram';
|
||||
RAISE NOTICE '2. Configurar webhook del bot';
|
||||
RAISE NOTICE '3. Agregar grupos de Telegram';
|
||||
RAISE NOTICE '4. Asignar chat IDs a staff';
|
||||
RAISE NOTICE '5. Implementar API de envío de mensajes';
|
||||
RAISE NOTICE '==========================================';
|
||||
END
|
||||
$$;
|
||||
Reference in New Issue
Block a user