Soportar autenticacion SSH por clave privada
This commit is contained in:
+7
-4
@@ -12,7 +12,7 @@ Navegador
|
||||
v
|
||||
Contenedor monitor-rpi
|
||||
|
|
||||
| SSH con usuario/password
|
||||
| SSH con usuario/password o clave privada
|
||||
v
|
||||
Raspberry Pi activas
|
||||
```
|
||||
@@ -118,7 +118,11 @@ Usuario SSH.
|
||||
|
||||
### `password`
|
||||
|
||||
Password SSH. Se guarda en `config.json`, por lo que el archivo debe permanecer protegido.
|
||||
Password SSH. Se guarda en `config.json`, por lo que el archivo debe permanecer protegido. Si `privateKeyPath` esta configurado, la clave privada tiene prioridad y no se usa `password`.
|
||||
|
||||
### `privateKeyPath`
|
||||
|
||||
Ruta de la clave privada SSH dentro del contenedor o del host donde corre el monitor. Debe apuntar a una clave privada legible por el proceso del monitor. Para uso en Docker, monta la clave o una carpeta `.ssh` dentro del contenedor. Las claves con passphrase requieren agente SSH disponible; para monitorizacion unattended suele usarse una clave sin passphrase protegida por permisos de archivo.
|
||||
|
||||
### `model`
|
||||
|
||||
@@ -197,7 +201,7 @@ Invoke-RestMethod http://192.168.0.53:8787/api/status -Headers @{Authorization="
|
||||
|
||||
Devuelve la configuracion actual.
|
||||
|
||||
Nota: actualmente devuelve tambien passwords porque `config.html` permite editarlas.
|
||||
Nota: actualmente devuelve tambien credenciales porque `config.html` permite editarlas. Protege el acceso HTTP y el archivo `config.json`.
|
||||
|
||||
### `POST /api/config`
|
||||
|
||||
@@ -254,4 +258,3 @@ Reconstruir:
|
||||
cd /home/yamaray/docker/monitorRPi
|
||||
docker compose up -d --build
|
||||
```
|
||||
|
||||
|
||||
@@ -37,6 +37,7 @@
|
||||
"host": "192.168.0.46",
|
||||
"username": "pi",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "Home Assistant Local",
|
||||
"port": 22,
|
||||
"id": "rpi5-ha-main",
|
||||
@@ -49,6 +50,7 @@
|
||||
"host": "192.168.0.57",
|
||||
"username": "pi",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "Home Assistant Cabaña",
|
||||
"port": 22,
|
||||
"id": "rpi4-ha-second",
|
||||
@@ -61,6 +63,7 @@
|
||||
"host": "192.168.0.53",
|
||||
"username": "yamaray",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "Docker",
|
||||
"port": 22,
|
||||
"id": "rpi5-jose-docker",
|
||||
@@ -73,6 +76,7 @@
|
||||
"host": "192.168.0.130",
|
||||
"username": "pi",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "Docker - Daniel",
|
||||
"port": 22,
|
||||
"id": "rpi5-dani-docker",
|
||||
@@ -85,6 +89,7 @@
|
||||
"host": "192.168.0.254",
|
||||
"username": "yamaray",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "nginx - wireguard ",
|
||||
"port": 22,
|
||||
"id": "rpi3-nginx",
|
||||
@@ -97,6 +102,7 @@
|
||||
"host": "192.168.0.37",
|
||||
"username": "pi",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "MQTT - varios",
|
||||
"port": 22,
|
||||
"id": "rpi3-pi3home",
|
||||
@@ -109,6 +115,7 @@
|
||||
"host": "192.168.0.60",
|
||||
"username": "yamaray",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "Meshcore-Interface",
|
||||
"port": 22,
|
||||
"id": "rpi3-meshcore",
|
||||
@@ -121,6 +128,7 @@
|
||||
"host": "192.168.1.47",
|
||||
"username": "pi",
|
||||
"password": "",
|
||||
"privateKeyPath": "",
|
||||
"role": "Reserva",
|
||||
"port": 22,
|
||||
"id": "rpi3-node-3",
|
||||
|
||||
+42
-2
@@ -192,6 +192,20 @@
|
||||
.field-id { grid-column: span 2; }
|
||||
.field-role { grid-column: span 2; }
|
||||
.field-location { grid-column: span 2; }
|
||||
.field-auth { grid-column: span 1; }
|
||||
.device[data-auth-method="password"] .key-credential { display: none; }
|
||||
.device[data-auth-method="key"] .password-credential { display: none; }
|
||||
|
||||
.credential-note {
|
||||
grid-column: span 2;
|
||||
color: var(--muted);
|
||||
font-size: 12px;
|
||||
line-height: 1.35;
|
||||
align-self: center;
|
||||
}
|
||||
|
||||
.device[data-auth-method="key"] .credential-note strong { color: var(--info); }
|
||||
.device[data-auth-method="password"] .credential-note strong { color: var(--ok); }
|
||||
|
||||
label {
|
||||
display: grid;
|
||||
@@ -547,8 +561,9 @@
|
||||
const dashboardLink = document.querySelector("#dashboardLink");
|
||||
|
||||
function deviceTemplate(device, index) {
|
||||
const authMethod = device.privateKeyPath ? "key" : "password";
|
||||
return `
|
||||
<div class="device" data-index="${index}">
|
||||
<div class="device" data-index="${index}" data-auth-method="${authMethod}">
|
||||
<label title="Si esta activo, este dispositivo se consulta por SSH. Si esta inactivo, se muestra deshabilitado y no se escanea.">Activo
|
||||
<input data-field="active" type="checkbox" ${device.active ? "checked" : ""}>
|
||||
</label>
|
||||
@@ -561,9 +576,21 @@
|
||||
<label title="Usuario SSH para este dispositivo.">Usuario
|
||||
<input data-field="username" value="${device.username || ""}">
|
||||
</label>
|
||||
<label title="Password SSH. Se guarda en config.json dentro del volumen persistente.">Password
|
||||
<label class="field-auth" title="Metodo de autenticacion SSH que usara el monitor para este dispositivo.">Auth SSH
|
||||
<select data-auth-method>
|
||||
<option value="password" ${authMethod === "password" ? "selected" : ""}>Password</option>
|
||||
<option value="key" ${authMethod === "key" ? "selected" : ""}>Clave privada</option>
|
||||
</select>
|
||||
</label>
|
||||
<label class="password-credential" title="Password SSH. Se guarda en config.json dentro del volumen persistente. Si eliges clave privada, se guardara vacio.">Password
|
||||
<input data-field="password" type="password" value="${device.password || ""}">
|
||||
</label>
|
||||
<label class="key-credential" title="Ruta de la clave privada SSH dentro del contenedor. Ejemplo: /ssh/carabanes_monitor_ed25519. Si eliges password, se guardara vacia.">Clave privada
|
||||
<input data-field="privateKeyPath" value="${device.privateKeyPath || ""}">
|
||||
</label>
|
||||
<div class="credential-note">
|
||||
Metodo activo: <strong>${authMethod === "key" ? "Clave privada" : "Password"}</strong>
|
||||
</div>
|
||||
<label class="field-id" title="Identificador interno unico. Se usa para tracking de metricas como red.">ID
|
||||
<input data-field="id" value="${device.id || ""}">
|
||||
</label>
|
||||
@@ -652,6 +679,9 @@
|
||||
else if (input.type === "number") device[field] = Number(input.value);
|
||||
else device[field] = input.value;
|
||||
});
|
||||
const authMethod = row.querySelector("[data-auth-method]")?.value || "password";
|
||||
if (authMethod === "key") device.password = "";
|
||||
else device.privateKeyPath = "";
|
||||
return device;
|
||||
});
|
||||
}
|
||||
@@ -702,6 +732,7 @@
|
||||
port: 22,
|
||||
username: "pi",
|
||||
password: "",
|
||||
privateKeyPath: "",
|
||||
model: "RPi 4",
|
||||
role: "",
|
||||
location: "Rack",
|
||||
@@ -731,6 +762,15 @@
|
||||
render();
|
||||
});
|
||||
|
||||
devicesEl.addEventListener("change", (event) => {
|
||||
const selector = event.target.closest("[data-auth-method]");
|
||||
if (!selector) return;
|
||||
const row = selector.closest(".device");
|
||||
row.dataset.authMethod = selector.value;
|
||||
const note = row.querySelector(".credential-note strong");
|
||||
if (note) note.textContent = selector.value === "key" ? "Clave privada" : "Password";
|
||||
});
|
||||
|
||||
load().catch((error) => {
|
||||
statusEl.textContent = `Error cargando configuracion: ${error.message}`;
|
||||
});
|
||||
|
||||
@@ -16,3 +16,4 @@ services:
|
||||
volumes:
|
||||
- /home/yamaray/docker/monitorRPi/app:/app:ro
|
||||
- /home/yamaray/docker/monitorRPi/data:/data
|
||||
- /home/yamaray/docker/monitorRPi/ssh:/ssh:ro
|
||||
|
||||
@@ -86,7 +86,7 @@
|
||||
<li><code>name</code>: nombre visible.</li>
|
||||
<li><code>host</code>: IP o DNS.</li>
|
||||
<li><code>port</code>: puerto SSH.</li>
|
||||
<li><code>username/password</code>: credenciales SSH.</li>
|
||||
<li><code>username/password</code> o <code>privateKeyPath</code>: credenciales SSH. Si hay clave privada configurada, tiene prioridad sobre el password.</li>
|
||||
<li><code>model</code>: modelo visible.</li>
|
||||
<li><code>role</code>: funcion del equipo.</li>
|
||||
<li><code>location</code>: ubicacion.</li>
|
||||
|
||||
@@ -364,17 +364,15 @@ function runCommandWithInput(command, args, input, timeoutMs) {
|
||||
}
|
||||
|
||||
async function sshMetrics(device) {
|
||||
if (!device.password) {
|
||||
throw new Error("Password SSH no configurada");
|
||||
const hasPrivateKey = Boolean(device.privateKeyPath);
|
||||
if (!hasPrivateKey && !device.password) {
|
||||
throw new Error("Credenciales SSH no configuradas");
|
||||
}
|
||||
|
||||
const timeoutMs = Math.max(2, config.sshTimeoutSeconds || 8) * 1000;
|
||||
const port = Number(device.port || 22);
|
||||
const destination = `${device.username || "pi"}@${device.host}`;
|
||||
const args = [
|
||||
"-p",
|
||||
device.password,
|
||||
"ssh",
|
||||
const sshArgs = [
|
||||
"-o",
|
||||
"StrictHostKeyChecking=no",
|
||||
"-o",
|
||||
@@ -388,7 +386,29 @@ async function sshMetrics(device) {
|
||||
"-s"
|
||||
];
|
||||
|
||||
const { stdout } = await runCommandWithInput("sshpass", args, REMOTE_METRICS_SCRIPT, timeoutMs);
|
||||
let command = "ssh";
|
||||
let args = sshArgs;
|
||||
if (hasPrivateKey) {
|
||||
args = [
|
||||
"-i",
|
||||
device.privateKeyPath,
|
||||
"-o",
|
||||
"BatchMode=yes",
|
||||
"-o",
|
||||
"IdentitiesOnly=yes",
|
||||
...sshArgs
|
||||
];
|
||||
} else {
|
||||
command = "sshpass";
|
||||
args = [
|
||||
"-p",
|
||||
device.password,
|
||||
"ssh",
|
||||
...sshArgs
|
||||
];
|
||||
}
|
||||
|
||||
const { stdout } = await runCommandWithInput(command, args, REMOTE_METRICS_SCRIPT, timeoutMs);
|
||||
const values = parseKeyValueOutput(stdout);
|
||||
const temp = toNumber(values.cpuTempC, 0);
|
||||
const loadAverage = String(values.loadAverage || "")
|
||||
|
||||
Reference in New Issue
Block a user