Compare commits
41 Commits
feature/ui
...
0a633258a8
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0a633258a8 | ||
| 36202e1a3a | |||
|
|
38bbd13c7f | ||
| add5f1f4f3 | |||
| 8ca57746df | |||
| c61c047cd8 | |||
| 4ad2573532 | |||
| b57afeddbd | |||
| 4ad98d578b | |||
| c875793314 | |||
| b357ccf978 | |||
| cd16edd486 | |||
| f791798044 | |||
| 2861de3c6e | |||
| d9556e3cfd | |||
| dcdc2873a6 | |||
| f5327a11c1 | |||
| 85058a336a | |||
| db983fb340 | |||
| 30895f4219 | |||
| 05a2ee970c | |||
| de98e74539 | |||
| f4517ae953 | |||
| a3bd3d06d1 | |||
| 77e40bf4ef | |||
| d896c2d19b | |||
| 0bbf4d66a8 | |||
| 60372f0708 | |||
| 4aaf9c2a49 | |||
| c31b976f01 | |||
| 869f123a5b | |||
| ca67f65242 | |||
| bc612daf22 | |||
| a3feacb842 | |||
| 94169b8d65 | |||
| 14764a79ad | |||
| 55fb63bec1 | |||
| 0e84b973ce | |||
| b23da0797e | |||
| 83bd23945c | |||
| cb8d47d7aa |
@@ -1,6 +1,5 @@
|
|||||||
node_modules
|
node_modules
|
||||||
dist
|
dist
|
||||||
.git
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
*.log
|
*.log
|
||||||
.claude
|
.claude
|
||||||
|
|||||||
10
.gitea/workflows/check.yml
Normal file
10
.gitea/workflows/check.yml
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
name: check
|
||||||
|
on: [push]
|
||||||
|
jobs:
|
||||||
|
check:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
# actions/checkout v4.2.2, 2026-02-22
|
||||||
|
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683
|
||||||
|
- run: docker build .
|
||||||
|
- run: docker build -f Dockerfile.backend .
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
backend/
|
||||||
dist/
|
dist/
|
||||||
node_modules/
|
node_modules/
|
||||||
yarn.lock
|
yarn.lock
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
{
|
{
|
||||||
"tabWidth": 4
|
"tabWidth": 4,
|
||||||
|
"proseWrap": "always"
|
||||||
}
|
}
|
||||||
|
|||||||
38
Dockerfile
38
Dockerfile
@@ -3,48 +3,16 @@ FROM node@sha256:e4bf2a82ad0a4037d28035ae71529873c069b13eb0455466ae0bc13363826e3
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY package.json yarn.lock ./
|
COPY package.json yarn.lock ./
|
||||||
RUN yarn install --frozen-lockfile
|
RUN yarn install --frozen-lockfile
|
||||||
|
RUN apk add --no-cache git
|
||||||
COPY . .
|
COPY . .
|
||||||
RUN yarn build
|
RUN yarn build
|
||||||
|
|
||||||
# nginx:stable-alpine as of 2026-02-22
|
# nginx:stable-alpine as of 2026-02-22
|
||||||
FROM nginx@sha256:15e96e59aa3b0aada3a121296e3bce117721f42d88f5f64217ef4b18f458c6ab
|
FROM nginx@sha256:15e96e59aa3b0aada3a121296e3bce117721f42d88f5f64217ef4b18f458c6ab
|
||||||
# Remove default config
|
|
||||||
RUN rm /etc/nginx/conf.d/default.conf
|
RUN rm /etc/nginx/conf.d/default.conf
|
||||||
# Config template — envsubst replaces $PORT at container start
|
COPY nginx.conf /etc/nginx/conf.d/netwatch.conf
|
||||||
COPY <<'EOF' /etc/nginx/netwatch.conf.template
|
|
||||||
server {
|
|
||||||
listen $PORT;
|
|
||||||
server_name _;
|
|
||||||
|
|
||||||
root /usr/share/nginx/html;
|
|
||||||
index index.html;
|
|
||||||
|
|
||||||
# Trust RFC1918 reverse proxies for X-Forwarded-For
|
|
||||||
set_real_ip_from 10.0.0.0/8;
|
|
||||||
set_real_ip_from 172.16.0.0/12;
|
|
||||||
set_real_ip_from 192.168.0.0/16;
|
|
||||||
real_ip_header X-Forwarded-For;
|
|
||||||
real_ip_recursive on;
|
|
||||||
|
|
||||||
# Access log to stdout (Docker best practice)
|
|
||||||
access_log /dev/stdout combined;
|
|
||||||
error_log /dev/stderr warn;
|
|
||||||
|
|
||||||
location / {
|
|
||||||
try_files $uri $uri/ /index.html;
|
|
||||||
}
|
|
||||||
|
|
||||||
# Cache static assets aggressively
|
|
||||||
location /assets/ {
|
|
||||||
expires 1y;
|
|
||||||
add_header Cache-Control "public, immutable";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
|
|
||||||
COPY --from=build /app/dist /usr/share/nginx/html
|
COPY --from=build /app/dist /usr/share/nginx/html
|
||||||
|
|
||||||
ENV PORT=8080
|
|
||||||
EXPOSE 8080
|
EXPOSE 8080
|
||||||
|
|
||||||
CMD ["/bin/sh", "-c", "envsubst '$PORT' < /etc/nginx/netwatch.conf.template > /etc/nginx/conf.d/netwatch.conf && exec nginx -g 'daemon off;'"]
|
CMD ["nginx", "-g", "daemon off;"]
|
||||||
|
|||||||
25
Dockerfile.backend
Normal file
25
Dockerfile.backend
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
# golang:1.25-alpine (2026-02-27)
|
||||||
|
FROM golang:1.25-alpine@sha256:f6751d823c26342f9506c03797d2527668d095b0a15f1862cddb4d927a7a4ced AS builder
|
||||||
|
|
||||||
|
RUN apk add --no-cache git make gcc musl-dev
|
||||||
|
|
||||||
|
# golangci-lint v2.7.2 (2026-02-27)
|
||||||
|
RUN CGO_ENABLED=0 go install github.com/golangci/golangci-lint/v2/cmd/golangci-lint@9f61b0f53f80672872fced07b6874397c3ed197b
|
||||||
|
|
||||||
|
WORKDIR /repo/backend
|
||||||
|
COPY backend/go.mod backend/go.sum ./
|
||||||
|
RUN go mod download
|
||||||
|
COPY .git /repo/.git
|
||||||
|
COPY backend/ .
|
||||||
|
|
||||||
|
RUN make check
|
||||||
|
RUN make build
|
||||||
|
|
||||||
|
# alpine:3.23 (2026-02-27)
|
||||||
|
FROM alpine:3.23@sha256:25109184c71bdad752c8312a8623239686a9a2071e8825f20acb8f2198c3f659
|
||||||
|
|
||||||
|
RUN apk add --no-cache ca-certificates
|
||||||
|
COPY --from=builder /repo/backend/netwatch-server /usr/local/bin/netwatch-server
|
||||||
|
|
||||||
|
EXPOSE 8080
|
||||||
|
ENTRYPOINT ["netwatch-server"]
|
||||||
5
Makefile
5
Makefile
@@ -1,4 +1,7 @@
|
|||||||
.PHONY: test lint fmt fmt-check check docker
|
.PHONY: dev test lint fmt fmt-check check docker
|
||||||
|
|
||||||
|
dev:
|
||||||
|
yarn dev
|
||||||
|
|
||||||
test:
|
test:
|
||||||
timeout 30 yarn build
|
timeout 30 yarn build
|
||||||
|
|||||||
73
README.md
73
README.md
@@ -1,4 +1,7 @@
|
|||||||
NetWatch is an MIT-licensed JavaScript single-page application by [@sneak](https://sneak.berlin) that provides real-time network latency monitoring to common internet hosts, displayed with color-coded figures and sparkline graphs, served from a static bucket or Docker container.
|
NetWatch is an MIT-licensed JavaScript single-page application by
|
||||||
|
[@sneak](https://sneak.berlin) that provides real-time network latency
|
||||||
|
monitoring to common internet hosts, displayed with color-coded figures and
|
||||||
|
sparkline graphs, served from a static bucket or Docker container.
|
||||||
|
|
||||||
## Getting Started
|
## Getting Started
|
||||||
|
|
||||||
@@ -22,27 +25,50 @@ docker run -p 8080:8080 netwatch
|
|||||||
|
|
||||||
## Rationale
|
## Rationale
|
||||||
|
|
||||||
When debugging network issues, it's useful to have a persistent at-a-glance view of latency and reachability to multiple well-known internet endpoints. NetWatch provides this as a zero-dependency SPA that can be deployed anywhere static files are served, with no backend required.
|
When debugging network issues, it's useful to have a persistent at-a-glance view
|
||||||
|
of latency and reachability to multiple well-known internet endpoints. NetWatch
|
||||||
|
provides this as a zero-dependency SPA that can be deployed anywhere static
|
||||||
|
files are served, with no backend required.
|
||||||
|
|
||||||
## Design
|
## Design
|
||||||
|
|
||||||
The application is a single-page app built with Vite and Tailwind CSS v4. All code lives in `src/main.js` with a class-based architecture:
|
The application is a single-page app built with Vite and Tailwind CSS v4. All
|
||||||
|
code lives in `src/main.js` with a class-based architecture:
|
||||||
|
|
||||||
- **`CONFIG`**: Frozen configuration object (update interval, timeouts, axis ticks, etc.)
|
- **`CONFIG`**: Frozen configuration object (update interval, timeouts, axis
|
||||||
- **`HostState`**: Per-host state management — history buffer, latency tracking, status transitions
|
ticks, etc.)
|
||||||
- **`AppState`**: Top-level state container — WAN hosts, local hosts, pause state, aggregate stats
|
- **`HostState`**: Per-host state management — history buffer, latency tracking,
|
||||||
- **`SparklineRenderer`**: Canvas 2D sparkline drawing with fixed axes, color-coded line segments, error regions, and DPR-aware scaling
|
status transitions
|
||||||
- **UI functions**: `buildUI()` constructs the DOM, `updateHostRow()` / `updateSummary()` / `updateHealthBox()` handle incremental updates
|
- **`AppState`**: Top-level state container — WAN hosts, local hosts, pause
|
||||||
- **`tick()`**: Main loop — measures all hosts in parallel via `Promise.all`, pushes samples, redraws UI. When paused, pushes blank markers (no probes, no false outage)
|
state, aggregate stats
|
||||||
|
- **`SparklineRenderer`**: Canvas 2D sparkline drawing with fixed axes,
|
||||||
|
color-coded line segments, error regions, and DPR-aware scaling
|
||||||
|
- **UI functions**: `buildUI()` constructs the DOM, `updateHostRow()` /
|
||||||
|
`updateSummary()` / `updateHealthBox()` handle incremental updates
|
||||||
|
- **`tick()`**: Main loop — measures all hosts in parallel via `Promise.all`,
|
||||||
|
pushes samples, redraws UI. When paused, pushes blank markers (no probes, no
|
||||||
|
false outage)
|
||||||
|
|
||||||
### Monitoring targets
|
### Monitoring targets
|
||||||
|
|
||||||
- **9 WAN hosts**: Google Cloud Console, AWS Console, GitHub, Cloudflare, Azure, DigitalOcean, Fastly, Akamai, datavi.be
|
- **22 WAN hosts**: datavi.be, Anthropic API, OpenAI API, AWS Console, GCP
|
||||||
- **1 Local host**: Local Gateway (192.168.100.1), tracked separately from WAN stats
|
Console, Azure, Cloudflare, Fastly, Akamai, GitHub, B2, 7 S3 regional
|
||||||
|
endpoints (Cape Town, London, Bahrain, Tokyo, Sydney, Oregon, São Paulo), 4
|
||||||
|
GCS locational endpoints (Iowa, Belgium, Singapore, Sydney)
|
||||||
|
- **Local CPE**: Cable modem at 192.168.100.1 (always monitored)
|
||||||
|
- **Local Gateway**: Auto-detected on startup by probing common default gateway
|
||||||
|
addresses (192.168.1.1, 192.168.0.1, 192.168.8.1, 10.0.0.1); first responder
|
||||||
|
wins. Note: modern browsers enforce Private Network Access restrictions that
|
||||||
|
block public-origin pages from reaching RFC1918 addresses, so local targets
|
||||||
|
only work when NetWatch is served from localhost or a private address.
|
||||||
|
|
||||||
|
Local hosts are tracked separately from WAN stats.
|
||||||
|
|
||||||
### Latency measurement
|
### Latency measurement
|
||||||
|
|
||||||
HEAD requests with `mode: 'no-cors'` and `cache: 'no-store'`, timed with `performance.now()`. 1-second timeout; anything over 1000ms is clamped to unreachable. IPv4 only.
|
HEAD requests with `mode: 'no-cors'` and `cache: 'no-store'`, timed with
|
||||||
|
`performance.now()`. 1-second timeout; anything over 1000ms is clamped to
|
||||||
|
unreachable. IPv4 only.
|
||||||
|
|
||||||
### Color coding
|
### Color coding
|
||||||
|
|
||||||
@@ -72,31 +98,40 @@ dist/
|
|||||||
- Summary stats: reachable count, min/max/avg latency across WAN hosts only
|
- Summary stats: reachable count, min/max/avg latency across WAN hosts only
|
||||||
- Fixed chart axes: Y-axis 0–1000ms, X-axis 0–300s
|
- Fixed chart axes: Y-axis 0–1000ms, X-axis 0–300s
|
||||||
- Color-coded latency figures and sparkline line segments
|
- Color-coded latency figures and sparkline line segments
|
||||||
- Play/pause: pause stops probes but history keeps scrolling (blank gaps, no false outage)
|
- Play/pause: pause stops probes but history keeps scrolling (blank gaps, no
|
||||||
|
false outage)
|
||||||
- Clickable service URLs
|
- Clickable service URLs
|
||||||
- Canvas-based sparkline rendering with devicePixelRatio scaling
|
- Canvas-based sparkline rendering with devicePixelRatio scaling
|
||||||
- Zero runtime dependencies: all resources bundled into build artifacts
|
- Zero runtime dependencies: all resources bundled into build artifacts
|
||||||
|
|
||||||
## Deployment
|
## Deployment
|
||||||
|
|
||||||
After running `yarn build`, deploy the contents of the `dist/` directory to any static file host (S3, GCS, Cloudflare Pages, Vercel, Netlify, GitHub Pages) or use the Docker image behind a reverse proxy.
|
After running `yarn build`, deploy the contents of the `dist/` directory to any
|
||||||
|
static file host (S3, GCS, Cloudflare Pages, Vercel, Netlify, GitHub Pages) or
|
||||||
|
use the Docker image behind a reverse proxy.
|
||||||
|
|
||||||
The Docker image:
|
The Docker image:
|
||||||
|
|
||||||
- Listens on port 8080 by default (override with `PORT` env var)
|
- Listens on port 8080 by default (override with `PORT` env var)
|
||||||
- Trusts `X-Forwarded-For` from RFC1918 reverse proxies (10/8, 172.16/12, 192.168/16)
|
- Trusts `X-Forwarded-For` from RFC1918 reverse proxies (10/8, 172.16/12,
|
||||||
|
192.168/16)
|
||||||
- Sends access logs to stdout
|
- Sends access logs to stdout
|
||||||
- Caches static assets with immutable headers
|
- Caches static assets with immutable headers
|
||||||
|
|
||||||
## Browser Compatibility
|
## Browser Compatibility
|
||||||
|
|
||||||
Requires a modern browser with ES modules, Fetch API, Canvas API, and CSS custom properties.
|
Requires a modern browser with ES modules, Fetch API, Canvas API, and CSS custom
|
||||||
|
properties.
|
||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
|
|
||||||
- **CORS**: Some hosts may block cross-origin HEAD requests. The app uses `no-cors` mode which allows the request but provides opaque responses. Latency is still measurable based on request timing.
|
- **CORS**: Some hosts may block cross-origin HEAD requests. The app uses
|
||||||
- **Local gateway**: The 192.168.100.1 endpoint requires the host to be accessible from your network.
|
`no-cors` mode which allows the request but provides opaque responses. Latency
|
||||||
- **Network conditions**: Measurements reflect browser-to-endpoint latency, which includes your local network, ISP, and internet routing.
|
is still measurable based on request timing.
|
||||||
|
- **Local gateway**: The 192.168.100.1 endpoint requires the host to be
|
||||||
|
accessible from your network.
|
||||||
|
- **Network conditions**: Measurements reflect browser-to-endpoint latency,
|
||||||
|
which includes your local network, ISP, and internet routing.
|
||||||
|
|
||||||
## TODO
|
## TODO
|
||||||
|
|
||||||
|
|||||||
@@ -1,23 +1,22 @@
|
|||||||
# Development Policies
|
# Development Policies
|
||||||
|
|
||||||
- Docker image references by tag are server-mutable, therefore using them is
|
- Docker image references by tag are server-mutable, therefore using them is an
|
||||||
an RCE vulnerability. All docker image references must use cryptographic
|
RCE vulnerability. All docker image references must use cryptographic hashes
|
||||||
hashes to securely specify the exact image that is expected.
|
to securely specify the exact image that is expected.
|
||||||
|
|
||||||
- Correspondingly, `go install` commands using things like '@latest' are
|
- Correspondingly, `go install` commands using things like '@latest' are also
|
||||||
also dangerous RCE. Whenever writing scripts or tools, ALWAYS specify go
|
dangerous RCE. Whenever writing scripts or tools, ALWAYS specify go install
|
||||||
install targets using commit hashes which are cryptographically secure.
|
targets using commit hashes which are cryptographically secure.
|
||||||
|
|
||||||
- Every repo with software in it must have a Makefile in the root. Each
|
- Every repo with software in it must have a Makefile in the root. Each such
|
||||||
such Makefile should support `make test` (runs the project-specific
|
Makefile should support `make test` (runs the project-specific tests),
|
||||||
tests), `make lint`, `make fmt` (writes), `make fmt-check` (readonly), and
|
`make lint`, `make fmt` (writes), `make fmt-check` (readonly), and
|
||||||
`make check` (has `test`, `lint`, and `fmt-check` as prereqs), `make
|
`make check` (has `test`, `lint`, and `fmt-check` as prereqs), `make docker`
|
||||||
docker` (builds docker image).
|
(builds docker image).
|
||||||
|
|
||||||
- Every repo should have a Dockerfile. If the repo contains non-server
|
- Every repo should have a Dockerfile. If the repo contains non-server software,
|
||||||
software, the Dockerfile should bring up a development environment and
|
the Dockerfile should bring up a development environment and `make check`
|
||||||
`make check` (i.e. the docker build should fail if the branch is not
|
(i.e. the docker build should fail if the branch is not green).
|
||||||
green).
|
|
||||||
|
|
||||||
- Platform-specific standard formatting should be used. `black` for python,
|
- Platform-specific standard formatting should be used. `black` for python,
|
||||||
`prettier` for js/css/etc, `go fmt` for go. The only changes to default
|
`prettier` for js/css/etc, `go fmt` for go. The only changes to default
|
||||||
@@ -25,27 +24,27 @@ docker` (builds docker image).
|
|||||||
everything except `go fmt`).
|
everything except `go fmt`).
|
||||||
|
|
||||||
- If local testing is possible (it is not always), `make check` should be a
|
- If local testing is possible (it is not always), `make check` should be a
|
||||||
pre-commit hook. If it is not possible, `make lint && make fmt-check`
|
pre-commit hook. If it is not possible, `make lint && make fmt-check` should
|
||||||
should be a pre-commit hook.
|
be a pre-commit hook.
|
||||||
|
|
||||||
- If a working `make test` takes more than 20 seconds, that's a bug that
|
- If a working `make test` takes more than 20 seconds, that's a bug that needs
|
||||||
needs fixing. In fact, there should be a timeout specified in the
|
fixing. In fact, there should be a timeout specified in the `Makefile` that
|
||||||
`Makefile` that fails it automatically if it takes >30s.
|
fails it automatically if it takes >30s.
|
||||||
|
|
||||||
- Docker builds should time out in 5 minutes or less.
|
- Docker builds should time out in 5 minutes or less.
|
||||||
|
|
||||||
- `main` must always pass `make check`, no exceptions.
|
- `main` must always pass `make check`, no exceptions.
|
||||||
|
|
||||||
- Do all changes on a feature branch. You can do whatever you want on a
|
- Do all changes on a feature branch. You can do whatever you want on a feature
|
||||||
feature branch.
|
branch.
|
||||||
|
|
||||||
- We have a standardized `.golangci.yml` which we reuse and is _NEVER_ to be
|
- We have a standardized `.golangci.yml` which we reuse and is _NEVER_ to be
|
||||||
modified by an agent, only manually by the user. It can be copied from
|
modified by an agent, only manually by the user. It can be copied from
|
||||||
`~/dev/upaas/.golangci.yml` if it exists at that location.
|
`~/dev/upaas/.golangci.yml` if it exists at that location.
|
||||||
|
|
||||||
- When specifying images or packages by hash in Dockerfiles or
|
- When specifying images or packages by hash in Dockerfiles or
|
||||||
`docker-compose.yml`, put a comment above the line and show the version
|
`docker-compose.yml`, put a comment above the line and show the version and
|
||||||
and date at which it was current.
|
date at which it was current.
|
||||||
|
|
||||||
- For javascript, always use `yarn` over `npm`.
|
- For javascript, always use `yarn` over `npm`.
|
||||||
|
|
||||||
@@ -54,11 +53,11 @@ docker` (builds docker image).
|
|||||||
- Simple projects should be configured with environment variables, as is
|
- Simple projects should be configured with environment variables, as is
|
||||||
standard for Dockerized applications.
|
standard for Dockerized applications.
|
||||||
|
|
||||||
- Dockerized web services should listen on the default HTTP port of 8080
|
- Dockerized web services should listen on the default HTTP port of 8080 unless
|
||||||
unless overridden with the `PORT` environment variable.
|
overridden with the `PORT` environment variable.
|
||||||
|
|
||||||
- The `README.md` is a project's primary documentation. It should contain
|
- The `README.md` is a project's primary documentation. It should contain at a
|
||||||
at a minimum the following sections:
|
minimum the following sections:
|
||||||
- Description
|
- Description
|
||||||
- Include a short and complete description of the functionality and
|
- Include a short and complete description of the functionality and
|
||||||
purpose of the software as the first line in the readme. It must
|
purpose of the software as the first line in the readme. It must
|
||||||
@@ -68,10 +67,10 @@ docker` (builds docker image).
|
|||||||
- the category (web server, SPA, command line tool, etc)
|
- the category (web server, SPA, command line tool, etc)
|
||||||
- the license
|
- the license
|
||||||
- the author
|
- the author
|
||||||
- eg: "µPaaS is an MIT-licensed Go web application by @sneak
|
- eg: "µPaaS is an MIT-licensed Go web application by @sneak that
|
||||||
that receives git-frontend webhooks and interacts with a
|
receives git-frontend webhooks and interacts with a Docker server
|
||||||
Docker server to build and deploy applications in realtime as
|
to build and deploy applications in realtime as certain branches
|
||||||
certain branches are updated."
|
are updated."
|
||||||
- Getting Started
|
- Getting Started
|
||||||
- a code block with copy-pasteable installation/use sections
|
- a code block with copy-pasteable installation/use sections
|
||||||
- Rationale
|
- Rationale
|
||||||
@@ -79,28 +78,27 @@ docker` (builds docker image).
|
|||||||
- Design
|
- Design
|
||||||
- how is the program structured?
|
- how is the program structured?
|
||||||
- TODO
|
- TODO
|
||||||
- This is your TODO list for the project - update it meticulously,
|
- This is your TODO list for the project - update it meticulously, even
|
||||||
even in between commits. Whenever planning, put your todo list in
|
in between commits. Whenever planning, put your todo list in the
|
||||||
the README so that a separate agent with new context can pick up
|
README so that a separate agent with new context can pick up where you
|
||||||
where you left off.
|
left off.
|
||||||
- License
|
- License
|
||||||
- GPL or MIT or WTFPL - ask the user when beginning a new project
|
- GPL or MIT or WTFPL - ask the user when beginning a new project and
|
||||||
and include a LICENSE file in the root and in a section in the
|
include a LICENSE file in the root and in a section in the README.
|
||||||
README.
|
|
||||||
- Author
|
- Author
|
||||||
- @sneak (link `@sneak` to `https://sneak.berlin`).
|
- @sneak (link `@sneak` to `https://sneak.berlin`).
|
||||||
|
|
||||||
- When beginning a new project, initialize a git repo and make the first
|
- When beginning a new project, initialize a git repo and make the first commit
|
||||||
commit simply the first version of the README.md in the root of the repo.
|
simply the first version of the README.md in the root of the repo.
|
||||||
|
|
||||||
- For Go packages, the module root is `sneak.berlin/go/...`, such
|
- For Go packages, the module root is `sneak.berlin/go/...`, such as
|
||||||
as `sneak.berlin/go/dnswatcher`.
|
`sneak.berlin/go/dnswatcher`.
|
||||||
|
|
||||||
- We use SemVer always.
|
- We use SemVer always.
|
||||||
|
|
||||||
- If no tag `1.0.0` or greater exists in the repository, modify the existing
|
- If no tag `1.0.0` or greater exists in the repository, modify the existing
|
||||||
migrations and assume no installed base or existing databases. If
|
migrations and assume no installed base or existing databases. If `>=1.0.0`,
|
||||||
`>=1.0.0`, database changes add new migration files.
|
database changes add new migration files.
|
||||||
|
|
||||||
- New repos must have at a minimum the following files:
|
- New repos must have at a minimum the following files:
|
||||||
- `README.md`, `.git`, `.gitignore`
|
- `README.md`, `.git`, `.gitignore`
|
||||||
|
|||||||
2
backend/.dockerignore
Normal file
2
backend/.dockerignore
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
node_modules
|
||||||
|
.DS_Store
|
||||||
12
backend/.editorconfig
Normal file
12
backend/.editorconfig
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 4
|
||||||
|
end_of_line = lf
|
||||||
|
charset = utf-8
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
insert_final_newline = true
|
||||||
|
|
||||||
|
[Makefile]
|
||||||
|
indent_style = tab
|
||||||
6
backend/.gitignore
vendored
Normal file
6
backend/.gitignore
vendored
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
/netwatch-server
|
||||||
|
*.log
|
||||||
|
*.out
|
||||||
|
*.test
|
||||||
|
.env
|
||||||
|
data/
|
||||||
32
backend/.golangci.yml
Normal file
32
backend/.golangci.yml
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
version: "2"
|
||||||
|
|
||||||
|
run:
|
||||||
|
timeout: 5m
|
||||||
|
modules-download-mode: readonly
|
||||||
|
|
||||||
|
linters:
|
||||||
|
default: all
|
||||||
|
disable:
|
||||||
|
# Genuinely incompatible with project patterns
|
||||||
|
- exhaustruct # Requires all struct fields
|
||||||
|
- depguard # Dependency allow/block lists
|
||||||
|
- godot # Requires comments to end with periods
|
||||||
|
- wsl # Deprecated, replaced by wsl_v5
|
||||||
|
- wrapcheck # Too verbose for internal packages
|
||||||
|
- varnamelen # Short names like db, id are idiomatic Go
|
||||||
|
|
||||||
|
linters-settings:
|
||||||
|
lll:
|
||||||
|
line-length: 88
|
||||||
|
funlen:
|
||||||
|
lines: 80
|
||||||
|
statements: 50
|
||||||
|
cyclop:
|
||||||
|
max-complexity: 15
|
||||||
|
dupl:
|
||||||
|
threshold: 100
|
||||||
|
|
||||||
|
issues:
|
||||||
|
exclude-use-default: false
|
||||||
|
max-issues-per-linter: 0
|
||||||
|
max-same-issues: 0
|
||||||
21
backend/LICENSE
Normal file
21
backend/LICENSE
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2026 @sneak (https://sneak.berlin)
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
53
backend/Makefile
Normal file
53
backend/Makefile
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
UNAME_S := $(shell uname -s)
|
||||||
|
VERSION := $(shell git describe --always --dirty)
|
||||||
|
BUILDARCH := $(shell uname -m)
|
||||||
|
BINARY := netwatch-server
|
||||||
|
|
||||||
|
GOLDFLAGS += -X main.Version=$(VERSION)
|
||||||
|
GOLDFLAGS += -X main.Buildarch=$(BUILDARCH)
|
||||||
|
|
||||||
|
ifeq ($(UNAME_S),Darwin)
|
||||||
|
GOFLAGS := -ldflags "$(GOLDFLAGS)"
|
||||||
|
else
|
||||||
|
GOFLAGS = -ldflags "-linkmode external -extldflags -static $(GOLDFLAGS)"
|
||||||
|
endif
|
||||||
|
|
||||||
|
.PHONY: all build test lint fmt fmt-check check docker hooks run clean
|
||||||
|
|
||||||
|
all: build
|
||||||
|
|
||||||
|
build: ./$(BINARY)
|
||||||
|
|
||||||
|
./$(BINARY): $(shell find . -name '*.go' -type f) go.mod go.sum
|
||||||
|
go build -o $@ $(GOFLAGS) ./cmd/netwatch-server/
|
||||||
|
|
||||||
|
test:
|
||||||
|
timeout 30 go test ./...
|
||||||
|
|
||||||
|
lint:
|
||||||
|
golangci-lint run ./...
|
||||||
|
|
||||||
|
fmt:
|
||||||
|
go fmt ./...
|
||||||
|
|
||||||
|
fmt-check:
|
||||||
|
@test -z "$$(gofmt -l .)" || \
|
||||||
|
(echo "Files not formatted:"; gofmt -l .; exit 1)
|
||||||
|
|
||||||
|
check: test lint fmt-check
|
||||||
|
|
||||||
|
docker:
|
||||||
|
timeout 300 docker build -t netwatch-server -f ../Dockerfile.backend ..
|
||||||
|
|
||||||
|
hooks:
|
||||||
|
@printf '#!/bin/sh\ncd backend && make check\n' > \
|
||||||
|
$$(git rev-parse --show-toplevel)/.git/hooks/pre-commit
|
||||||
|
@chmod +x \
|
||||||
|
$$(git rev-parse --show-toplevel)/.git/hooks/pre-commit
|
||||||
|
@echo "Pre-commit hook installed"
|
||||||
|
|
||||||
|
run: build
|
||||||
|
./$(BINARY)
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -f ./$(BINARY)
|
||||||
70
backend/README.md
Normal file
70
backend/README.md
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
netwatch-server is an MIT-licensed Go HTTP backend by
|
||||||
|
[@sneak](https://sneak.berlin) that receives telemetry reports from the NetWatch
|
||||||
|
SPA and persists them as zstd-compressed JSONL files on disk.
|
||||||
|
|
||||||
|
## Getting Started
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Build and run locally
|
||||||
|
make run
|
||||||
|
|
||||||
|
# Run tests, lint, and format check
|
||||||
|
make check
|
||||||
|
|
||||||
|
# Docker
|
||||||
|
docker build -t netwatch-server .
|
||||||
|
docker run -p 8080:8080 netwatch-server
|
||||||
|
```
|
||||||
|
|
||||||
|
## Rationale
|
||||||
|
|
||||||
|
The NetWatch frontend collects latency measurements from the browser but has no
|
||||||
|
way to persist or aggregate them. This backend provides a minimal
|
||||||
|
`POST /api/v1/reports` endpoint that buffers incoming reports in memory and
|
||||||
|
flushes them to compressed files on disk for later analysis.
|
||||||
|
|
||||||
|
## Design
|
||||||
|
|
||||||
|
The server is structured as an `fx`-wired Go application under `cmd/netwatch-server/`.
|
||||||
|
Internal packages in `internal/` follow standard Go project layout:
|
||||||
|
|
||||||
|
- **`config`**: Loads configuration from environment variables and config files
|
||||||
|
via Viper.
|
||||||
|
- **`handlers`**: HTTP request handlers for the API (health check, report
|
||||||
|
ingestion).
|
||||||
|
- **`reportbuf`**: In-memory buffer that accumulates JSONL report lines and
|
||||||
|
flushes to zstd-compressed files when the buffer reaches 10 MiB or every 60
|
||||||
|
seconds.
|
||||||
|
- **`server`**: Chi-based HTTP server with middleware wiring and route
|
||||||
|
registration.
|
||||||
|
- **`healthcheck`**, **`middleware`**, **`logger`**, **`globals`**: Supporting
|
||||||
|
infrastructure.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
| Variable | Default | Description |
|
||||||
|
| ---------- | ------------------ | --------------------------------- |
|
||||||
|
| `PORT` | `8080` | HTTP listen port |
|
||||||
|
| `DATA_DIR` | `./data/reports` | Directory for compressed reports |
|
||||||
|
| `DEBUG` | `false` | Enable debug logging |
|
||||||
|
|
||||||
|
### Report storage
|
||||||
|
|
||||||
|
Reports are written as `reports-<timestamp>.jsonl.zst` files in `DATA_DIR`.
|
||||||
|
Each file contains one JSON object per line, compressed with zstd. Files are
|
||||||
|
created with `O_EXCL` to prevent overwrites.
|
||||||
|
|
||||||
|
## TODO
|
||||||
|
|
||||||
|
- Add integration test that POSTs a report and verifies the compressed output
|
||||||
|
- Add report decompression/query endpoint
|
||||||
|
- Add metrics (Prometheus) for buffer size, flush count, report count
|
||||||
|
- Add retention policy to prune old report files
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
MIT. See [LICENSE](LICENSE).
|
||||||
|
|
||||||
|
## Author
|
||||||
|
|
||||||
|
[@sneak](https://sneak.berlin)
|
||||||
42
backend/cmd/netwatch-server/main.go
Normal file
42
backend/cmd/netwatch-server/main.go
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
// Package main is the entry point for netwatch-server.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sneak.berlin/go/netwatch/internal/config"
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
"sneak.berlin/go/netwatch/internal/handlers"
|
||||||
|
"sneak.berlin/go/netwatch/internal/healthcheck"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
"sneak.berlin/go/netwatch/internal/middleware"
|
||||||
|
"sneak.berlin/go/netwatch/internal/reportbuf"
|
||||||
|
"sneak.berlin/go/netwatch/internal/server"
|
||||||
|
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
//nolint:gochecknoglobals // set via ldflags at build time
|
||||||
|
var (
|
||||||
|
Appname = "netwatch-server"
|
||||||
|
Version string
|
||||||
|
Buildarch string
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
globals.Appname = Appname
|
||||||
|
globals.Version = Version
|
||||||
|
globals.Buildarch = Buildarch
|
||||||
|
|
||||||
|
fx.New(
|
||||||
|
fx.Provide(
|
||||||
|
config.New,
|
||||||
|
globals.New,
|
||||||
|
handlers.New,
|
||||||
|
healthcheck.New,
|
||||||
|
logger.New,
|
||||||
|
middleware.New,
|
||||||
|
reportbuf.New,
|
||||||
|
server.New,
|
||||||
|
),
|
||||||
|
fx.Invoke(func(*server.Server) {}),
|
||||||
|
).Run()
|
||||||
|
}
|
||||||
30
backend/go.mod
Normal file
30
backend/go.mod
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
module sneak.berlin/go/netwatch
|
||||||
|
|
||||||
|
go 1.25.5
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/go-chi/chi/v5 v5.2.5
|
||||||
|
github.com/go-chi/cors v1.2.2
|
||||||
|
github.com/joho/godotenv v1.5.1
|
||||||
|
github.com/klauspost/compress v1.18.4
|
||||||
|
github.com/spf13/viper v1.21.0
|
||||||
|
go.uber.org/fx v1.24.0
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/fsnotify/fsnotify v1.9.0 // indirect
|
||||||
|
github.com/go-viper/mapstructure/v2 v2.4.0 // indirect
|
||||||
|
github.com/pelletier/go-toml/v2 v2.2.4 // indirect
|
||||||
|
github.com/sagikazarmark/locafero v0.11.0 // indirect
|
||||||
|
github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 // indirect
|
||||||
|
github.com/spf13/afero v1.15.0 // indirect
|
||||||
|
github.com/spf13/cast v1.10.0 // indirect
|
||||||
|
github.com/spf13/pflag v1.0.10 // indirect
|
||||||
|
github.com/subosito/gotenv v1.6.0 // indirect
|
||||||
|
go.uber.org/dig v1.19.0 // indirect
|
||||||
|
go.uber.org/multierr v1.10.0 // indirect
|
||||||
|
go.uber.org/zap v1.26.0 // indirect
|
||||||
|
go.yaml.in/yaml/v3 v3.0.4 // indirect
|
||||||
|
golang.org/x/sys v0.29.0 // indirect
|
||||||
|
golang.org/x/text v0.28.0 // indirect
|
||||||
|
)
|
||||||
65
backend/go.sum
Normal file
65
backend/go.sum
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
|
||||||
|
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
|
||||||
|
github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k=
|
||||||
|
github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0=
|
||||||
|
github.com/go-chi/chi/v5 v5.2.5 h1:Eg4myHZBjyvJmAFjFvWgrqDTXFyOzjj7YIm3L3mu6Ug=
|
||||||
|
github.com/go-chi/chi/v5 v5.2.5/go.mod h1:X7Gx4mteadT3eDOMTsXzmI4/rwUpOwBHLpAfupzFJP0=
|
||||||
|
github.com/go-chi/cors v1.2.2 h1:Jmey33TE+b+rB7fT8MUy1u0I4L+NARQlK6LhzKPSyQE=
|
||||||
|
github.com/go-chi/cors v1.2.2/go.mod h1:sSbTewc+6wYHBBCW7ytsFSn836hqM7JxpglAy2Vzc58=
|
||||||
|
github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs=
|
||||||
|
github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
|
||||||
|
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
||||||
|
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||||
|
github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
|
||||||
|
github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
|
||||||
|
github.com/klauspost/compress v1.18.4 h1:RPhnKRAQ4Fh8zU2FY/6ZFDwTVTxgJ/EMydqSTzE9a2c=
|
||||||
|
github.com/klauspost/compress v1.18.4/go.mod h1:R0h/fSBs8DE4ENlcrlib3PsXS61voFxhIs2DeRhCvJ4=
|
||||||
|
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
|
||||||
|
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
|
||||||
|
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
|
||||||
|
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
|
||||||
|
github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4=
|
||||||
|
github.com/pelletier/go-toml/v2 v2.2.4/go.mod h1:2gIqNv+qfxSVS7cM2xJQKtLSTLUE9V8t9Stt+h56mCY=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
|
github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8=
|
||||||
|
github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs=
|
||||||
|
github.com/sagikazarmark/locafero v0.11.0 h1:1iurJgmM9G3PA/I+wWYIOw/5SyBtxapeHDcg+AAIFXc=
|
||||||
|
github.com/sagikazarmark/locafero v0.11.0/go.mod h1:nVIGvgyzw595SUSUE6tvCp3YYTeHs15MvlmU87WwIik=
|
||||||
|
github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 h1:+jumHNA0Wrelhe64i8F6HNlS8pkoyMv5sreGx2Ry5Rw=
|
||||||
|
github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8/go.mod h1:3n1Cwaq1E1/1lhQhtRK2ts/ZwZEhjcQeJQ1RuC6Q/8U=
|
||||||
|
github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I=
|
||||||
|
github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg=
|
||||||
|
github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY=
|
||||||
|
github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo=
|
||||||
|
github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk=
|
||||||
|
github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||||
|
github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU=
|
||||||
|
github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY=
|
||||||
|
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
|
||||||
|
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
|
||||||
|
github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8=
|
||||||
|
github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU=
|
||||||
|
go.uber.org/dig v1.19.0 h1:BACLhebsYdpQ7IROQ1AGPjrXcP5dF80U3gKoFzbaq/4=
|
||||||
|
go.uber.org/dig v1.19.0/go.mod h1:Us0rSJiThwCv2GteUN0Q7OKvU7n5J4dxZ9JKUXozFdE=
|
||||||
|
go.uber.org/fx v1.24.0 h1:wE8mruvpg2kiiL1Vqd0CC+tr0/24XIB10Iwp2lLWzkg=
|
||||||
|
go.uber.org/fx v1.24.0/go.mod h1:AmDeGyS+ZARGKM4tlH4FY2Jr63VjbEDJHtqXTGP5hbo=
|
||||||
|
go.uber.org/goleak v1.2.0 h1:xqgm/S+aQvhWFTtR0XK3Jvg7z8kGV8P4X14IzwN3Eqk=
|
||||||
|
go.uber.org/goleak v1.2.0/go.mod h1:XJYK+MuIchqpmGmUSAzotztawfKvYLUIgg7guXrwVUo=
|
||||||
|
go.uber.org/multierr v1.10.0 h1:S0h4aNzvfcFsC3dRF1jLoaov7oRaKqRGC/pUEJ2yvPQ=
|
||||||
|
go.uber.org/multierr v1.10.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
|
||||||
|
go.uber.org/zap v1.26.0 h1:sI7k6L95XOKS281NhVKOFCUNIvv9e0w4BF8N3u+tCRo=
|
||||||
|
go.uber.org/zap v1.26.0/go.mod h1:dtElttAiwGvoJ/vj4IwHBS/gXsEu/pZ50mUIRWuG0so=
|
||||||
|
go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc=
|
||||||
|
go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg=
|
||||||
|
golang.org/x/sys v0.29.0 h1:TPYlXGxvx1MGTn2GiZDhnjPA9wZzZeGKHHmKhHYvgaU=
|
||||||
|
golang.org/x/sys v0.29.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||||
|
golang.org/x/text v0.28.0 h1:rhazDwis8INMIwQ4tpjLDzUhx6RlXqZNPEM0huQojng=
|
||||||
|
golang.org/x/text v0.28.0/go.mod h1:U8nCwOR8jO/marOQ0QbDiOngZVEBB7MAiitBuMjXiNU=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
|
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
86
backend/internal/config/config.go
Normal file
86
backend/internal/config/config.go
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
// Package config loads application configuration from
|
||||||
|
// environment variables, .env files, and config files.
|
||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"log/slog"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
|
||||||
|
_ "github.com/joho/godotenv/autoload" // loads .env file
|
||||||
|
"github.com/spf13/viper"
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Params defines the dependencies for Config.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Globals *globals.Globals
|
||||||
|
Logger *logger.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Config holds the resolved application configuration.
|
||||||
|
type Config struct {
|
||||||
|
DataDir string
|
||||||
|
Debug bool
|
||||||
|
MetricsPassword string
|
||||||
|
MetricsUsername string
|
||||||
|
Port int
|
||||||
|
SentryDSN string
|
||||||
|
log *slog.Logger
|
||||||
|
params *Params
|
||||||
|
}
|
||||||
|
|
||||||
|
// New loads configuration from env, .env files, and config
|
||||||
|
// files, returning a fully resolved Config.
|
||||||
|
func New(
|
||||||
|
_ fx.Lifecycle,
|
||||||
|
params Params,
|
||||||
|
) (*Config, error) {
|
||||||
|
log := params.Logger.Get()
|
||||||
|
name := params.Globals.Appname
|
||||||
|
|
||||||
|
viper.SetConfigName(name)
|
||||||
|
viper.SetConfigType("yaml")
|
||||||
|
viper.AddConfigPath("/etc/" + name)
|
||||||
|
viper.AddConfigPath("$HOME/.config/" + name)
|
||||||
|
|
||||||
|
viper.AutomaticEnv()
|
||||||
|
|
||||||
|
viper.SetDefault("DATA_DIR", "./data/reports")
|
||||||
|
viper.SetDefault("DEBUG", "false")
|
||||||
|
viper.SetDefault("PORT", "8080")
|
||||||
|
viper.SetDefault("SENTRY_DSN", "")
|
||||||
|
viper.SetDefault("METRICS_USERNAME", "")
|
||||||
|
viper.SetDefault("METRICS_PASSWORD", "")
|
||||||
|
|
||||||
|
err := viper.ReadInConfig()
|
||||||
|
if err != nil {
|
||||||
|
var notFound viper.ConfigFileNotFoundError
|
||||||
|
if !errors.As(err, ¬Found) {
|
||||||
|
log.Error("config file malformed", "error", err)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
s := &Config{
|
||||||
|
DataDir: viper.GetString("DATA_DIR"),
|
||||||
|
Debug: viper.GetBool("DEBUG"),
|
||||||
|
MetricsPassword: viper.GetString("METRICS_PASSWORD"),
|
||||||
|
MetricsUsername: viper.GetString("METRICS_USERNAME"),
|
||||||
|
Port: viper.GetInt("PORT"),
|
||||||
|
SentryDSN: viper.GetString("SENTRY_DSN"),
|
||||||
|
log: log,
|
||||||
|
params: ¶ms,
|
||||||
|
}
|
||||||
|
|
||||||
|
if s.Debug {
|
||||||
|
params.Logger.EnableDebugLogging()
|
||||||
|
s.log = params.Logger.Get()
|
||||||
|
}
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
31
backend/internal/globals/globals.go
Normal file
31
backend/internal/globals/globals.go
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
// Package globals provides build-time variables injected via
|
||||||
|
// ldflags and made available through dependency injection.
|
||||||
|
package globals
|
||||||
|
|
||||||
|
import "go.uber.org/fx"
|
||||||
|
|
||||||
|
//nolint:gochecknoglobals // set from main before fx starts
|
||||||
|
var (
|
||||||
|
// Appname is the application name.
|
||||||
|
Appname string
|
||||||
|
// Version is the git version tag.
|
||||||
|
Version string
|
||||||
|
// Buildarch is the build architecture.
|
||||||
|
Buildarch string
|
||||||
|
)
|
||||||
|
|
||||||
|
// Globals holds build-time metadata for the application.
|
||||||
|
type Globals struct {
|
||||||
|
Appname string
|
||||||
|
Version string
|
||||||
|
Buildarch string
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Globals instance from package-level variables.
|
||||||
|
func New(_ fx.Lifecycle) (*Globals, error) {
|
||||||
|
return &Globals{
|
||||||
|
Appname: Appname,
|
||||||
|
Buildarch: Buildarch,
|
||||||
|
Version: Version,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
74
backend/internal/handlers/handlers.go
Normal file
74
backend/internal/handlers/handlers.go
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
// Package handlers implements HTTP request handlers for the
|
||||||
|
// netwatch-server API.
|
||||||
|
package handlers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"log/slog"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
"sneak.berlin/go/netwatch/internal/healthcheck"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
"sneak.berlin/go/netwatch/internal/reportbuf"
|
||||||
|
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
const jsonContentType = "application/json; charset=utf-8"
|
||||||
|
|
||||||
|
// Params defines the dependencies for Handlers.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Buffer *reportbuf.Buffer
|
||||||
|
Globals *globals.Globals
|
||||||
|
Healthcheck *healthcheck.Healthcheck
|
||||||
|
Logger *logger.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handlers provides HTTP handler factories for all endpoints.
|
||||||
|
type Handlers struct {
|
||||||
|
buf *reportbuf.Buffer
|
||||||
|
hc *healthcheck.Healthcheck
|
||||||
|
log *slog.Logger
|
||||||
|
params *Params
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Handlers instance.
|
||||||
|
func New(
|
||||||
|
lc fx.Lifecycle,
|
||||||
|
params Params,
|
||||||
|
) (*Handlers, error) {
|
||||||
|
s := new(Handlers)
|
||||||
|
s.buf = params.Buffer
|
||||||
|
s.params = ¶ms
|
||||||
|
s.log = params.Logger.Get()
|
||||||
|
s.hc = params.Healthcheck
|
||||||
|
|
||||||
|
lc.Append(fx.Hook{
|
||||||
|
OnStart: func(_ context.Context) error {
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Handlers) respondJSON(
|
||||||
|
w http.ResponseWriter,
|
||||||
|
_ *http.Request,
|
||||||
|
data any,
|
||||||
|
status int,
|
||||||
|
) {
|
||||||
|
w.Header().Set("Content-Type", jsonContentType)
|
||||||
|
w.WriteHeader(status)
|
||||||
|
|
||||||
|
if data != nil {
|
||||||
|
err := json.NewEncoder(w).Encode(data)
|
||||||
|
if err != nil {
|
||||||
|
s.log.Error("json encode error", "error", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
13
backend/internal/handlers/handlers_test.go
Normal file
13
backend/internal/handlers/handlers_test.go
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
package handlers_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
_ "sneak.berlin/go/netwatch/internal/handlers"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestImport(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
// Compilation check — verifies the package parses
|
||||||
|
// and all imports resolve.
|
||||||
|
}
|
||||||
11
backend/internal/handlers/healthcheck.go
Normal file
11
backend/internal/handlers/healthcheck.go
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
package handlers
|
||||||
|
|
||||||
|
import "net/http"
|
||||||
|
|
||||||
|
// HandleHealthCheck returns a handler for the health check
|
||||||
|
// endpoint.
|
||||||
|
func (s *Handlers) HandleHealthCheck() http.HandlerFunc {
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
s.respondJSON(w, r, s.hc.Check(), http.StatusOK)
|
||||||
|
}
|
||||||
|
}
|
||||||
82
backend/internal/handlers/report.go
Normal file
82
backend/internal/handlers/report.go
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
package handlers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
const maxReportBodyBytes = 1 << 20 // 1 MiB
|
||||||
|
|
||||||
|
type reportSample struct {
|
||||||
|
T int64 `json:"t"`
|
||||||
|
Latency *int `json:"latency"`
|
||||||
|
Error *string `json:"error"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type reportHost struct {
|
||||||
|
History []reportSample `json:"history"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
URL string `json:"url"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type report struct {
|
||||||
|
ClientID string `json:"clientId"`
|
||||||
|
Geo json.RawMessage `json:"geo"`
|
||||||
|
Hosts []reportHost `json:"hosts"`
|
||||||
|
Timestamp string `json:"timestamp"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleReport returns a handler that accepts telemetry
|
||||||
|
// reports from NetWatch clients.
|
||||||
|
func (s *Handlers) HandleReport() http.HandlerFunc {
|
||||||
|
type response struct {
|
||||||
|
Status string `json:"status"`
|
||||||
|
}
|
||||||
|
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
r.Body = http.MaxBytesReader(
|
||||||
|
w, r.Body, maxReportBodyBytes,
|
||||||
|
)
|
||||||
|
|
||||||
|
var rpt report
|
||||||
|
|
||||||
|
err := json.NewDecoder(r.Body).Decode(&rpt)
|
||||||
|
if err != nil {
|
||||||
|
s.log.Error("failed to decode report",
|
||||||
|
"error", err,
|
||||||
|
)
|
||||||
|
s.respondJSON(w, r,
|
||||||
|
&response{Status: "error"},
|
||||||
|
http.StatusBadRequest,
|
||||||
|
)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
totalSamples := 0
|
||||||
|
for _, h := range rpt.Hosts {
|
||||||
|
totalSamples += len(h.History)
|
||||||
|
}
|
||||||
|
|
||||||
|
s.log.Info("report received",
|
||||||
|
"client_id", rpt.ClientID,
|
||||||
|
"timestamp", rpt.Timestamp,
|
||||||
|
"host_count", len(rpt.Hosts),
|
||||||
|
"total_samples", totalSamples,
|
||||||
|
"geo", string(rpt.Geo),
|
||||||
|
)
|
||||||
|
|
||||||
|
bufErr := s.buf.Append(rpt)
|
||||||
|
if bufErr != nil {
|
||||||
|
s.log.Error("failed to buffer report",
|
||||||
|
"error", bufErr,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
s.respondJSON(w, r,
|
||||||
|
&response{Status: "ok"},
|
||||||
|
http.StatusOK,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
82
backend/internal/healthcheck/healthcheck.go
Normal file
82
backend/internal/healthcheck/healthcheck.go
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
// Package healthcheck provides a service that reports
|
||||||
|
// application health, uptime, and version information.
|
||||||
|
package healthcheck
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log/slog"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/config"
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Params defines the dependencies for Healthcheck.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Config *config.Config
|
||||||
|
Globals *globals.Globals
|
||||||
|
Logger *logger.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Healthcheck tracks startup time and builds health responses.
|
||||||
|
type Healthcheck struct {
|
||||||
|
StartupTime time.Time
|
||||||
|
log *slog.Logger
|
||||||
|
params *Params
|
||||||
|
}
|
||||||
|
|
||||||
|
// Response is the JSON payload returned by the health check
|
||||||
|
// endpoint.
|
||||||
|
type Response struct {
|
||||||
|
Appname string `json:"appname"`
|
||||||
|
Now string `json:"now"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
UptimeHuman string `json:"uptimeHuman"`
|
||||||
|
UptimeSeconds int64 `json:"uptimeSeconds"`
|
||||||
|
Version string `json:"version"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Healthcheck, recording startup time via an
|
||||||
|
// fx lifecycle hook.
|
||||||
|
func New(
|
||||||
|
lc fx.Lifecycle,
|
||||||
|
params Params,
|
||||||
|
) (*Healthcheck, error) {
|
||||||
|
s := new(Healthcheck)
|
||||||
|
s.params = ¶ms
|
||||||
|
s.log = params.Logger.Get()
|
||||||
|
|
||||||
|
lc.Append(fx.Hook{
|
||||||
|
OnStart: func(_ context.Context) error {
|
||||||
|
s.StartupTime = time.Now().UTC()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
OnStop: func(_ context.Context) error {
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check returns the current health status of the application.
|
||||||
|
func (s *Healthcheck) Check() *Response {
|
||||||
|
return &Response{
|
||||||
|
Appname: s.params.Globals.Appname,
|
||||||
|
Now: time.Now().UTC().Format(time.RFC3339Nano),
|
||||||
|
Status: "ok",
|
||||||
|
UptimeHuman: s.uptime().String(),
|
||||||
|
UptimeSeconds: int64(s.uptime().Seconds()),
|
||||||
|
Version: s.params.Globals.Version,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Healthcheck) uptime() time.Duration {
|
||||||
|
return time.Since(s.StartupTime)
|
||||||
|
}
|
||||||
85
backend/internal/logger/logger.go
Normal file
85
backend/internal/logger/logger.go
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
// Package logger provides a configured slog.Logger with TTY
|
||||||
|
// detection for development vs production output.
|
||||||
|
package logger
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log/slog"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Params defines the dependencies for Logger.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Globals *globals.Globals
|
||||||
|
}
|
||||||
|
|
||||||
|
// Logger wraps slog.Logger with dynamic level control.
|
||||||
|
type Logger struct {
|
||||||
|
log *slog.Logger
|
||||||
|
level *slog.LevelVar
|
||||||
|
params Params
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Logger with TTY-aware output formatting.
|
||||||
|
func New(_ fx.Lifecycle, params Params) (*Logger, error) {
|
||||||
|
l := new(Logger)
|
||||||
|
l.level = new(slog.LevelVar)
|
||||||
|
l.level.Set(slog.LevelInfo)
|
||||||
|
l.params = params
|
||||||
|
|
||||||
|
tty := false
|
||||||
|
|
||||||
|
if fileInfo, _ := os.Stdout.Stat(); fileInfo != nil {
|
||||||
|
if (fileInfo.Mode() & os.ModeCharDevice) != 0 {
|
||||||
|
tty = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var handler slog.Handler
|
||||||
|
if tty {
|
||||||
|
handler = slog.NewTextHandler(
|
||||||
|
os.Stdout,
|
||||||
|
&slog.HandlerOptions{
|
||||||
|
Level: l.level,
|
||||||
|
AddSource: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
handler = slog.NewJSONHandler(
|
||||||
|
os.Stdout,
|
||||||
|
&slog.HandlerOptions{
|
||||||
|
Level: l.level,
|
||||||
|
AddSource: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
l.log = slog.New(handler)
|
||||||
|
|
||||||
|
return l, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EnableDebugLogging sets the log level to debug.
|
||||||
|
func (l *Logger) EnableDebugLogging() {
|
||||||
|
l.level.Set(slog.LevelDebug)
|
||||||
|
l.log.Debug("debug logging enabled", "debug", true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get returns the underlying slog.Logger.
|
||||||
|
func (l *Logger) Get() *slog.Logger {
|
||||||
|
return l.log
|
||||||
|
}
|
||||||
|
|
||||||
|
// Identify logs the application's build-time metadata.
|
||||||
|
func (l *Logger) Identify() {
|
||||||
|
l.log.Info("starting",
|
||||||
|
"appname", l.params.Globals.Appname,
|
||||||
|
"version", l.params.Globals.Version,
|
||||||
|
"buildarch", l.params.Globals.Buildarch,
|
||||||
|
)
|
||||||
|
}
|
||||||
129
backend/internal/middleware/middleware.go
Normal file
129
backend/internal/middleware/middleware.go
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
// Package middleware provides HTTP middleware for logging,
|
||||||
|
// CORS, and other cross-cutting concerns.
|
||||||
|
package middleware
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log/slog"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/config"
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
|
||||||
|
"github.com/go-chi/chi/v5/middleware"
|
||||||
|
"github.com/go-chi/cors"
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
const corsMaxAgeSec = 300
|
||||||
|
|
||||||
|
// Params defines the dependencies for Middleware.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Config *config.Config
|
||||||
|
Globals *globals.Globals
|
||||||
|
Logger *logger.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Middleware holds shared state for middleware factories.
|
||||||
|
type Middleware struct {
|
||||||
|
log *slog.Logger
|
||||||
|
params *Params
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Middleware instance.
|
||||||
|
func New(
|
||||||
|
_ fx.Lifecycle,
|
||||||
|
params Params,
|
||||||
|
) (*Middleware, error) {
|
||||||
|
s := new(Middleware)
|
||||||
|
s.params = ¶ms
|
||||||
|
s.log = params.Logger.Get()
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type loggingResponseWriter struct {
|
||||||
|
http.ResponseWriter
|
||||||
|
|
||||||
|
statusCode int
|
||||||
|
}
|
||||||
|
|
||||||
|
func newLoggingResponseWriter(
|
||||||
|
w http.ResponseWriter,
|
||||||
|
) *loggingResponseWriter {
|
||||||
|
return &loggingResponseWriter{w, http.StatusOK}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (lrw *loggingResponseWriter) WriteHeader(code int) {
|
||||||
|
lrw.statusCode = code
|
||||||
|
lrw.ResponseWriter.WriteHeader(code)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ipFromHostPort(hostPort string) string {
|
||||||
|
host, _, err := net.SplitHostPort(hostPort)
|
||||||
|
if err != nil {
|
||||||
|
return hostPort
|
||||||
|
}
|
||||||
|
|
||||||
|
return host
|
||||||
|
}
|
||||||
|
|
||||||
|
// Logging returns middleware that logs each request with
|
||||||
|
// timing, status code, and client information.
|
||||||
|
func (s *Middleware) Logging() func(http.Handler) http.Handler {
|
||||||
|
return func(next http.Handler) http.Handler {
|
||||||
|
return http.HandlerFunc(
|
||||||
|
func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
start := time.Now().UTC()
|
||||||
|
lrw := newLoggingResponseWriter(w)
|
||||||
|
ctx := r.Context()
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
latency := time.Since(start)
|
||||||
|
s.log.InfoContext(ctx, "request",
|
||||||
|
"request_start", start,
|
||||||
|
"method", r.Method,
|
||||||
|
"url", r.URL.String(),
|
||||||
|
"useragent", r.UserAgent(),
|
||||||
|
"request_id",
|
||||||
|
ctx.Value(
|
||||||
|
middleware.RequestIDKey,
|
||||||
|
),
|
||||||
|
"referer", r.Referer(),
|
||||||
|
"proto", r.Proto,
|
||||||
|
"remote_ip",
|
||||||
|
ipFromHostPort(r.RemoteAddr),
|
||||||
|
"status", lrw.statusCode,
|
||||||
|
"latency_ms",
|
||||||
|
latency.Milliseconds(),
|
||||||
|
)
|
||||||
|
}()
|
||||||
|
|
||||||
|
next.ServeHTTP(lrw, r)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// CORS returns middleware that adds permissive CORS headers.
|
||||||
|
func (s *Middleware) CORS() func(http.Handler) http.Handler {
|
||||||
|
return cors.Handler(cors.Options{
|
||||||
|
AllowedOrigins: []string{"*"},
|
||||||
|
AllowedMethods: []string{
|
||||||
|
"GET", "POST", "PUT", "DELETE", "OPTIONS",
|
||||||
|
},
|
||||||
|
AllowedHeaders: []string{
|
||||||
|
"Accept",
|
||||||
|
"Authorization",
|
||||||
|
"Content-Type",
|
||||||
|
"X-CSRF-Token",
|
||||||
|
},
|
||||||
|
ExposedHeaders: []string{"Link"},
|
||||||
|
AllowCredentials: false,
|
||||||
|
MaxAge: corsMaxAgeSec,
|
||||||
|
})
|
||||||
|
}
|
||||||
199
backend/internal/reportbuf/reportbuf.go
Normal file
199
backend/internal/reportbuf/reportbuf.go
Normal file
@@ -0,0 +1,199 @@
|
|||||||
|
// Package reportbuf accumulates telemetry reports in memory
|
||||||
|
// and periodically flushes them to zstd-compressed JSONL files.
|
||||||
|
package reportbuf
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io/fs"
|
||||||
|
"log/slog"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/config"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
|
||||||
|
"github.com/klauspost/compress/zstd"
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
flushSizeThreshold = 10 << 20 // 10 MiB
|
||||||
|
flushInterval = 1 * time.Minute
|
||||||
|
defaultDataDir = "./data/reports"
|
||||||
|
dirPerms fs.FileMode = 0o750
|
||||||
|
filePerms fs.FileMode = 0o640
|
||||||
|
)
|
||||||
|
|
||||||
|
// Params defines the dependencies for Buffer.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Config *config.Config
|
||||||
|
Logger *logger.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Buffer accumulates JSON lines in memory and flushes them
|
||||||
|
// to zstd-compressed files on disk.
|
||||||
|
type Buffer struct {
|
||||||
|
buf bytes.Buffer
|
||||||
|
dataDir string
|
||||||
|
done chan struct{}
|
||||||
|
log *slog.Logger
|
||||||
|
mu sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Buffer and registers lifecycle hooks to
|
||||||
|
// manage the data directory and flush goroutine.
|
||||||
|
func New(
|
||||||
|
lc fx.Lifecycle,
|
||||||
|
params Params,
|
||||||
|
) (*Buffer, error) {
|
||||||
|
dir := params.Config.DataDir
|
||||||
|
if dir == "" {
|
||||||
|
dir = defaultDataDir
|
||||||
|
}
|
||||||
|
|
||||||
|
b := &Buffer{
|
||||||
|
dataDir: dir,
|
||||||
|
done: make(chan struct{}),
|
||||||
|
log: params.Logger.Get(),
|
||||||
|
}
|
||||||
|
|
||||||
|
lc.Append(fx.Hook{
|
||||||
|
OnStart: func(_ context.Context) error {
|
||||||
|
err := os.MkdirAll(b.dataDir, dirPerms)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("create data dir: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
go b.flushLoop()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
OnStop: func(_ context.Context) error {
|
||||||
|
close(b.done)
|
||||||
|
b.flushLocked()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return b, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Append marshals v as a single JSON line and appends it to
|
||||||
|
// the buffer. If the buffer reaches the size threshold, it is
|
||||||
|
// drained and written to disk asynchronously.
|
||||||
|
func (b *Buffer) Append(v any) error {
|
||||||
|
line, err := json.Marshal(v)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("marshal report: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.mu.Lock()
|
||||||
|
b.buf.Write(line)
|
||||||
|
b.buf.WriteByte('\n')
|
||||||
|
|
||||||
|
if b.buf.Len() >= flushSizeThreshold {
|
||||||
|
data := b.drainBuf()
|
||||||
|
b.mu.Unlock()
|
||||||
|
|
||||||
|
go b.writeFile(data)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
b.mu.Unlock()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// flushLoop runs a ticker that periodically flushes buffered
|
||||||
|
// data to disk until the done channel is closed.
|
||||||
|
func (b *Buffer) flushLoop() {
|
||||||
|
ticker := time.NewTicker(flushInterval)
|
||||||
|
defer ticker.Stop()
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-ticker.C:
|
||||||
|
b.flushLocked()
|
||||||
|
case <-b.done:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// flushLocked acquires the lock, drains the buffer, and
|
||||||
|
// writes the data to a compressed file.
|
||||||
|
func (b *Buffer) flushLocked() {
|
||||||
|
b.mu.Lock()
|
||||||
|
|
||||||
|
if b.buf.Len() == 0 {
|
||||||
|
b.mu.Unlock()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
data := b.drainBuf()
|
||||||
|
b.mu.Unlock()
|
||||||
|
|
||||||
|
b.writeFile(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
// drainBuf copies the buffer contents and resets it.
|
||||||
|
// The caller must hold b.mu.
|
||||||
|
func (b *Buffer) drainBuf() []byte {
|
||||||
|
data := make([]byte, b.buf.Len())
|
||||||
|
copy(data, b.buf.Bytes())
|
||||||
|
b.buf.Reset()
|
||||||
|
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
|
||||||
|
// writeFile creates a timestamped zstd-compressed JSONL file
|
||||||
|
// in the data directory.
|
||||||
|
func (b *Buffer) writeFile(data []byte) {
|
||||||
|
ts := time.Now().UTC().Format("2006-01-02T15-04-05.000Z")
|
||||||
|
name := fmt.Sprintf("reports-%s.jsonl.zst", ts)
|
||||||
|
path := filepath.Join(b.dataDir, name)
|
||||||
|
|
||||||
|
f, err := os.OpenFile( //nolint:gosec // path built from controlled dataDir + timestamp
|
||||||
|
path,
|
||||||
|
os.O_WRONLY|os.O_CREATE|os.O_EXCL,
|
||||||
|
filePerms,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
b.log.Error("create report file", "error", err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() { _ = f.Close() }()
|
||||||
|
|
||||||
|
enc, err := zstd.NewWriter(f)
|
||||||
|
if err != nil {
|
||||||
|
b.log.Error("create zstd encoder", "error", err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_, writeErr := enc.Write(data)
|
||||||
|
if writeErr != nil {
|
||||||
|
b.log.Error("write compressed data", "error", writeErr)
|
||||||
|
|
||||||
|
_ = enc.Close()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
closeErr := enc.Close()
|
||||||
|
if closeErr != nil {
|
||||||
|
b.log.Error("close zstd encoder", "error", closeErr)
|
||||||
|
}
|
||||||
|
}
|
||||||
13
backend/internal/reportbuf/reportbuf_test.go
Normal file
13
backend/internal/reportbuf/reportbuf_test.go
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
package reportbuf_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
_ "sneak.berlin/go/netwatch/internal/reportbuf"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestImport(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
// Compilation check — verifies the package parses
|
||||||
|
// and all imports resolve.
|
||||||
|
}
|
||||||
43
backend/internal/server/http.go
Normal file
43
backend/internal/server/http.go
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
readTimeout = 10 * time.Second
|
||||||
|
writeTimeout = 10 * time.Second
|
||||||
|
maxHeaderBytes = 1 << 20 // 1 MiB
|
||||||
|
)
|
||||||
|
|
||||||
|
func (s *Server) serveUntilShutdown() {
|
||||||
|
listenAddr := fmt.Sprintf(":%d", s.params.Config.Port)
|
||||||
|
|
||||||
|
s.httpServer = &http.Server{
|
||||||
|
Addr: listenAddr,
|
||||||
|
Handler: s,
|
||||||
|
MaxHeaderBytes: maxHeaderBytes,
|
||||||
|
ReadTimeout: readTimeout,
|
||||||
|
WriteTimeout: writeTimeout,
|
||||||
|
}
|
||||||
|
|
||||||
|
s.SetupRoutes()
|
||||||
|
|
||||||
|
s.log.Info("http begin listen",
|
||||||
|
"listenaddr", listenAddr,
|
||||||
|
"version", s.params.Globals.Version,
|
||||||
|
"buildarch", s.params.Globals.Buildarch,
|
||||||
|
)
|
||||||
|
|
||||||
|
err := s.httpServer.ListenAndServe()
|
||||||
|
if err != nil && !errors.Is(err, http.ErrServerClosed) {
|
||||||
|
s.log.Error("listen error", "error", err)
|
||||||
|
|
||||||
|
if s.cancelFunc != nil {
|
||||||
|
s.cancelFunc()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
31
backend/internal/server/routes.go
Normal file
31
backend/internal/server/routes.go
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/go-chi/chi/v5"
|
||||||
|
"github.com/go-chi/chi/v5/middleware"
|
||||||
|
)
|
||||||
|
|
||||||
|
const requestTimeout = 60 * time.Second
|
||||||
|
|
||||||
|
// SetupRoutes configures the chi router with middleware and
|
||||||
|
// all application routes.
|
||||||
|
func (s *Server) SetupRoutes() {
|
||||||
|
s.router = chi.NewRouter()
|
||||||
|
|
||||||
|
s.router.Use(middleware.Recoverer)
|
||||||
|
s.router.Use(middleware.RequestID)
|
||||||
|
s.router.Use(s.mw.Logging())
|
||||||
|
s.router.Use(s.mw.CORS())
|
||||||
|
s.router.Use(middleware.Timeout(requestTimeout))
|
||||||
|
|
||||||
|
s.router.Get(
|
||||||
|
"/.well-known/healthcheck",
|
||||||
|
s.h.HandleHealthCheck(),
|
||||||
|
)
|
||||||
|
|
||||||
|
s.router.Route("/api/v1", func(r chi.Router) {
|
||||||
|
r.Post("/reports", s.h.HandleReport())
|
||||||
|
})
|
||||||
|
}
|
||||||
147
backend/internal/server/server.go
Normal file
147
backend/internal/server/server.go
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
// Package server provides the HTTP server lifecycle,
|
||||||
|
// including startup, routing, signal handling, and graceful
|
||||||
|
// shutdown.
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log/slog"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"os/signal"
|
||||||
|
"syscall"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"sneak.berlin/go/netwatch/internal/config"
|
||||||
|
"sneak.berlin/go/netwatch/internal/globals"
|
||||||
|
"sneak.berlin/go/netwatch/internal/handlers"
|
||||||
|
"sneak.berlin/go/netwatch/internal/logger"
|
||||||
|
"sneak.berlin/go/netwatch/internal/middleware"
|
||||||
|
|
||||||
|
"github.com/go-chi/chi/v5"
|
||||||
|
"go.uber.org/fx"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Params defines the dependencies for Server.
|
||||||
|
type Params struct {
|
||||||
|
fx.In
|
||||||
|
|
||||||
|
Config *config.Config
|
||||||
|
Globals *globals.Globals
|
||||||
|
Handlers *handlers.Handlers
|
||||||
|
Logger *logger.Logger
|
||||||
|
Middleware *middleware.Middleware
|
||||||
|
}
|
||||||
|
|
||||||
|
// Server is the top-level HTTP server orchestrator.
|
||||||
|
type Server struct {
|
||||||
|
cancelFunc context.CancelFunc
|
||||||
|
exitCode int
|
||||||
|
h *handlers.Handlers
|
||||||
|
httpServer *http.Server
|
||||||
|
log *slog.Logger
|
||||||
|
mw *middleware.Middleware
|
||||||
|
params Params
|
||||||
|
router *chi.Mux
|
||||||
|
startupTime time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a Server and registers lifecycle hooks for
|
||||||
|
// starting and stopping it.
|
||||||
|
func New(
|
||||||
|
lc fx.Lifecycle,
|
||||||
|
params Params,
|
||||||
|
) (*Server, error) {
|
||||||
|
s := new(Server)
|
||||||
|
s.params = params
|
||||||
|
s.mw = params.Middleware
|
||||||
|
s.h = params.Handlers
|
||||||
|
s.log = params.Logger.Get()
|
||||||
|
|
||||||
|
lc.Append(fx.Hook{
|
||||||
|
OnStart: func(_ context.Context) error {
|
||||||
|
s.startupTime = time.Now().UTC()
|
||||||
|
|
||||||
|
go func() { //nolint:contextcheck // fx OnStart ctx is startup-only; run() creates its own
|
||||||
|
s.run()
|
||||||
|
}()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
OnStop: func(_ context.Context) error {
|
||||||
|
if s.cancelFunc != nil {
|
||||||
|
s.cancelFunc()
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ServeHTTP delegates to the chi router.
|
||||||
|
func (s *Server) ServeHTTP(
|
||||||
|
w http.ResponseWriter,
|
||||||
|
r *http.Request,
|
||||||
|
) {
|
||||||
|
s.router.ServeHTTP(w, r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) run() {
|
||||||
|
exitCode := s.serve()
|
||||||
|
os.Exit(exitCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) serve() int {
|
||||||
|
var ctx context.Context //nolint:wsl // ctx must be declared before multi-assign
|
||||||
|
|
||||||
|
ctx, s.cancelFunc = context.WithCancel(
|
||||||
|
context.Background(),
|
||||||
|
)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
c := make(chan os.Signal, 1)
|
||||||
|
|
||||||
|
signal.Ignore(syscall.SIGPIPE)
|
||||||
|
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
|
||||||
|
|
||||||
|
sig := <-c
|
||||||
|
s.log.Info("signal received", "signal", sig)
|
||||||
|
|
||||||
|
if s.cancelFunc != nil {
|
||||||
|
s.cancelFunc()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
s.serveUntilShutdown()
|
||||||
|
}()
|
||||||
|
|
||||||
|
<-ctx.Done()
|
||||||
|
s.cleanShutdown()
|
||||||
|
|
||||||
|
return s.exitCode
|
||||||
|
}
|
||||||
|
|
||||||
|
const shutdownTimeout = 5 * time.Second
|
||||||
|
|
||||||
|
func (s *Server) cleanShutdown() {
|
||||||
|
s.exitCode = 0
|
||||||
|
|
||||||
|
ctxShutdown, shutdownCancel := context.WithTimeout(
|
||||||
|
context.Background(),
|
||||||
|
shutdownTimeout,
|
||||||
|
)
|
||||||
|
defer shutdownCancel()
|
||||||
|
|
||||||
|
err := s.httpServer.Shutdown(ctxShutdown)
|
||||||
|
if err != nil {
|
||||||
|
s.log.Error(
|
||||||
|
"server clean shutdown failed",
|
||||||
|
"error", err,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
s.log.Info("server stopped")
|
||||||
|
}
|
||||||
28
nginx.conf
Normal file
28
nginx.conf
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
server {
|
||||||
|
listen 8080;
|
||||||
|
server_name _;
|
||||||
|
|
||||||
|
root /usr/share/nginx/html;
|
||||||
|
index index.html;
|
||||||
|
|
||||||
|
# Trust RFC1918 reverse proxies for X-Forwarded-For
|
||||||
|
set_real_ip_from 10.0.0.0/8;
|
||||||
|
set_real_ip_from 172.16.0.0/12;
|
||||||
|
set_real_ip_from 192.168.0.0/16;
|
||||||
|
real_ip_header X-Forwarded-For;
|
||||||
|
real_ip_recursive on;
|
||||||
|
|
||||||
|
# Access log to stdout (Docker best practice)
|
||||||
|
access_log /dev/stdout combined;
|
||||||
|
error_log /dev/stderr warn;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
try_files $uri $uri/ /index.html;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Cache static assets aggressively
|
||||||
|
location /assets/ {
|
||||||
|
expires 1y;
|
||||||
|
add_header Cache-Control "public, immutable";
|
||||||
|
}
|
||||||
|
}
|
||||||
924
src/main.js
924
src/main.js
File diff suppressed because it is too large
Load Diff
@@ -21,3 +21,54 @@ body {
|
|||||||
rgba(255, 255, 255, 0) 100%
|
rgba(255, 255, 255, 0) 100%
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Mobile-responsive host rows */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.host-row-inner {
|
||||||
|
flex-direction: column !important;
|
||||||
|
align-items: stretch !important;
|
||||||
|
gap: 0.5rem !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.host-row-inner .pin-btn {
|
||||||
|
align-self: flex-end;
|
||||||
|
}
|
||||||
|
|
||||||
|
.host-info-panel {
|
||||||
|
width: 100% !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.host-row .sparkline-container {
|
||||||
|
width: 100% !important;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Summary line: allow wrapping */
|
||||||
|
#summary {
|
||||||
|
display: flex !important;
|
||||||
|
flex-wrap: wrap !important;
|
||||||
|
gap: 0.25rem 0.5rem !important;
|
||||||
|
justify-content: center !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
#summary .text-gray-600.mx-3 {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Header: stack controls below title */
|
||||||
|
.header-top {
|
||||||
|
flex-direction: column !important;
|
||||||
|
align-items: flex-start !important;
|
||||||
|
gap: 1rem !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-controls {
|
||||||
|
align-items: flex-start !important;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#pause-btn {
|
||||||
|
width: 100%;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,8 +1,16 @@
|
|||||||
import { defineConfig } from "vite";
|
import { defineConfig } from "vite";
|
||||||
import tailwindcss from "@tailwindcss/vite";
|
import tailwindcss from "@tailwindcss/vite";
|
||||||
|
import { execSync } from "child_process";
|
||||||
|
|
||||||
|
const commitHash = execSync("git rev-parse --short HEAD").toString().trim();
|
||||||
|
const commitFull = execSync("git rev-parse HEAD").toString().trim();
|
||||||
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
plugins: [tailwindcss()],
|
plugins: [tailwindcss()],
|
||||||
|
define: {
|
||||||
|
__COMMIT_HASH__: JSON.stringify(commitHash),
|
||||||
|
__COMMIT_FULL__: JSON.stringify(commitFull),
|
||||||
|
},
|
||||||
build: {
|
build: {
|
||||||
target: "esnext",
|
target: "esnext",
|
||||||
minify: "esbuild",
|
minify: "esbuild",
|
||||||
|
|||||||
Reference in New Issue
Block a user