1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-11-10 01:19:53 +01:00
unleash.unleash/src/lib/features/traffic-data-usage/traffic-data-usage-store.ts
Gastón Fournier abe160eb7d
feat: Unleash v7 ESM migration (#9877)
We're migrating to ESM, which will allow us to import the latest
versions of our dependencies.

Co-Authored-By: Christopher Kolstad <chriswk@getunleash.io>
2025-05-14 09:47:12 +02:00

146 lines
4.6 KiB
TypeScript

import { endOfDay, endOfMonth, startOfDay, startOfMonth } from 'date-fns';
import type { Db } from '../../db/db.js';
import type { Logger, LogProvider } from '../../logger.js';
import type {
IStatMonthlyTrafficUsage,
IStatTrafficUsage,
IStatTrafficUsageKey,
ITrafficDataUsageStore,
} from './traffic-data-usage-store-type.js';
const TABLE = 'stat_traffic_usage';
const COLUMNS = ['day', 'traffic_group', 'status_code_series', 'count'];
const toRow = (trafficDataUsage: IStatTrafficUsage) => {
return {
day: trafficDataUsage.day,
traffic_group: trafficDataUsage.trafficGroup,
status_code_series: trafficDataUsage.statusCodeSeries,
count: trafficDataUsage.count,
};
};
const mapRow = (row: any): IStatTrafficUsage => {
return {
day: row.day,
trafficGroup: row.traffic_group,
statusCodeSeries: row.status_code_series,
count: Number.parseInt(row.count),
};
};
export class TrafficDataUsageStore implements ITrafficDataUsageStore {
private db: Db;
private logger: Logger;
constructor(db: Db, getLogger: LogProvider) {
this.db = db;
this.logger = getLogger('traffic-data-usage-store.ts');
}
async get(key: IStatTrafficUsageKey): Promise<IStatTrafficUsage> {
const row = await this.db
.table(TABLE)
.select()
.where({
day: key.day,
traffic_group: key.trafficGroup,
status_code_series: key.statusCodeSeries,
})
.first();
return mapRow(row);
}
async getAll(query = {}): Promise<IStatTrafficUsage[]> {
const rows = await this.db.select(COLUMNS).where(query).from(TABLE);
return rows.map(mapRow);
}
async exists(key: IStatTrafficUsageKey): Promise<boolean> {
const result = await this.db.raw(
`SELECT EXISTS (SELECT 1 FROM ${TABLE} WHERE
day = ? AND
traffic_group = ? AND
status_code_series ?) AS present`,
[key.day, key.trafficGroup, key.statusCodeSeries],
);
const { present } = result.rows[0];
return present;
}
async delete(key: IStatTrafficUsageKey): Promise<void> {
await this.db(TABLE)
.where({
day: key.day,
traffic_group: key.trafficGroup,
status_code_series: key.statusCodeSeries,
})
.del();
}
async deleteAll(): Promise<void> {
await this.db(TABLE).del();
}
destroy(): void {}
async upsert(trafficDataUsage: IStatTrafficUsage): Promise<void> {
const row = toRow(trafficDataUsage);
await this.db(TABLE)
.insert(row)
.onConflict(['day', 'traffic_group', 'status_code_series'])
.merge({
count: this.db.raw('stat_traffic_usage.count + EXCLUDED.count'),
});
}
async getDailyTrafficDataUsageForPeriod(
from: Date,
to: Date,
): Promise<IStatTrafficUsage[]> {
const rows = await this.db<IStatTrafficUsage>(TABLE)
.where('day', '>=', startOfDay(from))
.andWhere('day', '<=', endOfDay(to));
return rows.map(mapRow);
}
async getMonthlyTrafficDataUsageForPeriod(
from: Date,
to: Date,
): Promise<IStatMonthlyTrafficUsage[]> {
const rows = await this.db(TABLE)
.select(
'traffic_group',
'status_code_series',
this.db.raw(`to_char(day, 'YYYY-MM') AS month`),
this.db.raw(`SUM(count) AS count`),
)
.where('day', '>=', startOfDay(from))
.andWhere('day', '<=', endOfDay(to))
.groupBy([
'traffic_group',
this.db.raw(`to_char(day, 'YYYY-MM')`),
'status_code_series',
])
.orderBy([
{ column: 'month', order: 'desc' },
{ column: 'traffic_group', order: 'asc' },
]);
return rows.map(
({ traffic_group, status_code_series, month, count }) => ({
trafficGroup: traffic_group,
statusCodeSeries: status_code_series,
month,
count: Number.parseInt(count),
}),
);
}
async getTrafficDataUsageForPeriod(
period: string,
): Promise<IStatTrafficUsage[]> {
const month = new Date(period);
return this.getDailyTrafficDataUsageForPeriod(
startOfMonth(month),
endOfMonth(month),
);
}
}