first commit
This commit is contained in:
commit
db5ca09c1b
|
|
@ -0,0 +1,25 @@
|
|||
steps:
|
||||
- name: build-and-push
|
||||
image: woodpeckerci/plugin-docker-buildx
|
||||
settings:
|
||||
repo: "registry.af-east-1.dc.koldsoftware.com/${CI_REPO_OWNER}/nodejs-app"
|
||||
registry: "registry.af-east-1.dc.koldsoftware.com"
|
||||
tags:
|
||||
- "${CI_COMMIT_SHA:0:8}"
|
||||
- latest
|
||||
username:
|
||||
from_secret: registry_username
|
||||
password:
|
||||
from_secret: registry_password
|
||||
|
||||
- name: deploy
|
||||
image: hashicorp/nomad:latest
|
||||
commands:
|
||||
- nomad job run
|
||||
-var="version=${CI_COMMIT_SHA:0:8}"
|
||||
-var="domain=af-east-1.dc.koldsoftware.com"
|
||||
job.hcl
|
||||
environment:
|
||||
NOMAD_ADDR: "http://192.168.2.106:4646"
|
||||
when:
|
||||
branch: main
|
||||
|
|
@ -0,0 +1,14 @@
|
|||
# Stage 1: install production dependencies
|
||||
FROM node:20-alpine AS deps
|
||||
WORKDIR /app
|
||||
COPY package*.json ./
|
||||
RUN npm install --omit=dev
|
||||
|
||||
# Stage 2: final image — no dev tools, no npm cache
|
||||
FROM node:20-alpine
|
||||
WORKDIR /app
|
||||
COPY --from=deps /app/node_modules ./node_modules
|
||||
COPY src/ ./src/
|
||||
EXPOSE 3000
|
||||
USER node
|
||||
CMD ["node", "src/index.js"]
|
||||
|
|
@ -0,0 +1,65 @@
|
|||
variables {
|
||||
version = "latest"
|
||||
domain = "homelab.local"
|
||||
count = 1
|
||||
}
|
||||
|
||||
job "nodejs-app" {
|
||||
datacenters = ["*"]
|
||||
type = "service"
|
||||
|
||||
update {
|
||||
max_parallel = 1
|
||||
min_healthy_time = "10s"
|
||||
healthy_deadline = "3m"
|
||||
auto_revert = true
|
||||
}
|
||||
|
||||
group "nodejs-app" {
|
||||
count = var.count
|
||||
|
||||
network {
|
||||
port "http" { to = 3000 }
|
||||
}
|
||||
|
||||
task "nodejs-app" {
|
||||
driver = "docker"
|
||||
|
||||
config {
|
||||
image = "registry.${var.domain}/kurt/nodejs-app:${var.version}"
|
||||
ports = ["http"]
|
||||
}
|
||||
|
||||
env {
|
||||
NODE_ENV = "production"
|
||||
APP_VERSION = var.version
|
||||
PORT = "3000"
|
||||
}
|
||||
|
||||
resources {
|
||||
cpu = 256
|
||||
memory = 256
|
||||
}
|
||||
}
|
||||
|
||||
service {
|
||||
name = "nodejs-app"
|
||||
port = "http"
|
||||
provider = "nomad"
|
||||
|
||||
tags = [
|
||||
"traefik.enable=true",
|
||||
"traefik.http.routers.nodejs-app.rule=Host(`nodejs-app.${var.domain}`)",
|
||||
"traefik.http.routers.nodejs-app.tls=true",
|
||||
"traefik.http.routers.nodejs-app.tls.certresolver=letsencrypt",
|
||||
]
|
||||
|
||||
check {
|
||||
type = "http"
|
||||
path = "/health"
|
||||
interval = "10s"
|
||||
timeout = "2s"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,20 @@
|
|||
{
|
||||
"name": "nodejs-app",
|
||||
"version": "0.1.0",
|
||||
"description": "Example Node.js app for homelab-dc",
|
||||
"main": "src/index.js",
|
||||
"scripts": {
|
||||
"start": "node src/index.js",
|
||||
"dev": "nodemon src/index.js"
|
||||
},
|
||||
"dependencies": {
|
||||
"express": "^4.19.2",
|
||||
"prom-client": "^15.1.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"nodemon": "^3.1.0"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">=20"
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,65 @@
|
|||
'use strict'
|
||||
|
||||
const express = require('express')
|
||||
const promClient = require('prom-client')
|
||||
|
||||
const app = express()
|
||||
const PORT = process.env.PORT || 3000
|
||||
|
||||
// Prometheus metrics
|
||||
const collectDefaultMetrics = promClient.collectDefaultMetrics
|
||||
collectDefaultMetrics({ prefix: 'nodejs_app_' })
|
||||
|
||||
const httpRequestDuration = new promClient.Histogram({
|
||||
name: 'http_request_duration_seconds',
|
||||
help: 'Duration of HTTP requests in seconds',
|
||||
labelNames: ['method', 'route', 'status'],
|
||||
buckets: [0.01, 0.05, 0.1, 0.5, 1, 5],
|
||||
})
|
||||
|
||||
const httpRequestTotal = new promClient.Counter({
|
||||
name: 'http_requests_total',
|
||||
help: 'Total number of HTTP requests',
|
||||
labelNames: ['method', 'route', 'status'],
|
||||
})
|
||||
|
||||
// Metrics middleware
|
||||
app.use((req, res, next) => {
|
||||
const end = httpRequestDuration.startTimer()
|
||||
res.on('finish', () => {
|
||||
const route = req.route ? req.route.path : req.path
|
||||
const labels = { method: req.method, route, status: res.statusCode }
|
||||
end(labels)
|
||||
httpRequestTotal.inc(labels)
|
||||
})
|
||||
next()
|
||||
})
|
||||
|
||||
app.use(express.json())
|
||||
|
||||
app.get('/metrics', async (_req, res) => {
|
||||
res.set('Content-Type', promClient.register.contentType)
|
||||
res.end(await promClient.register.metrics())
|
||||
})
|
||||
|
||||
app.get('/health', (_req, res) => {
|
||||
res.json({ status: 'ok' })
|
||||
})
|
||||
|
||||
app.get('/', (_req, res) => {
|
||||
res.json({
|
||||
message: 'Hello from Node.js!',
|
||||
env: process.env.NODE_ENV || 'development',
|
||||
version: process.env.APP_VERSION || 'unknown',
|
||||
})
|
||||
})
|
||||
|
||||
app.listen(PORT, '0.0.0.0', () => {
|
||||
console.log(`Server listening on port ${PORT}`)
|
||||
})
|
||||
|
||||
// Graceful shutdown — important for Nomad rolling deploys
|
||||
process.on('SIGTERM', () => {
|
||||
console.log('SIGTERM received, shutting down gracefully')
|
||||
process.exit(0)
|
||||
})
|
||||
Loading…
Reference in New Issue