mirror of
https://github.com/kremalicious/umami.git
synced 2025-02-14 21:10:34 +01:00
* Initial Typescript models. * Re-add realtime data * get distinct sessions for session metrics * Add queries for new schema. * Fix Typo. * Add some api/team endpoints. * Fix destructure error. * Fix getWebsites call. * Ignore typescript build errors. * Fix enum issue. * add clickhouse route to deleteWebsite * Fix Website auth. * Updated lint-staged config. * Add permission checks. * Add user role api. * Fix error when updating website. * Fix isAdmin check. Fix Schema. * Initial conversion to react-basics. * Remove user/team transfer from website update. * delete website in relational query * Fix login secure token creation. * Add event type to event. * Allow user to be added to team with role. * Updated login form. * Add Role to TeamUser. * Add database migration. * Refactored permissions check. Updated redis lib. * Feat/um 114 roles and permissions (#1683) * Auth checkpoint. * Merge branch 'dev' into feat/um-114-roles-and-permissions * Add 02 migration. * Added lib/types. * Updated schema. * Updated roles and permissions logic. * Implement react-basics styles. Fix queries. * Update website details layout. * Add 01 migration. * Fix admin create. * Update react-basics. Co-authored-by: Francis Cao <franciscao@gmail.com> Co-authored-by: Mike Cao <mike@mikecao.com> Co-authored-by: Mike Cao <moocao@gmail.com>
97 lines
2.0 KiB
JavaScript
97 lines
2.0 KiB
JavaScript
import { parseToken } from 'next-basics';
|
|
import { validate } from 'uuid';
|
|
import { secret, uuid } from 'lib/crypto';
|
|
import cache from 'lib/cache';
|
|
import clickhouse from 'lib/clickhouse';
|
|
import { getClientInfo, getJsonBody } from 'lib/request';
|
|
import { createSession, getSession, getWebsite } from 'queries';
|
|
|
|
export async function findSession(req) {
|
|
const { payload } = getJsonBody(req);
|
|
|
|
if (!payload) {
|
|
return null;
|
|
}
|
|
|
|
// Check if cache token is passed
|
|
const cacheToken = req.headers['x-umami-cache'];
|
|
|
|
if (cacheToken) {
|
|
const result = await parseToken(cacheToken, secret());
|
|
|
|
if (result) {
|
|
return result;
|
|
}
|
|
}
|
|
|
|
// Verify payload
|
|
const { website: websiteId, hostname, screen, language } = payload;
|
|
|
|
if (!validate(websiteId)) {
|
|
return null;
|
|
}
|
|
|
|
// Find website
|
|
let website;
|
|
|
|
if (cache.enabled) {
|
|
website = await cache.fetchWebsite(websiteId);
|
|
} else {
|
|
website = await getWebsite({ id: websiteId });
|
|
}
|
|
|
|
if (!website || website.deletedAt) {
|
|
throw new Error(`Website not found: ${websiteId}`);
|
|
}
|
|
|
|
const { userAgent, browser, os, ip, country, device } = await getClientInfo(req, payload);
|
|
const sessionId = uuid(websiteId, hostname, ip, userAgent);
|
|
|
|
// Clickhouse does not require session lookup
|
|
if (clickhouse.enabled) {
|
|
return {
|
|
id: sessionId,
|
|
websiteId,
|
|
hostname,
|
|
browser,
|
|
os,
|
|
device,
|
|
screen,
|
|
language,
|
|
country,
|
|
};
|
|
}
|
|
|
|
// Find session
|
|
let session;
|
|
|
|
if (cache.enabled) {
|
|
session = await cache.fetchSession(sessionId);
|
|
} else {
|
|
session = await getSession({ id: sessionId });
|
|
}
|
|
|
|
// Create a session if not found
|
|
if (!session) {
|
|
try {
|
|
session = await createSession({
|
|
id: sessionId,
|
|
websiteId,
|
|
hostname,
|
|
browser,
|
|
os,
|
|
device,
|
|
screen,
|
|
language,
|
|
country,
|
|
});
|
|
} catch (e) {
|
|
if (!e.message.toLowerCase().includes('unique constraint')) {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
|
|
return session;
|
|
}
|