Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
147 changes: 147 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@
"ajv-formats": "^2.0.2",
"commander": "^7.2.0",
"dotenv": "^16.3.1",
"ioredis": "^5.8.1",
"jsonwebtoken": "^9.0.0",
"koa": "^2.11.0",
"koa-body": "^4.2.0",
Expand Down
1 change: 1 addition & 0 deletions src/main/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ export * from './openapi.js';
export * from './router.js';
export * from './schema.js';
export * from './services/index.js';
export * from './trial.js';
export * from './util.js';

export * from '@nodescript/logger';
Expand Down
89 changes: 89 additions & 0 deletions src/main/trial.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
import { Logger } from '@nodescript/logger';
import { Redis } from 'ioredis';
import { config } from 'mesh-config';
import { dep } from 'mesh-ioc';

import { AccessForbidden } from './ac-auth.js';

export interface TokenServiceRestriction {
serviceName: string;
requestLimit: number;
}

export interface TrialToken {
serviceRestrictions: Array<TokenServiceRestriction>;
[key: string]: any;
}

export class TrialClient {

@config() private REDIS_URL!: string;
@dep() private logger!: Logger;

private isRunning = false;
private trialKeyPrefix = 'cache:trialClient';

redisClient: Redis;

constructor() {
this.redisClient = this.createRedisClient();
}

async start() {
if (this.isRunning) {
return;
}
this.isRunning = true;
await this.redisClient.connect();
this.logger.info('TrialClient Redis connected');
}

async stop() {
try {
this.redisClient.disconnect();
this.logger.info('TrialClient Redis disconnected');
} finally {
this.isRunning = false;
}
}

isTrialToken(token: Record<string, any>): token is TrialToken {
return !!token.serviceRestrictions;
}

async requireValidServiceRestriction(token: TrialToken, serviceName: string) {
const serviceRestriction = token.serviceRestrictions.find((s: TokenServiceRestriction) => s.serviceName === serviceName);
if (!serviceRestriction) {
throw new AccessForbidden('Service access not configured on token');
}
const requestCount = await this.getRequestCount(token.clientId, serviceName);
if (requestCount >= serviceRestriction.requestLimit) {
throw new AccessForbidden('Trial token has exceeded request limit for service');
}
}

async incrementRequests(token: Record<string, any>, serviceName: string) {
const redisKey = this.getServiceKey(token.clientId, serviceName);
await this.redisClient.hincrby(redisKey, 'requestCount', 1);
}

async getRequestCount(clientId: string, serviceName: string) {
const redisKey = this.getServiceKey(clientId, serviceName);
const requestCountStr = await this.redisClient.hget(redisKey, 'requestCount');
if (requestCountStr == null) {
throw new AccessForbidden('Service access for token not configured');
}
return Number(requestCountStr);
}

private createRedisClient() {
return new Redis(this.REDIS_URL, {
lazyConnect: true,
disconnectTimeout: 10,
});
}

private getServiceKey(clientId: string, serviceName: string) {
return `${this.trialKeyPrefix}:${clientId}:${serviceName}`;
}
}