Lógica Backend
Motor de disponibilidad, callouts, procesos y validaciones por evento.
Motor de Disponibilidad
AppointmentAvailabilityEngine concentra la lógica que decide si una cita o un slot es válido para un empleado, servicio y fecha determinados.
Responsabilidades actuales
- Cargar una configuración activa por organización con fallback a organización
0. - Leer duración, precio e intervalo del servicio.
- Aplicar overrides de precio y duración por empleado-servicio.
- Validar que el empleado pueda prestar el servicio.
- Validar días libres de la organización.
- Validar días libres del empleado.
- Validar cobertura del horario semanal del empleado.
- Detectar conflictos con citas existentes pendientes o aprobadas.
- Aplicar límites diarios por empleado cuando correspondan.
- Crear slots persistentes para la UI de iDempiere.
Regla de conflicto
Una cita se considera en conflicto cuando otra cita activa del mismo empleado se cruza con el rango nuevo y tiene estado pendiente o aprobado.
existing.StartDateTime < newEnd
AND existing.EndDateTime > newStart
AND existing.Status IN ('PE', 'AP')
AND existing.IsActive = 'Y'
Generación de slots
La generación actual es específica por empleado. El proceso no genera para todos los empleados ni autoasigna el recurso; parte de una selección explícita de empleado, servicio y fecha.
Nota: esta decisión hace que el flujo de UI sea más claro: primero se elige quién atiende, luego qué servicio presta, después se generan horarios disponibles.
Reglas actuales vs reglas esperadas
| Regla | Estado actual | Comportamiento esperado |
|---|---|---|
| Empleado presta servicio | Implementado con CDS_EmployeeService. | Se mantiene como regla obligatoria. |
| Día libre de organización | Implementado por organización. Aunque CDS_OrgDayOff tiene CDS_AppointmentConfig_ID, el motor actual no filtra por configuración. | Debe filtrar por configuración cuando el servicio resuelva una configuración específica. |
| Día libre de empleado | Implementado con CDS_EmployeeDayOff. | Se mantiene como bloqueo específico del empleado. |
| Horario del empleado | Implementado con CDS_EmployeeScheduleLine. | Debe validarse después del horario macro de la configuración. |
| Horario macro de agenda | Pendiente. No existe aún una tabla equivalente a horario de configuración. | Debe definir los días y rangos generales permitidos, por encima del horario del empleado. |
| Anticipación mínima | Pendiente. Existe MinAdvanceHours, pero no se valida todavía. | Debe bloquear citas creadas con menos horas de anticipación que las configuradas. |
| Anticipación máxima | Pendiente. Existe MaxAdvanceDays, pero no se valida todavía. | Debe bloquear citas demasiado futuras. |
| Conflictos con citas existentes | Implementado para estados Pending y Approved. | Se mantiene como validación obligatoria. |
| Aprobación automática o forzada | Pendiente. Existen campos de configuración y servicio, pero no se aplican en el evento. | Debe resolverse en BeforeNew usando configuración y servicio. |
Roadmap del motor
- Resolver la configuración desde el servicio seleccionado, no solo desde la organización.
- Agregar validación de horario macro de configuración antes del horario del empleado.
- Aplicar
MinAdvanceHoursyMaxAdvanceDaysal generar slots y al guardar citas. - Filtrar días libres de organización por configuración cuando exista relación.
- Aplicar reglas de aprobación en el evento antes de validar el estado final de la cita.
Callouts y Flujo de UI
CalloutAppointment ayuda al usuario a capturar una cita desde iDempiere sin reemplazar la validación final del backend.
Comportamiento de callouts
| Campo | Comportamiento |
|---|---|
CDS_AppointmentService_ID | Copia precio y duración efectivos. Si existe empleado seleccionado, usa override de CDS_EmployeeService. Limpia slot y horas, pero no limpia empleado. |
AppointmentDate | Limpia slot, hora inicial y hora final. No limpia empleado ni servicio. |
CDS_AppointmentSlot_ID | Copia empleado, hora inicial, hora final, precio y duración desde el slot seleccionado. |
Flujo recomendado en ventana
- Seleccionar el empleado que atenderá la cita.
- Seleccionar el servicio filtrado para ese empleado.
- Seleccionar la fecha de la cita.
- Ejecutar el proceso de generación de slots.
- Seleccionar el slot generado.
- Guardar la cita.
Importante: si un usuario modifica manualmente horas, precio o duración, el evento de modelo vuelve a validar la consistencia antes de guardar.
Procesos y Eventos
El plugin mantiene pocos procesos activos y concentra las reglas críticas en eventos de modelo.
Procesos activos
| Proceso | Uso |
|---|---|
GenerateAppointmentSlots | Genera registros CDS_AppointmentSlot para el empleado, servicio y fecha seleccionados. Puede resolver valores desde la cita actual cuando se ejecuta desde la ventana. |
ReopenAppointment | Reabre una cita procesada cambiándola a pendiente y limpiando Processed mediante un contexto controlado. |
Proceso GenerateAppointmentSlots
Este proceso se usa para crear horarios disponibles persistentes que luego pueden consultarse desde la ventana iDempiere o desde la API leyendo la tabla CDS_AppointmentSlot.
| Parámetro | Requerido | Uso |
|---|---|---|
AD_Org_ID | Sí, directo o resuelto desde la cita | Organización para cargar configuración y validar disponibilidad. |
CDS_AppointmentEmployee_ID | Sí | Empleado para el cual se generan slots. |
CDS_AppointmentService_ID | Sí | Servicio que define duración, precio e intervalo del slot. |
AppointmentDate | Sí | Fecha para la cual se generan horarios disponibles. |
CDS_Appointment_ID | Opcional | Si se ejecuta desde una cita, permite resolver organización, empleado, servicio y fecha desde el registro actual. |
Antes de generar nuevos slots, el proceso desactiva slots previos activos del usuario actual. Si hay una cita, desactiva los slots asociados a esa cita. Si no hay cita, desactiva slots del mismo usuario, empleado, servicio y fecha.
Uso recomendado: desde la ventana de cita se selecciona empleado, servicio y fecha; luego se ejecuta GenerateAppointmentSlots; finalmente se selecciona un registro de CDS_AppointmentSlot.
Proceso ReopenAppointment
Este proceso reabre una cita procesada para permitir correcciones administrativas controladas.
| Parámetro | Requerido | Uso |
|---|---|---|
CDS_Appointment_ID | Sí, directo o resuelto desde el registro actual | Cita procesada que se desea reabrir. |
Si la cita ya está abierta, el proceso informa que no requiere reapertura. Si está procesada, activa un contexto temporal #CDS_ReopenAppointment, coloca Processed=N y cambia el estado a Pending.
Validación por evento actual
AppointmentModelValidator se ejecuta antes de crear o cambiar una cita. Para citas pendientes o aprobadas valida los datos mínimos y la disponibilidad real.
- Cliente requerido.
- Empleado requerido.
- Servicio requerido.
- Fecha de cita requerida.
- Hora inicial y final requeridas.
- Hora inicial menor que hora final.
- Fecha de cita igual al día de la hora inicial.
- Duración mayor que cero.
- Precio no negativo.
- Empleado habilitado para prestar el servicio.
- Slot consistente con empleado, servicio, horas, duración y precio.
- Sin bloqueo por día libre de organización o empleado.
- Horario cubierto por agenda del empleado.
- Sin conflicto con otra cita activa pendiente o aprobada.
Estados y procesamiento actual
- Si el estado viene vacío, se asigna
Draft. - Los estados finales marcan
Processed = true. - Los estados no finales limpian
Processed. - Una cita procesada no se puede modificar salvo mediante reapertura controlada.
Roadmap de validaciones en eventos
Las siguientes reglas deben agregarse al evento para que el comportamiento sea consistente desde ventanas iDempiere y desde REST/API.
| Validación pendiente | Momento sugerido | Regla esperada |
|---|---|---|
| Resolver configuración desde servicio | BeforeNew y BeforeChange | El servicio debe indicar qué configuración de agenda aplica. La cita usa esa configuración para anticipación, horario macro y aprobación. |
| Servicio con una sola configuración | Al guardar servicio o relación configuración-servicio | Un servicio no debe estar asociado a más de una configuración activa. |
| Anticipación mínima | BeforeNew y generación de slots | Si StartDateTime está antes de ahora + MinAdvanceHours, la cita o slot debe bloquearse. |
| Anticipación máxima | BeforeNew y generación de slots | Si StartDateTime supera hoy + MaxAdvanceDays, la cita o slot debe bloquearse. |
| Horario macro de configuración | BeforeNew, BeforeChange y generación de slots | El horario de agenda general debe contener el rango solicitado antes de evaluar el horario del empleado. |
| Forzar aprobación | BeforeNew | Si la configuración tiene ForceApproval=Y, la cita debe quedar pendiente aunque el servicio permita aprobación automática. |
| Autoaprobar si el servicio lo permite | BeforeNew | Si AutoApproveIfServiceAllows=Y y el servicio no requiere aprobación, la cita puede quedar Approved. |
Roadmap de workflows y correos
Además de las validaciones, falta definir workflows y plantillas de correo para notificar al cliente cuando cambie el estado de una cita.
| Flujo pendiente | Disparo sugerido | Resultado esperado |
|---|---|---|
| Notificar cita aprobada | Cambio de estado a Approved. | Enviar correo al cliente con fecha, hora, servicio y empleado. |
| Notificar cita cancelada | Cambio de estado a Cancelled. | Enviar correo al cliente con el motivo de cancelación cuando exista. |
| Notificar reprogramación | Cambio de fecha, hora o empleado. | Enviar correo con el nuevo horario confirmado. |
| Recordatorio de cita | Proceso programado antes de StartDateTime. | Enviar recordatorio al cliente según configuración. |
También falta crear las plantillas de correo, definir destinatarios, configurar remitente, parametrizar cuándo enviar cada notificación y evitar reenvíos duplicados con campos de control si aplica.
Orden recomendado de validación
- Resolver servicio, empleado y organización.
- Resolver configuración de agenda desde el servicio.
- Aplicar reglas de anticipación mínima y máxima.
- Validar horario macro de configuración.
- Validar días libres de organización vinculados a la configuración.
- Validar relación empleado-servicio.
- Validar horario y días libres del empleado.
- Validar slot seleccionado.
- Validar conflictos con citas existentes.
- Resolver estado de aprobación: pendiente o aprobada.