-- ===================================================== -- SCHEMA: sales -- PROPÓSITO: Gestión de ventas, cotizaciones, clientes -- MÓDULOS: MGN-007 (Ventas Básico) -- FECHA: 2025-11-24 -- ===================================================== -- Crear schema CREATE SCHEMA IF NOT EXISTS sales; -- ===================================================== -- TYPES (ENUMs) -- ===================================================== CREATE TYPE sales.order_status AS ENUM ( 'draft', 'sent', 'sale', 'done', 'cancelled' ); CREATE TYPE sales.quotation_status AS ENUM ( 'draft', 'sent', 'approved', 'rejected', 'converted', 'expired' ); CREATE TYPE sales.invoice_policy AS ENUM ( 'order', 'delivery' ); CREATE TYPE sales.delivery_status AS ENUM ( 'pending', 'partial', 'delivered' ); CREATE TYPE sales.invoice_status AS ENUM ( 'pending', 'partial', 'invoiced' ); -- ===================================================== -- TABLES -- ===================================================== -- Tabla: sales_orders (Órdenes de venta) CREATE TABLE sales.sales_orders ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, company_id UUID NOT NULL REFERENCES auth.companies(id) ON DELETE CASCADE, -- Numeración name VARCHAR(100) NOT NULL, client_order_ref VARCHAR(100), -- Referencia del cliente -- Cliente partner_id UUID NOT NULL REFERENCES core.partners(id), -- Fechas order_date DATE NOT NULL, validity_date DATE, commitment_date DATE, -- Configuración currency_id UUID NOT NULL REFERENCES core.currencies(id), pricelist_id UUID REFERENCES sales.pricelists(id), payment_term_id UUID REFERENCES financial.payment_terms(id), -- Usuario user_id UUID REFERENCES auth.users(id), sales_team_id UUID REFERENCES sales.sales_teams(id), -- Montos amount_untaxed DECIMAL(15, 2) NOT NULL DEFAULT 0, amount_tax DECIMAL(15, 2) NOT NULL DEFAULT 0, amount_total DECIMAL(15, 2) NOT NULL DEFAULT 0, -- Estado status sales.order_status NOT NULL DEFAULT 'draft', invoice_status sales.invoice_status NOT NULL DEFAULT 'pending', delivery_status sales.delivery_status NOT NULL DEFAULT 'pending', -- Facturación invoice_policy sales.invoice_policy DEFAULT 'order', -- Relaciones generadas picking_id UUID REFERENCES inventory.pickings(id), -- Notas notes TEXT, terms_conditions TEXT, -- Firma electrónica signature TEXT, -- base64 signature_date TIMESTAMP, signature_ip INET, -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, created_by UUID REFERENCES auth.users(id), updated_at TIMESTAMP, updated_by UUID REFERENCES auth.users(id), confirmed_at TIMESTAMP, confirmed_by UUID REFERENCES auth.users(id), cancelled_at TIMESTAMP, cancelled_by UUID REFERENCES auth.users(id), CONSTRAINT uq_sales_orders_name_company UNIQUE (company_id, name), CONSTRAINT chk_sales_orders_validity CHECK (validity_date IS NULL OR validity_date >= order_date) ); -- Tabla: sales_order_lines (Líneas de orden de venta) CREATE TABLE sales.sales_order_lines ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, order_id UUID NOT NULL REFERENCES sales.sales_orders(id) ON DELETE CASCADE, product_id UUID NOT NULL REFERENCES inventory.products(id), description TEXT NOT NULL, -- Cantidades quantity DECIMAL(12, 4) NOT NULL, qty_delivered DECIMAL(12, 4) DEFAULT 0, qty_invoiced DECIMAL(12, 4) DEFAULT 0, uom_id UUID NOT NULL REFERENCES core.uom(id), -- Precios price_unit DECIMAL(15, 4) NOT NULL, discount DECIMAL(5, 2) DEFAULT 0, -- Porcentaje de descuento -- Impuestos tax_ids UUID[] DEFAULT '{}', -- Montos amount_untaxed DECIMAL(15, 2) NOT NULL, amount_tax DECIMAL(15, 2) NOT NULL, amount_total DECIMAL(15, 2) NOT NULL, -- Analítica analytic_account_id UUID REFERENCES analytics.analytic_accounts(id), -- Distribución analítica -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP, CONSTRAINT chk_sales_order_lines_quantity CHECK (quantity > 0), CONSTRAINT chk_sales_order_lines_discount CHECK (discount >= 0 AND discount <= 100), CONSTRAINT chk_sales_order_lines_qty_delivered CHECK (qty_delivered >= 0 AND qty_delivered <= quantity), CONSTRAINT chk_sales_order_lines_qty_invoiced CHECK (qty_invoiced >= 0 AND qty_invoiced <= quantity) ); -- Índices para sales_order_lines CREATE INDEX idx_sales_order_lines_tenant_id ON sales.sales_order_lines(tenant_id); CREATE INDEX idx_sales_order_lines_order_id ON sales.sales_order_lines(order_id); CREATE INDEX idx_sales_order_lines_product_id ON sales.sales_order_lines(product_id); -- RLS para sales_order_lines ALTER TABLE sales.sales_order_lines ENABLE ROW LEVEL SECURITY; CREATE POLICY tenant_isolation_sales_order_lines ON sales.sales_order_lines USING (tenant_id = current_setting('app.current_tenant_id', true)::uuid); -- Tabla: quotations (Cotizaciones) CREATE TABLE sales.quotations ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, company_id UUID NOT NULL REFERENCES auth.companies(id) ON DELETE CASCADE, -- Numeración name VARCHAR(100) NOT NULL, -- Cliente potencial partner_id UUID NOT NULL REFERENCES core.partners(id), -- Fechas quotation_date DATE NOT NULL, validity_date DATE NOT NULL, -- Configuración currency_id UUID NOT NULL REFERENCES core.currencies(id), pricelist_id UUID REFERENCES sales.pricelists(id), -- Usuario user_id UUID REFERENCES auth.users(id), sales_team_id UUID REFERENCES sales.sales_teams(id), -- Montos amount_untaxed DECIMAL(15, 2) NOT NULL DEFAULT 0, amount_tax DECIMAL(15, 2) NOT NULL DEFAULT 0, amount_total DECIMAL(15, 2) NOT NULL DEFAULT 0, -- Estado status sales.quotation_status NOT NULL DEFAULT 'draft', -- Conversión sale_order_id UUID REFERENCES sales.sales_orders(id), -- Orden generada -- Notas notes TEXT, terms_conditions TEXT, -- Firma electrónica signature TEXT, -- base64 signature_date TIMESTAMP, signature_ip INET, -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, created_by UUID REFERENCES auth.users(id), updated_at TIMESTAMP, updated_by UUID REFERENCES auth.users(id), CONSTRAINT uq_quotations_name_company UNIQUE (company_id, name), CONSTRAINT chk_quotations_validity CHECK (validity_date >= quotation_date) ); -- Tabla: quotation_lines (Líneas de cotización) CREATE TABLE sales.quotation_lines ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, quotation_id UUID NOT NULL REFERENCES sales.quotations(id) ON DELETE CASCADE, product_id UUID REFERENCES inventory.products(id), description TEXT NOT NULL, quantity DECIMAL(12, 4) NOT NULL, uom_id UUID NOT NULL REFERENCES core.uom(id), price_unit DECIMAL(15, 4) NOT NULL, discount DECIMAL(5, 2) DEFAULT 0, tax_ids UUID[] DEFAULT '{}', amount_untaxed DECIMAL(15, 2) NOT NULL, amount_tax DECIMAL(15, 2) NOT NULL, amount_total DECIMAL(15, 2) NOT NULL, created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, CONSTRAINT chk_quotation_lines_quantity CHECK (quantity > 0), CONSTRAINT chk_quotation_lines_discount CHECK (discount >= 0 AND discount <= 100) ); -- Índices para quotation_lines CREATE INDEX idx_quotation_lines_tenant_id ON sales.quotation_lines(tenant_id); CREATE INDEX idx_quotation_lines_quotation_id ON sales.quotation_lines(quotation_id); CREATE INDEX idx_quotation_lines_product_id ON sales.quotation_lines(product_id); -- RLS para quotation_lines ALTER TABLE sales.quotation_lines ENABLE ROW LEVEL SECURITY; CREATE POLICY tenant_isolation_quotation_lines ON sales.quotation_lines USING (tenant_id = current_setting('app.current_tenant_id', true)::uuid); -- Tabla: pricelists (Listas de precios) CREATE TABLE sales.pricelists ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, company_id UUID REFERENCES auth.companies(id), name VARCHAR(255) NOT NULL, currency_id UUID NOT NULL REFERENCES core.currencies(id), -- Control active BOOLEAN NOT NULL DEFAULT TRUE, -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, created_by UUID REFERENCES auth.users(id), updated_at TIMESTAMP, updated_by UUID REFERENCES auth.users(id), CONSTRAINT uq_pricelists_name_tenant UNIQUE (tenant_id, name) ); -- Tabla: pricelist_items (Items de lista de precios) CREATE TABLE sales.pricelist_items ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, pricelist_id UUID NOT NULL REFERENCES sales.pricelists(id) ON DELETE CASCADE, product_id UUID REFERENCES inventory.products(id), product_category_id UUID REFERENCES core.product_categories(id), -- Precio price DECIMAL(15, 4) NOT NULL, -- Cantidad mínima min_quantity DECIMAL(12, 4) DEFAULT 1, -- Validez valid_from DATE, valid_to DATE, -- Control active BOOLEAN NOT NULL DEFAULT TRUE, -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, created_by UUID REFERENCES auth.users(id), CONSTRAINT chk_pricelist_items_price CHECK (price >= 0), CONSTRAINT chk_pricelist_items_min_qty CHECK (min_quantity > 0), CONSTRAINT chk_pricelist_items_dates CHECK (valid_to IS NULL OR valid_to >= valid_from), CONSTRAINT chk_pricelist_items_product_or_category CHECK ( (product_id IS NOT NULL AND product_category_id IS NULL) OR (product_id IS NULL AND product_category_id IS NOT NULL) ) ); -- Índices para pricelist_items CREATE INDEX idx_pricelist_items_tenant_id ON sales.pricelist_items(tenant_id); CREATE INDEX idx_pricelist_items_pricelist_id ON sales.pricelist_items(pricelist_id); CREATE INDEX idx_pricelist_items_product_id ON sales.pricelist_items(product_id); -- RLS para pricelist_items ALTER TABLE sales.pricelist_items ENABLE ROW LEVEL SECURITY; CREATE POLICY tenant_isolation_pricelist_items ON sales.pricelist_items USING (tenant_id = current_setting('app.current_tenant_id', true)::uuid); -- Tabla: customer_groups (Grupos de clientes) CREATE TABLE sales.customer_groups ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, name VARCHAR(255) NOT NULL, description TEXT, discount_percentage DECIMAL(5, 2) DEFAULT 0, -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, created_by UUID REFERENCES auth.users(id), CONSTRAINT uq_customer_groups_name_tenant UNIQUE (tenant_id, name), CONSTRAINT chk_customer_groups_discount CHECK (discount_percentage >= 0 AND discount_percentage <= 100) ); -- Tabla: customer_group_members (Miembros de grupos) CREATE TABLE sales.customer_group_members ( customer_group_id UUID NOT NULL REFERENCES sales.customer_groups(id) ON DELETE CASCADE, partner_id UUID NOT NULL REFERENCES core.partners(id) ON DELETE CASCADE, joined_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, PRIMARY KEY (customer_group_id, partner_id) ); -- Tabla: sales_teams (Equipos de ventas) CREATE TABLE sales.sales_teams ( id UUID PRIMARY KEY DEFAULT gen_random_uuid(), tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE, company_id UUID NOT NULL REFERENCES auth.companies(id) ON DELETE CASCADE, name VARCHAR(255) NOT NULL, code VARCHAR(50), team_leader_id UUID REFERENCES auth.users(id), -- Objetivos target_monthly DECIMAL(15, 2), target_annual DECIMAL(15, 2), -- Control active BOOLEAN NOT NULL DEFAULT TRUE, -- Auditoría created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, created_by UUID REFERENCES auth.users(id), updated_at TIMESTAMP, updated_by UUID REFERENCES auth.users(id), CONSTRAINT uq_sales_teams_code_company UNIQUE (company_id, code) ); -- Tabla: sales_team_members (Miembros de equipos) CREATE TABLE sales.sales_team_members ( sales_team_id UUID NOT NULL REFERENCES sales.sales_teams(id) ON DELETE CASCADE, user_id UUID NOT NULL REFERENCES auth.users(id) ON DELETE CASCADE, joined_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, PRIMARY KEY (sales_team_id, user_id) ); -- ===================================================== -- INDICES -- ===================================================== -- Sales Orders CREATE INDEX idx_sales_orders_tenant_id ON sales.sales_orders(tenant_id); CREATE INDEX idx_sales_orders_company_id ON sales.sales_orders(company_id); CREATE INDEX idx_sales_orders_partner_id ON sales.sales_orders(partner_id); CREATE INDEX idx_sales_orders_name ON sales.sales_orders(name); CREATE INDEX idx_sales_orders_status ON sales.sales_orders(status); CREATE INDEX idx_sales_orders_order_date ON sales.sales_orders(order_date); CREATE INDEX idx_sales_orders_user_id ON sales.sales_orders(user_id); CREATE INDEX idx_sales_orders_sales_team_id ON sales.sales_orders(sales_team_id); -- Sales Order Lines CREATE INDEX idx_sales_order_lines_order_id ON sales.sales_order_lines(order_id); CREATE INDEX idx_sales_order_lines_product_id ON sales.sales_order_lines(product_id); CREATE INDEX idx_sales_order_lines_analytic_account_id ON sales.sales_order_lines(analytic_account_id) WHERE analytic_account_id IS NOT NULL; -- Quotations CREATE INDEX idx_quotations_tenant_id ON sales.quotations(tenant_id); CREATE INDEX idx_quotations_company_id ON sales.quotations(company_id); CREATE INDEX idx_quotations_partner_id ON sales.quotations(partner_id); CREATE INDEX idx_quotations_status ON sales.quotations(status); CREATE INDEX idx_quotations_validity_date ON sales.quotations(validity_date); -- Quotation Lines CREATE INDEX idx_quotation_lines_quotation_id ON sales.quotation_lines(quotation_id); CREATE INDEX idx_quotation_lines_product_id ON sales.quotation_lines(product_id); -- Pricelists CREATE INDEX idx_pricelists_tenant_id ON sales.pricelists(tenant_id); CREATE INDEX idx_pricelists_active ON sales.pricelists(active) WHERE active = TRUE; -- Pricelist Items CREATE INDEX idx_pricelist_items_pricelist_id ON sales.pricelist_items(pricelist_id); CREATE INDEX idx_pricelist_items_product_id ON sales.pricelist_items(product_id); CREATE INDEX idx_pricelist_items_category_id ON sales.pricelist_items(product_category_id); -- Customer Groups CREATE INDEX idx_customer_groups_tenant_id ON sales.customer_groups(tenant_id); -- Sales Teams CREATE INDEX idx_sales_teams_tenant_id ON sales.sales_teams(tenant_id); CREATE INDEX idx_sales_teams_company_id ON sales.sales_teams(company_id); CREATE INDEX idx_sales_teams_leader_id ON sales.sales_teams(team_leader_id); -- ===================================================== -- FUNCTIONS -- ===================================================== -- Función: calculate_sales_order_totals CREATE OR REPLACE FUNCTION sales.calculate_sales_order_totals(p_order_id UUID) RETURNS VOID AS $$ DECLARE v_amount_untaxed DECIMAL; v_amount_tax DECIMAL; v_amount_total DECIMAL; BEGIN SELECT COALESCE(SUM(amount_untaxed), 0), COALESCE(SUM(amount_tax), 0), COALESCE(SUM(amount_total), 0) INTO v_amount_untaxed, v_amount_tax, v_amount_total FROM sales.sales_order_lines WHERE order_id = p_order_id; UPDATE sales.sales_orders SET amount_untaxed = v_amount_untaxed, amount_tax = v_amount_tax, amount_total = v_amount_total, updated_at = CURRENT_TIMESTAMP, updated_by = get_current_user_id() WHERE id = p_order_id; END; $$ LANGUAGE plpgsql; COMMENT ON FUNCTION sales.calculate_sales_order_totals IS 'Calcula los totales de una orden de venta'; -- Función: calculate_quotation_totals CREATE OR REPLACE FUNCTION sales.calculate_quotation_totals(p_quotation_id UUID) RETURNS VOID AS $$ DECLARE v_amount_untaxed DECIMAL; v_amount_tax DECIMAL; v_amount_total DECIMAL; BEGIN SELECT COALESCE(SUM(amount_untaxed), 0), COALESCE(SUM(amount_tax), 0), COALESCE(SUM(amount_total), 0) INTO v_amount_untaxed, v_amount_tax, v_amount_total FROM sales.quotation_lines WHERE quotation_id = p_quotation_id; UPDATE sales.quotations SET amount_untaxed = v_amount_untaxed, amount_tax = v_amount_tax, amount_total = v_amount_total, updated_at = CURRENT_TIMESTAMP, updated_by = get_current_user_id() WHERE id = p_quotation_id; END; $$ LANGUAGE plpgsql; COMMENT ON FUNCTION sales.calculate_quotation_totals IS 'Calcula los totales de una cotización'; -- Función: convert_quotation_to_order CREATE OR REPLACE FUNCTION sales.convert_quotation_to_order(p_quotation_id UUID) RETURNS UUID AS $$ DECLARE v_quotation RECORD; v_order_id UUID; BEGIN -- Obtener cotización SELECT * INTO v_quotation FROM sales.quotations WHERE id = p_quotation_id; IF NOT FOUND THEN RAISE EXCEPTION 'Quotation % not found', p_quotation_id; END IF; IF v_quotation.status != 'approved' THEN RAISE EXCEPTION 'Quotation must be approved before conversion'; END IF; -- Crear orden de venta INSERT INTO sales.sales_orders ( tenant_id, company_id, name, partner_id, order_date, currency_id, pricelist_id, user_id, sales_team_id, amount_untaxed, amount_tax, amount_total, notes, terms_conditions, signature, signature_date, signature_ip ) VALUES ( v_quotation.tenant_id, v_quotation.company_id, REPLACE(v_quotation.name, 'QT', 'SO'), v_quotation.partner_id, CURRENT_DATE, v_quotation.currency_id, v_quotation.pricelist_id, v_quotation.user_id, v_quotation.sales_team_id, v_quotation.amount_untaxed, v_quotation.amount_tax, v_quotation.amount_total, v_quotation.notes, v_quotation.terms_conditions, v_quotation.signature, v_quotation.signature_date, v_quotation.signature_ip ) RETURNING id INTO v_order_id; -- Copiar líneas INSERT INTO sales.sales_order_lines ( order_id, product_id, description, quantity, uom_id, price_unit, discount, tax_ids, amount_untaxed, amount_tax, amount_total ) SELECT v_order_id, product_id, description, quantity, uom_id, price_unit, discount, tax_ids, amount_untaxed, amount_tax, amount_total FROM sales.quotation_lines WHERE quotation_id = p_quotation_id; -- Actualizar cotización UPDATE sales.quotations SET status = 'converted', sale_order_id = v_order_id, updated_at = CURRENT_TIMESTAMP, updated_by = get_current_user_id() WHERE id = p_quotation_id; RETURN v_order_id; END; $$ LANGUAGE plpgsql; COMMENT ON FUNCTION sales.convert_quotation_to_order IS 'Convierte una cotización aprobada en orden de venta'; -- ===================================================== -- TRIGGERS -- ===================================================== CREATE TRIGGER trg_sales_orders_updated_at BEFORE UPDATE ON sales.sales_orders FOR EACH ROW EXECUTE FUNCTION auth.update_updated_at_column(); CREATE TRIGGER trg_quotations_updated_at BEFORE UPDATE ON sales.quotations FOR EACH ROW EXECUTE FUNCTION auth.update_updated_at_column(); CREATE TRIGGER trg_pricelists_updated_at BEFORE UPDATE ON sales.pricelists FOR EACH ROW EXECUTE FUNCTION auth.update_updated_at_column(); CREATE TRIGGER trg_sales_teams_updated_at BEFORE UPDATE ON sales.sales_teams FOR EACH ROW EXECUTE FUNCTION auth.update_updated_at_column(); -- Trigger: Actualizar totales de orden al cambiar líneas CREATE OR REPLACE FUNCTION sales.trg_update_so_totals() RETURNS TRIGGER AS $$ BEGIN IF TG_OP = 'DELETE' THEN PERFORM sales.calculate_sales_order_totals(OLD.order_id); ELSE PERFORM sales.calculate_sales_order_totals(NEW.order_id); END IF; RETURN NULL; END; $$ LANGUAGE plpgsql; CREATE TRIGGER trg_sales_order_lines_update_totals AFTER INSERT OR UPDATE OR DELETE ON sales.sales_order_lines FOR EACH ROW EXECUTE FUNCTION sales.trg_update_so_totals(); -- Trigger: Actualizar totales de cotización al cambiar líneas CREATE OR REPLACE FUNCTION sales.trg_update_quotation_totals() RETURNS TRIGGER AS $$ BEGIN IF TG_OP = 'DELETE' THEN PERFORM sales.calculate_quotation_totals(OLD.quotation_id); ELSE PERFORM sales.calculate_quotation_totals(NEW.quotation_id); END IF; RETURN NULL; END; $$ LANGUAGE plpgsql; CREATE TRIGGER trg_quotation_lines_update_totals AFTER INSERT OR UPDATE OR DELETE ON sales.quotation_lines FOR EACH ROW EXECUTE FUNCTION sales.trg_update_quotation_totals(); -- ===================================================== -- TRACKING AUTOMÁTICO (mail.thread pattern) -- ===================================================== -- Trigger: Tracking automático para órdenes de venta CREATE TRIGGER track_sales_order_changes AFTER INSERT OR UPDATE OR DELETE ON sales.sales_orders FOR EACH ROW EXECUTE FUNCTION system.track_field_changes(); COMMENT ON TRIGGER track_sales_order_changes ON sales.sales_orders IS 'Registra automáticamente cambios en órdenes de venta (estado, cliente, monto, fecha, facturación, entrega)'; -- ===================================================== -- ROW LEVEL SECURITY (RLS) -- ===================================================== ALTER TABLE sales.sales_orders ENABLE ROW LEVEL SECURITY; ALTER TABLE sales.quotations ENABLE ROW LEVEL SECURITY; ALTER TABLE sales.pricelists ENABLE ROW LEVEL SECURITY; ALTER TABLE sales.customer_groups ENABLE ROW LEVEL SECURITY; ALTER TABLE sales.sales_teams ENABLE ROW LEVEL SECURITY; CREATE POLICY tenant_isolation_sales_orders ON sales.sales_orders USING (tenant_id = get_current_tenant_id()); CREATE POLICY tenant_isolation_quotations ON sales.quotations USING (tenant_id = get_current_tenant_id()); CREATE POLICY tenant_isolation_pricelists ON sales.pricelists USING (tenant_id = get_current_tenant_id()); CREATE POLICY tenant_isolation_customer_groups ON sales.customer_groups USING (tenant_id = get_current_tenant_id()); CREATE POLICY tenant_isolation_sales_teams ON sales.sales_teams USING (tenant_id = get_current_tenant_id()); -- ===================================================== -- COMENTARIOS -- ===================================================== COMMENT ON SCHEMA sales IS 'Schema de gestión de ventas, cotizaciones y clientes'; COMMENT ON TABLE sales.sales_orders IS 'Órdenes de venta confirmadas'; COMMENT ON TABLE sales.sales_order_lines IS 'Líneas de órdenes de venta'; COMMENT ON TABLE sales.quotations IS 'Cotizaciones enviadas a clientes'; COMMENT ON TABLE sales.quotation_lines IS 'Líneas de cotizaciones'; COMMENT ON TABLE sales.pricelists IS 'Listas de precios para clientes'; COMMENT ON TABLE sales.pricelist_items IS 'Items de listas de precios por producto/categoría'; COMMENT ON TABLE sales.customer_groups IS 'Grupos de clientes para descuentos y segmentación'; COMMENT ON TABLE sales.sales_teams IS 'Equipos de ventas con objetivos'; -- ===================================================== -- FIN DEL SCHEMA SALES -- =====================================================