working docker compose
This commit is contained in:
parent
2a612bf0ee
commit
24bb9a5b92
37
docker-compose.yml
Normal file
37
docker-compose.yml
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
version: "3.7"
|
||||||
|
|
||||||
|
services:
|
||||||
|
webapp:
|
||||||
|
build:
|
||||||
|
context: ./webapp
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
ports:
|
||||||
|
- "3000:3000"
|
||||||
|
|
||||||
|
migrate:
|
||||||
|
build:
|
||||||
|
context: ./server/prisma
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
volumes:
|
||||||
|
- sqlite:/database/
|
||||||
|
environment:
|
||||||
|
- DATABASE_URL=file:/database/db.sqlite
|
||||||
|
|
||||||
|
server:
|
||||||
|
build:
|
||||||
|
context: ./server
|
||||||
|
dockerfile: Dockerfile
|
||||||
|
ports:
|
||||||
|
- "8080:8080"
|
||||||
|
volumes:
|
||||||
|
- sqlite:/database/
|
||||||
|
environment:
|
||||||
|
- DATABASE_URL=file:/database/db.sqlite
|
||||||
|
- FRONTEND_URL=http://0.0.0.0:3000
|
||||||
|
depends_on:
|
||||||
|
- migrate
|
||||||
|
|
||||||
|
## By default this config uses default local driver,
|
||||||
|
## For custom volumes replace with volume driver configuration.
|
||||||
|
volumes:
|
||||||
|
sqlite:
|
3
server/.dockerignore
Normal file
3
server/.dockerignore
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
node_modules
|
||||||
|
npm-debug.log
|
||||||
|
build
|
1
server/.gitignore
vendored
1
server/.gitignore
vendored
@ -1 +1,2 @@
|
|||||||
node_modules
|
node_modules
|
||||||
|
build
|
28
server/Dockerfile
Normal file
28
server/Dockerfile
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
FROM node:16-alpine AS BUILD_IMAGE
|
||||||
|
|
||||||
|
# install dependencies
|
||||||
|
WORKDIR /app
|
||||||
|
COPY package.json package-lock.json ./
|
||||||
|
RUN npm ci
|
||||||
|
|
||||||
|
# Copy all local files into the image.
|
||||||
|
COPY . .
|
||||||
|
COPY .env.docker .env
|
||||||
|
|
||||||
|
RUN npx prisma generate
|
||||||
|
|
||||||
|
RUN npm run build
|
||||||
|
|
||||||
|
# remove development dependencies
|
||||||
|
RUN npm prune --production
|
||||||
|
|
||||||
|
FROM node:16-alpine
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
COPY --from=0 /app .
|
||||||
|
# COPY . .
|
||||||
|
|
||||||
|
ENV PORT=8080
|
||||||
|
EXPOSE 8080
|
||||||
|
|
||||||
|
CMD ["node", "./build/server.js"]
|
@ -1,16 +1,99 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
||||||
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
||||||
|
return new (P || (P = Promise))(function (resolve, reject) {
|
||||||
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
||||||
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
||||||
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
||||||
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
||||||
|
});
|
||||||
|
};
|
||||||
var __importDefault = (this && this.__importDefault) || function (mod) {
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
||||||
return (mod && mod.__esModule) ? mod : { "default": mod };
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
||||||
};
|
};
|
||||||
Object.defineProperty(exports, "__esModule", { value: true });
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
require("dotenv/config");
|
||||||
const express_1 = __importDefault(require("express"));
|
const express_1 = __importDefault(require("express"));
|
||||||
|
const client_1 = require("@prisma/client");
|
||||||
|
const util_1 = require("./util");
|
||||||
|
const helmet_1 = __importDefault(require("helmet"));
|
||||||
|
const express_rate_limit_1 = __importDefault(require("express-rate-limit"));
|
||||||
|
// Initialize middleware clients
|
||||||
|
const prisma = new client_1.PrismaClient();
|
||||||
const app = (0, express_1.default)();
|
const app = (0, express_1.default)();
|
||||||
const port = 8080; // default port to listen
|
app.use(express_1.default.json());
|
||||||
// define a route handler for the default home page
|
app.use((0, helmet_1.default)({
|
||||||
app.get("/", (req, res) => {
|
crossOriginResourcePolicy: {
|
||||||
res.send("Hello world!");
|
policy: process.env.ENVIRONMENT == "dev" ? "cross-origin" : "same-origin",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
// Apply rate limiting
|
||||||
|
const postLimiter = (0, express_rate_limit_1.default)({
|
||||||
|
windowMs: parseInt(process.env.POST_LIMIT_WINDOW_SECONDS) * 1000,
|
||||||
|
max: parseInt(process.env.POST_LIMIT),
|
||||||
|
standardHeaders: true,
|
||||||
|
legacyHeaders: false, // Disable the `X-RateLimit-*` headers
|
||||||
});
|
});
|
||||||
// start the Express server
|
// start the Express server
|
||||||
app.listen(port, () => {
|
app.listen(process.env.PORT, () => {
|
||||||
console.log(`server started at http://localhost:${port}!`);
|
console.log(`server started at port ${process.env.PORT}`);
|
||||||
});
|
});
|
||||||
|
// Post new encrypted note
|
||||||
|
app.post("/note/", postLimiter, (req, res) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
try {
|
||||||
|
const note = req.body;
|
||||||
|
const savedNote = yield prisma.encryptedNote.create({
|
||||||
|
data: Object.assign(Object.assign({}, note), { expire_time: (0, util_1.addDays)(new Date(), 30) }),
|
||||||
|
});
|
||||||
|
console.log(`[POST] Saved note <${savedNote.id}> for <${req.ip}>`);
|
||||||
|
res.json({
|
||||||
|
view_url: `${process.env.FRONTEND_URL}/note/${savedNote.id}`,
|
||||||
|
expire_time: savedNote.expire_time,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
console.error(err.stack);
|
||||||
|
res.status(500).send("Something went wrong.");
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
// Get encrypted note
|
||||||
|
app.get("/note/:id", (req, res) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
const note = yield prisma.encryptedNote.findUnique({
|
||||||
|
where: { id: req.params.id },
|
||||||
|
});
|
||||||
|
if (note != null) {
|
||||||
|
res.send(note);
|
||||||
|
console.log(`[GET] Retrieved note <${note.id}> for <${req.ip}>`);
|
||||||
|
}
|
||||||
|
res.status(404).send();
|
||||||
|
}));
|
||||||
|
// Default response for any other request
|
||||||
|
app.use((req, res, next) => {
|
||||||
|
res.status(500).send();
|
||||||
|
});
|
||||||
|
// // Error handling
|
||||||
|
// app.use((err, req, res, next) => {
|
||||||
|
// console.error(err.stack);
|
||||||
|
// res.status(500).send("Something broke!");
|
||||||
|
// });
|
||||||
|
// Clean up expired notes periodically
|
||||||
|
const interval = Math.max(parseInt(process.env.CLEANUP_INTERVAL_SECONDS) || 1, 1) *
|
||||||
|
1000;
|
||||||
|
setInterval(() => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
try {
|
||||||
|
console.log("[Cleanup] Cleaning up expired notes...");
|
||||||
|
const deleted = yield prisma.encryptedNote.deleteMany({
|
||||||
|
where: {
|
||||||
|
expire_time: {
|
||||||
|
lte: new Date(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
console.log(`[Cleanup] Deleted ${deleted.count} expired notes.`);
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
console.error(`[Cleanup] Error cleaning expired notes:`);
|
||||||
|
console.error(err);
|
||||||
|
}
|
||||||
|
}), interval);
|
||||||
|
//# sourceMappingURL=server.js.map
|
31
server/prisma/Dockerfile
Normal file
31
server/prisma/Dockerfile
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
# FROM node:16-alpine AS BUILD_IMAGE
|
||||||
|
|
||||||
|
# # install dependencies
|
||||||
|
# WORKDIR /app
|
||||||
|
# COPY package.json package-lock.json ./
|
||||||
|
# RUN npm ci
|
||||||
|
|
||||||
|
# # Copy all local files into the image.
|
||||||
|
# COPY . .
|
||||||
|
# COPY .env.docker .env
|
||||||
|
|
||||||
|
# RUN npx prisma generate
|
||||||
|
|
||||||
|
# # remove development dependencies
|
||||||
|
# RUN npm prune --production
|
||||||
|
|
||||||
|
# CMD ["npx", "prisma migrate deploy
|
||||||
|
|
||||||
|
FROM node:16-alpine
|
||||||
|
RUN echo $DATABASE_URL
|
||||||
|
|
||||||
|
# Create app directory
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Install prisma for the migration
|
||||||
|
RUN npm install -g prisma --unsafe-perm
|
||||||
|
|
||||||
|
# Copy schema and migration folder
|
||||||
|
ADD ./ ./prisma/
|
||||||
|
|
||||||
|
CMD [ "prisma", "migrate", "deploy"]
|
Binary file not shown.
BIN
server/prisma/dev.db-journal
Normal file
BIN
server/prisma/dev.db-journal
Normal file
Binary file not shown.
14
server/prisma/migrations/20220701200456_devdev/migration.sql
Normal file
14
server/prisma/migrations/20220701200456_devdev/migration.sql
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
-- RedefineTables
|
||||||
|
PRAGMA foreign_keys=OFF;
|
||||||
|
CREATE TABLE "new_EncryptedNote" (
|
||||||
|
"id" TEXT NOT NULL PRIMARY KEY,
|
||||||
|
"insert_time" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"expire_time" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"ciphertext" TEXT NOT NULL,
|
||||||
|
"hmac" TEXT NOT NULL
|
||||||
|
);
|
||||||
|
INSERT INTO "new_EncryptedNote" ("ciphertext", "hmac", "id", "insert_time") SELECT "ciphertext", "hmac", "id", "insert_time" FROM "EncryptedNote";
|
||||||
|
DROP TABLE "EncryptedNote";
|
||||||
|
ALTER TABLE "new_EncryptedNote" RENAME TO "EncryptedNote";
|
||||||
|
PRAGMA foreign_key_check;
|
||||||
|
PRAGMA foreign_keys=ON;
|
@ -29,7 +29,7 @@ const postLimiter = rateLimit({
|
|||||||
|
|
||||||
// start the Express server
|
// start the Express server
|
||||||
app.listen(process.env.PORT, () => {
|
app.listen(process.env.PORT, () => {
|
||||||
console.log(`server started at http://localhost:${process.env.PORT}`);
|
console.log(`server started at port ${process.env.PORT}`);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Post new encrypted note
|
// Post new encrypted note
|
||||||
@ -55,6 +55,7 @@ app.get("/note/:id", async (req, res) => {
|
|||||||
});
|
});
|
||||||
if (note != null) {
|
if (note != null) {
|
||||||
res.send(note);
|
res.send(note);
|
||||||
|
console.log(`[GET] Retrieved note <${note.id}> for <${req.ip}>`);
|
||||||
}
|
}
|
||||||
res.status(404).send();
|
res.status(404).send();
|
||||||
});
|
});
|
||||||
@ -64,6 +65,12 @@ app.use((req, res, next) => {
|
|||||||
res.status(404).send();
|
res.status(404).send();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// // Error handling
|
||||||
|
// app.use((err, req, res, next) => {
|
||||||
|
// console.error(err.stack);
|
||||||
|
// res.status(500).send("Something broke!");
|
||||||
|
// });
|
||||||
|
|
||||||
// Clean up expired notes periodically
|
// Clean up expired notes periodically
|
||||||
const interval =
|
const interval =
|
||||||
Math.max(parseInt(<string>process.env.CLEANUP_INTERVAL_SECONDS) || 1, 1) *
|
Math.max(parseInt(<string>process.env.CLEANUP_INTERVAL_SECONDS) || 1, 1) *
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"sourceMap": true,
|
"sourceMap": true,
|
||||||
|
"outDir": "./build",
|
||||||
"lib": ["esnext"],
|
"lib": ["esnext"],
|
||||||
"target": "es2016" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */,
|
"target": "es2016" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */,
|
||||||
"module": "commonjs" /* Specify what module code is generated. */,
|
"module": "commonjs" /* Specify what module code is generated. */,
|
||||||
|
@ -9,6 +9,9 @@ RUN npm ci
|
|||||||
# Copy all local files into the image.
|
# Copy all local files into the image.
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
|
ENV VITE_BACKEND_URL="http://0.0.0.0:8080"
|
||||||
|
ENV VITE_BRANDING="Noteshare.space"
|
||||||
|
|
||||||
RUN npm run build
|
RUN npm run build
|
||||||
|
|
||||||
# remove development dependencies
|
# remove development dependencies
|
||||||
|
Loading…
Reference in New Issue
Block a user