mirror of
https://github.com/NginxProxyManager/nginx-proxy-manager.git
synced 2025-11-13 13:55:14 +00:00
backend test
This commit is contained in:
@@ -14,6 +14,7 @@ import schemaRoutes from "./schema.js";
|
|||||||
import settingsRoutes from "./settings.js";
|
import settingsRoutes from "./settings.js";
|
||||||
import tokensRoutes from "./tokens.js";
|
import tokensRoutes from "./tokens.js";
|
||||||
import usersRoutes from "./users.js";
|
import usersRoutes from "./users.js";
|
||||||
|
import versionRoutes from "./version.js";
|
||||||
|
|
||||||
const router = express.Router({
|
const router = express.Router({
|
||||||
caseSensitive: true,
|
caseSensitive: true,
|
||||||
@@ -46,6 +47,7 @@ router.use("/users", usersRoutes);
|
|||||||
router.use("/audit-log", auditLogRoutes);
|
router.use("/audit-log", auditLogRoutes);
|
||||||
router.use("/reports", reportsRoutes);
|
router.use("/reports", reportsRoutes);
|
||||||
router.use("/settings", settingsRoutes);
|
router.use("/settings", settingsRoutes);
|
||||||
|
router.use("/version", versionRoutes);
|
||||||
router.use("/nginx/proxy-hosts", proxyHostsRoutes);
|
router.use("/nginx/proxy-hosts", proxyHostsRoutes);
|
||||||
router.use("/nginx/redirection-hosts", redirectionHostsRoutes);
|
router.use("/nginx/redirection-hosts", redirectionHostsRoutes);
|
||||||
router.use("/nginx/dead-hosts", deadHostsRoutes);
|
router.use("/nginx/dead-hosts", deadHostsRoutes);
|
||||||
|
|||||||
77
backend/routes/version.js
Normal file
77
backend/routes/version.js
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
import express from "express";
|
||||||
|
import { debug, express as logger } from "../logger.js";
|
||||||
|
import pjson from "../package.json" with { type: "json" };
|
||||||
|
|
||||||
|
const router = express.Router({
|
||||||
|
caseSensitive: true,
|
||||||
|
strict: true,
|
||||||
|
mergeParams: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* /api/version/check
|
||||||
|
*/
|
||||||
|
router
|
||||||
|
.route("/check")
|
||||||
|
.options((_, res) => {
|
||||||
|
res.sendStatus(204);
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GET /api/version/check
|
||||||
|
*
|
||||||
|
* Check for available updates
|
||||||
|
*/
|
||||||
|
.get(async (req, res, next) => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
"https://api.github.com/repos/NginxProxyManager/nginx-proxy-manager/releases/latest"
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`GitHub API returned ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
const latestVersion = data.tag_name;
|
||||||
|
|
||||||
|
const version = pjson.version.split("-").shift().split(".");
|
||||||
|
const currentVersion = `v${version[0]}.${version[1]}.${version[2]}`;
|
||||||
|
|
||||||
|
res.status(200).send({
|
||||||
|
current: currentVersion,
|
||||||
|
latest: latestVersion,
|
||||||
|
updateAvailable: compareVersions(currentVersion, latestVersion),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
debug(logger, `${req.method.toUpperCase()} ${req.path}: ${error}`);
|
||||||
|
res.status(200).send({
|
||||||
|
current: null,
|
||||||
|
latest: null,
|
||||||
|
updateAvailable: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compare two version strings
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function compareVersions(current, latest) {
|
||||||
|
const cleanCurrent = current.replace(/^v/, "");
|
||||||
|
const cleanLatest = latest.replace(/^v/, "");
|
||||||
|
|
||||||
|
const currentParts = cleanCurrent.split(".").map(Number);
|
||||||
|
const latestParts = cleanLatest.split(".").map(Number);
|
||||||
|
|
||||||
|
for (let i = 0; i < Math.max(currentParts.length, latestParts.length); i++) {
|
||||||
|
const curr = currentParts[i] || 0;
|
||||||
|
const lat = latestParts[i] || 0;
|
||||||
|
|
||||||
|
if (lat > curr) return true;
|
||||||
|
if (lat < curr) return false;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default router;
|
||||||
@@ -2,23 +2,6 @@ import { useCallback, useEffect, useState } from "react";
|
|||||||
import { useHealth } from "src/hooks";
|
import { useHealth } from "src/hooks";
|
||||||
import { T } from "src/locale";
|
import { T } from "src/locale";
|
||||||
|
|
||||||
const compareVersions = (current: string, latest: string): boolean => {
|
|
||||||
const cleanCurrent = current.replace(/^v/, "");
|
|
||||||
const cleanLatest = latest.replace(/^v/, "");
|
|
||||||
|
|
||||||
const currentParts = cleanCurrent.split(".").map(Number);
|
|
||||||
const latestParts = cleanLatest.split(".").map(Number);
|
|
||||||
|
|
||||||
for (let i = 0; i < Math.max(currentParts.length, latestParts.length); i++) {
|
|
||||||
const curr = currentParts[i] || 0;
|
|
||||||
const lat = latestParts[i] || 0;
|
|
||||||
|
|
||||||
if (lat > curr) return true;
|
|
||||||
if (lat < curr) return false;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
export function SiteFooter() {
|
export function SiteFooter() {
|
||||||
const health = useHealth();
|
const health = useHealth();
|
||||||
const [latestVersion, setLatestVersion] = useState<string | null>(null);
|
const [latestVersion, setLatestVersion] = useState<string | null>(null);
|
||||||
@@ -35,18 +18,11 @@ export function SiteFooter() {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const checkForUpdates = async () => {
|
const checkForUpdates = async () => {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const response = await fetch("/api/version/check");
|
||||||
"https://api.github.com/repos/NginxProxyManager/nginx-proxy-manager/releases/latest"
|
|
||||||
);
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
const latest = data.tag_name;
|
setLatestVersion(data.latest);
|
||||||
setLatestVersion(latest);
|
setIsNewVersionAvailable(data.updateAvailable);
|
||||||
|
|
||||||
const currentVersion = getVersion();
|
|
||||||
if (currentVersion && compareVersions(currentVersion, latest)) {
|
|
||||||
setIsNewVersionAvailable(true);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.debug("Could not check for updates:", error);
|
console.debug("Could not check for updates:", error);
|
||||||
@@ -56,7 +32,7 @@ export function SiteFooter() {
|
|||||||
if (health.data) {
|
if (health.data) {
|
||||||
checkForUpdates();
|
checkForUpdates();
|
||||||
}
|
}
|
||||||
}, [health.data, getVersion]);
|
}, [health.data]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<footer className="footer d-print-none py-3">
|
<footer className="footer d-print-none py-3">
|
||||||
|
|||||||
Reference in New Issue
Block a user