Herramientas Informaticas

Autor: juliocesar20200413 Página 1 de 142

Desarrollador web apasionado y gusto por la buena musica

🚀 Apache NetBeans 29: ¿Qué mejoras trae en Git y vale la pena actualizar? 🤔

Entrada fija

Si eres desarrollador 👨‍💻 y trabajas todos los días con Git, seguramente te interesa saber qué cambió en Apache NetBeans 29.

Aunque a simple vista no parece una actualización “wow” ✨, sí trae mejoras importantes que hacen que trabajar con repositorios sea más rápido, estable y menos frustrante 😌.

Aquí te lo explico fácil y directo 👇

🔧 1. Mejor compatibilidad con Git moderno

Git ha cambiado bastante con los años.

Antes era común ver repositorios con la rama principal llamada:

master

Ahora la mayoría usa:

main

NetBeans 29 mejora la compatibilidad con estas configuraciones modernas ✅

¿Qué mejora?

  • ✔ Detecta mejor las ramas nuevas
  • ✔ Menos errores con repositorios recientes
  • ✔ Mejor soporte para nuevas configuraciones

👉 Si clonas proyectos modernos, todo funciona más fluido.


🔐 2. Mejor autenticación con GitHub (HTTPS y SSH)

GitHub ya no permite usar usuario y contraseña para hacer push o pull por HTTPS ❌

Ahora se usa:

  • 🔹 Token personal (PAT)
  • 🔹 Llave SSH

NetBeans 29 mejora ambos métodos 👇

📌 HTTPS con Token

Más estabilidad al hacer:

  • ✅ Push
  • ✅ Pull
  • ✅ Clone
  • ✅ Fetch

Menos errores de credenciales 🙌

🔑 SSH más confiable

Mejor soporte para llaves como:

id_rsa
id_ed25519

👉 Si ya configuraste tu carpeta .ssh, todo debería funcionar mejor.


⚡ 3. Más rápido en proyectos grandes

Si trabajas con proyectos pesados en:

  • 🐘 PHP
  • ☕ Java
  • 🟨 JavaScript

Seguro has notado lentitud al revisar cambios.

NetBeans 29 optimiza:

  • ✅ Status
  • ✅ Diff
  • ✅ Commit
  • ✅ Historial

👉 Menos tiempo esperando ⏳


🔍 4. Mejor comparación de archivos (Diff)

La herramienta para comparar archivos ahora es más confiable.

Te ayuda a ver mejor:

  • 🟢 líneas agregadas
  • 🔴 líneas eliminadas
  • 🟡 líneas modificadas

Y también mejora la resolución de conflictos 🔥

Ideal si trabajas en equipo 👥


🌐 5. Mejor soporte para otras plataformas Git

No todos usan GitHub.

Ahora NetBeans 29 trabaja mejor con:

  • 🦊 GitLab
  • 🏔 Codeberg
  • 🏢 servidores privados

👉 Más compatibilidad con repos remotos.


❌ Lo que NO cambió

Para no vender humo 😅

NetBeans 29 todavía NO tiene:

  • ❌ Login directo con GitHub tipo VS Code
  • ❌ Panel para Pull Requests
  • ❌ Integración con Issues
  • ❌ Marketplace de extensiones como VS Code

Sigue siendo una experiencia más clásica.


🤔 Entonces… ¿vale la pena actualizar?

Si usas Git diario:

✅ Sí vale la pena.

Porque aunque no se vea “bonito”, sí mejora lo importante:

  • 🚀 Más rápido
  • 🔐 Más estable
  • ⚡ Menos errores
  • 🌐 Mejor compatibilidad

Si vienes de NetBeans 26 o menor, sí notarás diferencia.


🧠 Conclusión

Apache NetBeans 29 no reinventa Git…

Pero sí hace que trabajar con repositorios sea:

  • ✔ más sólido
  • ✔ más confiable
  • ✔ más rápido

Y muchas veces eso vale más que una interfaz bonita 😎

Cómo usar Composer con symlinks para desarrollo local en CodeIgniter 4

Entrada fija

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.


🔥 El problema

Por defecto, Composer:

  • Descarga paquetes desde GitHub
  • Los copia en vendor/
  • Xdebug trabaja sobre esa copia

Resultado:

  • Código duplicado
  • Cambios no se reflejan
  • Debug incómodo

🧠 La solución: usar type: path

{
  "repositories": [
    {
      "type": "path",
      "url": "../boilerplateproducts",
      "options": {
        "symlink": true
      }
    }
  ]
}

⚙️ ¿Qué hace esto?

  • Usa tu carpeta local
  • No descarga desde GitHub
  • Crea un symlink en vendor

🔍 Verificar que funciona

composer update

Debes ver:

Installing julio101290/boilerplateproducts: Symlinking from ../boilerplateproducts

Y luego:

ls -l vendor/julio101290/boilerplateproducts

Resultado esperado:

boilerplateproducts -> ../../../boilerplateproducts

🚀 Beneficios

  • Debug directo en tu código
  • Cambios en tiempo real
  • Sin duplicación
  • Flujo más rápido

⚠️ Error común

No mezcles path y vcs:

{
  "type": "path",
  "url": "../boilerplateproducts"
},
{
  "type": "vcs",
  "url": "https://github.com/usuario/boilerplateproducts"
}

Composer puede ignorar el local y usar GitHub.


✅ Desarrollo correcto

rm -rf vendor composer.lock
composer update

🟢 Desarrollo vs Producción

Desarrollo

  • path
  • symlink
  • debug directo

Producción

  • vcs
  • descarga desde GitHub
  • entorno limpio

⚡ Scripts útiles

dev.sh

#!/bin/bash
rm -rf vendor composer.lock
COMPOSER=composer.local.json composer update

prod.sh

#!/bin/bash
rm -rf vendor composer.lock
composer update

🧠 Tip PRO

{
  "repositories": [
    {
      "type": "path",
      "url": "../boilerplateproducts",
      "options": { "symlink": true }
    }
  ],
  "config": {
    "preferred-install": "source"
  }
}

Ejecutar:

COMPOSER=composer.local.json composer update

💡 IDE

Puedes integrar esto con NetBeans usando scripts o herramientas externas.


✅ Conclusión

  • No debuggees código en vendor
  • Usa symlink
  • Automatiza tu flujo

Resultado: desarrollo más rápido y limpio 🚀

🚀 Solución definitiva al error “ONLYOFFICE Document Server is unavailable” en Nextcloud (Snap + Docker)

Entrada fija

¿Tu integración de ONLYOFFICE en Nextcloud se desconecta sola y tienes que estar guardando la configuración a cada rato? 😤
Tranquilo, no eres el único. Después de muchas pruebas, encontré un script automático que simula hacer clic en “Guardar” y mantiene la conexión viva. Y lo mejor: se ejecuta solo a la medianoche para no molestar a los usuarios. 🌙

🔍 ¿Qué causa el error?

Si usas Nextcloud instalado como Snap y el ONLYOFFICE Document Server en Docker (con dominio DDNS y puerto personalizado), el problema suele ser que el servidor de documentos no puede descargar los archivos desde la URL pública. Al guardar manualmente la configuración, todo vuelve a funcionar… hasta que falla otra vez.

✅ La solución: un script + cron nocturno

Este script hace exactamente lo mismo que tú harías en la interfaz web: establece la URL del Document Server, la URL interna (StorageUrl), desactiva la verificación SSL y verifica la conexión. Lo programamos con cron para que se ejecute cada noche a las 12:00 AM, cuando nadie está usando el sistema.

📝 Paso 1: Crear el script

Abre una terminal en tu servidor Ubuntu y crea el archivo:

sudo nano /usr/local/bin/onlyoffice_refresh.sh

Copia y pega este contenido (ajusta las URLs a las tuyas):

#!/bin/bash

# Script para guardar la configuración de ONLYOFFICE exactamente como en la interfaz web
# Esto simula hacer clic en "Save"

NEXTCLOUD_OCC="/snap/bin/nextcloud.occ"

# Valores según tu configuración (¡cámbialos si es necesario!)
DOCUMENT_SERVER_URL="https://tudominio.dyndns.org:4445/"
STORAGE_URL="https://tudominio.dyndns.org:444/"   # O usa la IP interna si funciona mejor
VERIFY_PEER_OFF="true"

# Establecer DocumentServerUrl
$NEXTCLOUD_OCC config:app:set onlyoffice DocumentServerUrl --value="$DOCUMENT_SERVER_URL"

# Establecer StorageUrl (dirección para peticiones internas desde ONLYOFFICE Docs)
$NEXTCLOUD_OCC config:app:set onlyoffice StorageUrl --value="$STORAGE_URL"

# Desactivar verificación de certificados
$NEXTCLOUD_OCC config:app:set onlyoffice verify_peer_off --value="$VERIFY_PEER_OFF"

# Limpiar InternalServerUrl (lo dejamos vacío)
$NEXTCLOUD_OCC config:app:delete onlyoffice InternalServerUrl >/dev/null 2>&1

# Verificar conexión
echo "Configuración aplicada. Verificando conexión..."
$NEXTCLOUD_OCC onlyoffice:documentserver --check

echo "Listo. Configuración guardada."

⚠️ Importante: Reemplaza tudominio.dyndns.org y los puertos por los tuyos. Si tu Nextcloud usa HTTP, cambia https por http. Si no usas puerto 444, ajústalo.

🔧 Paso 2: Dar permisos de ejecución

sudo chmod +x /usr/local/bin/onlyoffice_refresh.sh

🧪 Paso 3: Probar manualmente

sudo /usr/local/bin/onlyoffice_refresh.sh

Deberías ver un mensaje como Document Server is reachable. Si falla, revisa las URLs.

⏲️ Paso 4: Programar la tarea nocturna (cron)

sudo crontab -e

Agrega esta línea para que se ejecute todos los días a las 12:00 AM:

0 0 * * * /usr/local/bin/onlyoffice_refresh.sh >> /var/log/onlyoffice_refresh.log 2>&1

Con esto, cada medianoche el script se ejecutará y dejará la conexión lista para el día siguiente. Además, guarda un log en /var/log/onlyoffice_refresh.log por si quieres revisar que todo vaya bien.

🎯 ¿Por qué funciona?

El script fuerza la reescritura de los parámetros clave (DocumentServerUrl, StorageUrl, verify_peer_off) exactamente como lo harías desde el panel de administración de Nextcloud. Al ejecutarse una vez al día, evita que la conexión se caiga durante la jornada laboral. Es una solución sencilla, automática y sin impacto en el rendimiento.

🧠 Consejos adicionales

  • Si la conexión se sigue cayendo varias veces al día, puedes aumentar la frecuencia del cron a cada hora (0 * * * *) o cada 30 minutos (*/30 * * * *). El script es muy liviano.
  • Para una solución definitiva (sin cron), configura correctamente la variable StorageUrl apuntando a la IP interna de tu servidor (ej. http://192.168.0.1). Pero este script es un gran parche mientras encuentras la IP perfecta.
  • Mantén actualizados Nextcloud Snap, el conector ONLYOFFICE y el contenedor Docker del Document Server.

📢 ¿Te funcionó?
Si este tutorial te ayudó, comparte y comenta. Entre todos podemos hacer que Nextcloud + ONLYOFFICE funcione de maravilla. 💪


Etiquetas SEO sugeridas: Nextcloud ONLYOFFICE error, Document Server unavailable, Snap Nextcloud, Docker ONLYOFFICE, cron job, solución Nextcloud

Meta descripción: ¿El ONLYOFFICE de Nextcloud se desconecta solo? Aprende a crear un script automático que guarda la configuración cada noche y olvídate del error. Paso a paso para Snap y Docker.

🔧 Arquitectura híbrida con Composer + Git: desarrollo profesional de módulos en CodeIgniter 4

Entrada fija

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:

  • No se puede editar código fácilmente
  • Debug complicado
  • Cambios se pierden con composer update
  • Duplicación de código
  • Dificultad para mantener múltiples proyectos

💡 La solución: arquitectura híbrida

La solución moderna combina tres herramientas:

  • Composer como gestor de dependencias
  • Git como control de versiones
  • Symlinks para desarrollo en vivo

Estructura típica:

~/fuentes/
   ci4-project/
   boilerplateInventory/
   boilerplateProducts/

⚙️ Configuración en Composer

En el composer.json del proyecto principal:

"repositories": [
  {
    "type": "path",
    "url": "../boilerplateInventory",
    "options": {
      "symlink": true
    }
  },
  {
    "type": "vcs",
    "url": "https://github.com/julio101290/boilerplateInventory"
  }
]

📁 Path repository (desarrollo local)

Este tipo de repositorio indica a Composer que use una carpeta local en lugar de descargar el paquete.

Beneficios:

  • Cambios en tiempo real
  • Debug directo
  • No depende de vendor

🌐 VCS repository (GitHub)

Sirve como respaldo cuando el path no existe o en servidores de producción.


🔥 El problema de versiones

Muchos módulos requieren:

^1.0.0

Pero en desarrollo usamos:

dev-main

Esto genera conflictos.


🛠️ Solución: alias de versión

La solución correcta es:

"julio101290/boilerplateinventory": "dev-main as 1.2.6"

Esto le dice a Composer que trate el código local como versión estable.


⚡ Resultado del sistema

  • ✔ Desarrollo en vivo
  • ✔ Debug funcional
  • ✔ Compatibilidad con dependencias
  • ✔ Sin tocar vendor
  • ✔ Respaldo en GitHub

🧪 Flujo de trabajo real

cd ~/fuentes/boilerplateInventory
git add .
git commit -m "cambios"
git push

Los cambios se reflejan automáticamente en el proyecto principal.


⚠️ Buenas prácticas

  • Usar nombres consistentes en minúsculas
  • No trabajar dentro de vendor
  • Versionar módulos con tags
git tag v1.2.6
git push origin v1.2.6

🚀 Conclusión

Esta arquitectura permite construir sistemas modulares profesionales con:

  • Desarrollo rápido
  • Debug en vivo
  • Control de versiones limpio
  • Escalabilidad real

Es una forma moderna de trabajar con Composer sin depender de vendor como entorno de desarrollo.

🔥 Cómo configurar ONLYOFFICE en Docker con SSL y Nextcloud Snap (guía práctica sin errores) 🔥

Entrada fija

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

🔐 2. Generar certificado SSL con Certbot

sudo snap stop nextcloud
sudo certbot certonly --standalone -d tudominio.com
sudo snap start nextcloud

👉 Los certificados quedarán en algo como:

/etc/letsencrypt/live/tudominio.com-0001/

🌐 3. Configurar Apache como proxy SSL

Archivo:

/etc/apache2/sites-available/onlyoffice.conf

Contenido:

<VirtualHost *:4443>
    ServerName tudominio.com

    SSLEngine on
    SSLCertificateFile /etc/letsencrypt/live/tudominio.com-0001/fullchain.pem
    SSLCertificateKeyFile /etc/letsencrypt/live/tudominio.com-0001/privkey.pem

    ProxyPreserveHost On
    ProxyPass / http://localhost:8080/
    ProxyPassReverse / http://localhost:8080/

    RequestHeader set X-Forwarded-Proto "https"
</VirtualHost>

🔌 4. Habilitar el puerto en Apache

Editar:

/etc/apache2/ports.conf

Agregar:

Listen 4443

⚡ 5. Activar módulos necesarios

sudo a2enmod ssl proxy proxy_http headers

🚀 6. Activar sitio y reiniciar

sudo a2ensite onlyoffice.conf
sudo apachectl configtest
sudo systemctl restart apache2

🔥 7. Abrir firewall

sudo ufw allow 4443

✅ Resultado

Acceso funcionando en:

👉 https://tudominio.com:4443

✔ SSL válido
✔ Proxy funcionando
✔ Docker respondiendo correctamente
✔ Listo para integrarse con Nextcloud


🧪 Pruebas clave

curl http://localhost:8080/healthcheck   # → true
curl -k https://localhost:4443           # → 302 (correcto)

🧠 Conclusión

  • ❌ No uses SSL dentro de Docker
  • ✅ Usa proxy inverso en el host
  • ✅ Mantén servicios separados
  • ✅ Evita conflictos con Snap

🚀 Siguiente paso

Integrar ONLYOFFICE con Nextcloud usando JWT correctamente 🔐


Si estás montando tu propio entorno, esta arquitectura te va a ahorrar horas de debugging 💻🔥

Apóyame

Si esta guía te fue útil y quieres apoyar más contenido como este:

👉 https://www.patreon.com/u74078772?

Cómo instalar OnlyOffice en Nextcloud (instalado con Snap) – Guía completa

Entrada fija

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.

  • 📝 Editor de documentos (Word)
  • 📊 Hojas de cálculo (Excel)
  • 📽️ Presentaciones (PowerPoint)
  • 👥 Edición colaborativa en tiempo real

Requisitos

  • Nextcloud instalado vía Snap
  • Servidor Ubuntu (recomendado 2GB RAM mínimo)
  • Acceso SSH con sudo

Instalar OnlyOffice Document Server con Docker

1. Instalar Docker


sudo apt update
sudo apt install docker.io -y
sudo systemctl enable --now docker
  

2. Ejecutar OnlyOffice


sudo docker run -i -t -d -p 8080:80 \
--restart=always \
onlyoffice/documentserver
  

Esto levantará OnlyOffice en:

http://TU_IP:8080

Instalar la aplicación en Nextcloud

  1. Entra a tu panel de Nextcloud
  2. Ve a Apps (Aplicaciones)
  3. Busca: ONLYOFFICE
  4. Instala ONLYOFFICE Docs

Configurar conexión con OnlyOffice

Ve a:

Configuración → Administración → ONLYOFFICE

En “Document Editing Service”, coloca:

http://TU_IP:8080

Configurar seguridad con JWT (recomendado)

Para mayor seguridad, puedes activar autenticación JWT.


sudo docker stop $(sudo docker ps -q)

sudo docker run -i -t -d -p 8080:80 \
-e JWT_ENABLED=true \
-e JWT_SECRET=mi_clave_super_segura \
--restart=always \
onlyoffice/documentserver
  

Luego en Nextcloud:

  • Activa JWT
  • Introduce la misma clave

Solución de problemas (Nextcloud Snap)

Si Nextcloud no se conecta correctamente:


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:

👉 https://www.patreon.com/u74078772?

Cómo instalar Nextcloud en Ubuntu Server 24.04 (Guía completa + Script automático)

Entrada fija

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
&lt;VirtualHost *:80&gt;
    ServerName localhost
    DocumentRoot /var/www/nextcloud

    &lt;Directory /var/www/nextcloud/&gt;
        Require all granted
        AllowOverride All
        Options FollowSymLinks MultiViews
    &lt;/Directory&gt;

    ErrorLog \${APACHE_LOG_DIR}/nextcloud_error.log
    CustomLog \${APACHE_LOG_DIR}/nextcloud_access.log combined
&lt;/VirtualHost&gt;
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

  1. Crear el archivo: nano install_nextcloud.sh
  2. Pegar el script y guardar
  3. Dar permisos: chmod +x install_nextcloud.sh
  4. 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í:

👉 https://www.patreon.com/u74078772?

Validación en UDO de SAP Business One HANA: Solución definitiva

Entrada fija

Si estás desarrollando en SAP Business One sobre HANA y necesitas validar datos en una tabla de usuario tipo documento (UDO), es probable que hayas enfrentado el problema de que tu código en SBO_SP_TransactionNotification no se ejecuta. Aquí te explico por qué y cómo solucionarlo.

El problema

En SAP HANA, cuando una tabla de usuario (UDT) se registra formalmente como Objeto Definido por el Usuario (UDO) de tipo documento, el parámetro :object_type que recibe el procedimiento almacenado no es el nombre de la tabla física (como '@MTTOCONFIG'), sino el nombre del objeto registrado (el campo Name en la tabla OUDO).

Por eso, aunque tu validación funcionaba para algunas tablas (como USUARIOALMACEN), para otras simplemente no se ejecutaba.

Ejemplo real:
Tabla física: @MTTOCONFIG
Nombre del UDO registrado: ConfigMtto
✅ La condición correcta es: UPPER(:object_type) = 'CONFIGMTTO'

Solución: Código final que funciona

IF UPPER(:object_type) = 'CONFIGMTTO' 
   AND (:transaction_type = 'A' OR :transaction_type = 'U') THEN

    DECLARE v_U_Tipo VARCHAR(100);
    DECLARE v_U_Code VARCHAR(100);

    -- Obtener valores de la cabecera (la tabla @MTTOCONFIG)
    SELECT "U_Tipo", "U_Code"
      INTO v_U_Tipo, v_U_Code
      FROM "@MTTOCONFIG"
     WHERE "DocEntry" = :list_of_cols_val_tab_del;

    -- Validación según el tipo
    IF UPPER(v_U_Tipo) = 'TIPOMQ' THEN
        IF NOT EXISTS (SELECT 1 FROM "@TYPEMTTO" WHERE "Code" = v_U_Code) THEN
            error := 1;
            error_message := N'El código ' || IFNULL(v_U_Code, 'NULL') || N' no existe en @TYPEMTTO (TipoMQ).';
        END IF;
    ELSEIF UPPER(v_U_Tipo) = 'EQUIPO' THEN
        IF NOT EXISTS (SELECT 1 FROM OITM WHERE "ItemCode" = v_U_Code) THEN
            error := 1;
            error_message := N'El código ' || IFNULL(v_U_Code, 'NULL') || N' no existe en OITM (Equipo).';
        END IF;
    END IF;

END IF;

Puntos clave

  • Usa el nombre del UDO, no el de la tabla física. Para conocerlo, consulta: SELECT "Name" FROM "OUDO" WHERE "TableName" = '@TU_TABLA';
  • La clave primaria en documentos es DocEntry, y llega directamente en :list_of_cols_val_tab_del.
  • Usa UPPER() para evitar problemas de mayúsculas/minúsculas tanto en object_type como en los valores de U_Tipo.
  • Siempre asigna error := 1 y un error_message claro para que SAP muestre el mensaje y cancele la transacción.

¿Por qué funciona con USUARIOALMACEN y no con MTTOCONFIG?

Porque USUARIOALMACEN también es un UDO registrado, y su nombre coincide con el de la tabla. En cambio, ConfigMtto fue registrado con un nombre diferente al de la tabla física. Siempre verifica el campo Name en OUDO.

✅ Resultado: Con esta modificación, la validación se ejecuta correctamente y se impide guardar datos incorrectos (por ejemplo, un U_Code que no existe en @TYPEMTTO).


¿Te fue útil esta solución? Apoya mi trabajo en Patreon y accede a más contenido exclusivo sobre SAP Business One, HANA y automatizaciones.❤️ Apóyame en Patreon

🚀 Mejora PRO en SAP Business One: Validaciones Inteligentes de Odómetro y Horómetro

Entrada fija

Si trabajas con SAP Business One y manejas maquinaria o activos… esto te va a ahorrar MUCHOS dolores de cabeza 👇


🤯 El problema

Errores genéricos como:

❌ “Debe indicar el Odómetro al producto Aceite hidráulico”

Y tú:

  • 🤔 ¿Cuál línea?
  • 🤔 ¿Qué activo?
  • 🤔 ¿Dónde corrijo?

Resultado:

  • ⏳ Pierdes tiempo
  • 😤 Corriges mal
  • 🔁 El error vuelve a salir

💡 La solución

Ahora los mensajes muestran:

  • ✅ Código del activo
  • ✅ Nombre del activo
  • ✅ Línea exacta

🔥 Ejemplo:

Debe indicar el Odómetro al activo EQ-001 – Excavadora CAT (Línea 3)


🧠 ¿Qué se hizo?

  • ✔ Se usa el activo desde OcrCode
  • ✔ Se hace fallback al artículo si no hay activo
  • ✔ Se corrige LineNum + 1

🔧 Código completo listo para pegar

IF (:transaction_type = 'A') AND :object_type = '60' AND error_message = 'Ok'
THEN  

     select 
       max(
         case 

           when ifnull(t1."U_TypeMov",'') = '' 
                and ifnull(t1."BaseType",0) <> 202 then 
             'Seleccione el tipo de movimiento'

           when ifnull(t2."QryGroup11",'N') = 'Y' 
                and ifnull(t1."OcrCode",'') = '' 
                and ifnull(t1."U_TypeMov",'') = 'CONSUMO' then 
             'El articulo ' || t1."Dscription" || ' se especifico que debe seleccionarse la maquinaria'

           when ifnull(t4."QryGroup9",'N') = 'Y' 
                and ifnull(t2."QryGroup11",'N') = 'Y' 
                and ifnull(t1."U_Odometro",0) = 0 
                and ifnull(t1."U_TypeMov",'') = 'CONSUMO' then 
             'Debe indicar el Odometro al activo ' 
             || IFNULL(t4."ItemCode", t2."ItemCode")
             || ' - ' || IFNULL(t4."ItemName", t2."ItemName")
             || ' (Línea ' || (ifnull(t1."LineNum",0) + 1) || ')'

           when ifnull(t4."QryGroup9",'N') = 'N' 
                and ifnull(t1."U_Odometro",0) <> 0 then 
             'Quite el valor del campo Odometro en el activo ' 
             || IFNULL(t4."ItemCode", t2."ItemCode")
             || ' - ' || IFNULL(t4."ItemName", t2."ItemName")
             || ' (Línea ' || (ifnull(t1."LineNum",0) + 1) || ')'

           when ifnull(t4."QryGroup8",'N') = 'Y' 
                and ifnull(t2."QryGroup11",'N') = 'Y' 
                and ifnull(t1."U_Horometro",0) = 0  
                and ifnull(t1."U_TypeMov",'') = 'CONSUMO' then 
             'Debe indicar el Horometro al activo ' 
             || IFNULL(t4."ItemCode", t2."ItemCode")
             || ' - ' || IFNULL(t4."ItemName", t2."ItemName")
             || ' (Línea ' || (ifnull(t1."LineNum",0) + 1) || ')'

           when ifnull(t4."QryGroup8",'N') = 'N' 
                and ifnull(t1."U_Horometro",0) <> 0 then 
             'Quite el valor del campo Horometro en el activo ' 
             || IFNULL(t4."ItemCode", t2."ItemCode")
             || ' - ' || IFNULL(t4."ItemName", t2."ItemName")
             || ' (Línea ' || (ifnull(t1."LineNum",0) + 1) || ')'

           when ifnull(t2."QryGroup11",'N') = 'Y' 
                and (ifnull(t4."QryGroup8",'N') = 'Y' or ifnull(t4."QryGroup9",'N') = 'Y') 
                and ifnull(t1."U_Empleado",0) = 0  
                and ifnull(t1."U_TypeMov",'') = 'CONSUMO' then 
             'Debe indicar el empleado que recibe el producto ' || t1."Dscription"

           when ifnull(t2."QryGroup11",'N') = 'Y' 
                and (ifnull(t4."QryGroup8",'N') = 'Y' or ifnull(t4."QryGroup9",'N') = 'Y') 
                and ifnull(t1."U_HoraMovto",0) = 0  
                and ifnull(t1."U_TypeMov",'') = 'CONSUMO' then 
             'Debe indicar la hora en que se recibe el producto ' || t1."Dscription"

           when ifnull(t2."QryGroup11",'N') = 'Y' 
                and (ifnull(t4."QryGroup8",'N') = 'Y' or ifnull(t4."QryGroup9",'N') = 'Y') 
                and LENGTH(ifnull(t1."U_HoraMovto",0)) < 3  
                and ifnull(t1."U_TypeMov",'') = 'CONSUMO' then 
             'Verifique la hora del producto ' || t1."Dscription"

           else '' 
         end
       ) ERROR
     INTO error_message
     from OIGE t0
     inner join IGE1 t1 on t0."DocEntry" = t1."DocEntry"
     inner join OITM t2 on t2."ItemCode" = t1."ItemCode"
     left join OITM t4 on t4."ItemCode" = t1."OcrCode"
     left join "QUA_PermisosWhsUser" t3 
           on t3."WhsCode" = t1."WhsCode" 
          and t3."userId"  = t0."UserSign"
     where t0."DocEntry" = :list_of_cols_val_tab_del;

     IF IFNULL(:error_message, N'') <> N'' THEN 
        error := -10046;
     ELSE 
        error_message := N'Ok';
     END IF;
END IF;

📈 Beneficios

  • 🔥 Mensajes claros
  • 🔥 Menos errores
  • 🔥 Más productividad
  • 🔥 Mejor control de activos

🧩 Conclusión

Pequeño cambio… GRAN impacto 🚀

Tu sistema pasa de:

❌ Confuso
✅ Profesional y claro


💬 ¿Quieres más mejoras como esta?

Puedo ayudarte con:

  • ⚙️ Validaciones avanzadas
  • 🚀 Optimización de SQL
  • 🧠 Automatización en SAP

Solo dime 👇

🔐 Control de accesos a Saldos por Almacén — Código y guía práctica

Entrada fija

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

  1. 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).
  2. 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.
  3. Usuario con múltiples almacenes: Comprueba que aparecen filas de cualquiera de esos almacenes, y que no aparecen filas de almacenes no asignados.
  4. Busqueda global: Ejecuta una búsqueda por codigoProducto, lote o descripcion y valida que los resultados respetan el filtro por almacén.
  5. Paginación y orden: Revisa recordsTotal y recordsFiltered cuando aplicas orden y búsqueda; deben reflejar correctamente la cantidad total y la cantidad filtrada.
  6. 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):

&lt;div class="alert alert-info"&gt;
  &lt;strong&gt;Sin almacén asignado&lt;/strong&gt;&lt;br&gt;
  No tienes almacenes asignados. Contacta al administrador para que te asigne los permisos necesarios.
&lt;/div&gt;

🔐 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:

Release v1.2.3Commit d8ad77ad

Repositorio: https://github.com/julio101290/boilerplateInventory


📋 9) Checklist de despliegue

  • ✅ Probar en staging con usuarios de distintos permisos
  • ✅ Verificar índices y tiempos de respuesta
  • ✅ Añadir pruebas unitarias/integración que verifiquen permisos
  • ✅ Revisar logs de auditoría tras deploy

Página 1 de 142

Creado con WordPress & Tema de Anders Norén