Cuando trabajas con múltiples paquetes en PHP (como módulos propios), llega un punto donde necesitas debuggear directamente el código del paquete y no una copia dentro de vendor.
Si alguna vez terminaste debuggeando en vendor/ en lugar de tu proyecto real… esto es para ti.
En el desarrollo moderno de aplicaciones PHP con CodeIgniter 4, uno de los mayores retos es mantener un flujo de trabajo eficiente entre desarrollo local, control de versiones y gestión de dependencias.
Este artículo explica una arquitectura híbrida usando Composer + Git + symlinks que permite trabajar módulos de forma profesional sin depender del directorio vendor.
🧠 El problema clásico
Cuando trabajamos con módulos reutilizables (inventarios, CFDI, pagos, etc.), normalmente terminamos instalando todo en vendor/, lo que genera problemas:
Después de varios intentos y errores típicos (puertos, certificados, Docker, etc.), finalmente logré dejar funcionando ONLYOFFICE Document Server en Docker con acceso HTTPS y listo para integrarse con Nextcloud instalado vía Snap. Aquí te dejo el proceso completo con datos genéricos 👇
🚧 Problemas comunes
Error de conexión (connection refused o connection reset)
Certificados SSL no detectados
Confusión entre Apache, Nginx y Docker
Intentar usar HTTPS dentro del contenedor (mala idea 😅)
🧠 Lo importante que debes entender
ONLYOFFICE usa Nginx interno, no Apache
Docker debe correr en HTTP interno
El SSL se maneja mejor fuera del contenedor
Nextcloud Snap ya trae su propio Apache (aislado)
👉 La solución correcta: usar Apache HTTP Server del sistema como proxy inverso
⚙️ Configuración final
🐳 1. Ejecutar ONLYOFFICE en HTTP
docker run -d -p 8080:80 onlyoffice/documentserver
Si ya tienes Nextcloud instalado con Snap, el siguiente paso lógico es convertirlo en una suite completa tipo Google Workspace. Para eso, puedes integrar OnlyOffice, una poderosa herramienta que permite editar documentos Word, Excel y PowerPoint directamente desde tu nube.
¿Qué es OnlyOffice?
OnlyOffice es una suite ofimática online que se integra perfectamente con Nextcloud, permitiendo editar documentos en tiempo real, colaborar con otros usuarios y mantener compatibilidad con formatos de Microsoft Office.
sudo snap set nextcloud ports.http=80
sudo snap set nextcloud ports.https=443
sudo ufw allow 8080
Recomendaciones
🔒 Usa HTTPS (Let’s Encrypt)
⚡ Usa mínimo 2GB de RAM
🌐 Configura dominio si lo usarás fuera de red local
📦 Usa Docker para mayor estabilidad
Conclusión
Integrar OnlyOffice con Nextcloud transforma tu servidor en una plataforma completa de productividad, similar a Google Drive o Microsoft 365, pero con control total sobre tus datos.
Apóyame
Si esta guía te fue útil y quieres apoyar más contenido como este:
Si quieres tener tu propia nube privada tipo Google Drive o Dropbox, Nextcloud es una de las mejores opciones. En esta guía te mostraré cómo instalarlo en Ubuntu Server 24.04 de forma sencilla, incluyendo un script automático que hace casi todo por ti.
¿Qué es Nextcloud?
Nextcloud es una plataforma de almacenamiento en la nube de código abierto que te permite guardar archivos, sincronizarlos entre dispositivos, compartirlos y mucho más, todo en tu propio servidor.
📁 Almacenamiento privado
🔄 Sincronización de archivos
🔐 Control total de tus datos
👥 Compartir archivos fácilmente
Requisitos
Servidor con Ubuntu Server 24.04
Acceso SSH con permisos sudo
Conexión a internet
Instalación automática (Script)
Para facilitar todo el proceso, puedes usar el siguiente script que instala Apache, PHP, MariaDB y Nextcloud automáticamente.
#!/bin/bash
# ==========================================
# Instalador automático de Nextcloud
# Ubuntu Server 24.04
# Adaptado por julio101290
# ==========================================
set -e
DB_NAME="nextcloud"
DB_USER="ncuser"
DB_PASS="ncpassword123"
NC_DIR="/var/www/nextcloud"
echo "🔄 Actualizando sistema..."
sudo apt update && sudo apt upgrade -y
echo "🌐 Instalando Apache..."
sudo apt install apache2 -y
echo "🐘 Instalando PHP y extensiones..."
sudo apt install -y php php-cli php-common php-mysql php-zip php-gd php-mbstring php-curl php-xml php-bcmath php-intl php-imagick php-gmp libapache2-mod-php
echo "🗄️ Instalando MariaDB..."
sudo apt install mariadb-server -y
echo "🔐 Configurando base de datos..."
sudo mysql -e "CREATE DATABASE ${DB_NAME};"
sudo mysql -e "CREATE USER '${DB_USER}'@'localhost' IDENTIFIED BY '${DB_PASS}';"
sudo mysql -e "GRANT ALL PRIVILEGES ON ${DB_NAME}.* TO '${DB_USER}'@'localhost';"
sudo mysql -e "FLUSH PRIVILEGES;"
echo "⬇️ Descargando Nextcloud..."
cd /var/www/
sudo wget https://download.nextcloud.com/server/releases/latest.zip
sudo apt install unzip -y
sudo unzip latest.zip
sudo chown -R www-data:www-data nextcloud
sudo chmod -R 755 nextcloud
echo "⚙️ Configurando Apache..."
sudo bash -c 'cat > /etc/apache2/sites-available/nextcloud.conf <<EOF
<VirtualHost *:80>
ServerName localhost
DocumentRoot /var/www/nextcloud
<Directory /var/www/nextcloud/>
Require all granted
AllowOverride All
Options FollowSymLinks MultiViews
</Directory>
ErrorLog \${APACHE_LOG_DIR}/nextcloud_error.log
CustomLog \${APACHE_LOG_DIR}/nextcloud_access.log combined
</VirtualHost>
EOF'
sudo a2ensite nextcloud.conf
sudo a2enmod rewrite headers env dir mime
sudo systemctl restart apache2
echo "🧹 Limpiando..."
sudo rm latest.zip
IP=$(hostname -I | awk '{print $1}')
echo "========================================="
echo "✅ INSTALACIÓN COMPLETA"
echo "========================================="
echo "🌐 Accede en tu navegador:"
echo "http://$IP"
echo ""
echo "📌 Base de datos:"
echo "DB: $DB_NAME"
echo "Usuario: $DB_USER"
echo "Password: $DB_PASS"
echo "Host: localhost"
echo ""
echo "⚠️ IMPORTANTE: Completa la instalación en el navegador"
echo "========================================="
Cómo ejecutar el script
Crear el archivo: nano install_nextcloud.sh
Pegar el script y guardar
Dar permisos: chmod +x install_nextcloud.sh
Ejecutar: ./install_nextcloud.sh
Acceso a Nextcloud
Una vez terminado, abre tu navegador y entra a:
http://TU_IP
Desde ahí podrás crear el usuario administrador y conectar la base de datos.
Recomendaciones de seguridad
🔐 Cambiar la contraseña de la base de datos
🔒 Activar HTTPS con Let’s Encrypt
🧱 Configurar firewall (UFW)
💾 Hacer backups regularmente
Conclusión
Con este método puedes tener tu propia nube privada funcionando en pocos minutos. Es ideal para uso personal o incluso para pequeñas empresas que quieran tener control total sobre sus datos.
Apóyame
Si esta guía te fue útil y quieres apoyar más contenido como este, puedes hacerlo aquí:
Resumen: implementación limpia y los bloques de código bien acomodados para que en la vista de saldos solo se muestren los productos/lotes de los almacenes a los que el usuario tiene permiso.
🧾 1) Controller — Preparar empresas y almacenes del usuario
Este bloque es el handler principal (método index()). Se encarga de:
Recuperar empresas asociadas al usuario
Recuperar almacenes activos asignados al usuario
Construir el builder via mdlGetSaldos y preparar respuesta para DataTables (draw/records/paginación)
// Controller: index()
public function index() {
helper('auth');
// 1) Obtener usuario
$idUser = user()->id;
// 2) Empresas del usuario (fallback a [0] si no tiene)
$titulos["empresas"] = $this->empresa->mdlEmpresasPorUsuario($idUser);
$empresasID = count($titulos["empresas"]) === 0 ? [0] : array_column($titulos["empresas"], "id");
// 3) Almacenes (storages) asignados al usuario y activos (status = 'on')
$storagesUser = $this->storagesPerUser
->where("idUsuario", $idUser)
->where("status", "on")
->asArray()
->findAll();
$storagesUser = count($storagesUser) === 0 ? [0] : array_column($storagesUser, "idStorage");
// 4) Si es petición AJAX (DataTables) devolvemos JSON paginado
if ($this->request->isAJAX()) {
$request = service('request');
$draw = (int) $request->getGet('draw');
$start = (int) $request->getGet('start');
$length = (int) $request->getGet('length');
$searchValue = $request->getGet('search')['value'] ?? '';
$orderColumnIndex = (int) ($request->getGet('order')[0]['column'] ?? 0);
$orderDir = $request->getGet('order')[0]['dir'] ?? 'asc';
// Mapeo de columnas (orden)
$fields = [
'id' => 'a.id',
'nombreAlmacen' => 'c.name',
'lote' => 'a.lote',
'codigoProducto' => 'a.codigoProducto',
'descripcion' => 'a.descripcion',
'fullname' => 'e.fullname'
];
$orderField = $fields[$orderColumnIndex] ?? 'id';
// Builder desde el modelo con filtros de empresas y almacenes
$builder = $this->saldos->mdlGetSaldos($empresasID, $storagesUser);
// Conteo total (sin filtros de búsqueda)
$total = clone $builder;
$recordsTotal = $total->countAllResults(false);
// Filtro de búsqueda global
if (!empty($searchValue)) {
$builder->groupStart();
foreach ($fields as $field) {
$builder->orLike($field, $searchValue);
}
$builder->groupEnd();
}
// Conteo filtrado
$filteredBuilder = clone $builder;
$recordsFiltered = $filteredBuilder->countAllResults(false);
// Obtener página
$data = $builder->orderBy("a." . $orderField, $orderDir)
->get($length, $start)
->getResultArray();
// Respuesta JSON para DataTables
return $this->response->setJSON([
'draw' => $draw,
'recordsTotal' => $recordsTotal,
'recordsFiltered' => $recordsFiltered,
'data' => $data,
]);
}
// Vista normal (no-AJAX)
$titulos["title"] = "Info Productos";
$titulos["subtitle"] = "Extrae la información de los productos por el código de barras";
return view('julio101290\\boilerplateinventory\\Views\\saldos', $titulos);
}
🧱 2) Modelo — Builder con filtros por empresa y almacén
Este método devuelve un Query Builder ya filtrado por $idEmpresas y $storagesUser. Úsalo tal cual en el controller.
public function mdlGetSaldos($idEmpresas, $storagesUser) {
return $this->db->table('saldos a')
->select("
a.id,
a.idEmpresa,
a.idAlmacen,
a.idProducto,
a.codigoProducto,
a.lote,
a.descripcion,
a.cantidad,
a.created_at,
a.deleted_at,
a.updated_at,
b.nombre AS nombreEmpresa,
c.name AS nombreAlmacen,
COALESCE(e.fullname, 'Sin asignar') AS fullname
")
// JOINs para mostrar nombres legibles
->join('empresas b', 'a.idEmpresa = b.id')
->join('storages c', 'a.idAlmacen = c.id')
// LEFT JOINs para evitar romper si no hay relación
->join('productsemployes pe', 'pe.idProduct = a.id', 'left')
->join('employes e', 'e.id = pe.idEmploye', 'left')
// filtros de permisos: solo empresas/almacenes permitidos
->whereIn('a.idEmpresa', $idEmpresas)
->whereIn('a.idAlmacen', $storagesUser)
->orderBy('a.id', 'DESC');
}
🛠️ 3) Notas técnicas y buenas prácticas aplicadas
Back-end es la fuente de verdad: los arrays de IDs ($empresasID y $storagesUser) se construyen en el servidor a partir de user()->id. Nunca confíes en listas enviadas por cliente.
Fallback seguro: usar [0] cuando no hay empresas/almacenes evita que whereIn reciba un array vacío y genere errores SQL. En ese caso la consulta devuelve resultados vacíos.
LEFT JOINs: mantuvimos LEFT JOIN en relaciones opcionales para que la consulta no falle si no hay datos relacionados (por ejemplo, empleado no asignado).
Clonar builder: clonar el builder para conteo (countAllResults(false)) mantiene el flujo de DataTables sin re-ejecutar joins innecesarios.
🔍 4) Sugerencias de índices SQL (para rendimiento)
Recomiendo agregar índices (si no existen) para acelerar filtros y joins:
-- Índices recomendados
CREATE INDEX idx_saldos_idEmpresa ON saldos (idEmpresa);
CREATE INDEX idx_saldos_idAlmacen ON saldos (idAlmacen);
CREATE INDEX idx_saldos_codigoProducto ON saldos (codigoProducto);
CREATE INDEX idx_saldos_lote ON saldos (lote);
-- Índices en tablas relacionadas
CREATE INDEX idx_storages_id ON storages (id);
CREATE INDEX idx_empresas_id ON empresas (id);
Si usas PostgreSQL, considera índices compuestos o índices GIN si aplicarás búsquedas textuales complejas.
🧪 5) Pruebas recomendadas (QA) — pasos concretos
Usuario sin almacenes: Inicia sesión con un usuario sin almacenes asignados. La tabla debe venir vacía. Ver el mensaje UX (ver sección UX abajo).
Usuario con 1 almacén: Inicia sesión con acceso a un solo almacén; la vista debe mostrar únicamente los saldos de ese almacén.
Usuario con múltiples almacenes: Comprueba que aparecen filas de cualquiera de esos almacenes, y que no aparecen filas de almacenes no asignados.
Busqueda global: Ejecuta una búsqueda por codigoProducto, lote o descripcion y valida que los resultados respetan el filtro por almacén.
Paginación y orden: Revisa recordsTotal y recordsFiltered cuando aplicas orden y búsqueda; deben reflejar correctamente la cantidad total y la cantidad filtrada.
Seguridad: Intenta manipular parámetros GET/POST desde el cliente (por ejemplo, forzar otro idAlmacen) y verifica que no se muestran saldos si el usuario no tiene permiso.
💬 6) Mensajes UX sugeridos
Si el usuario no tiene almacenes asignados, muestra un mensaje amigable y accionable en la UI (evita pantalla en blanco):
<div class="alert alert-info">
<strong>Sin almacén asignado</strong><br>
No tienes almacenes asignados. Contacta al administrador para que te asigne los permisos necesarios.
</div>
🔐 7) Seguridad y consideraciones adicionales
Recalcula permisos siempre en servidor: no aceptes listas desde cliente.
Validar status: al desactivar un permiso (status != ‘on’), asegúrate que en la siguiente petición el usuario deje de ver los datos correspondientes.
Auditoría: opcionalmente loguea consultas sensibles (quién vio qué y cuándo) para trazabilidad.
📦 8) Release & commit (referencia)
Los cambios fueron incluidos en el release v1.2.3 y el commit con la implementación es este:
Generador Automático de CRUD para CodeIgniter 4 – julio101290/boilerplate
🚀 Generador Automático de CRUD para CodeIgniter 4: Crea Módulos Completos en 1 Minuto 🔥
¿Cansado de escribir el mismo código una y otra vez? ¿Tus proyectos se retrasan por la tediosa creación de modelos, controladores y vistas? ¡Tenemos la solución! Te presento el Generador Automático de CRUD para CodeIgniter 4, una herramienta integrada en mi boilerplate que transforma una tabla de base de datos en un módulo funcional, seguro y profesional en menos de 60 segundos. Ahorra cientos de horas y olvídate de los errores repetitivos.
Es un controlador inteligente que, a partir del nombre de una tabla existente en tu base de datos, genera de forma automática todos los archivos necesarios para un CRUD completo:
✅ Modelo con validaciones y soft delete
✅ Controlador con DataTables server-side
✅ Vistas (listado + modal) integradas con AdminLTE
✅ Archivos de idioma (inglés y español)
✅ Migración lista para ejecutar
✅ Rutas listas para copiar o integradas en tu paquete
✅ Permisos creados automáticamente (RBAC)
Todo esto con código limpio, indentado y siguiendo las mejores prácticas.
💡 Dato curioso: El generador lee la estructura de tu tabla y adapta los campos automáticamente. Si tu tabla tiene campos como created_at, updated_at, deleted_at, los maneja de forma especial para que el soft delete funcione perfectamente.
🏗️ El entorno perfecto: julio101290/boilerplate
Este generador vive dentro de mi fork del excelente boilerplate de agungsugiarto, adaptado y mejorado para proyectos reales. Incluye:
🔹 Lee el composer.json y extrae el namespace PSR-4 automáticamente.
🔹 Crea la estructura src/Models, src/Controllers, etc.
🔹 Actualiza el archivo src/Config/Routes.php del paquete con las nuevas rutas.
🔹 Tu paquete se vuelve autónomo y portable.
🔑 Permisos gestionados como profesionales
Antes: los permisos se creaban en caliente al generar el CRUD (poco ortodoxo). Ahora: cuando el destino es un paquete vendor, el generador actualiza el Seeder correspondiente (ej. BoilerplateCFDIDescargaMasiva.php), añadiendo la línea para crear el permiso y asignarlo al admin.
Así, la instalación de permisos se hace como Dios manda: con php spark db:seed.
⏱️ Ahorro de tiempo real
Tarea
Sin generador
Con generador
Ahorro
CRUD de 10 campos
45-60 min
1 min
~98%
20 tablas por proyecto
15-20 horas
20 minutos
¡Días!
Ese tiempo lo puedes reinvertir en lógica de negocio que realmente aporta valor. Además, todo el código generado sigue el mismo patrón, reduciendo la deuda técnica y facilitando el mantenimiento.
📝 Código completo del generador
Aquí tienes la clase AutoCrudControllerComposer en su versión final. Cópiala directamente en tu proyecto (julio101290/boilerplate/Controllers/).
db = \Config\Database::connect();
$this->authorize = Services::authorization();
$this->users = new UserModel();
helper('utilerias');
}
/**
* Método principal para generar el CRUD
*
* @param string $table Nombre de la tabla
* @param string|null $targetType 'app' o 'vendor' (por GET)
* @param string|null $vendorPackage Paquete vendor
* @param string|null $vendorNamespace Namespace (auto-detected)
*/
public function index($table, $targetType = null, $vendorPackage = null, $vendorNamespace = null)
{
// Leer de GET si no se pasaron como argumentos
if ($targetType === null) {
$targetType = $this->request->getGet('target') ?? 'app';
}
if ($vendorPackage === null && $targetType === 'vendor') {
$vendorPackage = $this->request->getGet('package');
}
if ($vendorNamespace === null && $targetType === 'vendor') {
$vendorNamespace = $this->request->getGet('namespace');
}
$this->targetType = $targetType;
if ($targetType === 'vendor' && $vendorPackage) {
$this->setupVendorPaths($vendorPackage, $vendorNamespace);
}
$this->generateModel($table);
$this->generateController($table);
$this->generateView($table);
$this->generateViewModal($table);
$this->generateLanguage($table);
$this->generateMigration($table);
$this->generateLanguageES($table);
if ($targetType === 'vendor') {
$this->generateVendorRoutesFile($table);
$this->updateSeederPermissions($table);
} else {
$this->generatePermissions($table);
}
$tableUpCase = ucfirst($table);
echo "";
echo "✅ CRUD generado exitosamente en: " . ($targetType === 'vendor' ? $this->vendorPackage : 'app') . "";
echo "";
}
// ... (el resto de métodos: setupVendorPaths, generateModel, generateController, etc.)
// Por brevedad, no repetimos todo el código aquí, pero en el artículo real debes incluir el código completo.
}
?>
⚠️ Nota: El código anterior es un resumen. Para obtener el código completo, visita el repositorio en GitHub o copia el bloque que aparece al final de este artículo.
🔌 Cómo usarlo
Agrega la ruta en app/Config/Routes.php: $routes->get('generateCRUDComposer/(:any)', 'julio101290\boilerplate\Controllers\AutoCrudControllerComposer::index/$1');
Genera un CRUD en app: http://tusitio.com/generateCRUDComposer/nombre_tabla
Genera en tu paquete vendor: http://tusitio.com/generateCRUDComposer/nombre_tabla?target=vendor&package=tu/paquete
¡Y listo! En segundos tendrás todo el código listo para usar.
🎯 Conclusión y llamado a la acción
El generador automático de CRUD ha evolucionado de un simple script a una herramienta profesional que:
Se ha implementado una mejora importante en el manejo de DataTables con procesamiento server-side para el módulo de Bitácora en el proyecto jcposUltimate, resolviendo problemas de ordenamiento, paginación y búsqueda que impedían un funcionamiento correcto con MariaDB / MySQL.
Esta guía detalla el despliegue del cliente de base de datos, componente esencial para conectar SAP B1, Crystal Reports, Excel y Power BI.
1 Ubicación del Kit de Instalación
En entornos de SAP Business One, el instalador reside dentro de los paquetes de datos del servidor. No busque instaladores genéricos; navegue a la siguiente ruta:
…\HANA 2.0 rev 56\DATA_UNITS\SAP HANA CLIENT 2.0 FOR B1
Dentro encontrará la arquitectura dual de SAP:
📂 NT_X64: Carpeta exclusiva para Windows (64 bits).
📂 LINX64SUSE: Carpeta para servidores Linux/SUSE.
2 Instalación Paso a Paso
Acceda a la carpeta NT_X64.
Localice el ejecutable hdbinst.exe.
Derechos de Administrador: Haga clic derecho y seleccione “Ejecutar como administrador”.
Consola: Presione Enter para aceptar la ruta por defecto C:\hdbclient.
⚠️ Verificación: La instalación finaliza correctamente cuando visualiza el mensaje: Installation done.
3 Configuración del PATH
Para ejecutar herramientas como hdbsql desde cualquier carpeta, configure la variable de entorno:
Busque “Variables de Entorno” en el menú Inicio.
En Variables del sistema, edite la variable Path.
Agregue una nueva línea con: C:\hdbclient.
4 Activación ODBC (64 bits)
Para habilitar la conexión en aplicaciones de terceros:
Abra el Administrador de orígenes de datos ODBC (64 bits).
Vaya a DSN de sistema > Agregar.
Seleccione el controlador HDBODBC.
Configure la IP del servidor y el puerto (ej: 30015).
// Prueba de fuego en CMDhdbsql -n 192.168.X.X:30015 -u SYSTEM -p Password123 "SELECT * FROM DUMMY"
🚀 Troubleshooting Rápido
¿No ve el driver? Instale “Visual C++ Redistributable 2013/2015”.
¿Error -10709? Verifique el Firewall del servidor en el puerto 30015.
Este manual técnico explica cómo enlazar sucursales locales con sucursales de SAP Business One (OBPL) utilizando ODBC (eODBC) sobre SAP HANA, evitando el uso del Service Layer para consultas de alto rendimiento.
¿Qué problema resuelve esta implementación?
En muchos proyectos con SAP Business One, el uso del Service Layer para catálogos o consultas simples introduce latencia innecesaria. Este módulo soluciona ese problema permitiendo:
Relacionar sucursales locales con sucursales SAP
Consultar la tabla OBPL directamente desde SAP HANA
Reducir tiempos de respuesta usando ODBC
Eliminar dependencias innecesarias del Service Layer
¿Por qué usar ODBC en SAP Business One?
El uso de ODBC en SAP HANA es ideal para consultas de solo lectura, catálogos y validaciones rápidas.
Característica
Service Layer
ODBC SAP HANA
Rendimiento
Medio
Alto
Latencia
Alta
Baja
Ideal para
CRUD complejo
Catálogos y consultas
Dependencia SAP
Alta
Media
Estructura del módulo de enlace SAP
La solución se divide en cuatro componentes principales:
Tabla de enlace entre sucursal local y SAP
Controlador con consultas ODBC
Vista con DataTables server-side
Formulario con Select2 dinámico
Tabla link_sap_branchoffice
Esta tabla permite mantener la relación entre el sistema local y SAP Business One sin duplicar información de SAP.
Campo
Descripción
id
ID interno
idEmpresa
Empresa local
idBranchOffice
Sucursal local
idBranchOfficeSAP
Sucursal SAP (OBPL.BPLId)
created_at
Fecha de creación
updated_at
Fecha de actualización
deleted_at
Borrado lógico
Conexión a SAP HANA usando ODBC
La conexión ODBC se realiza directamente contra SAP HANA, lo que permite consultas rápidas y estables.
Este paso es obligatorio para evitar errores como invalid table name en SAP HANA.
Consulta optimizada a la tabla OBPL
Para obtener una sucursal SAP se realiza una consulta directa a la tabla OBPL:
SELECT
"BPLId",
"BPLName"
FROM OBPL
WHERE "Disabled" <> 'Y'
AND "BPLId" = ?
Consulta directa
Sin bucles innecesarios
Ideal para formularios y edición
Listado con DataTables server-side
El módulo utiliza DataTables en modo server-side para manejar grandes volúmenes de datos:
Paginación eficiente
Búsqueda global
Ordenamiento dinámico
Filtrado por empresa del usuario
Select2 dinámico para sucursales
Sucursal local
Se cargan las sucursales locales vía AJAX usando Select2.
Sucursal SAP (OBPL)
Las sucursales SAP se consultan directamente desde SAP HANA vía ODBC, sin Service Layer.
Beneficios clave de esta arquitectura
Mayor rendimiento en SAP Business One
Menor latencia en consultas
Menos carga en el Service Layer
Arquitectura escalable y mantenible
Buenas prácticas recomendadas
Usar ODBC solo para consultas de lectura
Usar Service Layer para escrituras en SAP
Centralizar credenciales ODBC
Siempre definir el schema en SAP HANA
Conclusión
Este enfoque permite integrar SAP Business One con sistemas locales de forma eficiente, utilizando ODBC sobre SAP HANA como alternativa rápida y estable al Service Layer.
Ideal para catálogos, validaciones, reportes y sistemas híbridos SAP.
Usamos cookies en nuestro sitio web para brindarle la experiencia más relevante recordando sus preferencias y visitas repetidas. Al hacer clic en "Aceptar", acepta el uso de TODAS las cookies.
This website uses cookies to improve your experience while you navigate through the website. Out of these cookies, the cookies that are categorized as necessary are stored on your browser as they are essential for the working of basic functionalities of the website. We also use third-party cookies that help us analyze and understand how you use this website. These cookies will be stored in your browser only with your consent. You also have the option to opt-out of these cookies. But opting out of some of these cookies may have an effect on your browsing experience.
Necessary cookies are absolutely essential for the website to function properly. This category only includes cookies that ensures basic functionalities and security features of the website. These cookies do not store any personal information.
Any cookies that may not be particularly necessary for the website to function and is used specifically to collect user personal data via analytics, ads, other embedded contents are termed as non-necessary cookies. It is mandatory to procure user consent prior to running these cookies on your website.