flatten app to repo root and update docker deployment for single-stack runtime
This commit is contained in:
20
.dockerignore
Normal file
20
.dockerignore
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
# Build output and local caches
|
||||||
|
.next
|
||||||
|
.cache
|
||||||
|
|
||||||
|
# Dependencies
|
||||||
|
node_modules
|
||||||
|
|
||||||
|
# Local runtime data and environment
|
||||||
|
.env
|
||||||
|
.env.local
|
||||||
|
.env.*.local
|
||||||
|
data
|
||||||
|
|
||||||
|
# Editor/system files
|
||||||
|
.DS_Store
|
||||||
|
*.log
|
||||||
|
|
||||||
|
# Git
|
||||||
|
.git
|
||||||
|
.gitignore
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
# Optional API override. Leave empty to use same-origin internal API routes.
|
# Optional API override. Leave empty to use same-origin internal API routes.
|
||||||
NEXT_PUBLIC_API_URL=
|
NEXT_PUBLIC_API_URL=
|
||||||
|
APP_PORT=3000
|
||||||
|
|
||||||
# OpenClaw / ZeroClaw (OpenAI-compatible)
|
# OpenClaw / ZeroClaw (OpenAI-compatible)
|
||||||
OPENCLAW_BASE_URL=http://localhost:4000
|
OPENCLAW_BASE_URL=http://localhost:4000
|
||||||
|
|||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -37,4 +37,4 @@ out/
|
|||||||
*.sqlite
|
*.sqlite
|
||||||
|
|
||||||
# Local app runtime state
|
# Local app runtime state
|
||||||
frontend/data/*.json
|
data/*.json
|
||||||
|
|||||||
@@ -3,12 +3,13 @@ FROM node:20-alpine AS base
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
FROM base AS deps
|
FROM base AS deps
|
||||||
COPY package.json ./
|
COPY package.json package-lock.json ./
|
||||||
RUN npm install
|
RUN npm ci
|
||||||
|
|
||||||
FROM base AS builder
|
FROM base AS builder
|
||||||
ARG NEXT_PUBLIC_API_URL=
|
ARG NEXT_PUBLIC_API_URL=
|
||||||
ENV NEXT_PUBLIC_API_URL=${NEXT_PUBLIC_API_URL}
|
ENV NEXT_PUBLIC_API_URL=${NEXT_PUBLIC_API_URL}
|
||||||
|
ENV NEXT_TELEMETRY_DISABLED=1
|
||||||
COPY --from=deps /app/node_modules ./node_modules
|
COPY --from=deps /app/node_modules ./node_modules
|
||||||
COPY . .
|
COPY . .
|
||||||
RUN mkdir -p public && npm run build
|
RUN mkdir -p public && npm run build
|
||||||
@@ -19,18 +20,21 @@ WORKDIR /app
|
|||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
ARG NEXT_PUBLIC_API_URL=
|
ARG NEXT_PUBLIC_API_URL=
|
||||||
ENV NEXT_PUBLIC_API_URL=${NEXT_PUBLIC_API_URL}
|
ENV NEXT_PUBLIC_API_URL=${NEXT_PUBLIC_API_URL}
|
||||||
|
ENV NEXT_TELEMETRY_DISABLED=1
|
||||||
|
|
||||||
|
RUN apk add --no-cache su-exec
|
||||||
RUN addgroup --system --gid 1001 nodejs
|
RUN addgroup --system --gid 1001 nodejs
|
||||||
RUN adduser --system --uid 1001 nextjs
|
RUN adduser --system --uid 1001 nextjs
|
||||||
|
|
||||||
COPY --from=builder /app/public ./public
|
COPY --from=builder /app/public ./public
|
||||||
COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./
|
COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./
|
||||||
COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static
|
COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static
|
||||||
|
COPY docker-entrypoint.sh /usr/local/bin/docker-entrypoint.sh
|
||||||
USER nextjs
|
RUN chmod +x /usr/local/bin/docker-entrypoint.sh
|
||||||
|
|
||||||
EXPOSE 3000
|
EXPOSE 3000
|
||||||
|
|
||||||
ENV PORT=3000
|
ENV PORT=3000
|
||||||
|
|
||||||
|
ENTRYPOINT ["/usr/local/bin/docker-entrypoint.sh"]
|
||||||
CMD ["node", "server.js"]
|
CMD ["node", "server.js"]
|
||||||
14
README.md
14
README.md
@@ -13,7 +13,6 @@ Turbopack-first rebuild of a fiscal.ai-style terminal with OpenClaw integration.
|
|||||||
## Run locally
|
## Run locally
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd frontend
|
|
||||||
npm install
|
npm install
|
||||||
npm run dev
|
npm run dev
|
||||||
```
|
```
|
||||||
@@ -23,14 +22,23 @@ Open [http://localhost:3000](http://localhost:3000).
|
|||||||
## Production build
|
## Production build
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd frontend
|
|
||||||
npm run build
|
npm run build
|
||||||
npm run start
|
npm run start
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Docker deployment
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cp .env.example .env
|
||||||
|
docker compose up --build -d
|
||||||
|
```
|
||||||
|
|
||||||
|
Default app URL: `http://localhost:3000` (override with `APP_PORT` in `.env`).
|
||||||
|
Runtime data persists in the `app_data` volume (`/app/data` in container).
|
||||||
|
|
||||||
## Environment
|
## Environment
|
||||||
|
|
||||||
Use root `.env` or `frontend/.env.local`:
|
Use root `.env` or root `.env.local`:
|
||||||
|
|
||||||
```env
|
```env
|
||||||
# leave blank for same-origin API
|
# leave blank for same-origin API
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
services:
|
services:
|
||||||
app:
|
app:
|
||||||
build:
|
build:
|
||||||
context: ./frontend
|
context: .
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
args:
|
args:
|
||||||
NEXT_PUBLIC_API_URL: ${NEXT_PUBLIC_API_URL:-}
|
NEXT_PUBLIC_API_URL: ${NEXT_PUBLIC_API_URL:-}
|
||||||
@@ -17,5 +17,15 @@ services:
|
|||||||
OPENCLAW_API_KEY: ${OPENCLAW_API_KEY:-}
|
OPENCLAW_API_KEY: ${OPENCLAW_API_KEY:-}
|
||||||
OPENCLAW_MODEL: ${OPENCLAW_MODEL:-zeroclaw}
|
OPENCLAW_MODEL: ${OPENCLAW_MODEL:-zeroclaw}
|
||||||
SEC_USER_AGENT: ${SEC_USER_AGENT:-Fiscal Clone <support@fiscal.local>}
|
SEC_USER_AGENT: ${SEC_USER_AGENT:-Fiscal Clone <support@fiscal.local>}
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD-SHELL", "wget -q --spider http://localhost:3000/api/health || exit 1"]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
ports:
|
ports:
|
||||||
- '3000:3000'
|
- '${APP_PORT:-3000}:3000'
|
||||||
|
volumes:
|
||||||
|
- app_data:/app/data
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
app_data:
|
||||||
|
|||||||
7
docker-entrypoint.sh
Executable file
7
docker-entrypoint.sh
Executable file
@@ -0,0 +1,7 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
set -e
|
||||||
|
|
||||||
|
mkdir -p /app/data
|
||||||
|
chown -R nextjs:nodejs /app/data
|
||||||
|
|
||||||
|
exec su-exec nextjs "$@"
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { mkdir, readFile, writeFile } from 'node:fs/promises';
|
import { mkdir, readFile, rename, writeFile } from 'node:fs/promises';
|
||||||
import path from 'node:path';
|
import path from 'node:path';
|
||||||
import type { Filing, Holding, PortfolioInsight, Task, WatchlistItem } from '@/lib/types';
|
import type { Filing, Holding, PortfolioInsight, Task, WatchlistItem } from '@/lib/types';
|
||||||
|
|
||||||
@@ -74,7 +74,9 @@ async function readStore(): Promise<DataStore> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function writeStore(store: DataStore) {
|
async function writeStore(store: DataStore) {
|
||||||
await writeFile(STORE_PATH, JSON.stringify(store, null, 2), 'utf8');
|
const tempPath = `${STORE_PATH}.tmp`;
|
||||||
|
await writeFile(tempPath, JSON.stringify(store, null, 2), 'utf8');
|
||||||
|
await rename(tempPath, STORE_PATH);
|
||||||
}
|
}
|
||||||
|
|
||||||
function cloneStore(store: DataStore): DataStore {
|
function cloneStore(store: DataStore): DataStore {
|
||||||
0
frontend/next-env.d.ts → next-env.d.ts
vendored
0
frontend/next-env.d.ts → next-env.d.ts
vendored
Reference in New Issue
Block a user