cross-seed 7.0.0-1 → 7.0.0-11

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 (287) 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 +699 -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 +786 -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 +658 -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 +342 -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 +405 -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 +14 -0
  44. package/dist/db.js +287 -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/diagnostics/db.d.ts +21 -0
  53. package/dist/diagnostics/db.js +107 -0
  54. package/dist/diagnostics/db.js.map +1 -0
  55. package/dist/diff.d.ts +1 -0
  56. package/dist/diff.js +24 -0
  57. package/dist/diff.js.map +1 -0
  58. package/dist/errors.d.ts +3 -0
  59. package/dist/errors.js +7 -0
  60. package/dist/errors.js.map +1 -0
  61. package/dist/indexers.d.ts +105 -0
  62. package/dist/indexers.js +248 -0
  63. package/dist/indexers.js.map +1 -0
  64. package/dist/inject.d.ts +2 -0
  65. package/dist/inject.js +594 -0
  66. package/dist/inject.js.map +1 -0
  67. package/dist/jobs.d.ts +29 -0
  68. package/dist/jobs.js +151 -0
  69. package/dist/jobs.js.map +1 -0
  70. package/dist/logger.d.ts +29 -0
  71. package/dist/logger.js +157 -0
  72. package/dist/logger.js.map +1 -0
  73. package/dist/migrations/00-initialSchema.d.ts +9 -0
  74. package/dist/migrations/00-initialSchema.js +30 -0
  75. package/dist/migrations/00-initialSchema.js.map +1 -0
  76. package/dist/migrations/01-jobs.d.ts +9 -0
  77. package/dist/migrations/01-jobs.js +12 -0
  78. package/dist/migrations/01-jobs.js.map +1 -0
  79. package/dist/migrations/02-timestamps.d.ts +9 -0
  80. package/dist/migrations/02-timestamps.js +21 -0
  81. package/dist/migrations/02-timestamps.js.map +1 -0
  82. package/dist/migrations/03-rateLimits.d.ts +9 -0
  83. package/dist/migrations/03-rateLimits.js +14 -0
  84. package/dist/migrations/03-rateLimits.js.map +1 -0
  85. package/dist/migrations/04-auth.d.ts +9 -0
  86. package/dist/migrations/04-auth.js +13 -0
  87. package/dist/migrations/04-auth.js.map +1 -0
  88. package/dist/migrations/05-caps.d.ts +9 -0
  89. package/dist/migrations/05-caps.js +16 -0
  90. package/dist/migrations/05-caps.js.map +1 -0
  91. package/dist/migrations/06-uniqueDecisions.d.ts +9 -0
  92. package/dist/migrations/06-uniqueDecisions.js +29 -0
  93. package/dist/migrations/06-uniqueDecisions.js.map +1 -0
  94. package/dist/migrations/07-limits.d.ts +9 -0
  95. package/dist/migrations/07-limits.js +12 -0
  96. package/dist/migrations/07-limits.js.map +1 -0
  97. package/dist/migrations/08-rss.d.ts +9 -0
  98. package/dist/migrations/08-rss.js +15 -0
  99. package/dist/migrations/08-rss.js.map +1 -0
  100. package/dist/migrations/09-clientAndDataSearchees.d.ts +9 -0
  101. package/dist/migrations/09-clientAndDataSearchees.js +34 -0
  102. package/dist/migrations/09-clientAndDataSearchees.js.map +1 -0
  103. package/dist/migrations/10-indexerNameAudioBookCaps.d.ts +9 -0
  104. package/dist/migrations/10-indexerNameAudioBookCaps.js +18 -0
  105. package/dist/migrations/10-indexerNameAudioBookCaps.js.map +1 -0
  106. package/dist/migrations/11-trackers.d.ts +9 -0
  107. package/dist/migrations/11-trackers.js +38 -0
  108. package/dist/migrations/11-trackers.js.map +1 -0
  109. package/dist/migrations/12-user-auth.d.ts +9 -0
  110. package/dist/migrations/12-user-auth.js +22 -0
  111. package/dist/migrations/12-user-auth.js.map +1 -0
  112. package/dist/migrations/13-settings.d.ts +9 -0
  113. package/dist/migrations/13-settings.js +23 -0
  114. package/dist/migrations/13-settings.js.map +1 -0
  115. package/dist/migrations/14-indexer-enabled-flag.d.ts +9 -0
  116. package/dist/migrations/14-indexer-enabled-flag.js +12 -0
  117. package/dist/migrations/14-indexer-enabled-flag.js.map +1 -0
  118. package/dist/migrations/15-remove-url-unique-constraint.d.ts +9 -0
  119. package/dist/migrations/15-remove-url-unique-constraint.js +14 -0
  120. package/dist/migrations/15-remove-url-unique-constraint.js.map +1 -0
  121. package/dist/migrations/16-prune-inactive-indexers.d.ts +9 -0
  122. package/dist/migrations/16-prune-inactive-indexers.js +17 -0
  123. package/dist/migrations/16-prune-inactive-indexers.js.map +1 -0
  124. package/dist/migrations/migrations.d.ts +13 -0
  125. package/dist/migrations/migrations.js +41 -0
  126. package/dist/migrations/migrations.js.map +1 -0
  127. package/dist/parseTorrent.d.ts +53 -0
  128. package/dist/parseTorrent.js +128 -0
  129. package/dist/parseTorrent.js.map +1 -0
  130. package/dist/pipeline.d.ts +41 -0
  131. package/dist/pipeline.js +578 -0
  132. package/dist/pipeline.js.map +1 -0
  133. package/dist/preFilter.d.ts +25 -0
  134. package/dist/preFilter.js +250 -0
  135. package/dist/preFilter.js.map +1 -0
  136. package/dist/problems/linking.d.ts +2 -0
  137. package/dist/problems/linking.js +80 -0
  138. package/dist/problems/linking.js.map +1 -0
  139. package/dist/problems/path.d.ts +22 -0
  140. package/dist/problems/path.js +96 -0
  141. package/dist/problems/path.js.map +1 -0
  142. package/dist/problems.d.ts +13 -0
  143. package/dist/problems.js +48 -0
  144. package/dist/problems.js.map +1 -0
  145. package/dist/pushNotifier.d.ts +19 -0
  146. package/dist/pushNotifier.js +137 -0
  147. package/dist/pushNotifier.js.map +1 -0
  148. package/dist/routes/baseApi.d.ts +2 -0
  149. package/dist/routes/baseApi.js +354 -0
  150. package/dist/routes/baseApi.js.map +1 -0
  151. package/dist/routes/indexerApi.d.ts +6 -0
  152. package/dist/routes/indexerApi.js +165 -0
  153. package/dist/routes/indexerApi.js.map +1 -0
  154. package/dist/routes/staticFrontendPlugin.d.ts +4 -0
  155. package/dist/routes/staticFrontendPlugin.js +61 -0
  156. package/dist/routes/staticFrontendPlugin.js.map +1 -0
  157. package/dist/runtimeConfig.d.ts +6 -0
  158. package/dist/runtimeConfig.js +27 -0
  159. package/dist/runtimeConfig.js.map +1 -0
  160. package/dist/searchee.d.ts +108 -0
  161. package/dist/searchee.js +690 -0
  162. package/dist/searchee.js.map +1 -0
  163. package/dist/server.d.ts +4 -0
  164. package/dist/server.js +65 -0
  165. package/dist/server.js.map +1 -0
  166. package/dist/services/indexerService.d.ts +96 -0
  167. package/dist/services/indexerService.js +287 -0
  168. package/dist/services/indexerService.js.map +1 -0
  169. package/dist/sessionCookies.d.ts +5 -0
  170. package/dist/sessionCookies.js +27 -0
  171. package/dist/sessionCookies.js.map +1 -0
  172. package/dist/startup.d.ts +25 -0
  173. package/dist/startup.js +157 -0
  174. package/dist/startup.js.map +1 -0
  175. package/dist/torrent.d.ts +69 -0
  176. package/dist/torrent.js +665 -0
  177. package/dist/torrent.js.map +1 -0
  178. package/dist/torznab.d.ts +60 -0
  179. package/dist/torznab.js +711 -0
  180. package/dist/torznab.js.map +1 -0
  181. package/dist/trpc/fastifyAdapter.d.ts +2 -0
  182. package/dist/trpc/fastifyAdapter.js +9 -0
  183. package/dist/trpc/fastifyAdapter.js.map +1 -0
  184. package/dist/trpc/index.d.ts +49 -0
  185. package/dist/trpc/index.js +53 -0
  186. package/dist/trpc/index.js.map +1 -0
  187. package/dist/trpc/routers/auth.d.ts +43 -0
  188. package/dist/trpc/routers/auth.js +116 -0
  189. package/dist/trpc/routers/auth.js.map +1 -0
  190. package/dist/trpc/routers/clients.d.ts +21 -0
  191. package/dist/trpc/routers/clients.js +65 -0
  192. package/dist/trpc/routers/clients.js.map +1 -0
  193. package/dist/trpc/routers/health.d.ts +17 -0
  194. package/dist/trpc/routers/health.js +24 -0
  195. package/dist/trpc/routers/health.js.map +1 -0
  196. package/dist/trpc/routers/index.d.ts +394 -0
  197. package/dist/trpc/routers/index.js +23 -0
  198. package/dist/trpc/routers/index.js.map +1 -0
  199. package/dist/trpc/routers/indexers.d.ts +75 -0
  200. package/dist/trpc/routers/indexers.js +79 -0
  201. package/dist/trpc/routers/indexers.js.map +1 -0
  202. package/dist/trpc/routers/jobs.d.ts +33 -0
  203. package/dist/trpc/routers/jobs.js +84 -0
  204. package/dist/trpc/routers/jobs.js.map +1 -0
  205. package/dist/trpc/routers/logs.d.ts +27 -0
  206. package/dist/trpc/routers/logs.js +91 -0
  207. package/dist/trpc/routers/logs.js.map +1 -0
  208. package/dist/trpc/routers/searchees.d.ts +51 -0
  209. package/dist/trpc/routers/searchees.js +156 -0
  210. package/dist/trpc/routers/searchees.js.map +1 -0
  211. package/dist/trpc/routers/settings.d.ts +83 -0
  212. package/dist/trpc/routers/settings.js +92 -0
  213. package/dist/trpc/routers/settings.js.map +1 -0
  214. package/dist/trpc/routers/stats.d.ts +42 -0
  215. package/dist/trpc/routers/stats.js +102 -0
  216. package/dist/trpc/routers/stats.js.map +1 -0
  217. package/dist/userAuth.d.ts +21 -0
  218. package/dist/userAuth.js +86 -0
  219. package/dist/userAuth.js.map +1 -0
  220. package/dist/utils/authUtils.d.ts +10 -0
  221. package/dist/utils/authUtils.js +24 -0
  222. package/dist/utils/authUtils.js.map +1 -0
  223. package/dist/utils/logWatcher.d.ts +28 -0
  224. package/dist/utils/logWatcher.js +229 -0
  225. package/dist/utils/logWatcher.js.map +1 -0
  226. package/dist/utils/object.d.ts +1 -0
  227. package/dist/utils/object.js +4 -0
  228. package/dist/utils/object.js.map +1 -0
  229. package/dist/utils.d.ts +172 -0
  230. package/dist/utils.js +648 -0
  231. package/dist/utils.js.map +1 -0
  232. package/dist/webui/assets/{FieldInfo-Bxj_j8SJ.js → FieldInfo-sRlPRNSK.js} +1 -1
  233. package/dist/webui/assets/{Page-C3rteCZt.js → Page-B68mlTwU.js} +1 -1
  234. package/dist/webui/assets/{array-field-DVSC6nHP.js → array-field-BCFMrvoU.js} +1 -1
  235. package/dist/webui/assets/{badge-DTZMtS0e.js → badge-C5YCxEzP.js} +1 -1
  236. package/dist/webui/assets/check-NQsw6yBl.js +1 -0
  237. package/dist/webui/assets/{chevron-down-CRy8M0kJ.js → chevron-down-8PGvFYxV.js} +1 -1
  238. package/dist/webui/assets/{clients-CW8oEZoQ.js → clients-DnVpwApe.js} +1 -1
  239. package/dist/webui/assets/{connect-YBNsnjWT.js → connect-wMg2zyz6.js} +1 -1
  240. package/dist/webui/assets/debug-BrjwiEi2.js +1 -0
  241. package/dist/webui/assets/{directories-BSK28RgR.js → directories-CHpJCWNR.js} +1 -1
  242. package/dist/webui/assets/{duration-field-C6xoSlJg.js → duration-field-DIkKt3iw.js} +1 -1
  243. package/dist/webui/assets/{general-lJJxZhH7.js → general-uZrUIxbI.js} +1 -1
  244. package/dist/webui/assets/health-_MuvAyjo.js +1 -0
  245. package/dist/webui/assets/index-B41DM2T5.css +1 -0
  246. package/dist/webui/assets/{index-C2cH1Gst.js → index-BBzHsn7u.js} +1 -1
  247. package/dist/webui/assets/{index-Bi48hI2z.js → index-Ncy0-Qo7.js} +3 -3
  248. package/dist/webui/assets/index-pKWy6v1P.js +1 -0
  249. package/dist/webui/assets/{jobs-CxmNab9w.js → jobs-B8eat0YU.js} +1 -1
  250. package/dist/webui/assets/{library-vaj2W8sE.js → library-BB0jQ8zn.js} +1 -1
  251. package/dist/webui/assets/{loader-circle-M0gu1gZ-.js → loader-circle-Bz67bJa3.js} +1 -1
  252. package/dist/webui/assets/logs-CeP28848.js +1 -0
  253. package/dist/webui/assets/{search-2R5sIdT8.js → search-BRBIrqaX.js} +1 -1
  254. package/dist/webui/assets/{select-zHgqMzLj.js → select-GZr6C6eZ.js} +1 -1
  255. package/dist/webui/assets/{select-field-BCqNLDrJ.js → select-field-CvT0SYk8.js} +1 -1
  256. package/dist/webui/assets/settings-0ZdYY8g_.js +1 -0
  257. package/dist/webui/assets/{submit-button-BtcnyggQ.js → submit-button-D7DKHqAq.js} +1 -1
  258. package/dist/webui/assets/{switch-G0W3uJVN.js → switch-BeMrf8sh.js} +1 -1
  259. package/dist/webui/assets/{switch-field-IBd9ORNq.js → switch-field-qMXHRKhx.js} +1 -1
  260. package/dist/webui/assets/{table-DvgJU7Gh.js → table-qEFWauuw.js} +1 -1
  261. package/dist/webui/assets/{test-tube-BIwmoM45.js → test-tube-DhD6uWdp.js} +1 -1
  262. package/dist/webui/assets/{text-field-DruSbGhy.js → text-field-ZnKHDUks.js} +1 -1
  263. package/dist/webui/assets/{time-BSMZjmyW.js → time-BM9K_Fbp.js} +1 -1
  264. package/dist/webui/assets/{trackers-D-OpAe63.js → trackers-BjJuAdX3.js} +1 -1
  265. package/dist/webui/assets/{use-form-validation-context-BkAfWAh0.js → use-form-validation-context-D2oA54L_.js} +1 -1
  266. package/dist/webui/assets/{use-settings-form-submit-CDRh-E9U.js → use-settings-form-submit-CXwtE1sI.js} +2 -2
  267. package/dist/webui/assets/useQuery-DD10sbzn.js +1 -0
  268. package/dist/webui/index.html +2 -2
  269. package/node_modules/@cross-seed/shared/dist/configSchema.d.ts.map +1 -1
  270. package/node_modules/@cross-seed/shared/dist/configSchema.js.map +1 -1
  271. package/node_modules/@cross-seed/shared/dist/tsconfig.tsbuildinfo +1 -1
  272. package/node_modules/@cross-seed/shared/dist/utils.d.ts +3 -0
  273. package/node_modules/@cross-seed/shared/dist/utils.d.ts.map +1 -1
  274. package/node_modules/@cross-seed/shared/dist/utils.js +11 -0
  275. package/node_modules/@cross-seed/shared/dist/utils.js.map +1 -1
  276. package/node_modules/@cross-seed/shared/package.json +3 -1
  277. package/package.json +18 -59
  278. package/LICENSE +0 -201
  279. package/README.md +0 -33
  280. package/dist/webui/assets/check-Bu3ldi63.js +0 -1
  281. package/dist/webui/assets/debug-mz8-WYZj.js +0 -1
  282. package/dist/webui/assets/health-CXbsVrie.js +0 -1
  283. package/dist/webui/assets/index-C-Ul7GNg.css +0 -1
  284. package/dist/webui/assets/index-Cc5bDmJr.js +0 -1
  285. package/dist/webui/assets/logs-Cu9RyKS0.js +0 -1
  286. package/dist/webui/assets/settings-CMYjpTbZ.js +0 -1
  287. package/dist/webui/assets/useQuery-A4Hv_4uX.js +0 -1
@@ -0,0 +1,102 @@
1
+ import { authedProcedure, router } from "../index.js";
2
+ import { db } from "../../db.js";
3
+ import { estimateSearchString } from "../../torznab.js";
4
+ import { mapAsync } from "../../utils.js";
5
+ export const statsRouter = router({
6
+ getOverview: authedProcedure.query(async () => {
7
+ const [searcheeResult, searcheeNames, totalIndexerResult, healthyIndexerResult, timestampResult, decisionsByType, recentMatches, matchAggregates,] = await Promise.all([
8
+ db("searchee").count({ count: "*" }).first(),
9
+ db("searchee").select("name"),
10
+ db("indexer").count({ count: "*" }).first(),
11
+ db("indexer")
12
+ .where({ enabled: true, search_cap: true })
13
+ .where((qb) => qb
14
+ .whereNull("status")
15
+ .orWhere("status", "OK")
16
+ .orWhere("retry_after", "<", Date.now()))
17
+ .count({ count: "*" })
18
+ .first(),
19
+ db("timestamp").count({ count: "*" }).first(),
20
+ db("decision")
21
+ .select("decision")
22
+ .count({ count: "*" })
23
+ .groupBy("decision"),
24
+ db("decision")
25
+ .whereIn("decision", [
26
+ "MATCH",
27
+ "MATCH_SIZE_ONLY",
28
+ "MATCH_PARTIAL",
29
+ ])
30
+ .where("last_seen", ">", Date.now() - 24 * 60 * 60 * 1000) // last 24h
31
+ .count({ count: "*" })
32
+ .first(),
33
+ db("decision")
34
+ .whereNotNull("info_hash")
35
+ .select({
36
+ snatchCount: db.raw("COUNT(DISTINCT info_hash)"),
37
+ matchCount: db.raw("COUNT(DISTINCT CASE WHEN decision IN ('MATCH','MATCH_SIZE_ONLY','MATCH_PARTIAL') THEN info_hash END)"),
38
+ matchCountWithInfoHash: db.raw("COUNT(DISTINCT CASE WHEN decision IN ('MATCH','MATCH_SIZE_ONLY','MATCH_PARTIAL','SAME_INFO_HASH','INFO_HASH_ALREADY_EXISTS') THEN info_hash END)"),
39
+ })
40
+ .first(),
41
+ ]);
42
+ const snatchCount = Number(matchAggregates?.snatchCount ?? 0);
43
+ const matchCount = Number(matchAggregates?.matchCount ?? 0);
44
+ const matchCountWithInfoHash = Number(matchAggregates?.matchCountWithInfoHash ?? 0);
45
+ const totalMatches = matchCount;
46
+ const searcheeNameList = searcheeNames
47
+ .map((row) => row.name)
48
+ .filter((name) => typeof name === "string" && name.length > 0);
49
+ const queryCount = new Set(await mapAsync(searcheeNameList, estimateSearchString)).size;
50
+ const totalSearchees = Number(searcheeResult?.count || 0);
51
+ const matchRate = totalSearchees > 0
52
+ ? (totalMatches / totalSearchees).toFixed(2)
53
+ : "0";
54
+ const matchesPerSnatch = snatchCount > 0
55
+ ? (matchCountWithInfoHash / snatchCount).toFixed(3)
56
+ : "0";
57
+ const matchesPerQuery = queryCount > 0 ? (matchCount / queryCount).toFixed(3) : "0";
58
+ const queryIndexerCount = Number(timestampResult?.count || 0);
59
+ const matchesPerQueryIndexer = queryIndexerCount > 0
60
+ ? (matchCount / queryIndexerCount).toFixed(3)
61
+ : "0";
62
+ const wastedSnatchCount = Math.max(snatchCount - matchCountWithInfoHash, 0);
63
+ const wastedSnatchRate = snatchCount > 0
64
+ ? (wastedSnatchCount / snatchCount).toFixed(3)
65
+ : "0";
66
+ const totalIndexers = Number(totalIndexerResult?.count || 0);
67
+ const healthyIndexers = Number(healthyIndexerResult?.count || 0);
68
+ const unhealthyIndexers = Math.max(totalIndexers - healthyIndexers, 0);
69
+ const allIndexersHealthy = unhealthyIndexers === 0;
70
+ return {
71
+ totalSearchees,
72
+ totalMatches,
73
+ totalIndexers,
74
+ healthyIndexers,
75
+ recentMatches: recentMatches?.count || 0,
76
+ matchRate: parseFloat(matchRate),
77
+ matchesPerSnatch: parseFloat(matchesPerSnatch),
78
+ matchesPerQuery: parseFloat(matchesPerQuery),
79
+ matchesPerQueryIndexer: parseFloat(matchesPerQueryIndexer),
80
+ snatchCount,
81
+ queryCount,
82
+ queryIndexerCount,
83
+ wastedSnatchCount,
84
+ wastedSnatchRate: parseFloat(wastedSnatchRate),
85
+ unhealthyIndexers,
86
+ allIndexersHealthy,
87
+ decisionBreakdown: decisionsByType,
88
+ };
89
+ }),
90
+ getIndexerStats: authedProcedure.query(async () => {
91
+ const indexers = await db("indexer")
92
+ .select("id", "name", "enabled", "status")
93
+ .orderBy("name");
94
+ return indexers.map((indexer) => ({
95
+ id: indexer.id,
96
+ name: indexer.name || `Indexer ${indexer.id}`,
97
+ enabled: indexer.enabled,
98
+ status: indexer.status || "unknown",
99
+ }));
100
+ }),
101
+ });
102
+ //# sourceMappingURL=stats.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stats.js","sourceRoot":"","sources":["../../../src/trpc/routers/stats.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ export interface User {
2
+ id: number;
3
+ username: string;
4
+ password: string;
5
+ created_at: Date;
6
+ }
7
+ export interface Session {
8
+ id: string;
9
+ user_id: number;
10
+ expires_at: number;
11
+ created_at: number;
12
+ }
13
+ export declare function createUser(username: string, password: string): Promise<User>;
14
+ export declare function findUserByUsername(username: string): Promise<User | undefined>;
15
+ export declare function validateUserCredentials(username: string, password: string): Promise<User | null>;
16
+ export declare function createSession(userId: number): Promise<Session>;
17
+ export declare function validateSession(sessionId: string): Promise<User | null>;
18
+ export declare function removeSession(sessionId: string): Promise<void>;
19
+ export declare function hasUsers(): Promise<boolean>;
20
+ export declare function createInitialUserIfNeeded(username: string, password: string): Promise<User | null>;
21
+ export declare function resetUsers(): Promise<string>;
@@ -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 {};
@@ -0,0 +1,229 @@
1
+ import { promises as fs, watch } from "fs";
2
+ import { join } from "path";
3
+ import { appDir } from "../configuration.js";
4
+ import { logger } from "../logger.js";
5
+ function parseLocalTimestampToIso(timestamp) {
6
+ const match = /^(\d{4})-(\d{2})-(\d{2}) (\d{2}):(\d{2}):(\d{2})(?:\.(\d{3}))?$/.exec(timestamp);
7
+ if (!match) {
8
+ const parsed = new Date(timestamp);
9
+ return Number.isNaN(parsed.getTime())
10
+ ? timestamp
11
+ : parsed.toISOString();
12
+ }
13
+ const [, year, month, day, hour, minute, second, millisecond] = match;
14
+ return new Date(Number(year), Number(month) - 1, Number(day), Number(hour), Number(minute), Number(second), millisecond ? Number(millisecond) : 0).toISOString();
15
+ }
16
+ class LogWatcher {
17
+ watchers = new Map();
18
+ subscribers = new Set();
19
+ lastPositions = new Map();
20
+ retryTimers = new Map();
21
+ constructor() {
22
+ this.startWatching();
23
+ }
24
+ startWatching() {
25
+ const logFiles = ["verbose.current.log"];
26
+ for (const fileName of logFiles) {
27
+ const filePath = join(appDir(), "logs", fileName);
28
+ this.watchLogFile(filePath);
29
+ }
30
+ }
31
+ watchLogFile(filePath) {
32
+ // Initialize position to end of file (only watch new entries)
33
+ void this.initializePosition(filePath);
34
+ const clearRetryTimer = () => {
35
+ const existingTimer = this.retryTimers.get(filePath);
36
+ if (existingTimer) {
37
+ clearTimeout(existingTimer);
38
+ this.retryTimers.delete(filePath);
39
+ }
40
+ };
41
+ const scheduleRetry = () => {
42
+ if (this.retryTimers.has(filePath)) {
43
+ return;
44
+ }
45
+ const timeout = setTimeout(() => {
46
+ this.retryTimers.delete(filePath);
47
+ this.watchLogFile(filePath);
48
+ }, 1000);
49
+ if (typeof timeout.unref === "function") {
50
+ timeout.unref();
51
+ }
52
+ this.retryTimers.set(filePath, timeout);
53
+ };
54
+ try {
55
+ const watcher = watch(filePath, (eventType) => {
56
+ if (eventType === "change") {
57
+ void this.handleFileChange(filePath);
58
+ }
59
+ else if (eventType === "rename") {
60
+ // File was rotated or replaced - reset position and retry watching
61
+ this.lastPositions.set(filePath, 0);
62
+ this.watchers.get(filePath)?.close();
63
+ this.watchers.delete(filePath);
64
+ scheduleRetry();
65
+ }
66
+ });
67
+ watcher.on("error", (error) => {
68
+ console.error(`Error watching ${filePath}:`, error);
69
+ this.watchers.delete(filePath);
70
+ if (error.code === "ENOENT") {
71
+ scheduleRetry();
72
+ }
73
+ });
74
+ clearRetryTimer();
75
+ this.watchers.set(filePath, watcher);
76
+ }
77
+ catch (error) {
78
+ const err = error;
79
+ if (err.code === "ENOENT") {
80
+ scheduleRetry();
81
+ }
82
+ else {
83
+ console.error(`Failed to watch ${filePath}:`, error);
84
+ }
85
+ }
86
+ }
87
+ async initializePosition(filePath) {
88
+ try {
89
+ const stats = await fs.stat(filePath);
90
+ this.lastPositions.set(filePath, stats.size);
91
+ }
92
+ catch (error) {
93
+ // File doesn't exist yet, start from beginning
94
+ this.lastPositions.set(filePath, 0);
95
+ }
96
+ }
97
+ async handleFileChange(filePath) {
98
+ try {
99
+ const stats = await fs.stat(filePath);
100
+ const lastPosition = this.lastPositions.get(filePath) || 0;
101
+ if (stats.size <= lastPosition) {
102
+ // File was truncated or rotated, start from beginning
103
+ this.lastPositions.set(filePath, 0);
104
+ return;
105
+ }
106
+ // Read only the new content
107
+ const stream = await fs.open(filePath, "r");
108
+ const buffer = Buffer.alloc(stats.size - lastPosition);
109
+ await stream.read(buffer, 0, buffer.length, lastPosition);
110
+ await stream.close();
111
+ const newContent = buffer.toString("utf8");
112
+ const logEntries = this.parseLogContent(newContent);
113
+ for (const logEntry of logEntries) {
114
+ this.notifySubscribers(logEntry);
115
+ }
116
+ this.lastPositions.set(filePath, stats.size);
117
+ }
118
+ catch (error) {
119
+ console.error(`Error reading ${filePath}:`, error);
120
+ }
121
+ }
122
+ parseLogContent(content) {
123
+ const logEntries = [];
124
+ const lines = content.split("\n");
125
+ let currentEntry = null;
126
+ for (const line of lines) {
127
+ if (!line.trim())
128
+ continue;
129
+ const parsedEntry = this.tryParseLogLine(line);
130
+ if (parsedEntry) {
131
+ // This is a new log entry
132
+ if (currentEntry) {
133
+ logEntries.push(currentEntry);
134
+ }
135
+ currentEntry = parsedEntry;
136
+ }
137
+ else if (currentEntry) {
138
+ // This is a continuation line (stack trace, etc.)
139
+ currentEntry.message += "\n" + line;
140
+ }
141
+ else {
142
+ // Orphan line with no preceding log entry
143
+ logEntries.push({
144
+ timestamp: new Date().toISOString(),
145
+ level: "info",
146
+ label: "raw",
147
+ message: line,
148
+ });
149
+ }
150
+ }
151
+ if (currentEntry) {
152
+ logEntries.push(currentEntry);
153
+ }
154
+ return logEntries;
155
+ }
156
+ tryParseLogLine(line) {
157
+ // Parse winston text format: "2025-06-01 00:00:35.123 info: [scheduler] starting job: rss"
158
+ const logRegex = /^(\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}(?:\.\d{3})?) (\w+):\s*(?:\[([^\]]+)\])?\s*(.*)$/;
159
+ const match = line.match(logRegex);
160
+ if (match) {
161
+ const [, timestamp, level, label, message] = match;
162
+ return {
163
+ timestamp: parseLocalTimestampToIso(timestamp),
164
+ level,
165
+ label: label || undefined,
166
+ message: message.trim(),
167
+ };
168
+ }
169
+ return null;
170
+ }
171
+ notifySubscribers(logEntry) {
172
+ this.subscribers.forEach((callback) => {
173
+ try {
174
+ callback(logEntry);
175
+ }
176
+ catch (error) {
177
+ console.error("Error in log subscriber:", error);
178
+ }
179
+ });
180
+ }
181
+ subscribe(callback) {
182
+ this.subscribers.add(callback);
183
+ return () => this.subscribers.delete(callback);
184
+ }
185
+ async getRecentLogs(limit = 100) {
186
+ const filePath = join(appDir(), "logs", "verbose.current.log");
187
+ try {
188
+ const fileContent = await fs.readFile(filePath, "utf-8");
189
+ const allEntries = this.parseLogContent(fileContent);
190
+ return allEntries.slice(-limit);
191
+ }
192
+ catch (error) {
193
+ logger.error(`Error reading log file ${filePath}:`, error);
194
+ return [];
195
+ }
196
+ }
197
+ shouldIncludeLevel(logLevel, filterLevel) {
198
+ const levels = ["error", "warn", "info", "verbose", "debug"];
199
+ const logIndex = levels.indexOf(logLevel);
200
+ const filterIndex = levels.indexOf(filterLevel);
201
+ return logIndex <= filterIndex;
202
+ }
203
+ destroy() {
204
+ for (const watcher of this.watchers.values()) {
205
+ watcher.close();
206
+ }
207
+ this.watchers.clear();
208
+ this.subscribers.clear();
209
+ for (const timer of this.retryTimers.values()) {
210
+ clearTimeout(timer);
211
+ }
212
+ this.retryTimers.clear();
213
+ }
214
+ }
215
+ // Singleton instance
216
+ let logWatcher = null;
217
+ export function getLogWatcher() {
218
+ if (!logWatcher) {
219
+ logWatcher = new LogWatcher();
220
+ }
221
+ return logWatcher;
222
+ }
223
+ export function destroyLogWatcher() {
224
+ if (logWatcher) {
225
+ logWatcher.destroy();
226
+ logWatcher = null;
227
+ }
228
+ }
229
+ //# sourceMappingURL=logWatcher.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logWatcher.js","sourceRoot":"","sources":["../../src/utils/logWatcher.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare function omitUndefined<T extends Record<string, unknown>>(obj: T): Partial<T>;
@@ -0,0 +1,4 @@
1
+ export function omitUndefined(obj) {
2
+ return Object.fromEntries(Object.entries(obj).filter(([, value]) => value !== undefined));
3
+ }
4
+ //# sourceMappingURL=object.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"object.js","sourceRoot":"","sources":["../../src/utils/object.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,aAAa,CAC5B,GAAM;IAEN,OAAO,MAAM,CAAC,WAAW,CACxB,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,KAAK,KAAK,SAAS,CAAC,CAChD,CAAC;AACjB,CAAC"}