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.
Files changed (247) hide show
  1. package/README.md +12 -13
  2. package/dist/Result.d.ts +27 -0
  3. package/dist/action.d.ts +34 -0
  4. package/dist/action.js +2 -1
  5. package/dist/action.js.map +1 -1
  6. package/dist/arr.d.ts +31 -0
  7. package/dist/arr.js +107 -39
  8. package/dist/arr.js.map +1 -1
  9. package/dist/auth.d.ts +3 -0
  10. package/dist/auth.js +9 -6
  11. package/dist/auth.js.map +1 -1
  12. package/dist/clients/Deluge.d.ts +153 -0
  13. package/dist/clients/Deluge.js +6 -6
  14. package/dist/clients/Deluge.js.map +1 -1
  15. package/dist/clients/QBittorrent.d.ts +218 -0
  16. package/dist/clients/RTorrent.d.ts +43 -0
  17. package/dist/clients/RTorrent.js +6 -3
  18. package/dist/clients/RTorrent.js.map +1 -1
  19. package/dist/clients/TorrentClient.d.ts +108 -0
  20. package/dist/clients/TorrentClient.js +136 -67
  21. package/dist/clients/TorrentClient.js.map +1 -1
  22. package/dist/clients/Transmission.d.ts +43 -0
  23. package/dist/clients/Transmission.js +2 -2
  24. package/dist/clients/Transmission.js.map +1 -1
  25. package/dist/cmd.d.ts +2 -0
  26. package/dist/cmd.js +42 -110
  27. package/dist/cmd.js.map +1 -1
  28. package/dist/configSchema.d.ts +1 -0
  29. package/dist/configSchema.js +1 -666
  30. package/dist/configSchema.js.map +1 -1
  31. package/dist/configuration.d.ts +63 -0
  32. package/dist/configuration.js +263 -24
  33. package/dist/configuration.js.map +1 -1
  34. package/dist/constants.d.ts +108 -0
  35. package/dist/constants.js +2 -32
  36. package/dist/constants.js.map +1 -1
  37. package/dist/dataFiles.d.ts +8 -0
  38. package/dist/dataFiles.js +21 -6
  39. package/dist/dataFiles.js.map +1 -1
  40. package/dist/db.d.ts +3 -0
  41. package/dist/dbConfig.d.ts +4 -0
  42. package/dist/dbConfig.js +67 -0
  43. package/dist/dbConfig.js.map +1 -0
  44. package/dist/decide.d.ts +25 -0
  45. package/dist/decide.js +4 -4
  46. package/dist/decide.js.map +1 -1
  47. package/dist/diff.d.ts +1 -0
  48. package/dist/errors.d.ts +3 -0
  49. package/dist/errors.js +0 -9
  50. package/dist/errors.js.map +1 -1
  51. package/dist/indexers.d.ts +105 -0
  52. package/dist/indexers.js +82 -14
  53. package/dist/indexers.js.map +1 -1
  54. package/dist/inject.d.ts +2 -0
  55. package/dist/jobs.d.ts +29 -0
  56. package/dist/jobs.js +14 -9
  57. package/dist/jobs.js.map +1 -1
  58. package/dist/logger.d.ts +29 -0
  59. package/dist/logger.js +18 -4
  60. package/dist/logger.js.map +1 -1
  61. package/dist/migrations/00-initialSchema.d.ts +9 -0
  62. package/dist/migrations/01-jobs.d.ts +9 -0
  63. package/dist/migrations/02-timestamps.d.ts +9 -0
  64. package/dist/migrations/03-rateLimits.d.ts +9 -0
  65. package/dist/migrations/04-auth.d.ts +9 -0
  66. package/dist/migrations/04-auth.js +1 -1
  67. package/dist/migrations/04-auth.js.map +1 -1
  68. package/dist/migrations/05-caps.d.ts +9 -0
  69. package/dist/migrations/06-uniqueDecisions.d.ts +9 -0
  70. package/dist/migrations/07-limits.d.ts +9 -0
  71. package/dist/migrations/08-rss.d.ts +9 -0
  72. package/dist/migrations/09-clientAndDataSearchees.d.ts +9 -0
  73. package/dist/migrations/10-indexerNameAudioBookCaps.d.ts +9 -0
  74. package/dist/migrations/11-trackers.d.ts +9 -0
  75. package/dist/migrations/12-user-auth.d.ts +9 -0
  76. package/dist/migrations/12-user-auth.js +22 -0
  77. package/dist/migrations/12-user-auth.js.map +1 -0
  78. package/dist/migrations/13-settings.d.ts +9 -0
  79. package/dist/migrations/13-settings.js +23 -0
  80. package/dist/migrations/13-settings.js.map +1 -0
  81. package/dist/migrations/14-indexer-enabled-flag.d.ts +9 -0
  82. package/dist/migrations/14-indexer-enabled-flag.js +12 -0
  83. package/dist/migrations/14-indexer-enabled-flag.js.map +1 -0
  84. package/dist/migrations/15-remove-url-unique-constraint.d.ts +9 -0
  85. package/dist/migrations/15-remove-url-unique-constraint.js +14 -0
  86. package/dist/migrations/15-remove-url-unique-constraint.js.map +1 -0
  87. package/dist/migrations/16-prune-inactive-indexers.d.ts +9 -0
  88. package/dist/migrations/16-prune-inactive-indexers.js +17 -0
  89. package/dist/migrations/16-prune-inactive-indexers.js.map +1 -0
  90. package/dist/migrations/migrations.d.ts +13 -0
  91. package/dist/migrations/migrations.js +10 -0
  92. package/dist/migrations/migrations.js.map +1 -1
  93. package/dist/parseTorrent.d.ts +53 -0
  94. package/dist/pipeline.d.ts +41 -0
  95. package/dist/pipeline.js +57 -10
  96. package/dist/pipeline.js.map +1 -1
  97. package/dist/preFilter.d.ts +25 -0
  98. package/dist/preFilter.js +15 -15
  99. package/dist/preFilter.js.map +1 -1
  100. package/dist/problems/linking.d.ts +2 -0
  101. package/dist/problems/linking.js +80 -0
  102. package/dist/problems/linking.js.map +1 -0
  103. package/dist/problems/path.d.ts +22 -0
  104. package/dist/problems/path.js +96 -0
  105. package/dist/problems/path.js.map +1 -0
  106. package/dist/problems.d.ts +13 -0
  107. package/dist/problems.js +48 -0
  108. package/dist/problems.js.map +1 -0
  109. package/dist/pushNotifier.d.ts +19 -0
  110. package/dist/routes/baseApi.d.ts +2 -0
  111. package/dist/routes/baseApi.js +354 -0
  112. package/dist/routes/baseApi.js.map +1 -0
  113. package/dist/routes/indexerApi.d.ts +6 -0
  114. package/dist/routes/indexerApi.js +165 -0
  115. package/dist/routes/indexerApi.js.map +1 -0
  116. package/dist/routes/staticFrontendPlugin.d.ts +4 -0
  117. package/dist/routes/staticFrontendPlugin.js +61 -0
  118. package/dist/routes/staticFrontendPlugin.js.map +1 -0
  119. package/dist/runtimeConfig.d.ts +6 -0
  120. package/dist/runtimeConfig.js +17 -1
  121. package/dist/runtimeConfig.js.map +1 -1
  122. package/dist/searchee.d.ts +108 -0
  123. package/dist/searchee.js +36 -5
  124. package/dist/searchee.js.map +1 -1
  125. package/dist/server.d.ts +4 -0
  126. package/dist/server.js +38 -429
  127. package/dist/server.js.map +1 -1
  128. package/dist/services/indexerService.d.ts +96 -0
  129. package/dist/services/indexerService.js +287 -0
  130. package/dist/services/indexerService.js.map +1 -0
  131. package/dist/sessionCookies.d.ts +5 -0
  132. package/dist/sessionCookies.js +27 -0
  133. package/dist/sessionCookies.js.map +1 -0
  134. package/dist/startup.d.ts +25 -0
  135. package/dist/startup.js +105 -151
  136. package/dist/startup.js.map +1 -1
  137. package/dist/torrent.d.ts +69 -0
  138. package/dist/torrent.js +17 -13
  139. package/dist/torrent.js.map +1 -1
  140. package/dist/torznab.d.ts +60 -0
  141. package/dist/torznab.js +14 -89
  142. package/dist/torznab.js.map +1 -1
  143. package/dist/trpc/fastifyAdapter.d.ts +2 -0
  144. package/dist/trpc/fastifyAdapter.js +9 -0
  145. package/dist/trpc/fastifyAdapter.js.map +1 -0
  146. package/dist/trpc/index.d.ts +49 -0
  147. package/dist/trpc/index.js +53 -0
  148. package/dist/trpc/index.js.map +1 -0
  149. package/dist/trpc/routers/auth.d.ts +43 -0
  150. package/dist/trpc/routers/auth.js +116 -0
  151. package/dist/trpc/routers/auth.js.map +1 -0
  152. package/dist/trpc/routers/clients.d.ts +21 -0
  153. package/dist/trpc/routers/clients.js +65 -0
  154. package/dist/trpc/routers/clients.js.map +1 -0
  155. package/dist/trpc/routers/health.d.ts +14 -0
  156. package/dist/trpc/routers/health.js +20 -0
  157. package/dist/trpc/routers/health.js.map +1 -0
  158. package/dist/trpc/routers/index.d.ts +391 -0
  159. package/dist/trpc/routers/index.js +23 -0
  160. package/dist/trpc/routers/index.js.map +1 -0
  161. package/dist/trpc/routers/indexers.d.ts +75 -0
  162. package/dist/trpc/routers/indexers.js +79 -0
  163. package/dist/trpc/routers/indexers.js.map +1 -0
  164. package/dist/trpc/routers/jobs.d.ts +33 -0
  165. package/dist/trpc/routers/jobs.js +84 -0
  166. package/dist/trpc/routers/jobs.js.map +1 -0
  167. package/dist/trpc/routers/logs.d.ts +27 -0
  168. package/dist/trpc/routers/logs.js +91 -0
  169. package/dist/trpc/routers/logs.js.map +1 -0
  170. package/dist/trpc/routers/searchees.d.ts +51 -0
  171. package/dist/trpc/routers/searchees.js +156 -0
  172. package/dist/trpc/routers/searchees.js.map +1 -0
  173. package/dist/trpc/routers/settings.d.ts +83 -0
  174. package/dist/trpc/routers/settings.js +92 -0
  175. package/dist/trpc/routers/settings.js.map +1 -0
  176. package/dist/trpc/routers/stats.d.ts +42 -0
  177. package/dist/trpc/routers/stats.js +102 -0
  178. package/dist/trpc/routers/stats.js.map +1 -0
  179. package/dist/userAuth.d.ts +21 -0
  180. package/dist/userAuth.js +86 -0
  181. package/dist/userAuth.js.map +1 -0
  182. package/dist/utils/authUtils.d.ts +10 -0
  183. package/dist/utils/authUtils.js +24 -0
  184. package/dist/utils/authUtils.js.map +1 -0
  185. package/dist/utils/logWatcher.d.ts +28 -0
  186. package/dist/utils/logWatcher.js +218 -0
  187. package/dist/utils/logWatcher.js.map +1 -0
  188. package/dist/utils/object.d.ts +1 -0
  189. package/dist/utils/object.js +4 -0
  190. package/dist/utils/object.js.map +1 -0
  191. package/dist/utils.d.ts +175 -0
  192. package/dist/utils.js +61 -38
  193. package/dist/utils.js.map +1 -1
  194. package/dist/webui/assets/FieldInfo-Bxj_j8SJ.js +1 -0
  195. package/dist/webui/assets/Page-C3rteCZt.js +1 -0
  196. package/dist/webui/assets/array-field-DVSC6nHP.js +1 -0
  197. package/dist/webui/assets/badge-DTZMtS0e.js +1 -0
  198. package/dist/webui/assets/check-Bu3ldi63.js +1 -0
  199. package/dist/webui/assets/chevron-down-CRy8M0kJ.js +1 -0
  200. package/dist/webui/assets/clients-CW8oEZoQ.js +1 -0
  201. package/dist/webui/assets/connect-YBNsnjWT.js +1 -0
  202. package/dist/webui/assets/debug-mz8-WYZj.js +1 -0
  203. package/dist/webui/assets/directories-BSK28RgR.js +1 -0
  204. package/dist/webui/assets/duration-field-C6xoSlJg.js +1 -0
  205. package/dist/webui/assets/general-lJJxZhH7.js +1 -0
  206. package/dist/webui/assets/health-CXbsVrie.js +1 -0
  207. package/dist/webui/assets/index-Bi48hI2z.js +54 -0
  208. package/dist/webui/assets/index-C-Ul7GNg.css +1 -0
  209. package/dist/webui/assets/index-C2cH1Gst.js +1 -0
  210. package/dist/webui/assets/index-Cc5bDmJr.js +1 -0
  211. package/dist/webui/assets/jobs-CxmNab9w.js +1 -0
  212. package/dist/webui/assets/library-vaj2W8sE.js +1 -0
  213. package/dist/webui/assets/loader-circle-M0gu1gZ-.js +1 -0
  214. package/dist/webui/assets/logs-Cu9RyKS0.js +1 -0
  215. package/dist/webui/assets/search-2R5sIdT8.js +1 -0
  216. package/dist/webui/assets/select-field-BCqNLDrJ.js +1 -0
  217. package/dist/webui/assets/select-zHgqMzLj.js +1 -0
  218. package/dist/webui/assets/settings-CMYjpTbZ.js +1 -0
  219. package/dist/webui/assets/submit-button-BtcnyggQ.js +1 -0
  220. package/dist/webui/assets/switch-G0W3uJVN.js +1 -0
  221. package/dist/webui/assets/switch-field-IBd9ORNq.js +1 -0
  222. package/dist/webui/assets/table-DvgJU7Gh.js +1 -0
  223. package/dist/webui/assets/test-tube-BIwmoM45.js +1 -0
  224. package/dist/webui/assets/text-field-DruSbGhy.js +1 -0
  225. package/dist/webui/assets/time-BSMZjmyW.js +1 -0
  226. package/dist/webui/assets/trackers-D-OpAe63.js +7 -0
  227. package/dist/webui/assets/use-form-validation-context-BkAfWAh0.js +1 -0
  228. package/dist/webui/assets/use-settings-form-submit-CDRh-E9U.js +2 -0
  229. package/dist/webui/assets/useQuery-A4Hv_4uX.js +1 -0
  230. package/dist/webui/index.html +13 -0
  231. package/node_modules/@cross-seed/shared/dist/configSchema.d.ts +261 -0
  232. package/node_modules/@cross-seed/shared/dist/configSchema.d.ts.map +1 -0
  233. package/node_modules/@cross-seed/shared/dist/configSchema.js +53 -0
  234. package/node_modules/@cross-seed/shared/dist/configSchema.js.map +1 -0
  235. package/node_modules/@cross-seed/shared/dist/constants.d.ts +122 -0
  236. package/node_modules/@cross-seed/shared/dist/constants.d.ts.map +1 -0
  237. package/node_modules/@cross-seed/shared/dist/constants.js +127 -0
  238. package/node_modules/@cross-seed/shared/dist/constants.js.map +1 -0
  239. package/node_modules/@cross-seed/shared/dist/tsconfig.tsbuildinfo +1 -0
  240. package/node_modules/@cross-seed/shared/dist/utils.d.ts +6 -0
  241. package/node_modules/@cross-seed/shared/dist/utils.d.ts.map +1 -0
  242. package/node_modules/@cross-seed/shared/dist/utils.js +9 -0
  243. package/node_modules/@cross-seed/shared/dist/utils.js.map +1 -0
  244. package/node_modules/@cross-seed/shared/package.json +22 -0
  245. package/package.json +35 -11
  246. package/dist/config.template.cjs +0 -353
  247. 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":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACtD,OAAO,EAAE,EAAE,EAAE,MAAM,aAAa,CAAC;AACjC,OAAO,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AACxD,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE1C,MAAM,CAAC,MAAM,WAAW,GAAG,MAAM,CAAC;IACjC,WAAW,EAAE,eAAe,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE;QAC7C,MAAM,CACL,cAAc,EACd,aAAa,EACb,kBAAkB,EAClB,oBAAoB,EACpB,eAAe,EACf,eAAe,EACf,aAAa,EACb,eAAe,EACf,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;YACrB,EAAE,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC,KAAK,EAAE;YAC5C,EAAE,CAAC,UAAU,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC;YAC7B,EAAE,CAAC,SAAS,CAAC,CAAC,KAAK,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC,KAAK,EAAE;YAC3C,EAAE,CAAC,SAAS,CAAC;iBACX,KAAK,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC;iBAC1C,KAAK,CAAC,CAAC,EAAE,EAAE,EAAE,CACb,EAAE;iBACA,SAAS,CAAC,QAAQ,CAAC;iBACnB,OAAO,CAAC,QAAQ,EAAE,IAAI,CAAC;iBACvB,OAAO,CAAC,aAAa,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,CACzC;iBACA,KAAK,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;iBACrB,KAAK,EAAE;YACT,EAAE,CAAC,WAAW,CAAC,CAAC,KAAK,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC,KAAK,EAAE;YAC7C,EAAE,CAAC,UAAU,CAAC;iBACZ,MAAM,CAAC,UAAU,CAAC;iBAClB,KAAK,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;iBACrB,OAAO,CAAC,UAAU,CAAC;YACrB,EAAE,CAAC,UAAU,CAAC;iBACZ,OAAO,CAAC,UAAU,EAAE;gBACpB,OAAO;gBACP,iBAAiB;gBACjB,eAAe;aACf,CAAC;iBACD,KAAK,CAAC,WAAW,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,WAAW;iBACrE,KAAK,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;iBACrB,KAAK,EAAE;YACT,EAAE,CAAC,UAAU,CAAC;iBACZ,YAAY,CAAC,WAAW,CAAC;iBACzB,MAAM,CAAC;gBACP,WAAW,EAAE,EAAE,CAAC,GAAG,CAAC,2BAA2B,CAAC;gBAChD,UAAU,EAAE,EAAE,CAAC,GAAG,CACjB,sGAAsG,CACtG;gBACD,sBAAsB,EAAE,EAAE,CAAC,GAAG,CAC7B,kJAAkJ,CAClJ;aACD,CAAC;iBACD,KAAK,EAAE;SACT,CAAC,CAAC;QAEH,MAAM,WAAW,GAAG,MAAM,CAAC,eAAe,EAAE,WAAW,IAAI,CAAC,CAAC,CAAC;QAC9D,MAAM,UAAU,GAAG,MAAM,CAAC,eAAe,EAAE,UAAU,IAAI,CAAC,CAAC,CAAC;QAC5D,MAAM,sBAAsB,GAAG,MAAM,CACpC,eAAe,EAAE,sBAAsB,IAAI,CAAC,CAC5C,CAAC;QACF,MAAM,YAAY,GAAG,UAAU,CAAC;QAChC,MAAM,gBAAgB,GAAG,aAAa;aACpC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC;aACtB,MAAM,CACN,CAAC,IAAI,EAAkB,EAAE,CACxB,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,CAC5C,CAAC;QACH,MAAM,UAAU,GAAG,IAAI,GAAG,CACzB,MAAM,QAAQ,CAAC,gBAAgB,EAAE,oBAAoB,CAAC,CACtD,CAAC,IAAI,CAAC;QAEP,MAAM,cAAc,GAAG,MAAM,CAAC,cAAc,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC;QAC1D,MAAM,SAAS,GACd,cAAc,GAAG,CAAC;YACjB,CAAC,CAAC,CAAC,YAAY,GAAG,cAAc,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;YAC5C,CAAC,CAAC,GAAG,CAAC;QACR,MAAM,gBAAgB,GACrB,WAAW,GAAG,CAAC;YACd,CAAC,CAAC,CAAC,sBAAsB,GAAG,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;YACnD,CAAC,CAAC,GAAG,CAAC;QACR,MAAM,eAAe,GACpB,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,GAAG,UAAU,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;QAC7D,MAAM,iBAAiB,GAAG,MAAM,CAAC,eAAe,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC;QAC9D,MAAM,sBAAsB,GAC3B,iBAAiB,GAAG,CAAC;YACpB,CAAC,CAAC,CAAC,UAAU,GAAG,iBAAiB,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;YAC7C,CAAC,CAAC,GAAG,CAAC;QACR,MAAM,iBAAiB,GAAG,IAAI,CAAC,GAAG,CACjC,WAAW,GAAG,sBAAsB,EACpC,CAAC,CACD,CAAC;QACF,MAAM,gBAAgB,GACrB,WAAW,GAAG,CAAC;YACd,CAAC,CAAC,CAAC,iBAAiB,GAAG,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;YAC9C,CAAC,CAAC,GAAG,CAAC;QACR,MAAM,aAAa,GAAG,MAAM,CAAC,kBAAkB,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC;QAC7D,MAAM,eAAe,GAAG,MAAM,CAAC,oBAAoB,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC;QACjE,MAAM,iBAAiB,GAAG,IAAI,CAAC,GAAG,CAAC,aAAa,GAAG,eAAe,EAAE,CAAC,CAAC,CAAC;QACvE,MAAM,kBAAkB,GAAG,iBAAiB,KAAK,CAAC,CAAC;QAEnD,OAAO;YACN,cAAc;YACd,YAAY;YACZ,aAAa;YACb,eAAe;YACf,aAAa,EAAE,aAAa,EAAE,KAAK,IAAI,CAAC;YACxC,SAAS,EAAE,UAAU,CAAC,SAAS,CAAC;YAChC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC;YAC9C,eAAe,EAAE,UAAU,CAAC,eAAe,CAAC;YAC5C,sBAAsB,EAAE,UAAU,CAAC,sBAAsB,CAAC;YAC1D,WAAW;YACX,UAAU;YACV,iBAAiB;YACjB,iBAAiB;YACjB,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC;YAC9C,iBAAiB;YACjB,kBAAkB;YAClB,iBAAiB,EAAE,eAAe;SAClC,CAAC;IACH,CAAC,CAAC;IAEF,eAAe,EAAE,eAAe,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE;QACjD,MAAM,QAAQ,GAAG,MAAM,EAAE,CAAC,SAAS,CAAC;aAClC,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,CAAC;aACzC,OAAO,CAAC,MAAM,CAAC,CAAC;QAElB,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;YACjC,EAAE,EAAE,OAAO,CAAC,EAAE;YACd,IAAI,EAAE,OAAO,CAAC,IAAI,IAAI,WAAW,OAAO,CAAC,EAAE,EAAE;YAC7C,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,MAAM,EAAE,OAAO,CAAC,MAAM,IAAI,SAAS;SACnC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC;CACF,CAAC,CAAC"}
@@ -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>;
@@ -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 {};