Compare commits
158 Commits
5a27689e80
...
1bf6d3d564
Author | SHA1 | Date | |
---|---|---|---|
1bf6d3d564 | |||
7d5f3b092f | |||
0ef6d9cc48 | |||
48bef8db60 | |||
1d0d79f3da | |||
d20d738218 | |||
a03f1a438d | |||
75332983c2 | |||
a85d52437b | |||
a88c0389c1 | |||
43a4f1118d | |||
de91d491e5 | |||
2199cb0aef | |||
ee76f1c0e8 | |||
75b6489f8d | |||
389f6347c3 | |||
37afc10e44 | |||
82ced34750 | |||
5de81ad093 | |||
c1d784e29c | |||
4ca85a1f22 | |||
7a4238f1f7 | |||
fbe2b358bd | |||
532b5a56a5 | |||
18ede29ea5 | |||
ec4d75128b | |||
b2bd6173a5 | |||
cc68948a20 | |||
24de82f6df | |||
cf583a22fa | |||
b55d210aff | |||
adec2bcc5b | |||
3850bd9681 | |||
14b6651f96 | |||
4a21c1fb5c | |||
ca142376b3 | |||
314606addd | |||
a0a08f7724 | |||
cea5993049 | |||
3e940c2db5 | |||
631310f158 | |||
c3e3c6bed1 | |||
8f48d2593b | |||
23758e7a91 | |||
c7fd0593fb | |||
b19f18ada1 | |||
d742ccd581 | |||
d670b814a4 | |||
1a9c860188 | |||
f25ae9ba4f | |||
744faba7ee | |||
cdfd0e0d64 | |||
e25fc795fe | |||
2240a45a91 | |||
595a9213c1 | |||
428e2c38ce | |||
1d54fb085b | |||
6403e386ab | |||
65a8449ea3 | |||
b21dd6f0c0 | |||
445e96dcdf | |||
6237e62a03 | |||
b9e91502cd | |||
9dc336f0bb | |||
6a7e8ccc37 | |||
882065470a | |||
ff3a5b4545 | |||
d4293c164d | |||
145a08b1b4 | |||
dc485c02ea | |||
ebb0c5faca | |||
d89fcb84a2 | |||
388fc6ba6a | |||
bb4ea485fd | |||
5dc9edfe40 | |||
eb9473e230 | |||
476afc6a99 | |||
ed53627bbe | |||
efecffb72d | |||
3aae8f85c4 | |||
cc5a30980a | |||
c90f9f1dd4 | |||
15ed9f58d5 | |||
9db4344153 | |||
03b7e346ab | |||
0d8fbf1eca | |||
71228fbf33 | |||
97494aeaf7 | |||
4801e010b4 | |||
1b59d58c60 | |||
cad30c7f63 | |||
a8ec0142b0 | |||
30952aa14f | |||
2e4a4f1661 | |||
b9fd2379f4 | |||
1b1f8f2b09 | |||
39b932a81c | |||
ec69f6caf3 | |||
ad908a3555 | |||
3e6c7b6302 | |||
d0c5323cb6 | |||
fcb3e35b29 | |||
4705b5a0b4 | |||
0c6f3d1f12 | |||
ff178f9d77 | |||
e59630b17e | |||
20ec6e0cd6 | |||
e10a3947ba | |||
8d00487359 | |||
f304b86cb6 | |||
421ddc50ed | |||
c3aa88c212 | |||
10dbd233a0 | |||
c321da613a | |||
ff84209683 | |||
df3c231fd2 | |||
ac2da0af63 | |||
40fb081332 | |||
30928180e6 | |||
6aa1e0d573 | |||
aca3eaaeea | |||
615b54ec4f | |||
c07d40ae93 | |||
db5da3d3c2 | |||
0e003d2dc4 | |||
a1c3751164 | |||
359e955926 | |||
c391201570 | |||
e3980096e2 | |||
a7e27c6f6c | |||
bcb266e29b | |||
95f40a9c28 | |||
8bcaf710ad | |||
b8aebc14e8 | |||
5ccdfe1540 | |||
a1e3289a88 | |||
47e4f6cd7e | |||
36fbccb286 | |||
7429407843 | |||
10640f40aa | |||
1e625b0775 | |||
6cfaec8397 | |||
0f419625d2 | |||
a83a23a647 | |||
553a35bb8e | |||
ef3fcee2a9 | |||
61b2baaee7 | |||
31e7d074dc | |||
1fbddf5ef8 | |||
7a79f35b58 | |||
79e418f918 | |||
abb13045e6 | |||
d543dfb201 | |||
5eabc1fe18 | |||
473033aa50 | |||
effa79032b | |||
57f6775140 | |||
09decd5600 |
111
.drone.yml
Normal file
111
.drone.yml
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
---
|
||||||
|
kind: pipeline
|
||||||
|
name: tests:node_latest
|
||||||
|
clone:
|
||||||
|
disable: true
|
||||||
|
steps:
|
||||||
|
- name: checkout pr
|
||||||
|
image: alpine/git
|
||||||
|
commands:
|
||||||
|
- git clone $DRONE_REMOTE_URL .
|
||||||
|
- git checkout $DRONE_SOURCE_BRANCH
|
||||||
|
- mv .env.ci .env
|
||||||
|
- name: run tests
|
||||||
|
image: node:alpine
|
||||||
|
commands:
|
||||||
|
- yarn
|
||||||
|
- yarn test:ci
|
||||||
|
trigger:
|
||||||
|
event:
|
||||||
|
- pull_request
|
||||||
|
|
||||||
|
---
|
||||||
|
kind: pipeline
|
||||||
|
type: docker
|
||||||
|
name: build:dev
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: build dev
|
||||||
|
image: plugins/docker
|
||||||
|
depends_on: [clone]
|
||||||
|
settings:
|
||||||
|
username:
|
||||||
|
from_secret: DOCKER_REGISTRY_USER
|
||||||
|
password:
|
||||||
|
from_secret: DOCKER_REGISTRY_PASSWORD
|
||||||
|
repo: registry.odit.services/lfk/backend
|
||||||
|
tags:
|
||||||
|
- dev
|
||||||
|
registry: registry.odit.services
|
||||||
|
when:
|
||||||
|
branch:
|
||||||
|
- dev
|
||||||
|
event:
|
||||||
|
- push
|
||||||
|
|
||||||
|
trigger:
|
||||||
|
branch:
|
||||||
|
- dev
|
||||||
|
event:
|
||||||
|
- push
|
||||||
|
|
||||||
|
---
|
||||||
|
kind: pipeline
|
||||||
|
type: docker
|
||||||
|
name: build:latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: build latest
|
||||||
|
image: plugins/docker
|
||||||
|
depends_on: [clone]
|
||||||
|
settings:
|
||||||
|
username:
|
||||||
|
from_secret: DOCKER_REGISTRY_USER
|
||||||
|
password:
|
||||||
|
from_secret: DOCKER_REGISTRY_PASSWORD
|
||||||
|
repo: registry.odit.services/lfk/backend
|
||||||
|
tags:
|
||||||
|
- latest
|
||||||
|
registry: registry.odit.services
|
||||||
|
|
||||||
|
trigger:
|
||||||
|
branch:
|
||||||
|
- main
|
||||||
|
event:
|
||||||
|
- push
|
||||||
|
|
||||||
|
---
|
||||||
|
kind: pipeline
|
||||||
|
type: docker
|
||||||
|
name: build:tags
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: build $DRONE_TAG
|
||||||
|
image: plugins/docker
|
||||||
|
depends_on: [clone]
|
||||||
|
settings:
|
||||||
|
username:
|
||||||
|
from_secret: DOCKER_REGISTRY_USER
|
||||||
|
password:
|
||||||
|
from_secret: DOCKER_REGISTRY_PASSWORD
|
||||||
|
repo: registry.odit.services/lfk/backend
|
||||||
|
tags:
|
||||||
|
- $DRONE_TAG
|
||||||
|
registry: registry.odit.services
|
||||||
|
- name: trigger lib build
|
||||||
|
depends_on: [clone]
|
||||||
|
image: plugins/downstream
|
||||||
|
settings:
|
||||||
|
server: https://ci.odit.services/
|
||||||
|
token:
|
||||||
|
from_secret: BOT_DRONE_KEY
|
||||||
|
fork: false
|
||||||
|
repositories:
|
||||||
|
- lfk/lib
|
||||||
|
params:
|
||||||
|
- SOURCE_TAG: $DRONE_TAG
|
||||||
|
trigger:
|
||||||
|
branch:
|
||||||
|
- main
|
||||||
|
event:
|
||||||
|
- tag
|
8
.env.ci
Normal file
8
.env.ci
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
APP_PORT=4010
|
||||||
|
DB_TYPE=sqlite
|
||||||
|
DB_HOST=unused
|
||||||
|
DB_PORT=unused
|
||||||
|
DB_USER=unused
|
||||||
|
DB_PASSWORD=bla
|
||||||
|
DB_NAME=./test.sqlite
|
||||||
|
NODE_ENV=dev
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -133,3 +133,4 @@ build
|
|||||||
*.sqlite
|
*.sqlite
|
||||||
*.sqlite-jurnal
|
*.sqlite-jurnal
|
||||||
docs
|
docs
|
||||||
|
lib
|
20
Dockerfile
20
Dockerfile
@ -1,6 +1,16 @@
|
|||||||
FROM node:alpine
|
# Typescript Build
|
||||||
|
FROM node:14.15.1-alpine3.12
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY ./package.json ./
|
COPY package.json ./
|
||||||
RUN yarn
|
RUN npm i -g pnpm
|
||||||
COPY ./ ./
|
RUN pnpm i
|
||||||
ENTRYPOINT [ "yarn","dev" ]
|
COPY tsconfig.json ormconfig.js ./
|
||||||
|
COPY src ./src
|
||||||
|
RUN pnpm run build
|
||||||
|
# final image
|
||||||
|
FROM node:14.15.1-alpine3.12
|
||||||
|
COPY package.json ormconfig.js ./
|
||||||
|
RUN npm i -g pnpm
|
||||||
|
RUN pnpm i --prod
|
||||||
|
COPY --from=0 /app/dist dist
|
||||||
|
ENTRYPOINT ["node", "dist/app.js"]
|
@ -6,18 +6,26 @@ services:
|
|||||||
- 4010:4010
|
- 4010:4010
|
||||||
environment:
|
environment:
|
||||||
APP_PORT: 4010
|
APP_PORT: 4010
|
||||||
DB_TYPE: postgres
|
DB_TYPE: sqlite
|
||||||
DB_HOST: backend_db
|
DB_HOST: bla
|
||||||
DB_PORT: 5432
|
DB_PORT: bla
|
||||||
DB_USER: lfk
|
DB_USER: bla
|
||||||
DB_PASSWORD: changeme
|
DB_PASSWORD: bla
|
||||||
DB_NAME: lfk
|
DB_NAME: dev.sqlite
|
||||||
NODE_ENV: production
|
NODE_ENV: production
|
||||||
backend_db:
|
# APP_PORT: 4010
|
||||||
image: postgres:11-alpine
|
# DB_TYPE: postgres
|
||||||
environment:
|
# DB_HOST: backend_db
|
||||||
POSTGRES_DB: lfk
|
# DB_PORT: 5432
|
||||||
POSTGRES_PASSWORD: changeme
|
# DB_USER: lfk
|
||||||
POSTGRES_USER: lfk
|
# DB_PASSWORD: changeme
|
||||||
ports:
|
# DB_NAME: lfk
|
||||||
- 5432:5432
|
# NODE_ENV: production
|
||||||
|
# backend_db:
|
||||||
|
# image: postgres:11-alpine
|
||||||
|
# environment:
|
||||||
|
# POSTGRES_DB: lfk
|
||||||
|
# POSTGRES_PASSWORD: changeme
|
||||||
|
# POSTGRES_USER: lfk
|
||||||
|
# ports:
|
||||||
|
# - 5432:5432
|
||||||
|
16
ormconfig.js
Normal file
16
ormconfig.js
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
const dotenv = require('dotenv');
|
||||||
|
dotenv.config();
|
||||||
|
//
|
||||||
|
const SOURCE_PATH = process.env.NODE_ENV === 'production' ? 'dist' : 'src';
|
||||||
|
module.exports = {
|
||||||
|
type: process.env.DB_TYPE,
|
||||||
|
host: process.env.DB_HOST,
|
||||||
|
port: process.env.DB_PORT,
|
||||||
|
username: process.env.DB_USER,
|
||||||
|
password: process.env.DB_PASSWORD,
|
||||||
|
database: process.env.DB_NAME,
|
||||||
|
// entities: ["src/**/entities/*.ts"],
|
||||||
|
entities: [ `${SOURCE_PATH}/**/entities/*{.ts,.js}` ],
|
||||||
|
seeds: [ `${SOURCE_PATH}/**/seeds/*{.ts,.js}` ]
|
||||||
|
// seeds: ['src/seeds/*.ts'],
|
||||||
|
};
|
12
ormconfig.ts
12
ormconfig.ts
@ -1,12 +0,0 @@
|
|||||||
import { config } from 'dotenv';
|
|
||||||
config();
|
|
||||||
|
|
||||||
export default {
|
|
||||||
type: process.env.DB_TYPE,
|
|
||||||
host: process.env.DB_HOST,
|
|
||||||
port: process.env.DB_PORT,
|
|
||||||
username: process.env.DB_USER,
|
|
||||||
password: process.env.DB_PASSWORD,
|
|
||||||
database: process.env.DB_NAME,
|
|
||||||
entities: ["src/models/entities/*.ts"]
|
|
||||||
};
|
|
24
package.json
24
package.json
@ -28,37 +28,39 @@
|
|||||||
"class-validator": "^0.12.2",
|
"class-validator": "^0.12.2",
|
||||||
"class-validator-jsonschema": "^2.0.3",
|
"class-validator-jsonschema": "^2.0.3",
|
||||||
"consola": "^2.15.0",
|
"consola": "^2.15.0",
|
||||||
|
"cookie": "^0.4.1",
|
||||||
|
"cookie-parser": "^1.4.5",
|
||||||
"cors": "^2.8.5",
|
"cors": "^2.8.5",
|
||||||
|
"csvtojson": "^2.0.10",
|
||||||
"dotenv": "^8.2.0",
|
"dotenv": "^8.2.0",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"helmet": "^4.2.0",
|
|
||||||
"jsonwebtoken": "^8.5.1",
|
"jsonwebtoken": "^8.5.1",
|
||||||
"multer": "^1.4.2",
|
|
||||||
"mysql": "^2.18.1",
|
"mysql": "^2.18.1",
|
||||||
"pg": "^8.5.1",
|
"pg": "^8.5.1",
|
||||||
"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.1.0",
|
"routing-controllers-openapi": "^2.1.0",
|
||||||
|
"sqlite3": "^5.0.0",
|
||||||
"swagger-ui-express": "^4.1.5",
|
"swagger-ui-express": "^4.1.5",
|
||||||
"typeorm": "^0.2.29",
|
"typeorm": "^0.2.29",
|
||||||
"typeorm-routing-controllers-extensions": "^0.2.0",
|
"typeorm-routing-controllers-extensions": "^0.2.0",
|
||||||
|
"typeorm-seeding": "^1.6.1",
|
||||||
"uuid": "^8.3.1"
|
"uuid": "^8.3.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/cors": "^2.8.8",
|
"@types/cors": "^2.8.8",
|
||||||
"@types/dotenv": "^8.2.0",
|
"@types/csvtojson": "^1.1.5",
|
||||||
"@types/express": "^4.17.9",
|
"@types/express": "^4.17.9",
|
||||||
"@types/jest": "^26.0.16",
|
"@types/jest": "^26.0.16",
|
||||||
"@types/jsonwebtoken": "^8.5.0",
|
"@types/jsonwebtoken": "^8.5.0",
|
||||||
"@types/multer": "^1.4.4",
|
|
||||||
"@types/node": "^14.14.9",
|
"@types/node": "^14.14.9",
|
||||||
"@types/swagger-ui-express": "^4.1.2",
|
"@types/swagger-ui-express": "^4.1.2",
|
||||||
"@types/uuid": "^8.3.0",
|
"@types/uuid": "^8.3.0",
|
||||||
"axios": "^0.21.0",
|
"axios": "^0.21.0",
|
||||||
"dotenv-safe": "^8.2.0",
|
|
||||||
"jest": "^26.6.3",
|
"jest": "^26.6.3",
|
||||||
"nodemon": "^2.0.6",
|
"nodemon": "^2.0.6",
|
||||||
"sqlite3": "^5.0.0",
|
"rimraf": "^2.7.1",
|
||||||
|
"start-server-and-test": "^1.11.6",
|
||||||
"ts-jest": "^26.4.4",
|
"ts-jest": "^26.4.4",
|
||||||
"ts-node": "^9.0.0",
|
"ts-node": "^9.0.0",
|
||||||
"typedoc": "^0.19.2",
|
"typedoc": "^0.19.2",
|
||||||
@ -69,13 +71,15 @@
|
|||||||
"build": "tsc",
|
"build": "tsc",
|
||||||
"docs": "typedoc --out docs src",
|
"docs": "typedoc --out docs src",
|
||||||
"test": "jest",
|
"test": "jest",
|
||||||
"test:watch": "jest --watchAll"
|
"test:watch": "jest --watchAll",
|
||||||
|
"test:ci": "start-server-and-test dev http://localhost:4010/api/openapi.json test",
|
||||||
|
"seed": "ts-node ./node_modules/typeorm/cli.js schema:sync && ts-node ./node_modules/typeorm-seeding/dist/cli.js seed",
|
||||||
|
"openapi:export": "ts-node src/openapi_export.ts"
|
||||||
},
|
},
|
||||||
"nodemonConfig": {
|
"nodemonConfig": {
|
||||||
"ignore": [
|
"ignore": [
|
||||||
"src/tests/*",
|
"src/tests/*",
|
||||||
"docs/*"
|
"docs/*"
|
||||||
],
|
]
|
||||||
"delay": "2500"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,13 +6,14 @@ import { config, e as errors } from './config';
|
|||||||
import loaders from "./loaders/index";
|
import loaders from "./loaders/index";
|
||||||
import { ErrorHandler } from './middlewares/ErrorHandler';
|
import { ErrorHandler } from './middlewares/ErrorHandler';
|
||||||
|
|
||||||
|
const CONTROLLERS_FILE_EXTENSION = process.env.NODE_ENV === 'production' ? 'js' : 'ts';
|
||||||
const app = createExpressServer({
|
const app = createExpressServer({
|
||||||
authorizationChecker: authchecker,
|
authorizationChecker: authchecker,
|
||||||
middlewares: [ErrorHandler],
|
middlewares: [ErrorHandler],
|
||||||
development: config.development,
|
development: config.development,
|
||||||
cors: true,
|
cors: true,
|
||||||
routePrefix: "/api",
|
routePrefix: "/api",
|
||||||
controllers: [__dirname + "/controllers/*.ts"],
|
controllers: [`${__dirname}/controllers/*.${CONTROLLERS_FILE_EXTENSION}`],
|
||||||
});
|
});
|
||||||
|
|
||||||
async function main() {
|
async function main() {
|
||||||
|
@ -1,51 +1,72 @@
|
|||||||
|
import cookie from "cookie";
|
||||||
import * as jwt from "jsonwebtoken";
|
import * as jwt from "jsonwebtoken";
|
||||||
import { Action } from "routing-controllers";
|
import { Action } from "routing-controllers";
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { config } from './config';
|
import { config } from './config';
|
||||||
import { IllegalJWTError, NoPermissionError, UserNonexistantOrRefreshtokenInvalidError } from './errors/AuthError';
|
import { IllegalJWTError, NoPermissionError, UserNonexistantOrRefreshtokenInvalidError } from './errors/AuthError';
|
||||||
|
import { JwtCreator, JwtUser } from './jwtcreator';
|
||||||
import { User } from './models/entities/User';
|
import { User } from './models/entities/User';
|
||||||
// -----------
|
|
||||||
const authchecker = async (action: Action, permissions: string | string[]) => {
|
/**
|
||||||
let required_permissions = undefined
|
* Handels authorisation verification via jwt's for all api endpoints using the @Authorized decorator.
|
||||||
|
* @param action Routing-Controllers action object that provides request and response objects among other stuff.
|
||||||
|
* @param permissions The permissions that the endpoint using @Authorized requires.
|
||||||
|
*/
|
||||||
|
const authchecker = async (action: Action, permissions: string[] | string) => {
|
||||||
|
let required_permissions = undefined;
|
||||||
if (typeof permissions === "string") {
|
if (typeof permissions === "string") {
|
||||||
required_permissions = [permissions]
|
required_permissions = [permissions]
|
||||||
} else {
|
} else {
|
||||||
required_permissions = permissions
|
required_permissions = permissions
|
||||||
}
|
}
|
||||||
// const token = action.request.headers["authorization"];
|
|
||||||
const provided_token = action.request.query["auth"];
|
|
||||||
let jwtPayload = undefined
|
let jwtPayload = undefined
|
||||||
try {
|
try {
|
||||||
|
let provided_token = "" + action.request.headers["authorization"].replace("Bearer ", "");
|
||||||
jwtPayload = <any>jwt.verify(provided_token, config.jwt_secret);
|
jwtPayload = <any>jwt.verify(provided_token, config.jwt_secret);
|
||||||
|
jwtPayload = jwtPayload["userdetails"];
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new IllegalJWTError()
|
jwtPayload = await refresh(action);
|
||||||
}
|
}
|
||||||
const count = await getConnectionManager().get().getRepository(User).count({ id: jwtPayload["userdetails"]["id"], refreshTokenCount: jwtPayload["userdetails"]["refreshTokenCount"] })
|
|
||||||
if (count !== 1) {
|
const user = await getConnectionManager().get().getRepository(User).findOne({ id: jwtPayload["id"], refreshTokenCount: jwtPayload["refreshTokenCount"] }, { relations: ['permissions'] })
|
||||||
throw new UserNonexistantOrRefreshtokenInvalidError()
|
if (!user) { throw new UserNonexistantOrRefreshtokenInvalidError() }
|
||||||
|
if (!jwtPayload["permissions"]) { throw new NoPermissionError(); }
|
||||||
|
|
||||||
|
action.response.local = {}
|
||||||
|
action.response.local.jwtPayload = jwtPayload;
|
||||||
|
for (let required_permission of required_permissions) {
|
||||||
|
if (!(jwtPayload["permissions"].includes(required_permission))) { return false; }
|
||||||
}
|
}
|
||||||
if (jwtPayload.permissions) {
|
return true;
|
||||||
action.response.local = {}
|
}
|
||||||
action.response.local.jwtPayload = jwtPayload.permissions
|
|
||||||
required_permissions.forEach(r => {
|
/**
|
||||||
const permission_key = r.split(":")[0]
|
* Handels soft-refreshing of access-tokens.
|
||||||
const actual_accesslevel_for_permission = jwtPayload.permissions[permission_key]
|
* @param action Routing-Controllers action object that provides request and response objects among other stuff.
|
||||||
const permission_access_level = r.split(":")[1]
|
*/
|
||||||
if (actual_accesslevel_for_permission.includes(permission_access_level)) {
|
const refresh = async (action: Action) => {
|
||||||
return true;
|
let refresh_token = undefined;
|
||||||
} else {
|
|
||||||
throw new NoPermissionError()
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
throw new NoPermissionError()
|
|
||||||
}
|
|
||||||
//
|
|
||||||
try {
|
try {
|
||||||
jwt.verify(provided_token, config.jwt_secret);
|
refresh_token = cookie.parse(action.request.headers["cookie"])["lfk_backend__refresh_token"];
|
||||||
return true
|
|
||||||
} catch (error) {
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
catch {
|
||||||
|
throw new IllegalJWTError();
|
||||||
|
}
|
||||||
|
|
||||||
|
let jwtPayload = undefined;
|
||||||
|
try {
|
||||||
|
jwtPayload = <any>jwt.verify(refresh_token, config.jwt_secret);
|
||||||
|
} catch (error) {
|
||||||
|
throw new IllegalJWTError();
|
||||||
|
}
|
||||||
|
|
||||||
|
const user = await getConnectionManager().get().getRepository(User).findOne({ id: jwtPayload["id"], refreshTokenCount: jwtPayload["refreshTokenCount"] }, { relations: ['permissions', 'groups', 'groups.permissions'] })
|
||||||
|
if (!user) { throw new UserNonexistantOrRefreshtokenInvalidError() }
|
||||||
|
|
||||||
|
let newAccess = JwtCreator.createAccess(user);
|
||||||
|
action.response.header("authorization", "Bearer " + newAccess);
|
||||||
|
|
||||||
|
return await new JwtUser(user);
|
||||||
}
|
}
|
||||||
export default authchecker
|
export default authchecker
|
@ -1,4 +1,4 @@
|
|||||||
import { Body, JsonController, Post } from 'routing-controllers';
|
import { Body, CookieParam, JsonController, Post, Res } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { IllegalJWTError, InvalidCredentialsError, JwtNotProvidedError, PasswordNeededError, RefreshTokenCountInvalidError, UsernameOrEmailNeededError } from '../errors/AuthError';
|
import { IllegalJWTError, InvalidCredentialsError, JwtNotProvidedError, PasswordNeededError, RefreshTokenCountInvalidError, UsernameOrEmailNeededError } from '../errors/AuthError';
|
||||||
import { UserNotFoundError } from '../errors/UserErrors';
|
import { UserNotFoundError } from '../errors/UserErrors';
|
||||||
@ -21,14 +21,16 @@ export class AuthController {
|
|||||||
@ResponseSchema(PasswordNeededError)
|
@ResponseSchema(PasswordNeededError)
|
||||||
@ResponseSchema(InvalidCredentialsError)
|
@ResponseSchema(InvalidCredentialsError)
|
||||||
@OpenAPI({ description: 'Create a new access token object' })
|
@OpenAPI({ description: 'Create a new access token object' })
|
||||||
async login(@Body({ validate: true }) createAuth: CreateAuth) {
|
async login(@Body({ validate: true }) createAuth: CreateAuth, @Res() response: any) {
|
||||||
let auth;
|
let auth;
|
||||||
try {
|
try {
|
||||||
auth = await createAuth.toAuth();
|
auth = await createAuth.toAuth();
|
||||||
|
response.cookie('lfk_backend__refresh_token', auth.refresh_token, { expires: new Date(auth.refresh_token_expires_at * 1000), httpOnly: true });
|
||||||
|
response.cookie('lfk_backend__refresh_token_expires_at', auth.refresh_token_expires_at, { expires: new Date(auth.refresh_token_expires_at * 1000), httpOnly: true });
|
||||||
|
return response.send(auth)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
return auth
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post("/logout")
|
@Post("/logout")
|
||||||
@ -39,14 +41,20 @@ export class AuthController {
|
|||||||
@ResponseSchema(PasswordNeededError)
|
@ResponseSchema(PasswordNeededError)
|
||||||
@ResponseSchema(InvalidCredentialsError)
|
@ResponseSchema(InvalidCredentialsError)
|
||||||
@OpenAPI({ description: 'Create a new access token object' })
|
@OpenAPI({ description: 'Create a new access token object' })
|
||||||
async logout(@Body({ validate: true }) handleLogout: HandleLogout) {
|
async logout(@Body({ validate: true }) handleLogout: HandleLogout, @CookieParam("lfk_backend__refresh_token") refresh_token: string, @Res() response: any) {
|
||||||
|
if (refresh_token && refresh_token.length != 0 && handleLogout.token == undefined) {
|
||||||
|
handleLogout.token = refresh_token;
|
||||||
|
}
|
||||||
|
|
||||||
let logout;
|
let logout;
|
||||||
try {
|
try {
|
||||||
logout = await handleLogout.logout()
|
logout = await handleLogout.logout()
|
||||||
|
await response.cookie('lfk_backend__refresh_token', "expired", { expires: new Date(Date.now()), httpOnly: true });
|
||||||
|
response.cookie('lfk_backend__refresh_token_expires_at', "expired", { expires: new Date(Date.now()), httpOnly: true });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return error;
|
throw error;
|
||||||
}
|
}
|
||||||
return logout
|
return response.send(logout)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post("/refresh")
|
@Post("/refresh")
|
||||||
@ -56,13 +64,18 @@ export class AuthController {
|
|||||||
@ResponseSchema(UserNotFoundError)
|
@ResponseSchema(UserNotFoundError)
|
||||||
@ResponseSchema(RefreshTokenCountInvalidError)
|
@ResponseSchema(RefreshTokenCountInvalidError)
|
||||||
@OpenAPI({ description: 'refresh a access token' })
|
@OpenAPI({ description: 'refresh a access token' })
|
||||||
async refresh(@Body({ validate: true }) refreshAuth: RefreshAuth) {
|
async refresh(@Body({ validate: true }) refreshAuth: RefreshAuth, @CookieParam("lfk_backend__refresh_token") refresh_token: string, @Res() response: any) {
|
||||||
|
if (refresh_token && refresh_token.length != 0 && refreshAuth.token == undefined) {
|
||||||
|
refreshAuth.token = refresh_token;
|
||||||
|
}
|
||||||
let auth;
|
let auth;
|
||||||
try {
|
try {
|
||||||
auth = await refreshAuth.toAuth();
|
auth = await refreshAuth.toAuth();
|
||||||
|
response.cookie('lfk_backend__refresh_token', auth.refresh_token, { expires: new Date(auth.refresh_token_expires_at * 1000), httpOnly: true });
|
||||||
|
response.cookie('lfk_backend__refresh_token_expires_at', auth.refresh_token_expires_at, { expires: new Date(auth.refresh_token_expires_at * 1000), httpOnly: true });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return error;
|
throw error;
|
||||||
}
|
}
|
||||||
return auth
|
return response.send(auth)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
102
src/controllers/ImportController.ts
Normal file
102
src/controllers/ImportController.ts
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
import csv from 'csvtojson';
|
||||||
|
import { Authorized, Body, ContentType, Controller, Param, Post, QueryParam, Req, UseBefore } from 'routing-controllers';
|
||||||
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
|
import { RunnerGroupNeededError } from '../errors/RunnerErrors';
|
||||||
|
import { RunnerGroupNotFoundError } from '../errors/RunnerGroupErrors';
|
||||||
|
import RawBodyMiddleware from '../middlewares/RawBody';
|
||||||
|
import { ImportRunner } from '../models/actions/ImportRunner';
|
||||||
|
import { ResponseRunner } from '../models/responses/ResponseRunner';
|
||||||
|
import { RunnerController } from './RunnerController';
|
||||||
|
|
||||||
|
@Controller()
|
||||||
|
@Authorized(["RUNNER:IMPORT", "TEAM:IMPORT"])
|
||||||
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
|
export class ImportController {
|
||||||
|
private runnerController: RunnerController;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the repository of this controller's model/entity.
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
this.runnerController = new RunnerController();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Post('/runners/import')
|
||||||
|
@ContentType("application/json")
|
||||||
|
@ResponseSchema(ResponseRunner, { isArray: true, statusCode: 200 })
|
||||||
|
@ResponseSchema(RunnerGroupNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(RunnerGroupNeededError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Create new runners from json and insert them (or their teams) into the provided group" })
|
||||||
|
async postJSON(@Body({ validate: true, type: ImportRunner }) importRunners: ImportRunner[], @QueryParam("group") groupID: number) {
|
||||||
|
if (!groupID) { throw new RunnerGroupNeededError(); }
|
||||||
|
let responseRunners: ResponseRunner[] = new Array<ResponseRunner>();
|
||||||
|
for await (let runner of importRunners) {
|
||||||
|
responseRunners.push(await this.runnerController.post(await runner.toCreateRunner(groupID)));
|
||||||
|
}
|
||||||
|
return responseRunners;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Post('/organisations/:id/import')
|
||||||
|
@ContentType("application/json")
|
||||||
|
@ResponseSchema(ResponseRunner, { isArray: true, statusCode: 200 })
|
||||||
|
@ResponseSchema(RunnerGroupNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(RunnerGroupNeededError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Create new runners from json and insert them (or their teams) into the provided org" })
|
||||||
|
async postOrgsJSON(@Body({ validate: true, type: ImportRunner }) importRunners: ImportRunner[], @Param('id') id: number) {
|
||||||
|
return await this.postJSON(importRunners, id)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Post('/teams/:id/import')
|
||||||
|
@ContentType("application/json")
|
||||||
|
@ResponseSchema(ResponseRunner, { isArray: true, statusCode: 200 })
|
||||||
|
@ResponseSchema(RunnerGroupNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(RunnerGroupNeededError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Create new runners from json and insert them into the provided team" })
|
||||||
|
async postTeamsJSON(@Body({ validate: true, type: ImportRunner }) importRunners: ImportRunner[], @Param('id') id: number) {
|
||||||
|
return await this.postJSON(importRunners, id)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Post('/runners/import/csv')
|
||||||
|
@ContentType("application/json")
|
||||||
|
@UseBefore(RawBodyMiddleware)
|
||||||
|
@ResponseSchema(ResponseRunner, { isArray: true, statusCode: 200 })
|
||||||
|
@ResponseSchema(RunnerGroupNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(RunnerGroupNeededError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Create new runners from csv and insert them (or their teams) into the provided group" })
|
||||||
|
async postCSV(@Req() request: any, @QueryParam("group") groupID: number) {
|
||||||
|
let csvParse = await csv({ delimiter: [",", ";"], trim: true }).fromString(request.rawBody.toString());
|
||||||
|
let importRunners: ImportRunner[] = new Array<ImportRunner>();
|
||||||
|
for await (let runner of csvParse) {
|
||||||
|
let newImportRunner = new ImportRunner();
|
||||||
|
newImportRunner.firstname = runner.firstname;
|
||||||
|
newImportRunner.middlename = runner.middlename;
|
||||||
|
newImportRunner.lastname = runner.lastname;
|
||||||
|
if (runner.class === undefined) { newImportRunner.team = runner.team; }
|
||||||
|
else { newImportRunner.class = runner.class; }
|
||||||
|
importRunners.push(newImportRunner);
|
||||||
|
}
|
||||||
|
return await this.postJSON(importRunners, groupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Post('/organisations/:id/import/csv')
|
||||||
|
@ContentType("application/json")
|
||||||
|
@UseBefore(RawBodyMiddleware)
|
||||||
|
@ResponseSchema(ResponseRunner, { isArray: true, statusCode: 200 })
|
||||||
|
@ResponseSchema(RunnerGroupNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(RunnerGroupNeededError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Create new runners from csv and insert them (or their teams) into the provided org" })
|
||||||
|
async postOrgsCSV(@Req() request: any, @Param("id") id: number) {
|
||||||
|
return await this.postCSV(request, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Post('/teams/:id/import/csv')
|
||||||
|
@ContentType("application/json")
|
||||||
|
@UseBefore(RawBodyMiddleware)
|
||||||
|
@ResponseSchema(ResponseRunner, { isArray: true, statusCode: 200 })
|
||||||
|
@ResponseSchema(RunnerGroupNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(RunnerGroupNeededError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Create new runners from csv and insert them into the provided team" })
|
||||||
|
async postTeamsCSV(@Req() request: any, @Param("id") id: number) {
|
||||||
|
return await this.postCSV(request, id);
|
||||||
|
}
|
||||||
|
}
|
118
src/controllers/PermissionController.ts
Normal file
118
src/controllers/PermissionController.ts
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
||||||
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
|
import { PermissionIdsNotMatchingError, PermissionNeedsPrincipalError, PermissionNotFoundError } from '../errors/PermissionErrors';
|
||||||
|
import { PrincipalNotFoundError } from '../errors/PrincipalErrors';
|
||||||
|
import { CreatePermission } from '../models/actions/CreatePermission';
|
||||||
|
import { UpdatePermission } from '../models/actions/UpdatePermission';
|
||||||
|
import { Permission } from '../models/entities/Permission';
|
||||||
|
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
||||||
|
import { ResponsePermission } from '../models/responses/ResponsePermission';
|
||||||
|
import { ResponsePrincipal } from '../models/responses/ResponsePrincipal';
|
||||||
|
|
||||||
|
|
||||||
|
@JsonController('/permissions')
|
||||||
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
|
export class PermissionController {
|
||||||
|
private permissionRepository: Repository<Permission>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the repository of this controller's model/entity.
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
this.permissionRepository = getConnectionManager().get().getRepository(Permission);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Get()
|
||||||
|
@Authorized("PERMISSION:GET")
|
||||||
|
@ResponseSchema(ResponsePermission, { isArray: true })
|
||||||
|
@OpenAPI({ description: 'Lists all permissions.' })
|
||||||
|
async getAll() {
|
||||||
|
let responsePermissions: ResponsePermission[] = new Array<ResponsePermission>();
|
||||||
|
const permissions = await this.permissionRepository.find({ relations: ['principal'] });
|
||||||
|
permissions.forEach(permission => {
|
||||||
|
responsePermissions.push(new ResponsePermission(permission));
|
||||||
|
});
|
||||||
|
return responsePermissions;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Get('/:id')
|
||||||
|
@Authorized("PERMISSION:GET")
|
||||||
|
@ResponseSchema(ResponsePermission)
|
||||||
|
@ResponseSchema(PermissionNotFoundError, { statusCode: 404 })
|
||||||
|
@OnUndefined(PermissionNotFoundError)
|
||||||
|
@OpenAPI({ description: 'Returns a permissions of a specified id (if it exists)' })
|
||||||
|
async getOne(@Param('id') id: number) {
|
||||||
|
let permission = await this.permissionRepository.findOne({ id: id }, { relations: ['principal'] });
|
||||||
|
if (!permission) { throw new PermissionNotFoundError(); }
|
||||||
|
return new ResponsePermission(permission);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Post()
|
||||||
|
@Authorized("PERMISSION:CREATE")
|
||||||
|
@ResponseSchema(ResponsePermission)
|
||||||
|
@ResponseSchema(PrincipalNotFoundError, { statusCode: 404 })
|
||||||
|
@OpenAPI({ description: 'Create a new runnerTeam object (id will be generated automagicly).' })
|
||||||
|
async post(@Body({ validate: true }) createPermission: CreatePermission) {
|
||||||
|
let permission;
|
||||||
|
try {
|
||||||
|
permission = await createPermission.toPermission();
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
let existingPermission = await this.permissionRepository.findOne({ target: permission.target, action: permission.action, principal: permission.principal }, { relations: ['principal'] });
|
||||||
|
if (existingPermission) { return new ResponsePermission(existingPermission); }
|
||||||
|
|
||||||
|
permission = await this.permissionRepository.save(permission);
|
||||||
|
permission = await this.permissionRepository.findOne(permission, { relations: ['principal'] });
|
||||||
|
|
||||||
|
return new ResponsePermission(permission);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Put('/:id')
|
||||||
|
@Authorized("PERMISSION:UPDATE")
|
||||||
|
@ResponseSchema(ResponsePrincipal)
|
||||||
|
@ResponseSchema(PermissionNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(PrincipalNotFoundError, { statusCode: 404 })
|
||||||
|
@ResponseSchema(PermissionIdsNotMatchingError, { statusCode: 406 })
|
||||||
|
@ResponseSchema(PermissionNeedsPrincipalError, { statusCode: 406 })
|
||||||
|
@OpenAPI({ description: "Update a permission object (id can't be changed)." })
|
||||||
|
async put(@Param('id') id: number, @Body({ validate: true }) permission: UpdatePermission) {
|
||||||
|
let oldPermission = await this.permissionRepository.findOne({ id: id }, { relations: ['principal'] });
|
||||||
|
|
||||||
|
if (!oldPermission) {
|
||||||
|
throw new PermissionNotFoundError();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (oldPermission.id != permission.id) {
|
||||||
|
throw new PermissionIdsNotMatchingError();
|
||||||
|
}
|
||||||
|
let existingPermission = await this.permissionRepository.findOne({ target: permission.target, action: permission.action, principal: permission.principal }, { relations: ['principal'] });
|
||||||
|
if (existingPermission) {
|
||||||
|
await this.remove(permission.id, true);
|
||||||
|
return new ResponsePermission(existingPermission);
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.permissionRepository.save(await permission.updatePermission(oldPermission));
|
||||||
|
|
||||||
|
return new ResponsePermission(await this.permissionRepository.findOne({ id: permission.id }, { relations: ['principal'] }));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Delete('/:id')
|
||||||
|
@Authorized("PERMISSION:DELETE")
|
||||||
|
@ResponseSchema(ResponsePermission)
|
||||||
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
|
@OnUndefined(204)
|
||||||
|
@OpenAPI({ description: 'Delete a specified permission (if it exists).' })
|
||||||
|
async remove(@Param("id") id: number, @QueryParam("force") force: boolean) {
|
||||||
|
let permission = await this.permissionRepository.findOne({ id: id }, { relations: ['principal'] });
|
||||||
|
if (!permission) { return null; }
|
||||||
|
|
||||||
|
const responsePermission = new ResponsePermission(permission);
|
||||||
|
await this.permissionRepository.delete(permission);
|
||||||
|
return responsePermission;
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
import { RunnerGroupNeededError, RunnerIdsNotMatchingError, RunnerNotFoundError } from '../errors/RunnerErrors';
|
import { RunnerGroupNeededError, RunnerIdsNotMatchingError, RunnerNotFoundError } from '../errors/RunnerErrors';
|
||||||
@ -10,7 +10,7 @@ import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
|||||||
import { ResponseRunner } from '../models/responses/ResponseRunner';
|
import { ResponseRunner } from '../models/responses/ResponseRunner';
|
||||||
|
|
||||||
@JsonController('/runners')
|
@JsonController('/runners')
|
||||||
//@Authorized('RUNNERS:read')
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
export class RunnerController {
|
export class RunnerController {
|
||||||
private runnerRepository: Repository<Runner>;
|
private runnerRepository: Repository<Runner>;
|
||||||
|
|
||||||
@ -22,6 +22,7 @@ export class RunnerController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
|
@Authorized("RUNNER:GET")
|
||||||
@ResponseSchema(ResponseRunner, { isArray: true })
|
@ResponseSchema(ResponseRunner, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all runners.' })
|
@OpenAPI({ description: 'Lists all runners.' })
|
||||||
async getAll() {
|
async getAll() {
|
||||||
@ -34,6 +35,7 @@ export class RunnerController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
|
@Authorized("RUNNER:GET")
|
||||||
@ResponseSchema(ResponseRunner)
|
@ResponseSchema(ResponseRunner)
|
||||||
@ResponseSchema(RunnerNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(RunnerNotFoundError)
|
@OnUndefined(RunnerNotFoundError)
|
||||||
@ -45,6 +47,7 @@ export class RunnerController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Post()
|
@Post()
|
||||||
|
@Authorized("RUNNER:CREATE")
|
||||||
@ResponseSchema(ResponseRunner)
|
@ResponseSchema(ResponseRunner)
|
||||||
@ResponseSchema(RunnerGroupNeededError)
|
@ResponseSchema(RunnerGroupNeededError)
|
||||||
@ResponseSchema(RunnerGroupNotFoundError)
|
@ResponseSchema(RunnerGroupNotFoundError)
|
||||||
@ -62,6 +65,7 @@ export class RunnerController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Put('/:id')
|
@Put('/:id')
|
||||||
|
@Authorized("RUNNER:UPDATE")
|
||||||
@ResponseSchema(ResponseRunner)
|
@ResponseSchema(ResponseRunner)
|
||||||
@ResponseSchema(RunnerNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerNotFoundError, { statusCode: 404 })
|
||||||
@ResponseSchema(RunnerIdsNotMatchingError, { statusCode: 406 })
|
@ResponseSchema(RunnerIdsNotMatchingError, { statusCode: 406 })
|
||||||
@ -77,11 +81,12 @@ export class RunnerController {
|
|||||||
throw new RunnerIdsNotMatchingError();
|
throw new RunnerIdsNotMatchingError();
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.runnerRepository.update(oldRunner, await runner.toRunner());
|
await this.runnerRepository.save(await runner.updateRunner(oldRunner));
|
||||||
return new ResponseRunner(await this.runnerRepository.findOne({ id: id }, { relations: ['scans', 'group'] }));
|
return new ResponseRunner(await this.runnerRepository.findOne({ id: id }, { relations: ['scans', 'group'] }));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Delete('/:id')
|
@Delete('/:id')
|
||||||
|
@Authorized("RUNNER:DELETE")
|
||||||
@ResponseSchema(ResponseRunner)
|
@ResponseSchema(ResponseRunner)
|
||||||
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
@OnUndefined(204)
|
@OnUndefined(204)
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
import { EntityFromBody } from 'typeorm-routing-controllers-extensions';
|
|
||||||
import { RunnerOrganisationHasRunnersError, RunnerOrganisationHasTeamsError, RunnerOrganisationIdsNotMatchingError, RunnerOrganisationNotFoundError } from '../errors/RunnerOrganisationErrors';
|
import { RunnerOrganisationHasRunnersError, RunnerOrganisationHasTeamsError, RunnerOrganisationIdsNotMatchingError, RunnerOrganisationNotFoundError } from '../errors/RunnerOrganisationErrors';
|
||||||
import { CreateRunnerOrganisation } from '../models/actions/CreateRunnerOrganisation';
|
import { CreateRunnerOrganisation } from '../models/actions/CreateRunnerOrganisation';
|
||||||
|
import { UpdateRunnerOrganisation } from '../models/actions/UpdateRunnerOrganisation';
|
||||||
import { RunnerOrganisation } from '../models/entities/RunnerOrganisation';
|
import { RunnerOrganisation } from '../models/entities/RunnerOrganisation';
|
||||||
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
||||||
import { ResponseRunnerOrganisation } from '../models/responses/ResponseRunnerOrganisation';
|
import { ResponseRunnerOrganisation } from '../models/responses/ResponseRunnerOrganisation';
|
||||||
@ -12,7 +12,7 @@ import { RunnerTeamController } from './RunnerTeamController';
|
|||||||
|
|
||||||
|
|
||||||
@JsonController('/organisations')
|
@JsonController('/organisations')
|
||||||
//@Authorized('RUNNERS:read')
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
export class RunnerOrganisationController {
|
export class RunnerOrganisationController {
|
||||||
private runnerOrganisationRepository: Repository<RunnerOrganisation>;
|
private runnerOrganisationRepository: Repository<RunnerOrganisation>;
|
||||||
|
|
||||||
@ -24,6 +24,7 @@ export class RunnerOrganisationController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
|
@Authorized("ORGANISATION:GET")
|
||||||
@ResponseSchema(ResponseRunnerOrganisation, { isArray: true })
|
@ResponseSchema(ResponseRunnerOrganisation, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all runnerOrganisations.' })
|
@OpenAPI({ description: 'Lists all runnerOrganisations.' })
|
||||||
async getAll() {
|
async getAll() {
|
||||||
@ -36,6 +37,7 @@ export class RunnerOrganisationController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
|
@Authorized("ORGANISATION:GET")
|
||||||
@ResponseSchema(ResponseRunnerOrganisation)
|
@ResponseSchema(ResponseRunnerOrganisation)
|
||||||
@ResponseSchema(RunnerOrganisationNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerOrganisationNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(RunnerOrganisationNotFoundError)
|
@OnUndefined(RunnerOrganisationNotFoundError)
|
||||||
@ -47,6 +49,7 @@ export class RunnerOrganisationController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Post()
|
@Post()
|
||||||
|
@Authorized("ORGANISATION:CREATE")
|
||||||
@ResponseSchema(ResponseRunnerOrganisation)
|
@ResponseSchema(ResponseRunnerOrganisation)
|
||||||
@OpenAPI({ description: 'Create a new runnerOrganisation object (id will be generated automagicly).' })
|
@OpenAPI({ description: 'Create a new runnerOrganisation object (id will be generated automagicly).' })
|
||||||
async post(@Body({ validate: true }) createRunnerOrganisation: CreateRunnerOrganisation) {
|
async post(@Body({ validate: true }) createRunnerOrganisation: CreateRunnerOrganisation) {
|
||||||
@ -63,28 +66,29 @@ export class RunnerOrganisationController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Put('/:id')
|
@Put('/:id')
|
||||||
|
@Authorized("ORGANISATION:UPDATE")
|
||||||
@ResponseSchema(ResponseRunnerOrganisation)
|
@ResponseSchema(ResponseRunnerOrganisation)
|
||||||
@ResponseSchema(RunnerOrganisationNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerOrganisationNotFoundError, { statusCode: 404 })
|
||||||
@ResponseSchema(RunnerOrganisationIdsNotMatchingError, { statusCode: 406 })
|
@ResponseSchema(RunnerOrganisationIdsNotMatchingError, { statusCode: 406 })
|
||||||
@OpenAPI({ description: "Update a runnerOrganisation object (id can't be changed)." })
|
@OpenAPI({ description: "Update a runnerOrganisation object (id can't be changed)." })
|
||||||
async put(@Param('id') id: number, @EntityFromBody() runnerOrganisation: RunnerOrganisation) {
|
async put(@Param('id') id: number, @Body({ validate: true }) updateOrganisation: UpdateRunnerOrganisation) {
|
||||||
let oldRunnerOrganisation = await this.runnerOrganisationRepository.findOne({ id: id });
|
let oldRunnerOrganisation = await this.runnerOrganisationRepository.findOne({ id: id });
|
||||||
|
|
||||||
if (!oldRunnerOrganisation) {
|
if (!oldRunnerOrganisation) {
|
||||||
throw new RunnerOrganisationNotFoundError();
|
throw new RunnerOrganisationNotFoundError();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oldRunnerOrganisation.id != runnerOrganisation.id) {
|
if (oldRunnerOrganisation.id != updateOrganisation.id) {
|
||||||
throw new RunnerOrganisationIdsNotMatchingError();
|
throw new RunnerOrganisationIdsNotMatchingError();
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.runnerOrganisationRepository.update(oldRunnerOrganisation, runnerOrganisation);
|
await this.runnerOrganisationRepository.save(await updateOrganisation.updateRunnerOrganisation(oldRunnerOrganisation));
|
||||||
|
|
||||||
runnerOrganisation = await this.runnerOrganisationRepository.findOne(runnerOrganisation, { relations: ['address', 'contact', 'teams'] });
|
return new ResponseRunnerOrganisation(await this.runnerOrganisationRepository.findOne(id, { relations: ['address', 'contact', 'teams'] }));
|
||||||
return new ResponseRunnerOrganisation(runnerOrganisation);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Delete('/:id')
|
@Delete('/:id')
|
||||||
|
@Authorized("ORGANISATION:DELETE")
|
||||||
@ResponseSchema(ResponseRunnerOrganisation)
|
@ResponseSchema(ResponseRunnerOrganisation)
|
||||||
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
@ResponseSchema(RunnerOrganisationHasTeamsError, { statusCode: 406 })
|
@ResponseSchema(RunnerOrganisationHasTeamsError, { statusCode: 406 })
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
import { RunnerTeamHasRunnersError, RunnerTeamIdsNotMatchingError, RunnerTeamNotFoundError } from '../errors/RunnerTeamErrors';
|
import { RunnerTeamHasRunnersError, RunnerTeamIdsNotMatchingError, RunnerTeamNotFoundError } from '../errors/RunnerTeamErrors';
|
||||||
@ -11,7 +11,7 @@ import { RunnerController } from './RunnerController';
|
|||||||
|
|
||||||
|
|
||||||
@JsonController('/teams')
|
@JsonController('/teams')
|
||||||
//@Authorized('RUNNERS:read')
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
export class RunnerTeamController {
|
export class RunnerTeamController {
|
||||||
private runnerTeamRepository: Repository<RunnerTeam>;
|
private runnerTeamRepository: Repository<RunnerTeam>;
|
||||||
|
|
||||||
@ -23,6 +23,7 @@ export class RunnerTeamController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
|
@Authorized("TEAM:GET")
|
||||||
@ResponseSchema(ResponseRunnerTeam, { isArray: true })
|
@ResponseSchema(ResponseRunnerTeam, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all runnerTeams.' })
|
@OpenAPI({ description: 'Lists all runnerTeams.' })
|
||||||
async getAll() {
|
async getAll() {
|
||||||
@ -35,6 +36,7 @@ export class RunnerTeamController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
|
@Authorized("TEAM:GET")
|
||||||
@ResponseSchema(ResponseRunnerTeam)
|
@ResponseSchema(ResponseRunnerTeam)
|
||||||
@ResponseSchema(RunnerTeamNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerTeamNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(RunnerTeamNotFoundError)
|
@OnUndefined(RunnerTeamNotFoundError)
|
||||||
@ -46,6 +48,7 @@ export class RunnerTeamController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Post()
|
@Post()
|
||||||
|
@Authorized("TEAM:CREATE")
|
||||||
@ResponseSchema(ResponseRunnerTeam)
|
@ResponseSchema(ResponseRunnerTeam)
|
||||||
@OpenAPI({ description: 'Create a new runnerTeam object (id will be generated automagicly).' })
|
@OpenAPI({ description: 'Create a new runnerTeam object (id will be generated automagicly).' })
|
||||||
async post(@Body({ validate: true }) createRunnerTeam: CreateRunnerTeam) {
|
async post(@Body({ validate: true }) createRunnerTeam: CreateRunnerTeam) {
|
||||||
@ -63,6 +66,7 @@ export class RunnerTeamController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Put('/:id')
|
@Put('/:id')
|
||||||
|
@Authorized("TEAM:UPDATE")
|
||||||
@ResponseSchema(ResponseRunnerTeam)
|
@ResponseSchema(ResponseRunnerTeam)
|
||||||
@ResponseSchema(RunnerTeamNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerTeamNotFoundError, { statusCode: 404 })
|
||||||
@ResponseSchema(RunnerTeamIdsNotMatchingError, { statusCode: 406 })
|
@ResponseSchema(RunnerTeamIdsNotMatchingError, { statusCode: 406 })
|
||||||
@ -78,12 +82,13 @@ export class RunnerTeamController {
|
|||||||
throw new RunnerTeamIdsNotMatchingError();
|
throw new RunnerTeamIdsNotMatchingError();
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.runnerTeamRepository.update(oldRunnerTeam, await runnerTeam.toRunnerTeam());
|
await this.runnerTeamRepository.save(await runnerTeam.updateRunnerTeam(oldRunnerTeam));
|
||||||
|
|
||||||
return new ResponseRunnerTeam(await this.runnerTeamRepository.findOne({ id: runnerTeam.id }, { relations: ['parentGroup', 'contact'] }));
|
return new ResponseRunnerTeam(await this.runnerTeamRepository.findOne({ id: runnerTeam.id }, { relations: ['parentGroup', 'contact'] }));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Delete('/:id')
|
@Delete('/:id')
|
||||||
|
@Authorized("TEAM:DELETE")
|
||||||
@ResponseSchema(ResponseRunnerTeam)
|
@ResponseSchema(ResponseRunnerTeam)
|
||||||
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
@ResponseSchema(RunnerTeamHasRunnersError, { statusCode: 406 })
|
@ResponseSchema(RunnerTeamHasRunnersError, { statusCode: 406 })
|
||||||
|
22
src/controllers/StatusController.ts
Normal file
22
src/controllers/StatusController.ts
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import { Get, JsonController } from 'routing-controllers';
|
||||||
|
import { OpenAPI } from 'routing-controllers-openapi';
|
||||||
|
import { getConnection } from 'typeorm';
|
||||||
|
|
||||||
|
@JsonController('/status')
|
||||||
|
export class StatusController {
|
||||||
|
|
||||||
|
@Get()
|
||||||
|
@OpenAPI({ description: "Lists all tracks." })
|
||||||
|
get() {
|
||||||
|
let connection;
|
||||||
|
try {
|
||||||
|
connection = getConnection();
|
||||||
|
} catch {
|
||||||
|
throw new Error("sth is wrong, i can feel it....");
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
"controllers": "✔",
|
||||||
|
"database connection": "✔"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
import { EntityFromBody } from 'typeorm-routing-controllers-extensions';
|
import { EntityFromBody } from 'typeorm-routing-controllers-extensions';
|
||||||
@ -9,7 +9,7 @@ import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
|||||||
import { ResponseTrack } from '../models/responses/ResponseTrack';
|
import { ResponseTrack } from '../models/responses/ResponseTrack';
|
||||||
|
|
||||||
@JsonController('/tracks')
|
@JsonController('/tracks')
|
||||||
//@Authorized("TRACKS:read")
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
export class TrackController {
|
export class TrackController {
|
||||||
private trackRepository: Repository<Track>;
|
private trackRepository: Repository<Track>;
|
||||||
|
|
||||||
@ -21,8 +21,8 @@ export class TrackController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
|
@Authorized("TRACK:GET")
|
||||||
@ResponseSchema(ResponseTrack, { isArray: true })
|
@ResponseSchema(ResponseTrack, { isArray: true })
|
||||||
@OpenAPI({ description: "Lists all tracks." })
|
|
||||||
async getAll() {
|
async getAll() {
|
||||||
let responseTracks: ResponseTrack[] = new Array<ResponseTrack>();
|
let responseTracks: ResponseTrack[] = new Array<ResponseTrack>();
|
||||||
const tracks = await this.trackRepository.find();
|
const tracks = await this.trackRepository.find();
|
||||||
@ -33,6 +33,7 @@ export class TrackController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
|
@Authorized("TRACK:GET")
|
||||||
@ResponseSchema(ResponseTrack)
|
@ResponseSchema(ResponseTrack)
|
||||||
@ResponseSchema(TrackNotFoundError, { statusCode: 404 })
|
@ResponseSchema(TrackNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(TrackNotFoundError)
|
@OnUndefined(TrackNotFoundError)
|
||||||
@ -44,6 +45,7 @@ export class TrackController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Post()
|
@Post()
|
||||||
|
@Authorized("TRACK:CREATE")
|
||||||
@ResponseSchema(ResponseTrack)
|
@ResponseSchema(ResponseTrack)
|
||||||
@OpenAPI({ description: "Create a new track object (id will be generated automagicly)." })
|
@OpenAPI({ description: "Create a new track object (id will be generated automagicly)." })
|
||||||
async post(
|
async post(
|
||||||
@ -54,6 +56,7 @@ export class TrackController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Put('/:id')
|
@Put('/:id')
|
||||||
|
@Authorized("TRACK:UPDATE")
|
||||||
@ResponseSchema(ResponseTrack)
|
@ResponseSchema(ResponseTrack)
|
||||||
@ResponseSchema(TrackNotFoundError, { statusCode: 404 })
|
@ResponseSchema(TrackNotFoundError, { statusCode: 404 })
|
||||||
@ResponseSchema(TrackIdsNotMatchingError, { statusCode: 406 })
|
@ResponseSchema(TrackIdsNotMatchingError, { statusCode: 406 })
|
||||||
@ -69,11 +72,12 @@ export class TrackController {
|
|||||||
throw new TrackIdsNotMatchingError();
|
throw new TrackIdsNotMatchingError();
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.trackRepository.update(oldTrack, track);
|
await this.trackRepository.save(track);
|
||||||
return new ResponseTrack(track);
|
return new ResponseTrack(track);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Delete('/:id')
|
@Delete('/:id')
|
||||||
|
@Authorized("TRACK:DELETE")
|
||||||
@ResponseSchema(ResponseTrack)
|
@ResponseSchema(ResponseTrack)
|
||||||
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
@OnUndefined(204)
|
@OnUndefined(204)
|
||||||
|
@ -1,15 +1,18 @@
|
|||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
import { EntityFromBody } from 'typeorm-routing-controllers-extensions';
|
|
||||||
import { UserIdsNotMatchingError, UserNotFoundError } from '../errors/UserErrors';
|
import { UserIdsNotMatchingError, UserNotFoundError } from '../errors/UserErrors';
|
||||||
import { UserGroupNotFoundError } from '../errors/UserGroupErrors';
|
import { UserGroupNotFoundError } from '../errors/UserGroupErrors';
|
||||||
import { CreateUser } from '../models/actions/CreateUser';
|
import { CreateUser } from '../models/actions/CreateUser';
|
||||||
|
import { UpdateUser } from '../models/actions/UpdateUser';
|
||||||
import { User } from '../models/entities/User';
|
import { User } from '../models/entities/User';
|
||||||
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
||||||
|
import { ResponseUser } from '../models/responses/ResponseUser';
|
||||||
|
import { PermissionController } from './PermissionController';
|
||||||
|
|
||||||
|
|
||||||
@JsonController('/users')
|
@JsonController('/users')
|
||||||
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
export class UserController {
|
export class UserController {
|
||||||
private userRepository: Repository<User>;
|
private userRepository: Repository<User>;
|
||||||
|
|
||||||
@ -21,22 +24,32 @@ export class UserController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
|
@Authorized("USER:GET")
|
||||||
@ResponseSchema(User, { isArray: true })
|
@ResponseSchema(User, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all users.' })
|
@OpenAPI({ description: 'Lists all users.' })
|
||||||
getAll() {
|
async getAll() {
|
||||||
return this.userRepository.find();
|
let responseUsers: ResponseUser[] = new Array<ResponseUser>();
|
||||||
|
const users = await this.userRepository.find({ relations: ['permissions', 'groups'] });
|
||||||
|
users.forEach(user => {
|
||||||
|
responseUsers.push(new ResponseUser(user));
|
||||||
|
});
|
||||||
|
return responseUsers;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
|
@Authorized("USER:GET")
|
||||||
@ResponseSchema(User)
|
@ResponseSchema(User)
|
||||||
@ResponseSchema(UserNotFoundError, { statusCode: 404 })
|
@ResponseSchema(UserNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(UserNotFoundError)
|
@OnUndefined(UserNotFoundError)
|
||||||
@OpenAPI({ description: 'Returns a user of a specified id (if it exists)' })
|
@OpenAPI({ description: 'Returns a user of a specified id (if it exists)' })
|
||||||
getOne(@Param('id') id: number) {
|
async getOne(@Param('id') id: number) {
|
||||||
return this.userRepository.findOne({ id: id });
|
let user = await this.userRepository.findOne({ id: id }, { relations: ['permissions', 'groups'] })
|
||||||
|
if (!user) { throw new UserNotFoundError(); }
|
||||||
|
return new ResponseUser(user);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post()
|
@Post()
|
||||||
|
@Authorized("USER:CREATE")
|
||||||
@ResponseSchema(User)
|
@ResponseSchema(User)
|
||||||
@ResponseSchema(UserGroupNotFoundError)
|
@ResponseSchema(UserGroupNotFoundError)
|
||||||
@OpenAPI({ description: 'Create a new user object (id will be generated automagicly).' })
|
@OpenAPI({ description: 'Create a new user object (id will be generated automagicly).' })
|
||||||
@ -48,41 +61,48 @@ export class UserController {
|
|||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.userRepository.save(user);
|
user = await this.userRepository.save(user)
|
||||||
|
return new ResponseUser(await this.userRepository.findOne(user, { relations: ['permissions', 'groups'] }));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Put('/:id')
|
@Put('/:id')
|
||||||
|
@Authorized("USER:UPDATE")
|
||||||
@ResponseSchema(User)
|
@ResponseSchema(User)
|
||||||
@ResponseSchema(UserNotFoundError, { statusCode: 404 })
|
@ResponseSchema(UserNotFoundError, { statusCode: 404 })
|
||||||
@ResponseSchema(UserIdsNotMatchingError, { statusCode: 406 })
|
@ResponseSchema(UserIdsNotMatchingError, { statusCode: 406 })
|
||||||
@OpenAPI({ description: "Update a user object (id can't be changed)." })
|
@OpenAPI({ description: "Update a user object (id can't be changed)." })
|
||||||
async put(@Param('id') id: number, @EntityFromBody() user: User) {
|
async put(@Param('id') id: number, @Body({ validate: true }) updateUser: UpdateUser) {
|
||||||
let oldUser = await this.userRepository.findOne({ id: id });
|
let oldUser = await this.userRepository.findOne({ id: id });
|
||||||
|
|
||||||
if (!oldUser) {
|
if (!oldUser) {
|
||||||
throw new UserNotFoundError();
|
throw new UserNotFoundError();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oldUser.id != user.id) {
|
if (oldUser.id != updateUser.id) {
|
||||||
throw new UserIdsNotMatchingError();
|
throw new UserIdsNotMatchingError();
|
||||||
}
|
}
|
||||||
|
await this.userRepository.save(await updateUser.updateUser(oldUser));
|
||||||
|
|
||||||
await this.userRepository.update(oldUser, user);
|
return new ResponseUser(await this.userRepository.findOne({ id: id }, { relations: ['permissions', 'groups'] }));
|
||||||
return user;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Delete('/:id')
|
@Delete('/:id')
|
||||||
|
@Authorized("USER:DELETE")
|
||||||
@ResponseSchema(User)
|
@ResponseSchema(User)
|
||||||
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
@OnUndefined(204)
|
@OnUndefined(204)
|
||||||
@OpenAPI({ description: 'Delete a specified runner (if it exists).' })
|
@OpenAPI({ description: 'Delete a user runner (if it exists).' })
|
||||||
async remove(@Param("id") id: number) {
|
async remove(@Param("id") id: number, @QueryParam("force") force: boolean) {
|
||||||
let user = await this.userRepository.findOne({ id: id });
|
let user = await this.userRepository.findOne({ id: id });
|
||||||
if (!user) {
|
if (!user) { return null; }
|
||||||
return null;
|
const responseUser = await this.userRepository.findOne({ id: id }, { relations: ['permissions', 'groups'] });;
|
||||||
|
|
||||||
|
const permissionControler = new PermissionController();
|
||||||
|
for (let permission of responseUser.permissions) {
|
||||||
|
await permissionControler.remove(permission.id, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.userRepository.delete(user);
|
await this.userRepository.delete(user);
|
||||||
return user;
|
return new ResponseUser(responseUser);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { getConnectionManager, Repository } from 'typeorm';
|
||||||
import { EntityFromBody } from 'typeorm-routing-controllers-extensions';
|
import { EntityFromBody } from 'typeorm-routing-controllers-extensions';
|
||||||
@ -6,9 +6,12 @@ import { UserGroupIdsNotMatchingError, UserGroupNotFoundError } from '../errors/
|
|||||||
import { CreateUserGroup } from '../models/actions/CreateUserGroup';
|
import { CreateUserGroup } from '../models/actions/CreateUserGroup';
|
||||||
import { UserGroup } from '../models/entities/UserGroup';
|
import { UserGroup } from '../models/entities/UserGroup';
|
||||||
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
import { ResponseEmpty } from '../models/responses/ResponseEmpty';
|
||||||
|
import { ResponseUserGroup } from '../models/responses/ResponseUserGroup';
|
||||||
|
import { PermissionController } from './PermissionController';
|
||||||
|
|
||||||
|
|
||||||
@JsonController('/usergroups')
|
@JsonController('/usergroups')
|
||||||
|
@OpenAPI({ security: [{ "AuthToken": [] }] })
|
||||||
export class UserGroupController {
|
export class UserGroupController {
|
||||||
private userGroupsRepository: Repository<UserGroup>;
|
private userGroupsRepository: Repository<UserGroup>;
|
||||||
|
|
||||||
@ -20,6 +23,7 @@ export class UserGroupController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
|
@Authorized("USERGROUP:GET")
|
||||||
@ResponseSchema(UserGroup, { isArray: true })
|
@ResponseSchema(UserGroup, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all usergroups.' })
|
@OpenAPI({ description: 'Lists all usergroups.' })
|
||||||
getAll() {
|
getAll() {
|
||||||
@ -27,6 +31,7 @@ export class UserGroupController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
|
@Authorized("USERGROUP:GET")
|
||||||
@ResponseSchema(UserGroup)
|
@ResponseSchema(UserGroup)
|
||||||
@ResponseSchema(UserGroupNotFoundError, { statusCode: 404 })
|
@ResponseSchema(UserGroupNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(UserGroupNotFoundError)
|
@OnUndefined(UserGroupNotFoundError)
|
||||||
@ -36,6 +41,7 @@ export class UserGroupController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Post()
|
@Post()
|
||||||
|
@Authorized("USERGROUP:CREATE")
|
||||||
@ResponseSchema(UserGroup)
|
@ResponseSchema(UserGroup)
|
||||||
@ResponseSchema(UserGroupNotFoundError)
|
@ResponseSchema(UserGroupNotFoundError)
|
||||||
@OpenAPI({ description: 'Create a new usergroup object (id will be generated automagicly).' })
|
@OpenAPI({ description: 'Create a new usergroup object (id will be generated automagicly).' })
|
||||||
@ -51,6 +57,7 @@ export class UserGroupController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Put('/:id')
|
@Put('/:id')
|
||||||
|
@Authorized("USERGROUP:UPDATE")
|
||||||
@ResponseSchema(UserGroup)
|
@ResponseSchema(UserGroup)
|
||||||
@ResponseSchema(UserGroupNotFoundError, { statusCode: 404 })
|
@ResponseSchema(UserGroupNotFoundError, { statusCode: 404 })
|
||||||
@ResponseSchema(UserGroupIdsNotMatchingError, { statusCode: 406 })
|
@ResponseSchema(UserGroupIdsNotMatchingError, { statusCode: 406 })
|
||||||
@ -66,22 +73,27 @@ export class UserGroupController {
|
|||||||
throw new UserGroupIdsNotMatchingError();
|
throw new UserGroupIdsNotMatchingError();
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.userGroupsRepository.update(oldUserGroup, userGroup);
|
await this.userGroupsRepository.save(userGroup);
|
||||||
return userGroup;
|
return userGroup;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Delete('/:id')
|
@Delete('/:id')
|
||||||
@ResponseSchema(UserGroup)
|
@Authorized("USERGROUP:DELETE")
|
||||||
|
@ResponseSchema(ResponseUserGroup)
|
||||||
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
@ResponseSchema(ResponseEmpty, { statusCode: 204 })
|
||||||
@OnUndefined(204)
|
@OnUndefined(204)
|
||||||
@OpenAPI({ description: 'Delete a specified usergroup (if it exists).' })
|
@OpenAPI({ description: 'Delete a specified usergroup (if it exists).' })
|
||||||
async remove(@Param("id") id: number) {
|
async remove(@Param("id") id: number, @QueryParam("force") force: boolean) {
|
||||||
let group = await this.userGroupsRepository.findOne({ id: id });
|
let group = await this.userGroupsRepository.findOne({ id: id });
|
||||||
if (!group) {
|
if (!group) { return null; }
|
||||||
return null;
|
const responseGroup = await this.userGroupsRepository.findOne({ id: id }, { relations: ['permissions'] });;
|
||||||
|
|
||||||
|
const permissionControler = new PermissionController();
|
||||||
|
for (let permission of responseGroup.permissions) {
|
||||||
|
await permissionControler.remove(permission.id, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.userGroupsRepository.delete(group);
|
await this.userGroupsRepository.delete(group);
|
||||||
return group;
|
return new ResponseUserGroup(responseGroup);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,63 +1,57 @@
|
|||||||
import { IsString } from 'class-validator';
|
import { IsString } from 'class-validator';
|
||||||
import { ForbiddenError, NotAcceptableError, NotFoundError, UnauthorizedError } from 'routing-controllers';
|
import { ForbiddenError, NotAcceptableError, NotFoundError, UnauthorizedError } from 'routing-controllers';
|
||||||
|
|
||||||
/**
|
|
||||||
* Error to throw when a jwt is expired.
|
|
||||||
*/
|
|
||||||
export class ExpiredJWTError extends UnauthorizedError {
|
|
||||||
@IsString()
|
|
||||||
name = "ExpiredJWTError"
|
|
||||||
|
|
||||||
@IsString()
|
|
||||||
message = "your provided jwt is expired"
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a jwt could not be parsed.
|
* Error to throw when a jwt could not be parsed.
|
||||||
|
* For example: Wrong signature or expired.
|
||||||
*/
|
*/
|
||||||
export class IllegalJWTError extends UnauthorizedError {
|
export class IllegalJWTError extends UnauthorizedError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "IllegalJWTError"
|
name = "IllegalJWTError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "your provided jwt could not be parsed"
|
message = "Your provided jwt could not be parsed."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when user is nonexistant or refreshtoken is invalid.
|
* Error to throw when user is nonexistant or refreshtoken is invalid.
|
||||||
|
* This can happen if someone provides a JWT with a invalid user id or the refreshTokenCount of the user is higher that the provided jwt's is.
|
||||||
*/
|
*/
|
||||||
export class UserNonexistantOrRefreshtokenInvalidError extends UnauthorizedError {
|
export class UserNonexistantOrRefreshtokenInvalidError extends UnauthorizedError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "UserNonexistantOrRefreshtokenInvalidError"
|
name = "UserNonexistantOrRefreshtokenInvalidError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "user is nonexistant or refreshtoken is invalid"
|
message = "User is nonexistant or refreshtoken is invalid."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when provided credentials are invalid.
|
* Error to throw when provided credentials are invalid.
|
||||||
|
* We don't have seperate errors for username/mail and passwords to protect against guessing attacks.
|
||||||
*/
|
*/
|
||||||
export class InvalidCredentialsError extends UnauthorizedError {
|
export class InvalidCredentialsError extends UnauthorizedError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "InvalidCredentialsError"
|
name = "InvalidCredentialsError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "your provided credentials are invalid"
|
message = "Your provided credentials are invalid."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a jwt does not have permission for this route/action.
|
* Error to throw when a jwt does not have permission for this route/action.
|
||||||
|
* Mainly used be the @Authorized decorator (via the authchecker).
|
||||||
*/
|
*/
|
||||||
export class NoPermissionError extends ForbiddenError {
|
export class NoPermissionError extends ForbiddenError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "NoPermissionError"
|
name = "NoPermissionError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "your provided jwt does not have permission for this route/ action"
|
message = "Your provided jwt does not have permission for this route/ action."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when no username and no email is set.
|
* Error to throw when no username and no email is set.
|
||||||
|
* Because we have to identify users somehow.
|
||||||
*/
|
*/
|
||||||
export class UsernameOrEmailNeededError extends NotAcceptableError {
|
export class UsernameOrEmailNeededError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -68,47 +62,48 @@ export class UsernameOrEmailNeededError extends NotAcceptableError {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when no password is provided.
|
* Error to throw when no password is provided for a new user.
|
||||||
|
* Passwords are the minimum we need for user security.
|
||||||
*/
|
*/
|
||||||
export class PasswordNeededError extends NotAcceptableError {
|
export class PasswordNeededError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "PasswordNeededError"
|
name = "PasswordNeededError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "no password is provided - you need to provide it"
|
message = "No password is provided - you need to provide it."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when no user could be found mating the provided credential.
|
* Error to throw when no user could be found for a certain query.
|
||||||
*/
|
*/
|
||||||
export class UserNotFoundError extends NotFoundError {
|
export class UserNotFoundError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "UserNotFoundError"
|
name = "UserNotFoundError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "no user could be found for provided credential"
|
message = "The user you provided couldn't be located in the system. \n Please check your request."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when no jwt token was provided (but one had to be).
|
* Error to throw when no jwt was provided (but one had to be).
|
||||||
*/
|
*/
|
||||||
export class JwtNotProvidedError extends NotAcceptableError {
|
export class JwtNotProvidedError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "JwtNotProvidedError"
|
name = "JwtNotProvidedError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "no jwt token was provided"
|
message = "No jwt was provided."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when user was not found or refresh token count was invalid.
|
* Error to throw when user was not found or the jwt's refresh token count was invalid.
|
||||||
*/
|
*/
|
||||||
export class UserNotFoundOrRefreshTokenCountInvalidError extends NotAcceptableError {
|
export class UserNotFoundOrRefreshTokenCountInvalidError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "UserNotFoundOrRefreshTokenCountInvalidError"
|
name = "UserNotFoundOrRefreshTokenCountInvalidError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "user was not found or refresh token count was invalid"
|
message = "User was not found or the refresh token count is invalid."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -119,5 +114,5 @@ export class RefreshTokenCountInvalidError extends NotAcceptableError {
|
|||||||
name = "RefreshTokenCountInvalidError"
|
name = "RefreshTokenCountInvalidError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "refresh token count was invalid"
|
message = "Refresh token count is invalid."
|
||||||
}
|
}
|
36
src/errors/PermissionErrors.ts
Normal file
36
src/errors/PermissionErrors.ts
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import { IsString } from 'class-validator';
|
||||||
|
import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Error to throw when a permission couldn't be found.
|
||||||
|
*/
|
||||||
|
export class PermissionNotFoundError extends NotFoundError {
|
||||||
|
@IsString()
|
||||||
|
name = "PermissionNotFoundError"
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
message = "Permission not found!"
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Error to throw when two permissions' ids don't match.
|
||||||
|
* Usually occurs when a user tries to change a permission's id.
|
||||||
|
*/
|
||||||
|
export class PermissionIdsNotMatchingError extends NotAcceptableError {
|
||||||
|
@IsString()
|
||||||
|
name = "PermissionIdsNotMatchingError"
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
message = "The ids don't match! \n And if you wanted to change a permission's id: This isn't allowed!"
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Error to throw when a permission gets provided without a principal.
|
||||||
|
*/
|
||||||
|
export class PermissionNeedsPrincipalError extends NotAcceptableError {
|
||||||
|
@IsString()
|
||||||
|
name = "PermissionNeedsPrincipalError"
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
message = "You provided no principal for this permission."
|
||||||
|
}
|
24
src/errors/PrincipalErrors.ts
Normal file
24
src/errors/PrincipalErrors.ts
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
import { IsString } from 'class-validator';
|
||||||
|
import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Error to throw when a user couldn't be found.
|
||||||
|
*/
|
||||||
|
export class PrincipalNotFoundError extends NotFoundError {
|
||||||
|
@IsString()
|
||||||
|
name = "PrincipalNotFoundError"
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
message = "Principal not found!"
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Error to throw, when a provided runnerOrganisation doesn't belong to the accepted types.
|
||||||
|
*/
|
||||||
|
export class PrincipalWrongTypeError extends NotAcceptableError {
|
||||||
|
@IsString()
|
||||||
|
name = "PrincipalWrongTypeError"
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
message = "The princial must have an existing principal's id. \n You provided a object of another type."
|
||||||
|
}
|
@ -3,7 +3,6 @@ import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a runner couldn't be found.
|
* Error to throw when a runner couldn't be found.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerNotFoundError extends NotFoundError {
|
export class RunnerNotFoundError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -16,14 +15,13 @@ export class RunnerNotFoundError extends NotFoundError {
|
|||||||
/**
|
/**
|
||||||
* Error to throw when two runners' ids don't match.
|
* Error to throw when two runners' ids don't match.
|
||||||
* Usually occurs when a user tries to change a runner's id.
|
* Usually occurs when a user tries to change a runner's id.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerIdsNotMatchingError extends NotAcceptableError {
|
export class RunnerIdsNotMatchingError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "RunnerIdsNotMatchingError"
|
name = "RunnerIdsNotMatchingError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "The id's don't match!! \n And if you wanted to change a runner's id: This isn't allowed"
|
message = "The ids don't match! \n And if you wanted to change a runner's id: This isn't allowed!"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -3,7 +3,6 @@ import { NotFoundError } from 'routing-controllers';
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a runner group couldn't be found.
|
* Error to throw when a runner group couldn't be found.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerGroupNotFoundError extends NotFoundError {
|
export class RunnerGroupNotFoundError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
|
@ -3,7 +3,6 @@ import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a runner organisation couldn't be found.
|
* Error to throw when a runner organisation couldn't be found.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerOrganisationNotFoundError extends NotFoundError {
|
export class RunnerOrganisationNotFoundError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -15,39 +14,36 @@ export class RunnerOrganisationNotFoundError extends NotFoundError {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when two runner organisations' ids don't match.
|
* Error to throw when two runner organisations' ids don't match.
|
||||||
* Usually occurs when a user tries to change a runner's id.
|
* Usually occurs when a user tries to change a runner organisation's id.
|
||||||
* Implemented this way to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerOrganisationIdsNotMatchingError extends NotAcceptableError {
|
export class RunnerOrganisationIdsNotMatchingError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "RunnerOrganisationIdsNotMatchingError"
|
name = "RunnerOrganisationIdsNotMatchingError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "The id's don't match!! \n And if you wanted to change a runner's id: This isn't allowed"
|
message = "The ids don't match! \n And if you wanted to change a runner organisation's id: This isn't allowed!"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a organisation still has runners associated.
|
* Error to throw when a organisation still has runners associated.
|
||||||
* Implemented this waysto work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerOrganisationHasRunnersError extends NotAcceptableError {
|
export class RunnerOrganisationHasRunnersError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "RunnerOrganisationHasRunnersError"
|
name = "RunnerOrganisationHasRunnersError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "This organisation still has runners associated with it. \n If you want to delete this organisation with all it's runners and teams ass `?force` to your query."
|
message = "This organisation still has runners associated with it. \n If you want to delete this organisation with all it's runners and teams add `?force` to your query."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a organisation still has runners associated.
|
* Error to throw when a organisation still has teams associated.
|
||||||
* Implemented this waysto work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerOrganisationHasTeamsError extends NotAcceptableError {
|
export class RunnerOrganisationHasTeamsError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "RunnerOrganisationHasTeamsError"
|
name = "RunnerOrganisationHasTeamsError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "This organisation still has teams associated with it. \n If you want to delete this organisation with all it's runners and teams ass `?force` to your query."
|
message = "This organisation still has teams associated with it. \n If you want to delete this organisation with all it's runners and teams add `?force` to your query."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -3,7 +3,6 @@ import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a runner team couldn't be found.
|
* Error to throw when a runner team couldn't be found.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerTeamNotFoundError extends NotFoundError {
|
export class RunnerTeamNotFoundError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -15,32 +14,29 @@ export class RunnerTeamNotFoundError extends NotFoundError {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when two runner teams' ids don't match.
|
* Error to throw when two runner teams' ids don't match.
|
||||||
* Usually occurs when a user tries to change a runner's id.
|
* Usually occurs when a user tries to change a runner team's id.
|
||||||
* Implemented this way to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerTeamIdsNotMatchingError extends NotAcceptableError {
|
export class RunnerTeamIdsNotMatchingError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "RunnerTeamIdsNotMatchingError"
|
name = "RunnerTeamIdsNotMatchingError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "The id's don't match!! \n And if you wanted to change a runner's id: This isn't allowed"
|
message = "The ids don't match! \n And if you wanted to change a runner's id: This isn't allowed!"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a team still has runners associated.
|
* Error to throw when a team still has runners associated.
|
||||||
* Implemented this waysto work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerTeamHasRunnersError extends NotAcceptableError {
|
export class RunnerTeamHasRunnersError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "RunnerTeamHasRunnersError"
|
name = "RunnerTeamHasRunnersError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "This team still has runners associated with it. \n If you want to delete this team with all it's runners and teams ass `?force` to your query."
|
message = "This team still has runners associated with it. \n If you want to delete this team with all it's runners and teams add `?force` to your query."
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a team still has runners associated.
|
* Error to throw when a team still has runners associated.
|
||||||
* Implemented this waysto work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class RunnerTeamNeedsParentError extends NotAcceptableError {
|
export class RunnerTeamNeedsParentError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import { JsonController, Param, Body, Get, Post, Put, Delete, NotFoundError, OnUndefined, NotAcceptableError } from 'routing-controllers';
|
import { IsString } from 'class-validator';
|
||||||
import { IsInt, IsNotEmpty, IsPositive, IsString } from 'class-validator';
|
import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when a track couldn't be found.
|
* Error to throw when a track couldn't be found.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class TrackNotFoundError extends NotFoundError {
|
export class TrackNotFoundError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -16,12 +15,11 @@ export class TrackNotFoundError extends NotFoundError {
|
|||||||
/**
|
/**
|
||||||
* Error to throw when two tracks' ids don't match.
|
* Error to throw when two tracks' ids don't match.
|
||||||
* Usually occurs when a user tries to change a track's id.
|
* Usually occurs when a user tries to change a track's id.
|
||||||
* Implemented this ways to work with the json-schema conversion for openapi.
|
|
||||||
*/
|
*/
|
||||||
export class TrackIdsNotMatchingError extends NotAcceptableError {
|
export class TrackIdsNotMatchingError extends NotAcceptableError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "TrackIdsNotMatchingError"
|
name = "TrackIdsNotMatchingError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "The id's don't match!! \n And if you wanted to change a track's id: This isn't allowed"
|
message = "The ids don't match! \n And if you wanted to change a track's id: This isn't allowed"
|
||||||
}
|
}
|
@ -3,14 +3,15 @@ import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when no username or email is set
|
* Error to throw when no username or email is set.
|
||||||
|
* We somehow need to identify you :)
|
||||||
*/
|
*/
|
||||||
export class UsernameOrEmailNeededError extends NotFoundError {
|
export class UsernameOrEmailNeededError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "UsernameOrEmailNeededError"
|
name = "UsernameOrEmailNeededError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "no username or email is set!"
|
message = "No username or email is set!"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -33,5 +34,5 @@ export class UserIdsNotMatchingError extends NotAcceptableError {
|
|||||||
name = "UserIdsNotMatchingError"
|
name = "UserIdsNotMatchingError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "The id's don't match!! \n And if you wanted to change a user's id: This isn't allowed"
|
message = "The ids don't match!! \n And if you wanted to change a user's id: This isn't allowed!"
|
||||||
}
|
}
|
@ -2,14 +2,14 @@ import { IsString } from 'class-validator';
|
|||||||
import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
import { NotAcceptableError, NotFoundError } from 'routing-controllers';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Error to throw when no groupname is set
|
* Error to throw when no groupname is set.
|
||||||
*/
|
*/
|
||||||
export class GroupNameNeededError extends NotFoundError {
|
export class GroupNameNeededError extends NotFoundError {
|
||||||
@IsString()
|
@IsString()
|
||||||
name = "GroupNameNeededError"
|
name = "GroupNameNeededError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "no groupname is set!"
|
message = "No name is set for this group!"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -32,5 +32,5 @@ export class UserGroupIdsNotMatchingError extends NotAcceptableError {
|
|||||||
name = "UserGroupIdsNotMatchingError"
|
name = "UserGroupIdsNotMatchingError"
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
message = "The id's don't match!! \n If you wanted to change a usergroup's id: This isn't allowed"
|
message = "The ids don't match!! \n If you wanted to change a usergroup's id: This isn't allowed!"
|
||||||
}
|
}
|
114
src/jwtcreator.ts
Normal file
114
src/jwtcreator.ts
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
import { IsBoolean, IsEmail, IsInt, IsNotEmpty, IsOptional, IsString, IsUUID } from 'class-validator';
|
||||||
|
import * as jsonwebtoken from "jsonwebtoken";
|
||||||
|
import { config } from './config';
|
||||||
|
import { User } from './models/entities/User';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is responsible for all things JWT creation.
|
||||||
|
*/
|
||||||
|
export class JwtCreator {
|
||||||
|
/**
|
||||||
|
* Creates a new refresh token for a given user
|
||||||
|
* @param user User entity that the refresh token shall be created for
|
||||||
|
* @param expiry_timestamp Timestamp for the token expiry. Will be generated if not provided.
|
||||||
|
*/
|
||||||
|
public static createRefresh(user: User, expiry_timestamp?: number) {
|
||||||
|
if (!expiry_timestamp) { expiry_timestamp = Math.floor(Date.now() / 1000) + 10 * 36000; }
|
||||||
|
return jsonwebtoken.sign({
|
||||||
|
refreshTokenCount: user.refreshTokenCount,
|
||||||
|
id: user.id,
|
||||||
|
exp: expiry_timestamp
|
||||||
|
}, config.jwt_secret)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new access token for a given user
|
||||||
|
* @param user User entity that the access token shall be created for
|
||||||
|
* @param expiry_timestamp Timestamp for the token expiry. Will be generated if not provided.
|
||||||
|
*/
|
||||||
|
public static createAccess(user: User, expiry_timestamp?: number) {
|
||||||
|
if (!expiry_timestamp) { expiry_timestamp = Math.floor(Date.now() / 1000) + 10 * 36000; }
|
||||||
|
return jsonwebtoken.sign({
|
||||||
|
userdetails: new JwtUser(user),
|
||||||
|
exp: expiry_timestamp
|
||||||
|
}, config.jwt_secret)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Special variant of the user class that
|
||||||
|
*/
|
||||||
|
export class JwtUser {
|
||||||
|
@IsInt()
|
||||||
|
id: number;
|
||||||
|
|
||||||
|
@IsUUID(4)
|
||||||
|
uuid: string;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsEmail()
|
||||||
|
email?: string;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsString()
|
||||||
|
username?: string;
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
@IsNotEmpty()
|
||||||
|
firstname: string;
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
middlename?: string;
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
@IsNotEmpty()
|
||||||
|
lastname: string;
|
||||||
|
|
||||||
|
permissions: string[];
|
||||||
|
|
||||||
|
@IsBoolean()
|
||||||
|
enabled: boolean;
|
||||||
|
|
||||||
|
@IsInt()
|
||||||
|
@IsNotEmpty()
|
||||||
|
refreshTokenCount?: number;
|
||||||
|
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
profilePic?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new instance of this class based on a provided user entity.
|
||||||
|
* @param user User entity that shall be encapsulated in a jwt.
|
||||||
|
*/
|
||||||
|
public constructor(user: User) {
|
||||||
|
this.id = user.id;
|
||||||
|
this.firstname = user.firstname;
|
||||||
|
this.middlename = user.middlename;
|
||||||
|
this.lastname = user.lastname;
|
||||||
|
this.username = user.username;
|
||||||
|
this.email = user.email;
|
||||||
|
this.refreshTokenCount = user.refreshTokenCount;
|
||||||
|
this.uuid = user.uuid;
|
||||||
|
this.profilePic = user.profilePic;
|
||||||
|
this.permissions = this.getPermissions(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handels getting the permissions granted to this user (direct or indirect).
|
||||||
|
* @param user User which's permissions shall be gotten.
|
||||||
|
*/
|
||||||
|
public getPermissions(user: User): string[] {
|
||||||
|
let returnPermissions: string[] = new Array<string>();
|
||||||
|
for (let permission of user.permissions) {
|
||||||
|
returnPermissions.push(permission.toString());
|
||||||
|
}
|
||||||
|
for (let group of user.groups) {
|
||||||
|
for (let permission of group.permissions) {
|
||||||
|
returnPermissions.push(permission.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Array.from(new Set(returnPermissions));
|
||||||
|
}
|
||||||
|
}
|
@ -1,10 +1,16 @@
|
|||||||
import { createConnection } from "typeorm";
|
import { createConnection } from "typeorm";
|
||||||
|
import { runSeeder } from 'typeorm-seeding';
|
||||||
|
import { User } from '../models/entities/User';
|
||||||
|
import SeedUsers from '../seeds/SeedUsers';
|
||||||
/**
|
/**
|
||||||
* Loader for the database that creates the database connection and initializes the database tabels.
|
* Loader for the database that creates the database connection and initializes the database tabels.
|
||||||
|
* It also triggers the seeding process if no users got detected in the database.
|
||||||
*/
|
*/
|
||||||
export default async () => {
|
export default async () => {
|
||||||
const connection = await createConnection();
|
const connection = await createConnection();
|
||||||
connection.synchronize();
|
await connection.synchronize();
|
||||||
|
if (await connection.getRepository(User).count() === 0) {
|
||||||
|
await runSeeder(SeedUsers);
|
||||||
|
}
|
||||||
return connection;
|
return connection;
|
||||||
};
|
};
|
@ -1,10 +1,11 @@
|
|||||||
|
import cookieParser from "cookie-parser";
|
||||||
import { Application } from "express";
|
import { Application } from "express";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loader for express related configurations.
|
* Loader for express related configurations.
|
||||||
* Currently only enables the proxy trust.
|
* Configures proxy trusts, globally used middlewares and other express features.
|
||||||
*/
|
*/
|
||||||
export default async (app: Application) => {
|
export default async (app: Application) => {
|
||||||
app.enable('trust proxy');
|
app.enable('trust proxy');
|
||||||
|
app.use(cookieParser());
|
||||||
return app;
|
return app;
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ import openapiLoader from "./openapi";
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Index Loader that executes the other loaders in the right order.
|
* Index Loader that executes the other loaders in the right order.
|
||||||
|
* This basicly exists for abstraction and a overall better dev experience.
|
||||||
*/
|
*/
|
||||||
export default async (app: Application) => {
|
export default async (app: Application) => {
|
||||||
await databaseLoader();
|
await databaseLoader();
|
||||||
|
@ -5,7 +5,8 @@ import { routingControllersToSpec } from "routing-controllers-openapi";
|
|||||||
import * as swaggerUiExpress from "swagger-ui-express";
|
import * as swaggerUiExpress from "swagger-ui-express";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loader for everything openapi related - from creating the schema to serving it via a static route.
|
* Loader for everything openapi related - from creating the schema to serving it via a static route and swaggerUiExpress.
|
||||||
|
* All auth schema related stuff also has to be configured here
|
||||||
*/
|
*/
|
||||||
export default async (app: Application) => {
|
export default async (app: Application) => {
|
||||||
const storage = getMetadataArgsStorage();
|
const storage = getMetadataArgsStorage();
|
||||||
@ -26,7 +27,8 @@ export default async (app: Application) => {
|
|||||||
"AuthToken": {
|
"AuthToken": {
|
||||||
"type": "http",
|
"type": "http",
|
||||||
"scheme": "bearer",
|
"scheme": "bearer",
|
||||||
"bearerFormat": "JWT"
|
"bearerFormat": "JWT",
|
||||||
|
description: "A JWT based access token. Use /api/auth/login or /api/auth/refresh to get one."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { ExpressErrorMiddlewareInterface, Middleware } from "routing-controllers";
|
import { ExpressErrorMiddlewareInterface, Middleware } from "routing-controllers";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Our Error handling middlware that returns our custom httperrors to the user
|
* Our Error handling middlware that returns our custom httperrors to the user.
|
||||||
*/
|
*/
|
||||||
@Middleware({ type: "after" })
|
@Middleware({ type: "after" })
|
||||||
export class ErrorHandler implements ExpressErrorMiddlewareInterface {
|
export class ErrorHandler implements ExpressErrorMiddlewareInterface {
|
||||||
|
23
src/middlewares/RawBody.ts
Normal file
23
src/middlewares/RawBody.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
import { Request, Response } from 'express';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Custom express middleware that appends the raw body to the request obeject.
|
||||||
|
* Mainly used for parsing csvs from boddies.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const RawBodyMiddleware = (req: Request, res: Response, next: () => void) => {
|
||||||
|
const body = []
|
||||||
|
req.on('data', chunk => {
|
||||||
|
body.push(chunk)
|
||||||
|
})
|
||||||
|
req.on('end', () => {
|
||||||
|
const rawBody = Buffer.concat(body)
|
||||||
|
req['rawBody'] = rawBody
|
||||||
|
next()
|
||||||
|
})
|
||||||
|
req.on('error', () => {
|
||||||
|
res.sendStatus(400)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export default RawBodyMiddleware
|
@ -1,16 +1,19 @@
|
|||||||
import { IsNotEmpty, IsOptional, IsPostalCode, IsString } from 'class-validator';
|
import { IsNotEmpty, IsOptional, IsPostalCode, IsString } from 'class-validator';
|
||||||
import { Address } from '../entities/Address';
|
import { Address } from '../entities/Address';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new Address entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateAddress {
|
export class CreateAddress {
|
||||||
/**
|
/**
|
||||||
* The address's description.
|
* The newaddress's description.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
description?: string;
|
description?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's first line.
|
* The new address's first line.
|
||||||
* Containing the street and house number.
|
* Containing the street and house number.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -18,7 +21,7 @@ export class CreateAddress {
|
|||||||
address1: string;
|
address1: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's second line.
|
* The new address's second line.
|
||||||
* Containing optional information.
|
* Containing optional information.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -26,7 +29,9 @@ export class CreateAddress {
|
|||||||
address2?: string;
|
address2?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's postal code.
|
* The new address's postal code.
|
||||||
|
* This will get checked against the postal code syntax for the configured country.
|
||||||
|
* TODO: Implement the config option.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ -34,21 +39,21 @@ export class CreateAddress {
|
|||||||
postalcode: string;
|
postalcode: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's city.
|
* The new address's city.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
city: string;
|
city: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's country.
|
* The new address's country.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
country: string;
|
country: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a Address object based on this.
|
* Creates a new Address entity from this.
|
||||||
*/
|
*/
|
||||||
public toAddress(): Address {
|
public toAddress(): Address {
|
||||||
let newAddress: Address = new Address();
|
let newAddress: Address = new Address();
|
||||||
|
@ -1,24 +1,47 @@
|
|||||||
import * as argon2 from "argon2";
|
import * as argon2 from "argon2";
|
||||||
import { IsEmail, IsOptional, IsString } from 'class-validator';
|
import { IsEmail, IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
||||||
import * as jsonwebtoken from 'jsonwebtoken';
|
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { config } from '../../config';
|
|
||||||
import { InvalidCredentialsError, PasswordNeededError, UserNotFoundError } from '../../errors/AuthError';
|
import { InvalidCredentialsError, PasswordNeededError, UserNotFoundError } from '../../errors/AuthError';
|
||||||
import { UsernameOrEmailNeededError } from '../../errors/UserErrors';
|
import { UsernameOrEmailNeededError } from '../../errors/UserErrors';
|
||||||
|
import { JwtCreator } from '../../jwtcreator';
|
||||||
import { User } from '../entities/User';
|
import { User } from '../entities/User';
|
||||||
import { Auth } from '../responses/ResponseAuth';
|
import { Auth } from '../responses/ResponseAuth';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to create auth credentials based on user credentials provided in a json body (post request).
|
||||||
|
* To be a little bit more exact: Is takes in a username/email + password and creates a new access and refresh token for the user.
|
||||||
|
* It of course checks for user existance, password validity and so on.
|
||||||
|
*/
|
||||||
export class CreateAuth {
|
export class CreateAuth {
|
||||||
|
/**
|
||||||
|
* The username of the user that want's to login.
|
||||||
|
* Either username or email have to be provided.
|
||||||
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsString()
|
@IsString()
|
||||||
username?: string;
|
username?: string;
|
||||||
@IsString()
|
|
||||||
password: string;
|
/**
|
||||||
|
* The email address of the user that want's to login.
|
||||||
|
* Either username or email have to be provided.
|
||||||
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsEmail()
|
@IsEmail()
|
||||||
@IsString()
|
@IsString()
|
||||||
email?: string;
|
email?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's password.
|
||||||
|
* Will be checked against an argon2 hash.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
|
password: string;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new auth object based on this.
|
||||||
|
*/
|
||||||
public async toAuth(): Promise<Auth> {
|
public async toAuth(): Promise<Auth> {
|
||||||
let newAuth: Auth = new Auth();
|
let newAuth: Auth = new Auth();
|
||||||
|
|
||||||
@ -26,34 +49,24 @@ export class CreateAuth {
|
|||||||
throw new UsernameOrEmailNeededError();
|
throw new UsernameOrEmailNeededError();
|
||||||
}
|
}
|
||||||
if (!this.password) {
|
if (!this.password) {
|
||||||
throw new PasswordNeededError()
|
throw new PasswordNeededError();
|
||||||
}
|
}
|
||||||
const found_users = await getConnectionManager().get().getRepository(User).find({ relations: ['groups', 'permissions'], where: [{ username: this.username }, { email: this.email }] });
|
const found_user = await getConnectionManager().get().getRepository(User).findOne({ relations: ['groups', 'permissions', 'groups.permissions'], where: [{ username: this.username }, { email: this.email }] });
|
||||||
if (found_users.length === 0) {
|
if (!found_user) {
|
||||||
throw new UserNotFoundError()
|
throw new UserNotFoundError();
|
||||||
} else {
|
|
||||||
const found_user = found_users[0]
|
|
||||||
if (await argon2.verify(found_user.password, this.password + found_user.uuid)) {
|
|
||||||
const timestamp_accesstoken_expiry = Math.floor(Date.now() / 1000) + 5 * 60
|
|
||||||
found_user.permissions = found_user.permissions || []
|
|
||||||
delete found_user.password;
|
|
||||||
newAuth.access_token = jsonwebtoken.sign({
|
|
||||||
userdetails: found_user,
|
|
||||||
exp: timestamp_accesstoken_expiry
|
|
||||||
}, config.jwt_secret)
|
|
||||||
newAuth.access_token_expires_at = timestamp_accesstoken_expiry
|
|
||||||
//
|
|
||||||
const timestamp_refresh_expiry = Math.floor(Date.now() / 1000) + 10 * 36000
|
|
||||||
newAuth.refresh_token = jsonwebtoken.sign({
|
|
||||||
refreshtokencount: found_user.refreshTokenCount,
|
|
||||||
userid: found_user.id,
|
|
||||||
exp: timestamp_refresh_expiry
|
|
||||||
}, config.jwt_secret)
|
|
||||||
newAuth.refresh_token_expires_at = timestamp_refresh_expiry
|
|
||||||
} else {
|
|
||||||
throw new InvalidCredentialsError()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
if (!(await argon2.verify(found_user.password, this.password + found_user.uuid))) {
|
||||||
|
throw new InvalidCredentialsError();
|
||||||
|
}
|
||||||
|
|
||||||
|
//Create the access token
|
||||||
|
const timestamp_accesstoken_expiry = Math.floor(Date.now() / 1000) + 5 * 60
|
||||||
|
newAuth.access_token = JwtCreator.createAccess(found_user, timestamp_accesstoken_expiry);
|
||||||
|
newAuth.access_token_expires_at = timestamp_accesstoken_expiry
|
||||||
|
//Create the refresh token
|
||||||
|
const timestamp_refresh_expiry = Math.floor(Date.now() / 1000) + 10 * 36000
|
||||||
|
newAuth.refresh_token = JwtCreator.createRefresh(found_user, timestamp_refresh_expiry);
|
||||||
|
newAuth.refresh_token_expires_at = timestamp_refresh_expiry
|
||||||
return newAuth;
|
return newAuth;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -5,32 +5,34 @@ import { AddressNotFoundError, AddressWrongTypeError } from '../../errors/Addres
|
|||||||
import { Address } from '../entities/Address';
|
import { Address } from '../entities/Address';
|
||||||
import { GroupContact } from '../entities/GroupContact';
|
import { GroupContact } from '../entities/GroupContact';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new Group entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateGroupContact {
|
export class CreateGroupContact {
|
||||||
/**
|
/**
|
||||||
* The contact's first name.
|
* The new contact's first name.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@IsString()
|
@IsString()
|
||||||
firstname: string;
|
firstname: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's middle name.
|
* The new contact's middle name.
|
||||||
* Optional
|
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsString()
|
@IsString()
|
||||||
middlename?: string;
|
middlename?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's last name.
|
* The new contact's last name.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@IsString()
|
@IsString()
|
||||||
lastname: string;
|
lastname: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's address.
|
* The new contact's address.
|
||||||
* Optional
|
* Must be the address's id.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -38,7 +40,7 @@ export class CreateGroupContact {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's phone number.
|
* The contact's phone number.
|
||||||
* Optional
|
* This will be validated against the configured country phone numer syntax (default: international).
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsPhoneNumber(config.phone_validation_countrycode)
|
@IsPhoneNumber(config.phone_validation_countrycode)
|
||||||
@ -46,17 +48,16 @@ export class CreateGroupContact {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's email address.
|
* The contact's email address.
|
||||||
* Optional
|
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsEmail()
|
@IsEmail()
|
||||||
email?: string;
|
email?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get's this participant's address from this.address.
|
* Gets the new contact's address by it's id.
|
||||||
*/
|
*/
|
||||||
public async getAddress(): Promise<Address> {
|
public async getAddress(): Promise<Address> {
|
||||||
if (this.address === undefined) {
|
if (this.address === undefined || this.address === null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if (!isNaN(this.address)) {
|
if (!isNaN(this.address)) {
|
||||||
@ -69,7 +70,7 @@ export class CreateGroupContact {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a Address object based on this.
|
* Creates a new Address entity from this.
|
||||||
*/
|
*/
|
||||||
public async toGroupContact(): Promise<GroupContact> {
|
public async toGroupContact(): Promise<GroupContact> {
|
||||||
let contact: GroupContact = new GroupContact();
|
let contact: GroupContact = new GroupContact();
|
||||||
|
@ -4,6 +4,9 @@ import { config } from '../../config';
|
|||||||
import { AddressNotFoundError, AddressWrongTypeError } from '../../errors/AddressErrors';
|
import { AddressNotFoundError, AddressWrongTypeError } from '../../errors/AddressErrors';
|
||||||
import { Address } from '../entities/Address';
|
import { Address } from '../entities/Address';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new Participant entity from a json body (post request).
|
||||||
|
*/
|
||||||
export abstract class CreateParticipant {
|
export abstract class CreateParticipant {
|
||||||
/**
|
/**
|
||||||
* The new participant's first name.
|
* The new participant's first name.
|
||||||
@ -14,7 +17,6 @@ export abstract class CreateParticipant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The new participant's middle name.
|
* The new participant's middle name.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -29,7 +31,7 @@ export abstract class CreateParticipant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The new participant's phone number.
|
* The new participant's phone number.
|
||||||
* Optional.
|
* This will be validated against the configured country phone numer syntax (default: international).
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -38,7 +40,6 @@ export abstract class CreateParticipant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The new participant's e-mail address.
|
* The new participant's e-mail address.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -48,17 +49,16 @@ export abstract class CreateParticipant {
|
|||||||
/**
|
/**
|
||||||
* The new participant's address.
|
* The new participant's address.
|
||||||
* Must be of type number (address id).
|
* Must be of type number (address id).
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
address?: number;
|
address?: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get's this participant's address from this.address.
|
* Gets the new participant's address by it's address.
|
||||||
*/
|
*/
|
||||||
public async getAddress(): Promise<Address> {
|
public async getAddress(): Promise<Address> {
|
||||||
if (this.address === undefined) {
|
if (this.address === undefined || this.address === null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if (!isNaN(this.address)) {
|
if (!isNaN(this.address)) {
|
||||||
|
60
src/models/actions/CreatePermission.ts
Normal file
60
src/models/actions/CreatePermission.ts
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import {
|
||||||
|
IsEnum,
|
||||||
|
IsInt,
|
||||||
|
IsNotEmpty
|
||||||
|
} from "class-validator";
|
||||||
|
import { getConnectionManager } from 'typeorm';
|
||||||
|
import { PrincipalNotFoundError } from '../../errors/PrincipalErrors';
|
||||||
|
import { Permission } from '../entities/Permission';
|
||||||
|
import { Principal } from '../entities/Principal';
|
||||||
|
import { PermissionAction } from '../enums/PermissionAction';
|
||||||
|
import { PermissionTarget } from '../enums/PermissionTargets';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new Permission entity from a json body (post request).
|
||||||
|
*/
|
||||||
|
export class CreatePermission {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new permissions's principal's id.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
@IsNotEmpty()
|
||||||
|
principal: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new permissions's target.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsEnum(PermissionTarget)
|
||||||
|
target: PermissionTarget;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new permissions's action.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsEnum(PermissionAction)
|
||||||
|
action: PermissionAction;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new Permission entity from this.
|
||||||
|
*/
|
||||||
|
public async toPermission(): Promise<Permission> {
|
||||||
|
let newPermission: Permission = new Permission();
|
||||||
|
|
||||||
|
newPermission.principal = await this.getPrincipal();
|
||||||
|
newPermission.target = this.target;
|
||||||
|
newPermission.action = this.action;
|
||||||
|
|
||||||
|
return newPermission;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the new permission's principal by it's id.
|
||||||
|
*/
|
||||||
|
public async getPrincipal(): Promise<Principal> {
|
||||||
|
let principal = await getConnectionManager().get().getRepository(Principal).findOne({ id: this.principal })
|
||||||
|
if (!principal) { throw new PrincipalNotFoundError(); }
|
||||||
|
return principal;
|
||||||
|
}
|
||||||
|
}
|
@ -7,6 +7,9 @@ import { Runner } from '../entities/Runner';
|
|||||||
import { RunnerGroup } from '../entities/RunnerGroup';
|
import { RunnerGroup } from '../entities/RunnerGroup';
|
||||||
import { CreateParticipant } from './CreateParticipant';
|
import { CreateParticipant } from './CreateParticipant';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new Runner entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateRunner extends CreateParticipant {
|
export class CreateRunner extends CreateParticipant {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -16,7 +19,7 @@ export class CreateRunner extends CreateParticipant {
|
|||||||
group: number;
|
group: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a Runner entity from this.
|
* Creates a new Runner entity from this.
|
||||||
*/
|
*/
|
||||||
public async toRunner(): Promise<Runner> {
|
public async toRunner(): Promise<Runner> {
|
||||||
let newRunner: Runner = new Runner();
|
let newRunner: Runner = new Runner();
|
||||||
@ -33,10 +36,10 @@ export class CreateRunner extends CreateParticipant {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages all the different ways a group can be provided.
|
* Gets the new runner's group by it's id.
|
||||||
*/
|
*/
|
||||||
public async getGroup(): Promise<RunnerGroup> {
|
public async getGroup(): Promise<RunnerGroup> {
|
||||||
if (this.group === undefined) {
|
if (this.group === undefined || this.group === null) {
|
||||||
throw new RunnerTeamNeedsParentError();
|
throw new RunnerTeamNeedsParentError();
|
||||||
}
|
}
|
||||||
if (!isNaN(this.group)) {
|
if (!isNaN(this.group)) {
|
||||||
|
@ -3,16 +3,19 @@ import { getConnectionManager } from 'typeorm';
|
|||||||
import { GroupContactNotFoundError, GroupContactWrongTypeError } from '../../errors/GroupContactErrors';
|
import { GroupContactNotFoundError, GroupContactWrongTypeError } from '../../errors/GroupContactErrors';
|
||||||
import { GroupContact } from '../entities/GroupContact';
|
import { GroupContact } from '../entities/GroupContact';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new RunnerGroup entity from a json body (post request).
|
||||||
|
*/
|
||||||
export abstract class CreateRunnerGroup {
|
export abstract class CreateRunnerGroup {
|
||||||
/**
|
/**
|
||||||
* The group's name.
|
* The new group's name.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@IsString()
|
@IsString()
|
||||||
name: string;
|
name: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The group's contact.
|
* The new group's contact.
|
||||||
* Optional
|
* Optional
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@ -20,7 +23,7 @@ export abstract class CreateRunnerGroup {
|
|||||||
contact?: number;
|
contact?: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get's this group's contact from this.address.
|
* Gets the new group's contact by it's id.
|
||||||
*/
|
*/
|
||||||
public async getContact(): Promise<GroupContact> {
|
public async getContact(): Promise<GroupContact> {
|
||||||
if (this.contact === undefined || this.contact === null) {
|
if (this.contact === undefined || this.contact === null) {
|
||||||
|
@ -5,21 +5,23 @@ import { Address } from '../entities/Address';
|
|||||||
import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
||||||
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new RunnerOrganisation entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateRunnerOrganisation extends CreateRunnerGroup {
|
export class CreateRunnerOrganisation extends CreateRunnerGroup {
|
||||||
/**
|
/**
|
||||||
* The new organisation's address.
|
* The new organisation's address.
|
||||||
* Must be of type number (address id).
|
* Must be of type number (address id).
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
address?: number;
|
address?: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get's this org's address from this.address.
|
* Gets the org's address by it's id.
|
||||||
*/
|
*/
|
||||||
public async getAddress(): Promise<Address> {
|
public async getAddress(): Promise<Address> {
|
||||||
if (this.address === undefined) {
|
if (this.address === undefined || this.address === null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if (!isNaN(this.address)) {
|
if (!isNaN(this.address)) {
|
||||||
@ -32,7 +34,7 @@ export class CreateRunnerOrganisation extends CreateRunnerGroup {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a RunnerOrganisation entity from this.
|
* Creates a new RunnerOrganisation entity from this.
|
||||||
*/
|
*/
|
||||||
public async toRunnerOrganisation(): Promise<RunnerOrganisation> {
|
public async toRunnerOrganisation(): Promise<RunnerOrganisation> {
|
||||||
let newRunnerOrganisation: RunnerOrganisation = new RunnerOrganisation();
|
let newRunnerOrganisation: RunnerOrganisation = new RunnerOrganisation();
|
||||||
|
@ -6,17 +6,23 @@ import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
|||||||
import { RunnerTeam } from '../entities/RunnerTeam';
|
import { RunnerTeam } from '../entities/RunnerTeam';
|
||||||
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new RunnerTeam entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateRunnerTeam extends CreateRunnerGroup {
|
export class CreateRunnerTeam extends CreateRunnerGroup {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The team's parent group (organisation).
|
* The new team's parent group (organisation).
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
parentGroup: number;
|
parentGroup: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the new team's parent org based on it's id.
|
||||||
|
*/
|
||||||
public async getParent(): Promise<RunnerOrganisation> {
|
public async getParent(): Promise<RunnerOrganisation> {
|
||||||
if (this.parentGroup === undefined) {
|
if (this.parentGroup === undefined || this.parentGroup === null) {
|
||||||
throw new RunnerTeamNeedsParentError();
|
throw new RunnerTeamNeedsParentError();
|
||||||
}
|
}
|
||||||
if (!isNaN(this.parentGroup)) {
|
if (!isNaN(this.parentGroup)) {
|
||||||
@ -29,7 +35,7 @@ export class CreateRunnerTeam extends CreateRunnerGroup {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a RunnerTeam entity from this.
|
* Creates a new RunnerTeam entity from this.
|
||||||
*/
|
*/
|
||||||
public async toRunnerTeam(): Promise<RunnerTeam> {
|
public async toRunnerTeam(): Promise<RunnerTeam> {
|
||||||
let newRunnerTeam: RunnerTeam = new RunnerTeam();
|
let newRunnerTeam: RunnerTeam = new RunnerTeam();
|
||||||
|
@ -1,23 +1,26 @@
|
|||||||
import { IsInt, IsNotEmpty, IsPositive, IsString } from 'class-validator';
|
import { IsInt, IsNotEmpty, IsPositive, IsString } from 'class-validator';
|
||||||
import { Track } from '../entities/Track';
|
import { Track } from '../entities/Track';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new Track entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateTrack {
|
export class CreateTrack {
|
||||||
/**
|
/**
|
||||||
* The track's name.
|
* The new track's name.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
name: string;
|
name: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The track's distance in meters (must be greater than 0).
|
* The new track's distance in meters (must be greater than 0).
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsPositive()
|
@IsPositive()
|
||||||
distance: number;
|
distance: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts a Track object based on this.
|
* Creates a new Track entity from this.
|
||||||
*/
|
*/
|
||||||
public toTrack(): Track {
|
public toTrack(): Track {
|
||||||
let newTrack: Track = new Track();
|
let newTrack: Track = new Track();
|
||||||
|
@ -8,6 +8,9 @@ import { UserGroupNotFoundError } from '../../errors/UserGroupErrors';
|
|||||||
import { User } from '../entities/User';
|
import { User } from '../entities/User';
|
||||||
import { UserGroup } from '../entities/UserGroup';
|
import { UserGroup } from '../entities/UserGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new User entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateUser {
|
export class CreateUser {
|
||||||
/**
|
/**
|
||||||
* The new user's first name.
|
* The new user's first name.
|
||||||
@ -17,7 +20,6 @@ export class CreateUser {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The new user's middle name.
|
* The new user's middle name.
|
||||||
* Optinal.
|
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -48,7 +50,7 @@ export class CreateUser {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The new user's phone number.
|
* The new user's phone number.
|
||||||
* Optional
|
* This will be validated against the configured country phone numer syntax (default: international).
|
||||||
*/
|
*/
|
||||||
@IsPhoneNumber(config.phone_validation_countrycode)
|
@IsPhoneNumber(config.phone_validation_countrycode)
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -64,15 +66,14 @@ export class CreateUser {
|
|||||||
/**
|
/**
|
||||||
* The new user's groups' id(s).
|
* The new user's groups' id(s).
|
||||||
* You can provide either one groupId or an array of groupIDs.
|
* You can provide either one groupId or an array of groupIDs.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
groupId?: number[] | number
|
groups?: number[] | number
|
||||||
|
|
||||||
//TODO: ProfilePics
|
//TODO: ProfilePics
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts this to a User Entity.
|
* Converts this to a User entity.
|
||||||
*/
|
*/
|
||||||
public async toUser(): Promise<User> {
|
public async toUser(): Promise<User> {
|
||||||
let newUser: User = new User();
|
let newUser: User = new User();
|
||||||
@ -81,30 +82,6 @@ export class CreateUser {
|
|||||||
throw new UsernameOrEmailNeededError();
|
throw new UsernameOrEmailNeededError();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.groupId) {
|
|
||||||
if (!Array.isArray(this.groupId)) {
|
|
||||||
this.groupId = [this.groupId]
|
|
||||||
}
|
|
||||||
const groupIDs: number[] = this.groupId
|
|
||||||
let errors = 0
|
|
||||||
const validateusergroups = async () => {
|
|
||||||
let foundgroups = []
|
|
||||||
for (const g of groupIDs) {
|
|
||||||
const found = await getConnectionManager().get().getRepository(UserGroup).find({ id: g });
|
|
||||||
if (found.length === 0) {
|
|
||||||
errors++
|
|
||||||
} else {
|
|
||||||
foundgroups.push(found[0])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
newUser.groups = foundgroups
|
|
||||||
}
|
|
||||||
await validateusergroups()
|
|
||||||
if (errors !== 0) {
|
|
||||||
throw new UserGroupNotFoundError();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
newUser.email = this.email
|
newUser.email = this.email
|
||||||
newUser.username = this.username
|
newUser.username = this.username
|
||||||
newUser.firstname = this.firstname
|
newUser.firstname = this.firstname
|
||||||
@ -113,8 +90,26 @@ export class CreateUser {
|
|||||||
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 argon2.hash(this.password + newUser.uuid);
|
||||||
|
newUser.groups = await this.getGroups();
|
||||||
//TODO: ProfilePics
|
//TODO: ProfilePics
|
||||||
|
|
||||||
return newUser;
|
return newUser;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get's all groups for this user by their id's;
|
||||||
|
*/
|
||||||
|
public async getGroups() {
|
||||||
|
if (!this.groups) { return null; }
|
||||||
|
let groups = new Array<UserGroup>();
|
||||||
|
if (!Array.isArray(this.groups)) {
|
||||||
|
this.groups = [this.groups]
|
||||||
|
}
|
||||||
|
for (let group of this.groups) {
|
||||||
|
let found = await getConnectionManager().get().getRepository(UserGroup).findOne({ id: group });
|
||||||
|
if (!found) { throw new UserGroupNotFoundError(); }
|
||||||
|
groups.push(found);
|
||||||
|
}
|
||||||
|
return groups;
|
||||||
|
}
|
||||||
}
|
}
|
@ -1,6 +1,9 @@
|
|||||||
import { IsOptional, IsString } from 'class-validator';
|
import { IsOptional, IsString } from 'class-validator';
|
||||||
import { UserGroup } from '../entities/UserGroup';
|
import { UserGroup } from '../entities/UserGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This classed is used to create a new UserGroup entity from a json body (post request).
|
||||||
|
*/
|
||||||
export class CreateUserGroup {
|
export class CreateUserGroup {
|
||||||
/**
|
/**
|
||||||
* The new group's name.
|
* The new group's name.
|
||||||
@ -17,7 +20,7 @@ export class CreateUserGroup {
|
|||||||
description?: string;
|
description?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts this to a UserGroup entity.
|
* Creates a new UserGroup entity from this.
|
||||||
*/
|
*/
|
||||||
public async toUserGroup(): Promise<UserGroup> {
|
public async toUserGroup(): Promise<UserGroup> {
|
||||||
let newUserGroup: UserGroup = new UserGroup();
|
let newUserGroup: UserGroup = new UserGroup();
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { IsString } from 'class-validator';
|
import { IsOptional, IsString } from 'class-validator';
|
||||||
import * as jsonwebtoken from 'jsonwebtoken';
|
import * as jsonwebtoken from 'jsonwebtoken';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
@ -6,10 +6,23 @@ import { IllegalJWTError, JwtNotProvidedError, RefreshTokenCountInvalidError, Us
|
|||||||
import { User } from '../entities/User';
|
import { User } from '../entities/User';
|
||||||
import { Logout } from '../responses/ResponseLogout';
|
import { Logout } from '../responses/ResponseLogout';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class handels a user logging out of the system.
|
||||||
|
* Of course it check's the user's provided credential (token) before logging him out.
|
||||||
|
*/
|
||||||
export class HandleLogout {
|
export class HandleLogout {
|
||||||
|
/**
|
||||||
|
* A stringyfied jwt access token.
|
||||||
|
* Will get checked for validity.
|
||||||
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
token: string;
|
@IsOptional()
|
||||||
|
token?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logs the user out.
|
||||||
|
* This gets achived by increasing the user's refresh token count, thereby invalidateing all currently existing jwts for that user.
|
||||||
|
*/
|
||||||
public async logout(): Promise<Logout> {
|
public async logout(): Promise<Logout> {
|
||||||
let logout: Logout = new Logout();
|
let logout: Logout = new Logout();
|
||||||
if (!this.token || this.token === undefined) {
|
if (!this.token || this.token === undefined) {
|
||||||
@ -22,11 +35,11 @@ export class HandleLogout {
|
|||||||
throw new IllegalJWTError()
|
throw new IllegalJWTError()
|
||||||
}
|
}
|
||||||
logout.timestamp = Math.floor(Date.now() / 1000)
|
logout.timestamp = Math.floor(Date.now() / 1000)
|
||||||
let found_user: User = await getConnectionManager().get().getRepository(User).findOne({ id: decoded["userid"] });
|
let found_user: User = await getConnectionManager().get().getRepository(User).findOne({ id: decoded["id"] });
|
||||||
if (!found_user) {
|
if (!found_user) {
|
||||||
throw new UserNotFoundError()
|
throw new UserNotFoundError()
|
||||||
}
|
}
|
||||||
if (found_user.refreshTokenCount !== decoded["refreshtokencount"]) {
|
if (found_user.refreshTokenCount !== decoded["refreshTokenCount"]) {
|
||||||
throw new RefreshTokenCountInvalidError()
|
throw new RefreshTokenCountInvalidError()
|
||||||
}
|
}
|
||||||
found_user.refreshTokenCount++;
|
found_user.refreshTokenCount++;
|
||||||
|
97
src/models/actions/ImportRunner.ts
Normal file
97
src/models/actions/ImportRunner.ts
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
import { IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
||||||
|
import { getConnectionManager } from 'typeorm';
|
||||||
|
import { RunnerGroupNeededError } from '../../errors/RunnerErrors';
|
||||||
|
import { RunnerOrganisationNotFoundError } from '../../errors/RunnerOrganisationErrors';
|
||||||
|
import { RunnerGroup } from '../entities/RunnerGroup';
|
||||||
|
import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
||||||
|
import { RunnerTeam } from '../entities/RunnerTeam';
|
||||||
|
import { CreateRunner } from './CreateRunner';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Special class used to import runners from csv files - or json arrays created from csv to be exact.
|
||||||
|
* Why you ask? Because the past has shown us that a non excel/csv based workflow is too much for most schools.
|
||||||
|
*/
|
||||||
|
export class ImportRunner {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new runner's first name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsNotEmpty()
|
||||||
|
firstname: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new runner's middle name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
middlename?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new runner's last name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsNotEmpty()
|
||||||
|
lastname: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new runner's team's name (if not provided otherwise).
|
||||||
|
* The team will automaticly get generated if it doesn't exist in this org yet.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
team?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Just an alias for team, because this is usually only used for importing data from schools.
|
||||||
|
*/
|
||||||
|
@IsOptional()
|
||||||
|
@IsString()
|
||||||
|
public set class(value: string) {
|
||||||
|
this.team = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a CreateRunner object based on this.
|
||||||
|
* @param groupID Either the id of the new runner's group or the id of the org that the new runner's team is a part of.
|
||||||
|
*/
|
||||||
|
public async toCreateRunner(groupID: number): Promise<CreateRunner> {
|
||||||
|
let newRunner: CreateRunner = new CreateRunner();
|
||||||
|
|
||||||
|
newRunner.firstname = this.firstname;
|
||||||
|
newRunner.middlename = this.middlename;
|
||||||
|
newRunner.lastname = this.lastname;
|
||||||
|
newRunner.group = (await this.getGroup(groupID)).id;
|
||||||
|
|
||||||
|
return newRunner;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get's the new runners group.
|
||||||
|
* @param groupID Either the id of the new runner's group or the id of the org that the new runner's team is a part of.
|
||||||
|
*/
|
||||||
|
public async getGroup(groupID: number): Promise<RunnerGroup> {
|
||||||
|
if (this.team === undefined && groupID === undefined) {
|
||||||
|
throw new RunnerGroupNeededError();
|
||||||
|
}
|
||||||
|
|
||||||
|
let team = await getConnectionManager().get().getRepository(RunnerTeam).findOne({ id: groupID });
|
||||||
|
if (team) { return team; }
|
||||||
|
|
||||||
|
let org = await getConnectionManager().get().getRepository(RunnerOrganisation).findOne({ id: groupID });
|
||||||
|
if (!org) {
|
||||||
|
throw new RunnerOrganisationNotFoundError();
|
||||||
|
}
|
||||||
|
if (this.team === undefined) { return org; }
|
||||||
|
|
||||||
|
team = await getConnectionManager().get().getRepository(RunnerTeam).findOne({ name: this.team, parentGroup: org });
|
||||||
|
if (!team) {
|
||||||
|
let newRunnerTeam: RunnerTeam = new RunnerTeam();
|
||||||
|
newRunnerTeam.name = this.team;
|
||||||
|
newRunnerTeam.parentGroup = org;
|
||||||
|
team = await getConnectionManager().get().getRepository(RunnerTeam).save(newRunnerTeam);
|
||||||
|
}
|
||||||
|
|
||||||
|
return team;
|
||||||
|
}
|
||||||
|
}
|
@ -1,15 +1,29 @@
|
|||||||
import { IsString } from 'class-validator';
|
import { IsOptional, IsString } from 'class-validator';
|
||||||
import * as jsonwebtoken from 'jsonwebtoken';
|
import * as jsonwebtoken from 'jsonwebtoken';
|
||||||
import { getConnectionManager } from 'typeorm';
|
import { getConnectionManager } from 'typeorm';
|
||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
import { IllegalJWTError, JwtNotProvidedError, RefreshTokenCountInvalidError, UserNotFoundError } from '../../errors/AuthError';
|
import { IllegalJWTError, JwtNotProvidedError, RefreshTokenCountInvalidError, UserNotFoundError } from '../../errors/AuthError';
|
||||||
|
import { JwtCreator } from "../../jwtcreator";
|
||||||
import { User } from '../entities/User';
|
import { User } from '../entities/User';
|
||||||
import { Auth } from '../responses/ResponseAuth';
|
import { Auth } from '../responses/ResponseAuth';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to create refreshed auth credentials.
|
||||||
|
* To be a little bit more exact: Is takes in a refresh token and creates a new access and refresh token for it's user.
|
||||||
|
* It of course checks for user existance, jwt validity and so on.
|
||||||
|
*/
|
||||||
export class RefreshAuth {
|
export class RefreshAuth {
|
||||||
|
/**
|
||||||
|
* A stringyfied jwt refresh token.
|
||||||
|
* Will get checked for validity.
|
||||||
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
token: string;
|
@IsOptional()
|
||||||
|
token?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new auth object based on this.
|
||||||
|
*/
|
||||||
public async toAuth(): Promise<Auth> {
|
public async toAuth(): Promise<Auth> {
|
||||||
let newAuth: Auth = new Auth();
|
let newAuth: Auth = new Auth();
|
||||||
if (!this.token || this.token === undefined) {
|
if (!this.token || this.token === undefined) {
|
||||||
@ -21,31 +35,21 @@ export class RefreshAuth {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new IllegalJWTError()
|
throw new IllegalJWTError()
|
||||||
}
|
}
|
||||||
const found_user = await getConnectionManager().get().getRepository(User).findOne({ id: decoded["userid"] }, { relations: ['groups', 'permissions'] });
|
const found_user = await getConnectionManager().get().getRepository(User).findOne({ id: decoded["id"] }, { relations: ['groups', 'permissions', 'groups.permissions'] });
|
||||||
if (!found_user) {
|
if (!found_user) {
|
||||||
throw new UserNotFoundError()
|
throw new UserNotFoundError()
|
||||||
}
|
}
|
||||||
if (found_user.refreshTokenCount !== decoded["refreshtokencount"]) {
|
if (found_user.refreshTokenCount !== decoded["refreshTokenCount"]) {
|
||||||
throw new RefreshTokenCountInvalidError()
|
throw new RefreshTokenCountInvalidError()
|
||||||
}
|
}
|
||||||
found_user.permissions = found_user.permissions || []
|
//Create the auth token
|
||||||
delete found_user.password;
|
|
||||||
const timestamp_accesstoken_expiry = Math.floor(Date.now() / 1000) + 5 * 60
|
const timestamp_accesstoken_expiry = Math.floor(Date.now() / 1000) + 5 * 60
|
||||||
delete found_user.password;
|
newAuth.access_token = JwtCreator.createAccess(found_user, timestamp_accesstoken_expiry);
|
||||||
newAuth.access_token = jsonwebtoken.sign({
|
|
||||||
userdetails: found_user,
|
|
||||||
exp: timestamp_accesstoken_expiry
|
|
||||||
}, config.jwt_secret)
|
|
||||||
newAuth.access_token_expires_at = timestamp_accesstoken_expiry
|
newAuth.access_token_expires_at = timestamp_accesstoken_expiry
|
||||||
//
|
//Create the refresh token
|
||||||
const timestamp_refresh_expiry = Math.floor(Date.now() / 1000) + 10 * 36000
|
const timestamp_refresh_expiry = Math.floor(Date.now() / 1000) + 10 * 36000
|
||||||
newAuth.refresh_token = jsonwebtoken.sign({
|
newAuth.refresh_token = JwtCreator.createRefresh(found_user, timestamp_refresh_expiry);
|
||||||
refreshtokencount: found_user.refreshTokenCount,
|
newAuth.refresh_token_expires_at = timestamp_refresh_expiry;
|
||||||
userid: found_user.id,
|
|
||||||
exp: timestamp_refresh_expiry
|
|
||||||
}, config.jwt_secret)
|
|
||||||
newAuth.refresh_token_expires_at = timestamp_refresh_expiry
|
|
||||||
|
|
||||||
return newAuth;
|
return newAuth;
|
||||||
}
|
}
|
||||||
}
|
}
|
68
src/models/actions/UpdatePermission.ts
Normal file
68
src/models/actions/UpdatePermission.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import { IsInt, IsNotEmpty, IsObject } from 'class-validator';
|
||||||
|
import { getConnectionManager } from 'typeorm';
|
||||||
|
import { PermissionNeedsPrincipalError } from '../../errors/PermissionErrors';
|
||||||
|
import { PrincipalNotFoundError, PrincipalWrongTypeError } from '../../errors/PrincipalErrors';
|
||||||
|
import { Permission } from '../entities/Permission';
|
||||||
|
import { Principal } from '../entities/Principal';
|
||||||
|
import { PermissionAction } from '../enums/PermissionAction';
|
||||||
|
import { PermissionTarget } from '../enums/PermissionTargets';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to update a Permission entity (via put request).
|
||||||
|
*/
|
||||||
|
export class UpdatePermission {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated permission's id.
|
||||||
|
* This shouldn't have changed but it is here in case anyone ever wants to enable id changes (whyever they would want to).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
id: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated permissions's principal.
|
||||||
|
* Just has to contain the principal's id -everything else won't be checked or changed.
|
||||||
|
*/
|
||||||
|
@IsObject()
|
||||||
|
@IsNotEmpty()
|
||||||
|
principal: Principal;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The permissions's target.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
target: PermissionTarget;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The permissions's action.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
action: PermissionAction;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates a provided Permission entity based on this.
|
||||||
|
*/
|
||||||
|
public async updatePermission(permission: Permission): Promise<Permission> {
|
||||||
|
permission.principal = await this.getPrincipal();
|
||||||
|
permission.target = this.target;
|
||||||
|
permission.action = this.action;
|
||||||
|
|
||||||
|
return permission;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the updated permission's principal based on it's id.
|
||||||
|
*/
|
||||||
|
public async getPrincipal(): Promise<Principal> {
|
||||||
|
if (this.principal === undefined || this.principal === null) {
|
||||||
|
throw new PermissionNeedsPrincipalError();
|
||||||
|
}
|
||||||
|
if (!isNaN(this.principal.id)) {
|
||||||
|
let principal = await getConnectionManager().get().getRepository(Principal).findOne({ id: this.principal.id });
|
||||||
|
if (!principal) { throw new PrincipalNotFoundError(); }
|
||||||
|
return principal;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new PrincipalWrongTypeError();
|
||||||
|
}
|
||||||
|
}
|
@ -7,43 +7,45 @@ import { Runner } from '../entities/Runner';
|
|||||||
import { RunnerGroup } from '../entities/RunnerGroup';
|
import { RunnerGroup } from '../entities/RunnerGroup';
|
||||||
import { CreateParticipant } from './CreateParticipant';
|
import { CreateParticipant } from './CreateParticipant';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to update a Runner entity (via put request).
|
||||||
|
*/
|
||||||
export class UpdateRunner extends CreateParticipant {
|
export class UpdateRunner extends CreateParticipant {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The updated runner's id.
|
* The updated runner's id.
|
||||||
|
* This shouldn't have changed but it is here in case anyone ever wants to enable id changes (whyever they would want to).
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
id: number;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The updated runner's new team/org.
|
* The updated runner's new team/org.
|
||||||
|
* Just has to contain the group's id -everything else won't be checked or changed.
|
||||||
*/
|
*/
|
||||||
@IsObject()
|
@IsObject()
|
||||||
group: RunnerGroup;
|
group: RunnerGroup;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a Runner entity from this.
|
* Updates a provided Runner entity based on this.
|
||||||
*/
|
*/
|
||||||
public async toRunner(): Promise<Runner> {
|
public async updateRunner(runner: Runner): Promise<Runner> {
|
||||||
let newRunner: Runner = new Runner();
|
runner.firstname = this.firstname;
|
||||||
|
runner.middlename = this.middlename;
|
||||||
|
runner.lastname = this.lastname;
|
||||||
|
runner.phone = this.phone;
|
||||||
|
runner.email = this.email;
|
||||||
|
runner.group = await this.getGroup();
|
||||||
|
runner.address = await this.getAddress();
|
||||||
|
|
||||||
newRunner.id = this.id;
|
return runner;
|
||||||
newRunner.firstname = this.firstname;
|
|
||||||
newRunner.middlename = this.middlename;
|
|
||||||
newRunner.lastname = this.lastname;
|
|
||||||
newRunner.phone = this.phone;
|
|
||||||
newRunner.email = this.email;
|
|
||||||
newRunner.group = await this.getGroup();
|
|
||||||
newRunner.address = await this.getAddress();
|
|
||||||
|
|
||||||
return newRunner;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages all the different ways a group can be provided.
|
* Loads the updated runner's group based on it's id.
|
||||||
*/
|
*/
|
||||||
public async getGroup(): Promise<RunnerGroup> {
|
public async getGroup(): Promise<RunnerGroup> {
|
||||||
if (this.group === undefined) {
|
if (this.group === undefined || this.group === null) {
|
||||||
throw new RunnerTeamNeedsParentError();
|
throw new RunnerTeamNeedsParentError();
|
||||||
}
|
}
|
||||||
if (!isNaN(this.group.id)) {
|
if (!isNaN(this.group.id)) {
|
||||||
|
52
src/models/actions/UpdateRunnerOrganisation.ts
Normal file
52
src/models/actions/UpdateRunnerOrganisation.ts
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import { IsInt, IsOptional } from 'class-validator';
|
||||||
|
import { getConnectionManager } from 'typeorm';
|
||||||
|
import { AddressNotFoundError } from '../../errors/AddressErrors';
|
||||||
|
import { Address } from '../entities/Address';
|
||||||
|
import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
||||||
|
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to update a RunnerOrganisation entity (via put request).
|
||||||
|
*/
|
||||||
|
export class UpdateRunnerOrganisation extends CreateRunnerGroup {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated orgs's id.
|
||||||
|
* This shouldn't have changed but it is here in case anyone ever wants to enable id changes (whyever they would want to).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
id: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated organisation's address.
|
||||||
|
* Just has to contain the address's id - everything else won't be checked or changed.
|
||||||
|
* Optional.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
@IsOptional()
|
||||||
|
address?: Address;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the organisation's address based on it's id.
|
||||||
|
*/
|
||||||
|
public async getAddress(): Promise<Address> {
|
||||||
|
if (this.address === undefined || this.address === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
let address = await getConnectionManager().get().getRepository(Address).findOne({ id: this.address.id });
|
||||||
|
if (!address) { throw new AddressNotFoundError; }
|
||||||
|
return address;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates a provided RunnerOrganisation entity based on this.
|
||||||
|
*/
|
||||||
|
public async updateRunnerOrganisation(organisation: RunnerOrganisation): Promise<RunnerOrganisation> {
|
||||||
|
|
||||||
|
organisation.name = this.name;
|
||||||
|
organisation.contact = await this.getContact();
|
||||||
|
organisation.address = await this.getAddress();
|
||||||
|
|
||||||
|
return organisation;
|
||||||
|
}
|
||||||
|
}
|
@ -6,23 +6,31 @@ import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
|||||||
import { RunnerTeam } from '../entities/RunnerTeam';
|
import { RunnerTeam } from '../entities/RunnerTeam';
|
||||||
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
import { CreateRunnerGroup } from './CreateRunnerGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to update a RunnerTeam entity (via put request).
|
||||||
|
*/
|
||||||
export class UpdateRunnerTeam extends CreateRunnerGroup {
|
export class UpdateRunnerTeam extends CreateRunnerGroup {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The updated team's id.
|
* The updated team's id.
|
||||||
|
* This shouldn't have changed but it is here in case anyone ever wants to enable id changes (whyever they would want to).
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
id: number;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The team's parent group (organisation).
|
* The updated team's parentGroup.
|
||||||
|
* Just has to contain the organisation's id - everything else won't be checked or changed.
|
||||||
*/
|
*/
|
||||||
@IsObject()
|
@IsObject()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
parentGroup: RunnerOrganisation;
|
parentGroup: RunnerOrganisation;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the updated teams's parentGroup based on it's id.
|
||||||
|
*/
|
||||||
public async getParent(): Promise<RunnerOrganisation> {
|
public async getParent(): Promise<RunnerOrganisation> {
|
||||||
if (this.parentGroup === undefined) {
|
if (this.parentGroup === undefined || this.parentGroup === null) {
|
||||||
throw new RunnerTeamNeedsParentError();
|
throw new RunnerTeamNeedsParentError();
|
||||||
}
|
}
|
||||||
if (!isNaN(this.parentGroup.id)) {
|
if (!isNaN(this.parentGroup.id)) {
|
||||||
@ -35,16 +43,14 @@ export class UpdateRunnerTeam extends CreateRunnerGroup {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a RunnerTeam entity from this.
|
* Updates a provided RunnerTeam entity based on this.
|
||||||
*/
|
*/
|
||||||
public async toRunnerTeam(): Promise<RunnerTeam> {
|
public async updateRunnerTeam(team: RunnerTeam): Promise<RunnerTeam> {
|
||||||
let newRunnerTeam: RunnerTeam = new RunnerTeam();
|
|
||||||
|
|
||||||
newRunnerTeam.id = this.id;
|
team.name = this.name;
|
||||||
newRunnerTeam.name = this.name;
|
team.parentGroup = await this.getParent();
|
||||||
newRunnerTeam.parentGroup = await this.getParent();
|
team.contact = await this.getContact()
|
||||||
newRunnerTeam.contact = await this.getContact()
|
|
||||||
|
|
||||||
return newRunnerTeam;
|
return team;
|
||||||
}
|
}
|
||||||
}
|
}
|
130
src/models/actions/UpdateUser.ts
Normal file
130
src/models/actions/UpdateUser.ts
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
import * as argon2 from "argon2";
|
||||||
|
import { IsBoolean, IsEmail, IsInt, IsOptional, IsPhoneNumber, IsString } from 'class-validator';
|
||||||
|
import { getConnectionManager } from 'typeorm';
|
||||||
|
import { config } from '../../config';
|
||||||
|
import { UsernameOrEmailNeededError } from '../../errors/AuthError';
|
||||||
|
import { UserGroupNotFoundError } from '../../errors/UserGroupErrors';
|
||||||
|
import { User } from '../entities/User';
|
||||||
|
import { UserGroup } from '../entities/UserGroup';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is used to update a User entity (via put request).
|
||||||
|
*/
|
||||||
|
export class UpdateUser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's id.
|
||||||
|
* This shouldn't have changed but it is here in case anyone ever wants to enable id changes (whyever they would want to).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
id: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's first name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
firstname: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's middle name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
middlename?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's last name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
lastname: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's username.
|
||||||
|
* You have to provide at least one of: {email, username}.
|
||||||
|
*/
|
||||||
|
@IsOptional()
|
||||||
|
@IsString()
|
||||||
|
username?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's email address.
|
||||||
|
* You have to provide at least one of: {email, username}.
|
||||||
|
*/
|
||||||
|
@IsEmail()
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
email?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's phone number.
|
||||||
|
* This will be validated against the configured country phone numer syntax (default: international).
|
||||||
|
*/
|
||||||
|
@IsPhoneNumber(config.phone_validation_countrycode)
|
||||||
|
@IsOptional()
|
||||||
|
phone?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The new updated's password.
|
||||||
|
* Only provide it if you want it updated.
|
||||||
|
* Changeing the password will invalidate all of the user's jwts.
|
||||||
|
* This will of course not be saved in plaintext :)
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
password?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should the user be enabled?
|
||||||
|
*/
|
||||||
|
@IsBoolean()
|
||||||
|
enabled: boolean = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The updated user's groups.
|
||||||
|
* This just has to contain the group's id - everything else won't be changed.
|
||||||
|
*/
|
||||||
|
@IsOptional()
|
||||||
|
groups?: UserGroup[]
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates a provided User entity based on this.
|
||||||
|
*/
|
||||||
|
public async updateUser(user: User): Promise<User> {
|
||||||
|
user.email = this.email;
|
||||||
|
user.username = this.username;
|
||||||
|
if ((user.email === undefined || user.email === null) && (user.username === undefined || user.username === null)) {
|
||||||
|
throw new UsernameOrEmailNeededError();
|
||||||
|
}
|
||||||
|
if (this.password) {
|
||||||
|
user.password = await argon2.hash(this.password + user.uuid);
|
||||||
|
user.refreshTokenCount = user.refreshTokenCount + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
user.enabled = this.enabled;
|
||||||
|
user.firstname = this.firstname
|
||||||
|
user.middlename = this.middlename
|
||||||
|
user.lastname = this.lastname
|
||||||
|
user.phone = this.phone;
|
||||||
|
user.groups = await this.getGroups();
|
||||||
|
//TODO: ProfilePics
|
||||||
|
|
||||||
|
return user;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the updated user's groups based on their ids.
|
||||||
|
*/
|
||||||
|
public async getGroups() {
|
||||||
|
if (!this.groups) { return null; }
|
||||||
|
let groups = new Array<UserGroup>();
|
||||||
|
if (!Array.isArray(this.groups)) {
|
||||||
|
this.groups = [this.groups]
|
||||||
|
}
|
||||||
|
for (let group of this.groups) {
|
||||||
|
let found = await getConnectionManager().get().getRepository(UserGroup).findOne({ id: group.id });
|
||||||
|
if (!found) { throw new UserGroupNotFoundError(); }
|
||||||
|
groups.push(found);
|
||||||
|
}
|
||||||
|
return groups;
|
||||||
|
}
|
||||||
|
}
|
@ -10,7 +10,8 @@ import { Participant } from "./Participant";
|
|||||||
import { RunnerOrganisation } from "./RunnerOrganisation";
|
import { RunnerOrganisation } from "./RunnerOrganisation";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a address (to be used for contact information).
|
* Defines the Address entity.
|
||||||
|
* Implemented this way to prevent any formatting differences.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export class Address {
|
export class Address {
|
||||||
@ -23,6 +24,7 @@ export class Address {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's description.
|
* The address's description.
|
||||||
|
* Optional and mostly for UX.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -49,6 +51,8 @@ export class Address {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The address's postal code.
|
* The address's postal code.
|
||||||
|
* This will get checked against the postal code syntax for the configured country.
|
||||||
|
* TODO: Implement the config option.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsString()
|
@IsString()
|
||||||
|
@ -4,19 +4,21 @@ import { Donation } from "./Donation";
|
|||||||
import { Runner } from "./Runner";
|
import { Runner } from "./Runner";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a distance based donation.
|
* Defines the DistanceDonation entity.
|
||||||
* Here people donate a certain amout per kilometer
|
* For distanceDonations a donor pledges to donate a certain amount for each kilometer ran by a runner.
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class DistanceDonation extends Donation {
|
export class DistanceDonation extends Donation {
|
||||||
/**
|
/**
|
||||||
* The runner associated.
|
* The donation's associated runner.
|
||||||
|
* Used as the source of the donation's distance.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => Runner, runner => runner.distanceDonations)
|
@ManyToOne(() => Runner, runner => runner.distanceDonations)
|
||||||
runner: Runner;
|
runner: Runner;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* The donation's amount donated per distance.
|
||||||
* The amount the donor set to be donated per kilometer that the runner ran.
|
* The amount the donor set to be donated per kilometer that the runner ran.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@ -26,12 +28,12 @@ export class DistanceDonation extends Donation {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The donation's amount in cents (or whatever your currency's smallest unit is.).
|
* The donation's amount in cents (or whatever your currency's smallest unit is.).
|
||||||
* The exact implementation may differ for each type of donation.
|
* Get's calculated from the runner's distance ran and the amount donated per kilometer.
|
||||||
*/
|
*/
|
||||||
public get amount(): number {
|
public get amount(): number {
|
||||||
let calculatedAmount = -1;
|
let calculatedAmount = -1;
|
||||||
try {
|
try {
|
||||||
calculatedAmount = this.amountPerDistance * this.runner.distance;
|
calculatedAmount = this.amountPerDistance * (this.runner.distance / 1000);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,9 @@ import { Entity, ManyToOne, PrimaryGeneratedColumn, TableInheritance } from "typ
|
|||||||
import { Participant } from "./Participant";
|
import { Participant } from "./Participant";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the donation interface.
|
* Defines the Donation entity.
|
||||||
|
* A donation just associates a donor with a donation amount.
|
||||||
|
* The specifics of the amoun's determination has to be implemented in child classes.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
||||||
|
@ -3,13 +3,13 @@ import { ChildEntity, Column } from "typeorm";
|
|||||||
import { Participant } from "./Participant";
|
import { Participant } from "./Participant";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a donor.
|
* Defines the Donor entity.
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class Donor extends Participant {
|
export class Donor extends Participant {
|
||||||
/**
|
/**
|
||||||
* Does this donor need a receipt?.
|
* Does this donor need a receipt?
|
||||||
* Default: false
|
* Will later be used to automaticly generate donation receipts.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsBoolean()
|
@IsBoolean()
|
||||||
|
@ -3,7 +3,8 @@ import { ChildEntity, Column } from "typeorm";
|
|||||||
import { Donation } from "./Donation";
|
import { Donation } from "./Donation";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a fixed donation.
|
* Defines the FixedDonation entity.
|
||||||
|
* In the past there was no easy way to track fixed donations (eg. for creating donation receipts).
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class FixedDonation extends Donation {
|
export class FixedDonation extends Donation {
|
||||||
|
@ -13,13 +13,14 @@ import { Address } from "./Address";
|
|||||||
import { RunnerGroup } from "./RunnerGroup";
|
import { RunnerGroup } from "./RunnerGroup";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a group's contact.
|
* Defines the GroupContact entity.
|
||||||
|
* Mainly it's own class to reduce duplicate code and enable contact's to be associated with multiple groups.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export class GroupContact {
|
export class GroupContact {
|
||||||
/**
|
/**
|
||||||
* Autogenerated unique id (primary key).
|
* Autogenerated unique id (primary key).
|
||||||
*/
|
*/
|
||||||
@PrimaryGeneratedColumn()
|
@PrimaryGeneratedColumn()
|
||||||
@IsInt()
|
@IsInt()
|
||||||
id: number;
|
id: number;
|
||||||
@ -34,7 +35,6 @@ export class GroupContact {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's middle name.
|
* The contact's middle name.
|
||||||
* Optional
|
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -51,7 +51,7 @@ export class GroupContact {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's address.
|
* The contact's address.
|
||||||
* Optional
|
* This is a address object to prevent any formatting differences.
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ManyToOne(() => Address, address => address.participants, { nullable: true })
|
@ManyToOne(() => Address, address => address.participants, { nullable: true })
|
||||||
@ -59,7 +59,7 @@ export class GroupContact {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's phone number.
|
* The contact's phone number.
|
||||||
* Optional
|
* This will be validated against the configured country phone numer syntax (default: international).
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -68,7 +68,7 @@ export class GroupContact {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The contact's email address.
|
* The contact's email address.
|
||||||
* Optional
|
* Could later be used to automaticly send mails concerning the contact's associated groups.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
|
@ -13,7 +13,8 @@ import { Address } from "./Address";
|
|||||||
import { Donation } from "./Donation";
|
import { Donation } from "./Donation";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the participant interface.
|
* Defines the Participant entity.
|
||||||
|
* Participans can donate and therefor be associated with donation entities.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
||||||
@ -35,7 +36,6 @@ export abstract class Participant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's middle name.
|
* The participant's middle name.
|
||||||
* Optional
|
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -52,14 +52,14 @@ export abstract class Participant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's address.
|
* The participant's address.
|
||||||
* Optional
|
* This is a address object to prevent any formatting differences.
|
||||||
*/
|
*/
|
||||||
@ManyToOne(() => Address, address => address.participants, { nullable: true })
|
@ManyToOne(() => Address, address => address.participants, { nullable: true })
|
||||||
address?: Address;
|
address?: Address;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's phone number.
|
* The participant's phone number.
|
||||||
* Optional
|
* This will be validated against the configured country phone numer syntax (default: international).
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -68,7 +68,7 @@ export abstract class Participant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's email address.
|
* The participant's email address.
|
||||||
* Optional
|
* Can be used to contact the participant.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -77,6 +77,7 @@ export abstract class Participant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link the participant as the donor of a donation.
|
* Used to link the participant as the donor of a donation.
|
||||||
|
* Attention: Only runner's can be associated as a distanceDonations distance source.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => Donation, donation => donation.donor, { nullable: true })
|
@OneToMany(() => Donation, donation => donation.donor, { nullable: true })
|
||||||
donations: Donation[];
|
donations: Donation[];
|
||||||
|
@ -1,17 +1,19 @@
|
|||||||
import {
|
import {
|
||||||
|
IsEnum,
|
||||||
IsInt,
|
IsInt,
|
||||||
IsNotEmpty,
|
IsNotEmpty
|
||||||
|
|
||||||
IsString
|
|
||||||
} from "class-validator";
|
} from "class-validator";
|
||||||
import { Column, Entity, OneToMany, PrimaryGeneratedColumn } from "typeorm";
|
import { Column, Entity, ManyToOne, PrimaryGeneratedColumn } from "typeorm";
|
||||||
import { User } from './User';
|
import { PermissionAction } from '../enums/PermissionAction';
|
||||||
import { UserGroup } from './UserGroup';
|
import { PermissionTarget } from '../enums/PermissionTargets';
|
||||||
|
import { Principal } from './Principal';
|
||||||
/**
|
/**
|
||||||
* Defines the Permission interface.
|
* Defines the Permission entity.
|
||||||
|
* Permissions can be granted to principals.
|
||||||
|
* The permissions possible targets and actions are defined in enums.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export abstract class Permission {
|
export class Permission {
|
||||||
/**
|
/**
|
||||||
* Autogenerated unique id (primary key).
|
* Autogenerated unique id (primary key).
|
||||||
*/
|
*/
|
||||||
@ -20,30 +22,33 @@ export abstract class Permission {
|
|||||||
id: number;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* users
|
* The permission's principal.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => User, user => user.permissions, { nullable: true })
|
@ManyToOne(() => Principal, principal => principal.permissions)
|
||||||
users: User[]
|
principal: Principal;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* groups
|
* The permission's target.
|
||||||
|
* This get's stored as the enum value's string representation for compatability reasons.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => UserGroup, group => group.permissions, { nullable: true })
|
@Column({ type: 'varchar' })
|
||||||
groups: UserGroup[]
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The target
|
|
||||||
*/
|
|
||||||
@Column()
|
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@IsString()
|
@IsEnum(PermissionTarget)
|
||||||
target: string;
|
target: PermissionTarget;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The action type
|
* The permission's action.
|
||||||
|
* This get's stored as the enum value's string representation for compatability reasons.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column({ type: 'varchar' })
|
||||||
@IsNotEmpty()
|
@IsEnum(PermissionAction)
|
||||||
@IsString()
|
action: PermissionAction;
|
||||||
action: string;
|
|
||||||
|
/**
|
||||||
|
* Turn this into a string for exporting and jwts.
|
||||||
|
* Mainly used to shrink the size of jwts (otherwise the would contain entire objects).
|
||||||
|
*/
|
||||||
|
public toString(): string {
|
||||||
|
return this.target + ":" + this.action;
|
||||||
|
}
|
||||||
}
|
}
|
30
src/models/entities/Principal.ts
Normal file
30
src/models/entities/Principal.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import { IsInt } from 'class-validator';
|
||||||
|
import { Entity, OneToMany, PrimaryGeneratedColumn, TableInheritance } from 'typeorm';
|
||||||
|
import { ResponsePrincipal } from '../responses/ResponsePrincipal';
|
||||||
|
import { Permission } from './Permission';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the principal entity.
|
||||||
|
* A principal basicly is any entity that can receive permissions for the api (users and their groups).
|
||||||
|
*/
|
||||||
|
@Entity()
|
||||||
|
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
||||||
|
export abstract class Principal {
|
||||||
|
/**
|
||||||
|
* Autogenerated unique id (primary key).
|
||||||
|
*/
|
||||||
|
@PrimaryGeneratedColumn()
|
||||||
|
@IsInt()
|
||||||
|
id: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The participant's permissions.
|
||||||
|
*/
|
||||||
|
@OneToMany(() => Permission, permission => permission.principal, { nullable: true })
|
||||||
|
permissions: Permission[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turns this entity into it's response class.
|
||||||
|
*/
|
||||||
|
public abstract toResponse(): ResponsePrincipal;
|
||||||
|
}
|
@ -7,44 +7,52 @@ import { RunnerGroup } from "./RunnerGroup";
|
|||||||
import { Scan } from "./Scan";
|
import { Scan } from "./Scan";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a runner.
|
* Defines the runner entity.
|
||||||
|
* Runners differ from participants in being able to actually accumulate a ran distance through scans.
|
||||||
|
* Runner's get organized in groups.
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class Runner extends Participant {
|
export class Runner extends Participant {
|
||||||
/**
|
/**
|
||||||
* The runner's associated group.
|
* The runner's associated group.
|
||||||
|
* Can be a runner team or organisation.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => RunnerGroup, group => group.runners, { nullable: false })
|
@ManyToOne(() => RunnerGroup, group => group.runners, { nullable: false })
|
||||||
group: RunnerGroup;
|
group: RunnerGroup;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link runners to donations.
|
* The runner's associated distanceDonations.
|
||||||
|
* Used to link runners to distanceDonations in order to calculate the donation's amount based on the distance the runner ran.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => DistanceDonation, distanceDonation => distanceDonation.runner, { nullable: true })
|
@OneToMany(() => DistanceDonation, distanceDonation => distanceDonation.runner, { nullable: true })
|
||||||
distanceDonations: DistanceDonation[];
|
distanceDonations: DistanceDonation[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link runners to cards.
|
* The runner's associated cards.
|
||||||
|
* Used to link runners to cards - yes a runner be associated with multiple cards this came in handy in the past.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => RunnerCard, card => card.runner, { nullable: true })
|
@OneToMany(() => RunnerCard, card => card.runner, { nullable: true })
|
||||||
cards: RunnerCard[];
|
cards: RunnerCard[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link runners to a scans
|
* The runner's associated scans.
|
||||||
|
* Used to link runners to scans (valid and fraudulant).
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => Scan, scan => scan.runner, { nullable: true })
|
@OneToMany(() => Scan, scan => scan.runner, { nullable: true })
|
||||||
scans: Scan[];
|
scans: Scan[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns all valid scans associated with this runner.
|
* Returns all valid scans associated with this runner.
|
||||||
|
* This is implemented here to avoid duplicate code in other files.
|
||||||
*/
|
*/
|
||||||
public get validScans(): Scan[] {
|
public get validScans(): Scan[] {
|
||||||
return this.scans.filter(scan => { scan.valid === true });
|
return this.scans.filter(scan => { scan.valid === true });
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the total distance ran by this runner.
|
* Returns the total distance ran by this runner based on all his valid scans.
|
||||||
|
* This is implemented here to avoid duplicate code in other files.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
public get distance(): number {
|
public get distance(): number {
|
||||||
|
@ -11,7 +11,9 @@ import { Runner } from "./Runner";
|
|||||||
import { TrackScan } from "./TrackScan";
|
import { TrackScan } from "./TrackScan";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a card that can be scanned via a scanner station.
|
* Defines the RunnerCard entity.
|
||||||
|
* A runnerCard is a physical representation for a runner.
|
||||||
|
* It can be associated with a runner to create scans via the scan station's.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export class RunnerCard {
|
export class RunnerCard {
|
||||||
@ -23,7 +25,8 @@ export class RunnerCard {
|
|||||||
id: number;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The runner that is currently associated with this card.
|
* The card's currently associated runner.
|
||||||
|
* To increase reusability a card can be reassigned.
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ManyToOne(() => Runner, runner => runner.cards, { nullable: true })
|
@ManyToOne(() => Runner, runner => runner.cards, { nullable: true })
|
||||||
@ -32,7 +35,7 @@ export class RunnerCard {
|
|||||||
/**
|
/**
|
||||||
* The card's code.
|
* The card's code.
|
||||||
* This has to be able to being converted to something barcode compatible.
|
* This has to be able to being converted to something barcode compatible.
|
||||||
* could theoretically be autogenerated
|
* Will get automaticlly generated (not implemented yet).
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsEAN()
|
@IsEAN()
|
||||||
@ -49,7 +52,8 @@ export class RunnerCard {
|
|||||||
enabled: boolean = true;
|
enabled: boolean = true;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link cards to a track scans.
|
* The card's associated scans.
|
||||||
|
* Used to link cards to track scans.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => TrackScan, scan => scan.track, { nullable: true })
|
@OneToMany(() => TrackScan, scan => scan.track, { nullable: true })
|
||||||
scans: TrackScan[];
|
scans: TrackScan[];
|
||||||
|
@ -9,7 +9,8 @@ import { GroupContact } from "./GroupContact";
|
|||||||
import { Runner } from "./Runner";
|
import { Runner } from "./Runner";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the runnerGroup interface.
|
* Defines the RunnerGroup entity.
|
||||||
|
* This is used to group runners together (as the name suggests).
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
||||||
@ -31,13 +32,14 @@ export abstract class RunnerGroup {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The group's contact.
|
* The group's contact.
|
||||||
* Optional
|
* This is mostly a feature for the group managers and public relations.
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ManyToOne(() => GroupContact, contact => contact.groups, { nullable: true })
|
@ManyToOne(() => GroupContact, contact => contact.groups, { nullable: true })
|
||||||
contact?: GroupContact;
|
contact?: GroupContact;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* The group's associated runners.
|
||||||
* Used to link runners to a runner group.
|
* Used to link runners to a runner group.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => Runner, runner => runner.group, { nullable: true })
|
@OneToMany(() => Runner, runner => runner.group, { nullable: true })
|
||||||
|
@ -5,22 +5,23 @@ import { RunnerGroup } from "./RunnerGroup";
|
|||||||
import { RunnerTeam } from "./RunnerTeam";
|
import { RunnerTeam } from "./RunnerTeam";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a runner organisation (business or school for example).
|
* Defines the RunnerOrganisation entity.
|
||||||
|
* This usually is a school, club or company.
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class RunnerOrganisation extends RunnerGroup {
|
export class RunnerOrganisation extends RunnerGroup {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The organisations's address.
|
* The organisations's address.
|
||||||
* Optional
|
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ManyToOne(() => Address, address => address.groups, { nullable: true })
|
@ManyToOne(() => Address, address => address.groups, { nullable: true })
|
||||||
address?: Address;
|
address?: Address;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link teams to runner groups.
|
* The organisation's teams.
|
||||||
*/
|
* Used to link teams to a organisation.
|
||||||
|
*/
|
||||||
@OneToMany(() => RunnerTeam, team => team.parentGroup, { nullable: true })
|
@OneToMany(() => RunnerTeam, team => team.parentGroup, { nullable: true })
|
||||||
teams: RunnerTeam[];
|
teams: RunnerTeam[];
|
||||||
}
|
}
|
@ -4,14 +4,15 @@ import { RunnerGroup } from "./RunnerGroup";
|
|||||||
import { RunnerOrganisation } from "./RunnerOrganisation";
|
import { RunnerOrganisation } from "./RunnerOrganisation";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a runner team (class or deparment for example).
|
* Defines the RunnerTeam entity.
|
||||||
|
* This usually is a school class or department in a company.
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class RunnerTeam extends RunnerGroup {
|
export class RunnerTeam extends RunnerGroup {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The team's parent group.
|
* The team's parent group.
|
||||||
* Optional
|
* Every team has to be part of a runnerOrganisation - this get's checked on creation and update.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => RunnerOrganisation, org => org.teams, { nullable: true })
|
@ManyToOne(() => RunnerOrganisation, org => org.teams, { nullable: true })
|
||||||
|
@ -9,7 +9,8 @@ import { Column, Entity, ManyToOne, PrimaryGeneratedColumn, TableInheritance } f
|
|||||||
import { Runner } from "./Runner";
|
import { Runner } from "./Runner";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the scan interface.
|
* Defines the Scan entity.
|
||||||
|
* A scan basicly adds a certain distance to a runner's total ran distance.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
@TableInheritance({ column: { name: "type", type: "varchar" } })
|
||||||
@ -22,7 +23,8 @@ export abstract class Scan {
|
|||||||
id: number;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The associated runner.
|
* The scan's associated runner.
|
||||||
|
* This is important to link ran distances to runners.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => Runner, runner => runner.scans, { nullable: false })
|
@ManyToOne(() => Runner, runner => runner.scans, { nullable: false })
|
||||||
@ -30,15 +32,17 @@ export abstract class Scan {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The scan's distance in meters.
|
* The scan's distance in meters.
|
||||||
|
* Can be set manually or derived from another object.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsPositive()
|
@IsPositive()
|
||||||
abstract distance: number;
|
abstract distance: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Is the scan valid (for fraud reasons).
|
* Is the scan valid (for fraud reasons).
|
||||||
* Default: true
|
* The determination of validity will work differently for every child class.
|
||||||
*/
|
* Default: true
|
||||||
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsBoolean()
|
@IsBoolean()
|
||||||
valid: boolean = true;
|
valid: boolean = true;
|
||||||
|
@ -10,7 +10,8 @@ import { Track } from "./Track";
|
|||||||
import { TrackScan } from "./TrackScan";
|
import { TrackScan } from "./TrackScan";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ScannerStations have the ability to create scans for specific tracks.
|
* Defines the ScanStation entity.
|
||||||
|
* ScanStations get used to create TrackScans for runners based on a scan of their runnerCard.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export class ScanStation {
|
export class ScanStation {
|
||||||
@ -23,6 +24,7 @@ export class ScanStation {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The station's description.
|
* The station's description.
|
||||||
|
* Mostly for better UX when traceing back stuff.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -31,6 +33,7 @@ export class ScanStation {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The track this station is associated with.
|
* The track this station is associated with.
|
||||||
|
* All scans created by this station will also be associated with this track.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => Track, track => track.stations, { nullable: false })
|
@ManyToOne(() => Track, track => track.stations, { nullable: false })
|
||||||
@ -38,6 +41,7 @@ export class ScanStation {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The station's api key.
|
* The station's api key.
|
||||||
|
* This is used to authorize a station against the api (not implemented yet).
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ -45,7 +49,7 @@ export class ScanStation {
|
|||||||
key: string;
|
key: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Is the station enabled (for fraud reasons)?
|
* Is the station enabled (for fraud and setup reasons)?
|
||||||
* Default: true
|
* Default: true
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import {
|
import {
|
||||||
IsInt,
|
IsInt,
|
||||||
IsNotEmpty,
|
IsNotEmpty,
|
||||||
|
|
||||||
IsPositive,
|
IsPositive,
|
||||||
IsString
|
IsString
|
||||||
} from "class-validator";
|
} from "class-validator";
|
||||||
@ -10,7 +9,7 @@ import { ScanStation } from "./ScanStation";
|
|||||||
import { TrackScan } from "./TrackScan";
|
import { TrackScan } from "./TrackScan";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a track of given length.
|
* Defines the Track entity.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export class Track {
|
export class Track {
|
||||||
@ -23,6 +22,7 @@ export class Track {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The track's name.
|
* The track's name.
|
||||||
|
* Mainly here for UX.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -31,6 +31,7 @@ export class Track {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The track's length/distance in meters.
|
* The track's length/distance in meters.
|
||||||
|
* Will be used to calculate runner's ran distances.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@ -38,13 +39,15 @@ export class Track {
|
|||||||
distance: number;
|
distance: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link scan stations to track.
|
* Used to link scan stations to a certain track.
|
||||||
|
* This makes the configuration of the scan stations easier.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => ScanStation, station => station.track, { nullable: true })
|
@OneToMany(() => ScanStation, station => station.track, { nullable: true })
|
||||||
stations: ScanStation[];
|
stations: ScanStation[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to link track scans to a track.
|
* Used to link track scans to a track.
|
||||||
|
* The scan will derive it's distance from the track's distance.
|
||||||
*/
|
*/
|
||||||
@OneToMany(() => TrackScan, scan => scan.track, { nullable: true })
|
@OneToMany(() => TrackScan, scan => scan.track, { nullable: true })
|
||||||
scans: TrackScan[];
|
scans: TrackScan[];
|
||||||
|
@ -12,26 +12,30 @@ import { ScanStation } from "./ScanStation";
|
|||||||
import { Track } from "./Track";
|
import { Track } from "./Track";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the scan interface.
|
* Defines the TrackScan entity.
|
||||||
|
* A track scan usaually get's generated by a scan station.
|
||||||
*/
|
*/
|
||||||
@ChildEntity()
|
@ChildEntity()
|
||||||
export class TrackScan extends Scan {
|
export class TrackScan extends Scan {
|
||||||
/**
|
/**
|
||||||
* The associated track.
|
* The scan's associated track.
|
||||||
|
* This is used to determine the scan's distance.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => Track, track => track.scans, { nullable: true })
|
@ManyToOne(() => Track, track => track.scans, { nullable: true })
|
||||||
track: Track;
|
track: Track;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The associated card.
|
* The runnerCard associated with the scan.
|
||||||
|
* This get's saved for documentation and management purposes.
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => RunnerCard, card => card.scans, { nullable: true })
|
@ManyToOne(() => RunnerCard, card => card.scans, { nullable: true })
|
||||||
card: RunnerCard;
|
card: RunnerCard;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The scanning station.
|
* The scanning station that created the scan.
|
||||||
|
* Mainly used for logging and traceing back scans (or errors)
|
||||||
*/
|
*/
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ManyToOne(() => ScanStation, station => station.scans, { nullable: true })
|
@ManyToOne(() => ScanStation, station => station.scans, { nullable: true })
|
||||||
@ -39,6 +43,7 @@ export class TrackScan extends Scan {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The scan's distance in meters.
|
* The scan's distance in meters.
|
||||||
|
* This just get's loaded from it's track.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsPositive()
|
@IsPositive()
|
||||||
@ -48,6 +53,7 @@ export class TrackScan extends Scan {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The scan's creation timestamp.
|
* The scan's creation timestamp.
|
||||||
|
* Will be used to implement fraud detection.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsDateString()
|
@IsDateString()
|
||||||
|
@ -1,38 +1,36 @@
|
|||||||
import { IsBoolean, IsEmail, IsInt, IsNotEmpty, IsOptional, IsPhoneNumber, IsString, IsUUID } from "class-validator";
|
import { IsBoolean, IsEmail, IsInt, IsNotEmpty, IsOptional, IsPhoneNumber, IsString, IsUUID } from "class-validator";
|
||||||
import { Column, Entity, JoinTable, ManyToMany, ManyToOne, OneToMany, PrimaryGeneratedColumn } from "typeorm";
|
import { ChildEntity, Column, JoinTable, ManyToMany, OneToMany } from "typeorm";
|
||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
import { Permission } from './Permission';
|
import { ResponsePrincipal } from '../responses/ResponsePrincipal';
|
||||||
|
import { ResponseUser } from '../responses/ResponseUser';
|
||||||
|
import { Principal } from './Principal';
|
||||||
import { UserAction } from './UserAction';
|
import { UserAction } from './UserAction';
|
||||||
import { UserGroup } from './UserGroup';
|
import { UserGroup } from './UserGroup';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a admin user.
|
* Defines the User entity.
|
||||||
|
* Users are the ones that can use the "admin" webui and do stuff in the backend.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@ChildEntity()
|
||||||
export class User {
|
export class User extends Principal {
|
||||||
/**
|
/**
|
||||||
* autogenerated unique id (primary key).
|
* The user's uuid.
|
||||||
*/
|
* Mainly gets used as a per-user salt for the password hash.
|
||||||
@PrimaryGeneratedColumn()
|
|
||||||
@IsInt()
|
|
||||||
id: number;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* uuid
|
|
||||||
*/
|
*/
|
||||||
@Column({ unique: true })
|
@Column({ unique: true })
|
||||||
@IsUUID(4)
|
@IsUUID(4)
|
||||||
uuid: string;
|
uuid: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* user email
|
* The user's e-mail address.
|
||||||
|
* Either username or email has to be set (otherwise the user couldn't log in).
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true, unique: true })
|
@Column({ nullable: true, unique: true })
|
||||||
@IsEmail()
|
@IsEmail()
|
||||||
email?: string;
|
email?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* user phone
|
* The user's phone number.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -40,14 +38,15 @@ export class User {
|
|||||||
phone?: string;
|
phone?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* username
|
* The user's username.
|
||||||
|
* Either username or email has to be set (otherwise the user couldn't log in).
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true, unique: true })
|
@Column({ nullable: true, unique: true })
|
||||||
@IsString()
|
@IsString()
|
||||||
username?: string;
|
username?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* firstname
|
* The user's first name.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -55,7 +54,7 @@ export class User {
|
|||||||
firstname: string;
|
firstname: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* middlename
|
* The user's middle name.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -63,7 +62,7 @@ export class User {
|
|||||||
middlename?: string;
|
middlename?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* lastname
|
* The user's last name.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -71,7 +70,8 @@ export class User {
|
|||||||
lastname: string;
|
lastname: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* password
|
* The user's password.
|
||||||
|
* This is a argon2 hash salted with the user's uuid.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -79,14 +79,8 @@ export class User {
|
|||||||
password: string;
|
password: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* permissions
|
* The groups this user is a part of.
|
||||||
*/
|
* The user will inherit the groups permissions (without overwriting his own).
|
||||||
@IsOptional()
|
|
||||||
@ManyToOne(() => Permission, permission => permission.users, { nullable: true })
|
|
||||||
permissions?: Permission[];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* groups
|
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ManyToMany(() => UserGroup, { nullable: true })
|
@ManyToMany(() => UserGroup, { nullable: true })
|
||||||
@ -94,21 +88,23 @@ export class User {
|
|||||||
groups: UserGroup[];
|
groups: UserGroup[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* is user enabled?
|
* Is this user enabled?
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column()
|
||||||
@IsBoolean()
|
@IsBoolean()
|
||||||
enabled: boolean = true;
|
enabled: boolean = true;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* jwt refresh count
|
* The user's jwt refresh token count.
|
||||||
|
* Used to invalidate jwts.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@Column({ default: 1 })
|
@Column({ default: 1 })
|
||||||
refreshTokenCount?: number;
|
refreshTokenCount?: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* profilepic
|
* The user's profile picture.
|
||||||
|
* We haven't decided yet if this will be a bas64 encoded image or just a link to the profile picture.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true, unique: true })
|
@Column({ nullable: true, unique: true })
|
||||||
@IsString()
|
@IsString()
|
||||||
@ -116,27 +112,17 @@ export class User {
|
|||||||
profilePic?: string;
|
profilePic?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* actions
|
* The actions performed by this user.
|
||||||
|
* For documentation purposes only, will be implemented later.
|
||||||
*/
|
*/
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@OneToMany(() => UserAction, action => action.user, { nullable: true })
|
@OneToMany(() => UserAction, action => action.user, { nullable: true })
|
||||||
actions: UserAction[]
|
actions: UserAction[]
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* calculate all permissions
|
* Turns this entity into it's response class.
|
||||||
*/
|
*/
|
||||||
public get calc_permissions(): Permission[] {
|
public toResponse(): ResponsePrincipal {
|
||||||
let final_permissions = []
|
return new ResponseUser(this);
|
||||||
this.groups.forEach((permission) => {
|
|
||||||
if (!final_permissions.includes(permission)) {
|
|
||||||
final_permissions.push(permission)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
this.permissions.forEach((permission) => {
|
|
||||||
if (!final_permissions.includes(permission)) {
|
|
||||||
final_permissions.push(permission)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
return final_permissions
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,17 @@
|
|||||||
import {
|
import {
|
||||||
|
IsEnum,
|
||||||
IsInt,
|
IsInt,
|
||||||
IsNotEmpty,
|
IsNotEmpty,
|
||||||
IsOptional,
|
IsOptional,
|
||||||
IsString
|
IsString
|
||||||
} from "class-validator";
|
} from "class-validator";
|
||||||
import { Column, Entity, ManyToOne, PrimaryGeneratedColumn } from "typeorm";
|
import { Column, Entity, ManyToOne, PrimaryGeneratedColumn } from "typeorm";
|
||||||
|
import { PermissionAction } from '../enums/PermissionAction';
|
||||||
import { User } from './User';
|
import { User } from './User';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the UserAction interface.
|
* Defines the UserAction entity.
|
||||||
|
* Will later be used to document a user's actions.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@Entity()
|
||||||
export class UserAction {
|
export class UserAction {
|
||||||
@ -20,7 +23,7 @@ export class UserAction {
|
|||||||
id: number;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* user
|
* The user that performed the action.
|
||||||
*/
|
*/
|
||||||
@ManyToOne(() => User, user => user.actions)
|
@ManyToOne(() => User, user => user.actions)
|
||||||
user: User
|
user: User
|
||||||
@ -34,15 +37,16 @@ export class UserAction {
|
|||||||
target: string;
|
target: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The actions's action (e.g. UPDATE)
|
* The actions's action (e.g. UPDATE).
|
||||||
|
* Directly pulled from the PermissionAction Enum.
|
||||||
*/
|
*/
|
||||||
@Column()
|
@Column({ type: 'varchar' })
|
||||||
@IsNotEmpty()
|
@IsEnum(PermissionAction)
|
||||||
@IsString()
|
action: PermissionAction;
|
||||||
action: string;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The description of change (before-> after; e.g. distance:15->17)
|
* The description of the change (before-> after; e.g. distance:15->17).
|
||||||
|
* Will later be defined in more detail.
|
||||||
*/
|
*/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
|
@ -1,29 +1,19 @@
|
|||||||
import {
|
import {
|
||||||
IsInt,
|
|
||||||
IsNotEmpty,
|
IsNotEmpty,
|
||||||
IsOptional,
|
IsOptional,
|
||||||
IsString
|
IsString
|
||||||
} from "class-validator";
|
} from "class-validator";
|
||||||
import { Column, Entity, ManyToOne, PrimaryGeneratedColumn } from "typeorm";
|
import { ChildEntity, Column } from "typeorm";
|
||||||
import { Permission } from "./Permission";
|
import { ResponsePrincipal } from '../responses/ResponsePrincipal';
|
||||||
|
import { ResponseUserGroup } from '../responses/ResponseUserGroup';
|
||||||
|
import { Principal } from './Principal';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the UserGroup interface.
|
* Defines the UserGroup entity.
|
||||||
|
* This entity describes a group of users with a set of permissions.
|
||||||
*/
|
*/
|
||||||
@Entity()
|
@ChildEntity()
|
||||||
export class UserGroup {
|
export class UserGroup extends Principal {
|
||||||
/**
|
|
||||||
* Autogenerated unique id (primary key).
|
|
||||||
*/
|
|
||||||
@PrimaryGeneratedColumn()
|
|
||||||
@IsInt()
|
|
||||||
id: number;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* permissions
|
|
||||||
*/
|
|
||||||
@ManyToOne(() => Permission, permission => permission.groups, { nullable: true })
|
|
||||||
permissions: Permission[];
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The group's name
|
* The group's name
|
||||||
@ -40,4 +30,11 @@ export class UserGroup {
|
|||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsString()
|
@IsString()
|
||||||
description?: string;
|
description?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turns this entity into it's response class.
|
||||||
|
*/
|
||||||
|
public toResponse(): ResponsePrincipal {
|
||||||
|
return new ResponseUserGroup(this);
|
||||||
|
}
|
||||||
}
|
}
|
10
src/models/enums/PermissionAction.ts
Normal file
10
src/models/enums/PermissionAction.ts
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
/**
|
||||||
|
* This enum contains all posible actions for permissions.
|
||||||
|
*/
|
||||||
|
export enum PermissionAction {
|
||||||
|
GET = 'GET',
|
||||||
|
CREATE = 'CREATE',
|
||||||
|
UPDATE = 'UPDATE',
|
||||||
|
DELETE = 'DELETE',
|
||||||
|
IMPORT = 'IMPORT'
|
||||||
|
}
|
12
src/models/enums/PermissionTargets.ts
Normal file
12
src/models/enums/PermissionTargets.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
/**
|
||||||
|
* This enum contains all posible targets for permissions.
|
||||||
|
*/
|
||||||
|
export enum PermissionTarget {
|
||||||
|
RUNNER = 'RUNNER',
|
||||||
|
ORGANISATION = 'ORGANISATION',
|
||||||
|
TEAM = 'TEAM',
|
||||||
|
TRACK = 'TRACK',
|
||||||
|
USER = 'USER',
|
||||||
|
USERGROUP = 'USERGROUP',
|
||||||
|
PERMISSION = 'PERMISSION'
|
||||||
|
}
|
@ -1,26 +1,29 @@
|
|||||||
import { IsInt, IsString } from 'class-validator';
|
import { IsInt, IsString } from 'class-validator';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a auth object
|
* Defines the repsonse auth.
|
||||||
*/
|
*/
|
||||||
export class Auth {
|
export class Auth {
|
||||||
/**
|
/**
|
||||||
* access_token - JWT shortterm access token
|
* The access_token - JWT shortterm access token.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
access_token: string;
|
access_token: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* refresh_token - longterm refresh token (used for requesting new access tokens)
|
* The refresh_token - longterm refresh token (used for requesting new access tokens).
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
refresh_token: string;
|
refresh_token: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* access_token_expires_at - unix timestamp of access token expiry
|
* The unix timestamp for access the token's expiry.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
access_token_expires_at: number;
|
access_token_expires_at: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* refresh_token_expires_at - unix timestamp of access token expiry
|
* The unix unix timestamp for the access token's expiry.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
refresh_token_expires_at: number;
|
refresh_token_expires_at: number;
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { IsString } from 'class-validator';
|
import { IsString } from 'class-validator';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a empty response object
|
* Defines a empty response object.
|
||||||
*/
|
*/
|
||||||
export class ResponseEmpty {
|
export class ResponseEmpty {
|
||||||
@IsString()
|
@IsString()
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
import { IsString } from 'class-validator';
|
import { IsString } from 'class-validator';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a Logout object
|
* Defines the logout response.
|
||||||
*/
|
*/
|
||||||
export class Logout {
|
export class Logout {
|
||||||
/**
|
/**
|
||||||
* timestamp of logout
|
* The logout's timestamp.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
timestamp: number;
|
timestamp: number;
|
||||||
|
@ -1,21 +1,15 @@
|
|||||||
import {
|
import { IsInt, IsString } from "class-validator";
|
||||||
IsInt,
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
IsString
|
|
||||||
} from "class-validator";
|
|
||||||
import { Participant } from '../entities/Participant';
|
import { Participant } from '../entities/Participant';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a participant response.
|
* Defines the participant response.
|
||||||
*/
|
*/
|
||||||
export abstract class ResponseParticipant {
|
export abstract class ResponseParticipant {
|
||||||
/**
|
/**
|
||||||
* Autogenerated unique id (primary key).
|
* The participant's id.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
id: number;;
|
id: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's first name.
|
* The participant's first name.
|
||||||
@ -25,7 +19,6 @@ export abstract class ResponseParticipant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's middle name.
|
* The participant's middle name.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
middlename?: string;
|
middlename?: string;
|
||||||
@ -38,18 +31,20 @@ export abstract class ResponseParticipant {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's phone number.
|
* The participant's phone number.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
phone?: string;
|
phone?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The participant's e-mail address.
|
* The participant's e-mail address.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
email?: string;
|
email?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseParticipant object from a participant.
|
||||||
|
* @param participant The participant the response shall be build for.
|
||||||
|
*/
|
||||||
public constructor(participant: Participant) {
|
public constructor(participant: Participant) {
|
||||||
this.id = participant.id;
|
this.id = participant.id;
|
||||||
this.firstname = participant.firstname;
|
this.firstname = participant.firstname;
|
||||||
|
53
src/models/responses/ResponsePermission.ts
Normal file
53
src/models/responses/ResponsePermission.ts
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import {
|
||||||
|
IsEnum,
|
||||||
|
IsInt,
|
||||||
|
IsNotEmpty,
|
||||||
|
IsObject
|
||||||
|
} from "class-validator";
|
||||||
|
import { Permission } from '../entities/Permission';
|
||||||
|
import { PermissionAction } from '../enums/PermissionAction';
|
||||||
|
import { PermissionTarget } from '../enums/PermissionTargets';
|
||||||
|
import { ResponsePrincipal } from './ResponsePrincipal';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the permission response.
|
||||||
|
*/
|
||||||
|
export class ResponsePermission {
|
||||||
|
/**
|
||||||
|
* The permission's id.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
id: number;;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The permissions's principal.
|
||||||
|
*/
|
||||||
|
@IsObject()
|
||||||
|
@IsNotEmpty()
|
||||||
|
principal: ResponsePrincipal;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The permissions's target.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsEnum(PermissionTarget)
|
||||||
|
target: PermissionTarget;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The permissions's action.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsEnum(PermissionAction)
|
||||||
|
action: PermissionAction;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponsePermission object from a permission.
|
||||||
|
* @param permission The permission the response shall be build for.
|
||||||
|
*/
|
||||||
|
public constructor(permission: Permission) {
|
||||||
|
this.id = permission.id;
|
||||||
|
this.principal = permission.principal.toResponse();
|
||||||
|
this.target = permission.target;
|
||||||
|
this.action = permission.action;
|
||||||
|
}
|
||||||
|
}
|
24
src/models/responses/ResponsePrincipal.ts
Normal file
24
src/models/responses/ResponsePrincipal.ts
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
import {
|
||||||
|
IsInt
|
||||||
|
} from "class-validator";
|
||||||
|
import { Principal } from '../entities/Principal';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the principal response.
|
||||||
|
*/
|
||||||
|
export abstract class ResponsePrincipal {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The principal's id.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
id: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponsePrincipal object from a principal.
|
||||||
|
* @param principal The principal the response shall be build for.
|
||||||
|
*/
|
||||||
|
public constructor(principal: Principal) {
|
||||||
|
this.id = principal.id;
|
||||||
|
}
|
||||||
|
}
|
@ -7,13 +7,12 @@ import { RunnerGroup } from '../entities/RunnerGroup';
|
|||||||
import { ResponseParticipant } from './ResponseParticipant';
|
import { ResponseParticipant } from './ResponseParticipant';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines RunnerTeam's response class.
|
* Defines the runner response.
|
||||||
*/
|
*/
|
||||||
export class ResponseRunner extends ResponseParticipant {
|
export class ResponseRunner extends ResponseParticipant {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The runner's currently ran distance in meters.
|
* The runner's currently ran distance in meters.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
distance: number;
|
distance: number;
|
||||||
@ -24,6 +23,10 @@ export class ResponseRunner extends ResponseParticipant {
|
|||||||
@IsObject()
|
@IsObject()
|
||||||
group: RunnerGroup;
|
group: RunnerGroup;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseRunner object from a runner.
|
||||||
|
* @param runner The user the response shall be build for.
|
||||||
|
*/
|
||||||
public constructor(runner: Runner) {
|
public constructor(runner: Runner) {
|
||||||
super(runner);
|
super(runner);
|
||||||
this.distance = runner.scans.filter(scan => { scan.valid === true }).reduce((sum, current) => sum + current.distance, 0);
|
this.distance = runner.scans.filter(scan => { scan.valid === true }).reduce((sum, current) => sum + current.distance, 0);
|
||||||
|
@ -1,38 +1,20 @@
|
|||||||
import {
|
import { IsInt, IsNotEmpty, IsObject, IsOptional, IsString } from "class-validator";
|
||||||
IsInt,
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
IsNotEmpty,
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
IsObject,
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
IsOptional,
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
IsString
|
|
||||||
} from "class-validator";
|
|
||||||
import { GroupContact } from '../entities/GroupContact';
|
import { GroupContact } from '../entities/GroupContact';
|
||||||
import { RunnerGroup } from '../entities/RunnerGroup';
|
import { RunnerGroup } from '../entities/RunnerGroup';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a track of given length.
|
* Defines the runnerGroup response.
|
||||||
*/
|
*/
|
||||||
export abstract class ResponseRunnerGroup {
|
export abstract class ResponseRunnerGroup {
|
||||||
/**
|
/**
|
||||||
* Autogenerated unique id (primary key).
|
* The runnerGroup's id.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
id: number;;
|
id: number;;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The groups's name.
|
* The runnerGroup's name.
|
||||||
*/
|
*/
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@ -40,13 +22,16 @@ export abstract class ResponseRunnerGroup {
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The group's contact.
|
* The runnerGroup's contact.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsObject()
|
@IsObject()
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
contact?: GroupContact;
|
contact?: GroupContact;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseRunnerGroup object from a runnerGroup.
|
||||||
|
* @param group The runnerGroup the response shall be build for.
|
||||||
|
*/
|
||||||
public constructor(group: RunnerGroup) {
|
public constructor(group: RunnerGroup) {
|
||||||
this.id = group.id;
|
this.id = group.id;
|
||||||
this.name = group.name;
|
this.name = group.name;
|
||||||
|
@ -9,25 +9,27 @@ import { RunnerTeam } from '../entities/RunnerTeam';
|
|||||||
import { ResponseRunnerGroup } from './ResponseRunnerGroup';
|
import { ResponseRunnerGroup } from './ResponseRunnerGroup';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines RunnerOrgs's response class.
|
* Defines the runnerOrganisation response.
|
||||||
*/
|
*/
|
||||||
export class ResponseRunnerOrganisation extends ResponseRunnerGroup {
|
export class ResponseRunnerOrganisation extends ResponseRunnerGroup {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The orgs's address.
|
* The runnerOrganisation's address.
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsObject()
|
@IsObject()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
address?: Address;
|
address?: Address;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The orgs associated teams.
|
* The runnerOrganisation associated teams.
|
||||||
*/
|
*/
|
||||||
@IsArray()
|
@IsArray()
|
||||||
teams: RunnerTeam[];
|
teams: RunnerTeam[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseRunnerOrganisation object from a runnerOrganisation.
|
||||||
|
* @param org The runnerOrganisation the response shall be build for.
|
||||||
|
*/
|
||||||
public constructor(org: RunnerOrganisation) {
|
public constructor(org: RunnerOrganisation) {
|
||||||
super(org);
|
super(org);
|
||||||
this.address = org.address;
|
this.address = org.address;
|
||||||
|
@ -1,24 +1,24 @@
|
|||||||
import {
|
import { IsNotEmpty, IsObject } from "class-validator";
|
||||||
IsNotEmpty,
|
|
||||||
IsObject
|
|
||||||
} from "class-validator";
|
|
||||||
import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
import { RunnerOrganisation } from '../entities/RunnerOrganisation';
|
||||||
import { RunnerTeam } from '../entities/RunnerTeam';
|
import { RunnerTeam } from '../entities/RunnerTeam';
|
||||||
import { ResponseRunnerGroup } from './ResponseRunnerGroup';
|
import { ResponseRunnerGroup } from './ResponseRunnerGroup';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines RunnerTeam's response class.
|
* Defines the runnerTeam response.
|
||||||
*/
|
*/
|
||||||
export class ResponseRunnerTeam extends ResponseRunnerGroup {
|
export class ResponseRunnerTeam extends ResponseRunnerGroup {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The team's parent group (organisation).
|
* The runnerTeam's parent group (organisation).
|
||||||
* Optional.
|
|
||||||
*/
|
*/
|
||||||
@IsObject()
|
@IsObject()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
parentGroup: RunnerOrganisation;
|
parentGroup: RunnerOrganisation;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseRunnerTeam object from a runnerTeam.
|
||||||
|
* @param team The team the response shall be build for.
|
||||||
|
*/
|
||||||
public constructor(team: RunnerTeam) {
|
public constructor(team: RunnerTeam) {
|
||||||
super(team);
|
super(team);
|
||||||
this.parentGroup = team.parentGroup;
|
this.parentGroup = team.parentGroup;
|
||||||
|
@ -1,16 +1,12 @@
|
|||||||
import {
|
import { IsInt, IsString } from "class-validator";
|
||||||
IsInt,
|
|
||||||
|
|
||||||
IsString
|
|
||||||
} from "class-validator";
|
|
||||||
import { Track } from '../entities/Track';
|
import { Track } from '../entities/Track';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a track of given length.
|
* Defines the track response.
|
||||||
*/
|
*/
|
||||||
export class ResponseTrack {
|
export class ResponseTrack {
|
||||||
/**
|
/**
|
||||||
* Autogenerated unique id (primary key).
|
* The track's id.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
id: number;;
|
id: number;;
|
||||||
@ -27,6 +23,10 @@ export class ResponseTrack {
|
|||||||
@IsInt()
|
@IsInt()
|
||||||
distance: number;
|
distance: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseTrack object from a track.
|
||||||
|
* @param track The track the response shall be build for.
|
||||||
|
*/
|
||||||
public constructor(track: Track) {
|
public constructor(track: Track) {
|
||||||
this.id = track.id;
|
this.id = track.id;
|
||||||
this.name = track.name;
|
this.name = track.name;
|
||||||
|
97
src/models/responses/ResponseUser.ts
Normal file
97
src/models/responses/ResponseUser.ts
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
import {
|
||||||
|
IsArray,
|
||||||
|
IsBoolean,
|
||||||
|
|
||||||
|
IsOptional,
|
||||||
|
IsString
|
||||||
|
} from "class-validator";
|
||||||
|
import { Permission } from '../entities/Permission';
|
||||||
|
import { User } from '../entities/User';
|
||||||
|
import { UserGroup } from '../entities/UserGroup';
|
||||||
|
import { ResponsePrincipal } from './ResponsePrincipal';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the user response.
|
||||||
|
*/
|
||||||
|
export class ResponseUser extends ResponsePrincipal {
|
||||||
|
/**
|
||||||
|
* The user's first name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
firstname: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's middle name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
middlename?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's last name.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
lastname: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's phone number.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
phone?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's e-mail address.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
email?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's username.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
username?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is user enabled?
|
||||||
|
*/
|
||||||
|
@IsBoolean()
|
||||||
|
enabled: boolean = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's profile pic.
|
||||||
|
*/
|
||||||
|
@IsString()
|
||||||
|
@IsOptional()
|
||||||
|
profilePic?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The groups that the user is a part of.
|
||||||
|
*/
|
||||||
|
@IsArray()
|
||||||
|
@IsOptional()
|
||||||
|
groups: UserGroup[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The user's permissions.
|
||||||
|
*/
|
||||||
|
@IsArray()
|
||||||
|
@IsOptional()
|
||||||
|
permissions: Permission[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseUser object from a user.
|
||||||
|
* @param user The user the response shall be build for.
|
||||||
|
*/
|
||||||
|
public constructor(user: User) {
|
||||||
|
super(user);
|
||||||
|
this.firstname = user.firstname;
|
||||||
|
this.middlename = user.middlename;
|
||||||
|
this.lastname = user.lastname;
|
||||||
|
this.phone = user.phone;
|
||||||
|
this.email = user.email;
|
||||||
|
this.username = user.username;
|
||||||
|
this.enabled = user.enabled;
|
||||||
|
this.profilePic = user.profilePic;
|
||||||
|
this.groups = user.groups;
|
||||||
|
this.permissions = user.permissions;
|
||||||
|
}
|
||||||
|
}
|
41
src/models/responses/ResponseUserGroup.ts
Normal file
41
src/models/responses/ResponseUserGroup.ts
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
import { IsArray, IsNotEmpty, IsOptional, IsString } from "class-validator";
|
||||||
|
import { Permission } from '../entities/Permission';
|
||||||
|
import { UserGroup } from '../entities/UserGroup';
|
||||||
|
import { ResponsePrincipal } from './ResponsePrincipal';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the userGroup response.
|
||||||
|
*/
|
||||||
|
export class ResponseUserGroup extends ResponsePrincipal {
|
||||||
|
/**
|
||||||
|
* The userGroup's name.
|
||||||
|
*/
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
|
name: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The userGroup's description.
|
||||||
|
*/
|
||||||
|
@IsOptional()
|
||||||
|
@IsString()
|
||||||
|
description?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The userGroup's permissions.
|
||||||
|
*/
|
||||||
|
@IsArray()
|
||||||
|
@IsOptional()
|
||||||
|
permissions: Permission[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ResponseUserGroup object from a userGroup.
|
||||||
|
* @param group The userGroup the response shall be build for.
|
||||||
|
*/
|
||||||
|
public constructor(group: UserGroup) {
|
||||||
|
super(group);
|
||||||
|
this.name = group.name;
|
||||||
|
this.description = group.description;
|
||||||
|
this.permissions = group.permissions;
|
||||||
|
}
|
||||||
|
}
|
56
src/openapi_export.ts
Normal file
56
src/openapi_export.ts
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
import { validationMetadatasToSchemas } from 'class-validator-jsonschema';
|
||||||
|
import consola from "consola";
|
||||||
|
import fs from "fs";
|
||||||
|
import "reflect-metadata";
|
||||||
|
import { createExpressServer, getMetadataArgsStorage } from "routing-controllers";
|
||||||
|
import { routingControllersToSpec } from 'routing-controllers-openapi';
|
||||||
|
import authchecker from "./authchecker";
|
||||||
|
import { config } from './config';
|
||||||
|
import { ErrorHandler } from './middlewares/ErrorHandler';
|
||||||
|
|
||||||
|
const CONTROLLERS_FILE_EXTENSION = process.env.NODE_ENV === 'production' ? 'js' : 'ts';
|
||||||
|
createExpressServer({
|
||||||
|
authorizationChecker: authchecker,
|
||||||
|
middlewares: [ErrorHandler],
|
||||||
|
development: config.development,
|
||||||
|
cors: true,
|
||||||
|
routePrefix: "/api",
|
||||||
|
controllers: [`${__dirname}/controllers/*.${CONTROLLERS_FILE_EXTENSION}`],
|
||||||
|
});
|
||||||
|
|
||||||
|
const storage = getMetadataArgsStorage();
|
||||||
|
const schemas = validationMetadatasToSchemas({
|
||||||
|
refPointerPrefix: "#/components/schemas/",
|
||||||
|
});
|
||||||
|
|
||||||
|
//Spec creation based on the previously created schemas
|
||||||
|
const spec = routingControllersToSpec(
|
||||||
|
storage,
|
||||||
|
{
|
||||||
|
routePrefix: "/api"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
components: {
|
||||||
|
schemas,
|
||||||
|
"securitySchemes": {
|
||||||
|
"AuthToken": {
|
||||||
|
"type": "http",
|
||||||
|
"scheme": "bearer",
|
||||||
|
"bearerFormat": "JWT"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
info: {
|
||||||
|
description: "The the backend API for the LfK! runner system.",
|
||||||
|
title: "LfK! Backend API",
|
||||||
|
version: "1.0.0",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
try {
|
||||||
|
fs.writeFileSync("./openapi.json", JSON.stringify(spec), { encoding: "utf-8" });
|
||||||
|
consola.success("Exported openapi spec to openapi.json");
|
||||||
|
} catch (error) {
|
||||||
|
consola.error("Couldn't export the openapi spec");
|
||||||
|
}
|
52
src/seeds/SeedUsers.ts
Normal file
52
src/seeds/SeedUsers.ts
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import { Connection } from 'typeorm';
|
||||||
|
import { Factory, Seeder } from 'typeorm-seeding';
|
||||||
|
import { CreatePermission } from '../models/actions/CreatePermission';
|
||||||
|
import { CreateUser } from '../models/actions/CreateUser';
|
||||||
|
import { CreateUserGroup } from '../models/actions/CreateUserGroup';
|
||||||
|
import { Permission } from '../models/entities/Permission';
|
||||||
|
import { User } from '../models/entities/User';
|
||||||
|
import { UserGroup } from '../models/entities/UserGroup';
|
||||||
|
import { PermissionAction } from '../models/enums/PermissionAction';
|
||||||
|
import { PermissionTarget } from '../models/enums/PermissionTargets';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Seeds a admin group with a demo user into the database for initial setup and auto recovery.
|
||||||
|
* We know that the nameing isn't perfectly fitting. Feel free to change it.
|
||||||
|
*/
|
||||||
|
export default class SeedUsers implements Seeder {
|
||||||
|
public async run(factory: Factory, connection: Connection): Promise<any> {
|
||||||
|
let adminGroup: UserGroup = await this.createAdminGroup(connection);
|
||||||
|
await this.createUser(connection, adminGroup.id);
|
||||||
|
await this.createPermissions(connection, adminGroup.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async createAdminGroup(connection: Connection) {
|
||||||
|
let adminGroup = new CreateUserGroup();
|
||||||
|
adminGroup.name = "ADMINS";
|
||||||
|
adminGroup.description = "Have all possible permissions";
|
||||||
|
return await connection.getRepository(UserGroup).save(await adminGroup.toUserGroup());
|
||||||
|
}
|
||||||
|
|
||||||
|
public async createUser(connection: Connection, group: number) {
|
||||||
|
let initialUser = new CreateUser();
|
||||||
|
initialUser.firstname = "demo";
|
||||||
|
initialUser.lastname = "demo";
|
||||||
|
initialUser.username = "demo";
|
||||||
|
initialUser.password = "demo";
|
||||||
|
initialUser.groups = group;
|
||||||
|
return await connection.getRepository(User).save(await initialUser.toUser());
|
||||||
|
}
|
||||||
|
|
||||||
|
public async createPermissions(connection: Connection, principal: number) {
|
||||||
|
let repo = await connection.getRepository(Permission);
|
||||||
|
for (let target in PermissionTarget) {
|
||||||
|
for (let action in PermissionAction) {
|
||||||
|
let permission = new CreatePermission;
|
||||||
|
permission.target = <PermissionTarget>target;
|
||||||
|
permission.action = <PermissionAction>action;
|
||||||
|
permission.principal = principal;
|
||||||
|
await repo.save(await permission.toPermission());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -2,6 +2,18 @@ import axios from 'axios';
|
|||||||
import { config } from '../config';
|
import { config } from '../config';
|
||||||
const base = "http://localhost:" + config.internal_port
|
const base = "http://localhost:" + config.internal_port
|
||||||
|
|
||||||
|
let access_token;
|
||||||
|
let axios_config;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const res = await axios.post(base + '/api/auth/login', { username: "demo", password: "demo" });
|
||||||
|
access_token = res.data["access_token"];
|
||||||
|
axios_config = {
|
||||||
|
headers: { "authorization": "Bearer " + access_token },
|
||||||
|
validateStatus: undefined
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
describe('GET /api/openapi.json', () => {
|
describe('GET /api/openapi.json', () => {
|
||||||
it('is http 200', async () => {
|
it('is http 200', async () => {
|
||||||
const res = await axios.get(base + '/api/openapi.json');
|
const res = await axios.get(base + '/api/openapi.json');
|
||||||
@ -10,13 +22,13 @@ describe('GET /api/openapi.json', () => {
|
|||||||
});
|
});
|
||||||
describe('GET /', () => {
|
describe('GET /', () => {
|
||||||
it('is http 404', async () => {
|
it('is http 404', async () => {
|
||||||
const res = await axios.get(base + '/', { validateStatus: undefined });
|
const res = await axios.get(base + '/', axios_config);
|
||||||
expect(res.status).toEqual(404);
|
expect(res.status).toEqual(404);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
describe('GET /api/teams', () => {
|
describe('GET /api/teams', () => {
|
||||||
it('is http 200 && is json', async () => {
|
it('is http 200 && is json', async () => {
|
||||||
const res = await axios.get(base + '/api/teams', { validateStatus: undefined });
|
const res = await axios.get(base + '/api/teams', axios_config);
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
|
@ -2,9 +2,21 @@ import axios from 'axios';
|
|||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
const base = "http://localhost:" + config.internal_port
|
const base = "http://localhost:" + config.internal_port
|
||||||
|
|
||||||
|
let access_token;
|
||||||
|
let axios_config;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const res = await axios.post(base + '/api/auth/login', { username: "demo", password: "demo" });
|
||||||
|
access_token = res.data["access_token"];
|
||||||
|
axios_config = {
|
||||||
|
headers: { "authorization": "Bearer " + access_token },
|
||||||
|
validateStatus: undefined
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
describe('GET /api/organisations', () => {
|
describe('GET /api/organisations', () => {
|
||||||
it('basic get should return 200', async () => {
|
it('basic get should return 200', async () => {
|
||||||
const res = await axios.get(base + '/api/organisations');
|
const res = await axios.get(base + '/api/organisations', axios_config);
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
@ -14,14 +26,14 @@ describe('POST /api/organisations', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/organisations', {
|
const res = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
it('creating a new org with without a name should return 400', async () => {
|
it('creating a new org with without a name should return 400', async () => {
|
||||||
const res = await axios.post(base + '/api/organisations', {
|
const res = await axios.post(base + '/api/organisations', {
|
||||||
"name": null
|
"name": null
|
||||||
}, { validateStatus: undefined });
|
}, axios_config);
|
||||||
expect(res.status).toEqual(400);
|
expect(res.status).toEqual(400);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
@ -31,12 +43,12 @@ describe('adding + getting from all orgs', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/organisations', {
|
const res = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
it('check if org was added', async () => {
|
it('check if org was added', async () => {
|
||||||
const res = await axios.get(base + '/api/organisations');
|
const res = await axios.get(base + '/api/organisations', axios_config);
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
let added_org = res.data[res.data.length - 1]
|
let added_org = res.data[res.data.length - 1]
|
||||||
@ -55,14 +67,14 @@ describe('adding + getting explicitly', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res1 = await axios.post(base + '/api/organisations', {
|
const res1 = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
let added_org = res1.data
|
let added_org = res1.data
|
||||||
added_org_id = added_org.id;
|
added_org_id = added_org.id;
|
||||||
expect(res1.status).toEqual(200);
|
expect(res1.status).toEqual(200);
|
||||||
expect(res1.headers['content-type']).toContain("application/json")
|
expect(res1.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
it('check if org was added', async () => {
|
it('check if org was added', async () => {
|
||||||
const res2 = await axios.get(base + '/api/organisations/' + added_org_id);
|
const res2 = await axios.get(base + '/api/organisations/' + added_org_id, axios_config);
|
||||||
expect(res2.status).toEqual(200);
|
expect(res2.status).toEqual(200);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
let added_org2 = res2.data
|
let added_org2 = res2.data
|
||||||
|
@ -2,10 +2,22 @@ import axios from 'axios';
|
|||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
const base = "http://localhost:" + config.internal_port
|
const base = "http://localhost:" + config.internal_port
|
||||||
|
|
||||||
|
let access_token;
|
||||||
|
let axios_config;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const res = await axios.post(base + '/api/auth/login', { username: "demo", password: "demo" });
|
||||||
|
access_token = res.data["access_token"];
|
||||||
|
axios_config = {
|
||||||
|
headers: { "authorization": "Bearer " + access_token },
|
||||||
|
validateStatus: undefined
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
// ---------------
|
// ---------------
|
||||||
describe('adding + deletion (non-existant)', () => {
|
describe('adding + deletion (non-existant)', () => {
|
||||||
it('delete', async () => {
|
it('delete', async () => {
|
||||||
const res2 = await axios.delete(base + '/api/organisations/0', { validateStatus: undefined });
|
const res2 = await axios.delete(base + '/api/organisations/0', axios_config);
|
||||||
expect(res2.status).toEqual(204);
|
expect(res2.status).toEqual(204);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -16,14 +28,14 @@ describe('adding + deletion (successfull)', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res1 = await axios.post(base + '/api/organisations', {
|
const res1 = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
added_org = res1.data
|
added_org = res1.data
|
||||||
added_org_id = added_org.id;
|
added_org_id = added_org.id;
|
||||||
expect(res1.status).toEqual(200);
|
expect(res1.status).toEqual(200);
|
||||||
expect(res1.headers['content-type']).toContain("application/json")
|
expect(res1.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
it('delete', async () => {
|
it('delete', async () => {
|
||||||
const res2 = await axios.delete(base + '/api/organisations/' + added_org_id);
|
const res2 = await axios.delete(base + '/api/organisations/' + added_org_id, axios_config);
|
||||||
expect(res2.status).toEqual(200);
|
expect(res2.status).toEqual(200);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
let added_org2 = res2.data
|
let added_org2 = res2.data
|
||||||
@ -37,7 +49,7 @@ describe('adding + deletion (successfull)', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
it('check if org really was deleted', async () => {
|
it('check if org really was deleted', async () => {
|
||||||
const res3 = await axios.get(base + '/api/organisations/' + added_org_id, { validateStatus: undefined });
|
const res3 = await axios.get(base + '/api/organisations/' + added_org_id, axios_config);
|
||||||
expect(res3.status).toEqual(404);
|
expect(res3.status).toEqual(404);
|
||||||
expect(res3.headers['content-type']).toContain("application/json")
|
expect(res3.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
@ -51,7 +63,7 @@ describe('adding + deletion with teams still existing (without force)', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res1 = await axios.post(base + '/api/organisations', {
|
const res1 = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
added_org = res1.data;
|
added_org = res1.data;
|
||||||
added_org_id = added_org.id;
|
added_org_id = added_org.id;
|
||||||
expect(res1.status).toEqual(200);
|
expect(res1.status).toEqual(200);
|
||||||
@ -61,14 +73,14 @@ describe('adding + deletion with teams still existing (without force)', () => {
|
|||||||
const res2 = await axios.post(base + '/api/teams', {
|
const res2 = await axios.post(base + '/api/teams', {
|
||||||
"name": "test123",
|
"name": "test123",
|
||||||
"parentGroup": added_org_id
|
"parentGroup": added_org_id
|
||||||
});
|
}, axios_config);
|
||||||
added_team = res2.data;
|
added_team = res2.data;
|
||||||
added_team_id = added_team.id;
|
added_team_id = added_team.id;
|
||||||
expect(res2.status).toEqual(200);
|
expect(res2.status).toEqual(200);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
it('delete org - this should fail with a 406', async () => {
|
it('delete org - this should fail with a 406', async () => {
|
||||||
const res2 = await axios.delete(base + '/api/organisations/' + added_org_id, { validateStatus: undefined });
|
const res2 = await axios.delete(base + '/api/organisations/' + added_org_id, axios_config);
|
||||||
expect(res2.status).toEqual(406);
|
expect(res2.status).toEqual(406);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
@ -82,7 +94,7 @@ describe('adding + deletion with teams still existing (with force)', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res1 = await axios.post(base + '/api/organisations', {
|
const res1 = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
added_org = res1.data;
|
added_org = res1.data;
|
||||||
added_org_id = added_org.id;
|
added_org_id = added_org.id;
|
||||||
expect(res1.status).toEqual(200);
|
expect(res1.status).toEqual(200);
|
||||||
@ -92,14 +104,14 @@ describe('adding + deletion with teams still existing (with force)', () => {
|
|||||||
const res2 = await axios.post(base + '/api/teams', {
|
const res2 = await axios.post(base + '/api/teams', {
|
||||||
"name": "test123",
|
"name": "test123",
|
||||||
"parentGroup": added_org_id
|
"parentGroup": added_org_id
|
||||||
});
|
}, axios_config);
|
||||||
added_team = res2.data;
|
added_team = res2.data;
|
||||||
added_team_id = added_team.id;
|
added_team_id = added_team.id;
|
||||||
expect(res2.status).toEqual(200);
|
expect(res2.status).toEqual(200);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
it('delete', async () => {
|
it('delete', async () => {
|
||||||
const res2 = await axios.delete(base + '/api/organisations/' + added_org_id + '?force=true');
|
const res2 = await axios.delete(base + '/api/organisations/' + added_org_id + '?force=true', axios_config);
|
||||||
expect(res2.status).toEqual(200);
|
expect(res2.status).toEqual(200);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
let added_org2 = res2.data
|
let added_org2 = res2.data
|
||||||
@ -113,7 +125,7 @@ describe('adding + deletion with teams still existing (with force)', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
it('check if org really was deleted', async () => {
|
it('check if org really was deleted', async () => {
|
||||||
const res3 = await axios.get(base + '/api/organisations/' + added_org_id, { validateStatus: undefined });
|
const res3 = await axios.get(base + '/api/organisations/' + added_org_id, axios_config);
|
||||||
expect(res3.status).toEqual(404);
|
expect(res3.status).toEqual(404);
|
||||||
expect(res3.headers['content-type']).toContain("application/json")
|
expect(res3.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
|
@ -2,9 +2,21 @@ import axios from 'axios';
|
|||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
const base = "http://localhost:" + config.internal_port
|
const base = "http://localhost:" + config.internal_port
|
||||||
|
|
||||||
|
let access_token;
|
||||||
|
let axios_config;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const res = await axios.post(base + '/api/auth/login', { username: "demo", password: "demo" });
|
||||||
|
access_token = res.data["access_token"];
|
||||||
|
axios_config = {
|
||||||
|
headers: { "authorization": "Bearer " + access_token },
|
||||||
|
validateStatus: undefined
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
describe('GET /api/organisations', () => {
|
describe('GET /api/organisations', () => {
|
||||||
it('basic get should return 200', async () => {
|
it('basic get should return 200', async () => {
|
||||||
const res = await axios.get(base + '/api/organisations');
|
const res = await axios.get(base + '/api/organisations', axios_config);
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
@ -12,7 +24,7 @@ describe('GET /api/organisations', () => {
|
|||||||
// ---------------
|
// ---------------
|
||||||
describe('GET /api/organisations/0', () => {
|
describe('GET /api/organisations/0', () => {
|
||||||
it('basic get should return 404', async () => {
|
it('basic get should return 404', async () => {
|
||||||
const res = await axios.get(base + '/api/runners/0', { validateStatus: undefined });
|
const res = await axios.get(base + '/api/runners/0', axios_config);
|
||||||
expect(res.status).toEqual(404);
|
expect(res.status).toEqual(404);
|
||||||
expect(res.headers['content-type']).toContain("application/json")
|
expect(res.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
|
@ -2,6 +2,18 @@ import axios from 'axios';
|
|||||||
import { config } from '../../config';
|
import { config } from '../../config';
|
||||||
const base = "http://localhost:" + config.internal_port
|
const base = "http://localhost:" + config.internal_port
|
||||||
|
|
||||||
|
let access_token;
|
||||||
|
let axios_config;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const res = await axios.post(base + '/api/auth/login', { username: "demo", password: "demo" });
|
||||||
|
access_token = res.data["access_token"];
|
||||||
|
axios_config = {
|
||||||
|
headers: { "authorization": "Bearer " + access_token },
|
||||||
|
validateStatus: undefined
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
// ---------------
|
// ---------------
|
||||||
describe('adding + updating name', () => {
|
describe('adding + updating name', () => {
|
||||||
let added_org_id
|
let added_org_id
|
||||||
@ -9,7 +21,7 @@ describe('adding + updating name', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res1 = await axios.post(base + '/api/organisations', {
|
const res1 = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
added_org = res1.data
|
added_org = res1.data
|
||||||
added_org_id = added_org.id;
|
added_org_id = added_org.id;
|
||||||
expect(res1.status).toEqual(200);
|
expect(res1.status).toEqual(200);
|
||||||
@ -21,7 +33,7 @@ describe('adding + updating name', () => {
|
|||||||
"name": "testlelele",
|
"name": "testlelele",
|
||||||
"contact": null,
|
"contact": null,
|
||||||
"address": null,
|
"address": null,
|
||||||
});
|
}, axios_config);
|
||||||
expect(res2.status).toEqual(200);
|
expect(res2.status).toEqual(200);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
let added_org2 = res2.data
|
let added_org2 = res2.data
|
||||||
@ -42,7 +54,7 @@ describe('adding + try updating id (should return 406)', () => {
|
|||||||
it('creating a new org with just a name should return 200', async () => {
|
it('creating a new org with just a name should return 200', async () => {
|
||||||
const res1 = await axios.post(base + '/api/organisations', {
|
const res1 = await axios.post(base + '/api/organisations', {
|
||||||
"name": "test123"
|
"name": "test123"
|
||||||
});
|
}, axios_config);
|
||||||
added_org = res1.data
|
added_org = res1.data
|
||||||
added_org_id = added_org.id;
|
added_org_id = added_org.id;
|
||||||
expect(res1.status).toEqual(200);
|
expect(res1.status).toEqual(200);
|
||||||
@ -54,7 +66,7 @@ describe('adding + try updating id (should return 406)', () => {
|
|||||||
"name": "testlelele",
|
"name": "testlelele",
|
||||||
"contact": null,
|
"contact": null,
|
||||||
"address": null,
|
"address": null,
|
||||||
}, { validateStatus: undefined });
|
}, axios_config);
|
||||||
expect(res2.status).toEqual(406);
|
expect(res2.status).toEqual(406);
|
||||||
expect(res2.headers['content-type']).toContain("application/json")
|
expect(res2.headers['content-type']).toContain("application/json")
|
||||||
});
|
});
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user