mirror of
https://github.com/umami-software/umami.git
synced 2026-02-04 04:37:11 +01:00
Imported libraries, removed next-basics.
This commit is contained in:
parent
31266cb1ac
commit
113022ed17
44 changed files with 361 additions and 180 deletions
2
next-env.d.ts
vendored
2
next-env.d.ts
vendored
|
|
@ -2,4 +2,4 @@
|
|||
/// <reference types="next/image-types/global" />
|
||||
|
||||
// NOTE: This file should not be edited
|
||||
// see https://nextjs.org/docs/app/api-reference/config/typescript for more information.
|
||||
// see https://nextjs.org/docs/app/building-your-application/configuring/typescript for more information.
|
||||
|
|
|
|||
|
|
@ -76,6 +76,7 @@
|
|||
"@tanstack/react-query": "^5.28.6",
|
||||
"@umami/prisma-client": "^0.14.0",
|
||||
"@umami/redis-client": "^0.24.0",
|
||||
"bcryptjs": "^2.4.3",
|
||||
"chalk": "^4.1.1",
|
||||
"chart.js": "^4.4.2",
|
||||
"chartjs-adapter-date-fns": "^3.0.0",
|
||||
|
|
@ -97,14 +98,15 @@
|
|||
"is-docker": "^3.0.0",
|
||||
"is-localhost-ip": "^1.4.0",
|
||||
"isbot": "^5.1.16",
|
||||
"jsonwebtoken": "^9.0.2",
|
||||
"kafkajs": "^2.1.0",
|
||||
"maxmind": "^4.3.6",
|
||||
"md5": "^2.3.0",
|
||||
"next": "15.0.4",
|
||||
"next-basics": "^0.39.0",
|
||||
"node-fetch": "^3.2.8",
|
||||
"npm-run-all": "^4.1.5",
|
||||
"prisma": "6.1.0",
|
||||
"pure-rand": "^6.1.0",
|
||||
"react": "^19.0.0",
|
||||
"react-basics": "^0.126.0",
|
||||
"react-dom": "^19.0.0",
|
||||
|
|
|
|||
|
|
@ -1,94 +0,0 @@
|
|||
/* eslint-disable no-console */
|
||||
require('dotenv').config();
|
||||
const { hashPassword } = require('next-basics');
|
||||
const chalk = require('chalk');
|
||||
const prompts = require('prompts');
|
||||
const { PrismaClient } = require('@prisma/client');
|
||||
|
||||
const prisma = new PrismaClient();
|
||||
|
||||
const runQuery = async query => {
|
||||
return query.catch(e => {
|
||||
throw e;
|
||||
});
|
||||
};
|
||||
|
||||
const updateUserByUsername = (username, data) => {
|
||||
return runQuery(
|
||||
prisma.user.update({
|
||||
where: {
|
||||
username,
|
||||
},
|
||||
data,
|
||||
}),
|
||||
);
|
||||
};
|
||||
|
||||
const changePassword = async (username, newPassword) => {
|
||||
const password = hashPassword(newPassword);
|
||||
return updateUserByUsername(username, { password });
|
||||
};
|
||||
|
||||
const getUsernameAndPassword = async () => {
|
||||
let [username, password] = process.argv.slice(2);
|
||||
if (username && password) {
|
||||
return { username, password };
|
||||
}
|
||||
|
||||
const questions = [];
|
||||
if (!username) {
|
||||
questions.push({
|
||||
type: 'text',
|
||||
name: 'username',
|
||||
message: 'Enter user to change password',
|
||||
});
|
||||
}
|
||||
if (!password) {
|
||||
questions.push(
|
||||
{
|
||||
type: 'password',
|
||||
name: 'password',
|
||||
message: 'Enter new password',
|
||||
},
|
||||
{
|
||||
type: 'password',
|
||||
name: 'confirmation',
|
||||
message: 'Confirm new password',
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
const answers = await prompts(questions);
|
||||
if (answers.password !== answers.confirmation) {
|
||||
throw new Error(`Passwords don't match`);
|
||||
}
|
||||
|
||||
return {
|
||||
username: username || answers.username,
|
||||
password: answers.password,
|
||||
};
|
||||
};
|
||||
|
||||
(async () => {
|
||||
let username, password;
|
||||
|
||||
try {
|
||||
({ username, password } = await getUsernameAndPassword());
|
||||
} catch (error) {
|
||||
console.log(chalk.redBright(error.message));
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
await changePassword(username, password);
|
||||
console.log('Password changed for user', chalk.greenBright(username));
|
||||
} catch (error) {
|
||||
if (error.meta.cause.includes('Record to update not found')) {
|
||||
console.log('User not found:', chalk.redBright(username));
|
||||
} else {
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
prisma.$disconnect();
|
||||
})();
|
||||
|
|
@ -1,4 +1,5 @@
|
|||
'use client';
|
||||
import { useEffect } from 'react';
|
||||
import { Icon, Text } from 'react-basics';
|
||||
import Link from 'next/link';
|
||||
import classNames from 'classnames';
|
||||
|
|
@ -9,9 +10,8 @@ import ProfileButton from 'components/input/ProfileButton';
|
|||
import TeamsButton from 'components/input/TeamsButton';
|
||||
import Icons from 'components/icons';
|
||||
import { useMessages, useNavigation, useTeamUrl } from 'components/hooks';
|
||||
import { getItem, setItem } from 'lib/storage';
|
||||
import styles from './NavBar.module.css';
|
||||
import { useEffect } from 'react';
|
||||
import { getItem, setItem } from 'next-basics';
|
||||
|
||||
export function NavBar() {
|
||||
const { formatMessage, labels } = useMessages();
|
||||
|
|
|
|||
|
|
@ -1,7 +1,7 @@
|
|||
import { useEffect, useCallback, useState } from 'react';
|
||||
import { createPortal } from 'react-dom';
|
||||
import { Button } from 'react-basics';
|
||||
import { setItem } from 'next-basics';
|
||||
import { setItem } from 'lib/storage';
|
||||
import useStore, { checkVersion } from 'store/version';
|
||||
import { REPO_URL, VERSION_CHECK } from 'lib/constants';
|
||||
import { useMessages } from 'components/hooks';
|
||||
|
|
|
|||
|
|
@ -9,7 +9,7 @@ import {
|
|||
LoadingButton,
|
||||
} from 'react-basics';
|
||||
import { useContext, useState } from 'react';
|
||||
import { getRandomChars } from 'next-basics';
|
||||
import { getRandomChars } from 'lib/crypto';
|
||||
import { useApi, useMessages, useModified } from 'components/hooks';
|
||||
import { WebsiteContext } from 'app/(main)/websites/[websiteId]/WebsiteProvider';
|
||||
|
||||
|
|
|
|||
|
|
@ -9,7 +9,7 @@ import {
|
|||
Flexbox,
|
||||
useToasts,
|
||||
} from 'react-basics';
|
||||
import { getRandomChars } from 'next-basics';
|
||||
import { getRandomChars } from 'lib/crypto';
|
||||
import { useContext, useRef, useState } from 'react';
|
||||
import { useApi, useMessages, useModified } from 'components/hooks';
|
||||
import { TeamContext } from 'app/(main)/teams/[teamId]/TeamProvider';
|
||||
|
|
|
|||
|
|
@ -6,7 +6,6 @@ import Icons from 'components/icons';
|
|||
import { BROWSERS, OS_NAMES } from 'lib/constants';
|
||||
import { stringToColor } from 'lib/format';
|
||||
import { RealtimeData } from 'lib/types';
|
||||
import { safeDecodeURI } from 'next-basics';
|
||||
import { useContext, useMemo, useState } from 'react';
|
||||
import { Icon, SearchField, StatusLight, Text } from 'react-basics';
|
||||
import { FixedSizeList } from 'react-window';
|
||||
|
|
@ -99,7 +98,7 @@ export function RealtimeLog({ data }: { data: RealtimeData }) {
|
|||
target="_blank"
|
||||
rel="noreferrer noopener"
|
||||
>
|
||||
{safeDecodeURI(url)}
|
||||
{url}
|
||||
</a>
|
||||
);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,5 +1,6 @@
|
|||
import { z } from 'zod';
|
||||
import { checkPassword, createSecureToken } from 'next-basics';
|
||||
import { checkPassword } from 'lib/auth';
|
||||
import { createSecureToken } from 'lib/jwt';
|
||||
import { redisEnabled } from '@umami/redis-client';
|
||||
import { getUserByUsername } from 'queries';
|
||||
import { json, unauthorized } from 'lib/response';
|
||||
|
|
|
|||
|
|
@ -1,8 +1,8 @@
|
|||
import { z } from 'zod';
|
||||
import { checkPassword, hashPassword } from 'lib/auth';
|
||||
import { parseRequest } from 'lib/request';
|
||||
import { json, badRequest } from 'lib/response';
|
||||
import { getUser, updateUser } from 'queries/prisma/user';
|
||||
import { checkPassword, hashPassword } from 'next-basics';
|
||||
|
||||
export async function POST(request: Request) {
|
||||
const schema = z.object({
|
||||
|
|
|
|||
|
|
@ -1,14 +1,15 @@
|
|||
import { z } from 'zod';
|
||||
import { isbot } from 'isbot';
|
||||
import { createToken, parseToken, safeDecodeURI } from 'next-basics';
|
||||
import { createToken, parseToken } from 'lib/jwt';
|
||||
import { safeDecodeURI } from 'lib/url';
|
||||
import clickhouse from 'lib/clickhouse';
|
||||
import { parseRequest } from 'lib/request';
|
||||
import { badRequest, json, forbidden, serverError } from 'lib/response';
|
||||
import { fetchSession, fetchWebsite } from 'lib/load';
|
||||
import { getClientInfo, hasBlockedIp } from 'lib/detect';
|
||||
import { secret, uuid, visitSalt } from 'lib/crypto';
|
||||
import { createSession, saveEvent, saveSessionData } from 'queries';
|
||||
import { COLLECTION_TYPE } from 'lib/constants';
|
||||
import { createSession, saveEvent, saveSessionData } from 'queries';
|
||||
|
||||
export async function POST(request: Request) {
|
||||
// Bot check
|
||||
|
|
|
|||
|
|
@ -1,7 +1,7 @@
|
|||
import { json, notFound } from 'lib/response';
|
||||
import { getSharedWebsite } from 'queries';
|
||||
import { createToken } from 'next-basics';
|
||||
import { createToken } from 'lib/jwt';
|
||||
import { secret } from 'lib/crypto';
|
||||
import { getSharedWebsite } from 'queries';
|
||||
|
||||
export async function GET(request: Request, { params }: { params: Promise<{ shareId: string }> }) {
|
||||
const { shareId } = await params;
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
import { z } from 'zod';
|
||||
import { getRandomChars } from 'next-basics';
|
||||
import { getRandomChars } from 'lib/crypto';
|
||||
import { unauthorized, json } from 'lib/response';
|
||||
import { canCreateTeam } from 'lib/auth';
|
||||
import { uuid } from 'lib/crypto';
|
||||
|
|
|
|||
|
|
@ -2,7 +2,7 @@ import { z } from 'zod';
|
|||
import { canUpdateUser, canViewUser, canDeleteUser } from 'lib/auth';
|
||||
import { getUser, getUserByUsername, updateUser, deleteUser } from 'queries';
|
||||
import { json, unauthorized, badRequest, ok } from 'lib/response';
|
||||
import { hashPassword } from 'next-basics';
|
||||
import { hashPassword } from 'lib/auth';
|
||||
import { parseRequest } from 'lib/request';
|
||||
|
||||
export async function GET(request: Request, { params }: { params: Promise<{ userId: string }> }) {
|
||||
|
|
|
|||
|
|
@ -1,6 +1,5 @@
|
|||
import { z } from 'zod';
|
||||
import { hashPassword } from 'next-basics';
|
||||
import { canCreateUser } from 'lib/auth';
|
||||
import { hashPassword, canCreateUser } from 'lib/auth';
|
||||
import { ROLES } from 'lib/constants';
|
||||
import { uuid } from 'lib/crypto';
|
||||
import { parseRequest } from 'lib/request';
|
||||
|
|
|
|||
|
|
@ -1,6 +1,5 @@
|
|||
import classNames from 'classnames';
|
||||
import { useMessages, useNavigation } from 'components/hooks';
|
||||
import { safeDecodeURIComponent } from 'next-basics';
|
||||
import Link from 'next/link';
|
||||
import { ReactNode } from 'react';
|
||||
import { Icon, Icons } from 'react-basics';
|
||||
|
|
@ -39,7 +38,7 @@ export function FilterLink({
|
|||
{!value && `(${label || formatMessage(labels.unknown)})`}
|
||||
{value && (
|
||||
<Link href={renderUrl({ [id]: value })} className={styles.label} replace>
|
||||
{safeDecodeURIComponent(label || value)}
|
||||
{label || value}
|
||||
</Link>
|
||||
)}
|
||||
{externalUrl && (
|
||||
|
|
|
|||
|
|
@ -11,6 +11,7 @@ export function useTeams(userId: string) {
|
|||
queryFn: (params: any) => {
|
||||
return get(`/users/${userId}/teams`, params);
|
||||
},
|
||||
enabled: !!userId,
|
||||
});
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -1,7 +1,8 @@
|
|||
import { useCallback } from 'react';
|
||||
import * as reactQuery from '@tanstack/react-query';
|
||||
import { useApi as nextUseApi } from 'next-basics';
|
||||
import { getClientAuthToken } from 'lib/client';
|
||||
import { SHARE_TOKEN_HEADER } from 'lib/constants';
|
||||
import { httpGet, httpPost, httpPut, httpDelete } from 'lib/fetch';
|
||||
import useStore from 'store/app';
|
||||
|
||||
const selector = (state: { shareToken: { token?: string } }) => state.shareToken;
|
||||
|
|
@ -9,12 +10,50 @@ const selector = (state: { shareToken: { token?: string } }) => state.shareToken
|
|||
export function useApi() {
|
||||
const shareToken = useStore(selector);
|
||||
|
||||
const { get, post, put, del } = nextUseApi(
|
||||
{ authorization: `Bearer ${getClientAuthToken()}`, [SHARE_TOKEN_HEADER]: shareToken?.token },
|
||||
process.env.basePath,
|
||||
);
|
||||
const defaultHeaders = {
|
||||
authorization: `Bearer ${getClientAuthToken()}`,
|
||||
[SHARE_TOKEN_HEADER]: shareToken?.token,
|
||||
};
|
||||
const basePath = process.env.basePath;
|
||||
|
||||
return { get, post, put, del, ...reactQuery };
|
||||
function getUrl(url: string, basePath = ''): string {
|
||||
return url.startsWith('http') ? url : `${basePath}/api${url}`;
|
||||
}
|
||||
|
||||
const getHeaders = (headers: any = {}) => {
|
||||
return { ...defaultHeaders, ...headers };
|
||||
};
|
||||
|
||||
return {
|
||||
get: useCallback(
|
||||
async (url: string, params: object = {}, headers: object = {}) => {
|
||||
return httpGet(getUrl(url, basePath), params, getHeaders(headers));
|
||||
},
|
||||
[httpGet],
|
||||
),
|
||||
|
||||
post: useCallback(
|
||||
async (url: string, params: object = {}, headers: object = {}) => {
|
||||
return httpPost(getUrl(url, basePath), params, getHeaders(headers));
|
||||
},
|
||||
[httpPost],
|
||||
),
|
||||
|
||||
put: useCallback(
|
||||
async (url: string, params: object = {}, headers: object = {}) => {
|
||||
return httpPut(getUrl(url, basePath), params, getHeaders(headers));
|
||||
},
|
||||
[httpPut],
|
||||
),
|
||||
|
||||
del: useCallback(
|
||||
async (url: string, params: object = {}, headers: object = {}) => {
|
||||
return httpDelete(getUrl(url, basePath), params, getHeaders(headers));
|
||||
},
|
||||
[httpDelete],
|
||||
),
|
||||
...reactQuery,
|
||||
};
|
||||
}
|
||||
|
||||
export default useApi;
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
import { useState, useEffect } from 'react';
|
||||
import { httpGet } from 'next-basics';
|
||||
import { httpGet } from 'lib/fetch';
|
||||
import enUS from '../../../public/intl/country/en-US.json';
|
||||
|
||||
const countryNames = {
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
import { getMinimumUnit, parseDateRange } from 'lib/date';
|
||||
import { setItem } from 'next-basics';
|
||||
import { setItem } from 'lib/storage';
|
||||
import { DATE_RANGE_CONFIG, DEFAULT_DATE_COMPARE, DEFAULT_DATE_RANGE } from 'lib/constants';
|
||||
import websiteStore, { setWebsiteDateRange, setWebsiteDateCompare } from 'store/websites';
|
||||
import appStore, { setDateRange } from 'store/app';
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
import { useState, useEffect } from 'react';
|
||||
import { httpGet } from 'next-basics';
|
||||
import { httpGet } from 'lib/fetch';
|
||||
import enUS from '../../../public/intl/language/en-US.json';
|
||||
|
||||
const languageNames = {
|
||||
|
|
|
|||
|
|
@ -1,5 +1,6 @@
|
|||
import { useEffect } from 'react';
|
||||
import { httpGet, setItem } from 'next-basics';
|
||||
import { httpGet } from 'lib/fetch';
|
||||
import { setItem } from 'lib/storage';
|
||||
import { LOCALE_CONFIG } from 'lib/constants';
|
||||
import { getDateLocale, getTextDirection } from 'lib/lang';
|
||||
import useStore, { setLocale } from 'store/app';
|
||||
|
|
|
|||
|
|
@ -1,6 +1,6 @@
|
|||
import { useMemo } from 'react';
|
||||
import { usePathname, useRouter, useSearchParams } from 'next/navigation';
|
||||
import { buildUrl, safeDecodeURIComponent } from 'next-basics';
|
||||
import { buildUrl } from 'lib/url';
|
||||
|
||||
export function useNavigation(): {
|
||||
pathname: string;
|
||||
|
|
@ -16,7 +16,7 @@ export function useNavigation(): {
|
|||
const obj = {};
|
||||
|
||||
for (const [key, value] of params.entries()) {
|
||||
obj[key] = safeDecodeURIComponent(value);
|
||||
obj[key] = value;
|
||||
}
|
||||
|
||||
return obj;
|
||||
|
|
|
|||
|
|
@ -1,6 +1,6 @@
|
|||
import { useEffect, useMemo } from 'react';
|
||||
import useStore, { setTheme } from 'store/app';
|
||||
import { getItem, setItem } from 'next-basics';
|
||||
import { getItem, setItem } from 'lib/storage';
|
||||
import { DEFAULT_THEME, THEME_COLORS, THEME_CONFIG } from 'lib/constants';
|
||||
import { colord } from 'colord';
|
||||
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
import { setItem } from 'next-basics';
|
||||
import { setItem } from 'lib/storage';
|
||||
import { TIMEZONE_CONFIG } from 'lib/constants';
|
||||
import { formatInTimeZone, zonedTimeToUtc, utcToZonedTime } from 'date-fns-tz';
|
||||
import useStore, { setTimezone } from 'store/app';
|
||||
|
|
|
|||
|
|
@ -16,7 +16,7 @@ export function TeamsButton({
|
|||
}) {
|
||||
const { user } = useLogin();
|
||||
const { formatMessage, labels } = useMessages();
|
||||
const { result } = useTeams(user?.id);
|
||||
const { result } = useTeams(user.id);
|
||||
const { teamId } = useTeamUrl();
|
||||
const team = result?.data?.find(({ id }) => id === teamId);
|
||||
|
||||
|
|
|
|||
|
|
@ -1,5 +1,4 @@
|
|||
import { StatusLight } from 'react-basics';
|
||||
import { safeDecodeURIComponent } from 'next-basics';
|
||||
import { colord } from 'colord';
|
||||
import classNames from 'classnames';
|
||||
import { LegendItem } from 'chart.js/auto';
|
||||
|
|
@ -28,9 +27,7 @@ export function Legend({
|
|||
className={classNames(styles.label, { [styles.hidden]: hidden })}
|
||||
onClick={() => onClick(item)}
|
||||
>
|
||||
<StatusLight color={color.alpha(color.alpha() + 0.2).toHex()}>
|
||||
{safeDecodeURIComponent(text)}
|
||||
</StatusLight>
|
||||
<StatusLight color={color.alpha(color.alpha() + 0.2).toHex()}>{text}</StatusLight>
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
|
|
|
|||
|
|
@ -1,5 +1,4 @@
|
|||
import { useState } from 'react';
|
||||
import { safeDecodeURI } from 'next-basics';
|
||||
import FilterButtons from 'components/common/FilterButtons';
|
||||
import { emptyFilter, paramFilter } from 'lib/filters';
|
||||
import { FILTER_RAW, FILTER_COMBINED } from 'lib/constants';
|
||||
|
|
@ -39,8 +38,8 @@ export function QueryParametersTable({
|
|||
x
|
||||
) : (
|
||||
<div className={styles.item}>
|
||||
<div className={styles.param}>{safeDecodeURI(p)}</div>
|
||||
<div className={styles.value}>{safeDecodeURI(v)}</div>
|
||||
<div className={styles.param}>{p}</div>
|
||||
<div className={styles.value}>{v}</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
|
|
|||
5
src/declaration.d.ts
vendored
5
src/declaration.d.ts
vendored
|
|
@ -1,5 +1,6 @@
|
|||
declare module 'bcryptjs';
|
||||
declare module 'chartjs-adapter-date-fns';
|
||||
declare module 'cors';
|
||||
declare module 'debug';
|
||||
declare module 'chartjs-adapter-date-fns';
|
||||
declare module 'jsonwebtoken';
|
||||
declare module 'md5';
|
||||
declare module 'request-ip';
|
||||
|
|
|
|||
|
|
@ -1,26 +1,30 @@
|
|||
import bcrypt from 'bcryptjs';
|
||||
import { Report } from '@prisma/client';
|
||||
import { getClient, redisEnabled } from '@umami/redis-client';
|
||||
import debug from 'debug';
|
||||
import { PERMISSIONS, ROLE_PERMISSIONS, ROLES, SHARE_TOKEN_HEADER } from 'lib/constants';
|
||||
import { secret } from 'lib/crypto';
|
||||
import { NextApiRequest } from 'next';
|
||||
import {
|
||||
createSecureToken,
|
||||
ensureArray,
|
||||
getRandomChars,
|
||||
parseSecureToken,
|
||||
parseToken,
|
||||
} from 'next-basics';
|
||||
import { secret, getRandomChars } from 'lib/crypto';
|
||||
import { createSecureToken, parseSecureToken, parseToken } from 'lib/jwt';
|
||||
import { ensureArray } from 'lib/utils';
|
||||
import { getTeamUser, getUser, getWebsite } from 'queries';
|
||||
import { Auth } from './types';
|
||||
|
||||
const log = debug('umami:auth');
|
||||
const cloudMode = process.env.CLOUD_MODE;
|
||||
const SALT_ROUNDS = 10;
|
||||
|
||||
export function hashPassword(password: string, rounds = SALT_ROUNDS) {
|
||||
return bcrypt.hashSync(password, rounds);
|
||||
}
|
||||
|
||||
export function checkPassword(password: string, passwordHash: string) {
|
||||
return bcrypt.compareSync(password, passwordHash);
|
||||
}
|
||||
|
||||
export async function checkAuth(request: Request) {
|
||||
const token = request.headers.get('authorization')?.split(' ')?.[1];
|
||||
const payload = parseSecureToken(token, secret());
|
||||
const shareToken = await parseShareToken(request as any);
|
||||
const shareToken = await parseShareToken(request.headers);
|
||||
|
||||
let user = null;
|
||||
const { userId, authKey, grant } = payload || {};
|
||||
|
|
@ -73,17 +77,9 @@ export async function saveAuth(data: any, expire = 0) {
|
|||
return createSecureToken({ authKey }, secret());
|
||||
}
|
||||
|
||||
export function getAuthToken(req: NextApiRequest) {
|
||||
export function parseShareToken(headers: Headers) {
|
||||
try {
|
||||
return req.headers.authorization.split(' ')[1];
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export function parseShareToken(req: Request) {
|
||||
try {
|
||||
return parseToken(req.headers[SHARE_TOKEN_HEADER], secret());
|
||||
return parseToken(headers.get(SHARE_TOKEN_HEADER), secret());
|
||||
} catch (e) {
|
||||
log(e);
|
||||
return null;
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
import { getItem, setItem, removeItem } from 'next-basics';
|
||||
import { getItem, setItem, removeItem } from 'lib/storage';
|
||||
import { AUTH_TOKEN } from './constants';
|
||||
|
||||
export function getClientAuthToken() {
|
||||
|
|
|
|||
|
|
@ -1,7 +1,78 @@
|
|||
import crypto from 'crypto';
|
||||
import { startOfHour, startOfMonth } from 'date-fns';
|
||||
import { hash } from 'next-basics';
|
||||
import prand from 'pure-rand';
|
||||
import { v4, v5 } from 'uuid';
|
||||
|
||||
const ALGORITHM = 'aes-256-gcm';
|
||||
const IV_LENGTH = 16;
|
||||
const SALT_LENGTH = 64;
|
||||
const TAG_LENGTH = 16;
|
||||
const TAG_POSITION = SALT_LENGTH + IV_LENGTH;
|
||||
const ENC_POSITION = TAG_POSITION + TAG_LENGTH;
|
||||
|
||||
const HASH_ALGO = 'sha512';
|
||||
const HASH_ENCODING = 'hex';
|
||||
|
||||
const seed = Date.now() ^ (Math.random() * 0x100000000);
|
||||
const rng = prand.xoroshiro128plus(seed);
|
||||
|
||||
export function random(min: number, max: number) {
|
||||
return prand.unsafeUniformIntDistribution(min, max, rng);
|
||||
}
|
||||
|
||||
export function getRandomChars(
|
||||
n: number,
|
||||
chars = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ',
|
||||
) {
|
||||
const arr = chars.split('');
|
||||
let s = '';
|
||||
for (let i = 0; i < n; i++) {
|
||||
s += arr[random(0, arr.length - 1)];
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
const getKey = (password: string, salt: Buffer) =>
|
||||
crypto.pbkdf2Sync(password, salt, 10000, 32, 'sha512');
|
||||
|
||||
export function encrypt(value: any, secret: any) {
|
||||
const iv = crypto.randomBytes(IV_LENGTH);
|
||||
const salt = crypto.randomBytes(SALT_LENGTH);
|
||||
const key = getKey(secret, salt);
|
||||
|
||||
const cipher = crypto.createCipheriv(ALGORITHM, key, iv);
|
||||
|
||||
const encrypted = Buffer.concat([cipher.update(String(value), 'utf8'), cipher.final()]);
|
||||
|
||||
const tag = cipher.getAuthTag();
|
||||
|
||||
return Buffer.concat([salt, iv, tag, encrypted]).toString('base64');
|
||||
}
|
||||
|
||||
export function decrypt(value: any, secret: any) {
|
||||
const str = Buffer.from(String(value), 'base64');
|
||||
const salt = str.subarray(0, SALT_LENGTH);
|
||||
const iv = str.subarray(SALT_LENGTH, TAG_POSITION);
|
||||
const tag = str.subarray(TAG_POSITION, ENC_POSITION);
|
||||
const encrypted = str.subarray(ENC_POSITION);
|
||||
|
||||
const key = getKey(secret, salt);
|
||||
|
||||
const decipher = crypto.createDecipheriv(ALGORITHM, key, iv);
|
||||
|
||||
decipher.setAuthTag(tag);
|
||||
|
||||
return decipher.update(encrypted) + decipher.final('utf8');
|
||||
}
|
||||
|
||||
export function hash(...args: string[]) {
|
||||
return crypto.createHash(HASH_ALGO).update(args.join('')).digest(HASH_ENCODING);
|
||||
}
|
||||
|
||||
export function md5(...args: string[]) {
|
||||
return crypto.createHash('md5').update(args.join('')).digest('hex');
|
||||
}
|
||||
|
||||
export function secret() {
|
||||
return hash(process.env.APP_SECRET || process.env.DATABASE_URL);
|
||||
}
|
||||
|
|
|
|||
31
src/lib/fetch.ts
Normal file
31
src/lib/fetch.ts
Normal file
|
|
@ -0,0 +1,31 @@
|
|||
import { buildUrl } from 'lib/url';
|
||||
|
||||
export async function request(method: string, url: string, body?: string, headers: object = {}) {
|
||||
const res = await fetch(url, {
|
||||
method,
|
||||
cache: 'no-cache',
|
||||
headers: {
|
||||
Accept: 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
...headers,
|
||||
},
|
||||
body,
|
||||
});
|
||||
return res.json();
|
||||
}
|
||||
|
||||
export function httpGet(url: string, params: object = {}, headers: object = {}) {
|
||||
return request('GET', buildUrl(url, params), undefined, headers);
|
||||
}
|
||||
|
||||
export function httpDelete(url: string, params: object = {}, headers: object = {}) {
|
||||
return request('DELETE', buildUrl(url, params), undefined, headers);
|
||||
}
|
||||
|
||||
export function httpPost(url: string, params: object = {}, headers: object = {}) {
|
||||
return request('POST', url, JSON.stringify(params), headers);
|
||||
}
|
||||
|
||||
export function httpPut(url: string, params: object = {}, headers: object = {}) {
|
||||
return request('PUT', url, JSON.stringify(params), headers);
|
||||
}
|
||||
36
src/lib/jwt.ts
Normal file
36
src/lib/jwt.ts
Normal file
|
|
@ -0,0 +1,36 @@
|
|||
import jwt from 'jsonwebtoken';
|
||||
import { decrypt, encrypt } from 'lib/crypto';
|
||||
|
||||
export function createToken(payload: any, secret: any, options?: any) {
|
||||
return jwt.sign(payload, secret, options);
|
||||
}
|
||||
|
||||
export function parseToken(token: string, secret: any) {
|
||||
try {
|
||||
return jwt.verify(token, secret);
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export function createSecureToken(payload: any, secret: any, options?: any) {
|
||||
return encrypt(createToken(payload, secret, options), secret);
|
||||
}
|
||||
|
||||
export function parseSecureToken(token: string, secret: any) {
|
||||
try {
|
||||
return jwt.verify(decrypt(token, secret), secret);
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export async function parseAuthToken(req: Request, secret: string) {
|
||||
try {
|
||||
const token = req.headers.get('authorization')?.split(' ')?.[1];
|
||||
|
||||
return parseSecureToken(token as string, secret);
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
|
@ -1,9 +1,9 @@
|
|||
import { ZodObject } from 'zod';
|
||||
import { FILTER_COLUMNS } from 'lib/constants';
|
||||
import { badRequest, unauthorized } from 'lib/response';
|
||||
import { getAllowedUnits, getMinimumUnit } from './date';
|
||||
import { getWebsiteDateRange } from '../queries';
|
||||
import { getAllowedUnits, getMinimumUnit } from 'lib/date';
|
||||
import { checkAuth } from 'lib/auth';
|
||||
import { getWebsiteDateRange } from 'queries';
|
||||
|
||||
export async function getJsonBody(request: Request) {
|
||||
try {
|
||||
|
|
|
|||
21
src/lib/storage.ts
Normal file
21
src/lib/storage.ts
Normal file
|
|
@ -0,0 +1,21 @@
|
|||
export function setItem(key: string, data: any, session?: boolean): void {
|
||||
if (typeof window !== 'undefined' && data) {
|
||||
return (session ? sessionStorage : localStorage).setItem(key, JSON.stringify(data));
|
||||
}
|
||||
}
|
||||
|
||||
export function getItem(key: string, session?: boolean): any {
|
||||
if (typeof window !== 'undefined') {
|
||||
const value = (session ? sessionStorage : localStorage).getItem(key);
|
||||
|
||||
if (value !== 'undefined' && value !== null) {
|
||||
return JSON.parse(value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export function removeItem(key: string, session?: boolean): void {
|
||||
if (typeof window !== 'undefined') {
|
||||
return (session ? sessionStorage : localStorage).removeItem(key);
|
||||
}
|
||||
}
|
||||
40
src/lib/url.ts
Normal file
40
src/lib/url.ts
Normal file
|
|
@ -0,0 +1,40 @@
|
|||
export function getQueryString(params: object = {}): string {
|
||||
const searchParams = new URLSearchParams();
|
||||
|
||||
Object.entries(params).forEach(([key, value]) => {
|
||||
if (value !== undefined) {
|
||||
searchParams.append(key, value);
|
||||
}
|
||||
});
|
||||
|
||||
return searchParams.toString();
|
||||
}
|
||||
|
||||
export function buildUrl(url: string, params: object = {}): string {
|
||||
const queryString = getQueryString(params);
|
||||
return `${url}${queryString && '?' + queryString}`;
|
||||
}
|
||||
|
||||
export function safeDecodeURI(s: string | undefined | null): string | undefined | null {
|
||||
if (s === undefined || s === null) {
|
||||
return s;
|
||||
}
|
||||
|
||||
try {
|
||||
return decodeURI(s);
|
||||
} catch (e) {
|
||||
return s;
|
||||
}
|
||||
}
|
||||
|
||||
export function safeDecodeURIComponent(s: string | undefined | null): string | undefined | null {
|
||||
if (s === undefined || s === null) {
|
||||
return s;
|
||||
}
|
||||
|
||||
try {
|
||||
return decodeURIComponent(s);
|
||||
} catch (e) {
|
||||
return s;
|
||||
}
|
||||
}
|
||||
46
src/lib/utils.ts
Normal file
46
src/lib/utils.ts
Normal file
|
|
@ -0,0 +1,46 @@
|
|||
export function hook(
|
||||
_this: { [x: string]: any },
|
||||
method: string | number,
|
||||
callback: (arg0: any) => void,
|
||||
) {
|
||||
const orig = _this[method];
|
||||
|
||||
return (...args: any) => {
|
||||
callback.apply(_this, args);
|
||||
|
||||
return orig.apply(_this, args);
|
||||
};
|
||||
}
|
||||
|
||||
export function sleep(ms: number | undefined) {
|
||||
return new Promise(resolve => setTimeout(resolve, ms));
|
||||
}
|
||||
|
||||
export function shuffleArray(a) {
|
||||
const arr = a.slice();
|
||||
for (let i = arr.length - 1; i > 0; i--) {
|
||||
const j = Math.floor(Math.random() * (i + 1));
|
||||
const temp = arr[i];
|
||||
arr[i] = arr[j];
|
||||
arr[j] = temp;
|
||||
}
|
||||
return arr;
|
||||
}
|
||||
|
||||
export function chunkArray(arr: any[], size: number) {
|
||||
const chunks: any[] = [];
|
||||
|
||||
let index = 0;
|
||||
while (index < arr.length) {
|
||||
chunks.push(arr.slice(index, size + index));
|
||||
index += size;
|
||||
}
|
||||
|
||||
return chunks;
|
||||
}
|
||||
|
||||
export function ensureArray(arr?: any) {
|
||||
if (arr === undefined || arr === null) return [];
|
||||
if (Array.isArray(arr)) return arr;
|
||||
return [arr];
|
||||
}
|
||||
|
|
@ -1,7 +1,6 @@
|
|||
import clickhouse from 'lib/clickhouse';
|
||||
import { CLICKHOUSE, PRISMA, runQuery } from 'lib/db';
|
||||
import prisma from 'lib/prisma';
|
||||
import { safeDecodeURIComponent } from 'next-basics';
|
||||
|
||||
export async function getUTM(
|
||||
...args: [
|
||||
|
|
@ -84,7 +83,7 @@ function parseParameters(data: any[]) {
|
|||
|
||||
for (const [key, value] of searchParams) {
|
||||
if (key.match(/^utm_(\w+)$/)) {
|
||||
const name = safeDecodeURIComponent(value);
|
||||
const name = value;
|
||||
if (!obj[key]) {
|
||||
obj[key] = { [name]: Number(num) };
|
||||
} else if (!obj[key][name]) {
|
||||
|
|
|
|||
|
|
@ -2,7 +2,7 @@ import { Prisma } from '@prisma/client';
|
|||
import { ROLES } from 'lib/constants';
|
||||
import prisma from 'lib/prisma';
|
||||
import { PageResult, Role, User, PageParams } from 'lib/types';
|
||||
import { getRandomChars } from 'next-basics';
|
||||
import { getRandomChars } from 'lib/crypto';
|
||||
import UserFindManyArgs = Prisma.UserFindManyArgs;
|
||||
|
||||
export interface GetUserOptions {
|
||||
|
|
|
|||
|
|
@ -8,7 +8,7 @@ import {
|
|||
THEME_CONFIG,
|
||||
TIMEZONE_CONFIG,
|
||||
} from 'lib/constants';
|
||||
import { getItem } from 'next-basics';
|
||||
import { getItem } from 'lib/storage';
|
||||
import { getTimezone } from 'lib/date';
|
||||
|
||||
function getDefaultTheme() {
|
||||
|
|
|
|||
|
|
@ -1,6 +1,6 @@
|
|||
import { create } from 'zustand';
|
||||
import { DASHBOARD_CONFIG, DEFAULT_WEBSITE_LIMIT } from 'lib/constants';
|
||||
import { getItem, setItem } from 'next-basics';
|
||||
import { getItem, setItem } from 'lib/storage';
|
||||
|
||||
export const initialState = {
|
||||
showCharts: true,
|
||||
|
|
|
|||
|
|
@ -2,7 +2,7 @@ import { create } from 'zustand';
|
|||
import { produce } from 'immer';
|
||||
import semver from 'semver';
|
||||
import { CURRENT_VERSION, VERSION_CHECK, UPDATES_URL } from 'lib/constants';
|
||||
import { getItem } from 'next-basics';
|
||||
import { getItem } from 'lib/storage';
|
||||
|
||||
const initialState = {
|
||||
current: CURRENT_VERSION,
|
||||
|
|
|
|||
18
yarn.lock
18
yarn.lock
|
|
@ -7411,7 +7411,7 @@ jsonify@^0.0.1:
|
|||
resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.1.tgz#2aa3111dae3d34a0f151c63f3a45d995d9420978"
|
||||
integrity sha512-2/Ki0GcmuqSrgFyelQq9M05y7PS0mEwuIzrf3f1fPqkVDVRvZrPZtVSMHxdgo8Aq0sxAOb/cr2aqqA3LeWHVPg==
|
||||
|
||||
jsonwebtoken@^9.0.0:
|
||||
jsonwebtoken@^9.0.2:
|
||||
version "9.0.2"
|
||||
resolved "https://registry.yarnpkg.com/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz#65ff91f4abef1784697d40952bb1998c504caaf3"
|
||||
integrity sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==
|
||||
|
|
@ -8054,15 +8054,6 @@ natural-compare@^1.4.0:
|
|||
resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7"
|
||||
integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==
|
||||
|
||||
next-basics@^0.39.0:
|
||||
version "0.39.0"
|
||||
resolved "https://registry.yarnpkg.com/next-basics/-/next-basics-0.39.0.tgz#1ec448a1c12966a82067445bfb9319b7e883dd6a"
|
||||
integrity sha512-5HWf3u7jgx5n4auIkArFP5+EVdyz7kSvxs86o2V4y8/t3J4scdIHgI8BBE6UhzB17WMbMgVql44IfcJH1CQc/w==
|
||||
dependencies:
|
||||
bcryptjs "^2.4.3"
|
||||
jsonwebtoken "^9.0.0"
|
||||
pure-rand "^6.0.2"
|
||||
|
||||
next@15.0.4:
|
||||
version "15.0.4"
|
||||
resolved "https://registry.yarnpkg.com/next/-/next-15.0.4.tgz#7ddad7299204f16c132d7e524cf903f1a513588e"
|
||||
|
|
@ -9179,11 +9170,16 @@ punycode@^2.1.0:
|
|||
resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5"
|
||||
integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==
|
||||
|
||||
pure-rand@^6.0.0, pure-rand@^6.0.2:
|
||||
pure-rand@^6.0.0:
|
||||
version "6.0.4"
|
||||
resolved "https://registry.yarnpkg.com/pure-rand/-/pure-rand-6.0.4.tgz#50b737f6a925468679bff00ad20eade53f37d5c7"
|
||||
integrity sha512-LA0Y9kxMYv47GIPJy6MI84fqTd2HmYZI83W/kM/SkKfDlajnZYfmXFTxkbY+xSBPkLJxltMa9hIkmdc29eguMA==
|
||||
|
||||
pure-rand@^6.1.0:
|
||||
version "6.1.0"
|
||||
resolved "https://registry.yarnpkg.com/pure-rand/-/pure-rand-6.1.0.tgz#d173cf23258231976ccbdb05247c9787957604f2"
|
||||
integrity sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==
|
||||
|
||||
qs@6.13.0:
|
||||
version "6.13.0"
|
||||
resolved "https://registry.yarnpkg.com/qs/-/qs-6.13.0.tgz#6ca3bd58439f7e245655798997787b0d88a51906"
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue