Compare commits
185 Commits
6163f0a90b
...
1.2.0
Author | SHA1 | Date | |
---|---|---|---|
6764bf80ea
|
|||
b3a73b25e8
|
|||
bda1f971d1 | |||
765ef84903
|
|||
296ba8ddab
|
|||
6eff243803
|
|||
0f4c8b2051
|
|||
d842c14240
|
|||
a54cb287a4
|
|||
74d334f9b7
|
|||
cd3cd81360
|
|||
cf48c00ddb
|
|||
3192365793
|
|||
075d484f11
|
|||
5082b1b8b1
|
|||
50dd703a1b
|
|||
057a8ee699
|
|||
8d9418635d
|
|||
f2832a2dae
|
|||
0d21596e2b
|
|||
245827e9c6
|
|||
4608a36df6
|
|||
cb1305aa77
|
|||
12a9ae2493
|
|||
b9fe9f1c24
|
|||
b25b0db760
|
|||
fe59e3a557
|
|||
42c23a5883
|
|||
6ee5328dbc
|
|||
6f39ac42da
|
|||
301f334674
|
|||
fcee3909f4
|
|||
f0e20e4130
|
|||
80de188565 | |||
2f305e127c
|
|||
513d7f6fba
|
|||
244da61892
|
|||
2a72aea10e
|
|||
71ebce6f8e
|
|||
f60025b6de
|
|||
0fa663a341
|
|||
538622aa18
|
|||
86a21dbfa4
|
|||
1e9e24d99d
|
|||
4493c0e3d9
|
|||
f5d48fc638
|
|||
b35a2dd2fa
|
|||
a28ffe06e5
|
|||
d873674819
|
|||
37b2ac974b
|
|||
81aed1de40
|
|||
0f0c3c7214
|
|||
3909ed34f7
|
|||
b2ac70e0ae
|
|||
5f17e7f783
|
|||
a5a56a263a
|
|||
2d8f7528d9
|
|||
9581185b24
|
|||
2905884c02
|
|||
e9914e317b
|
|||
702070da66
|
|||
cc89ba8afb
|
|||
7c4ff42a3b
|
|||
8007117434
|
|||
23fa78eb9d
|
|||
3b3e68900b
|
|||
3ff666fd3e
|
|||
4e4435010f
|
|||
de9af5a909
|
|||
ac631f0af4
|
|||
6bbdd5bb04
|
|||
a8fc755840
|
|||
27e74e824c
|
|||
b5c0a288ac
|
|||
85dc3444ac
|
|||
d02743984d
|
|||
734c826fac
|
|||
33b25c9743
|
|||
6275aaa326
|
|||
2a94bfa622
|
|||
a64f6c9822
|
|||
93d43b7684
|
|||
16ce0a8480
|
|||
9a8d618ae4
|
|||
38da2d3318
|
|||
068deb4960
|
|||
13f093bb61
|
|||
6289f30740
|
|||
6ff764bc34
|
|||
ea87cc793b
|
|||
92517e3653
|
|||
ffee887ddf
|
|||
3bac75e7ab
|
|||
d05eddcae1 | |||
d5c689d693
|
|||
8fedd4ef3b
|
|||
e8b2e6f261
|
|||
39f3b0e01f | |||
edaf255e8f
|
|||
41c4ed4d0f | |||
f2bd88aadf
|
|||
67a3661448
|
|||
0c763a2dfd
|
|||
a7297ff933
|
|||
4cdba8bc77
|
|||
77c6303014
|
|||
2b641faa29 | |||
9fa8b93c08 | |||
4b676bc853 | |||
4433ddb1e1
|
|||
39aa7598b7
|
|||
19a290c3a9
|
|||
9bc80aac8a
|
|||
e184673963
|
|||
68cd746a9f
|
|||
69651d9f6c
|
|||
6fd246f43c | |||
ae14d6c74f | |||
2fa56b82d1
|
|||
9cc66eebdf
|
|||
4c10e20b91
|
|||
9217421221
|
|||
4570845b3e
|
|||
0e78951300
|
|||
6ad56b3126
|
|||
d95c6d3365
|
|||
1f2c8abb22
|
|||
a6d5693ccd
|
|||
31b258b4ce | |||
f19f2808d8 | |||
3b9cd2e1bb | |||
95320ca1bc | |||
f2d127fc98 | |||
eb526fb57f | |||
348fe52c42 | |||
eef0fa6952 | |||
8a82e059b7 | |||
2229cdf20d | |||
3220b194d4 | |||
278c4a6a41 | |||
ec50ac31c4 | |||
a2f0d814fc | |||
6468b35708 | |||
3558e99090 | |||
520608aef0 | |||
6df5f634f3 | |||
da266a8dd6 | |||
8ae4b85827 | |||
8fe3243693 | |||
49b174f29f | |||
30c6d3d8db | |||
6c14ed9c89 | |||
01ed51489e | |||
0636616dad | |||
34dbaaafe0 | |||
b4c31ee9b5 | |||
99307423c5 | |||
71542bc388 | |||
d64f470b60 | |||
b8fbb72fa0 | |||
0c61ff457d | |||
1d82f65b0d | |||
610988ec16 | |||
6e236ede14 | |||
b7ad5d3a31 | |||
a694ad225c | |||
5633e85f41 | |||
95e1eec313 | |||
377d5dadb2 | |||
4a294b1e17 | |||
720774fcf4 | |||
dcdbdd15ac | |||
132b48cf2a | |||
23bd432c5f | |||
71b33ab05b | |||
87f444c30d | |||
4a73eab134 | |||
f8baca5ab2 | |||
10221b9f2e | |||
1d8c8c8e9c | |||
4603a84f16 | |||
2cd8f3f7f3 | |||
107eeeae7f | |||
b8767b8bd4 | |||
bf686e89e0 |
188
.drone.yml
188
.drone.yml
@@ -1,188 +0,0 @@
|
|||||||
---
|
|
||||||
kind: secret
|
|
||||||
name: docker_username
|
|
||||||
get:
|
|
||||||
path: odit-registry-builder
|
|
||||||
name: username
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: secret
|
|
||||||
name: docker_password
|
|
||||||
get:
|
|
||||||
path: odit-registry-builder
|
|
||||||
name: password
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: secret
|
|
||||||
name: git_ssh
|
|
||||||
get:
|
|
||||||
path: odit-git-bot
|
|
||||||
name: sshkey
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: pipeline
|
|
||||||
type: kubernetes
|
|
||||||
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
|
|
||||||
- name: run tests
|
|
||||||
image: node:latest
|
|
||||||
commands:
|
|
||||||
- yarn
|
|
||||||
- yarn test:ci
|
|
||||||
trigger:
|
|
||||||
event:
|
|
||||||
- pull_request
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: pipeline
|
|
||||||
type: kubernetes
|
|
||||||
name: build:dev
|
|
||||||
clone:
|
|
||||||
disable: true
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: clone
|
|
||||||
image: alpine/git
|
|
||||||
commands:
|
|
||||||
- git clone $DRONE_REMOTE_URL .
|
|
||||||
- git checkout dev
|
|
||||||
- name: build dev
|
|
||||||
image: plugins/docker
|
|
||||||
depends_on: [clone]
|
|
||||||
settings:
|
|
||||||
username:
|
|
||||||
from_secret: docker_username
|
|
||||||
password:
|
|
||||||
from_secret: docker_password
|
|
||||||
repo: registry.odit.services/lfk/backend
|
|
||||||
tags:
|
|
||||||
- dev
|
|
||||||
registry: registry.odit.services
|
|
||||||
mtu: 1000
|
|
||||||
- name: run changelog export
|
|
||||||
depends_on: ["clone"]
|
|
||||||
image: node:latest
|
|
||||||
commands:
|
|
||||||
- npx auto-changelog --commit-limit false -p -u --hide-credit
|
|
||||||
- name: push new changelog to repo
|
|
||||||
depends_on: ["run changelog export"]
|
|
||||||
image: appleboy/drone-git-push
|
|
||||||
settings:
|
|
||||||
branch: dev
|
|
||||||
commit: true
|
|
||||||
commit_message: 🧾New changelog file version [CI SKIP] [skip ci]
|
|
||||||
author_email: bot@odit.services
|
|
||||||
remote: git@git.odit.services:lfk/backend.git
|
|
||||||
ssh_key:
|
|
||||||
from_secret: git_ssh
|
|
||||||
- name: run full license export
|
|
||||||
depends_on: ["clone"]
|
|
||||||
image: node:14.15.1-alpine3.12
|
|
||||||
commands:
|
|
||||||
- yarn
|
|
||||||
- yarn licenses:export
|
|
||||||
- name: push new licenses file to repo
|
|
||||||
depends_on: ["run full license export"]
|
|
||||||
image: appleboy/drone-git-push
|
|
||||||
settings:
|
|
||||||
branch: dev
|
|
||||||
commit: true
|
|
||||||
commit_message: 📖New license file version [CI SKIP] [skip ci]
|
|
||||||
author_email: bot@odit.services
|
|
||||||
remote: git@git.odit.services:lfk/backend.git
|
|
||||||
skip_verify: true
|
|
||||||
ssh_key:
|
|
||||||
from_secret: git_ssh
|
|
||||||
|
|
||||||
|
|
||||||
trigger:
|
|
||||||
branch:
|
|
||||||
- dev
|
|
||||||
event:
|
|
||||||
- push
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: pipeline
|
|
||||||
type: kubernetes
|
|
||||||
name: build:latest
|
|
||||||
clone:
|
|
||||||
disable: true
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: clone
|
|
||||||
image: alpine/git
|
|
||||||
commands:
|
|
||||||
- git clone $DRONE_REMOTE_URL .
|
|
||||||
- git checkout dev
|
|
||||||
- git merge main
|
|
||||||
- git checkout main
|
|
||||||
- name: build latest
|
|
||||||
depends_on: ["clone"]
|
|
||||||
image: plugins/docker
|
|
||||||
settings:
|
|
||||||
username:
|
|
||||||
from_secret: docker_username
|
|
||||||
password:
|
|
||||||
from_secret: docker_password
|
|
||||||
repo: registry.odit.services/lfk/backend
|
|
||||||
tags:
|
|
||||||
- latest
|
|
||||||
registry: registry.odit.services
|
|
||||||
mtu: 1000
|
|
||||||
- name: push merge to repo
|
|
||||||
depends_on: ["clone"]
|
|
||||||
image: appleboy/drone-git-push
|
|
||||||
settings:
|
|
||||||
branch: dev
|
|
||||||
commit: false
|
|
||||||
remote: git@git.odit.services:lfk/backend.git
|
|
||||||
ssh_key:
|
|
||||||
from_secret: git_ssh
|
|
||||||
|
|
||||||
trigger:
|
|
||||||
branch:
|
|
||||||
- main
|
|
||||||
event:
|
|
||||||
- push
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: pipeline
|
|
||||||
type: kubernetes
|
|
||||||
name: build:tags
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: build $DRONE_TAG
|
|
||||||
image: plugins/docker
|
|
||||||
depends_on: [clone]
|
|
||||||
settings:
|
|
||||||
username:
|
|
||||||
from_secret: docker_username
|
|
||||||
password:
|
|
||||||
from_secret: docker_password
|
|
||||||
repo: registry.odit.services/lfk/backend
|
|
||||||
tags:
|
|
||||||
- '${DRONE_TAG}'
|
|
||||||
registry: registry.odit.services
|
|
||||||
mtu: 1000
|
|
||||||
- name: trigger node lib build
|
|
||||||
image: idcooldi/drone-webhook
|
|
||||||
settings:
|
|
||||||
urls: https://ci.odit.services/api/repos/lfk/lfk-client-node/builds?SOURCE_TAG=${DRONE_TAG}
|
|
||||||
bearer:
|
|
||||||
from_secret: BOT_DRONE_KEY
|
|
||||||
- name: trigger js lib build
|
|
||||||
image: idcooldi/drone-webhook
|
|
||||||
settings:
|
|
||||||
urls: https://ci.odit.services/api/repos/lfk/lfk-client-js/builds?SOURCE_TAG=${DRONE_TAG}
|
|
||||||
bearer:
|
|
||||||
from_secret: BOT_DRONE_KEY
|
|
||||||
trigger:
|
|
||||||
event:
|
|
||||||
- tag
|
|
@@ -7,4 +7,5 @@ DB_PASSWORD=bla
|
|||||||
DB_NAME=./test.sqlite
|
DB_NAME=./test.sqlite
|
||||||
NODE_ENV=production
|
NODE_ENV=production
|
||||||
POSTALCODE_COUNTRYCODE=DE
|
POSTALCODE_COUNTRYCODE=DE
|
||||||
SEED_TEST_DATA=false
|
SEED_TEST_DATA=false
|
||||||
|
SELFSERVICE_URL=bla
|
2
.gitignore
vendored
2
.gitignore
vendored
@@ -135,4 +135,4 @@ build
|
|||||||
/docs
|
/docs
|
||||||
lib
|
lib
|
||||||
/oss-attribution
|
/oss-attribution
|
||||||
*.tmp
|
*.tmp
|
||||||
|
33
.woodpecker/build.yml
Normal file
33
.woodpecker/build.yml
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
steps:
|
||||||
|
- name: build latest
|
||||||
|
image: woodpeckerci/plugin-docker-buildx
|
||||||
|
settings:
|
||||||
|
repo: registry.odit.services/lfk/backend
|
||||||
|
tags:
|
||||||
|
- latest
|
||||||
|
registry: registry.odit.services
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
cache_from: registry.odit.services/lfk/backend:dev
|
||||||
|
username:
|
||||||
|
from_secret: odit-registry-builder-username
|
||||||
|
password:
|
||||||
|
from_secret: odit-registry-builder-password
|
||||||
|
when:
|
||||||
|
branch: main
|
||||||
|
- name: build dev
|
||||||
|
image: woodpeckerci/plugin-docker-buildx
|
||||||
|
settings:
|
||||||
|
repo: registry.odit.services/lfk/backend
|
||||||
|
tags:
|
||||||
|
- dev
|
||||||
|
registry: registry.odit.services
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
cache_from: registry.odit.services/lfk/backend:dev
|
||||||
|
username:
|
||||||
|
from_secret: odit-registry-builder-username
|
||||||
|
password:
|
||||||
|
from_secret: odit-registry-builder-password
|
||||||
|
when:
|
||||||
|
branch: dev
|
||||||
|
when:
|
||||||
|
event: push
|
17
.woodpecker/release.yml
Normal file
17
.woodpecker/release.yml
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
steps:
|
||||||
|
- name: build tag
|
||||||
|
image: woodpeckerci/plugin-docker-buildx
|
||||||
|
settings:
|
||||||
|
repo: registry.odit.services/lfk/backend
|
||||||
|
tags:
|
||||||
|
- "${CI_COMMIT_TAG}"
|
||||||
|
registry: registry.odit.services
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
cache_from: registry.odit.services/lfk/backend:dev
|
||||||
|
username:
|
||||||
|
from_secret: odit-registry-builder-username
|
||||||
|
password:
|
||||||
|
from_secret: odit-registry-builder-password
|
||||||
|
when:
|
||||||
|
event:
|
||||||
|
- tag
|
323
CHANGELOG.md
323
CHANGELOG.md
@@ -2,8 +2,331 @@
|
|||||||
|
|
||||||
All notable changes to this project will be documented in this file. Dates are displayed in UTC.
|
All notable changes to this project will be documented in this file. Dates are displayed in UTC.
|
||||||
|
|
||||||
|
#### [v1.1.4](https://git.odit.services/lfk/backend/compare/v1.1.3...v1.1.4)
|
||||||
|
|
||||||
|
- build: package lock [`50dd703`](https://git.odit.services/lfk/backend/commit/50dd703a1bd276a607cc10a087c7e90fd880847a)
|
||||||
|
- fix(deps): Bump sqlite3 [`cd3cd81`](https://git.odit.services/lfk/backend/commit/cd3cd81360777e8bc4d78e861354e58c8da79cc7)
|
||||||
|
- feat(ci)!: Switch to woodpecker [`3192365`](https://git.odit.services/lfk/backend/commit/3192365793fae59f2b89e3231db298654f0a28e9)
|
||||||
|
- fix(deps): Bumped argon2 to latest version for arm support [`cf48c00`](https://git.odit.services/lfk/backend/commit/cf48c00ddb2ac33263549876928db50ae152c12d)
|
||||||
|
- fix: updated README for pnpm, typos [`5082b1b`](https://git.odit.services/lfk/backend/commit/5082b1b8b1c0ae9e8ffa9c71c4d7923fd9223c87)
|
||||||
|
- ci: drop lfk-client-node [`075d484`](https://git.odit.services/lfk/backend/commit/075d484f1169bfc5c5b68cb9712116b0e270b471)
|
||||||
|
- fix(dependencies): Switch back to previous class-validator version to produce a working build [`74d334f`](https://git.odit.services/lfk/backend/commit/74d334f9b747a77115bd9b97729ef1120822e128)
|
||||||
|
|
||||||
|
#### [v1.1.3](https://git.odit.services/lfk/backend/compare/v1.1.2...v1.1.3)
|
||||||
|
|
||||||
|
> 10 May 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v1.1.3 [`057a8ee`](https://git.odit.services/lfk/backend/commit/057a8ee699d08c0e4a80cb50a8820f819569c9ac)
|
||||||
|
- feat(orgs): Also resolve child-teams' distances and add them to org total [`8d94186`](https://git.odit.services/lfk/backend/commit/8d9418635d3e381c0f55a2521a3334ba497c169a)
|
||||||
|
- fix(orgs): Removed unused log [`f2832a2`](https://git.odit.services/lfk/backend/commit/f2832a2daecc7bc7bbee4d4fceeab8db194730cf)
|
||||||
|
|
||||||
|
#### [v1.1.2](https://git.odit.services/lfk/backend/compare/v1.1.1...v1.1.2)
|
||||||
|
|
||||||
|
> 10 May 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v1.1.2 [`0d21596`](https://git.odit.services/lfk/backend/commit/0d21596e2b64a99258d4925ae2ad627d5cdbd984)
|
||||||
|
- feat(groups): Resolve the total group distance on group get single (aka get org and get team) [`245827e`](https://git.odit.services/lfk/backend/commit/245827e9c659cf76183dc33ab253becc22ddf032)
|
||||||
|
- chore(package): Formatting [`4608a36`](https://git.odit.services/lfk/backend/commit/4608a36df6b187520ca0c331b8dce615205257be)
|
||||||
|
|
||||||
|
#### [v1.1.1](https://git.odit.services/lfk/backend/compare/v1.1.0...v1.1.1)
|
||||||
|
|
||||||
|
> 19 April 2023
|
||||||
|
|
||||||
|
- feat(donors): Resolve donations with donors via pagination [`12a9ae2`](https://git.odit.services/lfk/backend/commit/12a9ae24933117acb3ff9815a7d72abca5eea7a7)
|
||||||
|
- 🚀Bumped version to v1.1.1 [`cb1305a`](https://git.odit.services/lfk/backend/commit/cb1305aa77c36aa9d7900f09e7413bc6d45f2c89)
|
||||||
|
|
||||||
|
#### [v1.1.0](https://git.odit.services/lfk/backend/compare/v1.0.1...v1.1.0)
|
||||||
|
|
||||||
|
> 19 April 2023
|
||||||
|
|
||||||
|
- feat(stats): Added donation count and donor count to stats [`6f39ac4`](https://git.odit.services/lfk/backend/commit/6f39ac42dafc2a589bbb2256b0417f3e774ae174)
|
||||||
|
- 🚀Bumped version to v1.1.0 [`b9fe9f1`](https://git.odit.services/lfk/backend/commit/b9fe9f1c24653b91255a6dbbdc32c30b1b411eeb)
|
||||||
|
- Added average donation per distance to stats [`fe59e3a`](https://git.odit.services/lfk/backend/commit/fe59e3a557903cf555d4c50098e935c49ca1fac4)
|
||||||
|
- Added hints [`b25b0db`](https://git.odit.services/lfk/backend/commit/b25b0db76071ef8d50cc60e950a399dc060a2a9f)
|
||||||
|
- Added calls to controller [`6ee5328`](https://git.odit.services/lfk/backend/commit/6ee5328dbc404603d19db3a5173ae4def560a9c9)
|
||||||
|
- Formatting [`42c23a5`](https://git.odit.services/lfk/backend/commit/42c23a5883dacda4e0147842d448b3ad35b197b1)
|
||||||
|
|
||||||
|
#### [v1.0.1](https://git.odit.services/lfk/backend/compare/v1.0.0...v1.0.1)
|
||||||
|
|
||||||
|
> 18 April 2023
|
||||||
|
|
||||||
|
- fix(pagination) page=0 resulted in false thx JS [`fcee390`](https://git.odit.services/lfk/backend/commit/fcee3909f4c4664115cc7ecb94f30e0dd8e78ce0)
|
||||||
|
- 🚀Bumped version to v1.0.1 [`301f334`](https://git.odit.services/lfk/backend/commit/301f33467489a8533bdac11fbd10efd1b791f5e3)
|
||||||
|
|
||||||
|
### [v1.0.0](https://git.odit.services/lfk/backend/compare/v0.15.4...v1.0.0)
|
||||||
|
|
||||||
|
> 18 April 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v1.0.0 [`f0e20e4`](https://git.odit.services/lfk/backend/commit/f0e20e413014fe446c97754d2765cdad92c2cc3b)
|
||||||
|
- Merge pull request 'feature/205-pagination' (#206) from feature/205-pagination into dev [`80de188`](https://git.odit.services/lfk/backend/commit/80de188565523d642407612272432ef07672b890)
|
||||||
|
- Added pagination for runner orgs [`538622a`](https://git.odit.services/lfk/backend/commit/538622aa1841e27256f304e15b4204c2f6d24d76)
|
||||||
|
- RunnerTeam Pagination [`0fa663a`](https://git.odit.services/lfk/backend/commit/0fa663a34104d438dd8fc9ab02458fdf289329f8)
|
||||||
|
- users pagination [`244da61`](https://git.odit.services/lfk/backend/commit/244da618926377f58bb12dbbd89b7bb39d84596e)
|
||||||
|
- Track pagination [`2a72aea`](https://git.odit.services/lfk/backend/commit/2a72aea10ef940fbdd4a9e6137b22933fdec7734)
|
||||||
|
- usergroup pagination [`513d7f6`](https://git.odit.services/lfk/backend/commit/513d7f6fbaebe39beab6ec95e6e42eb10c62296d)
|
||||||
|
- statsclient pagination [`71ebce6`](https://git.odit.services/lfk/backend/commit/71ebce6f8eebf110bb973a53b91dd6a49e1def99)
|
||||||
|
- scanstation pagination [`f60025b`](https://git.odit.services/lfk/backend/commit/f60025b6de79b0f5f89995bf59260194f5de9af0)
|
||||||
|
- Get all pagination for permissions [`86a21db`](https://git.odit.services/lfk/backend/commit/86a21dbfa4b50d8e80c611ea6e3eabfc2b8ae365)
|
||||||
|
- Pagination for group contacts [`1e9e24d`](https://git.odit.services/lfk/backend/commit/1e9e24d99d75ce6dc846ff662e62c886646ea974)
|
||||||
|
- Added pagination for get all donors [`4493c0e`](https://git.odit.services/lfk/backend/commit/4493c0e3d9beebbf7f601b39e1a2579771b4d152)
|
||||||
|
- Added pagination for donations [`f5d48fc`](https://git.odit.services/lfk/backend/commit/f5d48fc638080c9333efe474d86f131794c809af)
|
||||||
|
- Added pagination for runnercards [`b35a2dd`](https://git.odit.services/lfk/backend/commit/b35a2dd2fab708253373b3326f11ab574be18371)
|
||||||
|
- Added pagination for runners [`d873674`](https://git.odit.services/lfk/backend/commit/d873674819e6cb33cf89da4f8fdc30a0b41707e4)
|
||||||
|
- Added pagination for get all scans [`37b2ac9`](https://git.odit.services/lfk/backend/commit/37b2ac974b2276efd13538c127ba5ddda2537fe3)
|
||||||
|
- Updated test for attribute [`2f305e1`](https://git.odit.services/lfk/backend/commit/2f305e127c75e9e6ff8e9fc0cfc10cc3db44759d)
|
||||||
|
- Formatting [`a28ffe0`](https://git.odit.services/lfk/backend/commit/a28ffe06e5f3f69e4af6fdf0c66c9a1dfda10cfa)
|
||||||
|
|
||||||
|
#### [v0.15.4](https://git.odit.services/lfk/backend/compare/v0.15.3...v0.15.4)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- Fixed possible null [`0f0c3c7`](https://git.odit.services/lfk/backend/commit/0f0c3c7214f357d991518aafd015ffc4d387ce59)
|
||||||
|
- 🚀Bumped version to v0.15.4 [`81aed1d`](https://git.odit.services/lfk/backend/commit/81aed1de40166f4cefabdb478d7638017127b25c)
|
||||||
|
|
||||||
|
#### [v0.15.3](https://git.odit.services/lfk/backend/compare/v0.15.2...v0.15.3)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- Faster stats (not including donations) [`b2ac70e`](https://git.odit.services/lfk/backend/commit/b2ac70e0aec1064e54a5043a104e7892984b2338)
|
||||||
|
- 🚀Bumped version to v0.15.3 [`3909ed3`](https://git.odit.services/lfk/backend/commit/3909ed34f739e9fee90828f16757c75da90bab0f)
|
||||||
|
|
||||||
|
#### [v0.15.2](https://git.odit.services/lfk/backend/compare/v0.15.1...v0.15.2)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.15.2 [`5f17e7f`](https://git.odit.services/lfk/backend/commit/5f17e7f783a7e8e2efc8f7dbbf2c98bcd1d80240)
|
||||||
|
- Don't resolve runner group and parten with get all card requests [`2d8f752`](https://git.odit.services/lfk/backend/commit/2d8f7528d98144832e7609f5aa6fac8de4723c4a)
|
||||||
|
- Resolve groups again for card generation [`a5a56a2`](https://git.odit.services/lfk/backend/commit/a5a56a263a01dbd911a799ab57084166e17b80ac)
|
||||||
|
|
||||||
|
#### [v0.15.1](https://git.odit.services/lfk/backend/compare/v0.15.0...v0.15.1)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.15.1 [`9581185`](https://git.odit.services/lfk/backend/commit/9581185b24039338e7f238ecdcc3881bb5203759)
|
||||||
|
- Faster trackscan creation by only loading the latest scan [`e9914e3`](https://git.odit.services/lfk/backend/commit/e9914e317b7fd78863cfd8549bad65da9292b7ca)
|
||||||
|
- Log batch time in mass scan script [`2905884`](https://git.odit.services/lfk/backend/commit/2905884c024d7f275b3ad2c2858a2f0911adb95b)
|
||||||
|
- Dont load cards with get all runners request [`702070d`](https://git.odit.services/lfk/backend/commit/702070da669cc605b93e6f5b62d712c28f079dd0)
|
||||||
|
|
||||||
|
#### [v0.15.0](https://git.odit.services/lfk/backend/compare/v0.14.6...v0.15.0)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- Added test script for creating mass scans [`8007117`](https://git.odit.services/lfk/backend/commit/80071174342d87199fcbd981cd8c92300b0a51e4)
|
||||||
|
- 🚀Bumped version to v0.15.0 [`cc89ba8`](https://git.odit.services/lfk/backend/commit/cc89ba8afb3120569613a889baf962555612e95a)
|
||||||
|
- Get all scans speed improvement [`23fa78e`](https://git.odit.services/lfk/backend/commit/23fa78eb9dcc01ecc036347f6703aacc0d163d7d)
|
||||||
|
- More scan request optimizations [`7c4ff42`](https://git.odit.services/lfk/backend/commit/7c4ff42a3b3e7b186e16c85a97d9ecc854a32cb0)
|
||||||
|
|
||||||
|
#### [v0.14.6](https://git.odit.services/lfk/backend/compare/v0.14.5...v0.14.6)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.14.6 [`3b3e689`](https://git.odit.services/lfk/backend/commit/3b3e68900beca16cfff88dbef22540f77750d29b)
|
||||||
|
- Missing orm file [`3ff666f`](https://git.odit.services/lfk/backend/commit/3ff666fd3e84ac8cf41b30e9e17082b10548d55b)
|
||||||
|
|
||||||
|
#### [v0.14.5](https://git.odit.services/lfk/backend/compare/v0.14.4...v0.14.5)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.14.5 [`4e44350`](https://git.odit.services/lfk/backend/commit/4e4435010fd7095e3b9742e207cba1b68cd6da3b)
|
||||||
|
- Entrypoint fix [`de9af5a`](https://git.odit.services/lfk/backend/commit/de9af5a90907dcfc9bfb1d5a56420eed8bb59922)
|
||||||
|
- Fixed copy [`ac631f0`](https://git.odit.services/lfk/backend/commit/ac631f0af467446552478873b7b4802a9310f865)
|
||||||
|
|
||||||
|
#### [v0.14.4](https://git.odit.services/lfk/backend/compare/v0.14.3...v0.14.4)
|
||||||
|
|
||||||
|
> 15 April 2023
|
||||||
|
|
||||||
|
- Switched ci over to pnpm + cache [`6275aaa`](https://git.odit.services/lfk/backend/commit/6275aaa326f1c02c8dd42aa31608978408c44ab7)
|
||||||
|
- 🚀Bumped version to v0.14.4 [`6bbdd5b`](https://git.odit.services/lfk/backend/commit/6bbdd5bb04a1c38e4b3a150db24b76e9c96490dd)
|
||||||
|
- Back to ean13 based codes [`a8fc755`](https://git.odit.services/lfk/backend/commit/a8fc7558408b97da4b2c469ae5e73ab502b4fda0)
|
||||||
|
- install prod in first step [`d027439`](https://git.odit.services/lfk/backend/commit/d02743984dfea8057be3081bd3a32a8f67e610aa)
|
||||||
|
- Switched dockerfile to pnpm 8 with cache [`93d43b7`](https://git.odit.services/lfk/backend/commit/93d43b76843d7cb411f37fd2066c6a5364c05415)
|
||||||
|
- COPY by stage name [`a64f6c9`](https://git.odit.services/lfk/backend/commit/a64f6c9822af2b927e91b0b55f1f50176de30169)
|
||||||
|
- pinned pnpm version [`2a94bfa`](https://git.odit.services/lfk/backend/commit/2a94bfa6227d14f635b5fc2789b59c36d490937e)
|
||||||
|
- custom pnpm cache [`85dc344`](https://git.odit.services/lfk/backend/commit/85dc3444acc677ddd242f9f2543ce477fe427a7c)
|
||||||
|
- added missing ci env [`734c826`](https://git.odit.services/lfk/backend/commit/734c826face58dd5c3bb2607bda6e7f6d051012e)
|
||||||
|
- pinned pnpm to 8 [`27e74e8`](https://git.odit.services/lfk/backend/commit/27e74e824cd1e23d4d53c1a983a1668dd87f5d59)
|
||||||
|
- coherent baseimage [`b5c0a28`](https://git.odit.services/lfk/backend/commit/b5c0a288ac3c020f5d753c558aee160fea0bae14)
|
||||||
|
- bumped final pnpm version [`33b25c9`](https://git.odit.services/lfk/backend/commit/33b25c9743abb7cefb3538f08cc2f78a646905c8)
|
||||||
|
|
||||||
|
#### [v0.14.3](https://git.odit.services/lfk/backend/compare/v0.14.2...v0.14.3)
|
||||||
|
|
||||||
|
> 18 March 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.14.3 [`16ce0a8`](https://git.odit.services/lfk/backend/commit/16ce0a848050b74c4b6dd93f17e5a6e9024cdb7d)
|
||||||
|
- Adjusted modulo for new fixed card length [`9a8d618`](https://git.odit.services/lfk/backend/commit/9a8d618ae4584640e8be1ce9fe4bddd2ef7a92ae)
|
||||||
|
|
||||||
|
#### [v0.14.2](https://git.odit.services/lfk/backend/compare/v0.14.1...v0.14.2)
|
||||||
|
|
||||||
|
> 18 March 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.14.2 [`38da2d3`](https://git.odit.services/lfk/backend/commit/38da2d33187f4b24eef878642e153663ecd95de1)
|
||||||
|
- Back to modulo [`068deb4`](https://git.odit.services/lfk/backend/commit/068deb4960bd16decf99887ffbda7a7d3dd9ff0b)
|
||||||
|
|
||||||
|
#### [v0.14.1](https://git.odit.services/lfk/backend/compare/v0.14.0...v0.14.1)
|
||||||
|
|
||||||
|
> 18 March 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.14.1 [`13f093b`](https://git.odit.services/lfk/backend/commit/13f093bb6138a498f93a05ef6dd812ae92f2676a)
|
||||||
|
- Switched from card prefix replacement via modulo to regex [`6289f30`](https://git.odit.services/lfk/backend/commit/6289f307400aacaa9cfe03f3024c1e0d5554d4f2)
|
||||||
|
|
||||||
|
#### [v0.14.0](https://git.odit.services/lfk/backend/compare/v0.13.3...v0.14.0)
|
||||||
|
|
||||||
|
> 15 March 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.14.0 [`6ff764b`](https://git.odit.services/lfk/backend/commit/6ff764bc340ca25b3bdd62c6892259e228723973)
|
||||||
|
- Updated default length [`ea87cc7`](https://git.odit.services/lfk/backend/commit/ea87cc793b163bf0d4405a25bbe83fbc8e31c206)
|
||||||
|
- breaking(runnercards): shorter runnercard codes (padding to 12 was a bit tooo ambitious) [`ffee887`](https://git.odit.services/lfk/backend/commit/ffee887ddf6a71102ee39533d7cd504d1fd6698f)
|
||||||
|
- Removed sqlite journal [`92517e3`](https://git.odit.services/lfk/backend/commit/92517e365393f4baac3814f5668874b5752dc7c8)
|
||||||
|
|
||||||
|
#### [v0.13.3](https://git.odit.services/lfk/backend/compare/v0.13.2...v0.13.3)
|
||||||
|
|
||||||
|
> 15 February 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.13.3 [`3bac75e`](https://git.odit.services/lfk/backend/commit/3bac75e7ab9f16ecab1fbfa9915a7edb923883f6)
|
||||||
|
- Merge pull request 'feature/201-no_citizen-deletion' (#202) from feature/201-no_citizen-deletion into dev [`d05eddc`](https://git.odit.services/lfk/backend/commit/d05eddcae198427ce9a334096563b3aadcff2b56)
|
||||||
|
- Updated tests [`d5c689d`](https://git.odit.services/lfk/backend/commit/d5c689d6937288df7dca14ce26fbbd4f46a8752a)
|
||||||
|
- Added delete check for citizen org [`8fedd4e`](https://git.odit.services/lfk/backend/commit/8fedd4ef3bdd48dc42abc1d53006eefc145175e3)
|
||||||
|
|
||||||
|
#### [v0.13.2](https://git.odit.services/lfk/backend/compare/v0.13.1...v0.13.2)
|
||||||
|
|
||||||
|
> 3 February 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.13.2 [`e8b2e6f`](https://git.odit.services/lfk/backend/commit/e8b2e6f26140a18c06b017e4461742d7e7942f08)
|
||||||
|
- Merge pull request 'move selfservice magic link endpoint to 15min rate limit' (#200) from feature/runner-selfservice-login-link-rate-limit into dev [`39f3b0e`](https://git.odit.services/lfk/backend/commit/39f3b0e01f03bfbcfcb0ea08d697268ce068e63d)
|
||||||
|
- move to 15min limit [`edaf255`](https://git.odit.services/lfk/backend/commit/edaf255e8f609185dcd6c2c0cd2e8b007b785e0c)
|
||||||
|
- Merge pull request 'Releases 0.12.0 and 0.13.0' (#199) from dev into main [`41c4ed4`](https://git.odit.services/lfk/backend/commit/41c4ed4d0faaed382801bbe480f31dafa6f3912d)
|
||||||
|
|
||||||
|
#### [v0.13.1](https://git.odit.services/lfk/backend/compare/v0.13.0...v0.13.1)
|
||||||
|
|
||||||
|
> 2 February 2023
|
||||||
|
|
||||||
|
- 🚀Bumped version to v0.13.1 [`f2bd88a`](https://git.odit.services/lfk/backend/commit/f2bd88aadfcb6ffa0485ea6afac8c7664a37f5f4)
|
||||||
|
- Updated description [`67a3661`](https://git.odit.services/lfk/backend/commit/67a36614485b2ea83c2de41e0684708b95a05b32)
|
||||||
|
|
||||||
|
#### [v0.13.0](https://git.odit.services/lfk/backend/compare/v0.12.0...v0.13.0)
|
||||||
|
|
||||||
|
> 2 February 2023
|
||||||
|
|
||||||
|
- Added faker for testing [`e184673`](https://git.odit.services/lfk/backend/commit/e1846739638905aab6ba7e059fd2cbf8ff467bf3)
|
||||||
|
- 📖New license file version [CI SKIP] [skip ci] [`2b641fa`](https://git.odit.services/lfk/backend/commit/2b641faa29c47d95f69983770dc4ab37e674604f)
|
||||||
|
- 🚀Bumped version to v0.13.0 [`0c763a2`](https://git.odit.services/lfk/backend/commit/0c763a2dfd39607b480d9aff7d3c883791f41700)
|
||||||
|
- Updated selfservice tests to prevent email duplication [`9bc80aa`](https://git.odit.services/lfk/backend/commit/9bc80aac8aab9b4dedc26c9bc3ce705d7fe9c0bf)
|
||||||
|
- Moved license and changelog export to releaseit hooks [`77c6303`](https://git.odit.services/lfk/backend/commit/77c6303014578edbbadeeaa790f7974bde2a9764)
|
||||||
|
- Updated readme [`4cdba8b`](https://git.odit.services/lfk/backend/commit/4cdba8bc77ce543f6fb636711b8728bce794eac7)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`ae14d6c`](https://git.odit.services/lfk/backend/commit/ae14d6c74f9205440b41ca5fdbd052ca449148fc)
|
||||||
|
- Added selfservice runner create check to prevent duplicate email [`68cd746`](https://git.odit.services/lfk/backend/commit/68cd746a9f3360b3630a9ba570213d2aa62497b4)
|
||||||
|
- Updated tests for new login in selfservice [`39aa759`](https://git.odit.services/lfk/backend/commit/39aa7598b7cd0ecb0f077f50ebdd31c6e205f06d)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`9fa8b93`](https://git.odit.services/lfk/backend/commit/9fa8b93c08ee52335b18e743f9d205b19e6095c6)
|
||||||
|
- Moved changelog generation to package script [`a7297ff`](https://git.odit.services/lfk/backend/commit/a7297ff933ae1372a9d508cdae1a54d2ebbcc647)
|
||||||
|
- Merge pull request 'feature/197-duplicate_runner_mail' (#198) from feature/197-duplicate_runner_mail into dev [`4b676bc`](https://git.odit.services/lfk/backend/commit/4b676bc85336c2d494e9e74823d38deec5cc0400)
|
||||||
|
- Updated logo url [`4433ddb`](https://git.odit.services/lfk/backend/commit/4433ddb1e15a35481728670e22049200644bf337)
|
||||||
|
- depends_on: ["clone"] [`9cc66ee`](https://git.odit.services/lfk/backend/commit/9cc66eebdfe8e7a2888bbc97197d1756ff44de30)
|
||||||
|
- Fixed typo [`19a290c`](https://git.odit.services/lfk/backend/commit/19a290c3a931ead0d9ae9ebb0985bfbaac54df59)
|
||||||
|
- Rename selfservice forgot to login [`69651d9`](https://git.odit.services/lfk/backend/commit/69651d9f6cd826b6d4720f164897a2a72a57c851)
|
||||||
|
- 📖New license file version [CI SKIP] [skip ci] [`6fd246f`](https://git.odit.services/lfk/backend/commit/6fd246f43cb3f4d0ccb6e017ee699889ba17daac)
|
||||||
|
- Add git for changelog fun [`2fa56b8`](https://git.odit.services/lfk/backend/commit/2fa56b82d1e082a1deae943e5fca5101f24e3ef5)
|
||||||
|
|
||||||
|
#### [v0.12.0](https://git.odit.services/lfk/backend/compare/v0.11.1...v0.12.0)
|
||||||
|
|
||||||
|
> 2 February 2023
|
||||||
|
|
||||||
|
- Pinned versions [`a6d5693`](https://git.odit.services/lfk/backend/commit/a6d5693ccdeb25b15a09af8f7438142114268807)
|
||||||
|
- Drone -> Kaniko based builds [`0e78951`](https://git.odit.services/lfk/backend/commit/0e789513008085d0db94fc3b2dd9e74a5e583049)
|
||||||
|
- Drone images to odit registry [`6ad56b3`](https://git.odit.services/lfk/backend/commit/6ad56b31269bf19a740c1b6b1a303a8a9d7d59d0)
|
||||||
|
- Bumped container base images [`d95c6d3`](https://git.odit.services/lfk/backend/commit/d95c6d33657f6aa977a8ebfefad7e199bb1cc9c3)
|
||||||
|
- Enabled tag via release script [`9217421`](https://git.odit.services/lfk/backend/commit/92174212213f874e41c9472a927bcf87b963ac94)
|
||||||
|
- Pinned pnpm for builds [`4570845`](https://git.odit.services/lfk/backend/commit/4570845b3e1bd00c228fe1b09b658c24e20aba7f)
|
||||||
|
- 🚀Bumped version to v0.12.0 [`4c10e20`](https://git.odit.services/lfk/backend/commit/4c10e20b91a8101ee37b230373ceb3e024582b41)
|
||||||
|
- Ignore pnpm lock [`1f2c8ab`](https://git.odit.services/lfk/backend/commit/1f2c8abb22f3ff1e61b7350b517bd699c3e315f6)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`31b258b`](https://git.odit.services/lfk/backend/commit/31b258b4ce82213144160a4233b7fd127e456776)
|
||||||
|
|
||||||
|
#### [v0.11.1](https://git.odit.services/lfk/backend/compare/v0.11.0...v0.11.1)
|
||||||
|
|
||||||
|
> 22 April 2021
|
||||||
|
|
||||||
|
- Merge pull request 'Release 0.11.1' (#196) from dev into main [`f19f280`](https://git.odit.services/lfk/backend/commit/f19f2808d88414f1877c01f10996dac68b6f9617)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`2229cdf`](https://git.odit.services/lfk/backend/commit/2229cdf20db1a98f9f76a99fa9d3f463cdf6d804)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`348fe52`](https://git.odit.services/lfk/backend/commit/348fe52c42cfa32239b703041820f725e147154e)
|
||||||
|
- Now prefixing runnercards with 2 [`8a82e05`](https://git.odit.services/lfk/backend/commit/8a82e059b74ceabf43c9cbfe9c9b89ef6ce15a28)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`3b9cd2e`](https://git.odit.services/lfk/backend/commit/3b9cd2e1bbbe8e69c3883233a98f286d768c2b79)
|
||||||
|
- Added fix for the appended 2 [`eb526fb`](https://git.odit.services/lfk/backend/commit/eb526fb57faf631fd6e84af99af738ab1b3481c7)
|
||||||
|
- 🚀Bumped version to v0.11.1 [`95320ca`](https://git.odit.services/lfk/backend/commit/95320ca1bccc2886553accea6a428aadffda0a27)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`f2d127f`](https://git.odit.services/lfk/backend/commit/f2d127fc98d75ce658424624abd382c087737ca0)
|
||||||
|
|
||||||
|
#### [v0.11.0](https://git.odit.services/lfk/backend/compare/v0.10.2...v0.11.0)
|
||||||
|
|
||||||
|
> 14 April 2021
|
||||||
|
|
||||||
|
- Merge pull request 'Release 0.11.0' (#195) from dev into main [`3220b19`](https://git.odit.services/lfk/backend/commit/3220b194d4c704835d6d106ec4d9d54a17a38b62)
|
||||||
|
- Fixed spelling [`da266a8`](https://git.odit.services/lfk/backend/commit/da266a8dd68dbb575997ae343624982b690486ec)
|
||||||
|
- Updated tests [`01ed514`](https://git.odit.services/lfk/backend/commit/01ed51489eb92fff907d46a930ecf0b0eb5cad2b)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`520608a`](https://git.odit.services/lfk/backend/commit/520608aef05b21f4daadf55cfc8caddba06b8f01)
|
||||||
|
- Added payedDonationAmount to donor and responsedonor [`8ae4b85`](https://git.odit.services/lfk/backend/commit/8ae4b8582749332f4fb081eee0c520293347001f)
|
||||||
|
- Responses now contain the donation status [`34dbaaa`](https://git.odit.services/lfk/backend/commit/34dbaaafe0422234848eabe3f52b26879c9e5a49)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`278c4a6`](https://git.odit.services/lfk/backend/commit/278c4a6a415434487a92ff66f8114bb2547aac48)
|
||||||
|
- Marked payedAmount as optional during creation and/or update [`0636616`](https://git.odit.services/lfk/backend/commit/0636616dad5afb41ffe47a857d91ac75b4f2f20a)
|
||||||
|
- Added payed amount fileld to donation class [`b8fbb72`](https://git.odit.services/lfk/backend/commit/b8fbb72fa0b659c9acc406c72a8a59c2174351b4)
|
||||||
|
- Added status to tests [`30c6d3d`](https://git.odit.services/lfk/backend/commit/30c6d3d8db9fe37a51e596a73add8b87e8616e54)
|
||||||
|
- Added payed amount to response class [`d64f470`](https://git.odit.services/lfk/backend/commit/d64f470b608b3f179ec77da0210de51c328ef3f2)
|
||||||
|
- 📖New license file version [CI SKIP] [skip ci] [`a2f0d81`](https://git.odit.services/lfk/backend/commit/a2f0d814fc782ad440500e7d6ec779b6ab7f0ac6)
|
||||||
|
- 🚀Bumped version to v0.11.0 [`3558e99`](https://git.odit.services/lfk/backend/commit/3558e9909088647bd4f1f4334f50c07a5ef00214)
|
||||||
|
- Merge pull request 'Donation payment management feature/193-donation_payments' (#194) from feature/193-donation_payments into dev [`6df5f63`](https://git.odit.services/lfk/backend/commit/6df5f634f3123e04c015889573ccc5674a8bab27)
|
||||||
|
- Added payed amount to crealte classes [`71542bc`](https://git.odit.services/lfk/backend/commit/71542bc3887b97c15436d03280e49f7b3f0fcb06)
|
||||||
|
- Added donation status enum [`b4c31ee`](https://git.odit.services/lfk/backend/commit/b4c31ee9b5b35d6e11b07f50f3d30ca12e0f7728)
|
||||||
|
- Added payed amount to update classes [`9930742`](https://git.odit.services/lfk/backend/commit/99307423c533f8cde847b59a80bffc2ff42c9769)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`0c61ff4`](https://git.odit.services/lfk/backend/commit/0c61ff457d02f750efa457dd75464187683b037a)
|
||||||
|
- Added mssing check to tests [`6c14ed9`](https://git.odit.services/lfk/backend/commit/6c14ed9c89eadc1a10db8c912d8ea2711a518766)
|
||||||
|
- No longer answering with null, but 0 [`49b174f`](https://git.odit.services/lfk/backend/commit/49b174f29f63e963e600d74b6923a20211d832eb)
|
||||||
|
- Saved missing file [`8fe3243`](https://git.odit.services/lfk/backend/commit/8fe32436935d7cd6c17eae1e138383d3b714e1ba)
|
||||||
|
|
||||||
|
#### [v0.10.2](https://git.odit.services/lfk/backend/compare/v0.10.1...v0.10.2)
|
||||||
|
|
||||||
|
> 7 April 2021
|
||||||
|
|
||||||
|
- Merge pull request 'Release 0.10.2' (#192) from dev into main [`1d82f65`](https://git.odit.services/lfk/backend/commit/1d82f65b0d3a32d10c1a10c991353c18696d58bf)
|
||||||
|
- Added first selfservice test [`057ae0d`](https://git.odit.services/lfk/backend/commit/057ae0d79758cd627d6d128406a0d201b6b7ad9b)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`b7ad5d3`](https://git.odit.services/lfk/backend/commit/b7ad5d3a31b8b4f5960852d3ac38af133719ebcd)
|
||||||
|
- First try of the laptime sort [`4471e57`](https://git.odit.services/lfk/backend/commit/4471e57438582d55ff846fd69c2cfcc26b40df2a)
|
||||||
|
- Potential fix for all remaining errors [`377d5da`](https://git.odit.services/lfk/backend/commit/377d5dadb2a14cb2d70e0b2dc77026f51b3fb51c)
|
||||||
|
- At least one fewer test should fail now [`87f444c`](https://git.odit.services/lfk/backend/commit/87f444c30d69d65a9f918c63631a859a389eeee3)
|
||||||
|
- Tried workaround for no availdable stats [`8f0f795`](https://git.odit.services/lfk/backend/commit/8f0f795a709db216396998b68b8bbd64ff4d44ff)
|
||||||
|
- Reverted temp bugfix [`4603a84`](https://git.odit.services/lfk/backend/commit/4603a84f16fb53a14d1792447100f5b470969dd0)
|
||||||
|
- Fixed sorting algo [`988f17a`](https://git.odit.services/lfk/backend/commit/988f17a795bb2d867e9d1d8e78051dff1a14ec30)
|
||||||
|
- Added runners stats tests [`7111068`](https://git.odit.services/lfk/backend/commit/7111068361e00cc1308664a3ae650a56e28c015c)
|
||||||
|
- Added basic laptime endpoint [`cb71fcd`](https://git.odit.services/lfk/backend/commit/cb71fcd13bc61e6214e2fd7b70e72094749463d3)
|
||||||
|
- Added orgs by donations stats tests [`d4a02e7`](https://git.odit.services/lfk/backend/commit/d4a02e7db2ff4976be21605e31aac2f3c82a49c0)
|
||||||
|
- Added teams stats endpoint tests [`b9a7dc8`](https://git.odit.services/lfk/backend/commit/b9a7dc84f05441445453193974b2a793b5197fa5)
|
||||||
|
- Now resolving all missing relations [`257f320`](https://git.odit.services/lfk/backend/commit/257f320ee3bf6429c4314c64023520366f9f730b)
|
||||||
|
- Added min laptime to StatsRunner [`51daf96`](https://git.odit.services/lfk/backend/commit/51daf969cf74792b2c2f2f16ce4359d9fca47bc8)
|
||||||
|
- Fixed sorting [`7b15c2d`](https://git.odit.services/lfk/backend/commit/7b15c2d88b14e7279aad97b0c950202ddb5acaaa)
|
||||||
|
- Fixed top-ten bein top 9 [`a6a526d`](https://git.odit.services/lfk/backend/commit/a6a526dc5d8b1613ea34e82e477081349e764aec)
|
||||||
|
- added new ci secret [`5633e85`](https://git.odit.services/lfk/backend/commit/5633e85f41cb69b10fd8a86f57f1bd2f50848f7b)
|
||||||
|
- Added temp console log for test [`22cae39`](https://git.odit.services/lfk/backend/commit/22cae39bd351ca285880e50187ea0d46a7a26437)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`610988e`](https://git.odit.services/lfk/backend/commit/610988ec16b8df61cca61cf2252a469d30318d81)
|
||||||
|
- Added temp console log for ci debugging [`4a73eab`](https://git.odit.services/lfk/backend/commit/4a73eab134c3a9f58771be996bc8811b62cf378e)
|
||||||
|
- Temp disabled runners by donations test [`0b07a53`](https://git.odit.services/lfk/backend/commit/0b07a53ed209c6193ead3c4d199545e22333ab32)
|
||||||
|
- Updated default docker-compose [`f8baca5`](https://git.odit.services/lfk/backend/commit/f8baca5ab2c56b906751bc7edd71477456ad91f3)
|
||||||
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`dd6d799`](https://git.odit.services/lfk/backend/commit/dd6d799c847fc96aec1be8f2667ad371890076fb)
|
||||||
|
- Resolved missing parentgroup relation [`23bd432`](https://git.odit.services/lfk/backend/commit/23bd432c5f33a0863217120d97e2e4ea52a08baf)
|
||||||
|
- Removed console logs for now working tests [`71b33ab`](https://git.odit.services/lfk/backend/commit/71b33ab05b53b62c8b271bd2995c94b2fc212dfd)
|
||||||
|
- Fixed typo in test [`cbcb829`](https://git.odit.services/lfk/backend/commit/cbcb829fbde3a4a5e7f94de5dcf24d854c5fc257)
|
||||||
|
- Ptotential fix for stats failing [`dcdbdd1`](https://git.odit.services/lfk/backend/commit/dcdbdd15acfe6eef4220b7ed66db60d78107d1f9)
|
||||||
|
- 🚀Bumped version to v0.10.2 [`6e236ed`](https://git.odit.services/lfk/backend/commit/6e236ede145e164ee84543fb62404b4776550973)
|
||||||
|
- Merge pull request 'stats/runners/laptime feature/190-runners_laptime' (#191) from feature/190-runners_laptime into dev [`a694ad2`](https://git.odit.services/lfk/backend/commit/a694ad225c68fa23152402acba871c857433cc70)
|
||||||
|
- Removed all useless console.logs [`95e1eec`](https://git.odit.services/lfk/backend/commit/95e1eec313a79458dd75307a9d0f8319af0d0904)
|
||||||
|
- Pinned testing container tag to prod container tag [`10221b9`](https://git.odit.services/lfk/backend/commit/10221b9f2e4493080f3ff095d9772bcfd0ac50eb)
|
||||||
|
- Now resolving all relations for orgs by distance [`4a294b1`](https://git.odit.services/lfk/backend/commit/4a294b1e17c44294274b06748ec8141812c2d217)
|
||||||
|
- Added temp console log [`720774f`](https://git.odit.services/lfk/backend/commit/720774fcf47c38601ab88d5d74cfcd0e47b21acf)
|
||||||
|
- Removed console log for passing tests [`132b48c`](https://git.odit.services/lfk/backend/commit/132b48cf2a9e990a5e830c744ed8244bd25e8b3a)
|
||||||
|
- Removed console log [`1d8c8c8`](https://git.odit.services/lfk/backend/commit/1d8c8c8e9cefa58449f7abb2481d9396fe37ba20)
|
||||||
|
- Temp test logging workaround [`bf686e8`](https://git.odit.services/lfk/backend/commit/bf686e89e02998ccc80c838ef890c736c252634c)
|
||||||
|
- Temp test logging workaround [`6163f0a`](https://git.odit.services/lfk/backend/commit/6163f0a90b3721d3a1488f89cbb39ddff7152241)
|
||||||
|
- Removed test for content type [`63964fb`](https://git.odit.services/lfk/backend/commit/63964fbf2c41d9b90f995f056e9db65ab07d54a8)
|
||||||
|
|
||||||
#### [v0.10.1](https://git.odit.services/lfk/backend/compare/v0.10.0...v0.10.1)
|
#### [v0.10.1](https://git.odit.services/lfk/backend/compare/v0.10.0...v0.10.1)
|
||||||
|
|
||||||
|
> 3 April 2021
|
||||||
|
|
||||||
- Merge pull request 'Release 0.10.1' (#189) from dev into main [`e89e07d`](https://git.odit.services/lfk/backend/commit/e89e07d0fc99f14148b01204fb8ed39e2da77e38)
|
- Merge pull request 'Release 0.10.1' (#189) from dev into main [`e89e07d`](https://git.odit.services/lfk/backend/commit/e89e07d0fc99f14148b01204fb8ed39e2da77e38)
|
||||||
- 🧾New changelog file version [CI SKIP] [skip ci] [`69afd4d`](https://git.odit.services/lfk/backend/commit/69afd4d5877401eb46df430f43a7feb273abda1e)
|
- 🧾New changelog file version [CI SKIP] [skip ci] [`69afd4d`](https://git.odit.services/lfk/backend/commit/69afd4d5877401eb46df430f43a7feb273abda1e)
|
||||||
- 🚀Bumped version to v0.10.1 [`24d152f`](https://git.odit.services/lfk/backend/commit/24d152fdc8fe17fffa2f2a718d7145ba8a91d79c)
|
- 🚀Bumped version to v0.10.1 [`24d152f`](https://git.odit.services/lfk/backend/commit/24d152fdc8fe17fffa2f2a718d7145ba8a91d79c)
|
||||||
|
27
Dockerfile
27
Dockerfile
@@ -1,16 +1,23 @@
|
|||||||
# Typescript Build
|
# Typescript Build
|
||||||
FROM node:14.15.1-alpine3.12
|
FROM registry.odit.services/hub/library/node:21.1.0-alpine3.18 as build
|
||||||
|
ARG NPM_REGISTRY_URL=https://registry.npmjs.org
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
COPY package.json ./
|
COPY package.json ./
|
||||||
RUN npm i -g pnpm
|
RUN npm config set registry $NPM_REGISTRY_URL && npm i -g pnpm@8
|
||||||
RUN pnpm i
|
RUN mkdir /pnpm && pnpm config set store-dir /pnpm && pnpm i
|
||||||
|
|
||||||
COPY tsconfig.json ormconfig.js ./
|
COPY tsconfig.json ormconfig.js ./
|
||||||
COPY src ./src
|
COPY src ./src
|
||||||
RUN pnpm run build
|
RUN pnpm run build \
|
||||||
|
&& rm -rf /app/node_modules \
|
||||||
|
&& pnpm i --production --prefer-offline
|
||||||
|
|
||||||
# final image
|
# final image
|
||||||
FROM node:14.15.1-alpine3.12
|
FROM registry.odit.services/hub/library/node:21.1.0-alpine3.18 as final
|
||||||
COPY package.json ormconfig.js ./
|
WORKDIR /app
|
||||||
RUN npm i -g pnpm
|
COPY --from=build /app/package.json /app/package.json
|
||||||
RUN pnpm i --prod
|
COPY --from=build /app/ormconfig.js /app/ormconfig.js
|
||||||
COPY --from=0 /app/dist dist
|
COPY --from=build /app/dist /app/dist
|
||||||
ENTRYPOINT ["node", "dist/app.js"]
|
COPY --from=build /app/node_modules /app/node_modules
|
||||||
|
ENTRYPOINT ["node", "/app/dist/app.js"]
|
68
README.md
68
README.md
@@ -15,59 +15,53 @@ Backend Server
|
|||||||
|
|
||||||
1. Rename the .env.example file to .env (you can adjust app port and other settings, if needed)
|
1. Rename the .env.example file to .env (you can adjust app port and other settings, if needed)
|
||||||
2. Install Dependencies
|
2. Install Dependencies
|
||||||
```bash
|
```bash
|
||||||
yarn
|
pnpm i
|
||||||
```
|
```
|
||||||
3. Start the server
|
3. Start the server
|
||||||
```bash
|
```bash
|
||||||
yarn dev
|
pnpm dev
|
||||||
```
|
```
|
||||||
|
|
||||||
### Run Tests
|
### Run Tests
|
||||||
```bash
|
```bash
|
||||||
# Run tests once (server has to run)
|
# Run tests once (server has to run)
|
||||||
yarn test
|
pnpm test
|
||||||
|
|
||||||
# Run test in watch mode (reruns on change)
|
# Run test in watch mode (reruns on change)
|
||||||
yarn test:watch
|
pnpm test:watch
|
||||||
|
|
||||||
# Run test in ci mode (automaticly starts the dev server)
|
# Run test in ci mode (automaticly starts the dev server)
|
||||||
yarn test:ci
|
pnpm test:ci
|
||||||
```
|
```
|
||||||
|
|
||||||
### Use your own mail templates
|
|
||||||
> You use your own mail templates by replacing the default ones we provided (either in-code or by mounting them into the /app/static/mail_templates folder).
|
|
||||||
|
|
||||||
The mail templates always come in a .html and a .txt variant to provide compatability with legacy mail clients.
|
|
||||||
Currently the following templates exist:
|
|
||||||
* pw-reset.(html/txt)
|
|
||||||
|
|
||||||
### Generate Docs
|
### Generate Docs
|
||||||
```bash
|
```bash
|
||||||
yarn docs
|
pnpm docs
|
||||||
```
|
```
|
||||||
|
|
||||||
## ENV Vars
|
## ENV Vars
|
||||||
> You can provide them via .env file or docker env vars.
|
> You can provide them via .env file or docker env vars.
|
||||||
> You can use the `test:ci:generate_env` package script to generate a example env (uses bs data as test server and ignores the errors).
|
> You can use the `test:ci:generate_env` package script to generate a example env (uses bs data as test server and ignores the errors).
|
||||||
|
|
||||||
| Name | Type | Default | Description
|
| Name | Type | Default | Description |
|
||||||
| - | - | - | -
|
| ---------------------- | ------------------ | -------------------- | -------------------------------------------------------------------------------------------------------------- |
|
||||||
| APP_PORT | Number | 4010 | The port the backend server listens on. Is optional.
|
| APP_PORT | Number | 4010 | The port the backend server listens on. Is optional. |
|
||||||
| DB_TYPE | String | N/A | The type of the db u want to use. It has to be supported by typeorm. Possible: `sqlite`, `mysql`, `postgresql`
|
| DB_TYPE | String | N/A | The type of the db u want to use. It has to be supported by typeorm. Possible: `sqlite`, `mysql`, `postgresql` |
|
||||||
| DB_HOST | String | N/A | The db's host's ip-address/fqdn or file path for sqlite
|
| DB_HOST | String | N/A | The db's host's ip-address/fqdn or file path for sqlite |
|
||||||
| DB_PORT | String | N/A | The db's port
|
| DB_PORT | String | N/A | The db's port |
|
||||||
| DB_USER | String | N/A | The user for accessing the db
|
| DB_USER | String | N/A | The user for accessing the db |
|
||||||
| DB_PASSWORD | String | N/A | The user's password for accessing the db
|
| DB_PASSWORD | String | N/A | The user's password for accessing the db |
|
||||||
| DB_NAME | String | N/A | The db's name
|
| DB_NAME | String | N/A | The db's name |
|
||||||
| NODE_ENV | String | dev | The apps env - influences debug info. Also when the env is set to "test", mailing errors get ignored.
|
| NODE_ENV | String | dev | The apps env - influences debug info. Also when the env is set to "test", mailing errors get ignored. |
|
||||||
| POSTALCODE_COUNTRYCODE | String/CountryCode | N/A | The countrycode used to validate address's postal codes
|
| POSTALCODE_COUNTRYCODE | String/CountryCode | N/A | The countrycode used to validate address's postal codes |
|
||||||
| PHONE_COUNTRYCODE | String/CountryCode | null (international) | The countrycode used to validate phone numers
|
| PHONE_COUNTRYCODE | String/CountryCode | null (international) | The countrycode used to validate phone numers |
|
||||||
| SEED_TEST_DATA | Boolean | False | If you want the app to seed some example data set this to true
|
| SEED_TEST_DATA | Boolean | False | If you want the app to seed some example data set this to true |
|
||||||
| MAILER_URL | String(Url) | N/A | The mailer's base url (no trailing slash)
|
| MAILER_URL | String(Url) | N/A | The mailer's base url (no trailing slash) |
|
||||||
| MAILER_KEY | String | N/A | The mailer's api key.
|
| MAILER_KEY | String | N/A | The mailer's api key. |
|
||||||
| IMPRINT_URL | String(Url) | /imprint | The link to a imprint page for the system (Defaults to the frontend's imprint)
|
| SELFSERVICE_URL | String(Url) | N/A | The link to selfservice (no trailing slash) |
|
||||||
| PRIVACY_URL | String(Url) | /privacy | The link to a privacy page for the system (Defaults to the frontend's privacy page)
|
| IMPRINT_URL | String(Url) | /imprint | The link to a imprint page for the system (Defaults to the frontend's imprint) |
|
||||||
|
| PRIVACY_URL | String(Url) | /privacy | The link to a privacy page for the system (Defaults to the frontend's privacy page) |
|
||||||
|
|
||||||
|
|
||||||
## Recommended Editor
|
## Recommended Editor
|
||||||
@@ -85,10 +79,10 @@ yarn docs
|
|||||||
* A new release tag automaticly triggers the release ci pipeline
|
* A new release tag automaticly triggers the release ci pipeline
|
||||||
* main: Protected "release" branch
|
* main: Protected "release" branch
|
||||||
* The latest tag of the docker image get's build from this
|
* The latest tag of the docker image get's build from this
|
||||||
* New releases get created as tags from this
|
|
||||||
* dev: Current dev branch for merging the different feature branches and bugfixes
|
* dev: Current dev branch for merging the different feature branches and bugfixes
|
||||||
|
* New releases get created as tags from this
|
||||||
* The dev tag of the docker image get's build from this
|
* The dev tag of the docker image get's build from this
|
||||||
* Only push minor changes to this branch!
|
* Only push minor changes to this branch!
|
||||||
* To merge a feature branch into this please create a pull request
|
* To merge a feature branch into this please create a pull request
|
||||||
* feature/xyz: Feature branches - nameing scheme: `feature/issueid-title`
|
* feature/xyz: Feature branches - naming scheme: `feature/issueid-title`
|
||||||
* bugfix/xyz: Branches for bugfixes - nameing scheme:`bugfix/issueid-title`
|
* bugfix/xyz: Branches for bugfixes - naming scheme:`bugfix/issueid-title`
|
@@ -11,8 +11,12 @@ services:
|
|||||||
DB_PORT: bla
|
DB_PORT: bla
|
||||||
DB_USER: bla
|
DB_USER: bla
|
||||||
DB_PASSWORD: bla
|
DB_PASSWORD: bla
|
||||||
DB_NAME: dev.sqlite
|
DB_NAME: ./db.sqlite
|
||||||
NODE_ENV: production
|
NODE_ENV: production
|
||||||
|
POSTALCODE_COUNTRYCODE: DE
|
||||||
|
SEED_TEST_DATA: "false"
|
||||||
|
MAILER_URL: https://dev.lauf-fuer-kaya.de/mailer
|
||||||
|
MAILER_KEY: asdasd
|
||||||
# APP_PORT: 4010
|
# APP_PORT: 4010
|
||||||
# DB_TYPE: postgres
|
# DB_TYPE: postgres
|
||||||
# DB_HOST: backend_db
|
# DB_HOST: backend_db
|
||||||
|
117
licenses.md
117
licenses.md
@@ -444,6 +444,25 @@ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|||||||
**License**: MIT
|
**License**: MIT
|
||||||
**Description**: A node.js driver for mysql. It is written in JavaScript, does not require compiling, and is 100% MIT licensed.
|
**Description**: A node.js driver for mysql. It is written in JavaScript, does not require compiling, and is 100% MIT licensed.
|
||||||
## License Text
|
## License Text
|
||||||
|
Copyright (c) 2012 Felix Geisendörfer (felix@debuggable.com) and contributors
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
|
|
||||||
|
|
||||||
# pg
|
# pg
|
||||||
@@ -696,6 +715,75 @@ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|||||||
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
|
||||||
|
# @faker-js/faker
|
||||||
|
**Author**: undefined
|
||||||
|
**Repo**: [object Object]
|
||||||
|
**License**: MIT
|
||||||
|
**Description**: Generate massive amounts of fake contextual data
|
||||||
|
## License Text
|
||||||
|
Faker - Copyright (c) 2022
|
||||||
|
|
||||||
|
This software consists of voluntary contributions made by many individuals.
|
||||||
|
For exact contribution history, see the revision history
|
||||||
|
available at https://github.com/faker-js/faker
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
===
|
||||||
|
|
||||||
|
From: https://github.com/faker-js/faker/commit/a9f98046c7d5eeaabe12fc587024c06d683800b8
|
||||||
|
To: https://github.com/faker-js/faker/commit/29234378807c4141588861f69421bf20b5ac635e
|
||||||
|
|
||||||
|
Based on faker.js, copyright Marak Squires and contributor, what follows below is the original license.
|
||||||
|
|
||||||
|
===
|
||||||
|
|
||||||
|
faker.js - Copyright (c) 2020
|
||||||
|
Marak Squires
|
||||||
|
http://github.com/marak/faker.js/
|
||||||
|
|
||||||
|
faker.js was inspired by and has used data definitions from:
|
||||||
|
|
||||||
|
* https://github.com/stympy/faker/ - Copyright (c) 2007-2010 Benjamin Curtis
|
||||||
|
* http://search.cpan.org/~jasonk/Data-Faker-0.07/ - Copyright 2004-2005 by Jason Kohles
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
|
||||||
# @odit/license-exporter
|
# @odit/license-exporter
|
||||||
**Author**: ODIT.Services
|
**Author**: ODIT.Services
|
||||||
**Repo**: [object Object]
|
**Repo**: [object Object]
|
||||||
@@ -926,6 +1014,35 @@ OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|||||||
SOFTWARE
|
SOFTWARE
|
||||||
|
|
||||||
|
|
||||||
|
# auto-changelog
|
||||||
|
**Author**: Pete Cook <pete@cookpete.com> (https://github.com/cookpete)
|
||||||
|
**Repo**: [object Object]
|
||||||
|
**License**: MIT
|
||||||
|
**Description**: Command line tool for generating a changelog from git tags and commit history
|
||||||
|
## License Text
|
||||||
|
The MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2017 Pete Cook https://cookpete.com
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
|
|
||||||
|
|
||||||
# cp-cli
|
# cp-cli
|
||||||
**Author**: undefined
|
**Author**: undefined
|
||||||
**Repo**: [object Object]
|
**Repo**: [object Object]
|
||||||
|
223
package.json
223
package.json
@@ -1,106 +1,117 @@
|
|||||||
{
|
{
|
||||||
"name": "@odit/lfk-backend",
|
"name": "@odit/lfk-backend",
|
||||||
"version": "0.10.1",
|
"version": "1.2.0",
|
||||||
"main": "src/app.ts",
|
"main": "src/app.ts",
|
||||||
"repository": "https://git.odit.services/lfk/backend",
|
"repository": "https://git.odit.services/lfk/backend",
|
||||||
"author": {
|
"engines": {
|
||||||
"name": "ODIT.Services",
|
"pnpm": "8"
|
||||||
"email": "info@odit.services",
|
},
|
||||||
"url": "https://odit.services"
|
"author": {
|
||||||
},
|
"name": "ODIT.Services",
|
||||||
"contributors": [
|
"email": "info@odit.services",
|
||||||
{
|
"url": "https://odit.services"
|
||||||
"name": "Philipp Dormann",
|
},
|
||||||
"email": "philipp@philippdormann.de",
|
"contributors": [
|
||||||
"url": "https://philippdormann.de"
|
{
|
||||||
},
|
"name": "Philipp Dormann",
|
||||||
{
|
"email": "philipp@philippdormann.de",
|
||||||
"name": "Nicolai Ort",
|
"url": "https://philippdormann.de"
|
||||||
"email": "info@nicolai-ort.com",
|
},
|
||||||
"url": "https://nicolai-ort.com"
|
{
|
||||||
}
|
"name": "Nicolai Ort",
|
||||||
],
|
"email": "info@nicolai-ort.com",
|
||||||
"license": "CC-BY-NC-SA-4.0",
|
"url": "https://nicolai-ort.com"
|
||||||
"dependencies": {
|
}
|
||||||
"@odit/class-validator-jsonschema": "2.1.1",
|
],
|
||||||
"argon2": "^0.27.1",
|
"license": "CC-BY-NC-SA-4.0",
|
||||||
"axios": "^0.21.1",
|
"dependencies": {
|
||||||
"body-parser": "^1.19.0",
|
"@odit/class-validator-jsonschema": "2.1.1",
|
||||||
"check-password-strength": "^2.0.2",
|
"argon2": "0.31.2",
|
||||||
"class-transformer": "0.3.1",
|
"axios": "0.21.1",
|
||||||
"class-validator": "^0.13.1",
|
"body-parser": "1.19.0",
|
||||||
"consola": "^2.15.0",
|
"check-password-strength": "2.0.2",
|
||||||
"cookie": "^0.4.1",
|
"class-transformer": "0.3.1",
|
||||||
"cookie-parser": "^1.4.5",
|
"class-validator": "0.13.0",
|
||||||
"cors": "^2.8.5",
|
"consola": "2.15.0",
|
||||||
"csvtojson": "^2.0.10",
|
"cookie": "0.4.1",
|
||||||
"dotenv": "^8.2.0",
|
"cookie-parser": "1.4.5",
|
||||||
"express": "^4.17.1",
|
"cors": "2.8.5",
|
||||||
"jsonwebtoken": "^8.5.1",
|
"csvtojson": "2.0.10",
|
||||||
"libphonenumber-js": "^1.9.9",
|
"dotenv": "8.2.0",
|
||||||
"mysql": "^2.18.1",
|
"express": "4.17.1",
|
||||||
"pg": "^8.5.1",
|
"jsonwebtoken": "8.5.1",
|
||||||
"reflect-metadata": "^0.1.13",
|
"libphonenumber-js": "1.9.9",
|
||||||
"routing-controllers": "0.9.0-alpha.6",
|
"mysql": "2.18.1",
|
||||||
"routing-controllers-openapi": "^2.2.0",
|
"pg": "8.5.1",
|
||||||
"sqlite3": "5.0.0",
|
"reflect-metadata": "0.1.13",
|
||||||
"typeorm": "^0.2.30",
|
"routing-controllers": "0.9.0-alpha.6",
|
||||||
"typeorm-routing-controllers-extensions": "^0.2.0",
|
"routing-controllers-openapi": "2.2.0",
|
||||||
"typeorm-seeding": "^1.6.1",
|
"sqlite3": "5.1.6",
|
||||||
"uuid": "^8.3.2",
|
"typeorm": "0.2.30",
|
||||||
"validator": "^13.5.2"
|
"typeorm-routing-controllers-extensions": "0.2.0",
|
||||||
},
|
"typeorm-seeding": "1.6.1",
|
||||||
"devDependencies": {
|
"uuid": "8.3.2",
|
||||||
"@odit/license-exporter": "^0.0.9",
|
"validator": "13.5.2"
|
||||||
"@types/cors": "^2.8.9",
|
},
|
||||||
"@types/csvtojson": "^1.1.5",
|
"devDependencies": {
|
||||||
"@types/express": "^4.17.11",
|
"@faker-js/faker": "7.6.0",
|
||||||
"@types/jest": "^26.0.20",
|
"@odit/license-exporter": "0.0.9",
|
||||||
"@types/jsonwebtoken": "^8.5.0",
|
"@types/cors": "2.8.9",
|
||||||
"@types/node": "^14.14.22",
|
"@types/csvtojson": "1.1.5",
|
||||||
"@types/uuid": "^8.3.0",
|
"@types/express": "4.17.11",
|
||||||
"cp-cli": "^2.0.0",
|
"@types/jest": "26.0.20",
|
||||||
"jest": "^26.6.3",
|
"@types/jsonwebtoken": "8.5.0",
|
||||||
"nodemon": "^2.0.7",
|
"@types/node": "14.14.22",
|
||||||
"release-it": "^14.2.2",
|
"@types/uuid": "8.3.0",
|
||||||
"rimraf": "^3.0.2",
|
"auto-changelog": "2.4.0",
|
||||||
"start-server-and-test": "^1.11.7",
|
"cp-cli": "2.0.0",
|
||||||
"ts-jest": "^26.5.0",
|
"jest": "26.6.3",
|
||||||
"ts-node": "^9.1.1",
|
"nodemon": "2.0.7",
|
||||||
"typedoc": "^0.20.19",
|
"release-it": "14.2.2",
|
||||||
"typescript": "^4.1.3"
|
"rimraf": "3.0.2",
|
||||||
},
|
"start-server-and-test": "1.11.7",
|
||||||
"scripts": {
|
"ts-jest": "26.5.0",
|
||||||
"dev": "nodemon src/app.ts",
|
"ts-node": "9.1.1",
|
||||||
"build": "rimraf ./dist && tsc && cp-cli ./src/static ./dist/static",
|
"typedoc": "0.20.19",
|
||||||
"docs": "typedoc --out docs src",
|
"typescript": "4.1.3"
|
||||||
"test": "jest",
|
},
|
||||||
"test:watch": "jest --watchAll",
|
"scripts": {
|
||||||
"test:ci:generate_env": "ts-node scripts/create_testenv.ts",
|
"dev": "nodemon src/app.ts",
|
||||||
"test:ci:run": "start-server-and-test dev http://localhost:4010/api/docs/openapi.json test",
|
"build": "rimraf ./dist && tsc && cp-cli ./src/static ./dist/static",
|
||||||
"test:ci": "npm run test:ci:generate_env && npm run test:ci:run",
|
"docs": "typedoc --out docs src",
|
||||||
"seed": "ts-node ./node_modules/typeorm/cli.js schema:sync && ts-node ./node_modules/typeorm-seeding/dist/cli.js seed",
|
"test": "jest",
|
||||||
"openapi:export": "ts-node scripts/openapi_export.ts",
|
"test:watch": "jest --watchAll",
|
||||||
"licenses:export": "license-exporter --markdown",
|
"test:ci:generate_env": "ts-node scripts/create_testenv.ts",
|
||||||
"release": "release-it --only-version"
|
"test:ci:run": "start-server-and-test dev http://localhost:4010/api/docs/openapi.json test",
|
||||||
},
|
"test:ci": "npm run test:ci:generate_env && npm run test:ci:run",
|
||||||
"release-it": {
|
"seed": "ts-node ./node_modules/typeorm/cli.js schema:sync && ts-node ./node_modules/typeorm-seeding/dist/cli.js seed",
|
||||||
"git": {
|
"openapi:export": "ts-node scripts/openapi_export.ts",
|
||||||
"commit": true,
|
"licenses:export": "license-exporter --markdown",
|
||||||
"requireCleanWorkingDir": false,
|
"changelog:export": "auto-changelog --commit-limit false -p -u --hide-credit",
|
||||||
"commitMessage": "🚀Bumped version to v${version}",
|
"release": "release-it --only-version"
|
||||||
"requireBranch": "dev",
|
},
|
||||||
"push": false,
|
"release-it": {
|
||||||
"tag": false
|
"git": {
|
||||||
},
|
"commit": true,
|
||||||
"npm": {
|
"requireCleanWorkingDir": false,
|
||||||
"publish": false
|
"commitMessage": "🚀Bumped version to v${version}",
|
||||||
}
|
"requireBranch": "dev",
|
||||||
},
|
"push": true,
|
||||||
"nodemonConfig": {
|
"tag": true,
|
||||||
"ignore": [
|
"tagName": "v${version}",
|
||||||
"src/tests/*",
|
"tagAnnotation": "v${version}"
|
||||||
"docs/*"
|
},
|
||||||
]
|
"npm": {
|
||||||
}
|
"publish": false
|
||||||
}
|
},
|
||||||
|
"hooks": {
|
||||||
|
"after:bump": "npm run changelog:export && npm run licenses:export && git add CHANGELOG.md && git add licenses.md"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nodemonConfig": {
|
||||||
|
"ignore": [
|
||||||
|
"src/tests/*",
|
||||||
|
"docs/*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
7744
pnpm-lock.yaml
generated
Normal file
7744
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { DonationIdsNotMatchingError, DonationNotFoundError } from '../errors/DonationErrors';
|
import { DonationIdsNotMatchingError, DonationNotFoundError } from '../errors/DonationErrors';
|
||||||
import { DonorNotFoundError } from '../errors/DonorErrors';
|
import { DonorNotFoundError } from '../errors/DonorErrors';
|
||||||
import { RunnerNotFoundError } from '../errors/RunnerErrors';
|
import { RunnerNotFoundError } from '../errors/RunnerErrors';
|
||||||
@@ -36,9 +36,16 @@ export class DonationController {
|
|||||||
@ResponseSchema(ResponseDonation, { isArray: true })
|
@ResponseSchema(ResponseDonation, { isArray: true })
|
||||||
@ResponseSchema(ResponseDistanceDonation, { isArray: true })
|
@ResponseSchema(ResponseDistanceDonation, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all donations (fixed or distance based) from all donors. <br> This includes the donations\'s runner\'s distance ran(if distance donation).' })
|
@OpenAPI({ description: 'Lists all donations (fixed or distance based) from all donors. <br> This includes the donations\'s runner\'s distance ran(if distance donation).' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseDonations: ResponseDonation[] = new Array<ResponseDonation>();
|
let responseDonations: ResponseDonation[] = new Array<ResponseDonation>();
|
||||||
const donations = await this.donationRepository.find({ relations: ['runner', 'donor', 'runner.scans', 'runner.scans.track'] });
|
let donations: Array<Donation>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
donations = await this.donationRepository.find({ relations: ['runner', 'donor', 'runner.scans', 'runner.scans.track'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
donations = await this.donationRepository.find({ relations: ['runner', 'donor', 'runner.scans', 'runner.scans.track'] });
|
||||||
|
}
|
||||||
|
|
||||||
donations.forEach(donation => {
|
donations.forEach(donation => {
|
||||||
responseDonations.push(donation.toResponse());
|
responseDonations.push(donation.toResponse());
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { DonorHasDonationsError, DonorIdsNotMatchingError, DonorNotFoundError } from '../errors/DonorErrors';
|
import { DonorHasDonationsError, DonorIdsNotMatchingError, DonorNotFoundError } from '../errors/DonorErrors';
|
||||||
import { CreateDonor } from '../models/actions/create/CreateDonor';
|
import { CreateDonor } from '../models/actions/create/CreateDonor';
|
||||||
import { UpdateDonor } from '../models/actions/update/UpdateDonor';
|
import { UpdateDonor } from '../models/actions/update/UpdateDonor';
|
||||||
@@ -25,9 +25,16 @@ export class DonorController {
|
|||||||
@Authorized("DONOR:GET")
|
@Authorized("DONOR:GET")
|
||||||
@ResponseSchema(ResponseDonor, { isArray: true })
|
@ResponseSchema(ResponseDonor, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all donor. <br> This includes the donor\'s current donation amount.' })
|
@OpenAPI({ description: 'Lists all donor. <br> This includes the donor\'s current donation amount.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseDonors: ResponseDonor[] = new Array<ResponseDonor>();
|
let responseDonors: ResponseDonor[] = new Array<ResponseDonor>();
|
||||||
const donors = await this.donorRepository.find({ relations: ['donations', 'donations.runner', 'donations.runner.scans', 'donations.runner.scans.track'] });
|
let donors: Array<Donor>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
donors = await this.donorRepository.find({ relations: ['donations', 'donations.runner', 'donations.runner.scans', 'donations.runner.scans.track'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
donors = await this.donorRepository.find({ relations: ['donations', 'donations.runner', 'donations.runner.scans', 'donations.runner.scans.track'] });
|
||||||
|
}
|
||||||
|
|
||||||
donors.forEach(donor => {
|
donors.forEach(donor => {
|
||||||
responseDonors.push(new ResponseDonor(donor));
|
responseDonors.push(new ResponseDonor(donor));
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { getConnection, getConnectionManager, Repository } from 'typeorm';
|
import { Repository, getConnection, getConnectionManager } from 'typeorm';
|
||||||
import { GroupContactIdsNotMatchingError, GroupContactNotFoundError } from '../errors/GroupContactErrors';
|
import { GroupContactIdsNotMatchingError, GroupContactNotFoundError } from '../errors/GroupContactErrors';
|
||||||
import { RunnerGroupNotFoundError } from '../errors/RunnerGroupErrors';
|
import { RunnerGroupNotFoundError } from '../errors/RunnerGroupErrors';
|
||||||
import { CreateGroupContact } from '../models/actions/create/CreateGroupContact';
|
import { CreateGroupContact } from '../models/actions/create/CreateGroupContact';
|
||||||
@@ -26,9 +26,16 @@ export class GroupContactController {
|
|||||||
@Authorized("CONTACT:GET")
|
@Authorized("CONTACT:GET")
|
||||||
@ResponseSchema(ResponseGroupContact, { isArray: true })
|
@ResponseSchema(ResponseGroupContact, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all contacts. <br> This includes the contact\'s associated groups.' })
|
@OpenAPI({ description: 'Lists all contacts. <br> This includes the contact\'s associated groups.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseContacts: ResponseGroupContact[] = new Array<ResponseGroupContact>();
|
let responseContacts: ResponseGroupContact[] = new Array<ResponseGroupContact>();
|
||||||
const contacts = await this.contactRepository.find({ relations: ['groups', 'groups.parentGroup'] });
|
let contacts: Array<GroupContact>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
contacts = await this.contactRepository.find({ relations: ['groups', 'groups.parentGroup'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
contacts = await this.contactRepository.find({ relations: ['groups', 'groups.parentGroup'] });
|
||||||
|
}
|
||||||
|
|
||||||
contacts.forEach(contact => {
|
contacts.forEach(contact => {
|
||||||
responseContacts.push(contact.toResponse());
|
responseContacts.push(contact.toResponse());
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { PermissionIdsNotMatchingError, PermissionNeedsPrincipalError, PermissionNotFoundError } from '../errors/PermissionErrors';
|
import { PermissionIdsNotMatchingError, PermissionNeedsPrincipalError, PermissionNotFoundError } from '../errors/PermissionErrors';
|
||||||
import { PrincipalNotFoundError } from '../errors/PrincipalErrors';
|
import { PrincipalNotFoundError } from '../errors/PrincipalErrors';
|
||||||
import { CreatePermission } from '../models/actions/create/CreatePermission';
|
import { CreatePermission } from '../models/actions/create/CreatePermission';
|
||||||
@@ -27,9 +27,16 @@ export class PermissionController {
|
|||||||
@Authorized("PERMISSION:GET")
|
@Authorized("PERMISSION:GET")
|
||||||
@ResponseSchema(ResponsePermission, { isArray: true })
|
@ResponseSchema(ResponsePermission, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all permissions for all users and groups.' })
|
@OpenAPI({ description: 'Lists all permissions for all users and groups.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responsePermissions: ResponsePermission[] = new Array<ResponsePermission>();
|
let responsePermissions: ResponsePermission[] = new Array<ResponsePermission>();
|
||||||
const permissions = await this.permissionRepository.find({ relations: ['principal'] });
|
let permissions: Array<Permission>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
permissions = await this.permissionRepository.find({ relations: ['principal'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
permissions = await this.permissionRepository.find({ relations: ['principal'] });
|
||||||
|
}
|
||||||
|
|
||||||
permissions.forEach(permission => {
|
permissions.forEach(permission => {
|
||||||
responsePermissions.push(new ResponsePermission(permission));
|
responsePermissions.push(new ResponsePermission(permission));
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { RunnerCardHasScansError, RunnerCardIdsNotMatchingError, RunnerCardNotFoundError } from '../errors/RunnerCardErrors';
|
import { RunnerCardHasScansError, RunnerCardIdsNotMatchingError, RunnerCardNotFoundError } from '../errors/RunnerCardErrors';
|
||||||
import { RunnerNotFoundError } from '../errors/RunnerErrors';
|
import { RunnerNotFoundError } from '../errors/RunnerErrors';
|
||||||
import { CreateRunnerCard } from '../models/actions/create/CreateRunnerCard';
|
import { CreateRunnerCard } from '../models/actions/create/CreateRunnerCard';
|
||||||
@@ -26,9 +26,16 @@ export class RunnerCardController {
|
|||||||
@Authorized("CARD:GET")
|
@Authorized("CARD:GET")
|
||||||
@ResponseSchema(ResponseRunnerCard, { isArray: true })
|
@ResponseSchema(ResponseRunnerCard, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all card.' })
|
@OpenAPI({ description: 'Lists all card.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseCards: ResponseRunnerCard[] = new Array<ResponseRunnerCard>();
|
let responseCards: ResponseRunnerCard[] = new Array<ResponseRunnerCard>();
|
||||||
const cards = await this.cardRepository.find({ relations: ['runner', 'runner.group', 'runner.group.parentGroup'] });
|
let cards: Array<RunnerCard>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
cards = await this.cardRepository.find({ relations: ['runner', 'runner.group', 'runner.group.parentGroup'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
cards = await this.cardRepository.find({ relations: ['runner', 'runner.group', 'runner.group.parentGroup'] });
|
||||||
|
}
|
||||||
|
|
||||||
cards.forEach(card => {
|
cards.forEach(card => {
|
||||||
responseCards.push(new ResponseRunnerCard(card));
|
responseCards.push(new ResponseRunnerCard(card));
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { RunnerGroupNeededError, RunnerHasDistanceDonationsError, RunnerIdsNotMatchingError, RunnerNotFoundError } from '../errors/RunnerErrors';
|
import { RunnerGroupNeededError, RunnerHasDistanceDonationsError, RunnerIdsNotMatchingError, RunnerNotFoundError } from '../errors/RunnerErrors';
|
||||||
import { RunnerGroupNotFoundError } from '../errors/RunnerGroupErrors';
|
import { RunnerGroupNotFoundError } from '../errors/RunnerGroupErrors';
|
||||||
import { CreateRunner } from '../models/actions/create/CreateRunner';
|
import { CreateRunner } from '../models/actions/create/CreateRunner';
|
||||||
@@ -30,9 +30,16 @@ export class RunnerController {
|
|||||||
@Authorized("RUNNER:GET")
|
@Authorized("RUNNER:GET")
|
||||||
@ResponseSchema(ResponseRunner, { isArray: true })
|
@ResponseSchema(ResponseRunner, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all runners from all teams/orgs. <br> This includes the runner\'s group and distance ran.' })
|
@OpenAPI({ description: 'Lists all runners from all teams/orgs. <br> This includes the runner\'s group and distance ran.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseRunners: ResponseRunner[] = new Array<ResponseRunner>();
|
let responseRunners: ResponseRunner[] = new Array<ResponseRunner>();
|
||||||
const runners = await this.runnerRepository.find({ relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] });
|
let runners: Array<Runner>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
runners = await this.runnerRepository.find({ relations: ['scans', 'group', 'group.parentGroup', 'scans.track'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
runners = await this.runnerRepository.find({ relations: ['scans', 'group', 'group.parentGroup', 'scans.track'] });
|
||||||
|
}
|
||||||
|
|
||||||
runners.forEach(runner => {
|
runners.forEach(runner => {
|
||||||
responseRunners.push(new ResponseRunner(runner));
|
responseRunners.push(new ResponseRunner(runner));
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam } from 'routing-controllers';
|
import { Authorized, BadRequestError, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { RunnerOrganizationHasRunnersError, RunnerOrganizationHasTeamsError, RunnerOrganizationIdsNotMatchingError, RunnerOrganizationNotFoundError } from '../errors/RunnerOrganizationErrors';
|
import { RunnerOrganizationHasRunnersError, RunnerOrganizationHasTeamsError, RunnerOrganizationIdsNotMatchingError, RunnerOrganizationNotFoundError } from '../errors/RunnerOrganizationErrors';
|
||||||
import { CreateRunnerOrganization } from '../models/actions/create/CreateRunnerOrganization';
|
import { CreateRunnerOrganization } from '../models/actions/create/CreateRunnerOrganization';
|
||||||
import { UpdateRunnerOrganization } from '../models/actions/update/UpdateRunnerOrganization';
|
import { UpdateRunnerOrganization } from '../models/actions/update/UpdateRunnerOrganization';
|
||||||
@@ -29,13 +29,20 @@ export class RunnerOrganizationController {
|
|||||||
@Authorized("ORGANIZATION:GET")
|
@Authorized("ORGANIZATION:GET")
|
||||||
@ResponseSchema(ResponseRunnerOrganization, { isArray: true })
|
@ResponseSchema(ResponseRunnerOrganization, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all organizations. <br> This includes their address, contact and teams (if existing/associated).' })
|
@OpenAPI({ description: 'Lists all organizations. <br> This includes their address, contact and teams (if existing/associated).' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseTeams: ResponseRunnerOrganization[] = new Array<ResponseRunnerOrganization>();
|
let responseOrgs: ResponseRunnerOrganization[] = new Array<ResponseRunnerOrganization>();
|
||||||
const runners = await this.runnerOrganizationRepository.find({ relations: ['contact', 'teams'] });
|
let orgs: Array<RunnerOrganization>;
|
||||||
runners.forEach(runner => {
|
|
||||||
responseTeams.push(new ResponseRunnerOrganization(runner));
|
if (page != undefined) {
|
||||||
|
orgs = await this.runnerOrganizationRepository.find({ relations: ['contact', 'teams'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
orgs = await this.runnerOrganizationRepository.find({ relations: ['contact', 'teams'] });
|
||||||
|
}
|
||||||
|
|
||||||
|
orgs.forEach(org => {
|
||||||
|
responseOrgs.push(new ResponseRunnerOrganization(org));
|
||||||
});
|
});
|
||||||
return responseTeams;
|
return responseOrgs;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('/:id')
|
@Get('/:id')
|
||||||
@@ -45,7 +52,7 @@ export class RunnerOrganizationController {
|
|||||||
@OnUndefined(RunnerOrganizationNotFoundError)
|
@OnUndefined(RunnerOrganizationNotFoundError)
|
||||||
@OpenAPI({ description: 'Lists all information about the organization whose id got provided.' })
|
@OpenAPI({ description: 'Lists all information about the organization whose id got provided.' })
|
||||||
async getOne(@Param('id') id: number) {
|
async getOne(@Param('id') id: number) {
|
||||||
let runnerOrg = await this.runnerOrganizationRepository.findOne({ id: id }, { relations: ['contact', 'teams'] });
|
let runnerOrg = await this.runnerOrganizationRepository.findOne({ id: id }, { relations: ['contact', 'teams', 'teams.runners', 'teams.runners.scans', 'teams.runners.scans.track', 'runners', 'runners.scans', 'runners.scans.track'] });
|
||||||
if (!runnerOrg) { throw new RunnerOrganizationNotFoundError(); }
|
if (!runnerOrg) { throw new RunnerOrganizationNotFoundError(); }
|
||||||
return new ResponseRunnerOrganization(runnerOrg);
|
return new ResponseRunnerOrganization(runnerOrg);
|
||||||
}
|
}
|
||||||
@@ -114,6 +121,10 @@ export class RunnerOrganizationController {
|
|||||||
@OnUndefined(204)
|
@OnUndefined(204)
|
||||||
@OpenAPI({ description: 'Delete the organsisation whose id you provided. <br> If the organization still has runners and/or teams associated this will fail. <br> To delete the organization with all associated runners and teams set the force QueryParam to true (cascading deletion might take a while). <br> This won\'t delete the associated contact. <br> If no organization with this id exists it will just return 204(no content).' })
|
@OpenAPI({ description: 'Delete the organsisation whose id you provided. <br> If the organization still has runners and/or teams associated this will fail. <br> To delete the organization with all associated runners and teams set the force QueryParam to true (cascading deletion might take a while). <br> This won\'t delete the associated contact. <br> If no organization with this id exists it will just return 204(no content).' })
|
||||||
async remove(@Param("id") id: number, @QueryParam("force") force: boolean) {
|
async remove(@Param("id") id: number, @QueryParam("force") force: boolean) {
|
||||||
|
if (id == 1) {
|
||||||
|
throw new BadRequestError("You can't delete the citizen runner org.");
|
||||||
|
}
|
||||||
|
|
||||||
let organization = await this.runnerOrganizationRepository.findOne({ id: id });
|
let organization = await this.runnerOrganizationRepository.findOne({ id: id });
|
||||||
if (!organization) { return null; }
|
if (!organization) { return null; }
|
||||||
let runnerOrganization = await this.runnerOrganizationRepository.findOne(organization, { relations: ['contact', 'runners', 'teams'] });
|
let runnerOrganization = await this.runnerOrganizationRepository.findOne(organization, { relations: ['contact', 'runners', 'teams'] });
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Request } from "express";
|
import { Request } from "express";
|
||||||
import * as jwt from "jsonwebtoken";
|
import * as jwt from "jsonwebtoken";
|
||||||
import { Body, Delete, Get, JsonController, OnUndefined, Param, Post, QueryParam, Req, UseBefore } from 'routing-controllers';
|
import { BadRequestError, Body, Delete, Get, JsonController, OnUndefined, Param, Post, QueryParam, Req, UseBefore } 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 { config } from '../config';
|
import { config } from '../config';
|
||||||
@@ -116,10 +116,10 @@ export class RunnerSelfServiceController {
|
|||||||
return scan.toResponse();
|
return scan.toResponse();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('/runners/forgot')
|
@Post('/runners/login')
|
||||||
@ResponseSchema(RunnerNotFoundError, { statusCode: 404 })
|
@ResponseSchema(RunnerNotFoundError, { statusCode: 404 })
|
||||||
@OnUndefined(ResponseEmpty)
|
@OnUndefined(ResponseEmpty)
|
||||||
@OpenAPI({ description: 'Use this endpoint to reuqest a new selfservice token/link to be sent to your mail address (rate limited to one mail every 24hrs).' })
|
@OpenAPI({ description: 'Use this endpoint to reuqest a new selfservice magic-login-link to be sent to your mail address (rate limited to one mail every 15mins).' })
|
||||||
async requestNewToken(@QueryParam('mail') mail: string, @QueryParam("locale") locale: string = "en") {
|
async requestNewToken(@QueryParam('mail') mail: string, @QueryParam("locale") locale: string = "en") {
|
||||||
if (!mail) {
|
if (!mail) {
|
||||||
throw new RunnerNotFoundError();
|
throw new RunnerNotFoundError();
|
||||||
@@ -127,11 +127,11 @@ export class RunnerSelfServiceController {
|
|||||||
const runner = await this.runnerRepository.findOne({ email: mail });
|
const runner = await this.runnerRepository.findOne({ email: mail });
|
||||||
if (!runner) { throw new RunnerNotFoundError(); }
|
if (!runner) { throw new RunnerNotFoundError(); }
|
||||||
|
|
||||||
if (runner.resetRequestedTimestamp > (Math.floor(Date.now() / 1000) - 60 * 60 * 24)) { throw new RunnerSelfserviceTimeoutError(); }
|
if (runner.resetRequestedTimestamp > (Math.floor(Date.now() / 1000) - 60 * 15)) { throw new RunnerSelfserviceTimeoutError(); }
|
||||||
const token = JwtCreator.createSelfService(runner);
|
const token = JwtCreator.createSelfService(runner);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await Mailer.sendSelfserviceForgottenMail(runner.email, token, locale)
|
await Mailer.sendSelfserviceForgottenMail(runner.email, runner.id, runner.firstname, runner.middlename, runner.lastname, token, locale)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new MailSendingError();
|
throw new MailSendingError();
|
||||||
}
|
}
|
||||||
@@ -148,13 +148,16 @@ export class RunnerSelfServiceController {
|
|||||||
@OpenAPI({ description: 'Create a new selfservice runner in the citizen org. <br> This endpoint shoud be used to allow "everyday citizen" to register themselves. <br> You have to provide a mail address, b/c the future we\'ll implement email verification.' })
|
@OpenAPI({ description: 'Create a new selfservice runner in the citizen org. <br> This endpoint shoud be used to allow "everyday citizen" to register themselves. <br> You have to provide a mail address, b/c the future we\'ll implement email verification.' })
|
||||||
async registerRunner(@Body({ validate: true }) createRunner: CreateSelfServiceCitizenRunner, @QueryParam("locale") locale: string = "en") {
|
async registerRunner(@Body({ validate: true }) createRunner: CreateSelfServiceCitizenRunner, @QueryParam("locale") locale: string = "en") {
|
||||||
let runner = await createRunner.toEntity();
|
let runner = await createRunner.toEntity();
|
||||||
|
if (await this.getRunnerExistsByMail(runner.email)) {
|
||||||
|
throw new BadRequestError("E-Mail already registered")
|
||||||
|
}
|
||||||
runner = await this.runnerRepository.save(runner);
|
runner = await this.runnerRepository.save(runner);
|
||||||
|
|
||||||
let response = new ResponseSelfServiceRunner(await this.runnerRepository.findOne(runner, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards', 'distanceDonations', 'distanceDonations.donor', 'distanceDonations.runner', 'distanceDonations.runner.scans', 'distanceDonations.runner.scans.track'] }));
|
let response = new ResponseSelfServiceRunner(await this.runnerRepository.findOne(runner, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards', 'distanceDonations', 'distanceDonations.donor', 'distanceDonations.runner', 'distanceDonations.runner.scans', 'distanceDonations.runner.scans.track'] }));
|
||||||
response.token = JwtCreator.createSelfService(runner);
|
response.token = JwtCreator.createSelfService(runner);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await Mailer.sendSelfserviceWelcomeMail(runner.email, response.token, locale)
|
await Mailer.sendSelfserviceWelcomeMail(runner.email, runner.id, runner.firstname, runner.middlename, runner.lastname, response.token, locale)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new MailSendingError();
|
throw new MailSendingError();
|
||||||
}
|
}
|
||||||
@@ -170,13 +173,16 @@ export class RunnerSelfServiceController {
|
|||||||
const org = await this.getOrgansisation(token);
|
const org = await this.getOrgansisation(token);
|
||||||
|
|
||||||
let runner = await createRunner.toEntity(org);
|
let runner = await createRunner.toEntity(org);
|
||||||
|
if (await this.getRunnerExistsByMail(runner.email)) {
|
||||||
|
throw new BadRequestError("E-Mail already registered")
|
||||||
|
}
|
||||||
runner = await this.runnerRepository.save(runner);
|
runner = await this.runnerRepository.save(runner);
|
||||||
|
|
||||||
let response = new ResponseSelfServiceRunner(await this.runnerRepository.findOne(runner, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards', 'distanceDonations', 'distanceDonations.donor', 'distanceDonations.runner', 'distanceDonations.runner.scans', 'distanceDonations.runner.scans.track'] }));
|
let response = new ResponseSelfServiceRunner(await this.runnerRepository.findOne(runner, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards', 'distanceDonations', 'distanceDonations.donor', 'distanceDonations.runner', 'distanceDonations.runner.scans', 'distanceDonations.runner.scans.track'] }));
|
||||||
response.token = JwtCreator.createSelfService(runner);
|
response.token = JwtCreator.createSelfService(runner);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await Mailer.sendSelfserviceWelcomeMail(runner.email, response.token, locale)
|
await Mailer.sendSelfserviceWelcomeMail(runner.email, runner.id, runner.firstname, runner.middlename, runner.lastname, response.token, locale)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new MailSendingError();
|
throw new MailSendingError();
|
||||||
}
|
}
|
||||||
@@ -225,4 +231,14 @@ export class RunnerSelfServiceController {
|
|||||||
|
|
||||||
return organization;
|
return organization;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if a runner already exists
|
||||||
|
* @param email The runner's email address
|
||||||
|
* @returns Boolean (true if exists, false if not)
|
||||||
|
*/
|
||||||
|
private async getRunnerExistsByMail(email: string): Promise<boolean> {
|
||||||
|
const runner = await this.runnerRepository.findOne({ email });
|
||||||
|
return runner != undefined
|
||||||
|
}
|
||||||
}
|
}
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { RunnerTeamHasRunnersError, RunnerTeamIdsNotMatchingError, RunnerTeamNotFoundError } from '../errors/RunnerTeamErrors';
|
import { RunnerTeamHasRunnersError, RunnerTeamIdsNotMatchingError, RunnerTeamNotFoundError } from '../errors/RunnerTeamErrors';
|
||||||
import { CreateRunnerTeam } from '../models/actions/create/CreateRunnerTeam';
|
import { CreateRunnerTeam } from '../models/actions/create/CreateRunnerTeam';
|
||||||
import { UpdateRunnerTeam } from '../models/actions/update/UpdateRunnerTeam';
|
import { UpdateRunnerTeam } from '../models/actions/update/UpdateRunnerTeam';
|
||||||
@@ -27,11 +27,18 @@ export class RunnerTeamController {
|
|||||||
@Authorized("TEAM:GET")
|
@Authorized("TEAM:GET")
|
||||||
@ResponseSchema(ResponseRunnerTeam, { isArray: true })
|
@ResponseSchema(ResponseRunnerTeam, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all teams. <br> This includes their parent organization and contact (if existing/associated).' })
|
@OpenAPI({ description: 'Lists all teams. <br> This includes their parent organization and contact (if existing/associated).' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseTeams: ResponseRunnerTeam[] = new Array<ResponseRunnerTeam>();
|
let responseTeams: ResponseRunnerTeam[] = new Array<ResponseRunnerTeam>();
|
||||||
const runners = await this.runnerTeamRepository.find({ relations: ['parentGroup', 'contact'] });
|
let teams: Array<RunnerTeam>;
|
||||||
runners.forEach(runner => {
|
|
||||||
responseTeams.push(new ResponseRunnerTeam(runner));
|
if (page != undefined) {
|
||||||
|
teams = await this.runnerTeamRepository.find({ relations: ['parentGroup', 'contact'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
teams = await this.runnerTeamRepository.find({ relations: ['parentGroup', 'contact'] });
|
||||||
|
}
|
||||||
|
|
||||||
|
teams.forEach(team => {
|
||||||
|
responseTeams.push(new ResponseRunnerTeam(team));
|
||||||
});
|
});
|
||||||
return responseTeams;
|
return responseTeams;
|
||||||
}
|
}
|
||||||
@@ -43,7 +50,7 @@ export class RunnerTeamController {
|
|||||||
@OnUndefined(RunnerTeamNotFoundError)
|
@OnUndefined(RunnerTeamNotFoundError)
|
||||||
@OpenAPI({ description: 'Lists all information about the team whose id got provided.' })
|
@OpenAPI({ description: 'Lists all information about the team whose id got provided.' })
|
||||||
async getOne(@Param('id') id: number) {
|
async getOne(@Param('id') id: number) {
|
||||||
let runnerTeam = await this.runnerTeamRepository.findOne({ id: id }, { relations: ['parentGroup', 'contact'] });
|
let runnerTeam = await this.runnerTeamRepository.findOne({ id: id }, { relations: ['parentGroup', 'contact', 'runners', 'runners.scans', 'runners.scans.track'] });
|
||||||
if (!runnerTeam) { throw new RunnerTeamNotFoundError(); }
|
if (!runnerTeam) { throw new RunnerTeamNotFoundError(); }
|
||||||
return new ResponseRunnerTeam(runnerTeam);
|
return new ResponseRunnerTeam(runnerTeam);
|
||||||
}
|
}
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import { Request } from "express";
|
import { Request } from "express";
|
||||||
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam, Req, UseBefore } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, Put, QueryParam, Req, UseBefore } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { RunnerNotFoundError } from '../errors/RunnerErrors';
|
import { RunnerNotFoundError } from '../errors/RunnerErrors';
|
||||||
import { ScanIdsNotMatchingError, ScanNotFoundError } from '../errors/ScanErrors';
|
import { ScanIdsNotMatchingError, ScanNotFoundError } from '../errors/ScanErrors';
|
||||||
import { ScanStationNotFoundError } from '../errors/ScanStationErrors';
|
import { ScanStationNotFoundError } from '../errors/ScanStationErrors';
|
||||||
@@ -34,9 +34,16 @@ export class ScanController {
|
|||||||
@ResponseSchema(ResponseScan, { isArray: true })
|
@ResponseSchema(ResponseScan, { isArray: true })
|
||||||
@ResponseSchema(ResponseTrackScan, { isArray: true })
|
@ResponseSchema(ResponseTrackScan, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all scans (normal or track) from all runners. <br> This includes the scan\'s runner\'s distance ran.' })
|
@OpenAPI({ description: 'Lists all scans (normal or track) from all runners. <br> This includes the scan\'s runner\'s distance ran.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseScans: ResponseScan[] = new Array<ResponseScan>();
|
let responseScans: ResponseScan[] = new Array<ResponseScan>();
|
||||||
const scans = await this.scanRepository.find({ relations: ['runner', 'track', 'runner.scans', 'runner.group', 'runner.scans.track', 'card', 'station'] });
|
let scans: Array<Scan>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
scans = await this.scanRepository.find({ relations: ['runner', 'track'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
scans = await this.scanRepository.find({ relations: ['runner', 'track'] });
|
||||||
|
}
|
||||||
|
|
||||||
scans.forEach(scan => {
|
scans.forEach(scan => {
|
||||||
responseScans.push(scan.toResponse());
|
responseScans.push(scan.toResponse());
|
||||||
});
|
});
|
||||||
@@ -51,7 +58,7 @@ export class ScanController {
|
|||||||
@OnUndefined(ScanNotFoundError)
|
@OnUndefined(ScanNotFoundError)
|
||||||
@OpenAPI({ description: 'Lists all information about the scan whose id got provided. This includes the scan\'s runner\'s distance ran.' })
|
@OpenAPI({ description: 'Lists all information about the scan whose id got provided. This includes the scan\'s runner\'s distance ran.' })
|
||||||
async getOne(@Param('id') id: number) {
|
async getOne(@Param('id') id: number) {
|
||||||
let scan = await this.scanRepository.findOne({ id: id }, { relations: ['runner', 'track', 'runner.scans', 'runner.group', 'runner.scans.track', 'card', 'station'] })
|
let scan = await this.scanRepository.findOne({ id: id }, { relations: ['runner', 'track', 'runner.group', 'card', 'station'] })
|
||||||
if (!scan) { throw new ScanNotFoundError(); }
|
if (!scan) { throw new ScanNotFoundError(); }
|
||||||
return scan.toResponse();
|
return scan.toResponse();
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { ScanStationHasScansError, ScanStationIdsNotMatchingError, ScanStationNotFoundError } from '../errors/ScanStationErrors';
|
import { ScanStationHasScansError, ScanStationIdsNotMatchingError, ScanStationNotFoundError } from '../errors/ScanStationErrors';
|
||||||
import { TrackNotFoundError } from '../errors/TrackErrors';
|
import { TrackNotFoundError } from '../errors/TrackErrors';
|
||||||
import { CreateScanStation } from '../models/actions/create/CreateScanStation';
|
import { CreateScanStation } from '../models/actions/create/CreateScanStation';
|
||||||
@@ -26,9 +26,16 @@ export class ScanStationController {
|
|||||||
@Authorized("STATION:GET")
|
@Authorized("STATION:GET")
|
||||||
@ResponseSchema(ResponseScanStation, { isArray: true })
|
@ResponseSchema(ResponseScanStation, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all stations. <br> This includes their associated tracks.' })
|
@OpenAPI({ description: 'Lists all stations. <br> This includes their associated tracks.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseStations: ResponseScanStation[] = new Array<ResponseScanStation>();
|
let responseStations: ResponseScanStation[] = new Array<ResponseScanStation>();
|
||||||
const stations = await this.stationRepository.find({ relations: ['track'] });
|
let stations: Array<ScanStation>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
stations = await this.stationRepository.find({ relations: ['track'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
stations = await this.stationRepository.find({ relations: ['track'] });
|
||||||
|
}
|
||||||
|
|
||||||
stations.forEach(station => {
|
stations.forEach(station => {
|
||||||
responseStations.push(station.toResponse());
|
responseStations.push(station.toResponse());
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, QueryParam } from 'routing-controllers';
|
import { Authorized, Body, Delete, Get, JsonController, OnUndefined, Param, Post, QueryParam } from 'routing-controllers';
|
||||||
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
||||||
import { getConnectionManager, Repository } from 'typeorm';
|
import { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { StatsClientNotFoundError } from '../errors/StatsClientErrors';
|
import { StatsClientNotFoundError } from '../errors/StatsClientErrors';
|
||||||
import { TrackNotFoundError } from "../errors/TrackErrors";
|
import { TrackNotFoundError } from "../errors/TrackErrors";
|
||||||
import { CreateStatsClient } from '../models/actions/create/CreateStatsClient';
|
import { CreateStatsClient } from '../models/actions/create/CreateStatsClient';
|
||||||
@@ -24,9 +24,16 @@ export class StatsClientController {
|
|||||||
@Authorized("STATSCLIENT:GET")
|
@Authorized("STATSCLIENT:GET")
|
||||||
@ResponseSchema(ResponseStatsClient, { isArray: true })
|
@ResponseSchema(ResponseStatsClient, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all stats clients. Please remember that the key can only be viewed on creation.' })
|
@OpenAPI({ description: 'Lists all stats clients. Please remember that the key can only be viewed on creation.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseClients: ResponseStatsClient[] = new Array<ResponseStatsClient>();
|
let responseClients: ResponseStatsClient[] = new Array<ResponseStatsClient>();
|
||||||
const clients = await this.clientRepository.find();
|
let clients: Array<StatsClient>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
clients = await this.clientRepository.find({ skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
clients = await this.clientRepository.find();
|
||||||
|
}
|
||||||
|
|
||||||
clients.forEach(clients => {
|
clients.forEach(clients => {
|
||||||
responseClients.push(new ResponseStatsClient(clients));
|
responseClients.push(new ResponseStatsClient(clients));
|
||||||
});
|
});
|
||||||
|
@@ -3,6 +3,7 @@ import { OpenAPI, ResponseSchema } from 'routing-controllers-openapi';
|
|||||||
import { getConnection } from 'typeorm';
|
import { getConnection } from 'typeorm';
|
||||||
import StatsAuth from '../middlewares/StatsAuth';
|
import StatsAuth from '../middlewares/StatsAuth';
|
||||||
import { Donation } from '../models/entities/Donation';
|
import { Donation } from '../models/entities/Donation';
|
||||||
|
import { Donor } from '../models/entities/Donor';
|
||||||
import { Runner } from '../models/entities/Runner';
|
import { Runner } from '../models/entities/Runner';
|
||||||
import { RunnerOrganization } from '../models/entities/RunnerOrganization';
|
import { RunnerOrganization } from '../models/entities/RunnerOrganization';
|
||||||
import { RunnerTeam } from '../models/entities/RunnerTeam';
|
import { RunnerTeam } from '../models/entities/RunnerTeam';
|
||||||
@@ -21,14 +22,26 @@ export class StatsController {
|
|||||||
@ResponseSchema(ResponseStats)
|
@ResponseSchema(ResponseStats)
|
||||||
@OpenAPI({ description: "A very basic stats endpoint providing basic counters for a dashboard or simmilar" })
|
@OpenAPI({ description: "A very basic stats endpoint providing basic counters for a dashboard or simmilar" })
|
||||||
async get() {
|
async get() {
|
||||||
let connection = getConnection();
|
const connection = getConnection();
|
||||||
let runners = await connection.getRepository(Runner).find({ relations: ['scans', 'scans.track'] });
|
const runners = await connection.getRepository(Runner).count();
|
||||||
let teams = await connection.getRepository(RunnerTeam).find();
|
const teams = await connection.getRepository(RunnerTeam).count();
|
||||||
let orgs = await connection.getRepository(RunnerOrganization).find();
|
const orgs = await connection.getRepository(RunnerOrganization).count();
|
||||||
let users = await connection.getRepository(User).find();
|
const users = await connection.getRepository(User).count();
|
||||||
let scans = await connection.getRepository(Scan).find();
|
const scans = await connection.getRepository(Scan).count({ where: { valid: true } });
|
||||||
|
|
||||||
|
const distance_query = await connection.getRepository(Scan).createQueryBuilder('scan')
|
||||||
|
.leftJoinAndSelect("scan.track", "track").where("scan.valid = TRUE")
|
||||||
|
.select("SUM(track.distance)", "sum_track").addSelect("SUM(_distance)", "sum_distance")
|
||||||
|
.getRawOne();
|
||||||
|
let distace = parseInt(distance_query.sum_track)
|
||||||
|
if (distance_query.sum_distance) {
|
||||||
|
distace += parseInt(distance_query.sum_distance)
|
||||||
|
}
|
||||||
|
|
||||||
let donations = await connection.getRepository(Donation).find({ relations: ['runner', 'runner.scans', 'runner.scans.track'] });
|
let donations = await connection.getRepository(Donation).find({ relations: ['runner', 'runner.scans', 'runner.scans.track'] });
|
||||||
return new ResponseStats(runners, teams, orgs, users, scans, donations)
|
const donors = await connection.getRepository(Donor).count();
|
||||||
|
|
||||||
|
return new ResponseStats(runners, teams, orgs, users, scans, donations, distace, donors)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get("/runners/distance")
|
@Get("/runners/distance")
|
||||||
@@ -41,7 +54,6 @@ export class StatsController {
|
|||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
let topRunners = runners.sort((runner1, runner2) => runner2.distance - runner1.distance).slice(0, 10);
|
let topRunners = runners.sort((runner1, runner2) => runner2.distance - runner1.distance).slice(0, 10);
|
||||||
return runners;
|
|
||||||
let responseRunners: ResponseStatsRunner[] = new Array<ResponseStatsRunner>();
|
let responseRunners: ResponseStatsRunner[] = new Array<ResponseStatsRunner>();
|
||||||
topRunners.forEach(runner => {
|
topRunners.forEach(runner => {
|
||||||
responseRunners.push(new ResponseStatsRunner(runner));
|
responseRunners.push(new ResponseStatsRunner(runner));
|
||||||
@@ -54,7 +66,7 @@ export class StatsController {
|
|||||||
@ResponseSchema(ResponseStatsRunner, { isArray: true })
|
@ResponseSchema(ResponseStatsRunner, { isArray: true })
|
||||||
@OpenAPI({ description: "Returns the top ten runners by donations.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
@OpenAPI({ description: "Returns the top ten runners by donations.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
||||||
async getTopRunnersByDonations() {
|
async getTopRunnersByDonations() {
|
||||||
let runners = await getConnection().getRepository(Runner).find({ relations: ['scans', 'group', 'distanceDonations', 'scans.track'] });
|
let runners = await getConnection().getRepository(Runner).find({ relations: ['group', 'distanceDonations', 'distanceDonations.runner', 'distanceDonations.runner.scans', 'distanceDonations.runner.scans.track'] });
|
||||||
if (!runners || runners.length == 0) {
|
if (!runners || runners.length == 0) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
@@ -107,7 +119,7 @@ export class StatsController {
|
|||||||
@ResponseSchema(ResponseStatsTeam, { isArray: true })
|
@ResponseSchema(ResponseStatsTeam, { isArray: true })
|
||||||
@OpenAPI({ description: "Returns the top ten teams by distance.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
@OpenAPI({ description: "Returns the top ten teams by distance.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
||||||
async getTopTeamsByDistance() {
|
async getTopTeamsByDistance() {
|
||||||
let teams = await getConnection().getRepository(RunnerTeam).find({ relations: ['runners', 'runners.scans', 'runners.distanceDonations', 'runners.scans.track'] });
|
let teams = await getConnection().getRepository(RunnerTeam).find({ relations: ['parentGroup', 'runners', 'runners.scans', 'runners.scans.track'] });
|
||||||
if (!teams || teams.length == 0) {
|
if (!teams || teams.length == 0) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
@@ -124,7 +136,7 @@ export class StatsController {
|
|||||||
@ResponseSchema(ResponseStatsTeam, { isArray: true })
|
@ResponseSchema(ResponseStatsTeam, { isArray: true })
|
||||||
@OpenAPI({ description: "Returns the top ten teams by donations.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
@OpenAPI({ description: "Returns the top ten teams by donations.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
||||||
async getTopTeamsByDonations() {
|
async getTopTeamsByDonations() {
|
||||||
let teams = await getConnection().getRepository(RunnerTeam).find({ relations: ['runners', 'runners.scans', 'runners.distanceDonations', 'runners.scans.track'] });
|
let teams = await getConnection().getRepository(RunnerTeam).find({ relations: ['parentGroup', 'runners', 'runners.scans', 'runners.distanceDonations', 'runners.scans.track'] });
|
||||||
if (!teams || teams.length == 0) {
|
if (!teams || teams.length == 0) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
@@ -158,7 +170,7 @@ export class StatsController {
|
|||||||
@ResponseSchema(ResponseStatsOrgnisation, { isArray: true })
|
@ResponseSchema(ResponseStatsOrgnisation, { isArray: true })
|
||||||
@OpenAPI({ description: "Returns the top ten organizations by donations.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
@OpenAPI({ description: "Returns the top ten organizations by donations.", security: [{ "StatsApiToken": [] }, { "AuthToken": [] }, { "RefreshTokenCookie": [] }] })
|
||||||
async getTopOrgsByDonations() {
|
async getTopOrgsByDonations() {
|
||||||
let orgs = await getConnection().getRepository(RunnerOrganization).find({ relations: ['runners', 'runners.scans', 'runners.distanceDonations', 'runners.scans.track', 'teams', 'teams.runners', 'teams.runners.scans', 'teams.runners.distanceDonations', 'teams.runners.scans.track'] });
|
let orgs = await getConnection().getRepository(RunnerOrganization).find({ relations: ['runners', 'runners.distanceDonations', 'runners.distanceDonations.runner', 'runners.distanceDonations.runner.scans', 'runners.distanceDonations.runner.scans.track', 'teams', 'teams.runners', 'teams.runners.distanceDonations', 'teams.runners.distanceDonations.runner', 'teams.runners.distanceDonations.runner.scans', 'teams.runners.distanceDonations.runner.scans.track'] });
|
||||||
if (!orgs || orgs.length == 0) {
|
if (!orgs || orgs.length == 0) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { TrackHasScanStationsError, TrackIdsNotMatchingError, TrackLapTimeCantBeNegativeError, TrackNotFoundError } from "../errors/TrackErrors";
|
import { TrackHasScanStationsError, TrackIdsNotMatchingError, TrackLapTimeCantBeNegativeError, TrackNotFoundError } from "../errors/TrackErrors";
|
||||||
import { CreateTrack } from '../models/actions/create/CreateTrack';
|
import { CreateTrack } from '../models/actions/create/CreateTrack';
|
||||||
import { UpdateTrack } from '../models/actions/update/UpdateTrack';
|
import { UpdateTrack } from '../models/actions/update/UpdateTrack';
|
||||||
@@ -25,9 +25,17 @@ export class TrackController {
|
|||||||
@Authorized("TRACK:GET")
|
@Authorized("TRACK:GET")
|
||||||
@ResponseSchema(ResponseTrack, { isArray: true })
|
@ResponseSchema(ResponseTrack, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all tracks.' })
|
@OpenAPI({ description: 'Lists all tracks.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseTracks: ResponseTrack[] = new Array<ResponseTrack>();
|
let responseTracks: ResponseTrack[] = new Array<ResponseTrack>();
|
||||||
const tracks = await this.trackRepository.find();
|
let tracks: Array<Track>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
tracks = await this.trackRepository.find({ skip: page * page_size, take: page_size });
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
tracks = await this.trackRepository.find();
|
||||||
|
}
|
||||||
|
|
||||||
tracks.forEach(track => {
|
tracks.forEach(track => {
|
||||||
responseTracks.push(new ResponseTrack(track));
|
responseTracks.push(new ResponseTrack(track));
|
||||||
});
|
});
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { PasswordMustContainLowercaseLetterError, PasswordMustContainNumberError, PasswordMustContainUppercaseLetterError, PasswordTooShortError, UserDeletionNotConfirmedError, UserIdsNotMatchingError, UsernameContainsIllegalCharacterError, UserNotFoundError } from '../errors/UserErrors';
|
import { PasswordMustContainLowercaseLetterError, PasswordMustContainNumberError, PasswordMustContainUppercaseLetterError, PasswordTooShortError, UserDeletionNotConfirmedError, UserIdsNotMatchingError, UserNotFoundError, UsernameContainsIllegalCharacterError } from '../errors/UserErrors';
|
||||||
import { UserGroupNotFoundError } from '../errors/UserGroupErrors';
|
import { UserGroupNotFoundError } from '../errors/UserGroupErrors';
|
||||||
import { CreateUser } from '../models/actions/create/CreateUser';
|
import { CreateUser } from '../models/actions/create/CreateUser';
|
||||||
import { UpdateUser } from '../models/actions/update/UpdateUser';
|
import { UpdateUser } from '../models/actions/update/UpdateUser';
|
||||||
@@ -28,9 +28,17 @@ export class UserController {
|
|||||||
@Authorized("USER:GET")
|
@Authorized("USER:GET")
|
||||||
@ResponseSchema(ResponseUser, { isArray: true })
|
@ResponseSchema(ResponseUser, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all users. <br> This includes their groups and permissions granted to them.' })
|
@OpenAPI({ description: 'Lists all users. <br> This includes their groups and permissions granted to them.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseUsers: ResponseUser[] = new Array<ResponseUser>();
|
let responseUsers: ResponseUser[] = new Array<ResponseUser>();
|
||||||
const users = await this.userRepository.find({ relations: ['permissions', 'groups', 'groups.permissions'] });
|
let users: Array<User>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
users = await this.userRepository.find({ relations: ['permissions', 'groups', 'groups.permissions'], skip: page * page_size, take: page_size });
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
users = await this.userRepository.find({ relations: ['permissions', 'groups', 'groups.permissions'] });
|
||||||
|
}
|
||||||
|
|
||||||
users.forEach(user => {
|
users.forEach(user => {
|
||||||
responseUsers.push(new ResponseUser(user));
|
responseUsers.push(new ResponseUser(user));
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { Authorized, 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 { Repository, getConnectionManager } from 'typeorm';
|
||||||
import { UserGroupIdsNotMatchingError, UserGroupNotFoundError } from '../errors/UserGroupErrors';
|
import { UserGroupIdsNotMatchingError, UserGroupNotFoundError } from '../errors/UserGroupErrors';
|
||||||
import { CreateUserGroup } from '../models/actions/create/CreateUserGroup';
|
import { CreateUserGroup } from '../models/actions/create/CreateUserGroup';
|
||||||
import { UpdateUserGroup } from '../models/actions/update/UpdateUserGroup';
|
import { UpdateUserGroup } from '../models/actions/update/UpdateUserGroup';
|
||||||
@@ -27,9 +27,16 @@ export class UserGroupController {
|
|||||||
@Authorized("USERGROUP:GET")
|
@Authorized("USERGROUP:GET")
|
||||||
@ResponseSchema(ResponseUserGroup, { isArray: true })
|
@ResponseSchema(ResponseUserGroup, { isArray: true })
|
||||||
@OpenAPI({ description: 'Lists all groups. <br> The information provided might change while the project continues to evolve.' })
|
@OpenAPI({ description: 'Lists all groups. <br> The information provided might change while the project continues to evolve.' })
|
||||||
async getAll() {
|
async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) {
|
||||||
let responseGroups: ResponseUserGroup[] = new Array<ResponseUserGroup>();
|
let responseGroups: ResponseUserGroup[] = new Array<ResponseUserGroup>();
|
||||||
const groups = await this.userGroupsRepository.find({ relations: ['permissions'] });
|
let groups: Array<UserGroup>;
|
||||||
|
|
||||||
|
if (page != undefined) {
|
||||||
|
groups = await this.userGroupsRepository.find({ relations: ['permissions'], skip: page * page_size, take: page_size });
|
||||||
|
} else {
|
||||||
|
groups = await this.userGroupsRepository.find({ relations: ['permissions'] });
|
||||||
|
}
|
||||||
|
|
||||||
groups.forEach(group => {
|
groups.forEach(group => {
|
||||||
responseGroups.push(group.toResponse());
|
responseGroups.push(group.toResponse());
|
||||||
});
|
});
|
||||||
|
@@ -18,9 +18,19 @@ export class Mailer {
|
|||||||
*/
|
*/
|
||||||
public static async sendResetMail(to_address: string, token: string, locale: string = "en") {
|
public static async sendResetMail(to_address: string, token: string, locale: string = "en") {
|
||||||
try {
|
try {
|
||||||
await axios.post(`${Mailer.base}/reset?locale=${locale}&key=${Mailer.key}`, {
|
await axios.request({
|
||||||
address: to_address,
|
method: 'POST',
|
||||||
resetKey: token
|
url: `${Mailer.base}/api/v1/email`,
|
||||||
|
headers: {
|
||||||
|
authorization: `Bearer ${Mailer.key}`,
|
||||||
|
'content-type': 'application/json'
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
to: to_address,
|
||||||
|
templateName: 'password-reset',
|
||||||
|
language: locale,
|
||||||
|
data: { token: token }
|
||||||
|
}
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (Mailer.testing) { return true; }
|
if (Mailer.testing) { return true; }
|
||||||
@@ -32,12 +42,26 @@ export class Mailer {
|
|||||||
* Function for sending a runner selfservice welcome mail.
|
* Function for sending a runner selfservice welcome mail.
|
||||||
* @param to_address The address the mail will be sent to. Should always get pulled from a runner object.
|
* @param to_address The address the mail will be sent to. Should always get pulled from a runner object.
|
||||||
* @param token The requested selfservice token - will be combined with the app_url to generate a selfservice profile link.
|
* @param token The requested selfservice token - will be combined with the app_url to generate a selfservice profile link.
|
||||||
*/
|
*/
|
||||||
public static async sendSelfserviceWelcomeMail(to_address: string, token: string, locale: string = "en") {
|
public static async sendSelfserviceWelcomeMail(to_address: string, runner_id: number, firstname: string, middlename: string, lastname: string, token: string, locale: string = "en") {
|
||||||
try {
|
try {
|
||||||
await axios.post(`${Mailer.base}/registration?locale=${locale}&key=${Mailer.key}`, {
|
await axios.request({
|
||||||
address: to_address,
|
method: 'POST',
|
||||||
selfserviceToken: token
|
url: `${Mailer.base}/api/v1/email`,
|
||||||
|
headers: {
|
||||||
|
authorization: `Bearer ${Mailer.key}`,
|
||||||
|
'content-type': 'application/json'
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
to: to_address,
|
||||||
|
templateName: 'welcome',
|
||||||
|
language: locale,
|
||||||
|
data: {
|
||||||
|
name: `${firstname} ${middlename} ${lastname}`,
|
||||||
|
barcode_content: `${runner_id}`,
|
||||||
|
link: `${process.env.SELFSERVICE_URL}/profile/${token}`
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (Mailer.testing) { return true; }
|
if (Mailer.testing) { return true; }
|
||||||
@@ -49,12 +73,26 @@ export class Mailer {
|
|||||||
* Function for sending a runner selfservice link forgotten mail.
|
* Function for sending a runner selfservice link forgotten mail.
|
||||||
* @param to_address The address the mail will be sent to. Should always get pulled from a runner object.
|
* @param to_address The address the mail will be sent to. Should always get pulled from a runner object.
|
||||||
* @param token The requested selfservice token - will be combined with the app_url to generate a selfservice profile link.
|
* @param token The requested selfservice token - will be combined with the app_url to generate a selfservice profile link.
|
||||||
*/
|
*/
|
||||||
public static async sendSelfserviceForgottenMail(to_address: string, token: string, locale: string = "en") {
|
public static async sendSelfserviceForgottenMail(to_address: string, runner_id: number, firstname: string, middlename: string, lastname: string, token: string, locale: string = "en") {
|
||||||
try {
|
try {
|
||||||
await axios.post(`${Mailer.base}/registration_forgot?locale=${locale}&key=${Mailer.key}`, {
|
await axios.request({
|
||||||
address: to_address,
|
method: 'POST',
|
||||||
selfserviceToken: token
|
url: `${Mailer.base}/api/v1/email`,
|
||||||
|
headers: {
|
||||||
|
authorization: `Bearer ${Mailer.key}`,
|
||||||
|
'content-type': 'application/json'
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
to: to_address,
|
||||||
|
templateName: 'welcome',
|
||||||
|
language: locale,
|
||||||
|
data: {
|
||||||
|
name: `${firstname} ${middlename} ${lastname}`,
|
||||||
|
barcode_content: `${runner_id}`,
|
||||||
|
link: `${process.env.SELFSERVICE_URL}/profile/${token}`
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (Mailer.testing) { return true; }
|
if (Mailer.testing) { return true; }
|
||||||
|
@@ -33,6 +33,7 @@ export class CreateDistanceDonation extends CreateDonation {
|
|||||||
let newDonation = new DistanceDonation;
|
let newDonation = new DistanceDonation;
|
||||||
|
|
||||||
newDonation.amountPerDistance = this.amountPerDistance;
|
newDonation.amountPerDistance = this.amountPerDistance;
|
||||||
|
newDonation.paidAmount = this.paidAmount;
|
||||||
newDonation.donor = await this.getDonor();
|
newDonation.donor = await this.getDonor();
|
||||||
newDonation.runner = await this.getRunner();
|
newDonation.runner = await this.getRunner();
|
||||||
|
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { IsInt, IsPositive } from 'class-validator';
|
import { IsInt, IsOptional, IsPositive } from 'class-validator';
|
||||||
import { getConnection } from 'typeorm';
|
import { getConnection } from 'typeorm';
|
||||||
import { DonorNotFoundError } from '../../../errors/DonorErrors';
|
import { DonorNotFoundError } from '../../../errors/DonorErrors';
|
||||||
import { Donation } from '../../entities/Donation';
|
import { Donation } from '../../entities/Donation';
|
||||||
@@ -16,6 +16,13 @@ export abstract class CreateDonation {
|
|||||||
@IsPositive()
|
@IsPositive()
|
||||||
donor: number;
|
donor: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The donation's paid amount in the smalles unit of your currency (default: euro cent).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
@IsOptional()
|
||||||
|
paidAmount?: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new Donation entity from this.
|
* Creates a new Donation entity from this.
|
||||||
*/
|
*/
|
||||||
|
@@ -21,6 +21,7 @@ export class CreateFixedDonation extends CreateDonation {
|
|||||||
let newDonation = new FixedDonation;
|
let newDonation = new FixedDonation;
|
||||||
|
|
||||||
newDonation.amount = this.amount;
|
newDonation.amount = this.amount;
|
||||||
|
newDonation.paidAmount = this.paidAmount;
|
||||||
newDonation.donor = await this.getDonor();
|
newDonation.donor = await this.getDonor();
|
||||||
|
|
||||||
return newDonation;
|
return newDonation;
|
||||||
|
@@ -57,11 +57,12 @@ export class CreateTrackScan {
|
|||||||
* @returns The runnerCard whom's id you provided.
|
* @returns The runnerCard whom's id you provided.
|
||||||
*/
|
*/
|
||||||
public async getCard(): Promise<RunnerCard> {
|
public async getCard(): Promise<RunnerCard> {
|
||||||
const track = await getConnection().getRepository(RunnerCard).findOne({ id: this.card }, { relations: ["runner"] });
|
const id = this.card % 200000000000;
|
||||||
if (!track) {
|
const runnerCard = await getConnection().getRepository(RunnerCard).findOne({ id: id }, { relations: ["runner"] });
|
||||||
|
if (!runnerCard) {
|
||||||
throw new RunnerCardNotFoundError();
|
throw new RunnerCardNotFoundError();
|
||||||
}
|
}
|
||||||
return track;
|
return runnerCard;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -85,14 +86,13 @@ export class CreateTrackScan {
|
|||||||
* @returns The validated scan with it's laptime set.
|
* @returns The validated scan with it's laptime set.
|
||||||
*/
|
*/
|
||||||
public async validateScan(scan: TrackScan): Promise<TrackScan> {
|
public async validateScan(scan: TrackScan): Promise<TrackScan> {
|
||||||
const scans = await getConnection().getRepository(TrackScan).find({ where: { runner: scan.runner, valid: true }, relations: ["track"] });
|
const latestScan = await getConnection().getRepository(TrackScan).findOne({ where: { runner: scan.runner, valid: true }, relations: ["track"], order: { id: 'DESC' } });
|
||||||
if (scans.length == 0) {
|
if (!latestScan) {
|
||||||
scan.lapTime = 0;
|
scan.lapTime = 0;
|
||||||
scan.valid = true;
|
scan.valid = true;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
const newestScan = scans[scans.length - 1];
|
scan.lapTime = scan.timestamp - latestScan.timestamp;
|
||||||
scan.lapTime = scan.timestamp - newestScan.timestamp;
|
|
||||||
scan.valid = (scan.lapTime > scan.track.minimumLapTime);
|
scan.valid = (scan.lapTime > scan.track.minimumLapTime);
|
||||||
}
|
}
|
||||||
return scan;
|
return scan;
|
||||||
|
@@ -114,7 +114,7 @@ export class CreateUser {
|
|||||||
newUser.groups = await this.getGroups();
|
newUser.groups = await this.getGroups();
|
||||||
newUser.enabled = this.enabled;
|
newUser.enabled = this.enabled;
|
||||||
|
|
||||||
if (!this.profilePic) { newUser.profilePic = `https://dev.lauf-fuer-kaya.de/lfk-logo.png`; }
|
if (!this.profilePic) { newUser.profilePic = `https://lauf-fuer-kaya.de/lfk-logo.png`; }
|
||||||
else { newUser.profilePic = this.profilePic; }
|
else { newUser.profilePic = this.profilePic; }
|
||||||
|
|
||||||
return newUser;
|
return newUser;
|
||||||
|
@@ -32,6 +32,7 @@ export class UpdateDistanceDonation extends UpdateDonation {
|
|||||||
*/
|
*/
|
||||||
public async update(donation: DistanceDonation): Promise<DistanceDonation> {
|
public async update(donation: DistanceDonation): Promise<DistanceDonation> {
|
||||||
donation.amountPerDistance = this.amountPerDistance;
|
donation.amountPerDistance = this.amountPerDistance;
|
||||||
|
donation.paidAmount = this.paidAmount;
|
||||||
donation.donor = await this.getDonor();
|
donation.donor = await this.getDonor();
|
||||||
donation.runner = await this.getRunner();
|
donation.runner = await this.getRunner();
|
||||||
|
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { IsInt, IsPositive } from 'class-validator';
|
import { IsInt, IsOptional, IsPositive } from 'class-validator';
|
||||||
import { getConnection } from 'typeorm';
|
import { getConnection } from 'typeorm';
|
||||||
import { DonorNotFoundError } from '../../../errors/DonorErrors';
|
import { DonorNotFoundError } from '../../../errors/DonorErrors';
|
||||||
import { Donation } from '../../entities/Donation';
|
import { Donation } from '../../entities/Donation';
|
||||||
@@ -23,6 +23,13 @@ export abstract class UpdateDonation {
|
|||||||
@IsPositive()
|
@IsPositive()
|
||||||
donor: number;
|
donor: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The donation's paid amount in the smalles unit of your currency (default: euro cent).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
@IsOptional()
|
||||||
|
paidAmount?: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new Donation entity from this.
|
* Creates a new Donation entity from this.
|
||||||
*/
|
*/
|
||||||
|
@@ -20,6 +20,7 @@ export class UpdateFixedDonation extends UpdateDonation {
|
|||||||
*/
|
*/
|
||||||
public async update(donation: FixedDonation): Promise<FixedDonation> {
|
public async update(donation: FixedDonation): Promise<FixedDonation> {
|
||||||
donation.amount = this.amount;
|
donation.amount = this.amount;
|
||||||
|
donation.paidAmount = this.paidAmount;
|
||||||
donation.donor = await this.getDonor();
|
donation.donor = await this.getDonor();
|
||||||
|
|
||||||
return donation;
|
return donation;
|
||||||
|
@@ -124,7 +124,7 @@ export class UpdateUser {
|
|||||||
user.phone = this.phone;
|
user.phone = this.phone;
|
||||||
user.groups = await this.getGroups();
|
user.groups = await this.getGroups();
|
||||||
|
|
||||||
if (!this.profilePic) { user.profilePic = `https://dev.lauf-fuer-kaya.de/lfk-logo.png`; }
|
if (!this.profilePic) { user.profilePic = `https://lauf-fuer-kaya.de/lfk-logo.png`; }
|
||||||
else { user.profilePic = this.profilePic; }
|
else { user.profilePic = this.profilePic; }
|
||||||
|
|
||||||
return user;
|
return user;
|
||||||
|
@@ -2,7 +2,7 @@ import {
|
|||||||
IsInt,
|
IsInt,
|
||||||
IsNotEmpty
|
IsNotEmpty
|
||||||
} from "class-validator";
|
} from "class-validator";
|
||||||
import { Entity, ManyToOne, PrimaryGeneratedColumn, TableInheritance } from "typeorm";
|
import { Column, Entity, ManyToOne, PrimaryGeneratedColumn, TableInheritance } from "typeorm";
|
||||||
import { ResponseDonation } from '../responses/ResponseDonation';
|
import { ResponseDonation } from '../responses/ResponseDonation';
|
||||||
import { Donor } from './Donor';
|
import { Donor } from './Donor';
|
||||||
|
|
||||||
@@ -34,6 +34,13 @@ export abstract class Donation {
|
|||||||
*/
|
*/
|
||||||
public abstract get amount(): number;
|
public abstract get amount(): number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The donation's paid amount in cents (or whatever your currency's smallest unit is.).
|
||||||
|
* Used to mark donations as paid.
|
||||||
|
*/
|
||||||
|
@Column({ nullable: true })
|
||||||
|
@IsInt()
|
||||||
|
paidAmount: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Turns this entity into it's response class.
|
* Turns this entity into it's response class.
|
||||||
|
@@ -33,6 +33,15 @@ export class Donor extends Participant {
|
|||||||
return this.donations.reduce((sum, current) => sum + current.amount, 0);
|
return this.donations.reduce((sum, current) => sum + current.amount, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the total paid donations of a donor based on his linked donations.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
public get paidDonationAmount(): number {
|
||||||
|
if (!this.donations) { return 0; }
|
||||||
|
return this.donations.reduce((sum, current) => sum + current.paidAmount, 0);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Turns this entity into it's response class.
|
* Turns this entity into it's response class.
|
||||||
*/
|
*/
|
||||||
|
@@ -52,13 +52,7 @@ export class RunnerCard {
|
|||||||
* Generates a ean-13 compliant string for barcode generation.
|
* Generates a ean-13 compliant string for barcode generation.
|
||||||
*/
|
*/
|
||||||
public get code(): string {
|
public get code(): string {
|
||||||
const multiply = [1, 3];
|
return this.paddedId
|
||||||
let total = 0;
|
|
||||||
this.paddedId.split('').forEach((letter, index) => {
|
|
||||||
total += parseInt(letter, 10) * multiply[index % 2];
|
|
||||||
});
|
|
||||||
const checkSum = (Math.ceil(total / 10) * 10) - total;
|
|
||||||
return this.paddedId + checkSum.toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -67,10 +61,11 @@ export class RunnerCard {
|
|||||||
private get paddedId(): string {
|
private get paddedId(): string {
|
||||||
let id: string = this.id.toString();
|
let id: string = this.id.toString();
|
||||||
|
|
||||||
if (id.length > 12) {
|
if (id.length > 11) {
|
||||||
throw new RunnerCardIdOutOfRangeError();
|
throw new RunnerCardIdOutOfRangeError();
|
||||||
}
|
}
|
||||||
while (id.length < 12) { id = '0' + id; }
|
while (id.length < 11) { id = '0' + id; }
|
||||||
|
id = '2' + id;
|
||||||
|
|
||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
@@ -51,6 +51,9 @@ export abstract class RunnerGroup {
|
|||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
public get distance(): number {
|
public get distance(): number {
|
||||||
|
if (!this.runners || this.runners.length == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
return this.runners.reduce((sum, current) => sum + current.distance, 0);
|
return this.runners.reduce((sum, current) => sum + current.distance, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
7
src/models/enums/DonationStatus.ts
Normal file
7
src/models/enums/DonationStatus.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
/**
|
||||||
|
* This enum contains all status a donation can inherit regarding it's payment status.
|
||||||
|
*/
|
||||||
|
export enum DonationStatus {
|
||||||
|
OPEN = 'OPEN',
|
||||||
|
PAID = 'PAID'
|
||||||
|
}
|
@@ -1,5 +1,6 @@
|
|||||||
import { IsInt, IsNotEmpty, IsPositive } from "class-validator";
|
import { IsInt, IsNotEmpty, IsPositive } from "class-validator";
|
||||||
import { Donation } from '../entities/Donation';
|
import { Donation } from '../entities/Donation';
|
||||||
|
import { DonationStatus } from '../enums/DonationStatus';
|
||||||
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
||||||
import { IResponse } from './IResponse';
|
import { IResponse } from './IResponse';
|
||||||
import { ResponseDonor } from './ResponseDonor';
|
import { ResponseDonor } from './ResponseDonor';
|
||||||
@@ -15,6 +16,12 @@ export class ResponseDonation implements IResponse {
|
|||||||
*/
|
*/
|
||||||
responseType: ResponseObjectType = ResponseObjectType.DONATION;
|
responseType: ResponseObjectType = ResponseObjectType.DONATION;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The donation's payment status.
|
||||||
|
* Provides you with a quick indicator of it's payment status.
|
||||||
|
*/
|
||||||
|
status: DonationStatus;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The donation's id.
|
* The donation's id.
|
||||||
*/
|
*/
|
||||||
@@ -34,13 +41,28 @@ export class ResponseDonation implements IResponse {
|
|||||||
@IsInt()
|
@IsInt()
|
||||||
amount: number;
|
amount: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The donation's paid amount in the smalles unit of your currency (default: euro cent).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
paidAmount: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a ResponseDonation object from a scan.
|
* Creates a ResponseDonation object from a scan.
|
||||||
* @param donation The donation the response shall be build for.
|
* @param donation The donation the response shall be build for.
|
||||||
*/
|
*/
|
||||||
public constructor(donation: Donation) {
|
public constructor(donation: Donation) {
|
||||||
this.id = donation.id;
|
this.id = donation.id;
|
||||||
this.donor = donation.donor.toResponse();
|
if (donation.donor) {
|
||||||
|
this.donor = donation.donor.toResponse();
|
||||||
|
}
|
||||||
this.amount = donation.amount;
|
this.amount = donation.amount;
|
||||||
|
this.paidAmount = donation.paidAmount || 0;
|
||||||
|
if (this.paidAmount < this.amount) {
|
||||||
|
this.status = DonationStatus.OPEN;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.status = DonationStatus.PAID;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -4,6 +4,7 @@ import {
|
|||||||
import { Donor } from '../entities/Donor';
|
import { Donor } from '../entities/Donor';
|
||||||
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
||||||
import { IResponse } from './IResponse';
|
import { IResponse } from './IResponse';
|
||||||
|
import { ResponseDonation } from './ResponseDonation';
|
||||||
import { ResponseParticipant } from './ResponseParticipant';
|
import { ResponseParticipant } from './ResponseParticipant';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -28,6 +29,14 @@ export class ResponseDonor extends ResponseParticipant implements IResponse {
|
|||||||
@IsInt()
|
@IsInt()
|
||||||
donationAmount: number;
|
donationAmount: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the total paid donations of a donor based on his linked donations.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
paidDonationAmount: number;
|
||||||
|
|
||||||
|
donations: Array<ResponseDonation>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a ResponseRunner object from a runner.
|
* Creates a ResponseRunner object from a runner.
|
||||||
* @param runner The user the response shall be build for.
|
* @param runner The user the response shall be build for.
|
||||||
@@ -36,5 +45,12 @@ export class ResponseDonor extends ResponseParticipant implements IResponse {
|
|||||||
super(donor);
|
super(donor);
|
||||||
this.receiptNeeded = donor.receiptNeeded;
|
this.receiptNeeded = donor.receiptNeeded;
|
||||||
this.donationAmount = donor.donationAmount;
|
this.donationAmount = donor.donationAmount;
|
||||||
|
this.paidDonationAmount = donor.paidDonationAmount;
|
||||||
|
this.donations = new Array<ResponseDonation>();
|
||||||
|
if (donor.donations?.length > 0) {
|
||||||
|
for (const donation of donor.donations) {
|
||||||
|
this.donations.push(donation.toResponse())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { IsInt, IsNotEmpty, IsObject, IsOptional, IsString } from "class-validator";
|
import { IsInt, IsNotEmpty, IsNumber, IsObject, IsOptional, IsString } from "class-validator";
|
||||||
import { RunnerGroup } from '../entities/RunnerGroup';
|
import { RunnerGroup } from '../entities/RunnerGroup';
|
||||||
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
||||||
import { IResponse } from './IResponse';
|
import { IResponse } from './IResponse';
|
||||||
@@ -36,6 +36,10 @@ export abstract class ResponseRunnerGroup implements IResponse {
|
|||||||
@IsOptional()
|
@IsOptional()
|
||||||
contact?: ResponseGroupContact;
|
contact?: ResponseGroupContact;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsNumber()
|
||||||
|
total_distance: number
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a ResponseRunnerGroup object from a runnerGroup.
|
* Creates a ResponseRunnerGroup object from a runnerGroup.
|
||||||
* @param group The runnerGroup the response shall be build for.
|
* @param group The runnerGroup the response shall be build for.
|
||||||
@@ -44,5 +48,6 @@ export abstract class ResponseRunnerGroup implements IResponse {
|
|||||||
this.id = group.id;
|
this.id = group.id;
|
||||||
this.name = group.name;
|
this.name = group.name;
|
||||||
if (group.contact) { this.contact = group.contact.toResponse(); };
|
if (group.contact) { this.contact = group.contact.toResponse(); };
|
||||||
|
if (group.runners) { this.total_distance = group.runners.reduce((p, c) => p + c.distance, 0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -67,6 +67,9 @@ export class ResponseRunnerOrganization extends ResponseRunnerGroup implements I
|
|||||||
for (let team of org.teams) {
|
for (let team of org.teams) {
|
||||||
this.teams.push(team.toResponse());
|
this.teams.push(team.toResponse());
|
||||||
}
|
}
|
||||||
|
for (const team of this.teams) {
|
||||||
|
this.total_distance += team.total_distance;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!org.key) { this.registrationEnabled = false; }
|
if (!org.key) { this.registrationEnabled = false; }
|
||||||
|
@@ -2,11 +2,6 @@ import {
|
|||||||
IsInt
|
IsInt
|
||||||
} from "class-validator";
|
} from "class-validator";
|
||||||
import { Donation } from '../entities/Donation';
|
import { Donation } from '../entities/Donation';
|
||||||
import { Runner } from '../entities/Runner';
|
|
||||||
import { RunnerOrganization } from '../entities/RunnerOrganization';
|
|
||||||
import { RunnerTeam } from '../entities/RunnerTeam';
|
|
||||||
import { Scan } from '../entities/Scan';
|
|
||||||
import { User } from '../entities/User';
|
|
||||||
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
import { ResponseObjectType } from '../enums/ResponseObjectType';
|
||||||
import { IResponse } from './IResponse';
|
import { IResponse } from './IResponse';
|
||||||
|
|
||||||
@@ -63,12 +58,30 @@ export class ResponseStats implements IResponse {
|
|||||||
@IsInt()
|
@IsInt()
|
||||||
total_donation: number;
|
total_donation: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The total donation count (cent).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
total_donations: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The total donor count.
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
total_donors: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The average distance ran per runner.
|
* The average distance ran per runner.
|
||||||
*/
|
*/
|
||||||
@IsInt()
|
@IsInt()
|
||||||
average_distance: number;
|
average_distance: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The average donation per distance (cent).
|
||||||
|
*/
|
||||||
|
@IsInt()
|
||||||
|
average_donation: number;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new stats response containing some basic statistics for a dashboard or public display.
|
* Creates a new stats response containing some basic statistics for a dashboard or public display.
|
||||||
* @param runners Array containing all runners - the following relations have to be resolved: scans, scans.track
|
* @param runners Array containing all runners - the following relations have to be resolved: scans, scans.track
|
||||||
@@ -78,14 +91,17 @@ export class ResponseStats implements IResponse {
|
|||||||
* @param scans Array containing all scans - no relations have to be resolved.
|
* @param scans Array containing all scans - no relations have to be resolved.
|
||||||
* @param donations Array containing all donations - the following relations have to be resolved: runner, runner.scans, runner.scans.track
|
* @param donations Array containing all donations - the following relations have to be resolved: runner, runner.scans, runner.scans.track
|
||||||
*/
|
*/
|
||||||
public constructor(runners: Runner[], teams: RunnerTeam[], orgs: RunnerOrganization[], users: User[], scans: Scan[], donations: Donation[]) {
|
public constructor(runners: number, teams: number, orgs: number, users: number, scans: number, donations: Donation[], distance: number, donors: number) {
|
||||||
this.total_runners = runners.length;
|
this.total_runners = runners;
|
||||||
this.total_teams = teams.length;
|
this.total_teams = teams;
|
||||||
this.total_orgs = orgs.length;
|
this.total_orgs = orgs;
|
||||||
this.total_users = users.length;
|
this.total_users = users;
|
||||||
this.total_scans = scans.filter(scan => { scan.valid === true }).length;
|
this.total_scans = scans;
|
||||||
this.total_distance = runners.reduce((sum, current) => sum + current.distance, 0);
|
this.total_distance = distance;
|
||||||
this.total_donation = donations.reduce((sum, current) => sum + current.amount, 0);
|
this.total_donation = donations.reduce((sum, current) => sum + current.amount, 0);
|
||||||
|
this.total_donations = donations.length;
|
||||||
|
this.average_donation = this.total_donation / this.total_donations
|
||||||
|
this.total_donors = donors;
|
||||||
this.average_distance = this.total_distance / this.total_runners;
|
this.average_distance = this.total_distance / this.total_runners;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -49,7 +49,15 @@ export class ResponseStatsOrgnisation implements IResponse {
|
|||||||
public constructor(org: RunnerOrganization) {
|
public constructor(org: RunnerOrganization) {
|
||||||
this.name = org.name;
|
this.name = org.name;
|
||||||
this.id = org.id;
|
this.id = org.id;
|
||||||
this.distance = org.distance;
|
try {
|
||||||
this.donationAmount = org.distanceDonationAmount;
|
this.distance = org.distance;
|
||||||
|
} catch {
|
||||||
|
this.distance = -1;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
this.donationAmount = org.distanceDonationAmount;
|
||||||
|
} catch {
|
||||||
|
this.donationAmount = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -76,10 +76,22 @@ export class ResponseStatsRunner implements IResponse {
|
|||||||
public constructor(runner: Runner, laptime?: number) {
|
public constructor(runner: Runner, laptime?: number) {
|
||||||
this.id = runner.id;
|
this.id = runner.id;
|
||||||
this.firstname = runner.firstname;
|
this.firstname = runner.firstname;
|
||||||
this.middlename = runner.middlename;
|
if (runner.firstname) {
|
||||||
|
this.middlename = runner.middlename;
|
||||||
|
}
|
||||||
this.lastname = runner.lastname;
|
this.lastname = runner.lastname;
|
||||||
this.distance = runner.distance;
|
try {
|
||||||
this.donationAmount = runner.distanceDonationAmount;
|
this.distance = runner.distance;
|
||||||
|
}
|
||||||
|
catch {
|
||||||
|
this.distance = -1;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
this.donationAmount = runner.distanceDonationAmount;
|
||||||
|
}
|
||||||
|
catch {
|
||||||
|
this.donationAmount = -1;
|
||||||
|
}
|
||||||
if (laptime) {
|
if (laptime) {
|
||||||
this.minLaptime = laptime;
|
this.minLaptime = laptime;
|
||||||
}
|
}
|
||||||
|
@@ -57,7 +57,15 @@ export class ResponseStatsTeam implements IResponse {
|
|||||||
this.name = team.name;
|
this.name = team.name;
|
||||||
this.id = team.id;
|
this.id = team.id;
|
||||||
this.parent = team.parentGroup.toResponse();
|
this.parent = team.parentGroup.toResponse();
|
||||||
this.distance = team.distance;
|
try {
|
||||||
this.donationAmount = team.distanceDonationAmount;
|
this.distance = team.distance;
|
||||||
|
} catch {
|
||||||
|
this.distance = -1;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
this.donationAmount = team.distanceDonationAmount;
|
||||||
|
} catch {
|
||||||
|
this.donationAmount = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -11,7 +11,7 @@ import { PermissionAction } from '../models/enums/PermissionAction';
|
|||||||
import { PermissionTarget } from '../models/enums/PermissionTargets';
|
import { PermissionTarget } from '../models/enums/PermissionTargets';
|
||||||
/**
|
/**
|
||||||
* Seeds a admin group with a demo user into the database for initial setup and auto recovery.
|
* 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.
|
* We know that the naming isn't perfectly fitting. Feel free to change it.
|
||||||
*/
|
*/
|
||||||
export default class SeedUsers implements Seeder {
|
export default class SeedUsers implements Seeder {
|
||||||
public async run(factory: Factory, connection: Connection): Promise<any> {
|
public async run(factory: Factory, connection: Connection): Promise<any> {
|
||||||
|
19
src/tests/create_mass_scans.ts
Normal file
19
src/tests/create_mass_scans.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
console.time("batches")
|
||||||
|
for (let i = 0; i < 100; i++) {
|
||||||
|
const batch = [];
|
||||||
|
for (let i = 0; i < 6; i++) {
|
||||||
|
batch.push(axios.post('http://localhost:4010/api/scans/trackscans', { card: 200000000001, station: 2 }, {
|
||||||
|
headers: {
|
||||||
|
Authorization: 'Bearer 10F2E64.BB4F6CC5-2148-4CCF-88B5-0AA85D0508A9'
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
await Promise.all(batch)
|
||||||
|
console.timeLog("batches", `Finished batch ${i}`)
|
||||||
|
}
|
||||||
|
console.timeEnd("batches")
|
||||||
|
}
|
||||||
|
main();
|
@@ -170,7 +170,7 @@ describe('POST /api/donations/fixed successfully', () => {
|
|||||||
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 fixed donation should return 200', async () => {
|
it('creating a new fixed donation with more params should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/donations/fixed', {
|
const res = await axios.post(base + '/api/donations/fixed', {
|
||||||
"donor": added_donor.id,
|
"donor": added_donor.id,
|
||||||
"amount": 1000
|
"amount": 1000
|
||||||
@@ -181,6 +181,25 @@ describe('POST /api/donations/fixed successfully', () => {
|
|||||||
expect(res.data).toEqual({
|
expect(res.data).toEqual({
|
||||||
"donor": added_donor,
|
"donor": added_donor,
|
||||||
"amount": 1000,
|
"amount": 1000,
|
||||||
|
"paidAmount": 0,
|
||||||
|
"status": "OPEN",
|
||||||
|
"responseType": "DONATION"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('creating a new fixed donation with all params should return 200', async () => {
|
||||||
|
const res = await axios.post(base + '/api/donations/fixed', {
|
||||||
|
"donor": added_donor.id,
|
||||||
|
"amount": 1000,
|
||||||
|
"paidAmount": 1000
|
||||||
|
}, axios_config);
|
||||||
|
delete res.data.id;
|
||||||
|
expect(res.status).toEqual(200);
|
||||||
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
|
expect(res.data).toEqual({
|
||||||
|
"donor": added_donor,
|
||||||
|
"amount": 1000,
|
||||||
|
"paidAmount": 1000,
|
||||||
|
"status": "PAID",
|
||||||
"responseType": "DONATION"
|
"responseType": "DONATION"
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -219,7 +238,7 @@ describe('POST /api/donations/distance successfully', () => {
|
|||||||
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 fixed donation should return 200', async () => {
|
it('creating a new distance donation with most params should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/donations/distance', {
|
const res = await axios.post(base + '/api/donations/distance', {
|
||||||
"runner": added_runner.id,
|
"runner": added_runner.id,
|
||||||
"amountPerDistance": 100,
|
"amountPerDistance": 100,
|
||||||
@@ -233,6 +252,28 @@ describe('POST /api/donations/distance successfully', () => {
|
|||||||
"amountPerDistance": 100,
|
"amountPerDistance": 100,
|
||||||
"runner": added_runner,
|
"runner": added_runner,
|
||||||
"amount": 0,
|
"amount": 0,
|
||||||
|
"paidAmount": 0,
|
||||||
|
"status": "PAID",
|
||||||
|
"responseType": "DISTANCEDONATION"
|
||||||
|
})
|
||||||
|
});
|
||||||
|
it('creating a new distance donation with all params should return 200', async () => {
|
||||||
|
const res = await axios.post(base + '/api/donations/distance', {
|
||||||
|
"runner": added_runner.id,
|
||||||
|
"amountPerDistance": 100,
|
||||||
|
"donor": added_donor.id,
|
||||||
|
"paidAmount": 1000
|
||||||
|
}, axios_config);
|
||||||
|
delete res.data.id;
|
||||||
|
expect(res.status).toEqual(200);
|
||||||
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
|
expect(res.data).toEqual({
|
||||||
|
"donor": added_donor,
|
||||||
|
"amountPerDistance": 100,
|
||||||
|
"runner": added_runner,
|
||||||
|
"amount": 0,
|
||||||
|
"paidAmount": 1000,
|
||||||
|
"status": "PAID",
|
||||||
"responseType": "DISTANCEDONATION"
|
"responseType": "DISTANCEDONATION"
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
@@ -213,6 +213,17 @@ describe('adding + updating fixed donation valid', () => {
|
|||||||
expect(res.headers['content-type']).toContain("application/json");
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
expect(res.data.amount).toEqual(42);
|
expect(res.data.amount).toEqual(42);
|
||||||
});
|
});
|
||||||
|
it('updating paidAmount should return 200', async () => {
|
||||||
|
const res = await axios.put(base + '/api/donations/fixed/' + added_donation.id, {
|
||||||
|
"id": added_donation.id,
|
||||||
|
"donor": added_donor.id,
|
||||||
|
"amount": 42,
|
||||||
|
"paidAmount": 10
|
||||||
|
}, axios_config);
|
||||||
|
expect(res.status).toEqual(200);
|
||||||
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
|
expect(res.data.paidAmount).toEqual(10);
|
||||||
|
});
|
||||||
it('updating donor should return 200', async () => {
|
it('updating donor should return 200', async () => {
|
||||||
const res = await axios.put(base + '/api/donations/fixed/' + added_donation.id, {
|
const res = await axios.put(base + '/api/donations/fixed/' + added_donation.id, {
|
||||||
"id": added_donation.id,
|
"id": added_donation.id,
|
||||||
@@ -317,6 +328,19 @@ describe('adding + updating distance donation valid', () => {
|
|||||||
expect(res.headers['content-type']).toContain("application/json");
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
expect(res.data.amountPerDistance).toEqual(69);
|
expect(res.data.amountPerDistance).toEqual(69);
|
||||||
});
|
});
|
||||||
|
it('updating paidAmount should return 200', async () => {
|
||||||
|
const res = await axios.put(base + '/api/donations/distance/' + added_donation.id, {
|
||||||
|
"id": added_donation.id,
|
||||||
|
"runner": added_runner.id,
|
||||||
|
"amountPerDistance": 69,
|
||||||
|
"donor": added_donor.id,
|
||||||
|
"paidAmount": 10
|
||||||
|
}, axios_config);
|
||||||
|
delete res.data.donor.donationAmount;
|
||||||
|
expect(res.status).toEqual(200);
|
||||||
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
|
expect(res.data.paidAmount).toEqual(10);
|
||||||
|
});
|
||||||
it('updating runner should return 200', async () => {
|
it('updating runner should return 200', async () => {
|
||||||
const res = await axios.put(base + '/api/donations/distance/' + added_donation.id, {
|
const res = await axios.put(base + '/api/donations/distance/' + added_donation.id, {
|
||||||
"id": added_donation.id,
|
"id": added_donation.id,
|
||||||
|
@@ -22,6 +22,12 @@ describe('deletion (non-existant)', () => {
|
|||||||
expect(res2.status).toEqual(204);
|
expect(res2.status).toEqual(204);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
describe('deletion of citizen sould fail', () => {
|
||||||
|
it('delete', async () => {
|
||||||
|
const res3 = await axios.delete(base + '/api/organizations/1', axios_config);
|
||||||
|
expect(res3.status).toEqual(400);
|
||||||
|
});
|
||||||
|
});
|
||||||
// ---------------
|
// ---------------
|
||||||
describe('adding + deletion (successfull)', () => {
|
describe('adding + deletion (successfull)', () => {
|
||||||
let added_org_id
|
let added_org_id
|
||||||
|
@@ -66,6 +66,8 @@ describe('adding + getting scans', () => {
|
|||||||
const res = await axios.get(base + '/api/scans/' + added_scan.id, axios_config);
|
const res = await axios.get(base + '/api/scans/' + added_scan.id, 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");
|
||||||
|
delete res.data.runner.distance;
|
||||||
|
delete added_scan.runner.distance;
|
||||||
expect(res.data).toEqual(added_scan);
|
expect(res.data).toEqual(added_scan);
|
||||||
});
|
});
|
||||||
it('check if scans was added via the runner/scans endpoint.', async () => {
|
it('check if scans was added via the runner/scans endpoint.', async () => {
|
||||||
|
@@ -1,5 +1,7 @@
|
|||||||
|
import { faker } from '@faker-js/faker';
|
||||||
import axios from 'axios';
|
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 access_token;
|
||||||
@@ -21,7 +23,7 @@ describe('delete selfservice runner invalid', () => {
|
|||||||
const res = await axios.post(base + '/api/runners/register', {
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
}, axios_config);
|
}, axios_config);
|
||||||
added_runner = res.data;
|
added_runner = res.data;
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
@@ -50,7 +52,7 @@ describe('delete selfservice runner valid', () => {
|
|||||||
const res = await axios.post(base + '/api/runners/register', {
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
}, axios_config);
|
}, axios_config);
|
||||||
added_runner = res.data;
|
added_runner = res.data;
|
||||||
expect(res.status).toEqual(200);
|
expect(res.status).toEqual(200);
|
||||||
|
@@ -15,20 +15,20 @@ beforeAll(async () => {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('POST /api/runners/me/forgot invalid syntax/mail should fail', () => {
|
describe('POST /api/runners/me/login invalid syntax/mail should fail', () => {
|
||||||
it('get without mail return 404', async () => {
|
it('get without mail return 404', async () => {
|
||||||
const res = await axios.post(base + '/api/runners/forgot', null, axios_config);
|
const res = await axios.post(base + '/api/runners/login', null, 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");
|
||||||
});
|
});
|
||||||
it('get without bs mail return 404', async () => {
|
it('get without bs mail return 404', async () => {
|
||||||
const res = await axios.post(base + '/api/runners/forgot?mail=asdasdasdasdasd@tester.test.dev.lauf-fuer-kaya.de', null, axios_config);
|
const res = await axios.post(base + '/api/runners/login?mail=asdasdasdasdasd@tester.test.dev.lauf-fuer-kaya.de', null, 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");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
// ---------------
|
// ---------------
|
||||||
describe('POST /api/runners/me/forgot 2 times within timeout should fail', () => {
|
describe('POST /api/runners/me/login 2 times within timeout should fail', () => {
|
||||||
let added_runner;
|
let added_runner;
|
||||||
it('registering as citizen should return 200', async () => {
|
it('registering as citizen should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/runners/register', {
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
@@ -42,19 +42,19 @@ describe('POST /api/runners/me/forgot 2 times within timeout should fail', () =>
|
|||||||
added_runner = res.data;
|
added_runner = res.data;
|
||||||
});
|
});
|
||||||
it('post with valid mail should return 200', async () => {
|
it('post with valid mail should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/runners/forgot?mail=' + added_runner.email, null, axios_config);
|
const res = await axios.post(base + '/api/runners/login?mail=' + added_runner.email, null, 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('2nd post with valid mail should return 406', async () => {
|
it('2nd post with valid mail should return 406', async () => {
|
||||||
const res = await axios.post(base + '/api/runners/forgot?mail=' + added_runner.email, null, axios_config);
|
const res = await axios.post(base + '/api/runners/login?mail=' + added_runner.email, null, axios_config);
|
||||||
expect(res.status).toEqual(406);
|
expect(res.status).toEqual(406);
|
||||||
expect(res.headers['content-type']).toContain("application/json");
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ---------------
|
// ---------------
|
||||||
describe('POST /api/runners/me/forgot valid should return 200', () => {
|
describe('POST /api/runners/me/login valid should return 200', () => {
|
||||||
let added_runner;
|
let added_runner;
|
||||||
let new_token;
|
let new_token;
|
||||||
it('registering as citizen should return 200', async () => {
|
it('registering as citizen should return 200', async () => {
|
||||||
@@ -69,7 +69,7 @@ describe('POST /api/runners/me/forgot valid should return 200', () => {
|
|||||||
added_runner = res.data;
|
added_runner = res.data;
|
||||||
});
|
});
|
||||||
it('post with valid mail should return 200', async () => {
|
it('post with valid mail should return 200', async () => {
|
||||||
const res = await axios.post(base + '/api/runners/forgot?mail=' + added_runner.email, null, axios_config);
|
const res = await axios.post(base + '/api/runners/login?mail=' + added_runner.email, null, 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");
|
||||||
new_token = res.data.token;
|
new_token = res.data.token;
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
import { faker } from '@faker-js/faker';
|
||||||
import axios from 'axios';
|
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
|
||||||
@@ -30,7 +31,7 @@ describe('register + get should return 200', () => {
|
|||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
}, axios_config);
|
}, 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");
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
import { faker } from '@faker-js/faker';
|
||||||
import axios from 'axios';
|
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
|
||||||
@@ -39,7 +40,7 @@ describe('register invalid citizen', () => {
|
|||||||
const res = await axios.post(base + '/api/runners/register', {
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
}, axios_config);
|
}, 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");
|
||||||
@@ -48,7 +49,7 @@ describe('register invalid citizen', () => {
|
|||||||
const res = await axios.post(base + '/api/runners/register', {
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
}, axios_config);
|
}, 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");
|
||||||
@@ -59,7 +60,26 @@ describe('register invalid citizen', () => {
|
|||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"phone": "peter",
|
"phone": "peter",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
|
}, axios_config);
|
||||||
|
expect(res.status).toEqual(400);
|
||||||
|
expect(res.headers['content-type']).toContain("application/json");
|
||||||
|
});
|
||||||
|
it('registering as citizen with duplicate mail should return 400', async () => {
|
||||||
|
const mail = faker.internet.exampleEmail();
|
||||||
|
await axios.post(base + '/api/runners/register', {
|
||||||
|
"firstname": "string",
|
||||||
|
"middlename": "string",
|
||||||
|
"lastname": "string",
|
||||||
|
"phone": "peter",
|
||||||
|
"email": mail,
|
||||||
|
}, axios_config);
|
||||||
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
|
"firstname": "string",
|
||||||
|
"middlename": "string",
|
||||||
|
"lastname": "string",
|
||||||
|
"phone": "peter",
|
||||||
|
"email": mail,
|
||||||
}, axios_config);
|
}, 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");
|
||||||
@@ -71,7 +91,7 @@ describe('register citizen valid', () => {
|
|||||||
const res = await axios.post(base + '/api/runners/register', {
|
const res = await axios.post(base + '/api/runners/register', {
|
||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com"
|
"email": faker.internet.exampleEmail(),
|
||||||
}, axios_config);
|
}, 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");
|
||||||
@@ -81,7 +101,7 @@ describe('register citizen valid', () => {
|
|||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com",
|
"email": faker.internet.exampleEmail(),
|
||||||
"phone": "+4909132123456",
|
"phone": "+4909132123456",
|
||||||
"address": {
|
"address": {
|
||||||
address1: "Teststreet 1",
|
address1: "Teststreet 1",
|
||||||
@@ -187,7 +207,7 @@ describe('register valid company', () => {
|
|||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com",
|
"email": faker.internet.exampleEmail(),
|
||||||
"phone": "+4909132123456",
|
"phone": "+4909132123456",
|
||||||
"address": {
|
"address": {
|
||||||
address1: "Teststreet 1",
|
address1: "Teststreet 1",
|
||||||
@@ -214,7 +234,7 @@ describe('register valid company', () => {
|
|||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com",
|
"email": faker.internet.exampleEmail(),
|
||||||
"phone": "+4909132123456",
|
"phone": "+4909132123456",
|
||||||
"address": {
|
"address": {
|
||||||
address1: "Teststreet 1",
|
address1: "Teststreet 1",
|
||||||
@@ -232,7 +252,7 @@ describe('register valid company', () => {
|
|||||||
"firstname": "string",
|
"firstname": "string",
|
||||||
"middlename": "string",
|
"middlename": "string",
|
||||||
"lastname": "string",
|
"lastname": "string",
|
||||||
"email": "user@example.com",
|
"email": faker.internet.exampleEmail(),
|
||||||
"phone": "+4909132123456",
|
"phone": "+4909132123456",
|
||||||
"address": {
|
"address": {
|
||||||
address1: "Teststreet 1",
|
address1: "Teststreet 1",
|
||||||
|
@@ -52,7 +52,6 @@ describe('GET /api/stats/runners/* should return 200', () => {
|
|||||||
});
|
});
|
||||||
it('get by donations w/ auth should return 200', async () => {
|
it('get by donations w/ auth should return 200', async () => {
|
||||||
const res = await axios.get(base + '/api/stats/runners/donations', axios_config_stats);
|
const res = await axios.get(base + '/api/stats/runners/donations', axios_config_stats);
|
||||||
console.log(res.data)
|
|
||||||
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");
|
||||||
});
|
});
|
||||||
|
Reference in New Issue
Block a user