@sonordev/site-kit 1.2.7

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 (300) hide show
  1. package/README.md +376 -0
  2. package/dist/SetupWizard-Cki06kB0.d.mts +12 -0
  3. package/dist/SetupWizard-Cki06kB0.d.ts +12 -0
  4. package/dist/analytics/index.d.mts +93 -0
  5. package/dist/analytics/index.d.ts +93 -0
  6. package/dist/analytics/index.js +89 -0
  7. package/dist/analytics/index.js.map +1 -0
  8. package/dist/analytics/index.mjs +71 -0
  9. package/dist/analytics/index.mjs.map +1 -0
  10. package/dist/api-CWtoFJCO.d.mts +137 -0
  11. package/dist/api-CWtoFJCO.d.ts +137 -0
  12. package/dist/blog/index.d.mts +305 -0
  13. package/dist/blog/index.d.ts +305 -0
  14. package/dist/blog/index.js +1578 -0
  15. package/dist/blog/index.js.map +1 -0
  16. package/dist/blog/index.mjs +1562 -0
  17. package/dist/blog/index.mjs.map +1 -0
  18. package/dist/blog/server.d.mts +229 -0
  19. package/dist/blog/server.d.ts +229 -0
  20. package/dist/blog/server.js +692 -0
  21. package/dist/blog/server.js.map +1 -0
  22. package/dist/blog/server.mjs +666 -0
  23. package/dist/blog/server.mjs.map +1 -0
  24. package/dist/chunk-24277A3Q.mjs +968 -0
  25. package/dist/chunk-24277A3Q.mjs.map +1 -0
  26. package/dist/chunk-373TK6TZ.js +321 -0
  27. package/dist/chunk-373TK6TZ.js.map +1 -0
  28. package/dist/chunk-3MYZS6PD.js +30 -0
  29. package/dist/chunk-3MYZS6PD.js.map +1 -0
  30. package/dist/chunk-43GBM4SX.js +283 -0
  31. package/dist/chunk-43GBM4SX.js.map +1 -0
  32. package/dist/chunk-4XPGGLVP.mjs +53 -0
  33. package/dist/chunk-4XPGGLVP.mjs.map +1 -0
  34. package/dist/chunk-622GAQP5.js +2008 -0
  35. package/dist/chunk-622GAQP5.js.map +1 -0
  36. package/dist/chunk-6BIPAKL4.mjs +28 -0
  37. package/dist/chunk-6BIPAKL4.mjs.map +1 -0
  38. package/dist/chunk-6ZCISNAB.mjs +343 -0
  39. package/dist/chunk-6ZCISNAB.mjs.map +1 -0
  40. package/dist/chunk-72MQFHYJ.js +1429 -0
  41. package/dist/chunk-72MQFHYJ.js.map +1 -0
  42. package/dist/chunk-7557OTHW.js +62 -0
  43. package/dist/chunk-7557OTHW.js.map +1 -0
  44. package/dist/chunk-7FUV73JZ.js +981 -0
  45. package/dist/chunk-7FUV73JZ.js.map +1 -0
  46. package/dist/chunk-7RF6PVHA.mjs +324 -0
  47. package/dist/chunk-7RF6PVHA.mjs.map +1 -0
  48. package/dist/chunk-7RYCHO6D.mjs +134 -0
  49. package/dist/chunk-7RYCHO6D.mjs.map +1 -0
  50. package/dist/chunk-7UKPRW25.mjs +1999 -0
  51. package/dist/chunk-7UKPRW25.mjs.map +1 -0
  52. package/dist/chunk-7URAOG2M.js +14864 -0
  53. package/dist/chunk-7URAOG2M.js.map +1 -0
  54. package/dist/chunk-AFAO3TGS.mjs +810 -0
  55. package/dist/chunk-AFAO3TGS.mjs.map +1 -0
  56. package/dist/chunk-BYLIU6XG.js +343 -0
  57. package/dist/chunk-BYLIU6XG.js.map +1 -0
  58. package/dist/chunk-D63MUKZ6.mjs +4423 -0
  59. package/dist/chunk-D63MUKZ6.mjs.map +1 -0
  60. package/dist/chunk-DDKW2FNA.js +390 -0
  61. package/dist/chunk-DDKW2FNA.js.map +1 -0
  62. package/dist/chunk-DQYMKR27.mjs +341 -0
  63. package/dist/chunk-DQYMKR27.mjs.map +1 -0
  64. package/dist/chunk-DW5UJKHH.js +221 -0
  65. package/dist/chunk-DW5UJKHH.js.map +1 -0
  66. package/dist/chunk-EEZCR6E6.js +50 -0
  67. package/dist/chunk-EEZCR6E6.js.map +1 -0
  68. package/dist/chunk-GCJXQ4AG.mjs +59 -0
  69. package/dist/chunk-GCJXQ4AG.mjs.map +1 -0
  70. package/dist/chunk-JGNQK2G6.mjs +14845 -0
  71. package/dist/chunk-JGNQK2G6.mjs.map +1 -0
  72. package/dist/chunk-JTLOJLWQ.mjs +563 -0
  73. package/dist/chunk-JTLOJLWQ.mjs.map +1 -0
  74. package/dist/chunk-K23A4G76.mjs +202 -0
  75. package/dist/chunk-K23A4G76.mjs.map +1 -0
  76. package/dist/chunk-KKU3K7RG.js +336 -0
  77. package/dist/chunk-KKU3K7RG.js.map +1 -0
  78. package/dist/chunk-KUGMH4ZF.js +571 -0
  79. package/dist/chunk-KUGMH4ZF.js.map +1 -0
  80. package/dist/chunk-LBVWVP72.js +110 -0
  81. package/dist/chunk-LBVWVP72.js.map +1 -0
  82. package/dist/chunk-LIVWLY2P.js +138 -0
  83. package/dist/chunk-LIVWLY2P.js.map +1 -0
  84. package/dist/chunk-M2T6R7BA.mjs +1003 -0
  85. package/dist/chunk-M2T6R7BA.mjs.map +1 -0
  86. package/dist/chunk-MV3QN7PW.mjs +47 -0
  87. package/dist/chunk-MV3QN7PW.mjs.map +1 -0
  88. package/dist/chunk-OB7E654K.js +72 -0
  89. package/dist/chunk-OB7E654K.js.map +1 -0
  90. package/dist/chunk-OIIKTGRL.mjs +380 -0
  91. package/dist/chunk-OIIKTGRL.mjs.map +1 -0
  92. package/dist/chunk-P3UWIUJS.mjs +1427 -0
  93. package/dist/chunk-P3UWIUJS.mjs.map +1 -0
  94. package/dist/chunk-PKN27UMH.mjs +136 -0
  95. package/dist/chunk-PKN27UMH.mjs.map +1 -0
  96. package/dist/chunk-QXV4667R.mjs +105 -0
  97. package/dist/chunk-QXV4667R.mjs.map +1 -0
  98. package/dist/chunk-S7FRYNSU.mjs +315 -0
  99. package/dist/chunk-S7FRYNSU.mjs.map +1 -0
  100. package/dist/chunk-TFLQX7K7.mjs +68 -0
  101. package/dist/chunk-TFLQX7K7.mjs.map +1 -0
  102. package/dist/chunk-UWE5PCYJ.mjs +279 -0
  103. package/dist/chunk-UWE5PCYJ.mjs.map +1 -0
  104. package/dist/chunk-UYFDNX2F.js +4469 -0
  105. package/dist/chunk-UYFDNX2F.js.map +1 -0
  106. package/dist/chunk-W4PALSGM.js +350 -0
  107. package/dist/chunk-W4PALSGM.js.map +1 -0
  108. package/dist/chunk-WECQ6KOB.js +1008 -0
  109. package/dist/chunk-WECQ6KOB.js.map +1 -0
  110. package/dist/chunk-XQQWI6WB.js +814 -0
  111. package/dist/chunk-XQQWI6WB.js.map +1 -0
  112. package/dist/chunk-XZJOZJB6.js +140 -0
  113. package/dist/chunk-XZJOZJB6.js.map +1 -0
  114. package/dist/chunk-ZSMWDLMK.js +63 -0
  115. package/dist/chunk-ZSMWDLMK.js.map +1 -0
  116. package/dist/cli/index.js +37243 -0
  117. package/dist/cli/index.js.map +1 -0
  118. package/dist/cli/index.mjs +37209 -0
  119. package/dist/cli/index.mjs.map +1 -0
  120. package/dist/commerce/index.d.mts +170 -0
  121. package/dist/commerce/index.d.ts +170 -0
  122. package/dist/commerce/index.js +174 -0
  123. package/dist/commerce/index.js.map +1 -0
  124. package/dist/commerce/index.mjs +5 -0
  125. package/dist/commerce/index.mjs.map +1 -0
  126. package/dist/commerce/server.d.mts +107 -0
  127. package/dist/commerce/server.d.ts +107 -0
  128. package/dist/commerce/server.js +187 -0
  129. package/dist/commerce/server.js.map +1 -0
  130. package/dist/commerce/server.mjs +177 -0
  131. package/dist/commerce/server.mjs.map +1 -0
  132. package/dist/config/index.d.mts +43 -0
  133. package/dist/config/index.d.ts +43 -0
  134. package/dist/config/index.js +66 -0
  135. package/dist/config/index.js.map +1 -0
  136. package/dist/config/index.mjs +64 -0
  137. package/dist/config/index.mjs.map +1 -0
  138. package/dist/engage/index.d.mts +33 -0
  139. package/dist/engage/index.d.ts +33 -0
  140. package/dist/engage/index.js +22 -0
  141. package/dist/engage/index.js.map +1 -0
  142. package/dist/engage/index.mjs +5 -0
  143. package/dist/engage/index.mjs.map +1 -0
  144. package/dist/forms/index.d.mts +437 -0
  145. package/dist/forms/index.d.ts +437 -0
  146. package/dist/forms/index.js +1168 -0
  147. package/dist/forms/index.js.map +1 -0
  148. package/dist/forms/index.mjs +1142 -0
  149. package/dist/forms/index.mjs.map +1 -0
  150. package/dist/generators-2XKQMPKH.mjs +4 -0
  151. package/dist/generators-2XKQMPKH.mjs.map +1 -0
  152. package/dist/generators-DTMO36DV.js +33 -0
  153. package/dist/generators-DTMO36DV.js.map +1 -0
  154. package/dist/images/index.d.mts +4 -0
  155. package/dist/images/index.d.ts +4 -0
  156. package/dist/images/index.js +46 -0
  157. package/dist/images/index.js.map +1 -0
  158. package/dist/images/index.mjs +5 -0
  159. package/dist/images/index.mjs.map +1 -0
  160. package/dist/images/server.d.mts +69 -0
  161. package/dist/images/server.d.ts +69 -0
  162. package/dist/images/server.js +21 -0
  163. package/dist/images/server.js.map +1 -0
  164. package/dist/images/server.mjs +4 -0
  165. package/dist/images/server.mjs.map +1 -0
  166. package/dist/index.d.mts +846 -0
  167. package/dist/index.d.ts +846 -0
  168. package/dist/index.js +2623 -0
  169. package/dist/index.js.map +1 -0
  170. package/dist/index.mjs +2416 -0
  171. package/dist/index.mjs.map +1 -0
  172. package/dist/layout/index.d.mts +53 -0
  173. package/dist/layout/index.d.ts +53 -0
  174. package/dist/layout/index.js +187 -0
  175. package/dist/layout/index.js.map +1 -0
  176. package/dist/layout/index.mjs +185 -0
  177. package/dist/layout/index.mjs.map +1 -0
  178. package/dist/llms/index.d.mts +448 -0
  179. package/dist/llms/index.d.ts +448 -0
  180. package/dist/llms/index.js +581 -0
  181. package/dist/llms/index.js.map +1 -0
  182. package/dist/llms/index.mjs +529 -0
  183. package/dist/llms/index.mjs.map +1 -0
  184. package/dist/manifest/index.d.mts +62 -0
  185. package/dist/manifest/index.d.ts +62 -0
  186. package/dist/manifest/index.js +85 -0
  187. package/dist/manifest/index.js.map +1 -0
  188. package/dist/manifest/index.mjs +83 -0
  189. package/dist/manifest/index.mjs.map +1 -0
  190. package/dist/middleware/index.d.mts +63 -0
  191. package/dist/middleware/index.d.ts +63 -0
  192. package/dist/middleware/index.js +54 -0
  193. package/dist/middleware/index.js.map +1 -0
  194. package/dist/middleware/index.mjs +51 -0
  195. package/dist/middleware/index.mjs.map +1 -0
  196. package/dist/migrator-2MQHOFDQ.mjs +4 -0
  197. package/dist/migrator-2MQHOFDQ.mjs.map +1 -0
  198. package/dist/migrator-THJCF6MZ.js +37 -0
  199. package/dist/migrator-THJCF6MZ.js.map +1 -0
  200. package/dist/redirects/index.d.mts +78 -0
  201. package/dist/redirects/index.d.ts +78 -0
  202. package/dist/redirects/index.js +26 -0
  203. package/dist/redirects/index.js.map +1 -0
  204. package/dist/redirects/index.mjs +5 -0
  205. package/dist/redirects/index.mjs.map +1 -0
  206. package/dist/reputation/index.d.mts +57 -0
  207. package/dist/reputation/index.d.ts +57 -0
  208. package/dist/reputation/index.js +21 -0
  209. package/dist/reputation/index.js.map +1 -0
  210. package/dist/reputation/index.mjs +4 -0
  211. package/dist/reputation/index.mjs.map +1 -0
  212. package/dist/robots/index.d.mts +38 -0
  213. package/dist/robots/index.d.ts +38 -0
  214. package/dist/robots/index.js +52 -0
  215. package/dist/robots/index.js.map +1 -0
  216. package/dist/robots/index.mjs +50 -0
  217. package/dist/robots/index.mjs.map +1 -0
  218. package/dist/routing-B5XS-6_W.d.mts +118 -0
  219. package/dist/routing-DZYzyDHw.d.ts +118 -0
  220. package/dist/scanner-GAF5PO5F.js +53 -0
  221. package/dist/scanner-GAF5PO5F.js.map +1 -0
  222. package/dist/scanner-LKJKW7IT.mjs +4 -0
  223. package/dist/scanner-LKJKW7IT.mjs.map +1 -0
  224. package/dist/securityHeaders-nwZ6nP4g.d.mts +24 -0
  225. package/dist/securityHeaders-nwZ6nP4g.d.ts +24 -0
  226. package/dist/seo/index.d.mts +600 -0
  227. package/dist/seo/index.d.ts +600 -0
  228. package/dist/seo/index.js +883 -0
  229. package/dist/seo/index.js.map +1 -0
  230. package/dist/seo/index.mjs +773 -0
  231. package/dist/seo/index.mjs.map +1 -0
  232. package/dist/seo/register-sitemap-cli.js +151 -0
  233. package/dist/seo/register-sitemap-cli.js.map +1 -0
  234. package/dist/seo/register-sitemap-cli.mjs +144 -0
  235. package/dist/seo/register-sitemap-cli.mjs.map +1 -0
  236. package/dist/seo/server.d.mts +107 -0
  237. package/dist/seo/server.d.ts +107 -0
  238. package/dist/seo/server.js +207 -0
  239. package/dist/seo/server.js.map +1 -0
  240. package/dist/seo/server.mjs +186 -0
  241. package/dist/seo/server.mjs.map +1 -0
  242. package/dist/server-api-EWXKOQZA.mjs +4 -0
  243. package/dist/server-api-EWXKOQZA.mjs.map +1 -0
  244. package/dist/server-api-GJPNRYUP.js +81 -0
  245. package/dist/server-api-GJPNRYUP.js.map +1 -0
  246. package/dist/setup/client.d.mts +60 -0
  247. package/dist/setup/client.d.ts +60 -0
  248. package/dist/setup/client.js +31 -0
  249. package/dist/setup/client.js.map +1 -0
  250. package/dist/setup/client.mjs +6 -0
  251. package/dist/setup/client.mjs.map +1 -0
  252. package/dist/setup/index.d.mts +5 -0
  253. package/dist/setup/index.d.ts +5 -0
  254. package/dist/setup/index.js +35 -0
  255. package/dist/setup/index.js.map +1 -0
  256. package/dist/setup/index.mjs +6 -0
  257. package/dist/setup/index.mjs.map +1 -0
  258. package/dist/setup/server.d.mts +14 -0
  259. package/dist/setup/server.d.ts +14 -0
  260. package/dist/setup/server.js +13 -0
  261. package/dist/setup/server.js.map +1 -0
  262. package/dist/setup/server.mjs +4 -0
  263. package/dist/setup/server.mjs.map +1 -0
  264. package/dist/site-config/index.d.mts +24 -0
  265. package/dist/site-config/index.d.ts +24 -0
  266. package/dist/site-config/index.js +17 -0
  267. package/dist/site-config/index.js.map +1 -0
  268. package/dist/site-config/index.mjs +4 -0
  269. package/dist/site-config/index.mjs.map +1 -0
  270. package/dist/sitemap/index.d.mts +96 -0
  271. package/dist/sitemap/index.d.ts +96 -0
  272. package/dist/sitemap/index.js +288 -0
  273. package/dist/sitemap/index.js.map +1 -0
  274. package/dist/sitemap/index.mjs +285 -0
  275. package/dist/sitemap/index.mjs.map +1 -0
  276. package/dist/socket-loader-J26QHHOB.js +16 -0
  277. package/dist/socket-loader-J26QHHOB.js.map +1 -0
  278. package/dist/socket-loader-R7S2YJ2J.mjs +14 -0
  279. package/dist/socket-loader-R7S2YJ2J.mjs.map +1 -0
  280. package/dist/types-0dmq3k20.d.mts +168 -0
  281. package/dist/types-0dmq3k20.d.ts +168 -0
  282. package/dist/types-Blb2QNkV.d.mts +263 -0
  283. package/dist/types-Blb2QNkV.d.ts +263 -0
  284. package/dist/types-BnCwwUX3.d.mts +250 -0
  285. package/dist/types-BnCwwUX3.d.ts +250 -0
  286. package/dist/types-CGlnp43R.d.mts +312 -0
  287. package/dist/types-CGlnp43R.d.ts +312 -0
  288. package/dist/types-D08004rU.d.mts +179 -0
  289. package/dist/types-D08004rU.d.ts +179 -0
  290. package/dist/types-DNSYU7qI.d.mts +127 -0
  291. package/dist/types-DNSYU7qI.d.ts +127 -0
  292. package/dist/types-KZP_VWZp.d.mts +266 -0
  293. package/dist/types-KZP_VWZp.d.ts +266 -0
  294. package/dist/useEventModal-BVTx69XE.d.mts +274 -0
  295. package/dist/useEventModal-Dx1dItTJ.d.ts +274 -0
  296. package/dist/web-vitals-444RLW3B.js +252 -0
  297. package/dist/web-vitals-444RLW3B.js.map +1 -0
  298. package/dist/web-vitals-KPICZIEF.mjs +241 -0
  299. package/dist/web-vitals-KPICZIEF.mjs.map +1 -0
  300. package/package.json +192 -0
@@ -0,0 +1,288 @@
1
+ 'use strict';
2
+
3
+ var chunkDDKW2FNA_js = require('../chunk-DDKW2FNA.js');
4
+ var chunk7557OTHW_js = require('../chunk-7557OTHW.js');
5
+ require('../chunk-ZSMWDLMK.js');
6
+ var fs = require('fs');
7
+ var path = require('path');
8
+
9
+ var IGNORED_FOLDERS = [
10
+ "api",
11
+ "admin",
12
+ "_uptrade",
13
+ "%5Fuptrade",
14
+ // URL-encoded _uptrade
15
+ "sonor-setup",
16
+ "offline"
17
+ ];
18
+ var PAGE_FILES = ["page.tsx", "page.jsx", "page.js", "page.ts"];
19
+ function isExcluded(path, patterns) {
20
+ return patterns.some((pattern) => {
21
+ if (pattern.endsWith("/*")) {
22
+ const prefix = pattern.slice(0, -2);
23
+ return path.startsWith(prefix);
24
+ }
25
+ if (pattern.endsWith("*")) {
26
+ const prefix = pattern.slice(0, -1);
27
+ return path.startsWith(prefix);
28
+ }
29
+ return path === pattern;
30
+ });
31
+ }
32
+ function discoverPages(appDir, currentPath = "", pages = []) {
33
+ const fullPath = path.join(appDir, currentPath);
34
+ if (!fs.existsSync(fullPath)) {
35
+ return pages;
36
+ }
37
+ const entries = fs.readdirSync(fullPath);
38
+ const hasPage = entries.some((entry) => PAGE_FILES.includes(entry));
39
+ if (hasPage) {
40
+ let urlPath = "/" + currentPath;
41
+ urlPath = urlPath.replace(/\/\([^)]+\)/g, "");
42
+ urlPath = urlPath.replace(/\/+/g, "/");
43
+ if (urlPath !== "/" && urlPath.endsWith("/")) {
44
+ urlPath = urlPath.slice(0, -1);
45
+ }
46
+ pages.push(urlPath);
47
+ }
48
+ for (const entry of entries) {
49
+ const entryPath = path.join(fullPath, entry);
50
+ const stat = fs.statSync(entryPath);
51
+ if (stat.isDirectory()) {
52
+ const folderName = entry.toLowerCase();
53
+ if (IGNORED_FOLDERS.some((ignored) => folderName.includes(ignored))) {
54
+ continue;
55
+ }
56
+ if (entry.startsWith("[") && entry.endsWith("]")) {
57
+ continue;
58
+ }
59
+ if (entry.startsWith("_")) {
60
+ continue;
61
+ }
62
+ discoverPages(appDir, path.join(currentPath, entry), pages);
63
+ }
64
+ }
65
+ return pages;
66
+ }
67
+ function findAppDir() {
68
+ const cwd = process.cwd();
69
+ const candidates = [
70
+ path.join(cwd, "app"),
71
+ path.join(cwd, "src", "app")
72
+ ];
73
+ for (const candidate of candidates) {
74
+ if (fs.existsSync(candidate)) {
75
+ return candidate;
76
+ }
77
+ }
78
+ throw new Error(
79
+ '[site-kit] Could not find app directory. Ensure you have an "app" or "src/app" folder.'
80
+ );
81
+ }
82
+ function getPriority(path, config) {
83
+ if (config.priorities) {
84
+ for (const [pattern, priority] of Object.entries(config.priorities)) {
85
+ if (pattern.endsWith("/*")) {
86
+ const prefix = pattern.slice(0, -2);
87
+ if (path.startsWith(prefix)) return priority;
88
+ } else if (path === pattern) {
89
+ return priority;
90
+ }
91
+ }
92
+ }
93
+ if (path === "/") return 1;
94
+ const depth = path.split("/").filter(Boolean).length;
95
+ if (depth === 1) return 0.8;
96
+ if (depth === 2) return 0.6;
97
+ return config.defaultPriority ?? 0.5;
98
+ }
99
+ async function syncSitemapToPortal(entries, apiUrl, apiKey, options) {
100
+ if (!apiKey) {
101
+ console.warn("[site-kit] No API key provided, skipping sitemap sync");
102
+ return { success: false, created: 0, updated: 0 };
103
+ }
104
+ const normalizedEntries = entries.map((entry) => {
105
+ let path;
106
+ try {
107
+ const url = new URL(entry.url);
108
+ path = url.pathname;
109
+ } catch {
110
+ path = entry.url.startsWith("/") ? entry.url : `/${entry.url}`;
111
+ }
112
+ return {
113
+ path,
114
+ priority: entry.priority,
115
+ changefreq: entry.changeFrequency
116
+ };
117
+ });
118
+ try {
119
+ const body = { entries: normalizedEntries };
120
+ if (options?.optimizeMeta !== void 0) body.optimize_meta = options.optimizeMeta;
121
+ if (options?.awaitMeta !== void 0) body.await_meta = options.awaitMeta;
122
+ const response = await fetch(`${apiUrl}/api/public/seo/register-sitemap`, {
123
+ method: "POST",
124
+ headers: {
125
+ "Content-Type": "application/json",
126
+ "x-api-key": apiKey
127
+ },
128
+ body: JSON.stringify(body)
129
+ });
130
+ if (!response.ok) {
131
+ console.error("[site-kit] Sitemap sync failed:", response.status, await response.text());
132
+ return { success: false, created: 0, updated: 0 };
133
+ }
134
+ const result = await response.json();
135
+ return {
136
+ success: true,
137
+ created: result.created || 0,
138
+ updated: result.updated || 0
139
+ };
140
+ } catch (error) {
141
+ console.error("[site-kit] Sitemap sync error:", error);
142
+ return { success: false, created: 0, updated: 0 };
143
+ }
144
+ }
145
+ function createSitemap(config) {
146
+ return async () => {
147
+ const { exclude = [], defaultChangeFrequency = "weekly" } = config;
148
+ let baseUrl = config.baseUrl;
149
+ if (!baseUrl) {
150
+ const apiKey = config.apiKey ?? process.env.UPTRADE_API_KEY ?? process.env.NEXT_PUBLIC_UPTRADE_API_KEY;
151
+ const site = await chunk7557OTHW_js.getSiteConfig({ apiKey, apiUrl: config.apiUrl });
152
+ baseUrl = site?.site_url ?? process.env.NEXT_PUBLIC_SITE_URL ?? "https://example.com";
153
+ }
154
+ const normalizedBaseUrl = baseUrl.endsWith("/") ? baseUrl.slice(0, -1) : baseUrl;
155
+ const allExclusions = [
156
+ "/api/*",
157
+ "/admin/*",
158
+ "/_uptrade/*",
159
+ "/sonor-setup/*",
160
+ "/offline/*",
161
+ ...exclude
162
+ ];
163
+ let pages = [];
164
+ try {
165
+ const appDir = findAppDir();
166
+ pages = discoverPages(appDir);
167
+ } catch (error) {
168
+ console.warn("[site-kit] Failed to discover pages:", error);
169
+ pages = ["/"];
170
+ }
171
+ pages = pages.filter((page) => !isExcluded(page, allExclusions));
172
+ const entries = pages.map((path) => ({
173
+ url: `${normalizedBaseUrl}${path}`,
174
+ lastModified: /* @__PURE__ */ new Date(),
175
+ changeFrequency: defaultChangeFrequency,
176
+ priority: getPriority(path, config)
177
+ }));
178
+ if (config.additionalPaths) {
179
+ try {
180
+ const additional = await config.additionalPaths();
181
+ for (const item of additional) {
182
+ if (!isExcluded(item.path, allExclusions)) {
183
+ entries.push({
184
+ url: `${normalizedBaseUrl}${item.path}`,
185
+ lastModified: /* @__PURE__ */ new Date(),
186
+ changeFrequency: item.changeFrequency ?? defaultChangeFrequency,
187
+ priority: item.priority ?? getPriority(item.path, config)
188
+ });
189
+ }
190
+ }
191
+ } catch (error) {
192
+ console.warn("[site-kit] Failed to get additional paths:", error);
193
+ }
194
+ }
195
+ if (config.intelligentPriority) {
196
+ try {
197
+ const apiKey = config.apiKey ?? process.env.UPTRADE_API_KEY ?? process.env.NEXT_PUBLIC_UPTRADE_API_KEY ?? "";
198
+ const projectIdMatch = apiKey.match(/^uptrade_([0-9a-f-]{36})_/);
199
+ const projectId = projectIdMatch ? projectIdMatch[1] : null;
200
+ if (projectId) {
201
+ const signalApiUrl = process.env.SIGNAL_API_URL || process.env.NEXT_PUBLIC_SIGNAL_API_URL || "https://signal.uptrademedia.com";
202
+ const visResponse = await fetch(`${signalApiUrl}/skills/seo/visibility/${projectId}`, {
203
+ headers: { "Content-Type": "application/json", "x-api-key": apiKey }
204
+ }).catch(() => null);
205
+ if (visResponse?.ok) {
206
+ const visData = await visResponse.json();
207
+ const scores = visData?.data || visData;
208
+ const scoreMap = new Map(scores.map((s) => [s.page_path, s.overall_score]));
209
+ let appliedCount = 0;
210
+ for (const entry of entries) {
211
+ const path = new URL(entry.url).pathname;
212
+ const visScore = scoreMap.get(path);
213
+ if (visScore !== void 0) {
214
+ const depthBonus = path === "/" ? 1 : Math.max(0.3, 1 - path.split("/").filter(Boolean).length * 0.15);
215
+ entry.priority = Math.round((visScore * 0.4 + depthBonus * 0.3 + (entry.priority ?? 0.5) * 0.3) * 100) / 100;
216
+ appliedCount++;
217
+ }
218
+ }
219
+ if (appliedCount > 0) {
220
+ console.log(`[site-kit] Applied intelligent priority from ${appliedCount} visibility scores`);
221
+ }
222
+ }
223
+ }
224
+ } catch {
225
+ }
226
+ }
227
+ entries.sort((a, b) => (b.priority ?? 0) - (a.priority ?? 0));
228
+ console.log(`[site-kit] Generated sitemap with ${entries.length} pages`);
229
+ if (!config.disableSync) {
230
+ const apiUrl = config.apiUrl || process.env.NEXT_PUBLIC_UPTRADE_API_URL || "https://api.uptrademedia.com";
231
+ const apiKey = config.apiKey || process.env.UPTRADE_API_KEY || process.env.NEXT_PUBLIC_UPTRADE_API_KEY;
232
+ if (apiKey) {
233
+ try {
234
+ const syncOptions = {};
235
+ if (config.awaitMetaOptimization) {
236
+ syncOptions.optimizeMeta = true;
237
+ syncOptions.awaitMeta = true;
238
+ }
239
+ const result = await syncSitemapToPortal(entries, apiUrl, apiKey, syncOptions);
240
+ if (result.success) {
241
+ console.log(`[site-kit] Synced to Portal API: ${result.created} created, ${result.updated} updated`);
242
+ if (config.awaitMetaOptimization) {
243
+ try {
244
+ const pagesResponse = await fetch(`${apiUrl}/api/public/seo/pages`, {
245
+ headers: { "Content-Type": "application/json", "x-api-key": apiKey }
246
+ });
247
+ if (pagesResponse.ok) {
248
+ const pagesData = await pagesResponse.json();
249
+ const allPages = pagesData?.pages || pagesData?.data || [];
250
+ const optimized = allPages.filter((p) => p.managed_title && p.managed_meta_description);
251
+ console.log(`[site-kit] Meta optimization coverage: ${optimized.length}/${allPages.length} pages (${allPages.length ? Math.round(optimized.length / allPages.length * 100) : 0}%)`);
252
+ }
253
+ } catch {
254
+ console.warn("[site-kit] Could not verify meta optimization coverage");
255
+ }
256
+ }
257
+ }
258
+ if (config.optimizedLLMsTxt !== false) {
259
+ try {
260
+ const writeResult = await chunkDDKW2FNA_js.writeLLMsTxtToPublic({
261
+ apiUrl,
262
+ apiKey,
263
+ full: config.optimizedLLMsFullTxt ?? false,
264
+ fallbackToLocal: true,
265
+ getLocalData: config.getLocalData
266
+ });
267
+ if (writeResult.success) {
268
+ console.log(`[site-kit] Wrote llms.txt (optimized=${writeResult.optimized})`);
269
+ }
270
+ } catch (err) {
271
+ console.warn("[site-kit] Failed to write llms.txt:", err.message);
272
+ }
273
+ }
274
+ } catch (err) {
275
+ console.warn("[site-kit] Failed to sync sitemap to Portal API:", err.message);
276
+ }
277
+ } else {
278
+ console.log("[site-kit] No API key found, skipping Portal API sync");
279
+ }
280
+ }
281
+ return entries;
282
+ };
283
+ }
284
+
285
+ exports.createSitemap = createSitemap;
286
+ exports.syncSitemapToPortal = syncSitemapToPortal;
287
+ //# sourceMappingURL=index.js.map
288
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/sitemap/index.ts"],"names":["join","existsSync","readdirSync","statSync","getSiteConfig","writeLLMsTxtToPublic"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * @sonordev/site-kit - Sitemap Generator\n * \n * Automatically generates sitemap.xml from Next.js app directory structure.\n * Works at build time to discover all pages and sync them to Portal API.\n * \n * @example\n * ```ts\n * // app/sitemap.ts\n * import { createSitemap } from '@sonordev/site-kit/sitemap'\n * \n * export default createSitemap({\n * baseUrl: 'https://example.com',\n * // Optional: exclude patterns\n * exclude: ['/admin/*', '/api/*'],\n * // Optional: additional dynamic routes\n * additionalPaths: async () => [\n * { path: '/blog/post-1', priority: 0.7 },\n * ]\n * })\n * ```\n */\n\nimport { readdirSync, statSync, existsSync } from 'fs'\nimport { join } from 'path'\nimport { writeLLMsTxtToPublic } from '../llms/writeLLMsTxt'\nimport { getSiteConfig } from '../site-config'\n\nexport interface SitemapEntry {\n url: string\n lastModified?: Date | string\n changeFrequency?: 'always' | 'hourly' | 'daily' | 'weekly' | 'monthly' | 'yearly' | 'never'\n priority?: number\n}\n\nexport interface SitemapConfig {\n /** Base URL for the site (optional when apiKey is set; then resolved from Portal project-info) */\n baseUrl?: string\n /** Glob patterns to exclude (e.g., ['/admin/*', '/api/*']) */\n exclude?: string[]\n /** Default priority for pages */\n defaultPriority?: number\n /** Default change frequency */\n defaultChangeFrequency?: SitemapEntry['changeFrequency']\n /** Additional paths to include (for dynamic routes) */\n additionalPaths?: () => Promise<Array<{ path: string; priority?: number; changeFrequency?: SitemapEntry['changeFrequency'] }>>\n /** Priority overrides by path pattern */\n priorities?: Record<string, number>\n /** Portal API URL for syncing (defaults to NEXT_PUBLIC_UPTRADE_API_URL or https://api.uptrademedia.com) */\n apiUrl?: string\n /** Portal API key for syncing (defaults to NEXT_PUBLIC_UPTRADE_API_KEY) */\n apiKey?: string\n /** Disable auto-sync to Portal API */\n disableSync?: boolean\n /** When true, await meta optimization before build completes (for build-time llms.txt with managed_llm_schema) */\n awaitMetaOptimization?: boolean\n /** When true (default when API key present), write AI-optimized llms.txt to public/llms.txt after sitemap sync */\n optimizedLLMsTxt?: boolean\n /** When true with optimizedLLMsTxt, also write public/llms-full.txt */\n optimizedLLMsFullTxt?: boolean\n /** When Portal returns empty services, use this to supply local site data for llms.txt */\n getLocalData?: () => Promise<import('../llms/types').LLMsDataResponse | null>\n /** Use Signal AI visibility scores to compute page priority (40% visibility + 30% depth + 30% base) */\n intelligentPriority?: boolean\n}\n\n// Route groups and special folders to ignore\nconst IGNORED_FOLDERS = [\n 'api',\n 'admin',\n '_uptrade',\n '%5Fuptrade', // URL-encoded _uptrade\n 'sonor-setup',\n 'offline',\n]\n\n// File patterns that indicate a page\nconst PAGE_FILES = ['page.tsx', 'page.jsx', 'page.js', 'page.ts']\n\n/**\n * Check if a path matches any exclude pattern\n */\nfunction isExcluded(path: string, patterns: string[]): boolean {\n return patterns.some(pattern => {\n if (pattern.endsWith('/*')) {\n const prefix = pattern.slice(0, -2)\n return path.startsWith(prefix)\n }\n if (pattern.endsWith('*')) {\n const prefix = pattern.slice(0, -1)\n return path.startsWith(prefix)\n }\n return path === pattern\n })\n}\n\n/**\n * Recursively discover pages from the app directory\n */\nfunction discoverPages(\n appDir: string,\n currentPath: string = '',\n pages: string[] = []\n): string[] {\n const fullPath = join(appDir, currentPath)\n \n if (!existsSync(fullPath)) {\n return pages\n }\n\n const entries = readdirSync(fullPath)\n \n // Check if this directory has a page file\n const hasPage = entries.some(entry => PAGE_FILES.includes(entry))\n \n if (hasPage) {\n // Convert directory path to URL path\n let urlPath = '/' + currentPath\n \n // Handle route groups (parentheses) - remove them from URL\n urlPath = urlPath.replace(/\\/\\([^)]+\\)/g, '')\n \n // Clean up path\n urlPath = urlPath.replace(/\\/+/g, '/') // Remove double slashes\n if (urlPath !== '/' && urlPath.endsWith('/')) {\n urlPath = urlPath.slice(0, -1) // Remove trailing slash\n }\n \n pages.push(urlPath)\n }\n \n // Recurse into subdirectories\n for (const entry of entries) {\n const entryPath = join(fullPath, entry)\n const stat = statSync(entryPath)\n \n if (stat.isDirectory()) {\n // Skip ignored folders\n const folderName = entry.toLowerCase()\n if (IGNORED_FOLDERS.some(ignored => folderName.includes(ignored))) {\n continue\n }\n \n // Skip dynamic route segments for now (would need runtime data)\n if (entry.startsWith('[') && entry.endsWith(']')) {\n continue\n }\n \n // Skip private folders\n if (entry.startsWith('_')) {\n continue\n }\n \n discoverPages(appDir, join(currentPath, entry), pages)\n }\n }\n \n return pages\n}\n\n/**\n * Find the app directory from the current working directory\n */\nfunction findAppDir(): string {\n const cwd = process.cwd()\n \n // Try common locations\n const candidates = [\n join(cwd, 'app'),\n join(cwd, 'src', 'app'),\n ]\n \n for (const candidate of candidates) {\n if (existsSync(candidate)) {\n return candidate\n }\n }\n \n throw new Error(\n '[site-kit] Could not find app directory. Ensure you have an \"app\" or \"src/app\" folder.'\n )\n}\n\n/**\n * Get priority for a path based on config\n */\nfunction getPriority(path: string, config: SitemapConfig): number {\n // Check priority overrides\n if (config.priorities) {\n for (const [pattern, priority] of Object.entries(config.priorities)) {\n if (pattern.endsWith('/*')) {\n const prefix = pattern.slice(0, -2)\n if (path.startsWith(prefix)) return priority\n } else if (path === pattern) {\n return priority\n }\n }\n }\n \n // Home page gets highest priority\n if (path === '/') return 1.0\n \n // Shallow pages get higher priority\n const depth = path.split('/').filter(Boolean).length\n if (depth === 1) return 0.8\n if (depth === 2) return 0.6\n \n return config.defaultPriority ?? 0.5\n}\n\nexport interface SyncSitemapOptions {\n /** Trigger Signal AI meta optimization after sync */\n optimizeMeta?: boolean\n /** When true, wait for meta optimization to complete (for build-time llms.txt) */\n awaitMeta?: boolean\n}\n\n/**\n * Sync sitemap entries to Portal API\n * Called automatically by createSitemap at build time\n */\nexport async function syncSitemapToPortal(\n entries: SitemapEntry[],\n apiUrl: string,\n apiKey: string,\n options?: SyncSitemapOptions\n): Promise<{ success: boolean; created: number; updated: number }> {\n if (!apiKey) {\n console.warn('[site-kit] No API key provided, skipping sitemap sync')\n return { success: false, created: 0, updated: 0 }\n }\n \n const normalizedEntries = entries.map(entry => {\n let path: string\n try {\n const url = new URL(entry.url)\n path = url.pathname\n } catch {\n path = entry.url.startsWith('/') ? entry.url : `/${entry.url}`\n }\n \n return {\n path,\n priority: entry.priority,\n changefreq: entry.changeFrequency,\n }\n })\n \n try {\n const body: Record<string, unknown> = { entries: normalizedEntries }\n if (options?.optimizeMeta !== undefined) body.optimize_meta = options.optimizeMeta\n if (options?.awaitMeta !== undefined) body.await_meta = options.awaitMeta\n\n const response = await fetch(`${apiUrl}/api/public/seo/register-sitemap`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'x-api-key': apiKey,\n },\n body: JSON.stringify(body),\n })\n \n if (!response.ok) {\n console.error('[site-kit] Sitemap sync failed:', response.status, await response.text())\n return { success: false, created: 0, updated: 0 }\n }\n \n const result = await response.json()\n \n return {\n success: true,\n created: result.created || 0,\n updated: result.updated || 0,\n }\n } catch (error) {\n console.error('[site-kit] Sitemap sync error:', error)\n return { success: false, created: 0, updated: 0 }\n }\n}\n\n/**\n * Create a sitemap generator function for Next.js\n * Automatically syncs to Portal API at build time.\n * \n * @example\n * ```ts\n * // app/sitemap.ts\n * import { createSitemap } from '@sonordev/site-kit/sitemap'\n * \n * export default createSitemap({\n * baseUrl: 'https://example.com'\n * })\n * ```\n */\nexport function createSitemap(config: SitemapConfig): () => SitemapEntry[] | Promise<SitemapEntry[]> {\n return async () => {\n const { exclude = [], defaultChangeFrequency = 'weekly' } = config\n \n // Resolve baseUrl: from config, or from Portal via apiKey/env\n let baseUrl = config.baseUrl\n if (!baseUrl) {\n const apiKey = config.apiKey ?? process.env.UPTRADE_API_KEY ?? process.env.NEXT_PUBLIC_UPTRADE_API_KEY\n const site = await getSiteConfig({ apiKey, apiUrl: config.apiUrl })\n baseUrl = site?.site_url ?? process.env.NEXT_PUBLIC_SITE_URL ?? 'https://example.com'\n }\n const normalizedBaseUrl = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl\n \n // Default exclusions\n const allExclusions = [\n '/api/*',\n '/admin/*',\n '/_uptrade/*',\n '/sonor-setup/*',\n '/offline/*',\n ...exclude,\n ]\n \n // Discover pages from app directory\n let pages: string[] = []\n try {\n const appDir = findAppDir()\n pages = discoverPages(appDir)\n } catch (error) {\n console.warn('[site-kit] Failed to discover pages:', error)\n // Fall back to just the home page\n pages = ['/']\n }\n \n // Filter out excluded pages\n pages = pages.filter(page => !isExcluded(page, allExclusions))\n \n // Build sitemap entries\n const entries: SitemapEntry[] = pages.map(path => ({\n url: `${normalizedBaseUrl}${path}`,\n lastModified: new Date(),\n changeFrequency: defaultChangeFrequency,\n priority: getPriority(path, config),\n }))\n \n // Add additional dynamic paths if provided\n if (config.additionalPaths) {\n try {\n const additional = await config.additionalPaths()\n for (const item of additional) {\n if (!isExcluded(item.path, allExclusions)) {\n entries.push({\n url: `${normalizedBaseUrl}${item.path}`,\n lastModified: new Date(),\n changeFrequency: item.changeFrequency ?? defaultChangeFrequency,\n priority: item.priority ?? getPriority(item.path, config),\n })\n }\n }\n } catch (error) {\n console.warn('[site-kit] Failed to get additional paths:', error)\n }\n }\n \n // Intelligent priority: blend visibility scores with depth-based priority\n if (config.intelligentPriority) {\n try {\n const apiKey = config.apiKey ?? process.env.UPTRADE_API_KEY ?? process.env.NEXT_PUBLIC_UPTRADE_API_KEY ?? ''\n const projectIdMatch = apiKey.match(/^uptrade_([0-9a-f-]{36})_/)\n const projectId = projectIdMatch ? projectIdMatch[1] : null\n\n if (projectId) {\n const signalApiUrl = process.env.SIGNAL_API_URL || process.env.NEXT_PUBLIC_SIGNAL_API_URL || 'https://signal.uptrademedia.com'\n const visResponse = await fetch(`${signalApiUrl}/skills/seo/visibility/${projectId}`, {\n headers: { 'Content-Type': 'application/json', 'x-api-key': apiKey },\n }).catch(() => null)\n\n if (visResponse?.ok) {\n const visData = await visResponse.json()\n const scores = (visData?.data || visData) as Array<{ page_path: string; overall_score: number }>\n const scoreMap = new Map(scores.map(s => [s.page_path, s.overall_score]))\n\n let appliedCount = 0\n for (const entry of entries) {\n const path = new URL(entry.url).pathname\n const visScore = scoreMap.get(path)\n if (visScore !== undefined) {\n const depthBonus = path === '/' ? 1.0 : Math.max(0.3, 1.0 - (path.split('/').filter(Boolean).length * 0.15))\n // 40% visibility + 30% depth + 30% base\n entry.priority = Math.round((visScore * 0.4 + depthBonus * 0.3 + (entry.priority ?? 0.5) * 0.3) * 100) / 100\n appliedCount++\n }\n }\n if (appliedCount > 0) {\n console.log(`[site-kit] Applied intelligent priority from ${appliedCount} visibility scores`)\n }\n }\n }\n } catch {\n // Signal API unavailable — fall back to static priority\n }\n }\n\n // Sort by priority descending\n entries.sort((a, b) => (b.priority ?? 0) - (a.priority ?? 0))\n\n console.log(`[site-kit] Generated sitemap with ${entries.length} pages`)\n \n // Auto-sync to Portal API at build time (unless disabled)\n if (!config.disableSync) {\n const apiUrl = config.apiUrl || process.env.NEXT_PUBLIC_UPTRADE_API_URL || 'https://api.uptrademedia.com'\n // Server-side operations use UPTRADE_API_KEY, not NEXT_PUBLIC_\n const apiKey = config.apiKey || process.env.UPTRADE_API_KEY || process.env.NEXT_PUBLIC_UPTRADE_API_KEY\n \n if (apiKey) {\n try {\n const syncOptions: SyncSitemapOptions = {}\n if (config.awaitMetaOptimization) {\n syncOptions.optimizeMeta = true\n syncOptions.awaitMeta = true\n }\n const result = await syncSitemapToPortal(entries, apiUrl, apiKey, syncOptions)\n if (result.success) {\n console.log(`[site-kit] Synced to Portal API: ${result.created} created, ${result.updated} updated`)\n\n // Verification: report meta optimization coverage after sync\n if (config.awaitMetaOptimization) {\n try {\n const pagesResponse = await fetch(`${apiUrl}/api/public/seo/pages`, {\n headers: { 'Content-Type': 'application/json', 'x-api-key': apiKey },\n })\n if (pagesResponse.ok) {\n const pagesData = await pagesResponse.json()\n const allPages: Array<{ managed_title?: string; managed_meta_description?: string }> = pagesData?.pages || pagesData?.data || []\n const optimized = allPages.filter(p => p.managed_title && p.managed_meta_description)\n console.log(`[site-kit] Meta optimization coverage: ${optimized.length}/${allPages.length} pages (${allPages.length ? Math.round((optimized.length / allPages.length) * 100) : 0}%)`)\n }\n } catch {\n console.warn('[site-kit] Could not verify meta optimization coverage')\n }\n }\n }\n\n // Write AI-optimized llms.txt after sync (default true when API key present)\n if (config.optimizedLLMsTxt !== false) {\n try {\n const writeResult = await writeLLMsTxtToPublic({\n apiUrl,\n apiKey,\n full: config.optimizedLLMsFullTxt ?? false,\n fallbackToLocal: true,\n getLocalData: config.getLocalData,\n })\n if (writeResult.success) {\n console.log(`[site-kit] Wrote llms.txt (optimized=${writeResult.optimized})`)\n }\n } catch (err) {\n console.warn('[site-kit] Failed to write llms.txt:', (err as Error).message)\n }\n }\n } catch (err) {\n console.warn('[site-kit] Failed to sync sitemap to Portal API:', (err as Error).message)\n }\n } else {\n console.log('[site-kit] No API key found, skipping Portal API sync')\n }\n }\n \n return entries\n }\n}\n"]}
@@ -0,0 +1,285 @@
1
+ import { writeLLMsTxtToPublic } from '../chunk-OIIKTGRL.mjs';
2
+ import { getSiteConfig } from '../chunk-GCJXQ4AG.mjs';
3
+ import '../chunk-4XPGGLVP.mjs';
4
+ import { existsSync, readdirSync, statSync } from 'fs';
5
+ import { join } from 'path';
6
+
7
+ var IGNORED_FOLDERS = [
8
+ "api",
9
+ "admin",
10
+ "_uptrade",
11
+ "%5Fuptrade",
12
+ // URL-encoded _uptrade
13
+ "sonor-setup",
14
+ "offline"
15
+ ];
16
+ var PAGE_FILES = ["page.tsx", "page.jsx", "page.js", "page.ts"];
17
+ function isExcluded(path, patterns) {
18
+ return patterns.some((pattern) => {
19
+ if (pattern.endsWith("/*")) {
20
+ const prefix = pattern.slice(0, -2);
21
+ return path.startsWith(prefix);
22
+ }
23
+ if (pattern.endsWith("*")) {
24
+ const prefix = pattern.slice(0, -1);
25
+ return path.startsWith(prefix);
26
+ }
27
+ return path === pattern;
28
+ });
29
+ }
30
+ function discoverPages(appDir, currentPath = "", pages = []) {
31
+ const fullPath = join(appDir, currentPath);
32
+ if (!existsSync(fullPath)) {
33
+ return pages;
34
+ }
35
+ const entries = readdirSync(fullPath);
36
+ const hasPage = entries.some((entry) => PAGE_FILES.includes(entry));
37
+ if (hasPage) {
38
+ let urlPath = "/" + currentPath;
39
+ urlPath = urlPath.replace(/\/\([^)]+\)/g, "");
40
+ urlPath = urlPath.replace(/\/+/g, "/");
41
+ if (urlPath !== "/" && urlPath.endsWith("/")) {
42
+ urlPath = urlPath.slice(0, -1);
43
+ }
44
+ pages.push(urlPath);
45
+ }
46
+ for (const entry of entries) {
47
+ const entryPath = join(fullPath, entry);
48
+ const stat = statSync(entryPath);
49
+ if (stat.isDirectory()) {
50
+ const folderName = entry.toLowerCase();
51
+ if (IGNORED_FOLDERS.some((ignored) => folderName.includes(ignored))) {
52
+ continue;
53
+ }
54
+ if (entry.startsWith("[") && entry.endsWith("]")) {
55
+ continue;
56
+ }
57
+ if (entry.startsWith("_")) {
58
+ continue;
59
+ }
60
+ discoverPages(appDir, join(currentPath, entry), pages);
61
+ }
62
+ }
63
+ return pages;
64
+ }
65
+ function findAppDir() {
66
+ const cwd = process.cwd();
67
+ const candidates = [
68
+ join(cwd, "app"),
69
+ join(cwd, "src", "app")
70
+ ];
71
+ for (const candidate of candidates) {
72
+ if (existsSync(candidate)) {
73
+ return candidate;
74
+ }
75
+ }
76
+ throw new Error(
77
+ '[site-kit] Could not find app directory. Ensure you have an "app" or "src/app" folder.'
78
+ );
79
+ }
80
+ function getPriority(path, config) {
81
+ if (config.priorities) {
82
+ for (const [pattern, priority] of Object.entries(config.priorities)) {
83
+ if (pattern.endsWith("/*")) {
84
+ const prefix = pattern.slice(0, -2);
85
+ if (path.startsWith(prefix)) return priority;
86
+ } else if (path === pattern) {
87
+ return priority;
88
+ }
89
+ }
90
+ }
91
+ if (path === "/") return 1;
92
+ const depth = path.split("/").filter(Boolean).length;
93
+ if (depth === 1) return 0.8;
94
+ if (depth === 2) return 0.6;
95
+ return config.defaultPriority ?? 0.5;
96
+ }
97
+ async function syncSitemapToPortal(entries, apiUrl, apiKey, options) {
98
+ if (!apiKey) {
99
+ console.warn("[site-kit] No API key provided, skipping sitemap sync");
100
+ return { success: false, created: 0, updated: 0 };
101
+ }
102
+ const normalizedEntries = entries.map((entry) => {
103
+ let path;
104
+ try {
105
+ const url = new URL(entry.url);
106
+ path = url.pathname;
107
+ } catch {
108
+ path = entry.url.startsWith("/") ? entry.url : `/${entry.url}`;
109
+ }
110
+ return {
111
+ path,
112
+ priority: entry.priority,
113
+ changefreq: entry.changeFrequency
114
+ };
115
+ });
116
+ try {
117
+ const body = { entries: normalizedEntries };
118
+ if (options?.optimizeMeta !== void 0) body.optimize_meta = options.optimizeMeta;
119
+ if (options?.awaitMeta !== void 0) body.await_meta = options.awaitMeta;
120
+ const response = await fetch(`${apiUrl}/api/public/seo/register-sitemap`, {
121
+ method: "POST",
122
+ headers: {
123
+ "Content-Type": "application/json",
124
+ "x-api-key": apiKey
125
+ },
126
+ body: JSON.stringify(body)
127
+ });
128
+ if (!response.ok) {
129
+ console.error("[site-kit] Sitemap sync failed:", response.status, await response.text());
130
+ return { success: false, created: 0, updated: 0 };
131
+ }
132
+ const result = await response.json();
133
+ return {
134
+ success: true,
135
+ created: result.created || 0,
136
+ updated: result.updated || 0
137
+ };
138
+ } catch (error) {
139
+ console.error("[site-kit] Sitemap sync error:", error);
140
+ return { success: false, created: 0, updated: 0 };
141
+ }
142
+ }
143
+ function createSitemap(config) {
144
+ return async () => {
145
+ const { exclude = [], defaultChangeFrequency = "weekly" } = config;
146
+ let baseUrl = config.baseUrl;
147
+ if (!baseUrl) {
148
+ const apiKey = config.apiKey ?? process.env.UPTRADE_API_KEY ?? process.env.NEXT_PUBLIC_UPTRADE_API_KEY;
149
+ const site = await getSiteConfig({ apiKey, apiUrl: config.apiUrl });
150
+ baseUrl = site?.site_url ?? process.env.NEXT_PUBLIC_SITE_URL ?? "https://example.com";
151
+ }
152
+ const normalizedBaseUrl = baseUrl.endsWith("/") ? baseUrl.slice(0, -1) : baseUrl;
153
+ const allExclusions = [
154
+ "/api/*",
155
+ "/admin/*",
156
+ "/_uptrade/*",
157
+ "/sonor-setup/*",
158
+ "/offline/*",
159
+ ...exclude
160
+ ];
161
+ let pages = [];
162
+ try {
163
+ const appDir = findAppDir();
164
+ pages = discoverPages(appDir);
165
+ } catch (error) {
166
+ console.warn("[site-kit] Failed to discover pages:", error);
167
+ pages = ["/"];
168
+ }
169
+ pages = pages.filter((page) => !isExcluded(page, allExclusions));
170
+ const entries = pages.map((path) => ({
171
+ url: `${normalizedBaseUrl}${path}`,
172
+ lastModified: /* @__PURE__ */ new Date(),
173
+ changeFrequency: defaultChangeFrequency,
174
+ priority: getPriority(path, config)
175
+ }));
176
+ if (config.additionalPaths) {
177
+ try {
178
+ const additional = await config.additionalPaths();
179
+ for (const item of additional) {
180
+ if (!isExcluded(item.path, allExclusions)) {
181
+ entries.push({
182
+ url: `${normalizedBaseUrl}${item.path}`,
183
+ lastModified: /* @__PURE__ */ new Date(),
184
+ changeFrequency: item.changeFrequency ?? defaultChangeFrequency,
185
+ priority: item.priority ?? getPriority(item.path, config)
186
+ });
187
+ }
188
+ }
189
+ } catch (error) {
190
+ console.warn("[site-kit] Failed to get additional paths:", error);
191
+ }
192
+ }
193
+ if (config.intelligentPriority) {
194
+ try {
195
+ const apiKey = config.apiKey ?? process.env.UPTRADE_API_KEY ?? process.env.NEXT_PUBLIC_UPTRADE_API_KEY ?? "";
196
+ const projectIdMatch = apiKey.match(/^uptrade_([0-9a-f-]{36})_/);
197
+ const projectId = projectIdMatch ? projectIdMatch[1] : null;
198
+ if (projectId) {
199
+ const signalApiUrl = process.env.SIGNAL_API_URL || process.env.NEXT_PUBLIC_SIGNAL_API_URL || "https://signal.uptrademedia.com";
200
+ const visResponse = await fetch(`${signalApiUrl}/skills/seo/visibility/${projectId}`, {
201
+ headers: { "Content-Type": "application/json", "x-api-key": apiKey }
202
+ }).catch(() => null);
203
+ if (visResponse?.ok) {
204
+ const visData = await visResponse.json();
205
+ const scores = visData?.data || visData;
206
+ const scoreMap = new Map(scores.map((s) => [s.page_path, s.overall_score]));
207
+ let appliedCount = 0;
208
+ for (const entry of entries) {
209
+ const path = new URL(entry.url).pathname;
210
+ const visScore = scoreMap.get(path);
211
+ if (visScore !== void 0) {
212
+ const depthBonus = path === "/" ? 1 : Math.max(0.3, 1 - path.split("/").filter(Boolean).length * 0.15);
213
+ entry.priority = Math.round((visScore * 0.4 + depthBonus * 0.3 + (entry.priority ?? 0.5) * 0.3) * 100) / 100;
214
+ appliedCount++;
215
+ }
216
+ }
217
+ if (appliedCount > 0) {
218
+ console.log(`[site-kit] Applied intelligent priority from ${appliedCount} visibility scores`);
219
+ }
220
+ }
221
+ }
222
+ } catch {
223
+ }
224
+ }
225
+ entries.sort((a, b) => (b.priority ?? 0) - (a.priority ?? 0));
226
+ console.log(`[site-kit] Generated sitemap with ${entries.length} pages`);
227
+ if (!config.disableSync) {
228
+ const apiUrl = config.apiUrl || process.env.NEXT_PUBLIC_UPTRADE_API_URL || "https://api.uptrademedia.com";
229
+ const apiKey = config.apiKey || process.env.UPTRADE_API_KEY || process.env.NEXT_PUBLIC_UPTRADE_API_KEY;
230
+ if (apiKey) {
231
+ try {
232
+ const syncOptions = {};
233
+ if (config.awaitMetaOptimization) {
234
+ syncOptions.optimizeMeta = true;
235
+ syncOptions.awaitMeta = true;
236
+ }
237
+ const result = await syncSitemapToPortal(entries, apiUrl, apiKey, syncOptions);
238
+ if (result.success) {
239
+ console.log(`[site-kit] Synced to Portal API: ${result.created} created, ${result.updated} updated`);
240
+ if (config.awaitMetaOptimization) {
241
+ try {
242
+ const pagesResponse = await fetch(`${apiUrl}/api/public/seo/pages`, {
243
+ headers: { "Content-Type": "application/json", "x-api-key": apiKey }
244
+ });
245
+ if (pagesResponse.ok) {
246
+ const pagesData = await pagesResponse.json();
247
+ const allPages = pagesData?.pages || pagesData?.data || [];
248
+ const optimized = allPages.filter((p) => p.managed_title && p.managed_meta_description);
249
+ console.log(`[site-kit] Meta optimization coverage: ${optimized.length}/${allPages.length} pages (${allPages.length ? Math.round(optimized.length / allPages.length * 100) : 0}%)`);
250
+ }
251
+ } catch {
252
+ console.warn("[site-kit] Could not verify meta optimization coverage");
253
+ }
254
+ }
255
+ }
256
+ if (config.optimizedLLMsTxt !== false) {
257
+ try {
258
+ const writeResult = await writeLLMsTxtToPublic({
259
+ apiUrl,
260
+ apiKey,
261
+ full: config.optimizedLLMsFullTxt ?? false,
262
+ fallbackToLocal: true,
263
+ getLocalData: config.getLocalData
264
+ });
265
+ if (writeResult.success) {
266
+ console.log(`[site-kit] Wrote llms.txt (optimized=${writeResult.optimized})`);
267
+ }
268
+ } catch (err) {
269
+ console.warn("[site-kit] Failed to write llms.txt:", err.message);
270
+ }
271
+ }
272
+ } catch (err) {
273
+ console.warn("[site-kit] Failed to sync sitemap to Portal API:", err.message);
274
+ }
275
+ } else {
276
+ console.log("[site-kit] No API key found, skipping Portal API sync");
277
+ }
278
+ }
279
+ return entries;
280
+ };
281
+ }
282
+
283
+ export { createSitemap, syncSitemapToPortal };
284
+ //# sourceMappingURL=index.mjs.map
285
+ //# sourceMappingURL=index.mjs.map