cross-seed 6.13.6 → 7.0.0-2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +12 -13
- package/dist/Result.d.ts +27 -0
- package/dist/action.d.ts +34 -0
- package/dist/action.js +2 -1
- package/dist/action.js.map +1 -1
- package/dist/arr.d.ts +31 -0
- package/dist/arr.js +107 -39
- package/dist/arr.js.map +1 -1
- package/dist/auth.d.ts +3 -0
- package/dist/auth.js +9 -6
- package/dist/auth.js.map +1 -1
- package/dist/clients/Deluge.d.ts +153 -0
- package/dist/clients/Deluge.js +6 -6
- package/dist/clients/Deluge.js.map +1 -1
- package/dist/clients/QBittorrent.d.ts +218 -0
- package/dist/clients/RTorrent.d.ts +43 -0
- package/dist/clients/RTorrent.js +6 -3
- package/dist/clients/RTorrent.js.map +1 -1
- package/dist/clients/TorrentClient.d.ts +108 -0
- package/dist/clients/TorrentClient.js +136 -67
- package/dist/clients/TorrentClient.js.map +1 -1
- package/dist/clients/Transmission.d.ts +43 -0
- package/dist/clients/Transmission.js +2 -2
- package/dist/clients/Transmission.js.map +1 -1
- package/dist/cmd.d.ts +2 -0
- package/dist/cmd.js +42 -110
- package/dist/cmd.js.map +1 -1
- package/dist/configSchema.d.ts +1 -0
- package/dist/configSchema.js +1 -666
- package/dist/configSchema.js.map +1 -1
- package/dist/configuration.d.ts +63 -0
- package/dist/configuration.js +263 -24
- package/dist/configuration.js.map +1 -1
- package/dist/constants.d.ts +108 -0
- package/dist/constants.js +2 -32
- package/dist/constants.js.map +1 -1
- package/dist/dataFiles.d.ts +8 -0
- package/dist/dataFiles.js +21 -6
- package/dist/dataFiles.js.map +1 -1
- package/dist/db.d.ts +3 -0
- package/dist/dbConfig.d.ts +4 -0
- package/dist/dbConfig.js +67 -0
- package/dist/dbConfig.js.map +1 -0
- package/dist/decide.d.ts +25 -0
- package/dist/decide.js +4 -4
- package/dist/decide.js.map +1 -1
- package/dist/diff.d.ts +1 -0
- package/dist/errors.d.ts +3 -0
- package/dist/errors.js +0 -9
- package/dist/errors.js.map +1 -1
- package/dist/indexers.d.ts +105 -0
- package/dist/indexers.js +82 -14
- package/dist/indexers.js.map +1 -1
- package/dist/inject.d.ts +2 -0
- package/dist/jobs.d.ts +29 -0
- package/dist/jobs.js +14 -9
- package/dist/jobs.js.map +1 -1
- package/dist/logger.d.ts +29 -0
- package/dist/logger.js +18 -4
- package/dist/logger.js.map +1 -1
- package/dist/migrations/00-initialSchema.d.ts +9 -0
- package/dist/migrations/01-jobs.d.ts +9 -0
- package/dist/migrations/02-timestamps.d.ts +9 -0
- package/dist/migrations/03-rateLimits.d.ts +9 -0
- package/dist/migrations/04-auth.d.ts +9 -0
- package/dist/migrations/04-auth.js +1 -1
- package/dist/migrations/04-auth.js.map +1 -1
- package/dist/migrations/05-caps.d.ts +9 -0
- package/dist/migrations/06-uniqueDecisions.d.ts +9 -0
- package/dist/migrations/07-limits.d.ts +9 -0
- package/dist/migrations/08-rss.d.ts +9 -0
- package/dist/migrations/09-clientAndDataSearchees.d.ts +9 -0
- package/dist/migrations/10-indexerNameAudioBookCaps.d.ts +9 -0
- package/dist/migrations/11-trackers.d.ts +9 -0
- package/dist/migrations/12-user-auth.d.ts +9 -0
- package/dist/migrations/12-user-auth.js +22 -0
- package/dist/migrations/12-user-auth.js.map +1 -0
- package/dist/migrations/13-settings.d.ts +9 -0
- package/dist/migrations/13-settings.js +23 -0
- package/dist/migrations/13-settings.js.map +1 -0
- package/dist/migrations/14-indexer-enabled-flag.d.ts +9 -0
- package/dist/migrations/14-indexer-enabled-flag.js +12 -0
- package/dist/migrations/14-indexer-enabled-flag.js.map +1 -0
- package/dist/migrations/15-remove-url-unique-constraint.d.ts +9 -0
- package/dist/migrations/15-remove-url-unique-constraint.js +14 -0
- package/dist/migrations/15-remove-url-unique-constraint.js.map +1 -0
- package/dist/migrations/16-prune-inactive-indexers.d.ts +9 -0
- package/dist/migrations/16-prune-inactive-indexers.js +17 -0
- package/dist/migrations/16-prune-inactive-indexers.js.map +1 -0
- package/dist/migrations/migrations.d.ts +13 -0
- package/dist/migrations/migrations.js +10 -0
- package/dist/migrations/migrations.js.map +1 -1
- package/dist/parseTorrent.d.ts +53 -0
- package/dist/pipeline.d.ts +41 -0
- package/dist/pipeline.js +57 -10
- package/dist/pipeline.js.map +1 -1
- package/dist/preFilter.d.ts +25 -0
- package/dist/preFilter.js +15 -15
- package/dist/preFilter.js.map +1 -1
- package/dist/problems/linking.d.ts +2 -0
- package/dist/problems/linking.js +80 -0
- package/dist/problems/linking.js.map +1 -0
- package/dist/problems/path.d.ts +22 -0
- package/dist/problems/path.js +96 -0
- package/dist/problems/path.js.map +1 -0
- package/dist/problems.d.ts +13 -0
- package/dist/problems.js +48 -0
- package/dist/problems.js.map +1 -0
- package/dist/pushNotifier.d.ts +19 -0
- package/dist/routes/baseApi.d.ts +2 -0
- package/dist/routes/baseApi.js +354 -0
- package/dist/routes/baseApi.js.map +1 -0
- package/dist/routes/indexerApi.d.ts +6 -0
- package/dist/routes/indexerApi.js +165 -0
- package/dist/routes/indexerApi.js.map +1 -0
- package/dist/routes/staticFrontendPlugin.d.ts +4 -0
- package/dist/routes/staticFrontendPlugin.js +61 -0
- package/dist/routes/staticFrontendPlugin.js.map +1 -0
- package/dist/runtimeConfig.d.ts +6 -0
- package/dist/runtimeConfig.js +17 -1
- package/dist/runtimeConfig.js.map +1 -1
- package/dist/searchee.d.ts +108 -0
- package/dist/searchee.js +36 -5
- package/dist/searchee.js.map +1 -1
- package/dist/server.d.ts +4 -0
- package/dist/server.js +38 -429
- package/dist/server.js.map +1 -1
- package/dist/services/indexerService.d.ts +96 -0
- package/dist/services/indexerService.js +287 -0
- package/dist/services/indexerService.js.map +1 -0
- package/dist/sessionCookies.d.ts +5 -0
- package/dist/sessionCookies.js +27 -0
- package/dist/sessionCookies.js.map +1 -0
- package/dist/startup.d.ts +25 -0
- package/dist/startup.js +105 -151
- package/dist/startup.js.map +1 -1
- package/dist/torrent.d.ts +69 -0
- package/dist/torrent.js +17 -13
- package/dist/torrent.js.map +1 -1
- package/dist/torznab.d.ts +60 -0
- package/dist/torznab.js +14 -89
- package/dist/torznab.js.map +1 -1
- package/dist/trpc/fastifyAdapter.d.ts +2 -0
- package/dist/trpc/fastifyAdapter.js +9 -0
- package/dist/trpc/fastifyAdapter.js.map +1 -0
- package/dist/trpc/index.d.ts +49 -0
- package/dist/trpc/index.js +53 -0
- package/dist/trpc/index.js.map +1 -0
- package/dist/trpc/routers/auth.d.ts +43 -0
- package/dist/trpc/routers/auth.js +116 -0
- package/dist/trpc/routers/auth.js.map +1 -0
- package/dist/trpc/routers/clients.d.ts +21 -0
- package/dist/trpc/routers/clients.js +65 -0
- package/dist/trpc/routers/clients.js.map +1 -0
- package/dist/trpc/routers/health.d.ts +14 -0
- package/dist/trpc/routers/health.js +20 -0
- package/dist/trpc/routers/health.js.map +1 -0
- package/dist/trpc/routers/index.d.ts +391 -0
- package/dist/trpc/routers/index.js +23 -0
- package/dist/trpc/routers/index.js.map +1 -0
- package/dist/trpc/routers/indexers.d.ts +75 -0
- package/dist/trpc/routers/indexers.js +79 -0
- package/dist/trpc/routers/indexers.js.map +1 -0
- package/dist/trpc/routers/jobs.d.ts +33 -0
- package/dist/trpc/routers/jobs.js +84 -0
- package/dist/trpc/routers/jobs.js.map +1 -0
- package/dist/trpc/routers/logs.d.ts +27 -0
- package/dist/trpc/routers/logs.js +91 -0
- package/dist/trpc/routers/logs.js.map +1 -0
- package/dist/trpc/routers/searchees.d.ts +51 -0
- package/dist/trpc/routers/searchees.js +156 -0
- package/dist/trpc/routers/searchees.js.map +1 -0
- package/dist/trpc/routers/settings.d.ts +83 -0
- package/dist/trpc/routers/settings.js +92 -0
- package/dist/trpc/routers/settings.js.map +1 -0
- package/dist/trpc/routers/stats.d.ts +42 -0
- package/dist/trpc/routers/stats.js +102 -0
- package/dist/trpc/routers/stats.js.map +1 -0
- package/dist/userAuth.d.ts +21 -0
- package/dist/userAuth.js +86 -0
- package/dist/userAuth.js.map +1 -0
- package/dist/utils/authUtils.d.ts +10 -0
- package/dist/utils/authUtils.js +24 -0
- package/dist/utils/authUtils.js.map +1 -0
- package/dist/utils/logWatcher.d.ts +28 -0
- package/dist/utils/logWatcher.js +218 -0
- package/dist/utils/logWatcher.js.map +1 -0
- package/dist/utils/object.d.ts +1 -0
- package/dist/utils/object.js +4 -0
- package/dist/utils/object.js.map +1 -0
- package/dist/utils.d.ts +175 -0
- package/dist/utils.js +61 -38
- package/dist/utils.js.map +1 -1
- package/dist/webui/assets/FieldInfo-Bxj_j8SJ.js +1 -0
- package/dist/webui/assets/Page-C3rteCZt.js +1 -0
- package/dist/webui/assets/array-field-DVSC6nHP.js +1 -0
- package/dist/webui/assets/badge-DTZMtS0e.js +1 -0
- package/dist/webui/assets/check-Bu3ldi63.js +1 -0
- package/dist/webui/assets/chevron-down-CRy8M0kJ.js +1 -0
- package/dist/webui/assets/clients-CW8oEZoQ.js +1 -0
- package/dist/webui/assets/connect-YBNsnjWT.js +1 -0
- package/dist/webui/assets/debug-mz8-WYZj.js +1 -0
- package/dist/webui/assets/directories-BSK28RgR.js +1 -0
- package/dist/webui/assets/duration-field-C6xoSlJg.js +1 -0
- package/dist/webui/assets/general-lJJxZhH7.js +1 -0
- package/dist/webui/assets/health-CXbsVrie.js +1 -0
- package/dist/webui/assets/index-Bi48hI2z.js +54 -0
- package/dist/webui/assets/index-C-Ul7GNg.css +1 -0
- package/dist/webui/assets/index-C2cH1Gst.js +1 -0
- package/dist/webui/assets/index-Cc5bDmJr.js +1 -0
- package/dist/webui/assets/jobs-CxmNab9w.js +1 -0
- package/dist/webui/assets/library-vaj2W8sE.js +1 -0
- package/dist/webui/assets/loader-circle-M0gu1gZ-.js +1 -0
- package/dist/webui/assets/logs-Cu9RyKS0.js +1 -0
- package/dist/webui/assets/search-2R5sIdT8.js +1 -0
- package/dist/webui/assets/select-field-BCqNLDrJ.js +1 -0
- package/dist/webui/assets/select-zHgqMzLj.js +1 -0
- package/dist/webui/assets/settings-CMYjpTbZ.js +1 -0
- package/dist/webui/assets/submit-button-BtcnyggQ.js +1 -0
- package/dist/webui/assets/switch-G0W3uJVN.js +1 -0
- package/dist/webui/assets/switch-field-IBd9ORNq.js +1 -0
- package/dist/webui/assets/table-DvgJU7Gh.js +1 -0
- package/dist/webui/assets/test-tube-BIwmoM45.js +1 -0
- package/dist/webui/assets/text-field-DruSbGhy.js +1 -0
- package/dist/webui/assets/time-BSMZjmyW.js +1 -0
- package/dist/webui/assets/trackers-D-OpAe63.js +7 -0
- package/dist/webui/assets/use-form-validation-context-BkAfWAh0.js +1 -0
- package/dist/webui/assets/use-settings-form-submit-CDRh-E9U.js +2 -0
- package/dist/webui/assets/useQuery-A4Hv_4uX.js +1 -0
- package/dist/webui/index.html +13 -0
- package/node_modules/@cross-seed/shared/dist/configSchema.d.ts +261 -0
- package/node_modules/@cross-seed/shared/dist/configSchema.d.ts.map +1 -0
- package/node_modules/@cross-seed/shared/dist/configSchema.js +53 -0
- package/node_modules/@cross-seed/shared/dist/configSchema.js.map +1 -0
- package/node_modules/@cross-seed/shared/dist/constants.d.ts +122 -0
- package/node_modules/@cross-seed/shared/dist/constants.d.ts.map +1 -0
- package/node_modules/@cross-seed/shared/dist/constants.js +127 -0
- package/node_modules/@cross-seed/shared/dist/constants.js.map +1 -0
- package/node_modules/@cross-seed/shared/dist/tsconfig.tsbuildinfo +1 -0
- package/node_modules/@cross-seed/shared/dist/utils.d.ts +6 -0
- package/node_modules/@cross-seed/shared/dist/utils.d.ts.map +1 -0
- package/node_modules/@cross-seed/shared/dist/utils.js +9 -0
- package/node_modules/@cross-seed/shared/dist/utils.js.map +1 -0
- package/node_modules/@cross-seed/shared/package.json +22 -0
- package/package.json +35 -11
- package/dist/config.template.cjs +0 -353
- package/dist/config.template.cjs.map +0 -1
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
export declare const settingsRouter: import("@trpc/server").TRPCBuiltRouter<{
|
|
3
|
+
ctx: import("../index.js").Context;
|
|
4
|
+
meta: object;
|
|
5
|
+
errorShape: import("@trpc/server").TRPCDefaultErrorShape;
|
|
6
|
+
transformer: false;
|
|
7
|
+
}, import("@trpc/server").TRPCDecorateCreateRouterOptions<{
|
|
8
|
+
get: import("@trpc/server").TRPCQueryProcedure<{
|
|
9
|
+
input: void;
|
|
10
|
+
output: {
|
|
11
|
+
config: {
|
|
12
|
+
delay: number;
|
|
13
|
+
torznab: string[];
|
|
14
|
+
useClientTorrents: boolean;
|
|
15
|
+
dataDirs: string[];
|
|
16
|
+
matchMode: import("packages/shared/dist/constants.js").MatchMode;
|
|
17
|
+
skipRecheck: boolean;
|
|
18
|
+
autoResumeMaxDownload: number;
|
|
19
|
+
ignoreNonRelevantFilesToResume: boolean;
|
|
20
|
+
linkDirs: string[];
|
|
21
|
+
linkType: import("packages/shared/dist/constants.js").LinkType;
|
|
22
|
+
flatLinking: boolean;
|
|
23
|
+
maxDataDepth: number;
|
|
24
|
+
outputDir: string;
|
|
25
|
+
includeSingleEpisodes: boolean;
|
|
26
|
+
verbose: boolean;
|
|
27
|
+
includeNonVideos: boolean;
|
|
28
|
+
fuzzySizeThreshold: number;
|
|
29
|
+
action: import("packages/shared/dist/constants.js").Action;
|
|
30
|
+
torrentClients: string[];
|
|
31
|
+
duplicateCategories: boolean;
|
|
32
|
+
notificationWebhookUrls: string[];
|
|
33
|
+
torrents: string[];
|
|
34
|
+
blockList: string[];
|
|
35
|
+
sonarr: string[];
|
|
36
|
+
radarr: string[];
|
|
37
|
+
linkCategory?: string | undefined;
|
|
38
|
+
torrentDir?: string | undefined;
|
|
39
|
+
injectDir?: string | undefined;
|
|
40
|
+
ignoreTitles?: boolean | undefined;
|
|
41
|
+
seasonFromEpisodes?: number | undefined;
|
|
42
|
+
excludeOlder?: number | undefined;
|
|
43
|
+
excludeRecentSearch?: number | undefined;
|
|
44
|
+
port?: number | undefined;
|
|
45
|
+
host?: string | undefined;
|
|
46
|
+
basePath?: string | undefined;
|
|
47
|
+
searchCadence?: number | undefined;
|
|
48
|
+
rssCadence?: number | undefined;
|
|
49
|
+
snatchTimeout?: number | undefined;
|
|
50
|
+
searchTimeout?: number | undefined;
|
|
51
|
+
searchLimit?: number | undefined;
|
|
52
|
+
apiKey?: string | undefined;
|
|
53
|
+
};
|
|
54
|
+
apikey: string;
|
|
55
|
+
};
|
|
56
|
+
meta: object;
|
|
57
|
+
}>;
|
|
58
|
+
save: import("@trpc/server").TRPCMutationProcedure<{
|
|
59
|
+
input: z.objectInputType<{}, z.ZodTypeAny, "passthrough">;
|
|
60
|
+
output: {
|
|
61
|
+
success: boolean;
|
|
62
|
+
};
|
|
63
|
+
meta: object;
|
|
64
|
+
}>;
|
|
65
|
+
replace: import("@trpc/server").TRPCMutationProcedure<{
|
|
66
|
+
input: z.objectInputType<{}, z.ZodTypeAny, "passthrough">;
|
|
67
|
+
output: {
|
|
68
|
+
success: boolean;
|
|
69
|
+
};
|
|
70
|
+
meta: object;
|
|
71
|
+
}>;
|
|
72
|
+
validate: import("@trpc/server").TRPCQueryProcedure<{
|
|
73
|
+
input: void;
|
|
74
|
+
output: {
|
|
75
|
+
status: string;
|
|
76
|
+
validations: {
|
|
77
|
+
paths: boolean;
|
|
78
|
+
torznab: boolean;
|
|
79
|
+
};
|
|
80
|
+
};
|
|
81
|
+
meta: object;
|
|
82
|
+
}>;
|
|
83
|
+
}>>;
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { authedProcedure, router } from "../index.js";
|
|
2
|
+
import { Label, logger } from "../../logger.js";
|
|
3
|
+
import { setRuntimeConfig } from "../../runtimeConfig.js";
|
|
4
|
+
import { getApiKey } from "../../auth.js";
|
|
5
|
+
import { z } from "zod";
|
|
6
|
+
import { getDbConfig, setDbConfig, updateDbConfig } from "../../dbConfig.js";
|
|
7
|
+
import { getDefaultRuntimeConfig } from "../../configuration.js";
|
|
8
|
+
import { omitUndefined } from "../../utils/object.js";
|
|
9
|
+
import { parseRuntimeConfig } from "../../configSchema.js";
|
|
10
|
+
export const settingsRouter = router({
|
|
11
|
+
get: authedProcedure.query(async () => {
|
|
12
|
+
try {
|
|
13
|
+
const runtimeOverrides = await getDbConfig();
|
|
14
|
+
const runtimeConfig = {
|
|
15
|
+
...getDefaultRuntimeConfig(),
|
|
16
|
+
...runtimeOverrides,
|
|
17
|
+
};
|
|
18
|
+
const apikey = await getApiKey();
|
|
19
|
+
return {
|
|
20
|
+
config: runtimeConfig,
|
|
21
|
+
apikey,
|
|
22
|
+
};
|
|
23
|
+
}
|
|
24
|
+
catch (error) {
|
|
25
|
+
logger.error({ label: Label.SERVER, message: error.message });
|
|
26
|
+
throw new Error(`Failed to read config: ${error.message}`);
|
|
27
|
+
}
|
|
28
|
+
}),
|
|
29
|
+
// We'll need to implement the save functionality
|
|
30
|
+
save: authedProcedure
|
|
31
|
+
.input(z.object({}).passthrough())
|
|
32
|
+
.mutation(async ({ input }) => {
|
|
33
|
+
try {
|
|
34
|
+
logger.info({
|
|
35
|
+
label: Label.SERVER,
|
|
36
|
+
message: `Saving config updates...`,
|
|
37
|
+
});
|
|
38
|
+
// Save to database
|
|
39
|
+
await updateDbConfig(input);
|
|
40
|
+
// Update in-memory config with the merged result
|
|
41
|
+
const updatedOverrides = await getDbConfig();
|
|
42
|
+
setRuntimeConfig({
|
|
43
|
+
...getDefaultRuntimeConfig(),
|
|
44
|
+
...updatedOverrides,
|
|
45
|
+
});
|
|
46
|
+
return { success: true };
|
|
47
|
+
}
|
|
48
|
+
catch (error) {
|
|
49
|
+
logger.error({ label: Label.SERVER, message: error.message });
|
|
50
|
+
throw new Error(`Failed to save config: ${error.message}`);
|
|
51
|
+
}
|
|
52
|
+
}),
|
|
53
|
+
// Full replacement for debug page
|
|
54
|
+
replace: authedProcedure
|
|
55
|
+
.input(z.object({}).passthrough())
|
|
56
|
+
.mutation(async ({ input }) => {
|
|
57
|
+
try {
|
|
58
|
+
logger.info({
|
|
59
|
+
label: Label.SERVER,
|
|
60
|
+
message: `Replacing full config...`,
|
|
61
|
+
});
|
|
62
|
+
const parsedConfig = parseRuntimeConfig(input);
|
|
63
|
+
await setDbConfig(parsedConfig);
|
|
64
|
+
// Update in-memory config so changes are visible immediately
|
|
65
|
+
const sanitizedInput = omitUndefined(parsedConfig);
|
|
66
|
+
setRuntimeConfig({
|
|
67
|
+
...getDefaultRuntimeConfig(),
|
|
68
|
+
...sanitizedInput,
|
|
69
|
+
});
|
|
70
|
+
return { success: true };
|
|
71
|
+
}
|
|
72
|
+
catch (error) {
|
|
73
|
+
logger.error({ label: Label.SERVER, message: error.message });
|
|
74
|
+
throw new Error(`Failed to replace config: ${error.message}`);
|
|
75
|
+
}
|
|
76
|
+
}),
|
|
77
|
+
validate: authedProcedure.query(async () => {
|
|
78
|
+
try {
|
|
79
|
+
// This is a placeholder for config validation
|
|
80
|
+
// We need to implement proper validation logic
|
|
81
|
+
return {
|
|
82
|
+
status: "success",
|
|
83
|
+
validations: { paths: true, torznab: true },
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
catch (error) {
|
|
87
|
+
logger.error({ label: Label.SERVER, message: error.message });
|
|
88
|
+
throw new Error(`Failed to validate config: ${error.message}`);
|
|
89
|
+
}
|
|
90
|
+
}),
|
|
91
|
+
});
|
|
92
|
+
//# sourceMappingURL=settings.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"settings.js","sourceRoot":"","sources":["../../../src/trpc/routers/settings.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
export declare const statsRouter: import("@trpc/server").TRPCBuiltRouter<{
|
|
2
|
+
ctx: import("../index.js").Context;
|
|
3
|
+
meta: object;
|
|
4
|
+
errorShape: import("@trpc/server").TRPCDefaultErrorShape;
|
|
5
|
+
transformer: false;
|
|
6
|
+
}, import("@trpc/server").TRPCDecorateCreateRouterOptions<{
|
|
7
|
+
getOverview: import("@trpc/server").TRPCQueryProcedure<{
|
|
8
|
+
input: void;
|
|
9
|
+
output: {
|
|
10
|
+
totalSearchees: number;
|
|
11
|
+
totalMatches: number;
|
|
12
|
+
totalIndexers: number;
|
|
13
|
+
healthyIndexers: number;
|
|
14
|
+
recentMatches: string | number;
|
|
15
|
+
matchRate: number;
|
|
16
|
+
matchesPerSnatch: number;
|
|
17
|
+
matchesPerQuery: number;
|
|
18
|
+
matchesPerQueryIndexer: number;
|
|
19
|
+
snatchCount: number;
|
|
20
|
+
queryCount: number;
|
|
21
|
+
queryIndexerCount: number;
|
|
22
|
+
wastedSnatchCount: number;
|
|
23
|
+
wastedSnatchRate: number;
|
|
24
|
+
unhealthyIndexers: number;
|
|
25
|
+
allIndexersHealthy: boolean;
|
|
26
|
+
decisionBreakdown: (Pick<import("knex/types/tables.js").Decision, "decision"> & {
|
|
27
|
+
count?: string | number | undefined;
|
|
28
|
+
})[];
|
|
29
|
+
};
|
|
30
|
+
meta: object;
|
|
31
|
+
}>;
|
|
32
|
+
getIndexerStats: import("@trpc/server").TRPCQueryProcedure<{
|
|
33
|
+
input: void;
|
|
34
|
+
output: {
|
|
35
|
+
id: number;
|
|
36
|
+
name: string;
|
|
37
|
+
enabled: boolean;
|
|
38
|
+
status: string;
|
|
39
|
+
}[];
|
|
40
|
+
meta: object;
|
|
41
|
+
}>;
|
|
42
|
+
}>>;
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { authedProcedure, router } from "../index.js";
|
|
2
|
+
import { db } from "../../db.js";
|
|
3
|
+
import { estimateSearchString } from "../../torznab.js";
|
|
4
|
+
import { mapAsync } from "../../utils.js";
|
|
5
|
+
export const statsRouter = router({
|
|
6
|
+
getOverview: authedProcedure.query(async () => {
|
|
7
|
+
const [searcheeResult, searcheeNames, totalIndexerResult, healthyIndexerResult, timestampResult, decisionsByType, recentMatches, matchAggregates,] = await Promise.all([
|
|
8
|
+
db("searchee").count({ count: "*" }).first(),
|
|
9
|
+
db("searchee").select("name"),
|
|
10
|
+
db("indexer").count({ count: "*" }).first(),
|
|
11
|
+
db("indexer")
|
|
12
|
+
.where({ enabled: true, search_cap: true })
|
|
13
|
+
.where((qb) => qb
|
|
14
|
+
.whereNull("status")
|
|
15
|
+
.orWhere("status", "OK")
|
|
16
|
+
.orWhere("retry_after", "<", Date.now()))
|
|
17
|
+
.count({ count: "*" })
|
|
18
|
+
.first(),
|
|
19
|
+
db("timestamp").count({ count: "*" }).first(),
|
|
20
|
+
db("decision")
|
|
21
|
+
.select("decision")
|
|
22
|
+
.count({ count: "*" })
|
|
23
|
+
.groupBy("decision"),
|
|
24
|
+
db("decision")
|
|
25
|
+
.whereIn("decision", [
|
|
26
|
+
"MATCH",
|
|
27
|
+
"MATCH_SIZE_ONLY",
|
|
28
|
+
"MATCH_PARTIAL",
|
|
29
|
+
])
|
|
30
|
+
.where("last_seen", ">", Date.now() - 24 * 60 * 60 * 1000) // last 24h
|
|
31
|
+
.count({ count: "*" })
|
|
32
|
+
.first(),
|
|
33
|
+
db("decision")
|
|
34
|
+
.whereNotNull("info_hash")
|
|
35
|
+
.select({
|
|
36
|
+
snatchCount: db.raw("COUNT(DISTINCT info_hash)"),
|
|
37
|
+
matchCount: db.raw("COUNT(DISTINCT CASE WHEN decision IN ('MATCH','MATCH_SIZE_ONLY','MATCH_PARTIAL') THEN info_hash END)"),
|
|
38
|
+
matchCountWithInfoHash: db.raw("COUNT(DISTINCT CASE WHEN decision IN ('MATCH','MATCH_SIZE_ONLY','MATCH_PARTIAL','SAME_INFO_HASH','INFO_HASH_ALREADY_EXISTS') THEN info_hash END)"),
|
|
39
|
+
})
|
|
40
|
+
.first(),
|
|
41
|
+
]);
|
|
42
|
+
const snatchCount = Number(matchAggregates?.snatchCount ?? 0);
|
|
43
|
+
const matchCount = Number(matchAggregates?.matchCount ?? 0);
|
|
44
|
+
const matchCountWithInfoHash = Number(matchAggregates?.matchCountWithInfoHash ?? 0);
|
|
45
|
+
const totalMatches = matchCount;
|
|
46
|
+
const searcheeNameList = searcheeNames
|
|
47
|
+
.map((row) => row.name)
|
|
48
|
+
.filter((name) => typeof name === "string" && name.length > 0);
|
|
49
|
+
const queryCount = new Set(await mapAsync(searcheeNameList, estimateSearchString)).size;
|
|
50
|
+
const totalSearchees = Number(searcheeResult?.count || 0);
|
|
51
|
+
const matchRate = totalSearchees > 0
|
|
52
|
+
? (totalMatches / totalSearchees).toFixed(2)
|
|
53
|
+
: "0";
|
|
54
|
+
const matchesPerSnatch = snatchCount > 0
|
|
55
|
+
? (matchCountWithInfoHash / snatchCount).toFixed(3)
|
|
56
|
+
: "0";
|
|
57
|
+
const matchesPerQuery = queryCount > 0 ? (matchCount / queryCount).toFixed(3) : "0";
|
|
58
|
+
const queryIndexerCount = Number(timestampResult?.count || 0);
|
|
59
|
+
const matchesPerQueryIndexer = queryIndexerCount > 0
|
|
60
|
+
? (matchCount / queryIndexerCount).toFixed(3)
|
|
61
|
+
: "0";
|
|
62
|
+
const wastedSnatchCount = Math.max(snatchCount - matchCountWithInfoHash, 0);
|
|
63
|
+
const wastedSnatchRate = snatchCount > 0
|
|
64
|
+
? (wastedSnatchCount / snatchCount).toFixed(3)
|
|
65
|
+
: "0";
|
|
66
|
+
const totalIndexers = Number(totalIndexerResult?.count || 0);
|
|
67
|
+
const healthyIndexers = Number(healthyIndexerResult?.count || 0);
|
|
68
|
+
const unhealthyIndexers = Math.max(totalIndexers - healthyIndexers, 0);
|
|
69
|
+
const allIndexersHealthy = unhealthyIndexers === 0;
|
|
70
|
+
return {
|
|
71
|
+
totalSearchees,
|
|
72
|
+
totalMatches,
|
|
73
|
+
totalIndexers,
|
|
74
|
+
healthyIndexers,
|
|
75
|
+
recentMatches: recentMatches?.count || 0,
|
|
76
|
+
matchRate: parseFloat(matchRate),
|
|
77
|
+
matchesPerSnatch: parseFloat(matchesPerSnatch),
|
|
78
|
+
matchesPerQuery: parseFloat(matchesPerQuery),
|
|
79
|
+
matchesPerQueryIndexer: parseFloat(matchesPerQueryIndexer),
|
|
80
|
+
snatchCount,
|
|
81
|
+
queryCount,
|
|
82
|
+
queryIndexerCount,
|
|
83
|
+
wastedSnatchCount,
|
|
84
|
+
wastedSnatchRate: parseFloat(wastedSnatchRate),
|
|
85
|
+
unhealthyIndexers,
|
|
86
|
+
allIndexersHealthy,
|
|
87
|
+
decisionBreakdown: decisionsByType,
|
|
88
|
+
};
|
|
89
|
+
}),
|
|
90
|
+
getIndexerStats: authedProcedure.query(async () => {
|
|
91
|
+
const indexers = await db("indexer")
|
|
92
|
+
.select("id", "name", "enabled", "status")
|
|
93
|
+
.orderBy("name");
|
|
94
|
+
return indexers.map((indexer) => ({
|
|
95
|
+
id: indexer.id,
|
|
96
|
+
name: indexer.name || `Indexer ${indexer.id}`,
|
|
97
|
+
enabled: indexer.enabled,
|
|
98
|
+
status: indexer.status || "unknown",
|
|
99
|
+
}));
|
|
100
|
+
}),
|
|
101
|
+
});
|
|
102
|
+
//# sourceMappingURL=stats.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"stats.js","sourceRoot":"","sources":["../../../src/trpc/routers/stats.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export interface User {
|
|
2
|
+
id: number;
|
|
3
|
+
username: string;
|
|
4
|
+
password: string;
|
|
5
|
+
created_at: Date;
|
|
6
|
+
}
|
|
7
|
+
export interface Session {
|
|
8
|
+
id: string;
|
|
9
|
+
user_id: number;
|
|
10
|
+
expires_at: number;
|
|
11
|
+
created_at: number;
|
|
12
|
+
}
|
|
13
|
+
export declare function createUser(username: string, password: string): Promise<User>;
|
|
14
|
+
export declare function findUserByUsername(username: string): Promise<User | undefined>;
|
|
15
|
+
export declare function validateUserCredentials(username: string, password: string): Promise<User | null>;
|
|
16
|
+
export declare function createSession(userId: number): Promise<Session>;
|
|
17
|
+
export declare function validateSession(sessionId: string): Promise<User | null>;
|
|
18
|
+
export declare function removeSession(sessionId: string): Promise<void>;
|
|
19
|
+
export declare function hasUsers(): Promise<boolean>;
|
|
20
|
+
export declare function createInitialUserIfNeeded(username: string, password: string): Promise<User | null>;
|
|
21
|
+
export declare function resetUsers(): Promise<string>;
|
package/dist/userAuth.js
ADDED
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import { randomBytes } from "node:crypto";
|
|
2
|
+
import bcrypt from "bcryptjs";
|
|
3
|
+
import { db } from "./db.js";
|
|
4
|
+
import { Label, logger } from "./logger.js";
|
|
5
|
+
const SESSION_EXPIRY = 30 * 24 * 60 * 60 * 1000; // 30 days in milliseconds
|
|
6
|
+
export async function createUser(username, password) {
|
|
7
|
+
const hashedPassword = await bcrypt.hash(password, 10);
|
|
8
|
+
const [user] = await db("user")
|
|
9
|
+
.insert({
|
|
10
|
+
username,
|
|
11
|
+
password: hashedPassword,
|
|
12
|
+
})
|
|
13
|
+
.returning("*");
|
|
14
|
+
logger.info({
|
|
15
|
+
label: Label.AUTH,
|
|
16
|
+
message: `Created user: ${username}`,
|
|
17
|
+
});
|
|
18
|
+
return user;
|
|
19
|
+
}
|
|
20
|
+
export async function findUserByUsername(username) {
|
|
21
|
+
return db("user").where({ username }).first();
|
|
22
|
+
}
|
|
23
|
+
export async function validateUserCredentials(username, password) {
|
|
24
|
+
const user = await findUserByUsername(username);
|
|
25
|
+
if (!user) {
|
|
26
|
+
return null;
|
|
27
|
+
}
|
|
28
|
+
const isValid = await bcrypt.compare(password, user.password);
|
|
29
|
+
return isValid ? user : null;
|
|
30
|
+
}
|
|
31
|
+
export async function createSession(userId) {
|
|
32
|
+
const sessionId = randomBytes(32).toString("hex");
|
|
33
|
+
const now = Date.now();
|
|
34
|
+
const expiresAt = now + SESSION_EXPIRY;
|
|
35
|
+
const session = {
|
|
36
|
+
id: sessionId,
|
|
37
|
+
user_id: userId,
|
|
38
|
+
expires_at: expiresAt,
|
|
39
|
+
created_at: now,
|
|
40
|
+
};
|
|
41
|
+
await db("session").insert(session);
|
|
42
|
+
return session;
|
|
43
|
+
}
|
|
44
|
+
export async function validateSession(sessionId) {
|
|
45
|
+
const session = await db("session")
|
|
46
|
+
.where({
|
|
47
|
+
id: sessionId,
|
|
48
|
+
})
|
|
49
|
+
.where("expires_at", ">", Date.now())
|
|
50
|
+
.first();
|
|
51
|
+
if (!session) {
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
const user = await db("user").where({ id: session.user_id }).first();
|
|
55
|
+
return user || null;
|
|
56
|
+
}
|
|
57
|
+
export async function removeSession(sessionId) {
|
|
58
|
+
await db("session").where({ id: sessionId }).delete();
|
|
59
|
+
}
|
|
60
|
+
export async function hasUsers() {
|
|
61
|
+
const count = await db("user").count("* as count").first();
|
|
62
|
+
return count?.count > 0;
|
|
63
|
+
}
|
|
64
|
+
export async function createInitialUserIfNeeded(username, password) {
|
|
65
|
+
const hasExistingUsers = await hasUsers();
|
|
66
|
+
if (hasExistingUsers) {
|
|
67
|
+
logger.info({
|
|
68
|
+
label: Label.AUTH,
|
|
69
|
+
message: "Initial user already exists, skipping creation",
|
|
70
|
+
});
|
|
71
|
+
return null;
|
|
72
|
+
}
|
|
73
|
+
logger.info({
|
|
74
|
+
label: Label.AUTH,
|
|
75
|
+
message: "Creating initial user",
|
|
76
|
+
});
|
|
77
|
+
return createUser(username, password);
|
|
78
|
+
}
|
|
79
|
+
export async function resetUsers() {
|
|
80
|
+
const deletedSessions = await db("session").del();
|
|
81
|
+
const deletedUsers = await db("user").del();
|
|
82
|
+
const userLabel = deletedUsers === 1 ? "user" : "users";
|
|
83
|
+
const sessionLabel = deletedSessions === 1 ? "session" : "sessions";
|
|
84
|
+
return `Deleted ${deletedUsers} ${userLabel} and ${deletedSessions} ${sessionLabel}.`;
|
|
85
|
+
}
|
|
86
|
+
//# sourceMappingURL=userAuth.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"userAuth.js","sourceRoot":"","sources":["../src/userAuth.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { FastifyRequest, FastifyReply } from "fastify";
|
|
2
|
+
/**
|
|
3
|
+
* Checks all http API requests for authorized apiKey
|
|
4
|
+
* uses param `?apikey=` or as header `x-api-key`
|
|
5
|
+
*/
|
|
6
|
+
export declare function authorize(request: FastifyRequest<{
|
|
7
|
+
Querystring: {
|
|
8
|
+
apikey?: string;
|
|
9
|
+
};
|
|
10
|
+
}>, reply: FastifyReply): Promise<boolean>;
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { checkApiKey } from "../auth.js";
|
|
2
|
+
import { Label, logger } from "../logger.js";
|
|
3
|
+
/**
|
|
4
|
+
* Checks all http API requests for authorized apiKey
|
|
5
|
+
* uses param `?apikey=` or as header `x-api-key`
|
|
6
|
+
*/
|
|
7
|
+
export async function authorize(request, reply) {
|
|
8
|
+
const apiKey = request.headers["x-api-key"] || request.query.apikey || "";
|
|
9
|
+
const isAuthorized = await checkApiKey(apiKey);
|
|
10
|
+
if (!isAuthorized) {
|
|
11
|
+
const ipAddress = request.headers["x-forwarded-for"]
|
|
12
|
+
?.split(",")
|
|
13
|
+
.shift() || request.socket.remoteAddress;
|
|
14
|
+
logger.error({
|
|
15
|
+
label: Label.SERVER,
|
|
16
|
+
message: `Unauthorized API access attempt to ${request.url} from ${ipAddress}`,
|
|
17
|
+
});
|
|
18
|
+
void reply
|
|
19
|
+
.code(401)
|
|
20
|
+
.send("Specify the API key in an X-Api-Key header or an apikey query param.");
|
|
21
|
+
}
|
|
22
|
+
return isAuthorized;
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=authUtils.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"authUtils.js","sourceRoot":"","sources":["../../src/utils/authUtils.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AACzC,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAE7C;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,SAAS,CAC9B,OAEE,EACF,KAAmB;IAEnB,MAAM,MAAM,GACV,OAAO,CAAC,OAAO,CAAC,WAAW,CAAY,IAAI,OAAO,CAAC,KAAK,CAAC,MAAM,IAAI,EAAE,CAAC;IACxE,MAAM,YAAY,GAAG,MAAM,WAAW,CAAC,MAAM,CAAC,CAAC;IAC/C,IAAI,CAAC,YAAY,EAAE,CAAC;QACnB,MAAM,SAAS,GACb,OAAO,CAAC,OAAO,CAAC,iBAAiB,CAAY;YAC7C,EAAE,KAAK,CAAC,GAAG,CAAC;aACX,KAAK,EAAE,IAAI,OAAO,CAAC,MAAM,CAAC,aAAa,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC;YACZ,KAAK,EAAE,KAAK,CAAC,MAAM;YACnB,OAAO,EAAE,sCAAsC,OAAO,CAAC,GAAG,SAAS,SAAS,EAAE;SAC9E,CAAC,CAAC;QACH,KAAK,KAAK;aACR,IAAI,CAAC,GAAG,CAAC;aACT,IAAI,CACJ,sEAAsE,CACtE,CAAC;IACJ,CAAC;IACD,OAAO,YAAY,CAAC;AACrB,CAAC"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export interface LogEntry {
|
|
2
|
+
timestamp: string;
|
|
3
|
+
level: string;
|
|
4
|
+
label?: string;
|
|
5
|
+
message: string;
|
|
6
|
+
}
|
|
7
|
+
type LogCallback = (log: LogEntry) => void;
|
|
8
|
+
declare class LogWatcher {
|
|
9
|
+
private watchers;
|
|
10
|
+
private subscribers;
|
|
11
|
+
private lastPositions;
|
|
12
|
+
private retryTimers;
|
|
13
|
+
constructor();
|
|
14
|
+
private startWatching;
|
|
15
|
+
private watchLogFile;
|
|
16
|
+
private initializePosition;
|
|
17
|
+
private handleFileChange;
|
|
18
|
+
private parseLogContent;
|
|
19
|
+
private tryParseLogLine;
|
|
20
|
+
private notifySubscribers;
|
|
21
|
+
subscribe(callback: LogCallback): () => void;
|
|
22
|
+
getRecentLogs(limit?: number): Promise<LogEntry[]>;
|
|
23
|
+
private shouldIncludeLevel;
|
|
24
|
+
destroy(): void;
|
|
25
|
+
}
|
|
26
|
+
export declare function getLogWatcher(): LogWatcher;
|
|
27
|
+
export declare function destroyLogWatcher(): void;
|
|
28
|
+
export {};
|