fix: t-005 a t-0029 correctif
This commit is contained in:
@@ -15,7 +15,6 @@
|
|||||||
--color-m-success: rgb(var(--m-success));
|
--color-m-success: rgb(var(--m-success));
|
||||||
--color-m-accent: rgb(var(--m-accent));
|
--color-m-accent: rgb(var(--m-accent));
|
||||||
--color-m-warning: rgb(var(--m-warning));
|
--color-m-warning: rgb(var(--m-warning));
|
||||||
--color-m-succes: rgb(var(--m-success));
|
|
||||||
--font-display: "Outfit", system-ui, sans-serif;
|
--font-display: "Outfit", system-ui, sans-serif;
|
||||||
--font-mono: "JetBrains Mono", "Fira Code", monospace;
|
--font-mono: "JetBrains Mono", "Fira Code", monospace;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -77,7 +77,6 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, onMounted, ref } from "vue"
|
|
||||||
import { Icon as IconifyIcon } from "@iconify/vue"
|
import { Icon as IconifyIcon } from "@iconify/vue"
|
||||||
import { apiFetch } from "~/composables/useApiAuth"
|
import { apiFetch } from "~/composables/useApiAuth"
|
||||||
|
|
||||||
@@ -116,7 +115,6 @@ const active = ref<string | null>(null)
|
|||||||
const loading = ref(true)
|
const loading = ref(true)
|
||||||
const runningKey = ref<string | null>(null)
|
const runningKey = ref<string | null>(null)
|
||||||
const scripts = ref<BackupScript[]>([])
|
const scripts = ref<BackupScript[]>([])
|
||||||
const output = ref<string>("")
|
|
||||||
const message = ref<string>("")
|
const message = ref<string>("")
|
||||||
const isError = ref(false)
|
const isError = ref(false)
|
||||||
|
|
||||||
@@ -125,7 +123,6 @@ const statusClass = computed(() => (isError.value ? "status-error" : "status-suc
|
|||||||
const loadScripts = async () => {
|
const loadScripts = async () => {
|
||||||
loading.value = true
|
loading.value = true
|
||||||
message.value = ""
|
message.value = ""
|
||||||
output.value = ""
|
|
||||||
isError.value = false
|
isError.value = false
|
||||||
emit("result", {
|
emit("result", {
|
||||||
key: null,
|
key: null,
|
||||||
@@ -156,7 +153,6 @@ const loadScripts = async () => {
|
|||||||
const runScript = async (key: string) => {
|
const runScript = async (key: string) => {
|
||||||
active.value = key
|
active.value = key
|
||||||
runningKey.value = key
|
runningKey.value = key
|
||||||
output.value = ""
|
|
||||||
message.value = ""
|
message.value = ""
|
||||||
isError.value = false
|
isError.value = false
|
||||||
|
|
||||||
@@ -165,30 +161,17 @@ const runScript = async (key: string) => {
|
|||||||
method: "POST",
|
method: "POST",
|
||||||
body: { key }
|
body: { key }
|
||||||
})
|
})
|
||||||
|
const resultOutput = data.output || "Aucune sortie retournee."
|
||||||
message.value = `${data.label} execute avec succes`
|
message.value = `${data.label} execute avec succes`
|
||||||
output.value = data.output || "Aucune sortie retournee."
|
|
||||||
emit("result", {
|
emit("result", {
|
||||||
key: data.key,
|
key: data.key,
|
||||||
label: data.label,
|
label: data.label,
|
||||||
output: output.value,
|
output: resultOutput,
|
||||||
isError: false,
|
isError: false,
|
||||||
downloadFolders: data.downloadFolders || []
|
downloadFolders: data.downloadFolders || []
|
||||||
})
|
})
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
isError.value = true
|
message.value = (error as any)?.data?.statusMessage || "Erreur execution script"
|
||||||
const statusMessage =
|
|
||||||
typeof error === "object" &&
|
|
||||||
error !== null &&
|
|
||||||
"data" in error &&
|
|
||||||
typeof error.data === "object" &&
|
|
||||||
error.data !== null &&
|
|
||||||
"statusMessage" in error.data &&
|
|
||||||
typeof error.data.statusMessage === "string"
|
|
||||||
? error.data.statusMessage
|
|
||||||
: null
|
|
||||||
|
|
||||||
message.value = statusMessage || "Erreur execution script"
|
|
||||||
output.value = ""
|
|
||||||
emit("result", {
|
emit("result", {
|
||||||
key,
|
key,
|
||||||
label: scripts.value.find((item) => item.key === key)?.label || key,
|
label: scripts.value.find((item) => item.key === key)?.label || key,
|
||||||
|
|||||||
@@ -293,7 +293,7 @@ const visibleHistory = computed(() => {
|
|||||||
return history.value.filter((point) => point.sampledAt >= minTimestamp)
|
return history.value.filter((point) => point.sampledAt >= minTimestamp)
|
||||||
})
|
})
|
||||||
|
|
||||||
const scaleMax = computed(() => 100)
|
const scaleMax = 100
|
||||||
|
|
||||||
const formatValue = (value: number) => `${Math.round(value)}%`
|
const formatValue = (value: number) => `${Math.round(value)}%`
|
||||||
|
|
||||||
|
|||||||
@@ -146,7 +146,6 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {ref} from "vue"
|
|
||||||
import {Icon as IconifyIcon} from "@iconify/vue"
|
import {Icon as IconifyIcon} from "@iconify/vue"
|
||||||
import logoSrc from '~/assets/LOGO_CARRE_BLANC.png'
|
import logoSrc from '~/assets/LOGO_CARRE_BLANC.png'
|
||||||
|
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ export default defineNuxtConfig({
|
|||||||
head: {
|
head: {
|
||||||
link: [
|
link: [
|
||||||
{ rel: "preconnect", href: "https://fonts.googleapis.com" },
|
{ rel: "preconnect", href: "https://fonts.googleapis.com" },
|
||||||
{ rel: "preconnect", href: "https://fonts.gstatic.com ", crossorigin: "" },
|
{ rel: "preconnect", href: "https://fonts.gstatic.com", crossorigin: "" },
|
||||||
{
|
{
|
||||||
rel: "stylesheet",
|
rel: "stylesheet",
|
||||||
href: "https://fonts.googleapis.com/css2?family=JetBrains+Mono:wght@400;500;600;700&family=Outfit:wght@300;400;500;600;700;800;900&display=swap"
|
href: "https://fonts.googleapis.com/css2?family=JetBrains+Mono:wght@400;500;600;700&family=Outfit:wght@300;400;500;600;700;800;900&display=swap"
|
||||||
@@ -36,6 +36,8 @@ export default defineNuxtConfig({
|
|||||||
},
|
},
|
||||||
runtimeConfig: {
|
runtimeConfig: {
|
||||||
apiSecretKey: process.env.API_SECRET_KEY,
|
apiSecretKey: process.env.API_SECRET_KEY,
|
||||||
|
discordBotToken: process.env.DISCORD_BOT_TOKEN,
|
||||||
|
discordChannelId: process.env.DISCORD_CHANNEL_ID,
|
||||||
public: {
|
public: {
|
||||||
appVersion: getRepoVersion(),
|
appVersion: getRepoVersion(),
|
||||||
apiKey: process.env.API_SECRET_KEY
|
apiKey: process.env.API_SECRET_KEY
|
||||||
|
|||||||
8
package-lock.json
generated
8
package-lock.json
generated
@@ -1,10 +1,10 @@
|
|||||||
{
|
{
|
||||||
"name": "disk-monitor",
|
"name": "supervisor",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "disk-monitor",
|
"name": "supervisor",
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@iconify/vue": "^5.0.0",
|
"@iconify/vue": "^5.0.0",
|
||||||
@@ -15,11 +15,13 @@
|
|||||||
"@semantic-release/changelog": "^6.0.3",
|
"@semantic-release/changelog": "^6.0.3",
|
||||||
"@semantic-release/commit-analyzer": "^13.0.1",
|
"@semantic-release/commit-analyzer": "^13.0.1",
|
||||||
"@semantic-release/git": "^10.0.1",
|
"@semantic-release/git": "^10.0.1",
|
||||||
"@semantic-release/github": "^12.0.6",
|
|
||||||
"@semantic-release/release-notes-generator": "^14.1.0",
|
"@semantic-release/release-notes-generator": "^14.1.0",
|
||||||
"@tailwindcss/vite": "^4.2.1",
|
"@tailwindcss/vite": "^4.2.1",
|
||||||
"semantic-release": "^25.0.3",
|
"semantic-release": "^25.0.3",
|
||||||
"tailwindcss": "^4.2.1"
|
"tailwindcss": "^4.2.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=20"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@actions/core": {
|
"node_modules/@actions/core": {
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
{
|
{
|
||||||
"name": "disk-monitor",
|
"name": "supervisor",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=20"
|
||||||
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "nuxt build",
|
"build": "nuxt build",
|
||||||
"dev": "nuxt dev",
|
"dev": "nuxt dev",
|
||||||
@@ -20,7 +23,6 @@
|
|||||||
"@semantic-release/changelog": "^6.0.3",
|
"@semantic-release/changelog": "^6.0.3",
|
||||||
"@semantic-release/commit-analyzer": "^13.0.1",
|
"@semantic-release/commit-analyzer": "^13.0.1",
|
||||||
"@semantic-release/git": "^10.0.1",
|
"@semantic-release/git": "^10.0.1",
|
||||||
"@semantic-release/github": "^12.0.6",
|
|
||||||
"@semantic-release/release-notes-generator": "^14.1.0",
|
"@semantic-release/release-notes-generator": "^14.1.0",
|
||||||
"@tailwindcss/vite": "^4.2.1",
|
"@tailwindcss/vite": "^4.2.1",
|
||||||
"semantic-release": "^25.0.3",
|
"semantic-release": "^25.0.3",
|
||||||
|
|||||||
@@ -66,7 +66,6 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {computed, onMounted, ref} from "vue"
|
|
||||||
import { apiFetch } from "~/composables/useApiAuth"
|
import { apiFetch } from "~/composables/useApiAuth"
|
||||||
import type { SystemMetrics } from "~/types/system";
|
import type { SystemMetrics } from "~/types/system";
|
||||||
|
|
||||||
@@ -334,14 +333,6 @@ onBeforeUnmount(() => {
|
|||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
|
|
||||||
.backup-selector {
|
|
||||||
order: 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
.backup-list-mobile {
|
|
||||||
order: 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
.speedtest-card-mobile {
|
.speedtest-card-mobile {
|
||||||
order: 4;
|
order: 4;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
User-Agent: *
|
User-Agent: *
|
||||||
Disallow:
|
Disallow: /
|
||||||
|
|||||||
@@ -3,12 +3,10 @@ import {
|
|||||||
shellQuote,
|
shellQuote,
|
||||||
resolveFolderRemoteDir,
|
resolveFolderRemoteDir,
|
||||||
REMOTE_ROOT,
|
REMOTE_ROOT,
|
||||||
|
isSafeFolder,
|
||||||
} from "../utils/ssh.ts"
|
} from "../utils/ssh.ts"
|
||||||
|
|
||||||
import {process} from "std-env";
|
|
||||||
|
|
||||||
const MAX_FILES_PER_FOLDER = Math.max(1, Number(process.env.BACKUPS_MAX_FILES) || 50)
|
const MAX_FILES_PER_FOLDER = Math.max(1, Number(process.env.BACKUPS_MAX_FILES) || 50)
|
||||||
const isSafeFolder = (value: string) => /^[a-zA-Z0-9._-]+$/.test(value)
|
|
||||||
|
|
||||||
|
|
||||||
function isMissingPathError(error: unknown): boolean {
|
function isMissingPathError(error: unknown): boolean {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
export default defineEventHandler(async () => {
|
export default defineEventHandler(async (event) => {
|
||||||
const token = process.env.DISCORD_BOT_TOKEN
|
const config = useRuntimeConfig(event)
|
||||||
const channel = process.env.DISCORD_CHANNEL_ID
|
const token = config.discordBotToken
|
||||||
|
const channel = config.discordChannelId
|
||||||
|
|
||||||
if (!token || !channel) {
|
if (!token || !channel) {
|
||||||
throw createError({
|
throw createError({
|
||||||
|
|||||||
@@ -3,11 +3,10 @@ import {
|
|||||||
shellQuote,
|
shellQuote,
|
||||||
resolveFolderRemoteDir,
|
resolveFolderRemoteDir,
|
||||||
REMOTE_HOST,
|
REMOTE_HOST,
|
||||||
|
isSafeFolder
|
||||||
} from "../utils/ssh.ts"
|
} from "../utils/ssh.ts"
|
||||||
import { spawn } from "node:child_process"
|
import { spawn } from "node:child_process"
|
||||||
|
|
||||||
const isSafeFolder = (value: string) => /^[a-zA-Z0-9._-]+$/.test(value)
|
|
||||||
|
|
||||||
async function getLatestRemoteFile(remoteDir: string): Promise<string | null> {
|
async function getLatestRemoteFile(remoteDir: string): Promise<string | null> {
|
||||||
const output = await runSsh(`cd ${shellQuote(remoteDir)} && ls -1A | sort -r | head -n 1`)
|
const output = await runSsh(`cd ${shellQuote(remoteDir)} && ls -1A | sort -r | head -n 1`)
|
||||||
const fileName = output.trim()
|
const fileName = output.trim()
|
||||||
@@ -45,6 +44,9 @@ export default defineEventHandler(async (event) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const fileName = await getLatestRemoteFile(remoteDir)
|
const fileName = await getLatestRemoteFile(remoteDir)
|
||||||
|
if (!fileName || !isSafeFolder(fileName)) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
if (!fileName) {
|
if (!fileName) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -93,6 +95,6 @@ export default defineEventHandler(async (event) => {
|
|||||||
console.error(`Erreur archive SSH (${code}): ${stderr}`)
|
console.error(`Erreur archive SSH (${code}): ${stderr}`)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
event.node.res.on("close", () => child.kill())
|
||||||
return sendStream(event, child.stdout)
|
return sendStream(event, child.stdout)
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -3,13 +3,11 @@ import {
|
|||||||
shellQuote,
|
shellQuote,
|
||||||
resolveFolderRemoteDir,
|
resolveFolderRemoteDir,
|
||||||
REMOTE_HOST,
|
REMOTE_HOST,
|
||||||
|
isSafeFolder,
|
||||||
|
isSafeFile
|
||||||
} from "../utils/ssh.ts"
|
} from "../utils/ssh.ts"
|
||||||
import { spawn } from "node:child_process"
|
import { spawn } from "node:child_process"
|
||||||
|
|
||||||
const isSafeFolder = (value: string) => /^[a-zA-Z0-9._-]+$/.test(value)
|
|
||||||
const isSafeFile = (value: string) => /^[a-zA-Z0-9._-]+$/.test(value)
|
|
||||||
|
|
||||||
|
|
||||||
function buildContentDisposition(fileName: string) {
|
function buildContentDisposition(fileName: string) {
|
||||||
const asciiName = fileName.replace(/[^\x20-\x7E]/g, "_").replace(/["\\]/g, "_")
|
const asciiName = fileName.replace(/[^\x20-\x7E]/g, "_").replace(/["\\]/g, "_")
|
||||||
return `attachment; filename="${asciiName}"; filename*=UTF-8''${encodeURIComponent(fileName)}`
|
return `attachment; filename="${asciiName}"; filename*=UTF-8''${encodeURIComponent(fileName)}`
|
||||||
@@ -61,6 +59,6 @@ export default defineEventHandler(async (event) => {
|
|||||||
console.error(`Erreur téléchargement SSH (${code}): ${stderr}`)
|
console.error(`Erreur téléchargement SSH (${code}): ${stderr}`)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
event.node.res.on("close", () => child.kill())
|
||||||
return sendStream(event, child.stdout)
|
return sendStream(event, child.stdout)
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
export default defineEventHandler(async (event) => {
|
export default defineEventHandler(async (event) => {
|
||||||
const req = event.node.req
|
const req = event.node.req
|
||||||
|
const MAX_UPLOAD_BYTES = 100 * 1024 * 1024 // 100MB
|
||||||
let received = 0
|
let received = 0
|
||||||
|
|
||||||
for await (const chunk of req) {
|
for await (const chunk of req) {
|
||||||
|
if (received > MAX_UPLOAD_BYTES) throw createError({ statusCode: 413, statusMessage: "Fichier trop volumineux" })
|
||||||
received += chunk.length
|
received += chunk.length
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,18 @@
|
|||||||
import targets from "../config/version-status-targets.json"
|
import targets from "../config/version-status-targets.json"
|
||||||
|
|
||||||
|
const REQUEST_TIMEOUT_MS = 5000
|
||||||
|
|
||||||
export default defineEventHandler(async () => {
|
export default defineEventHandler(async () => {
|
||||||
const results = await Promise.all(
|
const results = await Promise.all(
|
||||||
targets.map(async (target) => {
|
targets.map(async (target) => {
|
||||||
|
const controller = new AbortController()
|
||||||
|
const timeoutId = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS)
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const response = await fetch(target.url, {
|
const response = await fetch(target.url, {
|
||||||
method: "GET",
|
method: "GET",
|
||||||
headers: { Accept: "application/json" }
|
headers: { Accept: "application/json" },
|
||||||
|
signal: controller.signal
|
||||||
})
|
})
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@@ -25,6 +31,8 @@ export default defineEventHandler(async () => {
|
|||||||
checkedAt: new Date().toISOString(),
|
checkedAt: new Date().toISOString(),
|
||||||
error: error instanceof Error ? error.message : String(error)
|
error: error instanceof Error ? error.message : String(error)
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
|
clearTimeout(timeoutId)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import {execFile} from "node:child_process"
|
import {execFile} from "node:child_process"
|
||||||
import {process} from "std-env";
|
|
||||||
import folderMap from "#server/config/backup-folders.json";
|
import folderMap from "#server/config/backup-folders.json";
|
||||||
|
|
||||||
export const REMOTE_HOST = process.env.BACKUPS_REMOTE_HOST
|
export const REMOTE_HOST = process.env.BACKUPS_REMOTE_HOST
|
||||||
export const REMOTE_ROOT = process.env.BACKUPS_REMOTE_ROOT || "/home/malio-b/backups"
|
export const REMOTE_ROOT = process.env.BACKUPS_REMOTE_ROOT || "/home/malio-b/backups"
|
||||||
export const FOLDER_MAP = folderMap as Record<string, string>
|
export const FOLDER_MAP = folderMap as Record<string, string>
|
||||||
|
export const isSafeFolder = (value: string) => /^[a-zA-Z0-9._-]+$/.test(value)
|
||||||
|
export const isSafeFile = (value: string) => /^[a-zA-Z0-9._-]+$/.test(value)
|
||||||
|
|
||||||
export const shellQuote = (value: string) => `'${value.replace(/'/g, `'\\''`)}'`
|
export const shellQuote = (value: string) => `'${value.replace(/'/g, `'\\''`)}'`
|
||||||
|
|
||||||
@@ -45,7 +45,7 @@ export async function resolveFolderRemoteDir(folderName: string): Promise<string
|
|||||||
return direct
|
return direct
|
||||||
}
|
}
|
||||||
|
|
||||||
const nested = `${REMOTE_ROOT}/bdd_recette/${folderName}`
|
const nested = `${REMOTE_ROOT}/bdd-recette/${folderName}`
|
||||||
if (await remoteDirExists(nested)) {
|
if (await remoteDirExists(nested)) {
|
||||||
return nested
|
return nested
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user