773 lines
26 KiB
PL/PgSQL
773 lines
26 KiB
PL/PgSQL
-- =====================================================
|
|
-- SCHEMA: inventory
|
|
-- PROPÓSITO: Gestión de inventarios, productos, almacenes, movimientos
|
|
-- MÓDULOS: MGN-005 (Inventario Básico)
|
|
-- FECHA: 2025-11-24
|
|
-- =====================================================
|
|
|
|
-- Crear schema
|
|
CREATE SCHEMA IF NOT EXISTS inventory;
|
|
|
|
-- =====================================================
|
|
-- TYPES (ENUMs)
|
|
-- =====================================================
|
|
|
|
CREATE TYPE inventory.product_type AS ENUM (
|
|
'storable',
|
|
'consumable',
|
|
'service'
|
|
);
|
|
|
|
CREATE TYPE inventory.tracking_type AS ENUM (
|
|
'none',
|
|
'lot',
|
|
'serial'
|
|
);
|
|
|
|
CREATE TYPE inventory.location_type AS ENUM (
|
|
'internal',
|
|
'customer',
|
|
'supplier',
|
|
'inventory',
|
|
'production',
|
|
'transit'
|
|
);
|
|
|
|
CREATE TYPE inventory.picking_type AS ENUM (
|
|
'incoming',
|
|
'outgoing',
|
|
'internal'
|
|
);
|
|
|
|
CREATE TYPE inventory.move_status AS ENUM (
|
|
'draft',
|
|
'confirmed',
|
|
'assigned',
|
|
'done',
|
|
'cancelled'
|
|
);
|
|
|
|
CREATE TYPE inventory.valuation_method AS ENUM (
|
|
'fifo',
|
|
'average',
|
|
'standard'
|
|
);
|
|
|
|
-- =====================================================
|
|
-- TABLES
|
|
-- =====================================================
|
|
|
|
-- Tabla: products (Productos)
|
|
CREATE TABLE inventory.products (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE,
|
|
|
|
-- Identificación
|
|
name VARCHAR(255) NOT NULL,
|
|
code VARCHAR(100),
|
|
barcode VARCHAR(100),
|
|
description TEXT,
|
|
|
|
-- Tipo
|
|
product_type inventory.product_type NOT NULL DEFAULT 'storable',
|
|
tracking inventory.tracking_type NOT NULL DEFAULT 'none',
|
|
|
|
-- Categoría
|
|
category_id UUID REFERENCES core.product_categories(id),
|
|
|
|
-- Unidades de medida
|
|
uom_id UUID NOT NULL REFERENCES core.uom(id), -- UoM de venta/uso
|
|
purchase_uom_id UUID REFERENCES core.uom(id), -- UoM de compra
|
|
|
|
-- Precios
|
|
cost_price DECIMAL(15, 4) DEFAULT 0,
|
|
list_price DECIMAL(15, 4) DEFAULT 0,
|
|
|
|
-- Configuración de inventario
|
|
valuation_method inventory.valuation_method DEFAULT 'fifo',
|
|
is_storable BOOLEAN GENERATED ALWAYS AS (product_type = 'storable') STORED,
|
|
|
|
-- Pesos y dimensiones
|
|
weight DECIMAL(12, 4),
|
|
volume DECIMAL(12, 4),
|
|
|
|
-- Proveedores y clientes
|
|
can_be_sold BOOLEAN DEFAULT TRUE,
|
|
can_be_purchased BOOLEAN DEFAULT TRUE,
|
|
|
|
-- Imagen
|
|
image_url VARCHAR(500),
|
|
|
|
-- 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),
|
|
deleted_at TIMESTAMP,
|
|
deleted_by UUID REFERENCES auth.users(id),
|
|
|
|
CONSTRAINT uq_products_code_tenant UNIQUE (tenant_id, code),
|
|
CONSTRAINT uq_products_barcode UNIQUE (barcode)
|
|
);
|
|
|
|
-- Tabla: product_variants (Variantes de producto)
|
|
CREATE TABLE inventory.product_variants (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
product_template_id UUID NOT NULL REFERENCES inventory.products(id) ON DELETE CASCADE,
|
|
|
|
-- Atributos (JSON)
|
|
-- Ejemplo: {"color": "red", "size": "XL"}
|
|
attribute_values JSONB NOT NULL DEFAULT '{}',
|
|
|
|
-- Identificación
|
|
name VARCHAR(255),
|
|
code VARCHAR(100),
|
|
barcode VARCHAR(100),
|
|
|
|
-- Precio diferencial
|
|
price_extra DECIMAL(15, 4) DEFAULT 0,
|
|
|
|
-- 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 uq_product_variants_barcode UNIQUE (barcode)
|
|
);
|
|
|
|
-- Tabla: warehouses (Almacenes)
|
|
CREATE TABLE inventory.warehouses (
|
|
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(20) NOT NULL,
|
|
|
|
-- Dirección
|
|
address_id UUID REFERENCES core.addresses(id),
|
|
|
|
-- Configuración
|
|
is_default BOOLEAN DEFAULT FALSE,
|
|
|
|
-- 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_warehouses_code_company UNIQUE (company_id, code)
|
|
);
|
|
|
|
-- Tabla: locations (Ubicaciones de inventario)
|
|
CREATE TABLE inventory.locations (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE,
|
|
|
|
warehouse_id UUID REFERENCES inventory.warehouses(id),
|
|
name VARCHAR(255) NOT NULL,
|
|
complete_name TEXT, -- Generado: "Warehouse / Zone A / Shelf 1"
|
|
location_type inventory.location_type NOT NULL DEFAULT 'internal',
|
|
|
|
-- Jerarquía
|
|
parent_id UUID REFERENCES inventory.locations(id),
|
|
|
|
-- Configuración
|
|
is_scrap_location BOOLEAN DEFAULT FALSE,
|
|
is_return_location BOOLEAN DEFAULT FALSE,
|
|
|
|
-- 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 chk_locations_no_self_parent CHECK (id != parent_id)
|
|
);
|
|
|
|
-- Tabla: lots (Lotes/Series) - DEBE IR ANTES DE stock_quants por FK
|
|
CREATE TABLE inventory.lots (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE,
|
|
|
|
product_id UUID NOT NULL REFERENCES inventory.products(id),
|
|
name VARCHAR(100) NOT NULL,
|
|
ref VARCHAR(100), -- Referencia externa
|
|
|
|
-- Fechas
|
|
manufacture_date DATE,
|
|
expiration_date DATE,
|
|
removal_date DATE,
|
|
alert_date DATE,
|
|
|
|
-- Notas
|
|
notes TEXT,
|
|
|
|
-- Auditoría
|
|
created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
|
created_by UUID REFERENCES auth.users(id),
|
|
|
|
CONSTRAINT uq_lots_name_product UNIQUE (product_id, name),
|
|
CONSTRAINT chk_lots_expiration CHECK (expiration_date IS NULL OR expiration_date > manufacture_date)
|
|
);
|
|
|
|
-- Tabla: stock_quants (Cantidades en stock)
|
|
CREATE TABLE inventory.stock_quants (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE,
|
|
|
|
product_id UUID NOT NULL REFERENCES inventory.products(id),
|
|
location_id UUID NOT NULL REFERENCES inventory.locations(id),
|
|
lot_id UUID REFERENCES inventory.lots(id),
|
|
|
|
-- Cantidades
|
|
quantity DECIMAL(12, 4) NOT NULL DEFAULT 0,
|
|
reserved_quantity DECIMAL(12, 4) NOT NULL DEFAULT 0,
|
|
available_quantity DECIMAL(12, 4) GENERATED ALWAYS AS (quantity - reserved_quantity) STORED,
|
|
|
|
-- Valoración
|
|
cost DECIMAL(15, 4) DEFAULT 0,
|
|
|
|
-- Auditoría
|
|
created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
|
updated_at TIMESTAMP,
|
|
|
|
CONSTRAINT chk_stock_quants_reserved CHECK (reserved_quantity >= 0 AND reserved_quantity <= quantity)
|
|
);
|
|
|
|
-- Unique index for stock_quants (allows expressions unlike UNIQUE constraint)
|
|
CREATE UNIQUE INDEX uq_stock_quants_product_location_lot
|
|
ON inventory.stock_quants (tenant_id, product_id, location_id, COALESCE(lot_id, '00000000-0000-0000-0000-000000000000'::UUID));
|
|
|
|
-- Índices para stock_quants
|
|
CREATE INDEX idx_stock_quants_tenant_id ON inventory.stock_quants(tenant_id);
|
|
CREATE INDEX idx_stock_quants_product_location ON inventory.stock_quants(product_id, location_id);
|
|
|
|
-- RLS para stock_quants
|
|
ALTER TABLE inventory.stock_quants ENABLE ROW LEVEL SECURITY;
|
|
CREATE POLICY tenant_isolation_stock_quants ON inventory.stock_quants
|
|
USING (tenant_id = current_setting('app.current_tenant_id', true)::uuid);
|
|
|
|
-- Tabla: pickings (Albaranes/Transferencias)
|
|
CREATE TABLE inventory.pickings (
|
|
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(100) NOT NULL,
|
|
picking_type inventory.picking_type NOT NULL,
|
|
|
|
-- Ubicaciones
|
|
location_id UUID NOT NULL REFERENCES inventory.locations(id), -- Origen
|
|
location_dest_id UUID NOT NULL REFERENCES inventory.locations(id), -- Destino
|
|
|
|
-- Partner (cliente/proveedor)
|
|
partner_id UUID REFERENCES core.partners(id),
|
|
|
|
-- Fechas
|
|
scheduled_date TIMESTAMP,
|
|
date_done TIMESTAMP,
|
|
|
|
-- Origen
|
|
origin VARCHAR(255), -- Referencia al documento origen (PO, SO, etc.)
|
|
|
|
-- Estado
|
|
status inventory.move_status NOT NULL DEFAULT 'draft',
|
|
|
|
-- Notas
|
|
notes TEXT,
|
|
|
|
-- 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),
|
|
validated_at TIMESTAMP,
|
|
validated_by UUID REFERENCES auth.users(id),
|
|
|
|
CONSTRAINT uq_pickings_name_company UNIQUE (company_id, name)
|
|
);
|
|
|
|
-- Tabla: stock_moves (Movimientos de inventario)
|
|
CREATE TABLE inventory.stock_moves (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE,
|
|
|
|
product_id UUID NOT NULL REFERENCES inventory.products(id),
|
|
product_uom_id UUID NOT NULL REFERENCES core.uom(id),
|
|
|
|
-- Ubicaciones
|
|
location_id UUID NOT NULL REFERENCES inventory.locations(id), -- Origen
|
|
location_dest_id UUID NOT NULL REFERENCES inventory.locations(id), -- Destino
|
|
|
|
-- Cantidades
|
|
product_qty DECIMAL(12, 4) NOT NULL,
|
|
quantity_done DECIMAL(12, 4) DEFAULT 0,
|
|
|
|
-- Lote/Serie
|
|
lot_id UUID REFERENCES inventory.lots(id),
|
|
|
|
-- Relación con picking
|
|
picking_id UUID REFERENCES inventory.pickings(id) ON DELETE CASCADE,
|
|
|
|
-- Origen del movimiento
|
|
origin VARCHAR(255),
|
|
ref VARCHAR(255),
|
|
|
|
-- Estado
|
|
status inventory.move_status NOT NULL DEFAULT 'draft',
|
|
|
|
-- Fechas
|
|
date_expected TIMESTAMP,
|
|
date TIMESTAMP,
|
|
|
|
-- Precio (para valoración)
|
|
price_unit DECIMAL(15, 4) DEFAULT 0,
|
|
|
|
-- 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,
|
|
created_by UUID REFERENCES auth.users(id),
|
|
updated_at TIMESTAMP,
|
|
updated_by UUID REFERENCES auth.users(id),
|
|
|
|
CONSTRAINT chk_stock_moves_quantity CHECK (product_qty > 0),
|
|
CONSTRAINT chk_stock_moves_quantity_done CHECK (quantity_done >= 0)
|
|
);
|
|
|
|
-- Tabla: inventory_adjustments (Ajustes de inventario)
|
|
CREATE TABLE inventory.inventory_adjustments (
|
|
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(100) NOT NULL,
|
|
|
|
-- Ubicación a ajustar
|
|
location_id UUID NOT NULL REFERENCES inventory.locations(id),
|
|
|
|
-- Fecha de conteo
|
|
date DATE NOT NULL,
|
|
|
|
-- Estado
|
|
status inventory.move_status NOT NULL DEFAULT 'draft',
|
|
|
|
-- Notas
|
|
notes TEXT,
|
|
|
|
-- 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),
|
|
validated_at TIMESTAMP,
|
|
validated_by UUID REFERENCES auth.users(id),
|
|
|
|
CONSTRAINT uq_inventory_adjustments_name_company UNIQUE (company_id, name)
|
|
);
|
|
|
|
-- Tabla: inventory_adjustment_lines (Líneas de ajuste)
|
|
CREATE TABLE inventory.inventory_adjustment_lines (
|
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
tenant_id UUID NOT NULL REFERENCES auth.tenants(id) ON DELETE CASCADE,
|
|
adjustment_id UUID NOT NULL REFERENCES inventory.inventory_adjustments(id) ON DELETE CASCADE,
|
|
|
|
product_id UUID NOT NULL REFERENCES inventory.products(id),
|
|
location_id UUID NOT NULL REFERENCES inventory.locations(id),
|
|
lot_id UUID REFERENCES inventory.lots(id),
|
|
|
|
-- Cantidades
|
|
theoretical_qty DECIMAL(12, 4) NOT NULL DEFAULT 0, -- Cantidad teórica del sistema
|
|
counted_qty DECIMAL(12, 4) NOT NULL, -- Cantidad contada físicamente
|
|
difference_qty DECIMAL(12, 4) GENERATED ALWAYS AS (counted_qty - theoretical_qty) STORED,
|
|
|
|
-- Auditoría
|
|
created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP
|
|
);
|
|
|
|
-- Índices para inventory_adjustment_lines
|
|
CREATE INDEX idx_inventory_adjustment_lines_tenant_id ON inventory.inventory_adjustment_lines(tenant_id);
|
|
|
|
-- RLS para inventory_adjustment_lines
|
|
ALTER TABLE inventory.inventory_adjustment_lines ENABLE ROW LEVEL SECURITY;
|
|
CREATE POLICY tenant_isolation_inventory_adjustment_lines ON inventory.inventory_adjustment_lines
|
|
USING (tenant_id = current_setting('app.current_tenant_id', true)::uuid);
|
|
|
|
-- =====================================================
|
|
-- INDICES
|
|
-- =====================================================
|
|
|
|
-- Products
|
|
CREATE INDEX idx_products_tenant_id ON inventory.products(tenant_id);
|
|
CREATE INDEX idx_products_code ON inventory.products(code);
|
|
CREATE INDEX idx_products_barcode ON inventory.products(barcode);
|
|
CREATE INDEX idx_products_category_id ON inventory.products(category_id);
|
|
CREATE INDEX idx_products_type ON inventory.products(product_type);
|
|
CREATE INDEX idx_products_active ON inventory.products(active) WHERE active = TRUE;
|
|
|
|
-- Product Variants
|
|
CREATE INDEX idx_product_variants_template_id ON inventory.product_variants(product_template_id);
|
|
CREATE INDEX idx_product_variants_barcode ON inventory.product_variants(barcode);
|
|
|
|
-- Warehouses
|
|
CREATE INDEX idx_warehouses_tenant_id ON inventory.warehouses(tenant_id);
|
|
CREATE INDEX idx_warehouses_company_id ON inventory.warehouses(company_id);
|
|
CREATE INDEX idx_warehouses_code ON inventory.warehouses(code);
|
|
|
|
-- Locations
|
|
CREATE INDEX idx_locations_tenant_id ON inventory.locations(tenant_id);
|
|
CREATE INDEX idx_locations_warehouse_id ON inventory.locations(warehouse_id);
|
|
CREATE INDEX idx_locations_parent_id ON inventory.locations(parent_id);
|
|
CREATE INDEX idx_locations_type ON inventory.locations(location_type);
|
|
|
|
-- Stock Quants
|
|
CREATE INDEX idx_stock_quants_product_id ON inventory.stock_quants(product_id);
|
|
CREATE INDEX idx_stock_quants_location_id ON inventory.stock_quants(location_id);
|
|
CREATE INDEX idx_stock_quants_lot_id ON inventory.stock_quants(lot_id);
|
|
CREATE INDEX idx_stock_quants_available ON inventory.stock_quants(product_id, location_id)
|
|
WHERE available_quantity > 0;
|
|
|
|
-- Lots
|
|
CREATE INDEX idx_lots_tenant_id ON inventory.lots(tenant_id);
|
|
CREATE INDEX idx_lots_product_id ON inventory.lots(product_id);
|
|
CREATE INDEX idx_lots_name ON inventory.lots(name);
|
|
CREATE INDEX idx_lots_expiration_date ON inventory.lots(expiration_date);
|
|
|
|
-- Pickings
|
|
CREATE INDEX idx_pickings_tenant_id ON inventory.pickings(tenant_id);
|
|
CREATE INDEX idx_pickings_company_id ON inventory.pickings(company_id);
|
|
CREATE INDEX idx_pickings_name ON inventory.pickings(name);
|
|
CREATE INDEX idx_pickings_type ON inventory.pickings(picking_type);
|
|
CREATE INDEX idx_pickings_status ON inventory.pickings(status);
|
|
CREATE INDEX idx_pickings_partner_id ON inventory.pickings(partner_id);
|
|
CREATE INDEX idx_pickings_origin ON inventory.pickings(origin);
|
|
CREATE INDEX idx_pickings_scheduled_date ON inventory.pickings(scheduled_date);
|
|
|
|
-- Stock Moves
|
|
CREATE INDEX idx_stock_moves_tenant_id ON inventory.stock_moves(tenant_id);
|
|
CREATE INDEX idx_stock_moves_product_id ON inventory.stock_moves(product_id);
|
|
CREATE INDEX idx_stock_moves_picking_id ON inventory.stock_moves(picking_id);
|
|
CREATE INDEX idx_stock_moves_location_id ON inventory.stock_moves(location_id);
|
|
CREATE INDEX idx_stock_moves_location_dest_id ON inventory.stock_moves(location_dest_id);
|
|
CREATE INDEX idx_stock_moves_status ON inventory.stock_moves(status);
|
|
CREATE INDEX idx_stock_moves_lot_id ON inventory.stock_moves(lot_id);
|
|
CREATE INDEX idx_stock_moves_analytic_account_id ON inventory.stock_moves(analytic_account_id) WHERE analytic_account_id IS NOT NULL;
|
|
|
|
-- Inventory Adjustments
|
|
CREATE INDEX idx_inventory_adjustments_tenant_id ON inventory.inventory_adjustments(tenant_id);
|
|
CREATE INDEX idx_inventory_adjustments_company_id ON inventory.inventory_adjustments(company_id);
|
|
CREATE INDEX idx_inventory_adjustments_location_id ON inventory.inventory_adjustments(location_id);
|
|
CREATE INDEX idx_inventory_adjustments_status ON inventory.inventory_adjustments(status);
|
|
CREATE INDEX idx_inventory_adjustments_date ON inventory.inventory_adjustments(date);
|
|
|
|
-- Inventory Adjustment Lines
|
|
CREATE INDEX idx_inventory_adjustment_lines_adjustment_id ON inventory.inventory_adjustment_lines(adjustment_id);
|
|
CREATE INDEX idx_inventory_adjustment_lines_product_id ON inventory.inventory_adjustment_lines(product_id);
|
|
|
|
-- =====================================================
|
|
-- FUNCTIONS
|
|
-- =====================================================
|
|
|
|
-- Función: update_stock_quant
|
|
-- Actualiza la cantidad en stock de un producto en una ubicación
|
|
CREATE OR REPLACE FUNCTION inventory.update_stock_quant(
|
|
p_product_id UUID,
|
|
p_location_id UUID,
|
|
p_lot_id UUID,
|
|
p_quantity DECIMAL
|
|
)
|
|
RETURNS VOID AS $$
|
|
BEGIN
|
|
INSERT INTO inventory.stock_quants (product_id, location_id, lot_id, quantity)
|
|
VALUES (p_product_id, p_location_id, p_lot_id, p_quantity)
|
|
ON CONFLICT (product_id, location_id, COALESCE(lot_id, '00000000-0000-0000-0000-000000000000'::UUID))
|
|
DO UPDATE SET
|
|
quantity = inventory.stock_quants.quantity + EXCLUDED.quantity,
|
|
updated_at = CURRENT_TIMESTAMP;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
COMMENT ON FUNCTION inventory.update_stock_quant IS 'Actualiza la cantidad en stock de un producto en una ubicación';
|
|
|
|
-- Función: reserve_quantity
|
|
-- Reserva cantidad de un producto en una ubicación
|
|
CREATE OR REPLACE FUNCTION inventory.reserve_quantity(
|
|
p_product_id UUID,
|
|
p_location_id UUID,
|
|
p_lot_id UUID,
|
|
p_quantity DECIMAL
|
|
)
|
|
RETURNS BOOLEAN AS $$
|
|
DECLARE
|
|
v_available DECIMAL;
|
|
BEGIN
|
|
-- Verificar disponibilidad
|
|
SELECT available_quantity INTO v_available
|
|
FROM inventory.stock_quants
|
|
WHERE product_id = p_product_id
|
|
AND location_id = p_location_id
|
|
AND (lot_id = p_lot_id OR (lot_id IS NULL AND p_lot_id IS NULL));
|
|
|
|
IF v_available IS NULL OR v_available < p_quantity THEN
|
|
RETURN FALSE;
|
|
END IF;
|
|
|
|
-- Reservar
|
|
UPDATE inventory.stock_quants
|
|
SET reserved_quantity = reserved_quantity + p_quantity,
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE product_id = p_product_id
|
|
AND location_id = p_location_id
|
|
AND (lot_id = p_lot_id OR (lot_id IS NULL AND p_lot_id IS NULL));
|
|
|
|
RETURN TRUE;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
COMMENT ON FUNCTION inventory.reserve_quantity IS 'Reserva cantidad de un producto en una ubicación';
|
|
|
|
-- Función: get_product_stock
|
|
-- Obtiene el stock disponible de un producto
|
|
CREATE OR REPLACE FUNCTION inventory.get_product_stock(
|
|
p_product_id UUID,
|
|
p_location_id UUID DEFAULT NULL
|
|
)
|
|
RETURNS TABLE(
|
|
location_id UUID,
|
|
location_name VARCHAR,
|
|
quantity DECIMAL,
|
|
reserved_quantity DECIMAL,
|
|
available_quantity DECIMAL
|
|
) AS $$
|
|
BEGIN
|
|
RETURN QUERY
|
|
SELECT
|
|
sq.location_id,
|
|
l.name AS location_name,
|
|
sq.quantity,
|
|
sq.reserved_quantity,
|
|
sq.available_quantity
|
|
FROM inventory.stock_quants sq
|
|
JOIN inventory.locations l ON sq.location_id = l.id
|
|
WHERE sq.product_id = p_product_id
|
|
AND (p_location_id IS NULL OR sq.location_id = p_location_id)
|
|
AND sq.quantity > 0;
|
|
END;
|
|
$$ LANGUAGE plpgsql STABLE;
|
|
|
|
COMMENT ON FUNCTION inventory.get_product_stock IS 'Obtiene el stock disponible de un producto por ubicación';
|
|
|
|
-- Función: process_stock_move
|
|
-- Procesa un movimiento de inventario (actualiza quants)
|
|
CREATE OR REPLACE FUNCTION inventory.process_stock_move(p_move_id UUID)
|
|
RETURNS VOID AS $$
|
|
DECLARE
|
|
v_move RECORD;
|
|
BEGIN
|
|
-- Obtener datos del movimiento
|
|
SELECT * INTO v_move
|
|
FROM inventory.stock_moves
|
|
WHERE id = p_move_id;
|
|
|
|
IF NOT FOUND THEN
|
|
RAISE EXCEPTION 'Stock move % not found', p_move_id;
|
|
END IF;
|
|
|
|
IF v_move.status != 'confirmed' THEN
|
|
RAISE EXCEPTION 'Stock move % is not in confirmed status', p_move_id;
|
|
END IF;
|
|
|
|
-- Decrementar en ubicación origen
|
|
PERFORM inventory.update_stock_quant(
|
|
v_move.product_id,
|
|
v_move.location_id,
|
|
v_move.lot_id,
|
|
-v_move.quantity_done
|
|
);
|
|
|
|
-- Incrementar en ubicación destino
|
|
PERFORM inventory.update_stock_quant(
|
|
v_move.product_id,
|
|
v_move.location_dest_id,
|
|
v_move.lot_id,
|
|
v_move.quantity_done
|
|
);
|
|
|
|
-- Actualizar estado del movimiento
|
|
UPDATE inventory.stock_moves
|
|
SET status = 'done',
|
|
date = CURRENT_TIMESTAMP,
|
|
updated_at = CURRENT_TIMESTAMP,
|
|
updated_by = get_current_user_id()
|
|
WHERE id = p_move_id;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
COMMENT ON FUNCTION inventory.process_stock_move IS 'Procesa un movimiento de inventario y actualiza los quants';
|
|
|
|
-- Función: update_location_complete_name
|
|
-- Actualiza el nombre completo de una ubicación
|
|
CREATE OR REPLACE FUNCTION inventory.update_location_complete_name()
|
|
RETURNS TRIGGER AS $$
|
|
DECLARE
|
|
v_parent_name TEXT;
|
|
BEGIN
|
|
IF NEW.parent_id IS NULL THEN
|
|
NEW.complete_name := NEW.name;
|
|
ELSE
|
|
SELECT complete_name INTO v_parent_name
|
|
FROM inventory.locations
|
|
WHERE id = NEW.parent_id;
|
|
|
|
NEW.complete_name := v_parent_name || ' / ' || NEW.name;
|
|
END IF;
|
|
|
|
RETURN NEW;
|
|
END;
|
|
$$ LANGUAGE plpgsql;
|
|
|
|
COMMENT ON FUNCTION inventory.update_location_complete_name IS 'Actualiza el nombre completo de la ubicación';
|
|
|
|
-- =====================================================
|
|
-- TRIGGERS
|
|
-- =====================================================
|
|
|
|
-- Trigger: Actualizar updated_at
|
|
CREATE TRIGGER trg_products_updated_at
|
|
BEFORE UPDATE ON inventory.products
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION auth.update_updated_at_column();
|
|
|
|
CREATE TRIGGER trg_warehouses_updated_at
|
|
BEFORE UPDATE ON inventory.warehouses
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION auth.update_updated_at_column();
|
|
|
|
CREATE TRIGGER trg_locations_updated_at
|
|
BEFORE UPDATE ON inventory.locations
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION auth.update_updated_at_column();
|
|
|
|
CREATE TRIGGER trg_pickings_updated_at
|
|
BEFORE UPDATE ON inventory.pickings
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION auth.update_updated_at_column();
|
|
|
|
CREATE TRIGGER trg_stock_moves_updated_at
|
|
BEFORE UPDATE ON inventory.stock_moves
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION auth.update_updated_at_column();
|
|
|
|
CREATE TRIGGER trg_inventory_adjustments_updated_at
|
|
BEFORE UPDATE ON inventory.inventory_adjustments
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION auth.update_updated_at_column();
|
|
|
|
-- Trigger: Actualizar complete_name de ubicación
|
|
CREATE TRIGGER trg_locations_update_complete_name
|
|
BEFORE INSERT OR UPDATE OF name, parent_id ON inventory.locations
|
|
FOR EACH ROW
|
|
EXECUTE FUNCTION inventory.update_location_complete_name();
|
|
|
|
-- =====================================================
|
|
-- TRACKING AUTOMÁTICO (mail.thread pattern)
|
|
-- =====================================================
|
|
|
|
-- Trigger: Tracking automático para movimientos de stock
|
|
CREATE TRIGGER track_stock_move_changes
|
|
AFTER INSERT OR UPDATE OR DELETE ON inventory.stock_moves
|
|
FOR EACH ROW EXECUTE FUNCTION system.track_field_changes();
|
|
|
|
COMMENT ON TRIGGER track_stock_move_changes ON inventory.stock_moves IS
|
|
'Registra automáticamente cambios en movimientos de stock (estado, producto, cantidad, ubicaciones)';
|
|
|
|
-- =====================================================
|
|
-- ROW LEVEL SECURITY (RLS)
|
|
-- =====================================================
|
|
|
|
ALTER TABLE inventory.products ENABLE ROW LEVEL SECURITY;
|
|
ALTER TABLE inventory.warehouses ENABLE ROW LEVEL SECURITY;
|
|
ALTER TABLE inventory.locations ENABLE ROW LEVEL SECURITY;
|
|
ALTER TABLE inventory.lots ENABLE ROW LEVEL SECURITY;
|
|
ALTER TABLE inventory.pickings ENABLE ROW LEVEL SECURITY;
|
|
ALTER TABLE inventory.stock_moves ENABLE ROW LEVEL SECURITY;
|
|
ALTER TABLE inventory.inventory_adjustments ENABLE ROW LEVEL SECURITY;
|
|
|
|
CREATE POLICY tenant_isolation_products ON inventory.products
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
CREATE POLICY tenant_isolation_warehouses ON inventory.warehouses
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
CREATE POLICY tenant_isolation_locations ON inventory.locations
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
CREATE POLICY tenant_isolation_lots ON inventory.lots
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
CREATE POLICY tenant_isolation_pickings ON inventory.pickings
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
CREATE POLICY tenant_isolation_stock_moves ON inventory.stock_moves
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
CREATE POLICY tenant_isolation_inventory_adjustments ON inventory.inventory_adjustments
|
|
USING (tenant_id = get_current_tenant_id());
|
|
|
|
-- =====================================================
|
|
-- COMENTARIOS
|
|
-- =====================================================
|
|
|
|
COMMENT ON SCHEMA inventory IS 'Schema de gestión de inventarios, productos, almacenes y movimientos';
|
|
COMMENT ON TABLE inventory.products IS 'Productos (almacenables, consumibles, servicios)';
|
|
COMMENT ON TABLE inventory.product_variants IS 'Variantes de productos (color, talla, etc.)';
|
|
COMMENT ON TABLE inventory.warehouses IS 'Almacenes físicos';
|
|
COMMENT ON TABLE inventory.locations IS 'Ubicaciones dentro de almacenes (estantes, zonas, etc.)';
|
|
COMMENT ON TABLE inventory.stock_quants IS 'Cantidades en stock por producto/ubicación/lote';
|
|
COMMENT ON TABLE inventory.lots IS 'Lotes de producción y números de serie';
|
|
COMMENT ON TABLE inventory.pickings IS 'Albaranes de entrada, salida y transferencia';
|
|
COMMENT ON TABLE inventory.stock_moves IS 'Movimientos individuales de inventario';
|
|
COMMENT ON TABLE inventory.inventory_adjustments IS 'Ajustes de inventario (conteos físicos)';
|
|
COMMENT ON TABLE inventory.inventory_adjustment_lines IS 'Líneas de ajuste de inventario';
|
|
|
|
-- =====================================================
|
|
-- VISTAS ÚTILES
|
|
-- =====================================================
|
|
|
|
-- Vista: stock_by_product (Stock por producto)
|
|
CREATE OR REPLACE VIEW inventory.stock_by_product_view AS
|
|
SELECT
|
|
p.id AS product_id,
|
|
p.code AS product_code,
|
|
p.name AS product_name,
|
|
l.id AS location_id,
|
|
l.complete_name AS location_name,
|
|
COALESCE(SUM(sq.quantity), 0) AS quantity,
|
|
COALESCE(SUM(sq.reserved_quantity), 0) AS reserved_quantity,
|
|
COALESCE(SUM(sq.available_quantity), 0) AS available_quantity
|
|
FROM inventory.products p
|
|
CROSS JOIN inventory.locations l
|
|
LEFT JOIN inventory.stock_quants sq ON sq.product_id = p.id AND sq.location_id = l.id
|
|
WHERE p.product_type = 'storable'
|
|
AND l.location_type = 'internal'
|
|
GROUP BY p.id, p.code, p.name, l.id, l.complete_name;
|
|
|
|
COMMENT ON VIEW inventory.stock_by_product_view IS 'Vista de stock disponible por producto y ubicación';
|
|
|
|
-- =====================================================
|
|
-- FIN DEL SCHEMA INVENTORY
|
|
-- =====================================================
|