UNPKG

unleash-server

Version:

Unleash is an enterprise ready feature flag service. It provides different strategies for handling feature flags.

220 lines • 8.21 kB
import metricsHelper from '../util/metrics-helper.js'; import { DB_TIME } from '../metric-events.js'; import NotFoundError from '../error/notfound-error.js'; import { ApiTokenType, } from '../types/index.js'; import { ALL_PROJECTS } from '../internals.js'; import { isAllProjects } from '../server-impl.js'; import { inTransaction } from './transaction.js'; const TABLE = 'api_tokens'; const API_LINK_TABLE = 'api_token_project'; const ALL = '*'; const tokenRowReducer = (acc, tokenRow) => { const { project, ...token } = tokenRow; if (!acc[tokenRow.secret]) { acc[tokenRow.secret] = { secret: token.secret, tokenName: token.token_name ? token.token_name : token.username, type: token.type.toLowerCase(), project: ALL, projects: [ALL], environment: token.environment ? token.environment : ALL, expiresAt: token.expires_at, createdAt: token.created_at, alias: token.alias, seenAt: token.seen_at, }; } const currentToken = acc[tokenRow.secret]; if (tokenRow.project) { if (isAllProjects(currentToken.projects)) { currentToken.projects = []; } currentToken.projects.push(tokenRow.project); currentToken.project = currentToken.projects.join(','); } return acc; }; const toRow = (newToken) => ({ username: newToken.tokenName, token_name: newToken.tokenName, secret: newToken.secret, type: newToken.type, environment: newToken.environment === ALL ? undefined : newToken.environment, expires_at: newToken.expiresAt, alias: newToken.alias || null, }); const toTokens = (rows) => { const tokens = rows.reduce(tokenRowReducer, {}); return Object.values(tokens); }; export class ApiTokenStore { constructor(db, eventBus, getLogger, flagResolver) { this.db = db; this.logger = getLogger('api-tokens.js'); this.timer = (action) => metricsHelper.wrapTimer(eventBus, DB_TIME, { store: 'api-tokens', action, }); this.flagResolver = flagResolver; } // helper function that we can move to utils async withTimer(timerName, fn) { const stopTimer = this.timer(timerName); try { return await fn(); } finally { stopTimer(); } } async count() { return this.db(TABLE) .count('*') .then((res) => Number(res[0].count)); } async countByType() { return this.db(TABLE) .select('type') .count('*') .groupBy('type') .then((res) => { const map = new Map(); res.forEach((row) => { map.set(row.type.toString(), Number(row.count)); }); return map; }); } async getAll() { const stopTimer = this.timer('getAll'); const rows = await this.makeTokenProjectQuery(); stopTimer(); return toTokens(rows); } async getAllActive() { const stopTimer = this.timer('getAllActive'); const rows = await this.makeTokenProjectQuery() .where('expires_at', 'IS', null) .orWhere('expires_at', '>', 'now()'); stopTimer(); return toTokens(rows); } makeTokenProjectQuery() { return this.db(`${TABLE} as tokens`) .leftJoin(`${API_LINK_TABLE} as token_project_link`, 'tokens.secret', 'token_project_link.secret') .select('tokens.secret', 'username', 'token_name', 'type', 'expires_at', 'created_at', 'alias', 'seen_at', 'environment', 'token_project_link.project'); } async insert(newToken) { const response = await inTransaction(this.db, async (tx) => { const [row] = await tx(TABLE).insert(toRow(newToken), ['created_at']); const updateProjectTasks = (newToken.projects || []) .filter((project) => { return project !== ALL_PROJECTS; }) .map((project) => { return tx.raw(`INSERT INTO ${API_LINK_TABLE} VALUES (?, ?)`, [newToken.secret, project]); }); await Promise.all(updateProjectTasks); return { ...newToken, alias: newToken.alias || null, project: newToken.projects?.join(',') || '*', createdAt: row.created_at, }; }); return response; } destroy() { } async exists(secret) { const result = await this.db.raw(`SELECT EXISTS (SELECT 1 FROM ${TABLE} WHERE secret = ?) AS present`, [secret]); const { present } = result.rows[0]; return present; } async get(key) { const stopTimer = this.timer('get-by-secret'); const row = await this.makeTokenProjectQuery().where('tokens.secret', key); stopTimer(); return toTokens(row)[0]; } async delete(secret) { return this.db(TABLE).where({ secret }).del(); } async deleteAll() { return this.db(TABLE).del(); } async setExpiry(secret, expiresAt) { const rows = await this.makeTokenProjectQuery() .update({ expires_at: expiresAt }) .where({ secret }) .returning('*'); if (rows.length > 0) { return toTokens(rows)[0]; } throw new NotFoundError('Could not find api-token.'); } async markSeenAt(secrets) { const now = new Date(); try { await this.db(TABLE) .whereIn('secret', secrets) .update({ seen_at: now }); } catch (err) { this.logger.error('Could not update lastSeen, error: ', err); } } async countDeprecatedTokens() { const allLegacyCount = this.withTimer('allLegacyCount', () => this.db(`${TABLE} as tokens`) .where('tokens.secret', 'NOT LIKE', '%:%') .count() .first() .then((res) => Number(res?.count) || 0)); const activeLegacyCount = this.withTimer('activeLegacyCount', () => this.db(`${TABLE} as tokens`) .where('tokens.secret', 'NOT LIKE', '%:%') .andWhereRaw("tokens.seen_at > NOW() - INTERVAL '3 MONTH'") .count() .first() .then((res) => Number(res?.count) || 0)); const orphanedTokensQuery = this.db(`${TABLE} as tokens`) .leftJoin(`${API_LINK_TABLE} as token_project_link`, 'tokens.secret', 'token_project_link.secret') .whereNull('token_project_link.project') .andWhere('tokens.secret', 'NOT LIKE', '*:%') // Exclude intentionally wildcard tokens .andWhere('tokens.secret', 'LIKE', '%:%') // Exclude legacy tokens .andWhere((builder) => { builder .where('tokens.type', ApiTokenType.CLIENT) .orWhere('tokens.type', ApiTokenType.FRONTEND); }); const allOrphanedCount = this.withTimer('allOrphanedCount', () => orphanedTokensQuery .clone() .count() .first() .then((res) => Number(res?.count) || 0)); const activeOrphanedCount = this.withTimer('activeOrphanedCount', () => orphanedTokensQuery .clone() .andWhereRaw("tokens.seen_at > NOW() - INTERVAL '3 MONTH'") .count() .first() .then((res) => Number(res?.count) || 0)); const [orphanedTokens, activeOrphanedTokens, legacyTokens, activeLegacyTokens,] = await Promise.all([ allOrphanedCount, activeOrphanedCount, allLegacyCount, activeLegacyCount, ]); return { orphanedTokens, activeOrphanedTokens, legacyTokens, activeLegacyTokens, }; } async countProjectTokens(projectId) { const count = await this.db(API_LINK_TABLE) .where({ project: projectId }) .count() .first(); return Number(count?.count ?? 0); } } //# sourceMappingURL=api-token-store.js.map