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,75 @@
|
|
|
1
|
+
export declare const indexersRouter: 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
|
+
getAll: import("@trpc/server").TRPCQueryProcedure<{
|
|
8
|
+
input: void;
|
|
9
|
+
output: import("../../indexers.js").Indexer[];
|
|
10
|
+
meta: object;
|
|
11
|
+
}>;
|
|
12
|
+
mergeDisabled: import("@trpc/server").TRPCMutationProcedure<{
|
|
13
|
+
input: {
|
|
14
|
+
sourceId: number;
|
|
15
|
+
targetId: number;
|
|
16
|
+
};
|
|
17
|
+
output: {
|
|
18
|
+
mergedCount: number;
|
|
19
|
+
deleted: boolean;
|
|
20
|
+
};
|
|
21
|
+
meta: object;
|
|
22
|
+
}>;
|
|
23
|
+
create: import("@trpc/server").TRPCMutationProcedure<{
|
|
24
|
+
input: {
|
|
25
|
+
url: string;
|
|
26
|
+
apikey: string;
|
|
27
|
+
name?: string | undefined;
|
|
28
|
+
enabled?: boolean | undefined;
|
|
29
|
+
};
|
|
30
|
+
output: import("../../indexers.js").Indexer;
|
|
31
|
+
meta: object;
|
|
32
|
+
}>;
|
|
33
|
+
update: import("@trpc/server").TRPCMutationProcedure<{
|
|
34
|
+
input: {
|
|
35
|
+
id: number;
|
|
36
|
+
name?: string | null | undefined;
|
|
37
|
+
url?: string | undefined;
|
|
38
|
+
apikey?: string | undefined;
|
|
39
|
+
enabled?: boolean | undefined;
|
|
40
|
+
};
|
|
41
|
+
output: import("../../indexers.js").Indexer;
|
|
42
|
+
meta: object;
|
|
43
|
+
}>;
|
|
44
|
+
delete: import("@trpc/server").TRPCMutationProcedure<{
|
|
45
|
+
input: {
|
|
46
|
+
id: number;
|
|
47
|
+
};
|
|
48
|
+
output: {
|
|
49
|
+
success: true;
|
|
50
|
+
indexer: import("../../indexers.js").Indexer;
|
|
51
|
+
};
|
|
52
|
+
meta: object;
|
|
53
|
+
}>;
|
|
54
|
+
testExisting: import("@trpc/server").TRPCMutationProcedure<{
|
|
55
|
+
input: {
|
|
56
|
+
id: number;
|
|
57
|
+
};
|
|
58
|
+
output: {
|
|
59
|
+
success: true;
|
|
60
|
+
message: string;
|
|
61
|
+
};
|
|
62
|
+
meta: object;
|
|
63
|
+
}>;
|
|
64
|
+
testNew: import("@trpc/server").TRPCMutationProcedure<{
|
|
65
|
+
input: {
|
|
66
|
+
url: string;
|
|
67
|
+
apikey: string;
|
|
68
|
+
};
|
|
69
|
+
output: {
|
|
70
|
+
success: true;
|
|
71
|
+
message: string;
|
|
72
|
+
};
|
|
73
|
+
meta: object;
|
|
74
|
+
}>;
|
|
75
|
+
}>>;
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
import { router, authedProcedure } from "../index.js";
|
|
3
|
+
import { indexerCreateSchema, indexerUpdateSchema, createIndexer, updateIndexer, deleteIndexer, testExistingIndexer, testNewIndexer, listAllIndexers, mergeDisabledIndexer, } from "../../services/indexerService.js";
|
|
4
|
+
import { TRPCError } from "@trpc/server";
|
|
5
|
+
export const indexersRouter = router({
|
|
6
|
+
// Get all indexers
|
|
7
|
+
getAll: authedProcedure.query(async () => {
|
|
8
|
+
const indexers = await listAllIndexers();
|
|
9
|
+
return indexers.sort((a, b) => (a.name || "").localeCompare(b.name || ""));
|
|
10
|
+
}),
|
|
11
|
+
mergeDisabled: authedProcedure
|
|
12
|
+
.input(z.object({
|
|
13
|
+
sourceId: z.number().int().positive(),
|
|
14
|
+
targetId: z.number().int().positive(),
|
|
15
|
+
}))
|
|
16
|
+
.mutation(async ({ input }) => {
|
|
17
|
+
const result = await mergeDisabledIndexer(input.sourceId, input.targetId);
|
|
18
|
+
if (result.isErr()) {
|
|
19
|
+
const err = result.unwrapErr();
|
|
20
|
+
throw new TRPCError({
|
|
21
|
+
code: "BAD_REQUEST",
|
|
22
|
+
message: err.message,
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
return result.unwrap();
|
|
26
|
+
}),
|
|
27
|
+
// Create new indexer
|
|
28
|
+
create: authedProcedure
|
|
29
|
+
.input(indexerCreateSchema)
|
|
30
|
+
.mutation(async ({ input }) => {
|
|
31
|
+
return createIndexer(input);
|
|
32
|
+
}),
|
|
33
|
+
// Update indexer
|
|
34
|
+
update: authedProcedure
|
|
35
|
+
.input(indexerUpdateSchema)
|
|
36
|
+
.mutation(async ({ input }) => {
|
|
37
|
+
const result = await updateIndexer(input);
|
|
38
|
+
if (result.isErr()) {
|
|
39
|
+
throw new Error(`Indexer with ID ${input.id} not found`);
|
|
40
|
+
}
|
|
41
|
+
return result.unwrap();
|
|
42
|
+
}),
|
|
43
|
+
// Delete indexer
|
|
44
|
+
delete: authedProcedure
|
|
45
|
+
.input(z.object({ id: z.number().int().positive() }))
|
|
46
|
+
.mutation(async ({ input }) => {
|
|
47
|
+
const result = await deleteIndexer(input.id);
|
|
48
|
+
if (result.isErr()) {
|
|
49
|
+
throw new Error(`Indexer with ID ${input.id} not found`);
|
|
50
|
+
}
|
|
51
|
+
return result.unwrap();
|
|
52
|
+
}),
|
|
53
|
+
// Test existing indexer connection
|
|
54
|
+
testExisting: authedProcedure
|
|
55
|
+
.input(z.object({ id: z.number().int().positive() }))
|
|
56
|
+
.mutation(async ({ input }) => {
|
|
57
|
+
const result = await testExistingIndexer(input.id);
|
|
58
|
+
if (result.isErr()) {
|
|
59
|
+
const err = result.unwrapErr();
|
|
60
|
+
throw new Error(err.message);
|
|
61
|
+
}
|
|
62
|
+
return result.unwrap();
|
|
63
|
+
}),
|
|
64
|
+
// Test new indexer connection before creating
|
|
65
|
+
testNew: authedProcedure
|
|
66
|
+
.input(z.object({
|
|
67
|
+
url: z.string().url(),
|
|
68
|
+
apikey: z.string().min(1),
|
|
69
|
+
}))
|
|
70
|
+
.mutation(async ({ input }) => {
|
|
71
|
+
const result = await testNewIndexer(input);
|
|
72
|
+
if (result.isErr()) {
|
|
73
|
+
const err = result.unwrapErr();
|
|
74
|
+
throw new Error(err.message);
|
|
75
|
+
}
|
|
76
|
+
return result.unwrap();
|
|
77
|
+
}),
|
|
78
|
+
});
|
|
79
|
+
//# sourceMappingURL=indexers.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"indexers.js","sourceRoot":"","sources":["../../../src/trpc/routers/indexers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,MAAM,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AACtD,OAAO,EACN,mBAAmB,EACnB,mBAAmB,EACnB,aAAa,EACb,aAAa,EACb,aAAa,EACb,mBAAmB,EACnB,cAAc,EACd,eAAe,EACf,oBAAoB,GACpB,MAAM,kCAAkC,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAEzC,MAAM,CAAC,MAAM,cAAc,GAAG,MAAM,CAAC;IACpC,mBAAmB;IACnB,MAAM,EAAE,eAAe,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE;QACxC,MAAM,QAAQ,GAAG,MAAM,eAAe,EAAE,CAAC;QACzC,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAC7B,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,CAC1C,CAAC;IACH,CAAC,CAAC;IAEF,aAAa,EAAE,eAAe;SAC5B,KAAK,CACL,CAAC,CAAC,MAAM,CAAC;QACR,QAAQ,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE;QACrC,QAAQ,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE;KACrC,CAAC,CACF;SACA,QAAQ,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;QAC7B,MAAM,MAAM,GAAG,MAAM,oBAAoB,CACxC,KAAK,CAAC,QAAQ,EACd,KAAK,CAAC,QAAQ,CACd,CAAC;QACF,IAAI,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;YACpB,MAAM,GAAG,GAAG,MAAM,CAAC,SAAS,EAAE,CAAC;YAC/B,MAAM,IAAI,SAAS,CAAC;gBACnB,IAAI,EAAE,aAAa;gBACnB,OAAO,EAAE,GAAG,CAAC,OAAO;aACpB,CAAC,CAAC;QACJ,CAAC;QACD,OAAO,MAAM,CAAC,MAAM,EAAE,CAAC;IACxB,CAAC,CAAC;IAEH,qBAAqB;IACrB,MAAM,EAAE,eAAe;SACrB,KAAK,CAAC,mBAAmB,CAAC;SAC1B,QAAQ,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;QAC7B,OAAO,aAAa,CAAC,KAAK,CAAC,CAAC;IAC7B,CAAC,CAAC;IAEH,iBAAiB;IACjB,MAAM,EAAE,eAAe;SACrB,KAAK,CAAC,mBAAmB,CAAC;SAC1B,QAAQ,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;QAC7B,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,KAAK,CAAC,CAAC;QAC1C,IAAI,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,mBAAmB,KAAK,CAAC,EAAE,YAAY,CAAC,CAAC;QAC1D,CAAC;QACD,OAAO,MAAM,CAAC,MAAM,EAAE,CAAC;IACxB,CAAC,CAAC;IAEH,iBAAiB;IACjB,MAAM,EAAE,eAAe;SACrB,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;SACpD,QAAQ,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;QAC7B,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC7C,IAAI,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,mBAAmB,KAAK,CAAC,EAAE,YAAY,CAAC,CAAC;QAC1D,CAAC;QACD,OAAO,MAAM,CAAC,MAAM,EAAE,CAAC;IACxB,CAAC,CAAC;IAEH,mCAAmC;IACnC,YAAY,EAAE,eAAe;SAC3B,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;SACpD,QAAQ,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;QAC7B,MAAM,MAAM,GAAG,MAAM,mBAAmB,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACnD,IAAI,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;YACpB,MAAM,GAAG,GAAG,MAAM,CAAC,SAAS,EAAE,CAAC;YAC/B,MAAM,IAAI,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;QACD,OAAO,MAAM,CAAC,MAAM,EAAE,CAAC;IACxB,CAAC,CAAC;IAEH,8CAA8C;IAC9C,OAAO,EAAE,eAAe;SACtB,KAAK,CACL,CAAC,CAAC,MAAM,CAAC;QACR,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE;QACrB,MAAM,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;KACzB,CAAC,CACF;SACA,QAAQ,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;QAC7B,MAAM,MAAM,GAAG,MAAM,cAAc,CAAC,KAAK,CAAC,CAAC;QAC3C,IAAI,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;YACpB,MAAM,GAAG,GAAG,MAAM,CAAC,SAAS,EAAE,CAAC;YAC/B,MAAM,IAAI,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;QACD,OAAO,MAAM,CAAC,MAAM,EAAE,CAAC;IACxB,CAAC,CAAC;CACH,CAAC,CAAC"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { JobName } from "../../jobs.js";
|
|
2
|
+
type JobStatus = {
|
|
3
|
+
name: JobName;
|
|
4
|
+
interval: string;
|
|
5
|
+
lastExecution: string | null;
|
|
6
|
+
lastDuration: string | null;
|
|
7
|
+
nextExecution: string;
|
|
8
|
+
isActive: boolean;
|
|
9
|
+
canRunNow: boolean;
|
|
10
|
+
};
|
|
11
|
+
export declare const jobsRouter: import("@trpc/server").TRPCBuiltRouter<{
|
|
12
|
+
ctx: import("../index.js").Context;
|
|
13
|
+
meta: object;
|
|
14
|
+
errorShape: import("@trpc/server").TRPCDefaultErrorShape;
|
|
15
|
+
transformer: false;
|
|
16
|
+
}, import("@trpc/server").TRPCDecorateCreateRouterOptions<{
|
|
17
|
+
getJobStatuses: import("@trpc/server").TRPCQueryProcedure<{
|
|
18
|
+
input: void;
|
|
19
|
+
output: JobStatus[];
|
|
20
|
+
meta: object;
|
|
21
|
+
}>;
|
|
22
|
+
triggerJob: import("@trpc/server").TRPCMutationProcedure<{
|
|
23
|
+
input: {
|
|
24
|
+
name: JobName;
|
|
25
|
+
};
|
|
26
|
+
output: {
|
|
27
|
+
success: boolean;
|
|
28
|
+
message: string;
|
|
29
|
+
};
|
|
30
|
+
meta: object;
|
|
31
|
+
}>;
|
|
32
|
+
}>>;
|
|
33
|
+
export {};
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
import ms from "ms";
|
|
3
|
+
import { router, authedProcedure } from "../index.js";
|
|
4
|
+
import { getJobs, getJobLastRun, JobName, checkJobs } from "../../jobs.js";
|
|
5
|
+
import { Label, logger } from "../../logger.js";
|
|
6
|
+
import { humanReadableDate } from "../../utils.js";
|
|
7
|
+
import { db } from "../../db.js";
|
|
8
|
+
export const jobsRouter = router({
|
|
9
|
+
// Get all job statuses
|
|
10
|
+
getJobStatuses: authedProcedure.query(async () => {
|
|
11
|
+
const jobs = getJobs();
|
|
12
|
+
const now = Date.now();
|
|
13
|
+
// Fetch all job last run times in a single query
|
|
14
|
+
const jobNames = jobs.map((job) => job.name);
|
|
15
|
+
const lastRunData = await db("job_log")
|
|
16
|
+
.select("name", "last_run")
|
|
17
|
+
.whereIn("name", jobNames);
|
|
18
|
+
// Create a map for quick lookups
|
|
19
|
+
const lastRunMap = new Map(lastRunData.map((row) => [row.name, row.last_run]));
|
|
20
|
+
const jobStatuses = jobs.map((job) => {
|
|
21
|
+
const lastRun = lastRunMap.get(job.name);
|
|
22
|
+
const eligibilityTs = lastRun ? lastRun + job.cadence : now;
|
|
23
|
+
// Format interval (cadence) as human readable
|
|
24
|
+
const interval = ms(job.cadence, { long: true });
|
|
25
|
+
// Format last execution - return timestamp for frontend to format
|
|
26
|
+
const lastExecution = lastRun
|
|
27
|
+
? new Date(lastRun).toISOString()
|
|
28
|
+
: null;
|
|
29
|
+
// Format next execution - return timestamp for frontend to format
|
|
30
|
+
const nextExecution = now >= eligibilityTs
|
|
31
|
+
? "now"
|
|
32
|
+
: new Date(eligibilityTs).toISOString();
|
|
33
|
+
// Check if job can run now (not active and eligible for early run)
|
|
34
|
+
// Jobs can run "ahead of schedule" if current time >= lastRun time
|
|
35
|
+
// Note: lastRun can be in the future when delayNextRun was used
|
|
36
|
+
const canRunNow = !job.isActive && (lastRun === undefined || now >= lastRun);
|
|
37
|
+
return {
|
|
38
|
+
name: job.name,
|
|
39
|
+
interval,
|
|
40
|
+
lastExecution,
|
|
41
|
+
lastDuration: null, // TODO: Add duration tracking in future
|
|
42
|
+
nextExecution,
|
|
43
|
+
isActive: job.isActive,
|
|
44
|
+
canRunNow,
|
|
45
|
+
};
|
|
46
|
+
});
|
|
47
|
+
return jobStatuses;
|
|
48
|
+
}),
|
|
49
|
+
// Trigger a job to run ahead of schedule
|
|
50
|
+
triggerJob: authedProcedure
|
|
51
|
+
.input(z.object({
|
|
52
|
+
name: z.nativeEnum(JobName),
|
|
53
|
+
}))
|
|
54
|
+
.mutation(async ({ input }) => {
|
|
55
|
+
const jobs = getJobs();
|
|
56
|
+
const job = jobs.find((j) => j.name === input.name);
|
|
57
|
+
if (!job) {
|
|
58
|
+
throw new Error(`${input.name}: unable to run, disabled in config`);
|
|
59
|
+
}
|
|
60
|
+
if (job.isActive) {
|
|
61
|
+
throw new Error(`${job.name}: already running`);
|
|
62
|
+
}
|
|
63
|
+
const lastRun = (await getJobLastRun(job.name)) ?? 0;
|
|
64
|
+
if (Date.now() < lastRun) {
|
|
65
|
+
throw new Error(`${job.name}: not eligible to run ahead of schedule, next scheduled run is at ${humanReadableDate(lastRun + job.cadence)} (triggering an early run is allowed after ${humanReadableDate(lastRun)})`);
|
|
66
|
+
}
|
|
67
|
+
job.runAheadOfSchedule = true;
|
|
68
|
+
if (job.name === JobName.SEARCH || job.name === JobName.RSS) {
|
|
69
|
+
job.delayNextRun = true;
|
|
70
|
+
}
|
|
71
|
+
// Clear any config overrides for manual runs
|
|
72
|
+
job.configOverride = {};
|
|
73
|
+
void checkJobs({ isFirstRun: false, useQueue: true });
|
|
74
|
+
logger.info({
|
|
75
|
+
label: Label.SCHEDULER,
|
|
76
|
+
message: `${job.name}: running ahead of schedule via web UI`,
|
|
77
|
+
});
|
|
78
|
+
return {
|
|
79
|
+
success: true,
|
|
80
|
+
message: `${job.name}: running ahead of schedule`,
|
|
81
|
+
};
|
|
82
|
+
}),
|
|
83
|
+
});
|
|
84
|
+
//# sourceMappingURL=jobs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"jobs.js","sourceRoot":"","sources":["../../../src/trpc/routers/jobs.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { type LogEntry } from "../../utils/logWatcher.js";
|
|
2
|
+
export declare const logsRouter: 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
|
+
getVerbose: import("@trpc/server").TRPCQueryProcedure<{
|
|
9
|
+
input: void;
|
|
10
|
+
output: string;
|
|
11
|
+
meta: object;
|
|
12
|
+
}>;
|
|
13
|
+
getRecentLogs: import("@trpc/server").TRPCQueryProcedure<{
|
|
14
|
+
input: {
|
|
15
|
+
limit?: number | undefined;
|
|
16
|
+
};
|
|
17
|
+
output: LogEntry[];
|
|
18
|
+
meta: object;
|
|
19
|
+
}>;
|
|
20
|
+
subscribe: import("@trpc/server").TRPCSubscriptionProcedure<{
|
|
21
|
+
input: {
|
|
22
|
+
limit?: number | undefined;
|
|
23
|
+
};
|
|
24
|
+
output: AsyncIterable<LogEntry, never, any>;
|
|
25
|
+
meta: object;
|
|
26
|
+
}>;
|
|
27
|
+
}>>;
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
import { authedProcedure, router } from "../index.js";
|
|
3
|
+
import { Label, logger } from "../../logger.js";
|
|
4
|
+
import { getLogWatcher } from "../../utils/logWatcher.js";
|
|
5
|
+
export const logsRouter = router({
|
|
6
|
+
getVerbose: authedProcedure.query(async () => {
|
|
7
|
+
try {
|
|
8
|
+
const logWatcher = getLogWatcher();
|
|
9
|
+
const logs = await logWatcher.getRecentLogs(1000);
|
|
10
|
+
// Convert back to text format for compatibility
|
|
11
|
+
return logs
|
|
12
|
+
.map((log) => `${log.timestamp} ${log.level}: ${log.label ? `[${log.label}] ` : ""}${log.message}`)
|
|
13
|
+
.join("\n");
|
|
14
|
+
}
|
|
15
|
+
catch (error) {
|
|
16
|
+
logger.error({
|
|
17
|
+
label: Label.SERVER,
|
|
18
|
+
message: `Failed to read verbose logs: ${error.message}`,
|
|
19
|
+
});
|
|
20
|
+
throw new Error(`Failed to read verbose logs: ${error.message}`);
|
|
21
|
+
}
|
|
22
|
+
}),
|
|
23
|
+
getRecentLogs: authedProcedure
|
|
24
|
+
.input(z.object({
|
|
25
|
+
limit: z.number().min(1).max(1000).default(100),
|
|
26
|
+
}))
|
|
27
|
+
.query(async ({ input }) => {
|
|
28
|
+
try {
|
|
29
|
+
const logWatcher = getLogWatcher();
|
|
30
|
+
const logs = await logWatcher.getRecentLogs(input.limit);
|
|
31
|
+
return logs.reverse(); // Return newest first
|
|
32
|
+
}
|
|
33
|
+
catch (error) {
|
|
34
|
+
logger.error({
|
|
35
|
+
label: Label.SERVER,
|
|
36
|
+
message: `Failed to read logs: ${error.message}`,
|
|
37
|
+
});
|
|
38
|
+
throw new Error(`Failed to read logs: ${error.message}`);
|
|
39
|
+
}
|
|
40
|
+
}),
|
|
41
|
+
subscribe: authedProcedure
|
|
42
|
+
.input(z.object({
|
|
43
|
+
limit: z.number().min(1).max(500).default(100),
|
|
44
|
+
}))
|
|
45
|
+
.subscription(async function* ({ input }) {
|
|
46
|
+
const logQueue = [];
|
|
47
|
+
let resolve = null;
|
|
48
|
+
// First, emit historical logs
|
|
49
|
+
try {
|
|
50
|
+
const logWatcher = getLogWatcher();
|
|
51
|
+
const historicalLogs = await logWatcher.getRecentLogs(input.limit);
|
|
52
|
+
// Emit each historical log
|
|
53
|
+
for (const log of historicalLogs) {
|
|
54
|
+
// reverse to get oldest first
|
|
55
|
+
yield log;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
catch (error) {
|
|
59
|
+
logger.error({
|
|
60
|
+
label: Label.SERVER,
|
|
61
|
+
message: `Failed to load historical logs: ${error.message}`,
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
// Then set up real-time streaming
|
|
65
|
+
const logWatcher = getLogWatcher();
|
|
66
|
+
const unsubscribe = logWatcher.subscribe((logEntry) => {
|
|
67
|
+
logQueue.push(logEntry);
|
|
68
|
+
if (resolve) {
|
|
69
|
+
resolve();
|
|
70
|
+
resolve = null;
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
try {
|
|
74
|
+
// Stream new logs indefinitely
|
|
75
|
+
while (true) {
|
|
76
|
+
if (logQueue.length > 0) {
|
|
77
|
+
yield logQueue.shift();
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
await new Promise((res) => {
|
|
81
|
+
resolve = res;
|
|
82
|
+
});
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
finally {
|
|
87
|
+
unsubscribe();
|
|
88
|
+
}
|
|
89
|
+
}),
|
|
90
|
+
});
|
|
91
|
+
//# sourceMappingURL=logs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logs.js","sourceRoot":"","sources":["../../../src/trpc/routers/logs.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Label } from "../../logger.js";
|
|
2
|
+
export declare const searcheesRouter: 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
|
+
list: import("@trpc/server").TRPCQueryProcedure<{
|
|
9
|
+
input: {
|
|
10
|
+
search?: string | undefined;
|
|
11
|
+
limit?: number | undefined;
|
|
12
|
+
offset?: number | undefined;
|
|
13
|
+
} | undefined;
|
|
14
|
+
output: {
|
|
15
|
+
total: number;
|
|
16
|
+
pagination: {
|
|
17
|
+
limit: number;
|
|
18
|
+
offset: number;
|
|
19
|
+
};
|
|
20
|
+
indexerTotals: {
|
|
21
|
+
configured: number;
|
|
22
|
+
enabled: number;
|
|
23
|
+
};
|
|
24
|
+
items: {
|
|
25
|
+
id: string | number;
|
|
26
|
+
name: string;
|
|
27
|
+
indexerCount: number;
|
|
28
|
+
firstSearchedAt: string | null;
|
|
29
|
+
lastSearchedAt: string | null;
|
|
30
|
+
label: Label.SEARCH | Label.RSS | Label.ANNOUNCE | Label.WEBHOOK | Label.INJECT | null;
|
|
31
|
+
source: import("../../searchee.js").SearcheeSource | null;
|
|
32
|
+
length: number | null;
|
|
33
|
+
clientHost: string | null;
|
|
34
|
+
}[];
|
|
35
|
+
};
|
|
36
|
+
meta: object;
|
|
37
|
+
}>;
|
|
38
|
+
bulkSearch: import("@trpc/server").TRPCMutationProcedure<{
|
|
39
|
+
input: {
|
|
40
|
+
names: string[];
|
|
41
|
+
force?: boolean | undefined;
|
|
42
|
+
};
|
|
43
|
+
output: {
|
|
44
|
+
requested: number;
|
|
45
|
+
attempted: number;
|
|
46
|
+
totalFound: number;
|
|
47
|
+
skipped: number;
|
|
48
|
+
};
|
|
49
|
+
meta: object;
|
|
50
|
+
}>;
|
|
51
|
+
}>>;
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
import { TRPCError } from "@trpc/server";
|
|
3
|
+
import { router, authedProcedure } from "../index.js";
|
|
4
|
+
import { db } from "../../db.js";
|
|
5
|
+
import { findAllSearchees, bulkSearchByNames } from "../../pipeline.js";
|
|
6
|
+
import { Label } from "../../logger.js";
|
|
7
|
+
import { getSearcheeSource } from "../../searchee.js";
|
|
8
|
+
import { getAllIndexers, getEnabledIndexers } from "../../indexers.js";
|
|
9
|
+
const DEFAULT_LIMIT = 50;
|
|
10
|
+
const MAX_BULK_SEARCH = 20;
|
|
11
|
+
export const searcheesRouter = router({
|
|
12
|
+
list: authedProcedure
|
|
13
|
+
.input(z
|
|
14
|
+
.object({
|
|
15
|
+
search: z.string().trim().min(1).max(200).optional(),
|
|
16
|
+
limit: z
|
|
17
|
+
.number()
|
|
18
|
+
.int()
|
|
19
|
+
.min(1)
|
|
20
|
+
.max(200)
|
|
21
|
+
.default(DEFAULT_LIMIT),
|
|
22
|
+
offset: z.number().int().min(0).default(0),
|
|
23
|
+
})
|
|
24
|
+
.default({
|
|
25
|
+
limit: DEFAULT_LIMIT,
|
|
26
|
+
offset: 0,
|
|
27
|
+
}))
|
|
28
|
+
.query(async ({ input }) => {
|
|
29
|
+
const { search, limit, offset } = input;
|
|
30
|
+
let query = db("searchee");
|
|
31
|
+
if (search) {
|
|
32
|
+
query = query.where("searchee.name", "like", `%${search}%`);
|
|
33
|
+
}
|
|
34
|
+
const [{ count }] = await query
|
|
35
|
+
.clone()
|
|
36
|
+
.count({ count: "*" });
|
|
37
|
+
const rows = await query
|
|
38
|
+
.clone()
|
|
39
|
+
.leftJoin("timestamp", "searchee.id", "timestamp.searchee_id")
|
|
40
|
+
.select({
|
|
41
|
+
dbId: "searchee.id",
|
|
42
|
+
name: "searchee.name",
|
|
43
|
+
})
|
|
44
|
+
.min({
|
|
45
|
+
firstSearched: "timestamp.first_searched",
|
|
46
|
+
})
|
|
47
|
+
.max({
|
|
48
|
+
lastSearched: "timestamp.last_searched",
|
|
49
|
+
})
|
|
50
|
+
.countDistinct({
|
|
51
|
+
indexerCount: "timestamp.indexer_id",
|
|
52
|
+
})
|
|
53
|
+
.groupBy("searchee.id", "searchee.name")
|
|
54
|
+
.orderBy("lastSearched", "desc")
|
|
55
|
+
.orderBy("firstSearched", "desc")
|
|
56
|
+
.orderBy("name", "asc")
|
|
57
|
+
.limit(limit)
|
|
58
|
+
.offset(offset);
|
|
59
|
+
const [allSearchees, allIndexers, enabledIndexers] = await Promise.all([
|
|
60
|
+
findAllSearchees(Label.SEARCH),
|
|
61
|
+
getAllIndexers(),
|
|
62
|
+
getEnabledIndexers(),
|
|
63
|
+
]);
|
|
64
|
+
// TODO: drop the in-memory metadata lookup when searchee rows expose persistent IDs directly via TRPC.
|
|
65
|
+
const searcheeMeta = new Map();
|
|
66
|
+
for (const searchee of allSearchees) {
|
|
67
|
+
const keyCandidates = [searchee.title, searchee.name].filter((candidate) => typeof candidate === "string" && candidate.length > 0);
|
|
68
|
+
for (const key of keyCandidates) {
|
|
69
|
+
if (!searcheeMeta.has(key)) {
|
|
70
|
+
searcheeMeta.set(key, searchee);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
const items = rows.map((row) => {
|
|
75
|
+
const displayName =
|
|
76
|
+
// @ts-expect-error something funky going on with knex types
|
|
77
|
+
typeof row.name === "string" && row.name.length
|
|
78
|
+
? row.name
|
|
79
|
+
: "(unknown)";
|
|
80
|
+
const meta = searcheeMeta.get(displayName) ??
|
|
81
|
+
searcheeMeta.get(displayName.trim());
|
|
82
|
+
const idCandidate = (typeof row.dbId === "number"
|
|
83
|
+
? row.dbId
|
|
84
|
+
: row.dbId != null
|
|
85
|
+
? Number(row.dbId)
|
|
86
|
+
: null) ??
|
|
87
|
+
meta?.infoHash ??
|
|
88
|
+
meta?.path ??
|
|
89
|
+
displayName;
|
|
90
|
+
return {
|
|
91
|
+
id: typeof idCandidate === "number"
|
|
92
|
+
? idCandidate
|
|
93
|
+
: String(idCandidate),
|
|
94
|
+
name: displayName,
|
|
95
|
+
indexerCount: typeof row.indexerCount === "number"
|
|
96
|
+
? row.indexerCount
|
|
97
|
+
: Number(row.indexerCount ?? 0),
|
|
98
|
+
firstSearchedAt: row.firstSearched != null
|
|
99
|
+
? new Date(Number(row.firstSearched)).toISOString()
|
|
100
|
+
: null,
|
|
101
|
+
lastSearchedAt: row.lastSearched != null
|
|
102
|
+
? new Date(Number(row.lastSearched)).toISOString()
|
|
103
|
+
: null,
|
|
104
|
+
label: meta?.label ?? null,
|
|
105
|
+
source: meta ? getSearcheeSource(meta) : null,
|
|
106
|
+
length: meta?.length ?? null,
|
|
107
|
+
clientHost: meta?.clientHost ?? null,
|
|
108
|
+
};
|
|
109
|
+
});
|
|
110
|
+
return {
|
|
111
|
+
total: Number(count ?? 0),
|
|
112
|
+
pagination: {
|
|
113
|
+
limit,
|
|
114
|
+
offset,
|
|
115
|
+
},
|
|
116
|
+
indexerTotals: {
|
|
117
|
+
configured: allIndexers.length,
|
|
118
|
+
enabled: enabledIndexers.length,
|
|
119
|
+
},
|
|
120
|
+
items,
|
|
121
|
+
};
|
|
122
|
+
}),
|
|
123
|
+
bulkSearch: authedProcedure
|
|
124
|
+
.input(z.object({
|
|
125
|
+
names: z
|
|
126
|
+
.array(z.string().trim().min(1).max(500))
|
|
127
|
+
.min(1, "Select at least one item")
|
|
128
|
+
.max(MAX_BULK_SEARCH, `You can only bulk search up to ${MAX_BULK_SEARCH} items at a time`),
|
|
129
|
+
force: z.boolean().optional().default(false),
|
|
130
|
+
}))
|
|
131
|
+
.mutation(async ({ input }) => {
|
|
132
|
+
const uniqueNames = Array.from(new Set(input.names
|
|
133
|
+
.map((name) => name.trim())
|
|
134
|
+
.filter((name) => name.length > 0)));
|
|
135
|
+
if (!uniqueNames.length) {
|
|
136
|
+
throw new TRPCError({
|
|
137
|
+
code: "BAD_REQUEST",
|
|
138
|
+
message: "No valid item names provided for bulk search",
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
const configOverride = input.force
|
|
142
|
+
? {
|
|
143
|
+
excludeRecentSearch: 1,
|
|
144
|
+
excludeOlder: Number.MAX_SAFE_INTEGER,
|
|
145
|
+
}
|
|
146
|
+
: undefined;
|
|
147
|
+
const { attempted, totalFound, requested } = await bulkSearchByNames(uniqueNames, configOverride ? { configOverride } : undefined);
|
|
148
|
+
return {
|
|
149
|
+
requested,
|
|
150
|
+
attempted,
|
|
151
|
+
totalFound,
|
|
152
|
+
skipped: Math.max(requested - attempted, 0),
|
|
153
|
+
};
|
|
154
|
+
}),
|
|
155
|
+
});
|
|
156
|
+
//# sourceMappingURL=searchees.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"searchees.js","sourceRoot":"","sources":["../../../src/trpc/routers/searchees.ts"],"names":[],"mappings":"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"}
|