Compare commits
	
		
			13 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 284cb0f8b3 | |||
| 6e63c57936 | |||
| 30b61db2c1 | |||
| 8237d5f210 | |||
| 03e0a29096 | |||
| a6afba93e2 | |||
| a41758cd9c | |||
| d6755ed134 | |||
| 599c75fc00 | |||
| bb213f001e | |||
| 5415cd38a7 | |||
| 175ba52ffa | |||
| 5c5000a218 | 
							
								
								
									
										45
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										45
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,8 +2,51 @@ | |||||||
|  |  | ||||||
| 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.3.3](https://git.odit.services/lfk/backend/compare/v1.3.2...v1.3.3) | #### [1.3.9](https://git.odit.services/lfk/backend/compare/1.3.8...1.3.9) | ||||||
|  |  | ||||||
|  | - feat(RunnerController.getAll): add created_via query param filter [`6e63c57`](https://git.odit.services/lfk/backend/commit/6e63c57936f06a29da5f1a94b1141d51b75df5f0) | ||||||
|  |  | ||||||
|  | #### [1.3.8](https://git.odit.services/lfk/backend/compare/1.3.7...1.3.8) | ||||||
|  |  | ||||||
|  | > 9 April 2025 | ||||||
|  |  | ||||||
|  | - feat(RunnerCardController): putByCode [`8237d5f`](https://git.odit.services/lfk/backend/commit/8237d5f21067c0872a7eff7c8d1506edf44ec10c) | ||||||
|  | - chore(release): 1.3.8 [`30b61db`](https://git.odit.services/lfk/backend/commit/30b61db2c160c019bac381f26cefdc6524ea465e) | ||||||
|  |  | ||||||
|  | #### [1.3.7](https://git.odit.services/lfk/backend/compare/1.3.6...1.3.7) | ||||||
|  |  | ||||||
|  | > 8 April 2025 | ||||||
|  |  | ||||||
|  | - feat(stats): Publish runners by kiosk stat [`a6afba9`](https://git.odit.services/lfk/backend/commit/a6afba93e243ca419c282a16cad023d06d864e0e) | ||||||
|  | - chore(release): 1.3.7 [`03e0a29`](https://git.odit.services/lfk/backend/commit/03e0a290965648579956ac1f8e8542c97a667ed8) | ||||||
|  |  | ||||||
|  | #### [1.3.6](https://git.odit.services/lfk/backend/compare/1.3.5...1.3.6) | ||||||
|  |  | ||||||
|  | > 8 April 2025 | ||||||
|  |  | ||||||
|  | - chore(release): 1.3.6 [`a41758c`](https://git.odit.services/lfk/backend/commit/a41758cd9c83105c3a4b407744bafe2f0f6fb48a) | ||||||
|  | - feat(runners): Allow created via being set via api [`d6755ed`](https://git.odit.services/lfk/backend/commit/d6755ed134071df635bc9d5821ceb2396c0f1d22) | ||||||
|  | - fix(participant): Switch to correct type [`599c75f`](https://git.odit.services/lfk/backend/commit/599c75fc00217eaec3cc87c0de50d059bdde685f) | ||||||
|  |  | ||||||
|  | #### [1.3.5](https://git.odit.services/lfk/backend/compare/1.3.4...1.3.5) | ||||||
|  |  | ||||||
|  | > 8 April 2025 | ||||||
|  |  | ||||||
|  | - feat(runners): Generate selfservice urls on runner if requested or create/update/get single [`5415cd3`](https://git.odit.services/lfk/backend/commit/5415cd38a727e76632a01a4d2634a1777df5542c) | ||||||
|  | - chore(release): 1.3.5 [`bb213f0`](https://git.odit.services/lfk/backend/commit/bb213f001eff2157abf8741128f624f9cc991afe) | ||||||
|  |  | ||||||
|  | #### [1.3.4](https://git.odit.services/lfk/backend/compare/1.3.3...1.3.4) | ||||||
|  |  | ||||||
|  | > 28 March 2025 | ||||||
|  |  | ||||||
|  | - feat: add runnersViaSelfservice to statsControllerGet [`5c5000a`](https://git.odit.services/lfk/backend/commit/5c5000a218b47815e6846ac8b857dcd1995bfa6f) | ||||||
|  | - chore(release): 1.3.4 [`175ba52`](https://git.odit.services/lfk/backend/commit/175ba52ffae8e6ba1fdc1603ac2f5eba15602046) | ||||||
|  |  | ||||||
|  | #### [1.3.3](https://git.odit.services/lfk/backend/compare/v1.3.2...1.3.3) | ||||||
|  |  | ||||||
|  | > 28 March 2025 | ||||||
|  |  | ||||||
|  | - chore(release): 1.3.3 [`d559d04`](https://git.odit.services/lfk/backend/commit/d559d0403191c703fd6da0e3f3dab53eec9258c0) | ||||||
| - ci: remove "v" prefix from tags [`2af682d`](https://git.odit.services/lfk/backend/commit/2af682d1dd09df496eb9f3a9111c50c0c4117356) | - ci: remove "v" prefix from tags [`2af682d`](https://git.odit.services/lfk/backend/commit/2af682d1dd09df496eb9f3a9111c50c0c4117356) | ||||||
|  |  | ||||||
| #### [v1.3.2](https://git.odit.services/lfk/backend/compare/v1.3.1...v1.3.2) | #### [v1.3.2](https://git.odit.services/lfk/backend/compare/v1.3.1...v1.3.2) | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@odit/lfk-backend", |   "name": "@odit/lfk-backend", | ||||||
|   "version": "1.3.3", |   "version": "1.3.9", | ||||||
|   "main": "src/app.ts", |   "main": "src/app.ts", | ||||||
|   "repository": "https://git.odit.services/lfk/backend", |   "repository": "https://git.odit.services/lfk/backend", | ||||||
|   "author": { |   "author": { | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import { RunnerCardHasScansError, RunnerCardIdsNotMatchingError, RunnerCardNotFo | |||||||
| import { RunnerNotFoundError } from '../errors/RunnerErrors'; | import { RunnerNotFoundError } from '../errors/RunnerErrors'; | ||||||
| import { CreateRunnerCard } from '../models/actions/create/CreateRunnerCard'; | import { CreateRunnerCard } from '../models/actions/create/CreateRunnerCard'; | ||||||
| import { UpdateRunnerCard } from '../models/actions/update/UpdateRunnerCard'; | import { UpdateRunnerCard } from '../models/actions/update/UpdateRunnerCard'; | ||||||
|  | import { UpdateRunnerCardByCode } from '../models/actions/update/UpdateRunnerCardByCode'; | ||||||
| import { RunnerCard } from '../models/entities/RunnerCard'; | import { RunnerCard } from '../models/entities/RunnerCard'; | ||||||
| import { ResponseEmpty } from '../models/responses/ResponseEmpty'; | import { ResponseEmpty } from '../models/responses/ResponseEmpty'; | ||||||
| import { ResponseRunnerCard } from '../models/responses/ResponseRunnerCard'; | import { ResponseRunnerCard } from '../models/responses/ResponseRunnerCard'; | ||||||
| @@ -112,6 +113,28 @@ export class RunnerCardController { | |||||||
| 		return (await this.cardRepository.findOne({ id: id }, { relations: ['runner', 'runner.group', 'runner.group.parentGroup'] })).toResponse(); | 		return (await this.cardRepository.findOne({ id: id }, { relations: ['runner', 'runner.group', 'runner.group.parentGroup'] })).toResponse(); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@Put('/:code') | ||||||
|  | 	@Authorized("CARD:UPDATE") | ||||||
|  | 	@ResponseSchema(ResponseRunnerCard) | ||||||
|  | 	@ResponseSchema(RunnerCardNotFoundError, { statusCode: 404 }) | ||||||
|  | 	@ResponseSchema(RunnerNotFoundError, { statusCode: 404 }) | ||||||
|  | 	@ResponseSchema(RunnerCardIdsNotMatchingError, { statusCode: 406 }) | ||||||
|  | 	@OpenAPI({ description: "Update the card whose code you provided." }) | ||||||
|  | 	async putByCode(@Param('code') code: string, @Body({ validate: true }) card: UpdateRunnerCardByCode) { | ||||||
|  | 		let oldCard = await this.cardRepository.findOne({ code: code }); | ||||||
|  |  | ||||||
|  | 		if (!oldCard) { | ||||||
|  | 			throw new RunnerCardNotFoundError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (oldCard.code != card.code) { | ||||||
|  | 			throw new RunnerCardIdsNotMatchingError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await this.cardRepository.save(await card.update(oldCard)); | ||||||
|  | 		return (await this.cardRepository.findOne({ code: code }, { relations: ['runner', 'runner.group', 'runner.group.parentGroup'] })).toResponse(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	@Delete('/:id') | 	@Delete('/:id') | ||||||
| 	@Authorized("CARD:DELETE") | 	@Authorized("CARD:DELETE") | ||||||
| 	@ResponseSchema(ResponseRunnerCard) | 	@ResponseSchema(ResponseRunnerCard) | ||||||
|   | |||||||
| @@ -30,7 +30,7 @@ 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(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100) { | 	async getAll(@QueryParam("page", { required: false }) page: number, @QueryParam("page_size", { required: false }) page_size: number = 100, @QueryParam("created_via", { required: false }) created_via: string = "all", @QueryParam("selfservice_links", { required: false }) selfservice_links: boolean = false) { | ||||||
| 		let responseRunners: ResponseRunner[] = new Array<ResponseRunner>(); | 		let responseRunners: ResponseRunner[] = new Array<ResponseRunner>(); | ||||||
| 		let runners: Array<Runner>; | 		let runners: Array<Runner>; | ||||||
|  |  | ||||||
| @@ -41,7 +41,13 @@ export class RunnerController { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		runners.forEach(runner => { | 		runners.forEach(runner => { | ||||||
| 			responseRunners.push(new ResponseRunner(runner)); | 			if (created_via === "all") { | ||||||
|  | 				responseRunners.push(new ResponseRunner(runner, selfservice_links)); | ||||||
|  | 			} else { | ||||||
|  | 				if (runner.created_via === created_via) { | ||||||
|  | 					responseRunners.push(new ResponseRunner(runner, selfservice_links)); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		}); | 		}); | ||||||
| 		return responseRunners; | 		return responseRunners; | ||||||
| 	} | 	} | ||||||
| @@ -55,7 +61,7 @@ export class RunnerController { | |||||||
| 	async getOne(@Param('id') id: number) { | 	async getOne(@Param('id') id: number) { | ||||||
| 		let runner = await this.runnerRepository.findOne({ id: id }, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] }) | 		let runner = await this.runnerRepository.findOne({ id: id }, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] }) | ||||||
| 		if (!runner) { throw new RunnerNotFoundError(); } | 		if (!runner) { throw new RunnerNotFoundError(); } | ||||||
| 		return new ResponseRunner(runner); | 		return new ResponseRunner(runner, true); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@Get('/:id/scans') | 	@Get('/:id/scans') | ||||||
| @@ -98,7 +104,7 @@ export class RunnerController { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		runner = await this.runnerRepository.save(runner) | 		runner = await this.runnerRepository.save(runner) | ||||||
| 		return new ResponseRunner(await this.runnerRepository.findOne(runner, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] })); | 		return new ResponseRunner(await this.runnerRepository.findOne(runner, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] }), true); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@Put('/:id') | 	@Put('/:id') | ||||||
| @@ -119,7 +125,7 @@ export class RunnerController { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		await this.runnerRepository.save(await runner.update(oldRunner)); | 		await this.runnerRepository.save(await runner.update(oldRunner)); | ||||||
| 		return new ResponseRunner(await this.runnerRepository.findOne({ id: id }, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] })); | 		return new ResponseRunner(await this.runnerRepository.findOne({ id: id }, { relations: ['scans', 'group', 'group.parentGroup', 'scans.track', 'cards'] }), true); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@Delete('/:id') | 	@Delete('/:id') | ||||||
|   | |||||||
| @@ -23,6 +23,8 @@ export class StatsController { | |||||||
|     @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() { | ||||||
|         const connection = getConnection(); |         const connection = getConnection(); | ||||||
|  |         const runnersViaSelfservice = await connection.getRepository(Runner).count({ where: { created_via: "selfservice" } }); | ||||||
|  |         const runnersViaKiosk = await connection.getRepository(Runner).count({ where: { created_via: "kiosk" } }); | ||||||
|         const runners = await connection.getRepository(Runner).count(); |         const runners = await connection.getRepository(Runner).count(); | ||||||
|         const teams = await connection.getRepository(RunnerTeam).count(); |         const teams = await connection.getRepository(RunnerTeam).count(); | ||||||
|         const orgs = await connection.getRepository(RunnerOrganization).count(); |         const orgs = await connection.getRepository(RunnerOrganization).count(); | ||||||
| @@ -41,7 +43,7 @@ export class StatsController { | |||||||
|         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'] }); | ||||||
|         const donors = await connection.getRepository(Donor).count(); |         const donors = await connection.getRepository(Donor).count(); | ||||||
|  |  | ||||||
|         return new ResponseStats(runners, teams, orgs, users, scans, donations, distace, donors) |         return new ResponseStats(runnersViaSelfservice, runners, teams, orgs, users, scans, donations, distace, donors, runnersViaKiosk) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Get("/runners/distance") |     @Get("/runners/distance") | ||||||
|   | |||||||
| @@ -50,4 +50,11 @@ export abstract class CreateParticipant { | |||||||
|     @IsOptional() |     @IsOptional() | ||||||
|     @IsObject() |     @IsObject() | ||||||
|     address?: Address; |     address?: Address; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * how the participant got into the system | ||||||
|  |      */ | ||||||
|  |     @IsOptional() | ||||||
|  |     @IsString() | ||||||
|  |     created_via?: string; | ||||||
| } | } | ||||||
| @@ -32,6 +32,9 @@ export class CreateRunner extends CreateParticipant { | |||||||
|         newRunner.email = this.email; |         newRunner.email = this.email; | ||||||
|         newRunner.group = await this.getGroup(); |         newRunner.group = await this.getGroup(); | ||||||
|         newRunner.address = this.address; |         newRunner.address = this.address; | ||||||
|  |         if (this.created_via) { | ||||||
|  |             newRunner.created_via = this.created_via; | ||||||
|  |         } | ||||||
|         Address.validate(newRunner.address); |         Address.validate(newRunner.address); | ||||||
|  |  | ||||||
|         return newRunner; |         return newRunner; | ||||||
|   | |||||||
							
								
								
									
										50
									
								
								src/models/actions/update/UpdateRunnerCardByCode.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/models/actions/update/UpdateRunnerCardByCode.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | import { IsBoolean, IsInt, IsNotEmpty, IsOptional, IsString } from 'class-validator'; | ||||||
|  | import { getConnection } from 'typeorm'; | ||||||
|  | import { RunnerNotFoundError } from '../../../errors/RunnerErrors'; | ||||||
|  | import { Runner } from '../../entities/Runner'; | ||||||
|  | import { RunnerCard } from '../../entities/RunnerCard'; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * This class is used to update a RunnerCard entity (via put request). | ||||||
|  |  */ | ||||||
|  | export class UpdateRunnerCardByCode { | ||||||
|  |     /** | ||||||
|  |      * The card's code. | ||||||
|  |      */ | ||||||
|  |     @IsString() | ||||||
|  |     @IsNotEmpty() | ||||||
|  |     code?: string; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * The runner's id. | ||||||
|  |      */ | ||||||
|  |     @IsInt() | ||||||
|  |     @IsOptional() | ||||||
|  |     runner?: number; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Is the updated card enabled (for fraud reasons)? | ||||||
|  |      * Default: true | ||||||
|  |      */ | ||||||
|  |     @IsBoolean() | ||||||
|  |     enabled: boolean = true; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Creates a new RunnerCard entity from this. | ||||||
|  |      */ | ||||||
|  |     public async update(card: RunnerCard): Promise<RunnerCard> { | ||||||
|  |         card.enabled = this.enabled; | ||||||
|  |         card.runner = await this.getRunner(); | ||||||
|  |  | ||||||
|  |         return card; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public async getRunner(): Promise<Runner> { | ||||||
|  |         if (!this.runner) { return null; } | ||||||
|  |         const runner = await getConnection().getRepository(Runner).findOne({ id: this.runner }); | ||||||
|  |         if (!runner) { | ||||||
|  |             throw new RunnerNotFoundError(); | ||||||
|  |         } | ||||||
|  |         return runner; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -80,7 +80,7 @@ export abstract class Participant { | |||||||
|    */ |    */ | ||||||
|   @Column({ nullable: true, default: "backend" }) |   @Column({ nullable: true, default: "backend" }) | ||||||
|   @IsOptional() |   @IsOptional() | ||||||
|   @IsEmail() |   @IsString() | ||||||
|   created_via?: string; |   created_via?: string; | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|   | |||||||
| @@ -1,7 +1,10 @@ | |||||||
| import { | import { | ||||||
|     IsInt, |     IsInt, | ||||||
|     IsObject |     IsObject, | ||||||
|  |     IsOptional, | ||||||
|  |     IsString | ||||||
| } from "class-validator"; | } from "class-validator"; | ||||||
|  | import { JwtCreator } from '../../jwtcreator'; | ||||||
| import { Runner } from '../entities/Runner'; | import { Runner } from '../entities/Runner'; | ||||||
| import { ResponseObjectType } from '../enums/ResponseObjectType'; | import { ResponseObjectType } from '../enums/ResponseObjectType'; | ||||||
| import { IResponse } from './IResponse'; | import { IResponse } from './IResponse'; | ||||||
| @@ -30,14 +33,26 @@ export class ResponseRunner extends ResponseParticipant implements IResponse { | |||||||
|     @IsObject() |     @IsObject() | ||||||
|     group: ResponseRunnerGroup; |     group: ResponseRunnerGroup; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * A selfservice link for our new runner. | ||||||
|  |      */ | ||||||
|  |     @IsOptional() | ||||||
|  |     @IsString() | ||||||
|  |     selfserviceLink: string; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 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. | ||||||
|      */ |      */ | ||||||
|     public constructor(runner: Runner) { |     public constructor(runner: Runner, generateSelfServiceLink: boolean = false) { | ||||||
|         super(runner); |         super(runner); | ||||||
|         if (!runner.scans) { this.distance = 0 } |         if (!runner.scans) { this.distance = 0 } | ||||||
|         else { this.distance = runner.validScans.reduce((sum, current) => sum + current.distance, 0); } |         else { this.distance = runner.validScans.reduce((sum, current) => sum + current.distance, 0); } | ||||||
|         if (runner.group) { this.group = runner.group.toResponse(); } |         if (runner.group) { this.group = runner.group.toResponse(); } | ||||||
|  |  | ||||||
|  |         if (generateSelfServiceLink) { | ||||||
|  |             const token = JwtCreator.createSelfService(runner); | ||||||
|  |             this.selfserviceLink = `${process.env.SELFSERVICE_URL}/profile/${token}`; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,6 +16,18 @@ export class ResponseStats implements IResponse { | |||||||
|     */ |     */ | ||||||
|     responseType: ResponseObjectType = ResponseObjectType.STATS; |     responseType: ResponseObjectType = ResponseObjectType.STATS; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * The amount of runners registered via selfservice. | ||||||
|  |      */ | ||||||
|  |     @IsInt() | ||||||
|  |     runnersViaSelfservice: number; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * The amount of runners registered via kiosk. | ||||||
|  |      */ | ||||||
|  |     @IsInt() | ||||||
|  |     runnersViaKiosk: number; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * The amount of runners registered in the system. |      * The amount of runners registered in the system. | ||||||
|      */ |      */ | ||||||
| @@ -84,14 +96,16 @@ export class ResponseStats implements IResponse { | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * 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 runnersViaSelfservice number of runners registered via selfservice | ||||||
|      * @param teams Array containing all teams - no relations have to be resolved. |      * @param runners number of runners | ||||||
|      * @param orgs Array containing all orgs - no relations have to be resolved. |      * @param teams number of teams - no relations have to be resolved. | ||||||
|      * @param users Array containing all users - no relations have to be resolved. |      * @param orgs number of orgs - no relations have to be resolved. | ||||||
|      * @param scans Array containing all scans - no relations have to be resolved. |      * @param users number of users - no relations have to be resolved. | ||||||
|  |      * @param scans number of 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: number, teams: number, orgs: number, users: number, scans: number, donations: Donation[], distance: number, donors: number) { |     public constructor(runnersViaSelfservice: number, runners: number, teams: number, orgs: number, users: number, scans: number, donations: Donation[], distance: number, donors: number, runnersViaKiosk: number) { | ||||||
|  |         this.runnersViaSelfservice = runnersViaSelfservice; | ||||||
|         this.total_runners = runners; |         this.total_runners = runners; | ||||||
|         this.total_teams = teams; |         this.total_teams = teams; | ||||||
|         this.total_orgs = orgs; |         this.total_orgs = orgs; | ||||||
| @@ -103,5 +117,6 @@ export class ResponseStats implements IResponse { | |||||||
|         this.average_donation = this.total_donation / this.total_donations |         this.average_donation = this.total_donation / this.total_donations | ||||||
|         this.total_donors = donors; |         this.total_donors = donors; | ||||||
|         this.average_distance = this.total_distance / this.total_runners; |         this.average_distance = this.total_distance / this.total_runners; | ||||||
|  |         this.runnersViaKiosk = runnersViaKiosk; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user