cross-seed 7.0.0-1 → 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.
Files changed (229) hide show
  1. package/dist/Result.d.ts +27 -0
  2. package/dist/Result.js +64 -0
  3. package/dist/Result.js.map +1 -0
  4. package/dist/action.d.ts +34 -0
  5. package/dist/action.js +694 -0
  6. package/dist/action.js.map +1 -0
  7. package/dist/arr.d.ts +31 -0
  8. package/dist/arr.js +267 -0
  9. package/dist/arr.js.map +1 -0
  10. package/dist/auth.d.ts +3 -0
  11. package/dist/auth.js +28 -0
  12. package/dist/auth.js.map +1 -0
  13. package/dist/clients/Deluge.d.ts +153 -0
  14. package/dist/clients/Deluge.js +698 -0
  15. package/dist/clients/Deluge.js.map +1 -0
  16. package/dist/clients/QBittorrent.d.ts +218 -0
  17. package/dist/clients/QBittorrent.js +785 -0
  18. package/dist/clients/QBittorrent.js.map +1 -0
  19. package/dist/clients/RTorrent.d.ts +43 -0
  20. package/dist/clients/RTorrent.js +657 -0
  21. package/dist/clients/RTorrent.js.map +1 -0
  22. package/dist/clients/TorrentClient.d.ts +108 -0
  23. package/dist/clients/TorrentClient.js +341 -0
  24. package/dist/clients/TorrentClient.js.map +1 -0
  25. package/dist/clients/Transmission.d.ts +43 -0
  26. package/dist/clients/Transmission.js +404 -0
  27. package/dist/clients/Transmission.js.map +1 -0
  28. package/dist/cmd.d.ts +2 -0
  29. package/dist/cmd.js +128 -0
  30. package/dist/cmd.js.map +1 -0
  31. package/dist/configSchema.d.ts +1 -0
  32. package/dist/configSchema.js +2 -0
  33. package/dist/configSchema.js.map +1 -0
  34. package/dist/configuration.d.ts +63 -0
  35. package/dist/configuration.js +321 -0
  36. package/dist/configuration.js.map +1 -0
  37. package/dist/constants.d.ts +108 -0
  38. package/dist/constants.js +251 -0
  39. package/dist/constants.js.map +1 -0
  40. package/dist/dataFiles.d.ts +8 -0
  41. package/dist/dataFiles.js +223 -0
  42. package/dist/dataFiles.js.map +1 -0
  43. package/dist/db.d.ts +3 -0
  44. package/dist/db.js +216 -0
  45. package/dist/db.js.map +1 -0
  46. package/dist/dbConfig.d.ts +4 -0
  47. package/dist/dbConfig.js +67 -0
  48. package/dist/dbConfig.js.map +1 -0
  49. package/dist/decide.d.ts +25 -0
  50. package/dist/decide.js +553 -0
  51. package/dist/decide.js.map +1 -0
  52. package/dist/diff.d.ts +1 -0
  53. package/dist/diff.js +24 -0
  54. package/dist/diff.js.map +1 -0
  55. package/dist/errors.d.ts +3 -0
  56. package/dist/errors.js +7 -0
  57. package/dist/errors.js.map +1 -0
  58. package/dist/indexers.d.ts +105 -0
  59. package/dist/indexers.js +248 -0
  60. package/dist/indexers.js.map +1 -0
  61. package/dist/inject.d.ts +2 -0
  62. package/dist/inject.js +594 -0
  63. package/dist/inject.js.map +1 -0
  64. package/dist/jobs.d.ts +29 -0
  65. package/dist/jobs.js +151 -0
  66. package/dist/jobs.js.map +1 -0
  67. package/dist/logger.d.ts +29 -0
  68. package/dist/logger.js +157 -0
  69. package/dist/logger.js.map +1 -0
  70. package/dist/migrations/00-initialSchema.d.ts +9 -0
  71. package/dist/migrations/00-initialSchema.js +30 -0
  72. package/dist/migrations/00-initialSchema.js.map +1 -0
  73. package/dist/migrations/01-jobs.d.ts +9 -0
  74. package/dist/migrations/01-jobs.js +12 -0
  75. package/dist/migrations/01-jobs.js.map +1 -0
  76. package/dist/migrations/02-timestamps.d.ts +9 -0
  77. package/dist/migrations/02-timestamps.js +21 -0
  78. package/dist/migrations/02-timestamps.js.map +1 -0
  79. package/dist/migrations/03-rateLimits.d.ts +9 -0
  80. package/dist/migrations/03-rateLimits.js +14 -0
  81. package/dist/migrations/03-rateLimits.js.map +1 -0
  82. package/dist/migrations/04-auth.d.ts +9 -0
  83. package/dist/migrations/04-auth.js +13 -0
  84. package/dist/migrations/04-auth.js.map +1 -0
  85. package/dist/migrations/05-caps.d.ts +9 -0
  86. package/dist/migrations/05-caps.js +16 -0
  87. package/dist/migrations/05-caps.js.map +1 -0
  88. package/dist/migrations/06-uniqueDecisions.d.ts +9 -0
  89. package/dist/migrations/06-uniqueDecisions.js +29 -0
  90. package/dist/migrations/06-uniqueDecisions.js.map +1 -0
  91. package/dist/migrations/07-limits.d.ts +9 -0
  92. package/dist/migrations/07-limits.js +12 -0
  93. package/dist/migrations/07-limits.js.map +1 -0
  94. package/dist/migrations/08-rss.d.ts +9 -0
  95. package/dist/migrations/08-rss.js +15 -0
  96. package/dist/migrations/08-rss.js.map +1 -0
  97. package/dist/migrations/09-clientAndDataSearchees.d.ts +9 -0
  98. package/dist/migrations/09-clientAndDataSearchees.js +34 -0
  99. package/dist/migrations/09-clientAndDataSearchees.js.map +1 -0
  100. package/dist/migrations/10-indexerNameAudioBookCaps.d.ts +9 -0
  101. package/dist/migrations/10-indexerNameAudioBookCaps.js +18 -0
  102. package/dist/migrations/10-indexerNameAudioBookCaps.js.map +1 -0
  103. package/dist/migrations/11-trackers.d.ts +9 -0
  104. package/dist/migrations/11-trackers.js +38 -0
  105. package/dist/migrations/11-trackers.js.map +1 -0
  106. package/dist/migrations/12-user-auth.d.ts +9 -0
  107. package/dist/migrations/12-user-auth.js +22 -0
  108. package/dist/migrations/12-user-auth.js.map +1 -0
  109. package/dist/migrations/13-settings.d.ts +9 -0
  110. package/dist/migrations/13-settings.js +23 -0
  111. package/dist/migrations/13-settings.js.map +1 -0
  112. package/dist/migrations/14-indexer-enabled-flag.d.ts +9 -0
  113. package/dist/migrations/14-indexer-enabled-flag.js +12 -0
  114. package/dist/migrations/14-indexer-enabled-flag.js.map +1 -0
  115. package/dist/migrations/15-remove-url-unique-constraint.d.ts +9 -0
  116. package/dist/migrations/15-remove-url-unique-constraint.js +14 -0
  117. package/dist/migrations/15-remove-url-unique-constraint.js.map +1 -0
  118. package/dist/migrations/16-prune-inactive-indexers.d.ts +9 -0
  119. package/dist/migrations/16-prune-inactive-indexers.js +17 -0
  120. package/dist/migrations/16-prune-inactive-indexers.js.map +1 -0
  121. package/dist/migrations/migrations.d.ts +13 -0
  122. package/dist/migrations/migrations.js +41 -0
  123. package/dist/migrations/migrations.js.map +1 -0
  124. package/dist/parseTorrent.d.ts +53 -0
  125. package/dist/parseTorrent.js +128 -0
  126. package/dist/parseTorrent.js.map +1 -0
  127. package/dist/pipeline.d.ts +41 -0
  128. package/dist/pipeline.js +574 -0
  129. package/dist/pipeline.js.map +1 -0
  130. package/dist/preFilter.d.ts +25 -0
  131. package/dist/preFilter.js +250 -0
  132. package/dist/preFilter.js.map +1 -0
  133. package/dist/problems/linking.d.ts +2 -0
  134. package/dist/problems/linking.js +80 -0
  135. package/dist/problems/linking.js.map +1 -0
  136. package/dist/problems/path.d.ts +22 -0
  137. package/dist/problems/path.js +96 -0
  138. package/dist/problems/path.js.map +1 -0
  139. package/dist/problems.d.ts +13 -0
  140. package/dist/problems.js +48 -0
  141. package/dist/problems.js.map +1 -0
  142. package/dist/pushNotifier.d.ts +19 -0
  143. package/dist/pushNotifier.js +137 -0
  144. package/dist/pushNotifier.js.map +1 -0
  145. package/dist/routes/baseApi.d.ts +2 -0
  146. package/dist/routes/baseApi.js +354 -0
  147. package/dist/routes/baseApi.js.map +1 -0
  148. package/dist/routes/indexerApi.d.ts +6 -0
  149. package/dist/routes/indexerApi.js +165 -0
  150. package/dist/routes/indexerApi.js.map +1 -0
  151. package/dist/routes/staticFrontendPlugin.d.ts +4 -0
  152. package/dist/routes/staticFrontendPlugin.js +61 -0
  153. package/dist/routes/staticFrontendPlugin.js.map +1 -0
  154. package/dist/runtimeConfig.d.ts +6 -0
  155. package/dist/runtimeConfig.js +27 -0
  156. package/dist/runtimeConfig.js.map +1 -0
  157. package/dist/searchee.d.ts +108 -0
  158. package/dist/searchee.js +689 -0
  159. package/dist/searchee.js.map +1 -0
  160. package/dist/server.d.ts +4 -0
  161. package/dist/server.js +65 -0
  162. package/dist/server.js.map +1 -0
  163. package/dist/services/indexerService.d.ts +96 -0
  164. package/dist/services/indexerService.js +287 -0
  165. package/dist/services/indexerService.js.map +1 -0
  166. package/dist/sessionCookies.d.ts +5 -0
  167. package/dist/sessionCookies.js +27 -0
  168. package/dist/sessionCookies.js.map +1 -0
  169. package/dist/startup.d.ts +25 -0
  170. package/dist/startup.js +157 -0
  171. package/dist/startup.js.map +1 -0
  172. package/dist/torrent.d.ts +69 -0
  173. package/dist/torrent.js +641 -0
  174. package/dist/torrent.js.map +1 -0
  175. package/dist/torznab.d.ts +60 -0
  176. package/dist/torznab.js +711 -0
  177. package/dist/torznab.js.map +1 -0
  178. package/dist/trpc/fastifyAdapter.d.ts +2 -0
  179. package/dist/trpc/fastifyAdapter.js +9 -0
  180. package/dist/trpc/fastifyAdapter.js.map +1 -0
  181. package/dist/trpc/index.d.ts +49 -0
  182. package/dist/trpc/index.js +53 -0
  183. package/dist/trpc/index.js.map +1 -0
  184. package/dist/trpc/routers/auth.d.ts +43 -0
  185. package/dist/trpc/routers/auth.js +116 -0
  186. package/dist/trpc/routers/auth.js.map +1 -0
  187. package/dist/trpc/routers/clients.d.ts +21 -0
  188. package/dist/trpc/routers/clients.js +65 -0
  189. package/dist/trpc/routers/clients.js.map +1 -0
  190. package/dist/trpc/routers/health.d.ts +14 -0
  191. package/dist/trpc/routers/health.js +20 -0
  192. package/dist/trpc/routers/health.js.map +1 -0
  193. package/dist/trpc/routers/index.d.ts +391 -0
  194. package/dist/trpc/routers/index.js +23 -0
  195. package/dist/trpc/routers/index.js.map +1 -0
  196. package/dist/trpc/routers/indexers.d.ts +75 -0
  197. package/dist/trpc/routers/indexers.js +79 -0
  198. package/dist/trpc/routers/indexers.js.map +1 -0
  199. package/dist/trpc/routers/jobs.d.ts +33 -0
  200. package/dist/trpc/routers/jobs.js +84 -0
  201. package/dist/trpc/routers/jobs.js.map +1 -0
  202. package/dist/trpc/routers/logs.d.ts +27 -0
  203. package/dist/trpc/routers/logs.js +91 -0
  204. package/dist/trpc/routers/logs.js.map +1 -0
  205. package/dist/trpc/routers/searchees.d.ts +51 -0
  206. package/dist/trpc/routers/searchees.js +156 -0
  207. package/dist/trpc/routers/searchees.js.map +1 -0
  208. package/dist/trpc/routers/settings.d.ts +83 -0
  209. package/dist/trpc/routers/settings.js +92 -0
  210. package/dist/trpc/routers/settings.js.map +1 -0
  211. package/dist/trpc/routers/stats.d.ts +42 -0
  212. package/dist/trpc/routers/stats.js +102 -0
  213. package/dist/trpc/routers/stats.js.map +1 -0
  214. package/dist/userAuth.d.ts +21 -0
  215. package/dist/userAuth.js +86 -0
  216. package/dist/userAuth.js.map +1 -0
  217. package/dist/utils/authUtils.d.ts +10 -0
  218. package/dist/utils/authUtils.js +24 -0
  219. package/dist/utils/authUtils.js.map +1 -0
  220. package/dist/utils/logWatcher.d.ts +28 -0
  221. package/dist/utils/logWatcher.js +218 -0
  222. package/dist/utils/logWatcher.js.map +1 -0
  223. package/dist/utils/object.d.ts +1 -0
  224. package/dist/utils/object.js +4 -0
  225. package/dist/utils/object.js.map +1 -0
  226. package/dist/utils.d.ts +175 -0
  227. package/dist/utils.js +660 -0
  228. package/dist/utils.js.map +1 -0
  229. package/package.json +2 -2
@@ -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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"}
@@ -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
+ }>>;