pnpm@10.7, node@23, argon->@node-rs/argon2
This commit is contained in:
parent
93e0cdf577
commit
78dcad0857
3
.vscode/settings.json
vendored
3
.vscode/settings.json
vendored
@ -9,8 +9,7 @@
|
|||||||
"[typescript]": {
|
"[typescript]": {
|
||||||
"editor.defaultFormatter": "vscode.typescript-language-features",
|
"editor.defaultFormatter": "vscode.typescript-language-features",
|
||||||
"editor.codeActionsOnSave": {
|
"editor.codeActionsOnSave": {
|
||||||
"source.organizeImports": true,
|
"source.organizeImports": "explicit"
|
||||||
// "source.fixAll": true
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"javascript.preferences.quoteStyle": "single",
|
"javascript.preferences.quoteStyle": "single",
|
||||||
|
10
Dockerfile
10
Dockerfile
@ -1,10 +1,12 @@
|
|||||||
# Typescript Build
|
# Typescript Build
|
||||||
FROM registry.odit.services/hub/library/node:21.1.0-alpine3.18 AS build
|
FROM registry.odit.services/hub/library/node:23.10.0-alpine3.21 AS build
|
||||||
ARG NPM_REGISTRY_URL=https://registry.npmjs.org
|
ARG NPM_REGISTRY_URL=https://registry.npmjs.org
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
COPY package.json ./
|
COPY package.json ./
|
||||||
RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8
|
COPY pnpm-workspace.yaml ./
|
||||||
|
COPY pnpm-lock.yaml ./
|
||||||
|
RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@10.7
|
||||||
RUN mkdir /pnpm && pnpm config set store-dir /pnpm && pnpm i
|
RUN mkdir /pnpm && pnpm config set store-dir /pnpm && pnpm i
|
||||||
|
|
||||||
COPY tsconfig.json ormconfig.js ./
|
COPY tsconfig.json ormconfig.js ./
|
||||||
@ -14,9 +16,11 @@ RUN pnpm run build \
|
|||||||
&& pnpm i --production --prefer-offline
|
&& pnpm i --production --prefer-offline
|
||||||
|
|
||||||
# final image
|
# final image
|
||||||
FROM registry.odit.services/hub/library/node:21.1.0-alpine3.18 AS final
|
FROM registry.odit.services/hub/library/node:23.10.0-alpine3.21 AS final
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=build /app/package.json /app/package.json
|
COPY --from=build /app/package.json /app/package.json
|
||||||
|
COPY --from=build /app/pnpm-lock.yaml /app/pnpm-lock.yaml
|
||||||
|
COPY --from=build /app/pnpm-workspace.yaml /app/pnpm-workspace.yaml
|
||||||
COPY --from=build /app/ormconfig.js /app/ormconfig.js
|
COPY --from=build /app/ormconfig.js /app/ormconfig.js
|
||||||
COPY --from=build /app/dist /app/dist
|
COPY --from=build /app/dist /app/dist
|
||||||
COPY --from=build /app/node_modules /app/node_modules
|
COPY --from=build /app/node_modules /app/node_modules
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
version: "3"
|
|
||||||
services:
|
services:
|
||||||
backend_server:
|
backend_server:
|
||||||
build: .
|
build: .
|
||||||
@ -14,7 +13,7 @@ services:
|
|||||||
DB_NAME: ./db.sqlite
|
DB_NAME: ./db.sqlite
|
||||||
NODE_ENV: production
|
NODE_ENV: production
|
||||||
POSTALCODE_COUNTRYCODE: DE
|
POSTALCODE_COUNTRYCODE: DE
|
||||||
SEED_TEST_DATA: "false"
|
SEED_TEST_DATA: "true"
|
||||||
MAILER_URL: https://dev.lauf-fuer-kaya.de/mailer
|
MAILER_URL: https://dev.lauf-fuer-kaya.de/mailer
|
||||||
MAILER_KEY: asdasd
|
MAILER_KEY: asdasd
|
||||||
# APP_PORT: 4010
|
# APP_PORT: 4010
|
||||||
|
@ -3,9 +3,6 @@
|
|||||||
"version": "1.3.0",
|
"version": "1.3.0",
|
||||||
"main": "src/app.ts",
|
"main": "src/app.ts",
|
||||||
"repository": "https://git.odit.services/lfk/backend",
|
"repository": "https://git.odit.services/lfk/backend",
|
||||||
"engines": {
|
|
||||||
"pnpm": "8"
|
|
||||||
},
|
|
||||||
"author": {
|
"author": {
|
||||||
"name": "ODIT.Services",
|
"name": "ODIT.Services",
|
||||||
"email": "info@odit.services",
|
"email": "info@odit.services",
|
||||||
@ -25,8 +22,8 @@
|
|||||||
],
|
],
|
||||||
"license": "CC-BY-NC-SA-4.0",
|
"license": "CC-BY-NC-SA-4.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@node-rs/argon2": "^2.0.2",
|
||||||
"@odit/class-validator-jsonschema": "2.1.1",
|
"@odit/class-validator-jsonschema": "2.1.1",
|
||||||
"argon2": "0.31.2",
|
|
||||||
"axios": "0.21.1",
|
"axios": "0.21.1",
|
||||||
"body-parser": "1.19.0",
|
"body-parser": "1.19.0",
|
||||||
"check-password-strength": "2.0.2",
|
"check-password-strength": "2.0.2",
|
||||||
@ -46,7 +43,7 @@
|
|||||||
"reflect-metadata": "0.1.13",
|
"reflect-metadata": "0.1.13",
|
||||||
"routing-controllers": "0.9.0-alpha.6",
|
"routing-controllers": "0.9.0-alpha.6",
|
||||||
"routing-controllers-openapi": "2.2.0",
|
"routing-controllers-openapi": "2.2.0",
|
||||||
"sqlite3": "5.1.6",
|
"sqlite3": "5.1.7",
|
||||||
"typeorm": "0.2.30",
|
"typeorm": "0.2.30",
|
||||||
"typeorm-routing-controllers-extensions": "0.2.0",
|
"typeorm-routing-controllers-extensions": "0.2.0",
|
||||||
"typeorm-seeding": "1.6.1",
|
"typeorm-seeding": "1.6.1",
|
||||||
|
10304
pnpm-lock.yaml
generated
10304
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
2
pnpm-workspace.yaml
Normal file
2
pnpm-workspace.yaml
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
onlyBuiltDependencies:
|
||||||
|
- sqlite3
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { verify } from '@node-rs/argon2';
|
||||||
import { Request, Response } from 'express';
|
import { Request, Response } from 'express';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { ScanStation } from '../models/entities/ScanStation';
|
import { ScanStation } from '../models/entities/ScanStation';
|
||||||
@ -58,7 +58,7 @@ const ScanAuth = async (req: Request, res: Response, next: () => void) => {
|
|||||||
if (station.enabled == false) {
|
if (station.enabled == false) {
|
||||||
res.status(401).send({ http_code: 401, short: "station_disabled", message: "Station is disabled." });
|
res.status(401).send({ http_code: 401, short: "station_disabled", message: "Station is disabled." });
|
||||||
}
|
}
|
||||||
if (!(await argon2.verify(station.key, provided_token))) {
|
if (!(await verify(station.key, provided_token))) {
|
||||||
res.status(401).send({ http_code: 401, short: "invalid_token", message: "Api token non-existent or invalid syntax." });
|
res.status(401).send({ http_code: 401, short: "invalid_token", message: "Api token non-existent or invalid syntax." });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { verify } from '@node-rs/argon2';
|
||||||
import { Request, Response } from 'express';
|
import { Request, Response } from 'express';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { StatsClient } from '../models/entities/StatsClient';
|
import { StatsClient } from '../models/entities/StatsClient';
|
||||||
@ -55,7 +55,7 @@ const StatsAuth = async (req: Request, res: Response, next: () => void) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (!(await argon2.verify(client.key, provided_token))) {
|
if (!(await verify(client.key, provided_token))) {
|
||||||
res.status(401).send("Api token invalid.");
|
res.status(401).send("Api token invalid.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { hash } from '@node-rs/argon2';
|
||||||
import { IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
import { IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
||||||
import * as jsonwebtoken from 'jsonwebtoken';
|
import * as jsonwebtoken from 'jsonwebtoken';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
@ -49,7 +49,7 @@ export class ResetPassword {
|
|||||||
if (found_user.refreshTokenCount !== decoded["refreshTokenCount"]) { throw new RefreshTokenCountInvalidError(); }
|
if (found_user.refreshTokenCount !== decoded["refreshTokenCount"]) { throw new RefreshTokenCountInvalidError(); }
|
||||||
|
|
||||||
found_user.refreshTokenCount = found_user.refreshTokenCount + 1;
|
found_user.refreshTokenCount = found_user.refreshTokenCount + 1;
|
||||||
found_user.password = await argon2.hash(this.password + found_user.uuid);
|
found_user.password = await hash(this.password + found_user.uuid);
|
||||||
await getConnectionManager().get().getRepository(User).save(found_user);
|
await getConnectionManager().get().getRepository(User).save(found_user);
|
||||||
|
|
||||||
return "password reset successfull";
|
return "password reset successfull";
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { verify } from '@node-rs/argon2';
|
||||||
import { IsEmail, IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
import { IsEmail, IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { InvalidCredentialsError, PasswordNeededError, UserDisabledError, UserNotFoundError } from '../../../errors/AuthError';
|
import { InvalidCredentialsError, PasswordNeededError, UserDisabledError, UserNotFoundError } from '../../../errors/AuthError';
|
||||||
@ -56,7 +56,7 @@ export class CreateAuth {
|
|||||||
throw new UserNotFoundError();
|
throw new UserNotFoundError();
|
||||||
}
|
}
|
||||||
if (found_user.enabled == false) { throw new UserDisabledError(); }
|
if (found_user.enabled == false) { throw new UserDisabledError(); }
|
||||||
if (!(await argon2.verify(found_user.password, this.password + found_user.uuid))) {
|
if (!(await verify(found_user.password, this.password + found_user.uuid))) {
|
||||||
throw new InvalidCredentialsError();
|
throw new InvalidCredentialsError();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { hash } from '@node-rs/argon2';
|
||||||
import { IsBoolean, IsInt, IsOptional, IsPositive, IsString } from 'class-validator';
|
import { IsBoolean, IsInt, IsOptional, IsPositive, IsString } from 'class-validator';
|
||||||
import crypto from 'crypto';
|
import crypto from 'crypto';
|
||||||
import { getConnection } from 'typeorm';
|
import { getConnection } from 'typeorm';
|
||||||
@ -44,7 +44,7 @@ export class CreateScanStation {
|
|||||||
|
|
||||||
let newUUID = uuid.v4().toUpperCase();
|
let newUUID = uuid.v4().toUpperCase();
|
||||||
newStation.prefix = crypto.createHash("sha3-512").update(newUUID).digest('hex').substring(0, 7).toUpperCase();
|
newStation.prefix = crypto.createHash("sha3-512").update(newUUID).digest('hex').substring(0, 7).toUpperCase();
|
||||||
newStation.key = await argon2.hash(newStation.prefix + "." + newUUID);
|
newStation.key = await hash(newStation.prefix + "." + newUUID);
|
||||||
newStation.cleartextkey = newStation.prefix + "." + newUUID;
|
newStation.cleartextkey = newStation.prefix + "." + newUUID;
|
||||||
|
|
||||||
return newStation;
|
return newStation;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { hash } from '@node-rs/argon2';
|
||||||
import { IsOptional, IsString } from 'class-validator';
|
import { IsOptional, IsString } from 'class-validator';
|
||||||
import crypto from 'crypto';
|
import crypto from 'crypto';
|
||||||
import * as uuid from 'uuid';
|
import * as uuid from 'uuid';
|
||||||
@ -25,7 +25,7 @@ export class CreateStatsClient {
|
|||||||
|
|
||||||
let newUUID = uuid.v4().toUpperCase();
|
let newUUID = uuid.v4().toUpperCase();
|
||||||
newClient.prefix = crypto.createHash("sha3-512").update(newUUID).digest('hex').substring(0, 7).toUpperCase();
|
newClient.prefix = crypto.createHash("sha3-512").update(newUUID).digest('hex').substring(0, 7).toUpperCase();
|
||||||
newClient.key = await argon2.hash(newClient.prefix + "." + newUUID);
|
newClient.key = await hash(newClient.prefix + "." + newUUID);
|
||||||
newClient.cleartextkey = newClient.prefix + "." + newUUID;
|
newClient.cleartextkey = newClient.prefix + "." + newUUID;
|
||||||
|
|
||||||
return newClient;
|
return newClient;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { hash } from "@node-rs/argon2";
|
||||||
import { passwordStrength } from "check-password-strength";
|
import { passwordStrength } from "check-password-strength";
|
||||||
import { IsBoolean, IsEmail, IsNotEmpty, IsOptional, IsPhoneNumber, IsString, IsUrl } from 'class-validator';
|
import { IsBoolean, IsEmail, IsNotEmpty, IsOptional, IsPhoneNumber, IsString, IsUrl } from 'class-validator';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
@ -110,7 +110,7 @@ export class CreateUser {
|
|||||||
newUser.lastname = this.lastname
|
newUser.lastname = this.lastname
|
||||||
newUser.uuid = uuid.v4()
|
newUser.uuid = uuid.v4()
|
||||||
newUser.phone = this.phone
|
newUser.phone = this.phone
|
||||||
newUser.password = await argon2.hash(this.password + newUser.uuid);
|
newUser.password = await hash(this.password + newUser.uuid);
|
||||||
newUser.groups = await this.getGroups();
|
newUser.groups = await this.getGroups();
|
||||||
newUser.enabled = this.enabled;
|
newUser.enabled = this.enabled;
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { hash } from '@node-rs/argon2';
|
||||||
import { passwordStrength } from "check-password-strength";
|
import { passwordStrength } from "check-password-strength";
|
||||||
import { IsBoolean, IsEmail, IsInt, IsNotEmpty, IsOptional, IsPhoneNumber, IsString, IsUrl } from 'class-validator';
|
import { IsBoolean, IsEmail, IsInt, IsNotEmpty, IsOptional, IsPhoneNumber, IsString, IsUrl } from 'class-validator';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
@ -111,7 +111,7 @@ export class UpdateUser {
|
|||||||
if (!password_strength.contains.includes("lowercase")) { throw new PasswordMustContainLowercaseLetterError(); }
|
if (!password_strength.contains.includes("lowercase")) { throw new PasswordMustContainLowercaseLetterError(); }
|
||||||
if (!password_strength.contains.includes("number")) { throw new PasswordMustContainNumberError(); }
|
if (!password_strength.contains.includes("number")) { throw new PasswordMustContainNumberError(); }
|
||||||
if (!(password_strength.length > 9)) { throw new PasswordTooShortError(); }
|
if (!(password_strength.length > 9)) { throw new PasswordTooShortError(); }
|
||||||
user.password = await argon2.hash(this.password + user.uuid);
|
user.password = await hash(this.password + user.uuid);
|
||||||
user.refreshTokenCount = user.refreshTokenCount + 1;
|
user.refreshTokenCount = user.refreshTokenCount + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as argon2 from "argon2";
|
import { hash } from '@node-rs/argon2';
|
||||||
import { Connection } from 'typeorm';
|
import { Connection } from 'typeorm';
|
||||||
import { Factory, Seeder } from 'typeorm-seeding';
|
import { Factory, Seeder } from 'typeorm-seeding';
|
||||||
import * as uuid from 'uuid';
|
import * as uuid from 'uuid';
|
||||||
@ -33,7 +33,7 @@ export default class SeedUsers implements Seeder {
|
|||||||
initialUser.lastname = "demo";
|
initialUser.lastname = "demo";
|
||||||
initialUser.username = "demo";
|
initialUser.username = "demo";
|
||||||
initialUser.uuid = uuid.v4();
|
initialUser.uuid = uuid.v4();
|
||||||
initialUser.password = await argon2.hash("demo" + initialUser.uuid);
|
initialUser.password = await hash("demo" + initialUser.uuid);
|
||||||
initialUser.email = "demo@dev.lauf-fuer-kaya.de"
|
initialUser.email = "demo@dev.lauf-fuer-kaya.de"
|
||||||
initialUser.groups = [group];
|
initialUser.groups = [group];
|
||||||
return await connection.getRepository(User).save(initialUser);
|
return await connection.getRepository(User).save(initialUser);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user