new version with admin

This commit is contained in:
user 2025-09-26 12:21:27 +03:00
parent 0f983eb34d
commit d08a02b6e7
10 changed files with 2357 additions and 186 deletions

184
package-lock.json generated
View File

@ -16,7 +16,6 @@
"axios": "^1.6.7",
"buffer": "^6.0.3",
"clsx": "^2.1.0",
"jssha": "^3.3.1",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-hook-form": "^7.51.0",
@ -25,6 +24,7 @@
"react-router-dom": "^6.22.2",
"react-tag-input": "^6.10.3",
"tailwind-merge": "^2.2.1",
"tus-js-client": "^4.3.1",
"zod": "^3.22.4",
"zustand": "^4.5.2"
},
@ -2298,6 +2298,12 @@
"ieee754": "^1.2.1"
}
},
"node_modules/buffer-from": {
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
"integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
"license": "MIT"
},
"node_modules/call-bind": {
"version": "1.0.7",
"resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz",
@ -2434,6 +2440,15 @@
"integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
"dev": true
},
"node_modules/combine-errors": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/combine-errors/-/combine-errors-3.0.3.tgz",
"integrity": "sha512-C8ikRNRMygCwaTx+Ek3Yr+OuZzgZjduCOfSQBjbM8V3MfgcjSTeto/GXP6PAwKvJz/v15b7GHZvx5rOlczFw/Q==",
"dependencies": {
"custom-error-instance": "2.1.1",
"lodash.uniqby": "4.5.0"
}
},
"node_modules/combined-stream": {
"version": "1.0.8",
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
@ -2496,6 +2511,12 @@
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz",
"integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw=="
},
"node_modules/custom-error-instance": {
"version": "2.1.1",
"resolved": "https://registry.npmjs.org/custom-error-instance/-/custom-error-instance-2.1.1.tgz",
"integrity": "sha512-p6JFxJc3M4OTD2li2qaHkDCw9SfMw82Ldr6OC9Je1aXiGfhx2W8p3GaoeaGrPJTUN9NirTM/KTxHWMUdR1rsUg==",
"license": "ISC"
},
"node_modules/debug": {
"version": "4.3.4",
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
@ -3683,6 +3704,12 @@
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/graceful-fs": {
"version": "4.2.11",
"resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz",
"integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==",
"license": "ISC"
},
"node_modules/graphemer": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz",
@ -4062,6 +4089,18 @@
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/is-stream": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz",
"integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==",
"license": "MIT",
"engines": {
"node": ">=8"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/is-string": {
"version": "1.0.7",
"resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz",
@ -4158,6 +4197,12 @@
"jiti": "bin/jiti.js"
}
},
"node_modules/js-base64": {
"version": "3.7.8",
"resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.8.tgz",
"integrity": "sha512-hNngCeKxIUQiEUN3GPJOkz4wF/YvdUdbNL9hsBcMQTkKzboD7T/q3OYOuuPZLUE6dBxSGpwhk5mwuDud7JVAow==",
"license": "BSD-3-Clause"
},
"node_modules/js-sha3": {
"version": "0.8.0",
"resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.8.0.tgz",
@ -4222,15 +4267,6 @@
"node": ">=6"
}
},
"node_modules/jssha": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/jssha/-/jssha-3.3.1.tgz",
"integrity": "sha512-VCMZj12FCFMQYcFLPRm/0lOBbLi8uM2BhXPTqw3U4YAfs4AZfiApOoBLoN8cQE60Z50m1MYMTQVCfgF/KaCVhQ==",
"license": "BSD-3-Clause",
"engines": {
"node": "*"
}
},
"node_modules/keyv": {
"version": "4.5.4",
"resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
@ -4294,12 +4330,74 @@
"integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==",
"license": "MIT"
},
"node_modules/lodash._baseiteratee": {
"version": "4.7.0",
"resolved": "https://registry.npmjs.org/lodash._baseiteratee/-/lodash._baseiteratee-4.7.0.tgz",
"integrity": "sha512-nqB9M+wITz0BX/Q2xg6fQ8mLkyfF7MU7eE+MNBNjTHFKeKaZAPEzEg+E8LWxKWf1DQVflNEn9N49yAuqKh2mWQ==",
"license": "MIT",
"dependencies": {
"lodash._stringtopath": "~4.8.0"
}
},
"node_modules/lodash._basetostring": {
"version": "4.12.0",
"resolved": "https://registry.npmjs.org/lodash._basetostring/-/lodash._basetostring-4.12.0.tgz",
"integrity": "sha512-SwcRIbyxnN6CFEEK4K1y+zuApvWdpQdBHM/swxP962s8HIxPO3alBH5t3m/dl+f4CMUug6sJb7Pww8d13/9WSw==",
"license": "MIT"
},
"node_modules/lodash._baseuniq": {
"version": "4.6.0",
"resolved": "https://registry.npmjs.org/lodash._baseuniq/-/lodash._baseuniq-4.6.0.tgz",
"integrity": "sha512-Ja1YevpHZctlI5beLA7oc5KNDhGcPixFhcqSiORHNsp/1QTv7amAXzw+gu4YOvErqVlMVyIJGgtzeepCnnur0A==",
"license": "MIT",
"dependencies": {
"lodash._createset": "~4.0.0",
"lodash._root": "~3.0.0"
}
},
"node_modules/lodash._createset": {
"version": "4.0.3",
"resolved": "https://registry.npmjs.org/lodash._createset/-/lodash._createset-4.0.3.tgz",
"integrity": "sha512-GTkC6YMprrJZCYU3zcqZj+jkXkrXzq3IPBcF/fIPpNEAB4hZEtXU8zp/RwKOvZl43NUmwDbyRk3+ZTbeRdEBXA==",
"license": "MIT"
},
"node_modules/lodash._root": {
"version": "3.0.1",
"resolved": "https://registry.npmjs.org/lodash._root/-/lodash._root-3.0.1.tgz",
"integrity": "sha512-O0pWuFSK6x4EXhM1dhZ8gchNtG7JMqBtrHdoUFUWXD7dJnNSUze1GuyQr5sOs0aCvgGeI3o/OJW8f4ca7FDxmQ==",
"license": "MIT"
},
"node_modules/lodash._stringtopath": {
"version": "4.8.0",
"resolved": "https://registry.npmjs.org/lodash._stringtopath/-/lodash._stringtopath-4.8.0.tgz",
"integrity": "sha512-SXL66C731p0xPDC5LZg4wI5H+dJo/EO4KTqOMwLYCH3+FmmfAKJEZCm6ohGpI+T1xwsDsJCfL4OnhorllvlTPQ==",
"license": "MIT",
"dependencies": {
"lodash._basetostring": "~4.12.0"
}
},
"node_modules/lodash.merge": {
"version": "4.6.2",
"resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
"integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
"dev": true
},
"node_modules/lodash.throttle": {
"version": "4.1.1",
"resolved": "https://registry.npmjs.org/lodash.throttle/-/lodash.throttle-4.1.1.tgz",
"integrity": "sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==",
"license": "MIT"
},
"node_modules/lodash.uniqby": {
"version": "4.5.0",
"resolved": "https://registry.npmjs.org/lodash.uniqby/-/lodash.uniqby-4.5.0.tgz",
"integrity": "sha512-IRt7cfTtHy6f1aRVA5n7kT8rgN3N1nH6MOWLcHfpWG2SH19E3JksLK38MktLxZDhlAjCP9jpIXkOnRXlu6oByQ==",
"license": "MIT",
"dependencies": {
"lodash._baseiteratee": "~4.7.0",
"lodash._baseuniq": "~4.6.0"
}
},
"node_modules/loose-envify": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz",
@ -5053,6 +5151,23 @@
"react-is": "^16.13.1"
}
},
"node_modules/proper-lockfile": {
"version": "4.1.2",
"resolved": "https://registry.npmjs.org/proper-lockfile/-/proper-lockfile-4.1.2.tgz",
"integrity": "sha512-TjNPblN4BwAWMXU8s9AEz4JmQxnD1NNL7bNOY/AKUzyamc379FWASUhc/K1pL2noVb+XmZKLL68cjzLsiOAMaA==",
"license": "MIT",
"dependencies": {
"graceful-fs": "^4.2.4",
"retry": "^0.12.0",
"signal-exit": "^3.0.2"
}
},
"node_modules/proper-lockfile/node_modules/signal-exit": {
"version": "3.0.7",
"resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz",
"integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==",
"license": "ISC"
},
"node_modules/proxy-from-env": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
@ -5067,6 +5182,12 @@
"node": ">=6"
}
},
"node_modules/querystringify": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz",
"integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==",
"license": "MIT"
},
"node_modules/queue-microtask": {
"version": "1.2.3",
"resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
@ -5341,6 +5462,12 @@
"resolved": "https://registry.npmjs.org/remove-accents/-/remove-accents-0.5.0.tgz",
"integrity": "sha512-8g3/Otx1eJaVD12e31UbJj1YzdtVvzH85HV7t+9MJYk/u3XmkOUJ5Ys9wQrf9PCPK8+xn4ymzqYCiZl6QWKn+A=="
},
"node_modules/requires-port": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz",
"integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==",
"license": "MIT"
},
"node_modules/resolve": {
"version": "1.22.8",
"resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz",
@ -5367,6 +5494,15 @@
"node": ">=4"
}
},
"node_modules/retry": {
"version": "0.12.0",
"resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz",
"integrity": "sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==",
"license": "MIT",
"engines": {
"node": ">= 4"
}
},
"node_modules/reusify": {
"version": "1.0.4",
"resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz",
@ -6040,6 +6176,24 @@
"integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==",
"dev": true
},
"node_modules/tus-js-client": {
"version": "4.3.1",
"resolved": "https://registry.npmjs.org/tus-js-client/-/tus-js-client-4.3.1.tgz",
"integrity": "sha512-ZLeYmjrkaU1fUsKbIi8JML52uAocjEZtBx4DKjRrqzrZa0O4MYwT6db+oqePlspV+FxXJAyFBc/L5gwUi2OFsg==",
"license": "MIT",
"dependencies": {
"buffer-from": "^1.1.2",
"combine-errors": "^3.0.3",
"is-stream": "^2.0.0",
"js-base64": "^3.7.2",
"lodash.throttle": "^4.1.1",
"proper-lockfile": "^4.1.2",
"url-parse": "^1.5.7"
},
"engines": {
"node": ">=18"
}
},
"node_modules/tweetnacl": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-1.0.3.tgz",
@ -6257,6 +6411,16 @@
"punycode": "^2.1.0"
}
},
"node_modules/url-parse": {
"version": "1.5.10",
"resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz",
"integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==",
"license": "MIT",
"dependencies": {
"querystringify": "^2.1.1",
"requires-port": "^1.0.0"
}
},
"node_modules/use-sync-external-store": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.0.tgz",

View File

@ -26,6 +26,7 @@
"react-router-dom": "^6.22.2",
"react-tag-input": "^6.10.3",
"tailwind-merge": "^2.2.1",
"tus-js-client": "^4.3.1",
"zod": "^3.22.4",
"zustand": "^4.5.2"
},

View File

@ -2,4 +2,5 @@ export const Routes = {
Root: "/uploadContent",
ViewContent: "/viewContent",
SentryCheck: "/sentryCheck",
Admin: "/admin",
};

View File

@ -3,6 +3,7 @@ import { createBrowserRouter, RouterProvider } from "react-router-dom";
import { Routes } from "~/app/router/constants";
import { RootPage } from "~/pages/root";
import { ViewContentPage } from "~/pages/view-content";
import { AdminPage } from "~/pages/admin";
import { ProtectedLayout } from "./protected-layout";
const router = createBrowserRouter([
@ -11,8 +12,8 @@ const router = createBrowserRouter([
children: [
{ path: Routes.Root, element: <RootPage /> },
{ path: Routes.ViewContent, element: <ViewContentPage /> },
{
path: Routes.SentryCheck,
{
path: Routes.SentryCheck,
element: (
<div className="flex h-screen items-center justify-center">
<button
@ -29,6 +30,7 @@ const router = createBrowserRouter([
},
],
},
{ path: Routes.Admin, element: <AdminPage /> },
]);
export const AppRouter = () => {

1085
src/pages/admin/index.tsx Normal file

File diff suppressed because it is too large Load Diff

View File

@ -8,7 +8,7 @@ import ReactPlayer from "react-player/lazy";
import { Button } from "~/shared/ui/button";
import { useRootStore } from "~/shared/stores/root";
import { FormLabel } from "~/shared/ui/form-label";
import { useUploadFile } from "~/shared/services/file";
import { useLegacyUploadFile, useTusUpload } from "~/shared/services/file";
import { Progress } from "~/shared/ui/progress";
import { useCreateNewContent } from "~/shared/services/content";
import { BackButton } from "~/shared/ui/back-button";
@ -31,8 +31,8 @@ export const PresubmitStep = ({ prevStep }: PresubmitStepProps) => {
const [isCoverExpanded, setCoverExpanded] = useState(false);
const uploadCover = useUploadFile();
const uploadFile = useUploadFile();
const uploadCover = useLegacyUploadFile();
const uploadFile = useTusUpload();
const createContent = useCreateNewContent();
@ -54,9 +54,21 @@ export const PresubmitStep = ({ prevStep }: PresubmitStepProps) => {
try {
let coverUploadResult = { content_id_v1: "" };
const fileUploadResult = await uploadFile.mutateAsync(
rootStore.file as File,
);
const fileUploadResult = await uploadFile.mutateAsync({
file: rootStore.file as File,
metadata: {
title: rootStore.name,
description: "",
content_type: rootStore.file?.type || "application/octet-stream",
preview_start_ms: 0,
preview_duration_ms: 30000,
downloadable: rootStore.allowDwnld ? "1" : "0",
},
});
if (!fileUploadResult.encryptedCid) {
throw new Error("Tus upload did not return encrypted cid");
}
if (rootStore.allowCover && rootStore.cover) {
coverUploadResult = await uploadCover.mutateAsync(
@ -76,7 +88,7 @@ export const PresubmitStep = ({ prevStep }: PresubmitStepProps) => {
// следует отметить что вы должны еще откомментровать AuthorsStep в RootPage
// authors: rootStore.authors,
downloadable: rootStore.allowDwnld,
content: fileUploadResult.content_id_v1,
content: fileUploadResult.encryptedCid,
image: coverUploadResult.content_id_v1,
price: String(rootStore.price * 10 ** 9),
hashtags: rootStore.hashtags,

View File

@ -1,17 +1,102 @@
import axios from 'axios';
import axios, { AxiosError, InternalAxiosRequestConfig } from 'axios';
export const APP_API_BASE_URL = import.meta.env.VITE_API_BASE_URL;
import { getAdminAuthSnapshot } from '~/shared/libs/admin-auth';
export const request = axios.create({
baseURL: APP_API_BASE_URL,
const API_BASE_PATH = '/api/v1';
const API_ENDPOINTS = [
'https://my-public-node-8.projscale.dev',
'http://localhost:3000',
] as const;
const API_BASE_URLS = API_ENDPOINTS.map((endpoint) => {
return `${endpoint.replace(/\/$/, '')}${API_BASE_PATH}`;
});
request.interceptors.request.use((config) => {
const auth_v1_token = localStorage.getItem('auth_v1_token');
const ENDPOINT_INDEX_KEY = '__apiEndpointIndex__';
if (auth_v1_token) {
config.headers.Authorization = auth_v1_token;
const isBrowser = typeof window !== 'undefined';
const readStorage = (key: string) => {
if (!isBrowser) {
return null;
}
try {
return window.localStorage.getItem(key);
} catch (error) {
console.error('Failed to access localStorage', error);
return null;
}
};
type RequestConfig = InternalAxiosRequestConfig & {
[ENDPOINT_INDEX_KEY]?: number;
};
export const request = axios.create({ withCredentials: true });
request.interceptors.request.use((config: RequestConfig) => {
config.headers = config.headers ?? {};
const headers = config.headers as Record<string, any>;
const authToken = readStorage('auth_v1_token');
if (authToken && !headers.Authorization) {
headers.Authorization = authToken;
}
const urlPath = config.url ?? '';
if (urlPath.startsWith('/admin')) {
const { token, headerName } = getAdminAuthSnapshot();
if (token) {
const headerKey = headerName || 'X-Admin-Token';
if (!headers[headerKey]) {
headers[headerKey] = token;
}
}
}
const hasCustomBaseUrl = Boolean(config.baseURL && !API_BASE_URLS.includes(config.baseURL));
if (hasCustomBaseUrl) {
return config;
}
const attempt = typeof config[ENDPOINT_INDEX_KEY] === 'number' ? (config[ENDPOINT_INDEX_KEY] as number) : 0;
config[ENDPOINT_INDEX_KEY] = attempt;
config.baseURL = API_BASE_URLS[attempt] ?? API_BASE_URLS[0];
return config;
});
request.interceptors.response.use(
(response) => response,
async (error: AxiosError) => {
const config = error.config as RequestConfig | undefined;
if (!config) {
return Promise.reject(error);
}
const hasCustomBaseUrl = Boolean(config.baseURL && !API_BASE_URLS.includes(config.baseURL));
if (hasCustomBaseUrl) {
return Promise.reject(error);
}
const isAbortError = error.code === 'ERR_CANCELED' || error.message === 'canceled';
const isNetworkError = !error.response && !isAbortError;
if (!isNetworkError) {
return Promise.reject(error);
}
const currentIndex = typeof config[ENDPOINT_INDEX_KEY] === 'number' ? (config[ENDPOINT_INDEX_KEY] as number) : 0;
const nextIndex = currentIndex + 1;
if (nextIndex >= API_BASE_URLS.length) {
return Promise.reject(error);
}
config[ENDPOINT_INDEX_KEY] = nextIndex;
config.baseURL = API_BASE_URLS[nextIndex];
return request(config);
},
);

View File

@ -0,0 +1,421 @@
import axios, { AxiosError } from 'axios';
import {
useMutation,
useQuery,
UseMutationOptions,
UseQueryOptions,
} from 'react-query';
import { request } from "~/shared/libs";
import { clearAdminAuth, persistAdminAuth } from "~/shared/libs/admin-auth";
export type AdminServiceState = {
name: string;
status?: string;
last_reported_seconds: number | null;
};
export type AdminOverviewResponse = {
project: {
host: string | null;
name: string;
privacy: string;
};
codebase: {
branch: string | null;
commit: string | null;
};
node: {
id: string;
service_wallet: string;
ton_master: string;
};
runtime: {
python: string;
implementation: string;
platform: string;
utc_now: string;
};
ipfs: Record<string, unknown> & {
identity?: Record<string, unknown> | { error: string };
bitswap?: Record<string, unknown> | { error: string };
repo?: Record<string, unknown> | { error: string };
};
content: {
encrypted_total: number;
upload_sessions_total: number;
derivatives_ready: number;
};
ton: {
host: string | null;
api_key_configured: boolean;
testnet: boolean;
};
services: AdminServiceState[];
};
export type AdminStorageResponse = {
directories: Array<{
label: string;
path: string;
exists: boolean;
file_count: number;
size_bytes: number;
}>;
disk: null | {
path: string;
total_bytes: number;
used_bytes: number;
free_bytes: number;
percent_used: number | null;
};
derivatives: {
ready: number;
processing: number;
pending: number;
failed: number;
total_bytes: number;
};
};
export type AdminUploadsResponse = {
total: number;
states: Record<string, number>;
recent: Array<{
id: string;
filename: string | null;
size_bytes: number | null;
state: string;
encrypted_cid: string | null;
error: string | null;
updated_at: string;
created_at: string;
}>;
};
export type AdminSystemResponse = {
env: Record<string, string | null | undefined>;
service_config: Array<{
key: string;
value: string | null;
raw: string | null;
}>;
services: AdminServiceState[];
blockchain_tasks: Record<string, number>;
latest_index_items: Array<{
encrypted_cid: string | null;
updated_at: string;
}>;
};
export type AdminBlockchainResponse = {
counts: Record<string, number>;
recent: Array<{
id: string;
destination: string | null;
amount: string | null;
status: string;
epoch: number | null;
seqno: number | null;
transaction_hash: string | null;
updated: string;
}>;
};
export type AdminNodesResponse = {
items: Array<{
ip: string | null;
port: number | null;
public_key: string | null;
role: 'trusted' | 'read-only' | 'deny';
version?: string | null;
last_seen?: string | null;
notes?: string | null;
}>;
};
export type AdminStatusResponse = {
ipfs: {
bitswap: Record<string, unknown>;
repo: Record<string, unknown>;
};
pin_counts: Record<string, number>;
derivatives: {
ready: number;
processing: number;
pending: number;
failed: number;
total_bytes: number;
};
convert_backlog: number;
limits: {
DERIVATIVE_CACHE_MAX_GB: number;
DERIVATIVE_CACHE_TTL_DAYS: number;
SYNC_MAX_CONCURRENT_PINS: number;
SYNC_DISK_LOW_WATERMARK_PCT: number;
};
};
export type AdminCacheSetLimitsPayload = {
max_gb: number;
ttl_days: number;
};
export type AdminCacheCleanupPayload =
| {
mode: 'ttl';
}
| {
mode?: 'fit';
max_gb: number;
};
export type AdminSyncSetLimitsPayload = {
max_concurrent_pins: number;
disk_low_watermark_pct: number;
};
export type AdminNodeSetRolePayload = {
public_key?: string;
host?: string;
role: 'trusted' | 'read-only' | 'deny';
};
const defaultQueryOptions = {
staleTime: 30_000,
refetchOnWindowFocus: false,
} as const;
type QueryOptions<TData, TQueryKey extends readonly unknown[]> = UseQueryOptions<
TData,
AxiosError,
TData,
TQueryKey
>;
type MutationOptions<TData, TVariables> = UseMutationOptions<
TData,
AxiosError,
TVariables
>;
export const useAdminOverview = (
options?: QueryOptions<AdminOverviewResponse, ['admin', 'overview']>,
) => {
return useQuery<AdminOverviewResponse, AxiosError, AdminOverviewResponse, ['admin', 'overview']>(
['admin', 'overview'],
async () => {
const { data } = await request.get<AdminOverviewResponse>('/admin.overview');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export const useAdminStorage = (
options?: QueryOptions<AdminStorageResponse, ['admin', 'storage']>,
) => {
return useQuery<AdminStorageResponse, AxiosError, AdminStorageResponse, ['admin', 'storage']>(
['admin', 'storage'],
async () => {
const { data } = await request.get<AdminStorageResponse>('/admin.storage');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export const useAdminUploads = (
options?: QueryOptions<AdminUploadsResponse, ['admin', 'uploads']>,
) => {
return useQuery<AdminUploadsResponse, AxiosError, AdminUploadsResponse, ['admin', 'uploads']>(
['admin', 'uploads'],
async () => {
const { data } = await request.get<AdminUploadsResponse>('/admin.uploads');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export const useAdminSystem = (
options?: QueryOptions<AdminSystemResponse, ['admin', 'system']>,
) => {
return useQuery<AdminSystemResponse, AxiosError, AdminSystemResponse, ['admin', 'system']>(
['admin', 'system'],
async () => {
const { data } = await request.get<AdminSystemResponse>('/admin.system');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export const useAdminBlockchain = (
options?: QueryOptions<AdminBlockchainResponse, ['admin', 'blockchain']>,
) => {
return useQuery<AdminBlockchainResponse, AxiosError, AdminBlockchainResponse, ['admin', 'blockchain']>(
['admin', 'blockchain'],
async () => {
const { data } = await request.get<AdminBlockchainResponse>('/admin.blockchain');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export const useAdminNodes = (
options?: QueryOptions<AdminNodesResponse, ['admin', 'nodes']>,
) => {
return useQuery<AdminNodesResponse, AxiosError, AdminNodesResponse, ['admin', 'nodes']>(
['admin', 'nodes'],
async () => {
const { data } = await request.get<AdminNodesResponse>('/admin.nodes');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export const useAdminStatus = (
options?: QueryOptions<AdminStatusResponse, ['admin', 'status']>,
) => {
return useQuery<AdminStatusResponse, AxiosError, AdminStatusResponse, ['admin', 'status']>(
['admin', 'status'],
async () => {
const { data } = await request.get<AdminStatusResponse>('/admin.status');
return data;
},
{
...defaultQueryOptions,
...options,
},
);
};
export type AdminLoginResponse = {
ok: true;
cookie_name?: string;
header_name?: string;
max_age?: number;
};
export const useAdminLogin = (
options?: MutationOptions<AdminLoginResponse, { secret: string }>,
) => {
return useMutation<AdminLoginResponse, AxiosError, { secret: string }>(
async ({ secret }) => {
const { data } = await request.post<AdminLoginResponse>('/admin.login', { secret });
persistAdminAuth({
token: secret,
headerName: data.header_name,
cookieName: data.cookie_name,
maxAge: data.max_age,
});
return data;
},
options,
);
};
export const useAdminLogout = (
options?: MutationOptions<{ ok: true }, void>,
) => {
return useMutation<{ ok: true }, AxiosError, void>(
async () => {
const { data } = await request.post<{ ok: true }>('/admin.logout');
clearAdminAuth();
return data;
},
options,
);
};
export const useAdminCacheSetLimits = (
options?: MutationOptions<{ ok: true }, AdminCacheSetLimitsPayload>,
) => {
return useMutation<{ ok: true }, AxiosError, AdminCacheSetLimitsPayload>(
async (payload) => {
const { data } = await request.post<{ ok: true }>('/admin.cache.setLimits', payload);
return data;
},
options,
);
};
export const useAdminCacheCleanup = (
options?: MutationOptions<{ ok: true; removed?: number }, AdminCacheCleanupPayload>,
) => {
return useMutation<{ ok: true; removed?: number }, AxiosError, AdminCacheCleanupPayload>(
async (payload) => {
const { data } = await request.post<{ ok: true; removed?: number }>('/admin.cache.cleanup', payload);
return data;
},
options,
);
};
export const useAdminSyncSetLimits = (
options?: MutationOptions<{ ok: true }, AdminSyncSetLimitsPayload>,
) => {
return useMutation<{ ok: true }, AxiosError, AdminSyncSetLimitsPayload>(
async (payload) => {
const { data } = await request.post<{ ok: true }>('/admin.sync.setLimits', payload);
return data;
},
options,
);
};
export const useAdminNodeSetRole = (
options?: MutationOptions<{ ok: true; node: { ip: string | null; public_key: string | null; role: string } }, AdminNodeSetRolePayload>,
) => {
return useMutation<
{ ok: true; node: { ip: string | null; public_key: string | null; role: string } },
AxiosError,
AdminNodeSetRolePayload
>(
async (payload) => {
const { data } = await request.post<{ ok: true; node: { ip: string | null; public_key: string | null; role: string } }>(
'/admin.node.setRole',
payload,
);
return data;
},
options,
);
};
export const useAdminSyncLimits = () => {
const cacheSetLimits = useAdminCacheSetLimits();
const syncSetLimits = useAdminSyncSetLimits();
return { cacheSetLimits, syncSetLimits };
};
export const isUnauthorizedError = (error: unknown) => {
if (!error) {
return false;
}
if (axios.isAxiosError(error)) {
return error.response?.status === 401;
}
return false;
};

View File

@ -1,201 +1,468 @@
import { useMutation } from "react-query";
import { useState } from "react";
import { useRef, useState } from "react";
import { Upload } from "tus-js-client";
import { request } from "~/shared/libs";
const STORAGE_API_URL = import.meta.env.VITE_API_BASE_STORAGE_URL;
const MAX_CHUNK_SIZE = 80 * 1024 * 1024; // 80 MB
const TUS_ENDPOINT = import.meta.env.VITE_TUS_ENDPOINT?.trim();
export const useUploadFile = () => {
const TUS_STATUS_POLL_INTERVAL_MS = 2000;
const TUS_STATUS_POLL_TIMEOUT_MS = 5 * 60 * 1000;
const sleep = (ms: number) => new Promise<void>((resolve) => {
setTimeout(resolve, ms);
});
const normalizeError = (error: unknown, fallbackMessage: string) => {
if (error instanceof Error) {
return error;
}
if (typeof error === "string") {
return new Error(error);
}
if (error && typeof error === "object" && "message" in error) {
const message = (error as { message?: unknown }).message;
if (typeof message === "string") {
return new Error(message);
}
}
return new Error(fallbackMessage);
};
const sanitizeMetadataKey = (key: string) => {
return key.replace(/[^A-Za-z0-9_.-]/g, "_");
};
const normalizeMetadata = (
metadata?: Record<string, string | number | boolean | undefined>,
) => {
if (!metadata) {
return {} as Record<string, string>;
}
return Object.entries(metadata).reduce((acc, [rawKey, rawValue]) => {
if (rawValue === undefined || rawValue === null) {
return acc;
}
const key = sanitizeMetadataKey(rawKey);
if (!key) {
return acc;
}
acc[key] = String(rawValue);
return acc;
}, {} as Record<string, string>);
};
const extractUploadId = (uploadUrl: string) => {
try {
const url = new URL(uploadUrl);
const parts = url.pathname.split("/").filter(Boolean);
const uploadId = parts[parts.length - 1];
if (!uploadId) {
throw new Error("Empty upload id");
}
return uploadId;
} catch (error) {
const normalized = uploadUrl.split("/").filter(Boolean);
const uploadId = normalized[normalized.length - 1];
if (!uploadId) {
throw new Error("Unable to extract upload id from tus Location");
}
return uploadId;
}
};
type UploadSessionState = "uploading" | "processing" | "pinned" | "failed";
type UploadStatusResponse = {
id: string;
state: UploadSessionState;
encrypted_cid?: string | null;
size_bytes?: number | null;
error?: string | null;
};
const pollUploadStatus = async (
uploadId: string,
signal?: AbortSignal,
): Promise<UploadStatusResponse> => {
const startedAt = Date.now();
while (true) {
if (signal?.aborted) {
throw new DOMException("Tus status polling aborted", "AbortError");
}
try {
const { data } = await request.get<UploadStatusResponse>(
`/upload.status/${uploadId}`,
{
headers: { "Cache-Control": "no-store" },
},
);
if (data.state === "failed") {
throw new Error(data.error || "Tus upload failed on server");
}
if (data.state === "pinned" && data.encrypted_cid) {
return data;
}
} catch (error) {
const maybeAxios = error as { response?: { status?: number } };
if (maybeAxios?.response?.status === 404) {
// Hook has not recorded the upload session yet; continue polling.
} else if (error instanceof DOMException && error.name === "AbortError") {
throw error;
} else {
throw normalizeError(error, "Tus status polling failed");
}
}
if (Date.now() - startedAt > TUS_STATUS_POLL_TIMEOUT_MS) {
throw new Error("Timed out waiting for tus upload finalization");
}
await sleep(TUS_STATUS_POLL_INTERVAL_MS);
}
};
type TusUploadArgs = {
file: File;
metadata?: Record<string, string | number | boolean | undefined>;
signal?: AbortSignal;
};
type TusUploadResult = {
kind: "tus";
uploadId: string;
encryptedCid: string;
sizeBytes?: number;
state: UploadSessionState;
};
export const useTusUpload = () => {
const [uploadProgress, setUploadProgress] = useState(0);
const [isUploading, setIsUploading] = useState(false);
const [uploadError, setUploadError] = useState<Error | null>(null);
const activeUploadRef = useRef<Upload | null>(null);
const mutation = useMutation<TusUploadResult, Error, TusUploadArgs>(
["upload-file", "tus"],
async ({ file, metadata, signal }) => {
if (!file) {
throw new Error("File is required for tus upload");
}
if (!TUS_ENDPOINT) {
throw new Error("Tus endpoint is not configured");
}
setIsUploading(true);
setUploadProgress(0);
setUploadError(null);
try {
const result = await new Promise<TusUploadResult>((resolve, reject) => {
const token = localStorage.getItem("auth_v1_token") ?? "";
const normalizedMeta = normalizeMetadata(metadata);
const headers: Record<string, string> = { "Cache-Control": "no-store" };
if (token) {
headers.Authorization = token;
}
const upload = new Upload(file, {
endpoint: TUS_ENDPOINT,
metadata: {
filename: file.name,
content_type: file.type || "application/octet-stream",
...normalizedMeta,
},
headers,
retryDelays: [0, 1000, 3000, 5000],
removeFingerprintOnSuccess: true,
uploadDataDuringCreation: true,
onError: (err) => {
const normalized = normalizeError(err, "Tus upload failed");
setUploadError(normalized);
setIsUploading(false);
setUploadProgress(0);
activeUploadRef.current = null;
reject(normalized);
},
onProgress: (bytesUploaded, bytesTotal) => {
const percent = bytesTotal
? Math.floor((bytesUploaded / bytesTotal) * 100)
: 0;
setUploadProgress(Math.min(99, Math.max(0, percent)));
},
onSuccess: () => {
(async () => {
try {
setUploadProgress(99);
const uploadUrl = upload.url;
if (!uploadUrl) {
throw new Error("Tus upload finished without Location header");
}
const uploadId = extractUploadId(uploadUrl);
const status = await pollUploadStatus(uploadId, signal);
const encryptedCid = status.encrypted_cid;
if (!encryptedCid) {
throw new Error("Tus upload finalized without encrypted CID");
}
setUploadProgress(100);
setIsUploading(false);
activeUploadRef.current = null;
resolve({
kind: "tus",
uploadId,
encryptedCid,
sizeBytes: status.size_bytes ?? undefined,
state: status.state,
});
} catch (statusError) {
const normalized = normalizeError(
statusError,
"Failed to finalize tus upload",
);
setUploadError(normalized);
setIsUploading(false);
activeUploadRef.current = null;
reject(normalized);
}
})();
},
});
activeUploadRef.current = upload;
if (signal) {
signal.addEventListener(
"abort",
() => {
upload.abort();
const abortError = new DOMException(
"Tus upload aborted",
"AbortError",
);
setUploadError(abortError);
setIsUploading(false);
activeUploadRef.current = null;
reject(abortError);
},
{ once: true },
);
}
upload
.findPreviousUploads()
.then((previousUploads) => {
if (previousUploads.length > 0) {
upload.resumeFromPreviousUpload(previousUploads[0]);
}
upload.start();
})
.catch((resumeError) => {
const normalized = normalizeError(
resumeError,
"Failed to resume tus upload",
);
setUploadError(normalized);
setIsUploading(false);
activeUploadRef.current = null;
reject(normalized);
});
});
return result;
} finally {
activeUploadRef.current = null;
setIsUploading(false);
}
},
);
const resetUploadError = () => setUploadError(null);
const abortUpload = () => {
if (activeUploadRef.current) {
activeUploadRef.current.abort();
activeUploadRef.current = null;
setIsUploading(false);
setUploadProgress(0);
}
};
return {
...mutation,
uploadProgress,
isUploading,
uploadError,
resetUploadError,
abortUpload,
};
};
export const useLegacyUploadFile = () => {
const [uploadProgress, setUploadProgress] = useState(0);
const [isUploading, setIsUploading] = useState(false);
const [uploadError, setUploadError] = useState<Error | null>(null);
const mutation = useMutation(["upload-file"], async (file: File) => {
console.log(`Начинаем загрузку файла: ${file.name} (${file.size} байт)`);
const mutation = useMutation(["upload-file", "legacy"], async (file: File) => {
setIsUploading(true);
setUploadProgress(0);
setUploadError(null); // Сбрасываем ошибку перед началом новой загрузки
setUploadError(null);
try {
// Для маленьких файлов используем обычную загрузку, но с теми же заголовками
if (file.size <= MAX_CHUNK_SIZE) {
console.log("Используем обычную загрузку (файл <= MAX_CHUNK_SIZE)");
// Подготавливаем заголовки - такие же, как для чанковой загрузки
const headers: Record<string, string> = {
"X-File-Name": btoa(unescape(encodeURIComponent(file.name))), // Имя файла в base64
"X-Chunk-Start": "0", // Начинаем с позиции 0
"X-File-Name": btoa(unescape(encodeURIComponent(file.name))),
"X-Chunk-Start": "0",
"Content-Type": file.type || "application/octet-stream",
"X-Last-Chunk": "1" // Это единственный и последний чанк
"X-Last-Chunk": "1",
};
// Добавляем заголовок авторизации
const authToken = localStorage.getItem('auth_v1_token');
const authToken = localStorage.getItem("auth_v1_token");
if (authToken) {
headers["Authorization"] = authToken;
}
console.log("Заголовки запроса:", headers);
try {
const response = await request.post<{
upload_id?: string;
content_sha256?: string;
content_id?: string;
content_id_v1?: string;
content_url?: string;
}>("", file, { // Отправляем файл напрямую вместо FormData
baseURL: STORAGE_API_URL,
headers,
onUploadProgress: (progressEvent) => {
const percentCompleted = Math.round(
(progressEvent.loaded * 100) / (progressEvent?.total as number || file.size)
);
setUploadProgress(Math.min(99, percentCompleted));
},
});
console.log("Ответ на обычную загрузку:", response.data);
setUploadProgress(100);
return {
content_sha256: response.data.content_sha256 || "",
content_id_v1: response.data.content_id_v1 || response.data.content_id || "",
content_url: response.data.content_url || ""
};
} catch (error) {
console.error("Ошибка при обычной загрузке:", error);
setUploadError(error instanceof Error ? error : new Error('Ошибка при загрузке файла'));
throw error;
headers.Authorization = authToken;
}
const response = await request.post<{
upload_id?: string;
content_sha256?: string;
content_id?: string;
content_id_v1?: string;
content_url?: string;
}>("", file, {
baseURL: STORAGE_API_URL,
headers,
onUploadProgress: (progressEvent) => {
const total = progressEvent?.total ?? file.size;
const percentCompleted = Math.round(
(progressEvent.loaded * 100) / total,
);
setUploadProgress(Math.min(99, percentCompleted));
},
});
setUploadProgress(100);
return {
content_sha256: response.data.content_sha256 || "",
content_id_v1:
response.data.content_id_v1 || response.data.content_id || "",
content_url: response.data.content_url || "",
};
}
// Для больших файлов используем чанковую загрузку
console.log("Используем чанкированную загрузку (файл > MAX_CHUNK_SIZE)");
let offset = 0;
let uploadId: string | null = null;
let chunkNumber = 0;
// Загружаем файл по чанкам
while (offset < file.size) {
chunkNumber++;
const chunkEnd = Math.min(offset + MAX_CHUNK_SIZE, file.size);
const chunk = file.slice(offset, chunkEnd);
console.log(`Загрузка чанка #${chunkNumber} начиная с байта ${offset}`);
// Определяем, является ли текущий чанк последним
const isLastChunk = chunkEnd === file.size;
// Подготавливаем заголовки
const headers: Record<string, string> = {
"X-File-Name": btoa(unescape(encodeURIComponent(file.name))), // Имя файла в base64
"X-File-Name": btoa(unescape(encodeURIComponent(file.name))),
"X-Chunk-Start": offset.toString(),
"Content-Type": file.type || "application/octet-stream"
"Content-Type": file.type || "application/octet-stream",
};
// Добавляем маркер последнего чанка, если это последний чанк
if (isLastChunk) {
headers["X-Last-Chunk"] = "1";
}
// Добавляем заголовок авторизации
const authToken = localStorage.getItem('auth_v1_token');
const authToken = localStorage.getItem("auth_v1_token");
if (authToken) {
headers["Authorization"] = authToken;
headers.Authorization = authToken;
}
// Если есть uploadId, добавляем его в заголовки
if (uploadId) {
headers["X-Upload-ID"] = uploadId;
}
console.log("Заголовки запроса:", headers);
try {
const response = await request.post<{
upload_id?: string;
current_size?: number;
content_id?: string;
content_sha256?: string;
content_id_v1?: string;
content_url?: string;
}>("", chunk, {
baseURL: STORAGE_API_URL,
headers,
onUploadProgress: (progressEvent) => {
// Прогресс загрузки текущего чанка
const overallProgress = offset + progressEvent.loaded;
const percentCompleted = Math.round((overallProgress / file.size) * 100);
setUploadProgress(Math.min(99, percentCompleted));
},
});
console.log(`Ответ на чанк #${chunkNumber}:`, response.data);
// Сохраняем uploadId из первого ответа, если не установлен
if (!uploadId && response.data.upload_id) {
uploadId = response.data.upload_id;
console.log("Получен upload_id:", uploadId);
}
// Проверка на наличие content_id
if (response.data.content_id) {
console.log("Загрузка завершена. ID файла:", response.data.content_id);
setUploadProgress(100);
return {
content_sha256: response.data.content_sha256 || "",
content_id_v1: response.data.content_id_v1 || response.data.content_id || "",
content_url: response.data.content_url || ""
};
}
// Обновляем смещение на основе ответа сервера
if (response.data.current_size !== undefined) {
offset = response.data.current_size;
console.log(`Сервер сообщает current_size: ${offset}`);
} else {
console.warn("Неожиданный ответ от сервера, отсутствует current_size");
const error = new Error("Missing current_size in response");
setUploadError(error);
throw error;
}
} catch (error: any) {
console.error(`Ошибка при загрузке чанка #${chunkNumber}:`, error);
if (error.response) {
console.error("Ответ сервера:", error.response.status, error.response.data);
}
setUploadError(error instanceof Error ? error : new Error(`Ошибка при загрузке чанка #${chunkNumber}`));
const response = await request.post<{
upload_id?: string;
current_size?: number;
content_id?: string;
content_sha256?: string;
content_id_v1?: string;
content_url?: string;
}>("", chunk, {
baseURL: STORAGE_API_URL,
headers,
onUploadProgress: (progressEvent) => {
const overallProgress = offset + progressEvent.loaded;
const percentCompleted = Math.round(
(overallProgress / file.size) * 100,
);
setUploadProgress(Math.min(99, percentCompleted));
},
});
if (!uploadId && response.data.upload_id) {
uploadId = response.data.upload_id;
}
if (response.data.content_id) {
setUploadProgress(100);
return {
content_sha256: response.data.content_sha256 || "",
content_id_v1:
response.data.content_id_v1 || response.data.content_id || "",
content_url: response.data.content_url || "",
};
}
if (response.data.current_size !== undefined) {
offset = response.data.current_size;
} else {
const error = new Error("Missing current_size in response");
setUploadError(error);
throw error;
}
}
const error = new Error("Ошибка загрузки файла: все чанки загружены, но content_id не получен");
console.error(error.message);
const error = new Error(
"All chunks uploaded but server did not return content_id",
);
setUploadError(error);
throw error;
} catch (error: any) {
console.error("Ошибка при загрузке:", error);
if (error.response) {
console.error("Ответ сервера:", error.response.status, error.response.data);
}
setUploadError(error instanceof Error ? error : new Error("Неизвестная ошибка при загрузке"));
} catch (error) {
const normalized = normalizeError(
error,
"Unknown error during legacy upload",
);
setUploadError(normalized);
setIsUploading(false);
throw error;
throw normalized;
} finally {
setIsUploading(false);
}
});
// Сбросить ошибку
const resetUploadError = () => setUploadError(null);
return {
...mutation,
uploadProgress,
isUploading,
return {
...mutation,
uploadProgress,
isUploading,
uploadError,
resetUploadError
resetUploadError,
};
};
};
export { useTusUpload as useUploadFile };
export type { TusUploadArgs, TusUploadResult, UploadSessionState };

155
yarn.lock
View File

@ -225,10 +225,10 @@
"@babel/helper-validator-identifier" "^7.22.20"
to-fast-properties "^2.0.0"
"@esbuild/win32-x64@0.19.12":
"@esbuild/darwin-arm64@0.19.12":
version "0.19.12"
resolved "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.19.12.tgz"
integrity sha512-T1QyPSDCyMXaO3pzBkF96E8xMkiRYbUEZADd29SyPGabqxMViNoii+NcK7eWJAEoU6RZyEm5lVSIjTmcdoB9HA==
resolved "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.19.12.tgz"
integrity sha512-B6IeSgZgtEzGC42jsI+YYu9Z3HKRxp8ZT3cqhvliEHovq8HSX2YX8lNocDn79gCKJXOSaEot9MVYky7AKjCs8g==
"@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0":
version "4.4.0"
@ -381,10 +381,10 @@
resolved "https://registry.npmjs.org/@remix-run/router/-/router-1.15.2.tgz"
integrity sha512-+Rnav+CaoTE5QJc4Jcwh5toUpnVLKYbpU6Ys0zqbakqbaLQHeglLVHPfxOiQqdNmUy5C2lXz5dwC6tQNX2JW2Q==
"@rollup/rollup-win32-x64-msvc@4.12.0":
"@rollup/rollup-darwin-arm64@4.12.0":
version "4.12.0"
resolved "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.12.0.tgz"
integrity sha512-ZYmr5mS2wd4Dew/JjT0Fqi2NPB/ZhZ2VvPp7SmvPZb4Y1CG/LRcS6tcRo2cYU7zLK5A7cdbhWnnWmUjoI4qapg==
resolved "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.12.0.tgz"
integrity sha512-X64tZd8dRE/QTrBIEs63kaOBG0b5GVEd3ccoLtyf6IdXtHdh8h+I56C2yC3PtC9Ucnv0CpNFJLqKFVgCYe0lOQ==
"@sentry-internal/browser-utils@9.1.0":
version "9.1.0"
@ -976,6 +976,11 @@ browserslist@^4.22.2, browserslist@^4.23.0, "browserslist@>= 4.21.0":
node-releases "^2.0.14"
update-browserslist-db "^1.0.13"
buffer-from@^1.1.2:
version "1.1.2"
resolved "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz"
integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==
buffer@^6.0.3:
version "6.0.3"
resolved "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz"
@ -1081,6 +1086,14 @@ color-name@1.1.3:
resolved "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz"
integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==
combine-errors@^3.0.3:
version "3.0.3"
resolved "https://registry.npmjs.org/combine-errors/-/combine-errors-3.0.3.tgz"
integrity sha512-C8ikRNRMygCwaTx+Ek3Yr+OuZzgZjduCOfSQBjbM8V3MfgcjSTeto/GXP6PAwKvJz/v15b7GHZvx5rOlczFw/Q==
dependencies:
custom-error-instance "2.1.1"
lodash.uniqby "4.5.0"
combined-stream@^1.0.8:
version "1.0.8"
resolved "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz"
@ -1122,6 +1135,11 @@ csstype@^3.0.2, csstype@^3.1.1:
resolved "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz"
integrity sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==
custom-error-instance@2.1.1:
version "2.1.1"
resolved "https://registry.npmjs.org/custom-error-instance/-/custom-error-instance-2.1.1.tgz"
integrity sha512-p6JFxJc3M4OTD2li2qaHkDCw9SfMw82Ldr6OC9Je1aXiGfhx2W8p3GaoeaGrPJTUN9NirTM/KTxHWMUdR1rsUg==
debug@^3.2.7:
version "3.2.7"
resolved "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz"
@ -1637,6 +1655,11 @@ fs.realpath@^1.0.0:
resolved "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz"
integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==
fsevents@~2.3.2, fsevents@~2.3.3:
version "2.3.3"
resolved "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz"
integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==
function-bind@^1.1.2:
version "1.1.2"
resolved "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz"
@ -1769,6 +1792,11 @@ gopd@^1.0.1:
dependencies:
get-intrinsic "^1.1.3"
graceful-fs@^4.2.4:
version "4.2.11"
resolved "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz"
integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==
graphemer@^1.4.0:
version "1.4.0"
resolved "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz"
@ -1975,6 +2003,11 @@ is-shared-array-buffer@^1.0.2, is-shared-array-buffer@^1.0.3:
dependencies:
call-bind "^1.0.7"
is-stream@^2.0.0:
version "2.0.1"
resolved "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz"
integrity sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==
is-string@^1.0.5, is-string@^1.0.7:
version "1.0.7"
resolved "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz"
@ -2027,6 +2060,11 @@ jiti@^1.19.1:
resolved "https://registry.npmjs.org/jiti/-/jiti-1.21.0.tgz"
integrity sha512-gFqAIbuKyyso/3G2qhiO2OM6shY6EPP/R0+mkDbyspxKazh8BXDC5FiFsUjlczgdNz/vfra0da2y+aHrusLG/Q==
js-base64@^3.7.2:
version "3.7.8"
resolved "https://registry.npmjs.org/js-base64/-/js-base64-3.7.8.tgz"
integrity sha512-hNngCeKxIUQiEUN3GPJOkz4wF/YvdUdbNL9hsBcMQTkKzboD7T/q3OYOuuPZLUE6dBxSGpwhk5mwuDud7JVAow==
js-sha3@0.8.0:
version "0.8.0"
resolved "https://registry.npmjs.org/js-sha3/-/js-sha3-0.8.0.tgz"
@ -2076,11 +2114,6 @@ json5@^2.2.3:
resolved "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz"
integrity sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==
jssha@^3.3.1:
version "3.3.1"
resolved "https://registry.npmjs.org/jssha/-/jssha-3.3.1.tgz"
integrity sha512-VCMZj12FCFMQYcFLPRm/0lOBbLi8uM2BhXPTqw3U4YAfs4AZfiApOoBLoN8cQE60Z50m1MYMTQVCfgF/KaCVhQ==
jssha@3.2.0:
version "3.2.0"
resolved "https://registry.npmjs.org/jssha/-/jssha-3.2.0.tgz"
@ -2133,11 +2166,61 @@ lodash-es@^4.17.21:
resolved "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz"
integrity sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==
lodash._baseiteratee@~4.7.0:
version "4.7.0"
resolved "https://registry.npmjs.org/lodash._baseiteratee/-/lodash._baseiteratee-4.7.0.tgz"
integrity sha512-nqB9M+wITz0BX/Q2xg6fQ8mLkyfF7MU7eE+MNBNjTHFKeKaZAPEzEg+E8LWxKWf1DQVflNEn9N49yAuqKh2mWQ==
dependencies:
lodash._stringtopath "~4.8.0"
lodash._basetostring@~4.12.0:
version "4.12.0"
resolved "https://registry.npmjs.org/lodash._basetostring/-/lodash._basetostring-4.12.0.tgz"
integrity sha512-SwcRIbyxnN6CFEEK4K1y+zuApvWdpQdBHM/swxP962s8HIxPO3alBH5t3m/dl+f4CMUug6sJb7Pww8d13/9WSw==
lodash._baseuniq@~4.6.0:
version "4.6.0"
resolved "https://registry.npmjs.org/lodash._baseuniq/-/lodash._baseuniq-4.6.0.tgz"
integrity sha512-Ja1YevpHZctlI5beLA7oc5KNDhGcPixFhcqSiORHNsp/1QTv7amAXzw+gu4YOvErqVlMVyIJGgtzeepCnnur0A==
dependencies:
lodash._createset "~4.0.0"
lodash._root "~3.0.0"
lodash._createset@~4.0.0:
version "4.0.3"
resolved "https://registry.npmjs.org/lodash._createset/-/lodash._createset-4.0.3.tgz"
integrity sha512-GTkC6YMprrJZCYU3zcqZj+jkXkrXzq3IPBcF/fIPpNEAB4hZEtXU8zp/RwKOvZl43NUmwDbyRk3+ZTbeRdEBXA==
lodash._root@~3.0.0:
version "3.0.1"
resolved "https://registry.npmjs.org/lodash._root/-/lodash._root-3.0.1.tgz"
integrity sha512-O0pWuFSK6x4EXhM1dhZ8gchNtG7JMqBtrHdoUFUWXD7dJnNSUze1GuyQr5sOs0aCvgGeI3o/OJW8f4ca7FDxmQ==
lodash._stringtopath@~4.8.0:
version "4.8.0"
resolved "https://registry.npmjs.org/lodash._stringtopath/-/lodash._stringtopath-4.8.0.tgz"
integrity sha512-SXL66C731p0xPDC5LZg4wI5H+dJo/EO4KTqOMwLYCH3+FmmfAKJEZCm6ohGpI+T1xwsDsJCfL4OnhorllvlTPQ==
dependencies:
lodash._basetostring "~4.12.0"
lodash.merge@^4.6.2:
version "4.6.2"
resolved "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz"
integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==
lodash.throttle@^4.1.1:
version "4.1.1"
resolved "https://registry.npmjs.org/lodash.throttle/-/lodash.throttle-4.1.1.tgz"
integrity sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==
lodash.uniqby@4.5.0:
version "4.5.0"
resolved "https://registry.npmjs.org/lodash.uniqby/-/lodash.uniqby-4.5.0.tgz"
integrity sha512-IRt7cfTtHy6f1aRVA5n7kT8rgN3N1nH6MOWLcHfpWG2SH19E3JksLK38MktLxZDhlAjCP9jpIXkOnRXlu6oByQ==
dependencies:
lodash._baseiteratee "~4.7.0"
lodash._baseuniq "~4.6.0"
loose-envify@^1.1.0, loose-envify@^1.4.0:
version "1.4.0"
resolved "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz"
@ -2544,6 +2627,15 @@ prop-types@^15.7.2:
object-assign "^4.1.1"
react-is "^16.13.1"
proper-lockfile@^4.1.2:
version "4.1.2"
resolved "https://registry.npmjs.org/proper-lockfile/-/proper-lockfile-4.1.2.tgz"
integrity sha512-TjNPblN4BwAWMXU8s9AEz4JmQxnD1NNL7bNOY/AKUzyamc379FWASUhc/K1pL2noVb+XmZKLL68cjzLsiOAMaA==
dependencies:
graceful-fs "^4.2.4"
retry "^0.12.0"
signal-exit "^3.0.2"
proxy-from-env@^1.1.0:
version "1.1.0"
resolved "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz"
@ -2554,6 +2646,11 @@ punycode@^2.1.0:
resolved "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz"
integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==
querystringify@^2.1.1:
version "2.2.0"
resolved "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz"
integrity sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==
queue-microtask@^1.2.2:
version "1.2.3"
resolved "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz"
@ -2697,6 +2794,11 @@ remove-accents@0.5.0:
resolved "https://registry.npmjs.org/remove-accents/-/remove-accents-0.5.0.tgz"
integrity sha512-8g3/Otx1eJaVD12e31UbJj1YzdtVvzH85HV7t+9MJYk/u3XmkOUJ5Ys9wQrf9PCPK8+xn4ymzqYCiZl6QWKn+A==
requires-port@^1.0.0:
version "1.0.0"
resolved "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz"
integrity sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==
resolve-from@^4.0.0:
version "4.0.0"
resolved "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz"
@ -2711,6 +2813,11 @@ resolve@^1.1.7, resolve@^1.22.2, resolve@^1.22.4:
path-parse "^1.0.7"
supports-preserve-symlinks-flag "^1.0.0"
retry@^0.12.0:
version "0.12.0"
resolved "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz"
integrity sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==
reusify@^1.0.4:
version "1.0.4"
resolved "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz"
@ -2834,6 +2941,11 @@ side-channel@^1.0.4:
get-intrinsic "^1.2.4"
object-inspect "^1.13.1"
signal-exit@^3.0.2:
version "3.0.7"
resolved "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz"
integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==
signal-exit@^4.0.1:
version "4.1.0"
resolved "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz"
@ -3080,6 +3192,19 @@ tslib@^2.6.2:
resolved "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz"
integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==
tus-js-client@^4.3.1:
version "4.3.1"
resolved "https://registry.npmjs.org/tus-js-client/-/tus-js-client-4.3.1.tgz"
integrity sha512-ZLeYmjrkaU1fUsKbIi8JML52uAocjEZtBx4DKjRrqzrZa0O4MYwT6db+oqePlspV+FxXJAyFBc/L5gwUi2OFsg==
dependencies:
buffer-from "^1.1.2"
combine-errors "^3.0.3"
is-stream "^2.0.0"
js-base64 "^3.7.2"
lodash.throttle "^4.1.1"
proper-lockfile "^4.1.2"
url-parse "^1.5.7"
tweetnacl-util@^0.15.1:
version "0.15.1"
resolved "https://registry.npmjs.org/tweetnacl-util/-/tweetnacl-util-0.15.1.tgz"
@ -3194,6 +3319,14 @@ uri-js@^4.2.2:
dependencies:
punycode "^2.1.0"
url-parse@^1.5.7:
version "1.5.10"
resolved "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz"
integrity sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==
dependencies:
querystringify "^2.1.1"
requires-port "^1.0.0"
use-sync-external-store@1.2.0:
version "1.2.0"
resolved "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.0.tgz"