feat: Add Docker containerization, production serving, and Gitea CI/CD workflow.
Some checks failed
Build and Deploy / build (push) Failing after 2m44s
Some checks failed
Build and Deploy / build (push) Failing after 2m44s
This commit is contained in:
8
.dockerignore
Normal file
8
.dockerignore
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
.git
|
||||||
|
node_modules
|
||||||
|
src/node_modules
|
||||||
|
src/dist
|
||||||
|
src/client/dist
|
||||||
|
.env
|
||||||
|
.DS_Store
|
||||||
|
coverage
|
||||||
40
.gitea/workflows/build.yml
Normal file
40
.gitea/workflows/build.yml
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
name: Build and Deploy
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [main]
|
||||||
|
pull_request:
|
||||||
|
branches: [main]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Log in to Gitea Container Registry
|
||||||
|
if: github.event_name == 'push'
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: ${{ vars.PACKAGES_REGISTRY }}
|
||||||
|
username: ${{ secrets.USERNAME }}
|
||||||
|
password: ${{ secrets.TOKEN }}
|
||||||
|
|
||||||
|
- name: Extract metadata
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: ${{ vars.PACKAGES_REGISTRY }}/${{ gitea.repository }}
|
||||||
|
|
||||||
|
- name: Build and push Docker image
|
||||||
|
uses: docker/build-push-action@v5
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
push: ${{ github.event_name == 'push' }}
|
||||||
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
27
Dockerfile
Normal file
27
Dockerfile
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# Use Node.js LTS (Alpine for smaller size)
|
||||||
|
FROM node:20-alpine
|
||||||
|
|
||||||
|
# Set working directory
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Copy package definition first to cache dependencies
|
||||||
|
COPY src/package.json src/package-lock.json ./
|
||||||
|
|
||||||
|
# Install dependencies
|
||||||
|
# Using npm install instead of ci to ensure updated package.json is respected
|
||||||
|
RUN npm install
|
||||||
|
|
||||||
|
# Copy the rest of the source code
|
||||||
|
COPY src/ ./
|
||||||
|
|
||||||
|
# Build the frontend (Production build)
|
||||||
|
RUN npm run build
|
||||||
|
|
||||||
|
# Remove development dependencies to keep image small
|
||||||
|
RUN npm prune --production
|
||||||
|
|
||||||
|
# Expose the application port
|
||||||
|
EXPOSE 3000
|
||||||
|
|
||||||
|
# Start the application
|
||||||
|
CMD ["npm", "start"]
|
||||||
@@ -8,7 +8,7 @@
|
|||||||
"server": "tsx watch server/index.ts",
|
"server": "tsx watch server/index.ts",
|
||||||
"client": "vite",
|
"client": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
"start": "node server/dist/index.js"
|
"start": "NODE_ENV=production tsx server/index.ts"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"express": "^4.21.2",
|
"express": "^4.21.2",
|
||||||
@@ -16,7 +16,8 @@
|
|||||||
"react": "^19.0.0",
|
"react": "^19.0.0",
|
||||||
"react-dom": "^19.0.0",
|
"react-dom": "^19.0.0",
|
||||||
"socket.io": "^4.8.1",
|
"socket.io": "^4.8.1",
|
||||||
"socket.io-client": "^4.8.1"
|
"socket.io-client": "^4.8.1",
|
||||||
|
"tsx": "^4.19.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/express": "^4.17.21",
|
"@types/express": "^4.17.21",
|
||||||
@@ -28,8 +29,7 @@
|
|||||||
"concurrently": "^9.1.0",
|
"concurrently": "^9.1.0",
|
||||||
"postcss": "^8.4.49",
|
"postcss": "^8.4.49",
|
||||||
"tailwindcss": "^3.4.16",
|
"tailwindcss": "^3.4.16",
|
||||||
"tsx": "^4.19.2",
|
|
||||||
"typescript": "^5.7.2",
|
"typescript": "^5.7.2",
|
||||||
"vite": "^6.0.3"
|
"vite": "^6.0.3"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -36,6 +36,13 @@ app.get('/api/health', (_req: Request, res: Response) => {
|
|||||||
res.json({ status: 'ok', message: 'Server is running' });
|
res.json({ status: 'ok', message: 'Server is running' });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Serve Frontend in Production
|
||||||
|
if (process.env.NODE_ENV === 'production') {
|
||||||
|
const distPath = path.resolve(process.cwd(), 'dist');
|
||||||
|
app.use(express.static(distPath));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
app.post('/api/cards/cache', async (req: Request, res: Response) => {
|
app.post('/api/cards/cache', async (req: Request, res: Response) => {
|
||||||
try {
|
try {
|
||||||
const { cards } = req.body;
|
const { cards } = req.body;
|
||||||
@@ -201,6 +208,19 @@ io.on('connection', (socket) => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
// Handle Client-Side Routing (Catch-All) - Must be last
|
||||||
|
if (process.env.NODE_ENV === 'production') {
|
||||||
|
app.get('*', (_req: Request, res: Response) => {
|
||||||
|
// Check if request is for API
|
||||||
|
if (_req.path.startsWith('/api') || _req.path.startsWith('/socket.io')) {
|
||||||
|
return res.status(404).json({ error: 'Not found' });
|
||||||
|
}
|
||||||
|
const distPath = path.resolve(process.cwd(), 'dist');
|
||||||
|
res.sendFile(path.join(distPath, 'index.html'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
import os from 'os';
|
import os from 'os';
|
||||||
|
|
||||||
httpServer.listen(Number(PORT), '0.0.0.0', () => {
|
httpServer.listen(Number(PORT), '0.0.0.0', () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user