Compare commits
33 Commits
feature/ui
...
4ad98d578b
| Author | SHA1 | Date | |
|---|---|---|---|
| 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
|
||||||
|
|||||||
@@ -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;"]
|
||||||
|
|||||||
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`
|
||||||
|
|||||||
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";
|
||||||
|
}
|
||||||
|
}
|
||||||
889
src/main.js
889
src/main.js
File diff suppressed because it is too large
Load Diff
@@ -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