This commit is contained in:
Mike Cao 2022-11-20 21:04:53 -08:00
commit df50d9cefc
24 changed files with 136 additions and 122 deletions

View File

@ -1,10 +1,3 @@
export interface User {
id: string;
username: string;
isAdmin: boolean;
createdAt: string;
}
export interface Website { export interface Website {
id: string; id: string;
userId: string; userId: string;

7
interface/enum.d.ts vendored
View File

@ -1,7 +0,0 @@
/* eslint-disable no-unused-vars */
export namespace UmamiApi {
enum EventType {
Pageview = 1,
Event = 2,
}
}

View File

@ -1,8 +1,9 @@
import { parseSecureToken, parseToken } from 'next-basics';
import { getUser, getWebsite } from 'queries';
import debug from 'debug'; import debug from 'debug';
import { SHARE_TOKEN_HEADER, TYPE_USER, TYPE_WEBSITE } from 'lib/constants'; import { NextApiRequestAuth } from 'interface/api/nextApi';
import { SHARE_TOKEN_HEADER, UmamiApi } from 'lib/constants';
import { secret } from 'lib/crypto'; import { secret } from 'lib/crypto';
import { parseSecureToken, parseToken } from 'next-basics';
import { getUser, getUserWebsite } from 'queries';
const log = debug('umami:auth'); const log = debug('umami:auth');
@ -47,30 +48,38 @@ export function isValidToken(token, validation) {
return false; return false;
} }
export async function allowQuery(req, type) { export async function allowQuery(
const { id } = req.query; req: NextApiRequestAuth,
type: UmamiApi.AuthType,
typeId?: string,
) {
const { id } = req.query as { id: string };
const { user, shareToken } = req.auth; const { user, shareToken } = req.auth;
if (user?.isAdmin) {
return true;
}
if (shareToken) { if (shareToken) {
return isValidToken(shareToken, { id }); return isValidToken(shareToken, { id });
} }
if (user?.id) { if (user?.id) {
if (type === TYPE_WEBSITE) { if (type === UmamiApi.AuthType.Website) {
const website = await getWebsite({ id }); const userWebsite = await getUserWebsite({
userId: user.id,
websiteId: typeId ?? id,
isDeleted: false,
});
return website && website.userId === user.id; return userWebsite;
} else if (type === TYPE_USER) { } else if (type === UmamiApi.AuthType.User) {
const user = await getUser({ id }); const user = await getUser({ id });
return user && user.id === id; return user && user.id === id;
} }
} }
if (user?.isAdmin) {
return true;
}
return false; return false;
} }

View File

@ -1,5 +1,6 @@
import { getWebsite, getUser, getSession } from '../queries'; import { getWebsite, getUser, getSession } from '../queries';
import redis, { DELETED } from 'lib/redis'; import redis, { DELETED } from 'lib/redis';
import { Role, Team, TeamUser, User, UserRole, UserWebsite, Website } from '@prisma/client';
async function fetchObject(key, query) { async function fetchObject(key, query) {
const obj = await redis.get(key); const obj = await redis.get(key);
@ -40,8 +41,14 @@ async function deleteWebsite(id) {
return deleteObject(`website:${id}`); return deleteObject(`website:${id}`);
} }
async function fetchUser(id) { async function fetchUser(id): Promise<
return fetchObject(`user:${id}`, () => getUser({ id })); User & {
userRole?: (UserRole & { role: Role })[];
teamUser?: (TeamUser & { team: Team })[];
userWebsite?: (UserWebsite & { website: Website })[];
}
> {
return fetchObject(`user:${id}`, () => getUser({ id }, true));
} }
async function storeUser(data) { async function storeUser(data) {

View File

@ -1,3 +1,15 @@
/* eslint-disable no-unused-vars */
export namespace UmamiApi {
export enum EventType {
Pageview = 1,
Event = 2,
}
export enum AuthType {
Website,
User,
}
}
export const CURRENT_VERSION = process.env.currentVersion; export const CURRENT_VERSION = process.env.currentVersion;
export const AUTH_TOKEN = 'umami.auth'; export const AUTH_TOKEN = 'umami.auth';
export const LOCALE_CONFIG = 'umami.locale'; export const LOCALE_CONFIG = 'umami.locale';

View File

@ -7,12 +7,11 @@ import {
methodNotAllowed, methodNotAllowed,
getRandomChars, getRandomChars,
} from 'next-basics'; } from 'next-basics';
import { getUser } from 'queries'; import { getUser, User } from 'queries';
import { secret } from 'lib/crypto'; import { secret } from 'lib/crypto';
import redis from 'lib/redis'; import redis from 'lib/redis';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
import { User } from 'interface/api/models';
export interface LoginRequestBody { export interface LoginRequestBody {
username: string; username: string;

View File

@ -1,18 +1,17 @@
import { getUser, updateUser } from 'queries'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth';
import { UmamiApi } from 'lib/constants';
import { useAuth } from 'lib/middleware'; import { useAuth } from 'lib/middleware';
import { NextApiResponse } from 'next';
import { import {
badRequest, badRequest,
checkPassword,
hashPassword,
methodNotAllowed, methodNotAllowed,
ok, ok,
unauthorized, unauthorized,
checkPassword,
hashPassword,
} from 'next-basics'; } from 'next-basics';
import { allowQuery } from 'lib/auth'; import { getUser, updateUser, User } from 'queries';
import { TYPE_USER } from 'lib/constants';
import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { NextApiResponse } from 'next';
import { User } from 'interface/api/models';
export interface UserPasswordRequestQuery { export interface UserPasswordRequestQuery {
id: string; id: string;
@ -32,7 +31,7 @@ export default async (
const { current_password, new_password } = req.body; const { current_password, new_password } = req.body;
const { id } = req.query; const { id } = req.query;
if (!(await allowQuery(req, TYPE_USER))) { if (!(await allowQuery(req, UmamiApi.AuthType.User))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,11 +1,11 @@
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { allowQuery } from 'lib/auth';
import { useAuth, useCors } from 'lib/middleware';
import { getActiveVisitors } from 'queries';
import { TYPE_WEBSITE } from 'lib/constants';
import { WebsiteActive } from 'interface/api/models'; import { WebsiteActive } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth';
import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { getActiveVisitors } from 'queries';
export interface WebsiteActiveRequestQuery { export interface WebsiteActiveRequestQuery {
id: string; id: string;
@ -19,7 +19,7 @@ export default async (
await useAuth(req, res); await useAuth(req, res);
if (req.method === 'GET') { if (req.method === 'GET') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,12 +1,11 @@
import moment from 'moment-timezone';
import { getEventData } from 'queries';
import { ok, badRequest, methodNotAllowed, unauthorized } from 'next-basics';
import { allowQuery } from 'lib/auth';
import { useAuth, useCors } from 'lib/middleware';
import { TYPE_WEBSITE } from 'lib/constants';
import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { NextApiResponse } from 'next';
import { WebsiteMetric } from 'interface/api/models'; import { WebsiteMetric } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth';
import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware';
import { NextApiResponse } from 'next';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { getEventData } from 'queries';
export interface WebsiteEventDataRequestQuery { export interface WebsiteEventDataRequestQuery {
id: string; id: string;
@ -15,7 +14,6 @@ export interface WebsiteEventDataRequestQuery {
export interface WebsiteEventDataRequestBody { export interface WebsiteEventDataRequestBody {
start_at: string; start_at: string;
end_at: string; end_at: string;
timezone: string;
event_name: string; event_name: string;
columns: { [key: string]: 'count' | 'max' | 'min' | 'avg' | 'sum' }; columns: { [key: string]: 'count' | 'max' | 'min' | 'avg' | 'sum' };
filters?: { [key: string]: any }; filters?: { [key: string]: any };
@ -29,17 +27,13 @@ export default async (
await useAuth(req, res); await useAuth(req, res);
if (req.method === 'POST') { if (req.method === 'POST') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }
const { id: websiteId } = req.query; const { id: websiteId } = req.query;
const { start_at, end_at, timezone, event_name: eventName, columns, filters } = req.body; const { start_at, end_at, event_name: eventName, columns, filters } = req.body;
if (!moment.tz.zone(timezone)) {
return badRequest(res);
}
const startDate = new Date(+start_at); const startDate = new Date(+start_at);
const endDate = new Date(+end_at); const endDate = new Date(+end_at);
@ -47,7 +41,6 @@ export default async (
const events = await getEventData(websiteId, { const events = await getEventData(websiteId, {
startDate, startDate,
endDate, endDate,
timezone,
eventName, eventName,
columns, columns,
filters, filters,

View File

@ -1,7 +1,7 @@
import { WebsiteMetric } from 'interface/api/models'; import { WebsiteMetric } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth'; import { allowQuery } from 'lib/auth';
import { TYPE_WEBSITE } from 'lib/constants'; import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware'; import { useAuth, useCors } from 'lib/middleware';
import moment from 'moment-timezone'; import moment from 'moment-timezone';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
@ -28,7 +28,7 @@ export default async (
await useAuth(req, res); await useAuth(req, res);
if (req.method === 'GET') { if (req.method === 'GET') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,11 +1,11 @@
import { Website } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth'; import { allowQuery } from 'lib/auth';
import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware'; import { useAuth, useCors } from 'lib/middleware';
import { NextApiResponse } from 'next';
import { methodNotAllowed, ok, serverError, unauthorized } from 'next-basics'; import { methodNotAllowed, ok, serverError, unauthorized } from 'next-basics';
import { deleteWebsite, getWebsite, updateWebsite } from 'queries'; import { deleteWebsite, getWebsite, updateWebsite } from 'queries';
import { TYPE_WEBSITE } from 'lib/constants';
import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { NextApiResponse } from 'next';
import { Website } from 'interface/api/models';
export interface WebsiteRequestQuery { export interface WebsiteRequestQuery {
id: string; id: string;
@ -26,7 +26,7 @@ export default async (
const { id: websiteId } = req.query; const { id: websiteId } = req.query;
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }
@ -45,7 +45,7 @@ export default async (
domain, domain,
shareId, shareId,
}); });
} catch (e) { } catch (e: any) {
if (e.message.includes('Unique constraint') && e.message.includes('share_id')) { if (e.message.includes('Unique constraint') && e.message.includes('share_id')) {
return serverError(res, 'That share ID is already taken.'); return serverError(res, 'That share ID is already taken.');
} }
@ -55,7 +55,7 @@ export default async (
} }
if (req.method === 'DELETE') { if (req.method === 'DELETE') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,7 +1,7 @@
import { WebsiteMetric } from 'interface/api/models'; import { WebsiteMetric } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth'; import { allowQuery } from 'lib/auth';
import { FILTER_IGNORED, TYPE_WEBSITE } from 'lib/constants'; import { FILTER_IGNORED, UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware'; import { useAuth, useCors } from 'lib/middleware';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
import { badRequest, methodNotAllowed, ok, unauthorized } from 'next-basics'; import { badRequest, methodNotAllowed, ok, unauthorized } from 'next-basics';
@ -57,7 +57,7 @@ export default async (
await useAuth(req, res); await useAuth(req, res);
if (req.method === 'GET') { if (req.method === 'GET') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,7 +1,7 @@
import { WebsitePageviews } from 'interface/api/models'; import { WebsitePageviews } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth'; import { allowQuery } from 'lib/auth';
import { TYPE_WEBSITE } from 'lib/constants'; import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware'; import { useAuth, useCors } from 'lib/middleware';
import moment from 'moment-timezone'; import moment from 'moment-timezone';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
@ -33,7 +33,7 @@ export default async (
await useAuth(req, res); await useAuth(req, res);
if (req.method === 'GET') { if (req.method === 'GET') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,10 +1,10 @@
import { resetWebsite } from 'queries';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { allowQuery } from 'lib/auth';
import { useAuth, useCors } from 'lib/middleware';
import { TYPE_WEBSITE } from 'lib/constants';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth';
import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { resetWebsite } from 'queries';
export interface WebsiteResetRequestQuery { export interface WebsiteResetRequestQuery {
id: string; id: string;
@ -20,7 +20,7 @@ export default async (
const { id: websiteId } = req.query; const { id: websiteId } = req.query;
if (req.method === 'POST') { if (req.method === 'POST') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,11 +1,11 @@
import { getWebsiteStats } from 'queries';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { allowQuery } from 'lib/auth';
import { useAuth, useCors } from 'lib/middleware';
import { TYPE_WEBSITE } from 'lib/constants';
import { WebsiteStats } from 'interface/api/models'; import { WebsiteStats } from 'interface/api/models';
import { NextApiRequestQueryBody } from 'interface/api/nextApi'; import { NextApiRequestQueryBody } from 'interface/api/nextApi';
import { allowQuery } from 'lib/auth';
import { UmamiApi } from 'lib/constants';
import { useAuth, useCors } from 'lib/middleware';
import { NextApiResponse } from 'next'; import { NextApiResponse } from 'next';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { getWebsiteStats } from 'queries';
export interface WebsiteStatsRequestQuery { export interface WebsiteStatsRequestQuery {
id: string; id: string;
@ -28,7 +28,7 @@ export default async (
await useAuth(req, res); await useAuth(req, res);
if (req.method === 'GET') { if (req.method === 'GET') {
if (!(await allowQuery(req, TYPE_WEBSITE))) { if (!(await allowQuery(req, UmamiApi.AuthType.Website))) {
return unauthorized(res); return unauthorized(res);
} }

View File

@ -1,5 +1,5 @@
import { Prisma } from '@prisma/client'; import { Prisma } from '@prisma/client';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
import cache from 'lib/cache'; import cache from 'lib/cache';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
@ -82,8 +82,7 @@ export async function updateUser(
data: Prisma.UserUpdateInput, data: Prisma.UserUpdateInput,
where: Prisma.UserWhereUniqueInput, where: Prisma.UserWhereUniqueInput,
): Promise<User> { ): Promise<User> {
return prisma.client.user return prisma.client.user.update({
.update({
where, where,
data, data,
select: { select: {
@ -92,11 +91,6 @@ export async function updateUser(
createdAt: true, createdAt: true,
userRole: true, userRole: true,
}, },
})
.then(user => {
const { userRole, ...rest } = user;
return { ...rest, isAdmin: userRole.some(a => a.roleId === UmamiApi.SystemRole.Admin) };
}); });
} }

View File

@ -9,10 +9,8 @@ export async function createUserWebsite(
}); });
} }
export async function getUserWebsite( export async function getUserWebsite(where: Prisma.UserWebsiteWhereInput): Promise<UserWebsite> {
where: Prisma.UserWebsiteWhereUniqueInput, return prisma.client.userWebsite.findFirst({
): Promise<UserWebsite> {
return prisma.client.userWebsite.findUnique({
where, where,
}); });
} }

View File

@ -1,6 +1,7 @@
import { Prisma, Website } from '@prisma/client'; import { Prisma, Website } from '@prisma/client';
import cache from 'lib/cache'; import cache from 'lib/cache';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
import { runQuery, CLICKHOUSE, PRISMA } from 'lib/db';
export async function createWebsiteByUser( export async function createWebsiteByUser(
userId: string, userId: string,
@ -150,7 +151,14 @@ export async function getAllWebsites(): Promise<(Website & { user: string })[]>
export async function deleteWebsite( export async function deleteWebsite(
websiteId: string, websiteId: string,
): Promise<[Prisma.BatchPayload, Prisma.BatchPayload, Website]> { ) {
return runQuery({
[PRISMA]: () => deleteWebsiteRelationalQuery(websiteId),
[CLICKHOUSE]: () => deleteWebsiteClickhouseQuery(websiteId),
});
}
async function deleteWebsiteRelationalQuery(websiteId): Promise<[Prisma.BatchPayload, Prisma.BatchPayload, Website]> {
const { client, transaction } = prisma; const { client, transaction } = prisma;
return transaction([ return transaction([
@ -174,3 +182,12 @@ export async function deleteWebsite(
return data; return data;
}); });
} }
async function deleteWebsiteClickhouseQuery(websiteId): Promise<Website> {
return prisma.client.website.update({
data: {
isDeleted: true,
},
where: { id: websiteId },
});
}

View File

@ -3,7 +3,7 @@ import { CLICKHOUSE, PRISMA, runQuery } from 'lib/db';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
import cache from 'lib/cache'; import cache from 'lib/cache';
import { WebsiteMetric } from 'interface/api/models'; import { WebsiteMetric } from 'interface/api/models';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
export async function getEventData( export async function getEventData(
...args: [ ...args: [
@ -11,7 +11,7 @@ export async function getEventData(
data: { data: {
startDate: Date; startDate: Date;
endDate: Date; endDate: Date;
event_name: string; eventName: string;
columns: any; columns: any;
filters: object; filters: object;
}, },
@ -32,12 +32,12 @@ async function relationalQuery(
data: { data: {
startDate: Date; startDate: Date;
endDate: Date; endDate: Date;
event_name: string; eventName: string;
columns: any; columns: any;
filters: object; filters: object;
}, },
) { ) {
const { startDate, endDate, event_name, columns, filters } = data; const { startDate, endDate, eventName, columns, filters } = data;
const { rawQuery, getEventDataColumnsQuery, getEventDataFilterQuery } = prisma; const { rawQuery, getEventDataColumnsQuery, getEventDataFilterQuery } = prisma;
const params = [startDate, endDate]; const params = [startDate, endDate];
@ -48,7 +48,7 @@ async function relationalQuery(
where website_id ='${websiteId}' where website_id ='${websiteId}'
and created_at between $1 and $2 and created_at between $1 and $2
and event_type = ${UmamiApi.EventType.Event} and event_type = ${UmamiApi.EventType.Event}
${event_name ? `and event_name = ${event_name}` : ''} ${eventName ? `and eventName = ${eventName}` : ''}
${ ${
Object.keys(filters).length > 0 Object.keys(filters).length > 0
? `and ${getEventDataFilterQuery('event_data', filters)}` ? `and ${getEventDataFilterQuery('event_data', filters)}`
@ -63,12 +63,12 @@ async function clickhouseQuery(
data: { data: {
startDate: Date; startDate: Date;
endDate: Date; endDate: Date;
event_name: string; eventName: string;
columns: any; columns: any;
filters: object; filters: object;
}, },
) { ) {
const { startDate, endDate, event_name, columns, filters } = data; const { startDate, endDate, eventName, columns, filters } = data;
const { rawQuery, getBetweenDates, getEventDataColumnsQuery, getEventDataFilterQuery } = const { rawQuery, getBetweenDates, getEventDataColumnsQuery, getEventDataFilterQuery } =
clickhouse; clickhouse;
const website = await cache.fetchWebsite(websiteId); const website = await cache.fetchWebsite(websiteId);
@ -81,7 +81,7 @@ async function clickhouseQuery(
where website_id = $1 where website_id = $1
and rev_id = $2 and rev_id = $2
and event_type = ${UmamiApi.EventType.Event} and event_type = ${UmamiApi.EventType.Event}
${event_name ? `and event_name = ${event_name}` : ''} ${eventName ? `and eventName = ${eventName}` : ''}
and ${getBetweenDates('created_at', startDate, endDate)} and ${getBetweenDates('created_at', startDate, endDate)}
${ ${
Object.keys(filters).length > 0 Object.keys(filters).length > 0

View File

@ -3,7 +3,7 @@ import clickhouse from 'lib/clickhouse';
import { runQuery, CLICKHOUSE, PRISMA } from 'lib/db'; import { runQuery, CLICKHOUSE, PRISMA } from 'lib/db';
import cache from 'lib/cache'; import cache from 'lib/cache';
import { WebsiteEventMetric } from 'interface/api/models'; import { WebsiteEventMetric } from 'interface/api/models';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
export async function getEventMetrics( export async function getEventMetrics(
...args: [ ...args: [

View File

@ -4,7 +4,7 @@ import kafka from 'lib/kafka';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
import { uuid } from 'lib/crypto'; import { uuid } from 'lib/crypto';
import cache from 'lib/cache'; import cache from 'lib/cache';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
export async function saveEvent(args: { export async function saveEvent(args: {
id: string; id: string;

View File

@ -3,7 +3,7 @@ import clickhouse from 'lib/clickhouse';
import { runQuery, CLICKHOUSE, PRISMA } from 'lib/db'; import { runQuery, CLICKHOUSE, PRISMA } from 'lib/db';
import cache from 'lib/cache'; import cache from 'lib/cache';
import { Prisma } from '@prisma/client'; import { Prisma } from '@prisma/client';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
export async function getPageviewMetrics( export async function getPageviewMetrics(
...args: [ ...args: [

View File

@ -2,7 +2,7 @@ import cache from 'lib/cache';
import clickhouse from 'lib/clickhouse'; import clickhouse from 'lib/clickhouse';
import { CLICKHOUSE, PRISMA, runQuery } from 'lib/db'; import { CLICKHOUSE, PRISMA, runQuery } from 'lib/db';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
export async function getPageviewStats( export async function getPageviewStats(
...args: [ ...args: [

View File

@ -4,7 +4,7 @@ import kafka from 'lib/kafka';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
import cache from 'lib/cache'; import cache from 'lib/cache';
import { uuid } from 'lib/crypto'; import { uuid } from 'lib/crypto';
import { UmamiApi } from 'interface/enum'; import { UmamiApi } from 'lib/constants';
export async function savePageView(args: { export async function savePageView(args: {
id: string; id: string;