@brainfish-ai/devdoc 0.1.21

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 (268) hide show
  1. package/LICENSE +33 -0
  2. package/README.md +415 -0
  3. package/bin/devdoc.js +13 -0
  4. package/dist/cli/commands/build.d.ts +5 -0
  5. package/dist/cli/commands/build.js +87 -0
  6. package/dist/cli/commands/check.d.ts +1 -0
  7. package/dist/cli/commands/check.js +143 -0
  8. package/dist/cli/commands/create.d.ts +24 -0
  9. package/dist/cli/commands/create.js +387 -0
  10. package/dist/cli/commands/deploy.d.ts +9 -0
  11. package/dist/cli/commands/deploy.js +433 -0
  12. package/dist/cli/commands/dev.d.ts +6 -0
  13. package/dist/cli/commands/dev.js +139 -0
  14. package/dist/cli/commands/init.d.ts +11 -0
  15. package/dist/cli/commands/init.js +238 -0
  16. package/dist/cli/commands/keys.d.ts +12 -0
  17. package/dist/cli/commands/keys.js +165 -0
  18. package/dist/cli/commands/start.d.ts +5 -0
  19. package/dist/cli/commands/start.js +56 -0
  20. package/dist/cli/commands/upload.d.ts +13 -0
  21. package/dist/cli/commands/upload.js +238 -0
  22. package/dist/cli/commands/whoami.d.ts +8 -0
  23. package/dist/cli/commands/whoami.js +91 -0
  24. package/dist/cli/index.d.ts +1 -0
  25. package/dist/cli/index.js +106 -0
  26. package/dist/config/index.d.ts +80 -0
  27. package/dist/config/index.js +133 -0
  28. package/dist/constants.d.ts +9 -0
  29. package/dist/constants.js +13 -0
  30. package/dist/index.d.ts +7 -0
  31. package/dist/index.js +12 -0
  32. package/dist/utils/logger.d.ts +16 -0
  33. package/dist/utils/logger.js +61 -0
  34. package/dist/utils/paths.d.ts +16 -0
  35. package/dist/utils/paths.js +50 -0
  36. package/package.json +51 -0
  37. package/renderer/app/api/assets/[...path]/route.ts +123 -0
  38. package/renderer/app/api/assets/route.ts +124 -0
  39. package/renderer/app/api/assets/upload/route.ts +177 -0
  40. package/renderer/app/api/auth-schemes/route.ts +77 -0
  41. package/renderer/app/api/chat/route.ts +858 -0
  42. package/renderer/app/api/codegen/route.ts +72 -0
  43. package/renderer/app/api/collections/route.ts +1016 -0
  44. package/renderer/app/api/debug/route.ts +53 -0
  45. package/renderer/app/api/deploy/route.ts +234 -0
  46. package/renderer/app/api/device/route.ts +42 -0
  47. package/renderer/app/api/docs/route.ts +187 -0
  48. package/renderer/app/api/keys/regenerate/route.ts +80 -0
  49. package/renderer/app/api/openapi-spec/route.ts +151 -0
  50. package/renderer/app/api/projects/[slug]/route.ts +153 -0
  51. package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
  52. package/renderer/app/api/projects/register/route.ts +152 -0
  53. package/renderer/app/api/proxy/route.ts +149 -0
  54. package/renderer/app/api/proxy-stream/route.ts +168 -0
  55. package/renderer/app/api/redirects/route.ts +47 -0
  56. package/renderer/app/api/schema/route.ts +65 -0
  57. package/renderer/app/api/subdomains/check/route.ts +172 -0
  58. package/renderer/app/api/suggestions/route.ts +144 -0
  59. package/renderer/app/favicon.ico +0 -0
  60. package/renderer/app/globals.css +1103 -0
  61. package/renderer/app/layout.tsx +47 -0
  62. package/renderer/app/llms-full.txt/route.ts +346 -0
  63. package/renderer/app/llms.txt/route.ts +279 -0
  64. package/renderer/app/page.tsx +14 -0
  65. package/renderer/app/robots.txt/route.ts +84 -0
  66. package/renderer/app/sitemap.xml/route.ts +199 -0
  67. package/renderer/components/docs/index.ts +12 -0
  68. package/renderer/components/docs/mdx/accordion.tsx +169 -0
  69. package/renderer/components/docs/mdx/badge.tsx +132 -0
  70. package/renderer/components/docs/mdx/callouts.tsx +154 -0
  71. package/renderer/components/docs/mdx/cards.tsx +213 -0
  72. package/renderer/components/docs/mdx/changelog.tsx +120 -0
  73. package/renderer/components/docs/mdx/code-block.tsx +186 -0
  74. package/renderer/components/docs/mdx/code-group.tsx +421 -0
  75. package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
  76. package/renderer/components/docs/mdx/frame.tsx +112 -0
  77. package/renderer/components/docs/mdx/highlight.tsx +151 -0
  78. package/renderer/components/docs/mdx/iframe.tsx +134 -0
  79. package/renderer/components/docs/mdx/image.tsx +235 -0
  80. package/renderer/components/docs/mdx/index.ts +204 -0
  81. package/renderer/components/docs/mdx/mermaid.tsx +240 -0
  82. package/renderer/components/docs/mdx/param-field.tsx +200 -0
  83. package/renderer/components/docs/mdx/steps.tsx +113 -0
  84. package/renderer/components/docs/mdx/tabs.tsx +86 -0
  85. package/renderer/components/docs/mdx-renderer.tsx +100 -0
  86. package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
  87. package/renderer/components/docs/navigation/index.ts +8 -0
  88. package/renderer/components/docs/navigation/page-nav.tsx +64 -0
  89. package/renderer/components/docs/navigation/sidebar.tsx +515 -0
  90. package/renderer/components/docs/navigation/toc.tsx +113 -0
  91. package/renderer/components/docs/notice.tsx +105 -0
  92. package/renderer/components/docs-header.tsx +274 -0
  93. package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
  94. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
  95. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
  96. package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
  97. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
  98. package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
  99. package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
  100. package/renderer/components/docs-viewer/agent/index.tsx +6 -0
  101. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
  102. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
  103. package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
  104. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
  105. package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
  106. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
  107. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
  108. package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
  109. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
  110. package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
  111. package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
  112. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
  113. package/renderer/components/docs-viewer/content/index.tsx +29 -0
  114. package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
  115. package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
  116. package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
  117. package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
  118. package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
  119. package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
  120. package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
  121. package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
  122. package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
  123. package/renderer/components/docs-viewer/index.tsx +1466 -0
  124. package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
  125. package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
  126. package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
  127. package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
  128. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
  129. package/renderer/components/docs-viewer/playground/index.tsx +671 -0
  130. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
  131. package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
  132. package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
  133. package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
  134. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
  135. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
  136. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
  137. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
  138. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
  139. package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
  140. package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
  141. package/renderer/components/docs-viewer/search/index.ts +2 -0
  142. package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
  143. package/renderer/components/docs-viewer/search/use-search.ts +117 -0
  144. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
  145. package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
  146. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
  147. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
  148. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
  149. package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
  150. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
  151. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
  152. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
  153. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
  154. package/renderer/components/theme-provider.tsx +11 -0
  155. package/renderer/components/theme-toggle.tsx +76 -0
  156. package/renderer/components/ui/badge.tsx +46 -0
  157. package/renderer/components/ui/button.tsx +59 -0
  158. package/renderer/components/ui/dialog.tsx +118 -0
  159. package/renderer/components/ui/dropdown-menu.tsx +257 -0
  160. package/renderer/components/ui/input.tsx +21 -0
  161. package/renderer/components/ui/label.tsx +24 -0
  162. package/renderer/components/ui/navigation-menu.tsx +168 -0
  163. package/renderer/components/ui/select.tsx +190 -0
  164. package/renderer/components/ui/spinner.tsx +114 -0
  165. package/renderer/components/ui/tabs.tsx +66 -0
  166. package/renderer/components/ui/tooltip.tsx +61 -0
  167. package/renderer/hooks/use-code-copy.ts +88 -0
  168. package/renderer/hooks/use-openapi-title.ts +44 -0
  169. package/renderer/lib/api-docs/agent/index.ts +6 -0
  170. package/renderer/lib/api-docs/agent/indexer.ts +323 -0
  171. package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
  172. package/renderer/lib/api-docs/agent/types.ts +116 -0
  173. package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
  174. package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
  175. package/renderer/lib/api-docs/auth/crypto.ts +89 -0
  176. package/renderer/lib/api-docs/auth/index.ts +4 -0
  177. package/renderer/lib/api-docs/code-editor/db.ts +164 -0
  178. package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
  179. package/renderer/lib/api-docs/code-editor/index.ts +6 -0
  180. package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
  181. package/renderer/lib/api-docs/code-editor/types.ts +105 -0
  182. package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
  183. package/renderer/lib/api-docs/codegen/har.ts +251 -0
  184. package/renderer/lib/api-docs/codegen/index.ts +159 -0
  185. package/renderer/lib/api-docs/factories.ts +151 -0
  186. package/renderer/lib/api-docs/index.ts +17 -0
  187. package/renderer/lib/api-docs/mobile-context.tsx +112 -0
  188. package/renderer/lib/api-docs/navigation-context.tsx +88 -0
  189. package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
  190. package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
  191. package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
  192. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
  193. package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
  194. package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
  195. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
  196. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
  197. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
  198. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
  199. package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
  200. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
  201. package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
  202. package/renderer/lib/api-docs/playground/context.tsx +107 -0
  203. package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
  204. package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
  205. package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
  206. package/renderer/lib/api-docs/playground/types.ts +35 -0
  207. package/renderer/lib/api-docs/types.ts +269 -0
  208. package/renderer/lib/api-docs/utils.ts +311 -0
  209. package/renderer/lib/cache.ts +193 -0
  210. package/renderer/lib/docs/config/index.ts +29 -0
  211. package/renderer/lib/docs/config/loader.ts +142 -0
  212. package/renderer/lib/docs/config/schema.ts +298 -0
  213. package/renderer/lib/docs/index.ts +12 -0
  214. package/renderer/lib/docs/mdx/compiler.ts +176 -0
  215. package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
  216. package/renderer/lib/docs/mdx/index.ts +26 -0
  217. package/renderer/lib/docs/navigation/generator.ts +348 -0
  218. package/renderer/lib/docs/navigation/index.ts +12 -0
  219. package/renderer/lib/docs/navigation/types.ts +123 -0
  220. package/renderer/lib/docs-navigation-context.tsx +80 -0
  221. package/renderer/lib/multi-tenant/context.ts +105 -0
  222. package/renderer/lib/storage/blob.ts +845 -0
  223. package/renderer/lib/utils.ts +6 -0
  224. package/renderer/next.config.ts +76 -0
  225. package/renderer/package.json +66 -0
  226. package/renderer/postcss.config.mjs +5 -0
  227. package/renderer/public/assets/images/screenshot.png +0 -0
  228. package/renderer/public/assets/logo/dark.svg +9 -0
  229. package/renderer/public/assets/logo/light.svg +9 -0
  230. package/renderer/public/assets/logo.svg +9 -0
  231. package/renderer/public/file.svg +1 -0
  232. package/renderer/public/globe.svg +1 -0
  233. package/renderer/public/icon.png +0 -0
  234. package/renderer/public/logo.svg +9 -0
  235. package/renderer/public/window.svg +1 -0
  236. package/renderer/tsconfig.json +28 -0
  237. package/templates/basic/README.md +139 -0
  238. package/templates/basic/assets/favicon.svg +4 -0
  239. package/templates/basic/assets/logo.svg +9 -0
  240. package/templates/basic/docs.json +47 -0
  241. package/templates/basic/guides/configuration.mdx +149 -0
  242. package/templates/basic/guides/overview.mdx +96 -0
  243. package/templates/basic/index.mdx +39 -0
  244. package/templates/basic/package.json +14 -0
  245. package/templates/basic/quickstart.mdx +92 -0
  246. package/templates/basic/vercel.json +6 -0
  247. package/templates/graphql/README.md +139 -0
  248. package/templates/graphql/api-reference/schema.graphql +305 -0
  249. package/templates/graphql/assets/favicon.svg +4 -0
  250. package/templates/graphql/assets/logo.svg +9 -0
  251. package/templates/graphql/docs.json +54 -0
  252. package/templates/graphql/guides/configuration.mdx +149 -0
  253. package/templates/graphql/guides/overview.mdx +96 -0
  254. package/templates/graphql/index.mdx +39 -0
  255. package/templates/graphql/package.json +14 -0
  256. package/templates/graphql/quickstart.mdx +92 -0
  257. package/templates/graphql/vercel.json +6 -0
  258. package/templates/openapi/README.md +139 -0
  259. package/templates/openapi/api-reference/openapi.json +419 -0
  260. package/templates/openapi/assets/favicon.svg +4 -0
  261. package/templates/openapi/assets/logo.svg +9 -0
  262. package/templates/openapi/docs.json +61 -0
  263. package/templates/openapi/guides/configuration.mdx +149 -0
  264. package/templates/openapi/guides/overview.mdx +96 -0
  265. package/templates/openapi/index.mdx +39 -0
  266. package/templates/openapi/package.json +14 -0
  267. package/templates/openapi/quickstart.mdx +92 -0
  268. package/templates/openapi/vercel.json +6 -0
@@ -0,0 +1,5 @@
1
+ interface StartOptions {
2
+ port: string;
3
+ }
4
+ export declare function start(options: StartOptions): Promise<void>;
5
+ export {};
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.start = start;
7
+ const child_process_1 = require("child_process");
8
+ const path_1 = __importDefault(require("path"));
9
+ const fs_extra_1 = __importDefault(require("fs-extra"));
10
+ const logger_1 = require("../../utils/logger");
11
+ const paths_1 = require("../../utils/paths");
12
+ async function start(options) {
13
+ const projectRoot = process.cwd();
14
+ logger_1.logger.info('Starting production server...');
15
+ // Check for build output
16
+ const buildDir = path_1.default.join(projectRoot, 'dist', '.next');
17
+ if (!fs_extra_1.default.existsSync(buildDir)) {
18
+ logger_1.logger.error('No build found. Run "devdoc build" first.');
19
+ process.exit(1);
20
+ }
21
+ // Get the renderer directory
22
+ const rendererDir = (0, paths_1.getRendererDir)();
23
+ if (!rendererDir || !fs_extra_1.default.existsSync(rendererDir)) {
24
+ logger_1.logger.error('Renderer not found. Package may be corrupted.');
25
+ process.exit(1);
26
+ }
27
+ // Set environment variables
28
+ const env = {
29
+ ...process.env,
30
+ STARTER_PATH: projectRoot,
31
+ PORT: options.port,
32
+ };
33
+ logger_1.logger.info(`Starting server at http://localhost:${options.port}`);
34
+ // Start Next.js production server
35
+ const child = (0, child_process_1.spawn)('npx', ['next', 'start', '-p', options.port], {
36
+ cwd: rendererDir,
37
+ env,
38
+ stdio: 'inherit',
39
+ shell: true,
40
+ });
41
+ child.on('error', (error) => {
42
+ logger_1.logger.error(`Failed to start server: ${error.message}`);
43
+ process.exit(1);
44
+ });
45
+ child.on('exit', (code) => {
46
+ process.exit(code || 0);
47
+ });
48
+ // Handle termination
49
+ process.on('SIGINT', () => {
50
+ child.kill('SIGINT');
51
+ });
52
+ process.on('SIGTERM', () => {
53
+ child.kill('SIGTERM');
54
+ });
55
+ }
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhcnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY2xpL2NvbW1hbmRzL3N0YXJ0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBVUEsc0JBMERDO0FBcEVELGlEQUFzQztBQUN0QyxnREFBd0I7QUFDeEIsd0RBQTBCO0FBQzFCLCtDQUE0QztBQUM1Qyw2Q0FBbUQ7QUFNNUMsS0FBSyxVQUFVLEtBQUssQ0FBQyxPQUFxQjtJQUMvQyxNQUFNLFdBQVcsR0FBRyxPQUFPLENBQUMsR0FBRyxFQUFFLENBQUM7SUFFbEMsZUFBTSxDQUFDLElBQUksQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO0lBRTdDLHlCQUF5QjtJQUN6QixNQUFNLFFBQVEsR0FBRyxjQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxNQUFNLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDekQsSUFBSSxDQUFDLGtCQUFFLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUM7UUFDN0IsZUFBTSxDQUFDLEtBQUssQ0FBQywyQ0FBMkMsQ0FBQyxDQUFDO1FBQzFELE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDbEIsQ0FBQztJQUVELDZCQUE2QjtJQUM3QixNQUFNLFdBQVcsR0FBRyxJQUFBLHNCQUFjLEdBQUUsQ0FBQztJQUVyQyxJQUFJLENBQUMsV0FBVyxJQUFJLENBQUMsa0JBQUUsQ0FBQyxVQUFVLENBQUMsV0FBVyxDQUFDLEVBQUUsQ0FBQztRQUNoRCxlQUFNLENBQUMsS0FBSyxDQUFDLCtDQUErQyxDQUFDLENBQUM7UUFDOUQsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsQixDQUFDO0lBRUQsNEJBQTRCO0lBQzVCLE1BQU0sR0FBRyxHQUFHO1FBQ1YsR0FBRyxPQUFPLENBQUMsR0FBRztRQUNkLFlBQVksRUFBRSxXQUFXO1FBQ3pCLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSTtLQUNuQixDQUFDO0lBRUYsZUFBTSxDQUFDLElBQUksQ0FBQyx1Q0FBdUMsT0FBTyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7SUFFbkUsa0NBQWtDO0lBQ2xDLE1BQU0sS0FBSyxHQUFHLElBQUEscUJBQUssRUFDakIsS0FBSyxFQUNMLENBQUMsTUFBTSxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUUsT0FBTyxDQUFDLElBQUksQ0FBQyxFQUNyQztRQUNFLEdBQUcsRUFBRSxXQUFXO1FBQ2hCLEdBQUc7UUFDSCxLQUFLLEVBQUUsU0FBUztRQUNoQixLQUFLLEVBQUUsSUFBSTtLQUNaLENBQ0YsQ0FBQztJQUVGLEtBQUssQ0FBQyxFQUFFLENBQUMsT0FBTyxFQUFFLENBQUMsS0FBSyxFQUFFLEVBQUU7UUFDMUIsZUFBTSxDQUFDLEtBQUssQ0FBQywyQkFBMkIsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFDekQsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsQixDQUFDLENBQUMsQ0FBQztJQUVILEtBQUssQ0FBQyxFQUFFLENBQUMsTUFBTSxFQUFFLENBQUMsSUFBSSxFQUFFLEVBQUU7UUFDeEIsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxDQUFDLENBQUM7SUFDMUIsQ0FBQyxDQUFDLENBQUM7SUFFSCxxQkFBcUI7SUFDckIsT0FBTyxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsR0FBRyxFQUFFO1FBQ3hCLEtBQUssQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDdkIsQ0FBQyxDQUFDLENBQUM7SUFFSCxPQUFPLENBQUMsRUFBRSxDQUFDLFNBQVMsRUFBRSxHQUFHLEVBQUU7UUFDekIsS0FBSyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUN4QixDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBzcGF3biB9IGZyb20gJ2NoaWxkX3Byb2Nlc3MnO1xuaW1wb3J0IHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgZnMgZnJvbSAnZnMtZXh0cmEnO1xuaW1wb3J0IHsgbG9nZ2VyIH0gZnJvbSAnLi4vLi4vdXRpbHMvbG9nZ2VyJztcbmltcG9ydCB7IGdldFJlbmRlcmVyRGlyIH0gZnJvbSAnLi4vLi4vdXRpbHMvcGF0aHMnO1xuXG5pbnRlcmZhY2UgU3RhcnRPcHRpb25zIHtcbiAgcG9ydDogc3RyaW5nO1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gc3RhcnQob3B0aW9uczogU3RhcnRPcHRpb25zKTogUHJvbWlzZTx2b2lkPiB7XG4gIGNvbnN0IHByb2plY3RSb290ID0gcHJvY2Vzcy5jd2QoKTtcbiAgXG4gIGxvZ2dlci5pbmZvKCdTdGFydGluZyBwcm9kdWN0aW9uIHNlcnZlci4uLicpO1xuICBcbiAgLy8gQ2hlY2sgZm9yIGJ1aWxkIG91dHB1dFxuICBjb25zdCBidWlsZERpciA9IHBhdGguam9pbihwcm9qZWN0Um9vdCwgJ2Rpc3QnLCAnLm5leHQnKTtcbiAgaWYgKCFmcy5leGlzdHNTeW5jKGJ1aWxkRGlyKSkge1xuICAgIGxvZ2dlci5lcnJvcignTm8gYnVpbGQgZm91bmQuIFJ1biBcImRldmRvYyBidWlsZFwiIGZpcnN0LicpO1xuICAgIHByb2Nlc3MuZXhpdCgxKTtcbiAgfVxuICBcbiAgLy8gR2V0IHRoZSByZW5kZXJlciBkaXJlY3RvcnlcbiAgY29uc3QgcmVuZGVyZXJEaXIgPSBnZXRSZW5kZXJlckRpcigpO1xuICBcbiAgaWYgKCFyZW5kZXJlckRpciB8fCAhZnMuZXhpc3RzU3luYyhyZW5kZXJlckRpcikpIHtcbiAgICBsb2dnZXIuZXJyb3IoJ1JlbmRlcmVyIG5vdCBmb3VuZC4gUGFja2FnZSBtYXkgYmUgY29ycnVwdGVkLicpO1xuICAgIHByb2Nlc3MuZXhpdCgxKTtcbiAgfVxuICBcbiAgLy8gU2V0IGVudmlyb25tZW50IHZhcmlhYmxlc1xuICBjb25zdCBlbnYgPSB7XG4gICAgLi4ucHJvY2Vzcy5lbnYsXG4gICAgU1RBUlRFUl9QQVRIOiBwcm9qZWN0Um9vdCxcbiAgICBQT1JUOiBvcHRpb25zLnBvcnQsXG4gIH07XG4gIFxuICBsb2dnZXIuaW5mbyhgU3RhcnRpbmcgc2VydmVyIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6JHtvcHRpb25zLnBvcnR9YCk7XG4gIFxuICAvLyBTdGFydCBOZXh0LmpzIHByb2R1Y3Rpb24gc2VydmVyXG4gIGNvbnN0IGNoaWxkID0gc3Bhd24oXG4gICAgJ25weCcsXG4gICAgWyduZXh0JywgJ3N0YXJ0JywgJy1wJywgb3B0aW9ucy5wb3J0XSxcbiAgICB7XG4gICAgICBjd2Q6IHJlbmRlcmVyRGlyLFxuICAgICAgZW52LFxuICAgICAgc3RkaW86ICdpbmhlcml0JyxcbiAgICAgIHNoZWxsOiB0cnVlLFxuICAgIH1cbiAgKTtcbiAgXG4gIGNoaWxkLm9uKCdlcnJvcicsIChlcnJvcikgPT4ge1xuICAgIGxvZ2dlci5lcnJvcihgRmFpbGVkIHRvIHN0YXJ0IHNlcnZlcjogJHtlcnJvci5tZXNzYWdlfWApO1xuICAgIHByb2Nlc3MuZXhpdCgxKTtcbiAgfSk7XG4gIFxuICBjaGlsZC5vbignZXhpdCcsIChjb2RlKSA9PiB7XG4gICAgcHJvY2Vzcy5leGl0KGNvZGUgfHwgMCk7XG4gIH0pO1xuICBcbiAgLy8gSGFuZGxlIHRlcm1pbmF0aW9uXG4gIHByb2Nlc3Mub24oJ1NJR0lOVCcsICgpID0+IHtcbiAgICBjaGlsZC5raWxsKCdTSUdJTlQnKTtcbiAgfSk7XG4gIFxuICBwcm9jZXNzLm9uKCdTSUdURVJNJywgKCkgPT4ge1xuICAgIGNoaWxkLmtpbGwoJ1NJR1RFUk0nKTtcbiAgfSk7XG59XG4iXX0=
@@ -0,0 +1,13 @@
1
+ interface UploadOptions {
2
+ url?: string;
3
+ apiKey?: string;
4
+ }
5
+ /**
6
+ * Upload assets to DevDoc storage
7
+ */
8
+ export declare function upload(files: string[], options: UploadOptions): Promise<void>;
9
+ /**
10
+ * List uploaded assets
11
+ */
12
+ export declare function listAssets(options: UploadOptions): Promise<void>;
13
+ export {};
@@ -0,0 +1,238 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.upload = upload;
7
+ exports.listAssets = listAssets;
8
+ const path_1 = __importDefault(require("path"));
9
+ const fs_extra_1 = __importDefault(require("fs-extra"));
10
+ const logger_1 = require("../../utils/logger");
11
+ const constants_1 = require("../../constants");
12
+ // Max file size: 25MB
13
+ const MAX_FILE_SIZE = 25 * 1024 * 1024;
14
+ /**
15
+ * Format file size for display
16
+ */
17
+ function formatSize(bytes) {
18
+ if (bytes < 1024)
19
+ return `${bytes} B`;
20
+ if (bytes < 1024 * 1024)
21
+ return `${(bytes / 1024).toFixed(1)} KB`;
22
+ return `${(bytes / (1024 * 1024)).toFixed(2)} MB`;
23
+ }
24
+ /**
25
+ * Create a progress bar string
26
+ */
27
+ function createProgressBar(progress, width = 30) {
28
+ const filled = Math.round(width * progress);
29
+ const empty = width - filled;
30
+ const bar = '█'.repeat(filled) + '░'.repeat(empty);
31
+ const percentage = Math.round(progress * 100);
32
+ return `[${bar}] ${percentage}%`;
33
+ }
34
+ /**
35
+ * Upload a file with progress tracking using XMLHttpRequest-like approach
36
+ */
37
+ async function uploadFileWithProgress(file, apiUrl, slug, apiKey, onProgress) {
38
+ const filePath = path_1.default.resolve(file);
39
+ if (!fs_extra_1.default.existsSync(filePath)) {
40
+ throw new Error(`File not found: ${filePath}`);
41
+ }
42
+ const stats = fs_extra_1.default.statSync(filePath);
43
+ const fileSize = stats.size;
44
+ const fileName = path_1.default.basename(filePath);
45
+ // Check file size
46
+ if (fileSize > MAX_FILE_SIZE) {
47
+ throw new Error(`File size (${formatSize(fileSize)}) exceeds maximum allowed size of 25MB`);
48
+ }
49
+ // Read file
50
+ const fileBuffer = fs_extra_1.default.readFileSync(filePath);
51
+ // Create form data manually for Node.js
52
+ const boundary = `----DevDocUpload${Date.now()}`;
53
+ const CRLF = '\r\n';
54
+ // Build multipart form data
55
+ const header = [
56
+ `--${boundary}`,
57
+ `Content-Disposition: form-data; name="file"; filename="${fileName}"`,
58
+ `Content-Type: application/octet-stream`,
59
+ '',
60
+ ''
61
+ ].join(CRLF);
62
+ const slugPart = [
63
+ '',
64
+ `--${boundary}`,
65
+ `Content-Disposition: form-data; name="slug"`,
66
+ '',
67
+ slug,
68
+ ''
69
+ ].join(CRLF);
70
+ const footer = `${CRLF}--${boundary}--${CRLF}`;
71
+ const headerBuffer = Buffer.from(header);
72
+ const slugBuffer = Buffer.from(slugPart);
73
+ const footerBuffer = Buffer.from(footer);
74
+ const totalSize = headerBuffer.length + fileBuffer.length + slugBuffer.length + footerBuffer.length;
75
+ // Combine all parts
76
+ const body = Buffer.concat([headerBuffer, fileBuffer, slugBuffer, footerBuffer]);
77
+ // Simulate progress for better UX (since fetch doesn't support upload progress natively)
78
+ let uploaded = 0;
79
+ const chunkSize = Math.ceil(totalSize / 20); // Simulate 20 progress updates
80
+ const progressInterval = setInterval(() => {
81
+ if (uploaded < totalSize * 0.9) { // Only go to 90% until response
82
+ uploaded = Math.min(uploaded + chunkSize, totalSize * 0.9);
83
+ onProgress(uploaded / totalSize, uploaded, totalSize);
84
+ }
85
+ }, 100);
86
+ try {
87
+ const response = await fetch(`${apiUrl}/api/assets/upload`, {
88
+ method: 'POST',
89
+ headers: {
90
+ 'Content-Type': `multipart/form-data; boundary=${boundary}`,
91
+ 'Authorization': `Bearer ${apiKey}`,
92
+ },
93
+ body,
94
+ });
95
+ clearInterval(progressInterval);
96
+ // Complete progress
97
+ onProgress(1, totalSize, totalSize);
98
+ if (!response.ok) {
99
+ const error = await response.json().catch(() => ({ error: 'Upload failed' }));
100
+ throw new Error(error.error || `HTTP ${response.status}`);
101
+ }
102
+ return await response.json();
103
+ }
104
+ catch (error) {
105
+ clearInterval(progressInterval);
106
+ throw error;
107
+ }
108
+ }
109
+ /**
110
+ * Upload assets to DevDoc storage
111
+ */
112
+ async function upload(files, options) {
113
+ const projectRoot = process.cwd();
114
+ const apiUrl = options.url || process.env.DEVDOC_API_URL || constants_1.DEFAULT_API_URL;
115
+ if (files.length === 0) {
116
+ logger_1.logger.error('No files specified');
117
+ logger_1.logger.info('Usage: devdoc upload <file1> [file2] ...');
118
+ process.exit(1);
119
+ }
120
+ // Load .devdoc.json to get project info
121
+ const devdocConfigPath = path_1.default.join(projectRoot, '.devdoc.json');
122
+ if (!fs_extra_1.default.existsSync(devdocConfigPath)) {
123
+ logger_1.logger.error('.devdoc.json not found');
124
+ logger_1.logger.info('Run "devdoc init" or "devdoc deploy" first to initialize your project');
125
+ process.exit(1);
126
+ }
127
+ let devdocConfig;
128
+ try {
129
+ devdocConfig = fs_extra_1.default.readJsonSync(devdocConfigPath);
130
+ }
131
+ catch {
132
+ logger_1.logger.error('Failed to read .devdoc.json');
133
+ process.exit(1);
134
+ }
135
+ const slug = devdocConfig.slug;
136
+ const apiKey = options.apiKey || process.env.DEVDOC_API_KEY || devdocConfig.apiKey;
137
+ if (!slug) {
138
+ logger_1.logger.error('Project slug not found in .devdoc.json');
139
+ process.exit(1);
140
+ }
141
+ if (!apiKey) {
142
+ logger_1.logger.error('API key not found');
143
+ logger_1.logger.info('Provide via --api-key, DEVDOC_API_KEY env var, or in .devdoc.json');
144
+ process.exit(1);
145
+ }
146
+ logger_1.logger.info(`Uploading ${files.length} file(s) to ${slug}...\n`);
147
+ const results = [];
148
+ for (const file of files) {
149
+ const fileName = path_1.default.basename(file);
150
+ process.stdout.write(` ${fileName}: `);
151
+ try {
152
+ const result = await uploadFileWithProgress(file, apiUrl, slug, apiKey, (progress, loaded, total) => {
153
+ // Clear line and rewrite progress
154
+ process.stdout.clearLine(0);
155
+ process.stdout.cursorTo(0);
156
+ process.stdout.write(` ${fileName}: ${createProgressBar(progress)} ${formatSize(loaded)}/${formatSize(total)}`);
157
+ });
158
+ // Clear line and show success
159
+ process.stdout.clearLine(0);
160
+ process.stdout.cursorTo(0);
161
+ console.log(` ${logger_1.logger.green('✓')} ${fileName} (${formatSize(result.size)})`);
162
+ console.log(` ${logger_1.logger.cyan(result.url)}`);
163
+ results.push({ file: fileName, success: true, url: result.url });
164
+ }
165
+ catch (error) {
166
+ const message = error instanceof Error ? error.message : String(error);
167
+ // Clear line and show error
168
+ process.stdout.clearLine(0);
169
+ process.stdout.cursorTo(0);
170
+ console.log(` ${logger_1.logger.red('✗')} ${fileName}: ${message}`);
171
+ results.push({ file: fileName, success: false, error: message });
172
+ }
173
+ }
174
+ // Summary
175
+ console.log('');
176
+ const successful = results.filter(r => r.success).length;
177
+ const failed = results.filter(r => !r.success).length;
178
+ if (failed === 0) {
179
+ logger_1.logger.success(`All ${successful} file(s) uploaded successfully!`);
180
+ }
181
+ else if (successful === 0) {
182
+ logger_1.logger.error(`All ${failed} file(s) failed to upload`);
183
+ process.exit(1);
184
+ }
185
+ else {
186
+ logger_1.logger.warn(`${successful} file(s) uploaded, ${failed} failed`);
187
+ }
188
+ }
189
+ /**
190
+ * List uploaded assets
191
+ */
192
+ async function listAssets(options) {
193
+ const projectRoot = process.cwd();
194
+ const apiUrl = options.url || process.env.DEVDOC_API_URL || constants_1.DEFAULT_API_URL;
195
+ // Load .devdoc.json
196
+ const devdocConfigPath = path_1.default.join(projectRoot, '.devdoc.json');
197
+ if (!fs_extra_1.default.existsSync(devdocConfigPath)) {
198
+ logger_1.logger.error('.devdoc.json not found');
199
+ process.exit(1);
200
+ }
201
+ const devdocConfig = fs_extra_1.default.readJsonSync(devdocConfigPath);
202
+ const slug = devdocConfig.slug;
203
+ const apiKey = options.apiKey || process.env.DEVDOC_API_KEY || devdocConfig.apiKey;
204
+ if (!slug || !apiKey) {
205
+ logger_1.logger.error('Project slug or API key not found');
206
+ process.exit(1);
207
+ }
208
+ logger_1.logger.info(`Fetching assets for ${slug}...`);
209
+ try {
210
+ const response = await fetch(`${apiUrl}/api/assets?slug=${slug}`, {
211
+ headers: {
212
+ 'Authorization': `Bearer ${apiKey}`,
213
+ },
214
+ });
215
+ if (!response.ok) {
216
+ throw new Error(`HTTP ${response.status}`);
217
+ }
218
+ const data = await response.json();
219
+ if (data.assets.length === 0) {
220
+ logger_1.logger.info('No assets found');
221
+ return;
222
+ }
223
+ console.log('');
224
+ console.log('Assets:');
225
+ for (const asset of data.assets) {
226
+ console.log(` ${asset.fileName} (${formatSize(asset.size)})`);
227
+ console.log(` ${logger_1.logger.cyan(asset.url)}`);
228
+ }
229
+ console.log('');
230
+ logger_1.logger.info(`Total: ${data.assets.length} asset(s)`);
231
+ }
232
+ catch (error) {
233
+ const message = error instanceof Error ? error.message : String(error);
234
+ logger_1.logger.error(`Failed to list assets: ${message}`);
235
+ process.exit(1);
236
+ }
237
+ }
238
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"upload.js","sourceRoot":"","sources":["../../../src/cli/commands/upload.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\n// Max file size: 25MB\nconst MAX_FILE_SIZE = 25 * 1024 * 1024\n\ninterface DevDocConfig {\n  slug: string\n  apiKey?: string\n  url?: string\n}\n\ninterface UploadOptions {\n  url?: string\n  apiKey?: string\n}\n\ninterface UploadResponse {\n  success: boolean\n  url: string\n  path: string\n  fileName: string\n  size: number\n  contentType: string\n  error?: string\n}\n\n/**\n * Format file size for display\n */\nfunction formatSize(bytes: number): string {\n  if (bytes < 1024) return `${bytes} B`\n  if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`\n  return `${(bytes / (1024 * 1024)).toFixed(2)} MB`\n}\n\n/**\n * Create a progress bar string\n */\nfunction createProgressBar(progress: number, width: number = 30): string {\n  const filled = Math.round(width * progress)\n  const empty = width - filled\n  const bar = '█'.repeat(filled) + '░'.repeat(empty)\n  const percentage = Math.round(progress * 100)\n  return `[${bar}] ${percentage}%`\n}\n\n/**\n * Upload a file with progress tracking using XMLHttpRequest-like approach\n */\nasync function uploadFileWithProgress(\n  file: string,\n  apiUrl: string,\n  slug: string,\n  apiKey: string,\n  onProgress: (progress: number, loaded: number, total: number) => void\n): Promise<UploadResponse> {\n  const filePath = path.resolve(file)\n  \n  if (!fs.existsSync(filePath)) {\n    throw new Error(`File not found: ${filePath}`)\n  }\n  \n  const stats = fs.statSync(filePath)\n  const fileSize = stats.size\n  const fileName = path.basename(filePath)\n  \n  // Check file size\n  if (fileSize > MAX_FILE_SIZE) {\n    throw new Error(`File size (${formatSize(fileSize)}) exceeds maximum allowed size of 25MB`)\n  }\n  \n  // Read file\n  const fileBuffer = fs.readFileSync(filePath)\n  \n  // Create form data manually for Node.js\n  const boundary = `----DevDocUpload${Date.now()}`\n  const CRLF = '\\r\\n'\n  \n  // Build multipart form data\n  const header = [\n    `--${boundary}`,\n    `Content-Disposition: form-data; name=\"file\"; filename=\"${fileName}\"`,\n    `Content-Type: application/octet-stream`,\n    '',\n    ''\n  ].join(CRLF)\n  \n  const slugPart = [\n    '',\n    `--${boundary}`,\n    `Content-Disposition: form-data; name=\"slug\"`,\n    '',\n    slug,\n    ''\n  ].join(CRLF)\n  \n  const footer = `${CRLF}--${boundary}--${CRLF}`\n  \n  const headerBuffer = Buffer.from(header)\n  const slugBuffer = Buffer.from(slugPart)\n  const footerBuffer = Buffer.from(footer)\n  \n  const totalSize = headerBuffer.length + fileBuffer.length + slugBuffer.length + footerBuffer.length\n  \n  // Combine all parts\n  const body = Buffer.concat([headerBuffer, fileBuffer, slugBuffer, footerBuffer])\n  \n  // Simulate progress for better UX (since fetch doesn't support upload progress natively)\n  let uploaded = 0\n  const chunkSize = Math.ceil(totalSize / 20) // Simulate 20 progress updates\n  \n  const progressInterval = setInterval(() => {\n    if (uploaded < totalSize * 0.9) { // Only go to 90% until response\n      uploaded = Math.min(uploaded + chunkSize, totalSize * 0.9)\n      onProgress(uploaded / totalSize, uploaded, totalSize)\n    }\n  }, 100)\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/assets/upload`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': `multipart/form-data; boundary=${boundary}`,\n        'Authorization': `Bearer ${apiKey}`,\n      },\n      body,\n    })\n    \n    clearInterval(progressInterval)\n    \n    // Complete progress\n    onProgress(1, totalSize, totalSize)\n    \n    if (!response.ok) {\n      const error = await response.json().catch(() => ({ error: 'Upload failed' })) as { error?: string }\n      throw new Error(error.error || `HTTP ${response.status}`)\n    }\n    \n    return await response.json() as UploadResponse\n  } catch (error) {\n    clearInterval(progressInterval)\n    throw error\n  }\n}\n\n/**\n * Upload assets to DevDoc storage\n */\nexport async function upload(files: string[], options: UploadOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  if (files.length === 0) {\n    logger.error('No files specified')\n    logger.info('Usage: devdoc upload <file1> [file2] ...')\n    process.exit(1)\n  }\n  \n  // Load .devdoc.json to get project info\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  \n  if (!fs.existsSync(devdocConfigPath)) {\n    logger.error('.devdoc.json not found')\n    logger.info('Run \"devdoc init\" or \"devdoc deploy\" first to initialize your project')\n    process.exit(1)\n  }\n  \n  let devdocConfig: DevDocConfig\n  try {\n    devdocConfig = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n  } catch {\n    logger.error('Failed to read .devdoc.json')\n    process.exit(1)\n  }\n  \n  const slug = devdocConfig.slug\n  const apiKey = options.apiKey || process.env.DEVDOC_API_KEY || devdocConfig.apiKey\n  \n  if (!slug) {\n    logger.error('Project slug not found in .devdoc.json')\n    process.exit(1)\n  }\n  \n  if (!apiKey) {\n    logger.error('API key not found')\n    logger.info('Provide via --api-key, DEVDOC_API_KEY env var, or in .devdoc.json')\n    process.exit(1)\n  }\n  \n  logger.info(`Uploading ${files.length} file(s) to ${slug}...\\n`)\n  \n  const results: { file: string; success: boolean; url?: string; error?: string }[] = []\n  \n  for (const file of files) {\n    const fileName = path.basename(file)\n    \n    process.stdout.write(`  ${fileName}: `)\n    \n    try {\n      const result = await uploadFileWithProgress(\n        file,\n        apiUrl,\n        slug,\n        apiKey,\n        (progress, loaded, total) => {\n          // Clear line and rewrite progress\n          process.stdout.clearLine(0)\n          process.stdout.cursorTo(0)\n          process.stdout.write(`  ${fileName}: ${createProgressBar(progress)} ${formatSize(loaded)}/${formatSize(total)}`)\n        }\n      )\n      \n      // Clear line and show success\n      process.stdout.clearLine(0)\n      process.stdout.cursorTo(0)\n      console.log(`  ${logger.green('✓')} ${fileName} (${formatSize(result.size)})`)\n      console.log(`    ${logger.cyan(result.url)}`)\n      \n      results.push({ file: fileName, success: true, url: result.url })\n    } catch (error) {\n      const message = error instanceof Error ? error.message : String(error)\n      \n      // Clear line and show error\n      process.stdout.clearLine(0)\n      process.stdout.cursorTo(0)\n      console.log(`  ${logger.red('✗')} ${fileName}: ${message}`)\n      \n      results.push({ file: fileName, success: false, error: message })\n    }\n  }\n  \n  // Summary\n  console.log('')\n  const successful = results.filter(r => r.success).length\n  const failed = results.filter(r => !r.success).length\n  \n  if (failed === 0) {\n    logger.success(`All ${successful} file(s) uploaded successfully!`)\n  } else if (successful === 0) {\n    logger.error(`All ${failed} file(s) failed to upload`)\n    process.exit(1)\n  } else {\n    logger.warn(`${successful} file(s) uploaded, ${failed} failed`)\n  }\n}\n\n/**\n * List uploaded assets\n */\nexport async function listAssets(options: UploadOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  \n  // Load .devdoc.json\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  \n  if (!fs.existsSync(devdocConfigPath)) {\n    logger.error('.devdoc.json not found')\n    process.exit(1)\n  }\n  \n  const devdocConfig = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n  const slug = devdocConfig.slug\n  const apiKey = options.apiKey || process.env.DEVDOC_API_KEY || devdocConfig.apiKey\n  \n  if (!slug || !apiKey) {\n    logger.error('Project slug or API key not found')\n    process.exit(1)\n  }\n  \n  logger.info(`Fetching assets for ${slug}...`)\n  \n  try {\n    const response = await fetch(`${apiUrl}/api/assets?slug=${slug}`, {\n      headers: {\n        'Authorization': `Bearer ${apiKey}`,\n      },\n    })\n    \n    if (!response.ok) {\n      throw new Error(`HTTP ${response.status}`)\n    }\n    \n    const data = await response.json() as { assets: Array<{ fileName: string; size: number; url: string; uploadedAt: string }> }\n    \n    if (data.assets.length === 0) {\n      logger.info('No assets found')\n      return\n    }\n    \n    console.log('')\n    console.log('Assets:')\n    for (const asset of data.assets) {\n      console.log(`  ${asset.fileName} (${formatSize(asset.size)})`)\n      console.log(`    ${logger.cyan(asset.url)}`)\n    }\n    console.log('')\n    logger.info(`Total: ${data.assets.length} asset(s)`)\n  } catch (error) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to list assets: ${message}`)\n    process.exit(1)\n  }\n}\n"]}
@@ -0,0 +1,8 @@
1
+ interface WhoamiOptions {
2
+ url?: string;
3
+ }
4
+ /**
5
+ * Show current project information
6
+ */
7
+ export declare function whoami(options: WhoamiOptions): Promise<void>;
8
+ export {};
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.whoami = whoami;
7
+ const path_1 = __importDefault(require("path"));
8
+ const fs_extra_1 = __importDefault(require("fs-extra"));
9
+ const logger_1 = require("../../utils/logger");
10
+ const constants_1 = require("../../constants");
11
+ /**
12
+ * Show current project information
13
+ */
14
+ async function whoami(options) {
15
+ const projectRoot = process.cwd();
16
+ const apiUrl = options.url || process.env.DEVDOC_API_URL || constants_1.DEFAULT_API_URL;
17
+ const devdocConfigPath = path_1.default.join(projectRoot, '.devdoc.json');
18
+ const docsConfigPath = path_1.default.join(projectRoot, 'docs.json');
19
+ console.log('');
20
+ console.log('DevDoc Project Information');
21
+ console.log('========================');
22
+ console.log('');
23
+ // Check for docs.json
24
+ if (fs_extra_1.default.existsSync(docsConfigPath)) {
25
+ try {
26
+ const docsConfig = fs_extra_1.default.readJsonSync(docsConfigPath);
27
+ console.log(` Project Name: ${logger_1.logger.cyan(docsConfig.name || 'Untitled')}`);
28
+ }
29
+ catch {
30
+ console.log(` Project Name: ${logger_1.logger.yellow('Unable to read docs.json')}`);
31
+ }
32
+ }
33
+ else {
34
+ console.log(` ${logger_1.logger.yellow('No docs.json found in current directory')}`);
35
+ console.log('');
36
+ process.exit(0);
37
+ }
38
+ // Check for .devdoc.json
39
+ if (!fs_extra_1.default.existsSync(devdocConfigPath)) {
40
+ console.log(` Status: ${logger_1.logger.yellow('Not deployed')}`);
41
+ console.log('');
42
+ logger_1.logger.info('Run "devdoc deploy" to deploy your documentation.');
43
+ console.log('');
44
+ process.exit(0);
45
+ }
46
+ try {
47
+ const config = fs_extra_1.default.readJsonSync(devdocConfigPath);
48
+ console.log(` Project Slug: ${config.slug || logger_1.logger.yellow('Unknown')}`);
49
+ console.log(` Status: ${logger_1.logger.green('Deployed')}`);
50
+ if (config.url) {
51
+ console.log(` URL: ${logger_1.logger.cyan(config.url)}`);
52
+ }
53
+ if (config.apiKey) {
54
+ const maskedKey = config.apiKey.substring(0, 12) + '...' + config.apiKey.substring(config.apiKey.length - 4);
55
+ console.log(` API Key: ${maskedKey}`);
56
+ }
57
+ if (config.lastDeployed) {
58
+ const date = new Date(config.lastDeployed);
59
+ console.log(` Last Deploy: ${date.toLocaleString()}`);
60
+ }
61
+ console.log('');
62
+ console.log(` Platform: ${apiUrl}`);
63
+ console.log('');
64
+ // Optional: verify with server
65
+ if (config.slug && config.apiKey) {
66
+ try {
67
+ const response = await fetch(`${apiUrl}/api/deploy?slug=${config.slug}`, {
68
+ method: 'GET',
69
+ headers: {
70
+ 'Authorization': `Bearer ${config.apiKey}`,
71
+ },
72
+ });
73
+ if (response.ok) {
74
+ const data = await response.json();
75
+ if (data.exists) {
76
+ logger_1.logger.success('✓ Project verified on server');
77
+ }
78
+ }
79
+ }
80
+ catch {
81
+ // Silent fail - server verification is optional
82
+ }
83
+ }
84
+ }
85
+ catch (error) {
86
+ const message = error instanceof Error ? error.message : String(error);
87
+ logger_1.logger.error(`Failed to read project config: ${message}`);
88
+ process.exit(1);
89
+ }
90
+ }
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"whoami.js","sourceRoot":"","sources":["../../../src/cli/commands/whoami.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path'\nimport fs from 'fs-extra'\nimport { logger } from '../../utils/logger'\nimport { DEFAULT_API_URL } from '../../constants'\n\ninterface WhoamiOptions {\n  url?: string\n}\n\ninterface DevDocConfig {\n  slug?: string\n  apiKey?: string\n  url?: string\n  lastDeployed?: string\n}\n\n/**\n * Show current project information\n */\nexport async function whoami(options: WhoamiOptions): Promise<void> {\n  const projectRoot = process.cwd()\n  const apiUrl = options.url || process.env.DEVDOC_API_URL || DEFAULT_API_URL\n  const devdocConfigPath = path.join(projectRoot, '.devdoc.json')\n  const docsConfigPath = path.join(projectRoot, 'docs.json')\n  \n  console.log('')\n  console.log('DevDoc Project Information')\n  console.log('========================')\n  console.log('')\n  \n  // Check for docs.json\n  if (fs.existsSync(docsConfigPath)) {\n    try {\n      const docsConfig = fs.readJsonSync(docsConfigPath)\n      console.log(`  Project Name: ${logger.cyan(docsConfig.name || 'Untitled')}`)\n    } catch {\n      console.log(`  Project Name: ${logger.yellow('Unable to read docs.json')}`)\n    }\n  } else {\n    console.log(`  ${logger.yellow('No docs.json found in current directory')}`)\n    console.log('')\n    process.exit(0)\n  }\n  \n  // Check for .devdoc.json\n  if (!fs.existsSync(devdocConfigPath)) {\n    console.log(`  Status:       ${logger.yellow('Not deployed')}`)\n    console.log('')\n    logger.info('Run \"devdoc deploy\" to deploy your documentation.')\n    console.log('')\n    process.exit(0)\n  }\n  \n  try {\n    const config = fs.readJsonSync(devdocConfigPath) as DevDocConfig\n    \n    console.log(`  Project Slug: ${config.slug || logger.yellow('Unknown')}`)\n    console.log(`  Status:       ${logger.green('Deployed')}`)\n    \n    if (config.url) {\n      console.log(`  URL:          ${logger.cyan(config.url)}`)\n    }\n    \n    if (config.apiKey) {\n      const maskedKey = config.apiKey.substring(0, 12) + '...' + config.apiKey.substring(config.apiKey.length - 4)\n      console.log(`  API Key:      ${maskedKey}`)\n    }\n    \n    if (config.lastDeployed) {\n      const date = new Date(config.lastDeployed)\n      console.log(`  Last Deploy:  ${date.toLocaleString()}`)\n    }\n    \n    console.log('')\n    console.log(`  Platform:     ${apiUrl}`)\n    console.log('')\n    \n    // Optional: verify with server\n    if (config.slug && config.apiKey) {\n      try {\n        const response = await fetch(`${apiUrl}/api/deploy?slug=${config.slug}`, {\n          method: 'GET',\n          headers: {\n            'Authorization': `Bearer ${config.apiKey}`,\n          },\n        })\n        \n        if (response.ok) {\n          const data = await response.json() as { exists?: boolean }\n          if (data.exists) {\n            logger.success('✓ Project verified on server')\n          }\n        }\n      } catch {\n        // Silent fail - server verification is optional\n      }\n    }\n    \n  } catch (error: unknown) {\n    const message = error instanceof Error ? error.message : String(error)\n    logger.error(`Failed to read project config: ${message}`)\n    process.exit(1)\n  }\n}\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,106 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const commander_1 = require("commander");
4
+ const create_1 = require("./commands/create");
5
+ const init_1 = require("./commands/init");
6
+ const dev_1 = require("./commands/dev");
7
+ const build_1 = require("./commands/build");
8
+ const start_1 = require("./commands/start");
9
+ const check_1 = require("./commands/check");
10
+ const deploy_1 = require("./commands/deploy");
11
+ const keys_1 = require("./commands/keys");
12
+ const whoami_1 = require("./commands/whoami");
13
+ const upload_1 = require("./commands/upload");
14
+ const packageJson = require('../../package.json');
15
+ const program = new commander_1.Command();
16
+ program
17
+ .name('devdoc')
18
+ .description('Documentation framework for developers')
19
+ .version(packageJson.version);
20
+ program
21
+ .command('create [project-directory]')
22
+ .description('Create a new DevDoc documentation site')
23
+ .option('-t, --template <type>', 'Template to use (basic, openapi, graphql)')
24
+ .option('-s, --subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')
25
+ .option('--no-git', 'Skip git initialization')
26
+ .option('--no-install', 'Skip installing dependencies')
27
+ .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')
28
+ .action(create_1.create);
29
+ program
30
+ .command('init')
31
+ .description('Initialize/register project with Brainfish')
32
+ .option('-s, --slug <slug>', 'Project slug')
33
+ .option('--subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')
34
+ .option('-f, --force', 'Overwrite existing .devdoc.json')
35
+ .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')
36
+ .action(init_1.init);
37
+ program
38
+ .command('dev')
39
+ .description('Start development server with hot reload')
40
+ .option('-p, --port <port>', 'Port to run the server on', '3333')
41
+ .option('-H, --host <host>', 'Host to bind the server to', 'localhost')
42
+ .action(dev_1.dev);
43
+ program
44
+ .command('build')
45
+ .description('Build documentation for production')
46
+ .option('-o, --output <dir>', 'Output directory', 'dist')
47
+ .action(build_1.build);
48
+ program
49
+ .command('start')
50
+ .description('Start production server')
51
+ .option('-p, --port <port>', 'Port to run the server on', '3000')
52
+ .action(start_1.start);
53
+ program
54
+ .command('check')
55
+ .description('Validate docs.json and MDX files')
56
+ .action(check_1.check);
57
+ program
58
+ .command('deploy')
59
+ .description('Deploy documentation to DevDoc platform')
60
+ .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')
61
+ .option('-k, --api-key <key>', 'API key for authentication')
62
+ .action(deploy_1.deploy);
63
+ // Keys management
64
+ const keysCmd = program
65
+ .command('keys')
66
+ .description('Manage API keys for your project');
67
+ keysCmd
68
+ .command('list')
69
+ .description('Show current API key info')
70
+ .option('-u, --url <url>', 'API URL')
71
+ .action(keys_1.listKeys);
72
+ keysCmd
73
+ .command('regenerate')
74
+ .description('Generate a new API key (invalidates the old one)')
75
+ .option('-u, --url <url>', 'API URL')
76
+ .action(keys_1.regenerateKey);
77
+ program
78
+ .command('whoami')
79
+ .description('Show current project information')
80
+ .option('-u, --url <url>', 'API URL')
81
+ .action(whoami_1.whoami);
82
+ // Assets management
83
+ const assetsCmd = program
84
+ .command('assets')
85
+ .description('Manage project assets (images, files)');
86
+ assetsCmd
87
+ .command('upload <files...>')
88
+ .description('Upload assets to storage (max 25MB per file)')
89
+ .option('-u, --url <url>', 'API URL')
90
+ .option('-k, --api-key <key>', 'API key for authentication')
91
+ .action(upload_1.upload);
92
+ assetsCmd
93
+ .command('list')
94
+ .description('List uploaded assets')
95
+ .option('-u, --url <url>', 'API URL')
96
+ .option('-k, --api-key <key>', 'API key for authentication')
97
+ .action(upload_1.listAssets);
98
+ // Shortcut for upload
99
+ program
100
+ .command('upload <files...>')
101
+ .description('Upload assets to storage (shortcut for "devdoc assets upload")')
102
+ .option('-u, --url <url>', 'API URL')
103
+ .option('-k, --api-key <key>', 'API key for authentication')
104
+ .action(upload_1.upload);
105
+ program.parse(process.argv);
106
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command } from 'commander';\nimport { create } from './commands/create';\nimport { init } from './commands/init';\nimport { dev } from './commands/dev';\nimport { build } from './commands/build';\nimport { start } from './commands/start';\nimport { check } from './commands/check';\nimport { deploy } from './commands/deploy';\nimport { listKeys, regenerateKey } from './commands/keys';\nimport { whoami } from './commands/whoami';\nimport { upload, listAssets } from './commands/upload';\n\nconst packageJson = require('../../package.json');\n\nconst program = new Command();\n\nprogram\n  .name('devdoc')\n  .description('Documentation framework for developers')\n  .version(packageJson.version);\n\nprogram\n  .command('create [project-directory]')\n  .description('Create a new DevDoc documentation site')\n  .option('-t, --template <type>', 'Template to use (basic, openapi, graphql)')\n  .option('-s, --subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')\n  .option('--no-git', 'Skip git initialization')\n  .option('--no-install', 'Skip installing dependencies')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .action(create);\n\nprogram\n  .command('init')\n  .description('Initialize/register project with Brainfish')\n  .option('-s, --slug <slug>', 'Project slug')\n  .option('--subdomain <subdomain>', 'Subdomain for your docs (e.g., my-docs.devdoc.sh)')\n  .option('-f, --force', 'Overwrite existing .devdoc.json')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .action(init);\n\nprogram\n  .command('dev')\n  .description('Start development server with hot reload')\n  .option('-p, --port <port>', 'Port to run the server on', '3333')\n  .option('-H, --host <host>', 'Host to bind the server to', 'localhost')\n  .action(dev);\n\nprogram\n  .command('build')\n  .description('Build documentation for production')\n  .option('-o, --output <dir>', 'Output directory', 'dist')\n  .action(build);\n\nprogram\n  .command('start')\n  .description('Start production server')\n  .option('-p, --port <port>', 'Port to run the server on', '3000')\n  .action(start);\n\nprogram\n  .command('check')\n  .description('Validate docs.json and MDX files')\n  .action(check);\n\nprogram\n  .command('deploy')\n  .description('Deploy documentation to DevDoc platform')\n  .option('-u, --url <url>', 'API URL (default: https://devdoc.sh)')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(deploy);\n\n// Keys management\nconst keysCmd = program\n  .command('keys')\n  .description('Manage API keys for your project');\n\nkeysCmd\n  .command('list')\n  .description('Show current API key info')\n  .option('-u, --url <url>', 'API URL')\n  .action(listKeys);\n\nkeysCmd\n  .command('regenerate')\n  .description('Generate a new API key (invalidates the old one)')\n  .option('-u, --url <url>', 'API URL')\n  .action(regenerateKey);\n\nprogram\n  .command('whoami')\n  .description('Show current project information')\n  .option('-u, --url <url>', 'API URL')\n  .action(whoami);\n\n// Assets management\nconst assetsCmd = program\n  .command('assets')\n  .description('Manage project assets (images, files)');\n\nassetsCmd\n  .command('upload <files...>')\n  .description('Upload assets to storage (max 25MB per file)')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(upload);\n\nassetsCmd\n  .command('list')\n  .description('List uploaded assets')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(listAssets);\n\n// Shortcut for upload\nprogram\n  .command('upload <files...>')\n  .description('Upload assets to storage (shortcut for \"devdoc assets upload\")')\n  .option('-u, --url <url>', 'API URL')\n  .option('-k, --api-key <key>', 'API key for authentication')\n  .action(upload);\n\nprogram.parse(process.argv);\n"]}