doclific 0.1.0

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 (231) hide show
  1. package/.gitattributes +2 -0
  2. package/.prettierignore +5 -0
  3. package/.prettierrc +9 -0
  4. package/.vscode/settings.json +13 -0
  5. package/dist/bin/doclific.d.ts +3 -0
  6. package/dist/bin/doclific.d.ts.map +1 -0
  7. package/dist/bin/doclific.js +11 -0
  8. package/dist/core/codebase.js +31 -0
  9. package/dist/core/docs.js +75 -0
  10. package/dist/core/git.js +47 -0
  11. package/dist/server/index.d.ts +2 -0
  12. package/dist/server/index.d.ts.map +1 -0
  13. package/dist/server/index.js +46 -0
  14. package/dist/server/router.d.ts +9 -0
  15. package/dist/server/router.d.ts.map +1 -0
  16. package/dist/server/router.js +55 -0
  17. package/frontend/README.md +73 -0
  18. package/frontend/components.json +24 -0
  19. package/frontend/eslint.config.js +23 -0
  20. package/frontend/index.html +25 -0
  21. package/frontend/package-lock.json +15754 -0
  22. package/frontend/package.json +122 -0
  23. package/frontend/public/logo.svg +1 -0
  24. package/frontend/src/App.tsx +21 -0
  25. package/frontend/src/components/app-sidebar.tsx +393 -0
  26. package/frontend/src/components/editor/editor-base-kit.tsx +43 -0
  27. package/frontend/src/components/editor/editor-kit.tsx +93 -0
  28. package/frontend/src/components/editor/plugins/align-base-kit.tsx +16 -0
  29. package/frontend/src/components/editor/plugins/align-kit.tsx +18 -0
  30. package/frontend/src/components/editor/plugins/autoformat-kit.tsx +236 -0
  31. package/frontend/src/components/editor/plugins/basic-blocks-base-kit.tsx +35 -0
  32. package/frontend/src/components/editor/plugins/basic-blocks-kit.tsx +88 -0
  33. package/frontend/src/components/editor/plugins/basic-marks-base-kit.tsx +27 -0
  34. package/frontend/src/components/editor/plugins/basic-marks-kit.tsx +41 -0
  35. package/frontend/src/components/editor/plugins/basic-nodes-kit.tsx +6 -0
  36. package/frontend/src/components/editor/plugins/block-menu-kit.tsx +14 -0
  37. package/frontend/src/components/editor/plugins/block-placeholder-kit.tsx +17 -0
  38. package/frontend/src/components/editor/plugins/block-selection-kit.tsx +32 -0
  39. package/frontend/src/components/editor/plugins/callout-base-kit.tsx +7 -0
  40. package/frontend/src/components/editor/plugins/callout-kit.tsx +7 -0
  41. package/frontend/src/components/editor/plugins/code-block-base-kit.tsx +23 -0
  42. package/frontend/src/components/editor/plugins/code-block-kit.tsx +26 -0
  43. package/frontend/src/components/editor/plugins/codebase-kit.tsx +23 -0
  44. package/frontend/src/components/editor/plugins/column-base-kit.tsx +11 -0
  45. package/frontend/src/components/editor/plugins/column-kit.tsx +10 -0
  46. package/frontend/src/components/editor/plugins/comment-base-kit.tsx +7 -0
  47. package/frontend/src/components/editor/plugins/comment-kit.tsx +97 -0
  48. package/frontend/src/components/editor/plugins/cursor-overlay-kit.tsx +13 -0
  49. package/frontend/src/components/editor/plugins/date-base-kit.tsx +5 -0
  50. package/frontend/src/components/editor/plugins/date-kit.tsx +7 -0
  51. package/frontend/src/components/editor/plugins/discussion-kit.tsx +148 -0
  52. package/frontend/src/components/editor/plugins/dnd-kit.tsx +28 -0
  53. package/frontend/src/components/editor/plugins/docx-kit.tsx +6 -0
  54. package/frontend/src/components/editor/plugins/emoji-kit.tsx +13 -0
  55. package/frontend/src/components/editor/plugins/excalidraw-kit.tsx +9 -0
  56. package/frontend/src/components/editor/plugins/exit-break-kit.tsx +12 -0
  57. package/frontend/src/components/editor/plugins/floating-toolbar-kit.tsx +19 -0
  58. package/frontend/src/components/editor/plugins/font-base-kit.tsx +20 -0
  59. package/frontend/src/components/editor/plugins/font-kit.tsx +29 -0
  60. package/frontend/src/components/editor/plugins/indent-base-kit.tsx +19 -0
  61. package/frontend/src/components/editor/plugins/indent-kit.tsx +22 -0
  62. package/frontend/src/components/editor/plugins/line-height-base-kit.tsx +14 -0
  63. package/frontend/src/components/editor/plugins/line-height-kit.tsx +16 -0
  64. package/frontend/src/components/editor/plugins/link-base-kit.tsx +5 -0
  65. package/frontend/src/components/editor/plugins/link-kit.tsx +15 -0
  66. package/frontend/src/components/editor/plugins/list-base-kit.tsx +23 -0
  67. package/frontend/src/components/editor/plugins/list-kit.tsx +26 -0
  68. package/frontend/src/components/editor/plugins/markdown-kit.tsx +46 -0
  69. package/frontend/src/components/editor/plugins/math-base-kit.tsx +11 -0
  70. package/frontend/src/components/editor/plugins/math-kit.tsx +13 -0
  71. package/frontend/src/components/editor/plugins/media-base-kit.tsx +31 -0
  72. package/frontend/src/components/editor/plugins/media-kit.tsx +43 -0
  73. package/frontend/src/components/editor/plugins/mention-base-kit.tsx +7 -0
  74. package/frontend/src/components/editor/plugins/mention-kit.tsx +15 -0
  75. package/frontend/src/components/editor/plugins/slash-kit.tsx +18 -0
  76. package/frontend/src/components/editor/plugins/suggestion-base-kit.tsx +7 -0
  77. package/frontend/src/components/editor/plugins/suggestion-kit.tsx +90 -0
  78. package/frontend/src/components/editor/plugins/table-base-kit.tsx +20 -0
  79. package/frontend/src/components/editor/plugins/table-kit.tsx +22 -0
  80. package/frontend/src/components/editor/plugins/toc-base-kit.tsx +5 -0
  81. package/frontend/src/components/editor/plugins/toc-kit.tsx +14 -0
  82. package/frontend/src/components/editor/plugins/toggle-base-kit.tsx +7 -0
  83. package/frontend/src/components/editor/plugins/toggle-kit.tsx +11 -0
  84. package/frontend/src/components/editor/transforms.ts +194 -0
  85. package/frontend/src/components/markdown-to-slate-demo.tsx +50 -0
  86. package/frontend/src/components/mode-toggle.tsx +15 -0
  87. package/frontend/src/components/theme-provider.tsx +73 -0
  88. package/frontend/src/components/ui/alert-dialog.tsx +155 -0
  89. package/frontend/src/components/ui/align-toolbar-button.tsx +84 -0
  90. package/frontend/src/components/ui/avatar.tsx +51 -0
  91. package/frontend/src/components/ui/block-context-menu.tsx +199 -0
  92. package/frontend/src/components/ui/block-discussion.tsx +365 -0
  93. package/frontend/src/components/ui/block-draggable.tsx +512 -0
  94. package/frontend/src/components/ui/block-list-static.tsx +80 -0
  95. package/frontend/src/components/ui/block-list.tsx +87 -0
  96. package/frontend/src/components/ui/block-selection.tsx +42 -0
  97. package/frontend/src/components/ui/block-suggestion.tsx +473 -0
  98. package/frontend/src/components/ui/blockquote-node-static.tsx +11 -0
  99. package/frontend/src/components/ui/blockquote-node.tsx +13 -0
  100. package/frontend/src/components/ui/button.tsx +62 -0
  101. package/frontend/src/components/ui/calendar.tsx +218 -0
  102. package/frontend/src/components/ui/callout-node-static.tsx +36 -0
  103. package/frontend/src/components/ui/callout-node.tsx +63 -0
  104. package/frontend/src/components/ui/caption.tsx +63 -0
  105. package/frontend/src/components/ui/checkbox.tsx +30 -0
  106. package/frontend/src/components/ui/code-block-node-static.tsx +35 -0
  107. package/frontend/src/components/ui/code-block-node.tsx +287 -0
  108. package/frontend/src/components/ui/code-node-static.tsx +15 -0
  109. package/frontend/src/components/ui/code-node.tsx +17 -0
  110. package/frontend/src/components/ui/codebase-snippet-node.tsx +237 -0
  111. package/frontend/src/components/ui/column-node-static.tsx +29 -0
  112. package/frontend/src/components/ui/column-node.tsx +317 -0
  113. package/frontend/src/components/ui/command.tsx +182 -0
  114. package/frontend/src/components/ui/comment-node-static.tsx +15 -0
  115. package/frontend/src/components/ui/comment-node.tsx +45 -0
  116. package/frontend/src/components/ui/comment-toolbar-button.tsx +24 -0
  117. package/frontend/src/components/ui/comment.tsx +618 -0
  118. package/frontend/src/components/ui/context-menu.tsx +250 -0
  119. package/frontend/src/components/ui/cursor-overlay.tsx +66 -0
  120. package/frontend/src/components/ui/date-node-static.tsx +45 -0
  121. package/frontend/src/components/ui/date-node.tsx +93 -0
  122. package/frontend/src/components/ui/dialog.tsx +143 -0
  123. package/frontend/src/components/ui/dropdown-menu.tsx +255 -0
  124. package/frontend/src/components/ui/dynamic-icon.tsx +12 -0
  125. package/frontend/src/components/ui/editor-static.tsx +53 -0
  126. package/frontend/src/components/ui/editor.tsx +130 -0
  127. package/frontend/src/components/ui/emoji-node.tsx +69 -0
  128. package/frontend/src/components/ui/emoji-toolbar-button.tsx +628 -0
  129. package/frontend/src/components/ui/equation-node-static.tsx +98 -0
  130. package/frontend/src/components/ui/equation-node.tsx +235 -0
  131. package/frontend/src/components/ui/equation-toolbar-button.tsx +25 -0
  132. package/frontend/src/components/ui/excalidraw-node.tsx +36 -0
  133. package/frontend/src/components/ui/export-toolbar-button.tsx +174 -0
  134. package/frontend/src/components/ui/file-selector.tsx +339 -0
  135. package/frontend/src/components/ui/floating-toolbar-buttons.tsx +73 -0
  136. package/frontend/src/components/ui/floating-toolbar.tsx +85 -0
  137. package/frontend/src/components/ui/font-color-toolbar-button.tsx +831 -0
  138. package/frontend/src/components/ui/font-size-toolbar-button.tsx +152 -0
  139. package/frontend/src/components/ui/heading-node-static.tsx +68 -0
  140. package/frontend/src/components/ui/heading-node.tsx +58 -0
  141. package/frontend/src/components/ui/highlight-node-static.tsx +11 -0
  142. package/frontend/src/components/ui/highlight-node.tsx +13 -0
  143. package/frontend/src/components/ui/history-toolbar-button.tsx +50 -0
  144. package/frontend/src/components/ui/hr-node-static.tsx +20 -0
  145. package/frontend/src/components/ui/hr-node.tsx +33 -0
  146. package/frontend/src/components/ui/import-toolbar-button.tsx +97 -0
  147. package/frontend/src/components/ui/indent-toolbar-button.tsx +30 -0
  148. package/frontend/src/components/ui/inline-combobox.tsx +414 -0
  149. package/frontend/src/components/ui/input.tsx +21 -0
  150. package/frontend/src/components/ui/insert-toolbar-button.tsx +254 -0
  151. package/frontend/src/components/ui/kbd-node-static.tsx +15 -0
  152. package/frontend/src/components/ui/kbd-node.tsx +17 -0
  153. package/frontend/src/components/ui/layout-header.tsx +35 -0
  154. package/frontend/src/components/ui/line-height-toolbar-button.tsx +68 -0
  155. package/frontend/src/components/ui/link-node-static.tsx +21 -0
  156. package/frontend/src/components/ui/link-node.tsx +39 -0
  157. package/frontend/src/components/ui/link-toolbar-button.tsx +22 -0
  158. package/frontend/src/components/ui/link-toolbar.tsx +206 -0
  159. package/frontend/src/components/ui/list-toolbar-button.tsx +204 -0
  160. package/frontend/src/components/ui/mark-toolbar-button.tsx +19 -0
  161. package/frontend/src/components/ui/media-audio-node-static.tsx +17 -0
  162. package/frontend/src/components/ui/media-audio-node.tsx +39 -0
  163. package/frontend/src/components/ui/media-embed-node.tsx +136 -0
  164. package/frontend/src/components/ui/media-file-node-static.tsx +29 -0
  165. package/frontend/src/components/ui/media-file-node.tsx +47 -0
  166. package/frontend/src/components/ui/media-image-node-static.tsx +39 -0
  167. package/frontend/src/components/ui/media-image-node.tsx +80 -0
  168. package/frontend/src/components/ui/media-placeholder-node.tsx +249 -0
  169. package/frontend/src/components/ui/media-preview-dialog.tsx +152 -0
  170. package/frontend/src/components/ui/media-toolbar-button.tsx +225 -0
  171. package/frontend/src/components/ui/media-toolbar.tsx +115 -0
  172. package/frontend/src/components/ui/media-upload-toast.tsx +66 -0
  173. package/frontend/src/components/ui/media-video-node-static.tsx +30 -0
  174. package/frontend/src/components/ui/media-video-node.tsx +121 -0
  175. package/frontend/src/components/ui/mention-node-static.tsx +36 -0
  176. package/frontend/src/components/ui/mention-node.tsx +194 -0
  177. package/frontend/src/components/ui/mode-toolbar-button.tsx +123 -0
  178. package/frontend/src/components/ui/more-toolbar-button.tsx +80 -0
  179. package/frontend/src/components/ui/paragraph-node-static.tsx +13 -0
  180. package/frontend/src/components/ui/paragraph-node.tsx +15 -0
  181. package/frontend/src/components/ui/popover.tsx +46 -0
  182. package/frontend/src/components/ui/resize-handle.tsx +87 -0
  183. package/frontend/src/components/ui/separator.tsx +28 -0
  184. package/frontend/src/components/ui/sheet.tsx +139 -0
  185. package/frontend/src/components/ui/sidebar.tsx +726 -0
  186. package/frontend/src/components/ui/skeleton.tsx +13 -0
  187. package/frontend/src/components/ui/slash-node.tsx +233 -0
  188. package/frontend/src/components/ui/sonner.tsx +38 -0
  189. package/frontend/src/components/ui/suggestion-node-static.tsx +35 -0
  190. package/frontend/src/components/ui/suggestion-node.tsx +162 -0
  191. package/frontend/src/components/ui/suggestion-toolbar-button.tsx +25 -0
  192. package/frontend/src/components/ui/table-icons.tsx +862 -0
  193. package/frontend/src/components/ui/table-node-static.tsx +98 -0
  194. package/frontend/src/components/ui/table-node.tsx +656 -0
  195. package/frontend/src/components/ui/table-toolbar-button.tsx +264 -0
  196. package/frontend/src/components/ui/toc-node-static.tsx +92 -0
  197. package/frontend/src/components/ui/toc-node.tsx +55 -0
  198. package/frontend/src/components/ui/toggle-node-static.tsx +18 -0
  199. package/frontend/src/components/ui/toggle-node.tsx +36 -0
  200. package/frontend/src/components/ui/toggle-toolbar-button.tsx +22 -0
  201. package/frontend/src/components/ui/toolbar.tsx +387 -0
  202. package/frontend/src/components/ui/tooltip.tsx +59 -0
  203. package/frontend/src/components/ui/turn-into-toolbar-button.tsx +188 -0
  204. package/frontend/src/hooks/use-debounce.ts +18 -0
  205. package/frontend/src/hooks/use-is-touch-device.ts +24 -0
  206. package/frontend/src/hooks/use-mobile.ts +19 -0
  207. package/frontend/src/hooks/use-mounted.ts +11 -0
  208. package/frontend/src/hooks/use-upload-file.ts +128 -0
  209. package/frontend/src/index.css +128 -0
  210. package/frontend/src/layout.tsx +42 -0
  211. package/frontend/src/lib/markdown-joiner-transform.ts +239 -0
  212. package/frontend/src/lib/orpc.ts +13 -0
  213. package/frontend/src/lib/uploadthing.ts +19 -0
  214. package/frontend/src/lib/utils.ts +6 -0
  215. package/frontend/src/main.tsx +13 -0
  216. package/frontend/src/pages/editor.tsx +44 -0
  217. package/frontend/src/types/docs.d.ts +6 -0
  218. package/frontend/src/types/global.d.ts +9 -0
  219. package/frontend/src/types/router.d.ts +4 -0
  220. package/frontend/tsconfig.app.json +33 -0
  221. package/frontend/tsconfig.json +10 -0
  222. package/frontend/tsconfig.node.json +26 -0
  223. package/frontend/vite.config.ts +14 -0
  224. package/package.json +30 -0
  225. package/src/bin/doclific.ts +17 -0
  226. package/src/core/codebase.ts +39 -0
  227. package/src/core/docs.ts +90 -0
  228. package/src/core/git.ts +48 -0
  229. package/src/server/index.ts +55 -0
  230. package/src/server/router.ts +65 -0
  231. package/tsconfig.json +15 -0
@@ -0,0 +1,10 @@
1
+ {
2
+ "files": [],
3
+ "references": [{ "path": "./tsconfig.app.json" }, { "path": "./tsconfig.node.json" }],
4
+ "compilerOptions": {
5
+ "baseUrl": ".",
6
+ "paths": {
7
+ "@/*": ["./src/*"]
8
+ }
9
+ }
10
+ }
@@ -0,0 +1,26 @@
1
+ {
2
+ "compilerOptions": {
3
+ "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
4
+ "target": "ES2023",
5
+ "lib": ["ES2023"],
6
+ "module": "ESNext",
7
+ "types": ["node"],
8
+ "skipLibCheck": true,
9
+
10
+ /* Bundler mode */
11
+ "moduleResolution": "bundler",
12
+ "allowImportingTsExtensions": true,
13
+ "verbatimModuleSyntax": true,
14
+ "moduleDetection": "force",
15
+ "noEmit": true,
16
+
17
+ /* Linting */
18
+ "strict": true,
19
+ "noUnusedLocals": true,
20
+ "noUnusedParameters": true,
21
+ "erasableSyntaxOnly": true,
22
+ "noFallthroughCasesInSwitch": true,
23
+ "noUncheckedSideEffectImports": true
24
+ },
25
+ "include": ["vite.config.ts"]
26
+ }
@@ -0,0 +1,14 @@
1
+ import { defineConfig } from 'vite';
2
+ import react from '@vitejs/plugin-react-swc';
3
+ import tailwindcss from '@tailwindcss/vite';
4
+ import path from 'path';
5
+
6
+ // https://vite.dev/config/
7
+ export default defineConfig({
8
+ plugins: [react(), tailwindcss()],
9
+ resolve: {
10
+ alias: {
11
+ '@': path.resolve(__dirname, './src'),
12
+ },
13
+ },
14
+ });
package/package.json ADDED
@@ -0,0 +1,30 @@
1
+ {
2
+ "name": "doclific",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "bin": {
6
+ "doclific": "./dist/bin/doclific.js"
7
+ },
8
+ "scripts": {
9
+ "build": "tsc",
10
+ "format": "prettier --write \"src/**/*.{ts,tsx,js,jsx,json}\"",
11
+ "format:check": "prettier --check \"src/**/*.{ts,tsx,js,jsx,json}\""
12
+ },
13
+ "dependencies": {
14
+ "@mdx-js/mdx": "^3.1.1",
15
+ "@orpc/client": "^1.12.2",
16
+ "@orpc/server": "^1.12.2",
17
+ "child_process": "^1.0.2",
18
+ "cors": "^2.8.5",
19
+ "express": "^5.2.1",
20
+ "util": "^0.12.5",
21
+ "zod": "^4.1.13"
22
+ },
23
+ "devDependencies": {
24
+ "@types/cors": "^2.8.19",
25
+ "@types/express": "^5.0.6",
26
+ "@types/node": "^20.6.0",
27
+ "prettier": "^3.7.4",
28
+ "typescript": "^5.4.5"
29
+ }
30
+ }
@@ -0,0 +1,17 @@
1
+ #!/usr/bin/env node
2
+ import { spawn } from 'child_process';
3
+ import { startServer } from '../server/index.js';
4
+
5
+ const port =
6
+ process.argv.find((arg) => arg.startsWith('-p=') || arg.startsWith('--port='))?.split('=')[1] ||
7
+ 6767;
8
+
9
+ const cwd = process.cwd();
10
+ console.log('Doclific CLI running in directory:', cwd);
11
+
12
+ startServer(Number(port));
13
+
14
+ // Optional: open browser automatically
15
+ const open =
16
+ process.platform === 'win32' ? 'start' : process.platform === 'darwin' ? 'open' : 'xdg-open';
17
+ spawn(open, [`http://localhost:${port}`]);
@@ -0,0 +1,39 @@
1
+ import path from 'path';
2
+ import fs from 'fs/promises';
3
+
4
+ interface FileNode {
5
+ path: string;
6
+ name: string;
7
+ type: 'file' | 'directory';
8
+ children?: FileNode[];
9
+ }
10
+
11
+ /**
12
+ * Get all contents of a folder given a filePath.
13
+ * @param filePath - The relative path to the folder
14
+ */
15
+ export const getFolderContents = async (filePath: string): Promise<FileNode[]> => {
16
+ const currentDir = process.cwd();
17
+ let fullPath = '';
18
+ if (filePath !== '') {
19
+ fullPath = path.join(currentDir, filePath);
20
+ } else {
21
+ fullPath = currentDir;
22
+ }
23
+ const entries = await fs.readdir(fullPath, { withFileTypes: true });
24
+ const nodes: FileNode[] = [];
25
+ for (const entry of entries) {
26
+ nodes.push({
27
+ path: path.join(filePath, entry.name),
28
+ name: entry.name,
29
+ type: entry.isDirectory() ? 'directory' : 'file',
30
+ });
31
+ }
32
+ return nodes;
33
+ };
34
+
35
+ export const getFileContents = async (filePath: string) => {
36
+ const currentDir = process.cwd();
37
+ const fullPath = path.join(currentDir, filePath);
38
+ return fs.readFile(fullPath, 'utf8');
39
+ };
@@ -0,0 +1,90 @@
1
+ import fs from 'fs/promises';
2
+ import path from 'path';
3
+
4
+ type FolderStructure = {
5
+ name: string;
6
+ title: string;
7
+ icon?: string;
8
+ children: FolderStructure[];
9
+ };
10
+
11
+ const scanDirectory = async (dirPath: string): Promise<FolderStructure[]> => {
12
+ const entries = await fs.readdir(dirPath, { withFileTypes: true });
13
+ const folders: FolderStructure[] = [];
14
+
15
+ for (const entry of entries) {
16
+ if (entry.isDirectory()) {
17
+ const fullPath = path.join(dirPath, entry.name);
18
+ const children = await scanDirectory(fullPath);
19
+ let config: { title: string; icon: string | undefined } = {
20
+ title: entry.name,
21
+ icon: undefined,
22
+ };
23
+ // get the config.json file
24
+ const configPath = path.join(fullPath, 'config.json');
25
+ try {
26
+ const configFile = await fs.readFile(configPath, 'utf8');
27
+ config = JSON.parse(configFile);
28
+ } catch {
29
+ throw new Error(`Config file not found for ${entry.name}`);
30
+ }
31
+ folders.push({
32
+ name: entry.name,
33
+ title: config.title,
34
+ icon: config.icon,
35
+ children,
36
+ });
37
+ }
38
+ }
39
+
40
+ return folders;
41
+ };
42
+
43
+ export const getDocs = async (): Promise<FolderStructure[]> => {
44
+ const docsFolder = path.join(process.cwd(), 'doclific');
45
+ try {
46
+ await fs.access(docsFolder);
47
+ } catch {
48
+ return [];
49
+ }
50
+ return scanDirectory(docsFolder);
51
+ };
52
+
53
+ /**
54
+ * This function will return the content of the .mdx file in the directory with the given filePath.
55
+ * @param filePath - The path to the file
56
+ * @returns
57
+ */
58
+ export const getDoc = async (filePath: string) => {
59
+ const currentDir = process.cwd();
60
+ const fullPath = path.join(currentDir, 'doclific', filePath);
61
+ try {
62
+ return fs.readFile(path.join(fullPath, 'content.mdx'), 'utf8');
63
+ } catch {
64
+ return undefined;
65
+ }
66
+ };
67
+
68
+ export const updateDoc = async (filePath: string, content: string) => {
69
+ const currentDir = process.cwd();
70
+ const fullPath = path.join(currentDir, 'doclific', filePath);
71
+ await fs.writeFile(path.join(fullPath, 'content.mdx'), content);
72
+ };
73
+
74
+ export const createDoc = async (filePath: string, title: string, icon: string | undefined) => {
75
+ const currentDir = process.cwd();
76
+ const fullPath = path.join(currentDir, 'doclific', filePath);
77
+
78
+ const newFolderName = crypto.randomUUID();
79
+ const newFolderPath = path.join(fullPath, newFolderName);
80
+ await fs.mkdir(newFolderPath, { recursive: true });
81
+ // create content.mdx file
82
+ await fs.writeFile(path.join(newFolderPath, 'content.mdx'), '# Hello World');
83
+ // create config.json file
84
+ await fs.writeFile(
85
+ path.join(newFolderPath, 'config.json'),
86
+ JSON.stringify({ title, icon }, null, 2)
87
+ );
88
+ // return the new folder path
89
+ return newFolderPath;
90
+ };
@@ -0,0 +1,48 @@
1
+ import util from 'util';
2
+ import { exec } from 'child_process';
3
+
4
+ const execAsync = util.promisify(exec);
5
+
6
+ export const getGitRepositoryName = async (): Promise<string | null> => {
7
+ try {
8
+ const GET_REPO_NAME_COMMAND = `git config --get remote.origin.url | sed -E "s|.*/(.+)\\.git|\\1|"`;
9
+ const repoName = await execAsync(GET_REPO_NAME_COMMAND);
10
+ return repoName.stdout.toString().trim();
11
+ } catch (error) {
12
+ console.error(error);
13
+ return null;
14
+ }
15
+ };
16
+
17
+ export const getGitRepositoryBranch = async (): Promise<string | null> => {
18
+ try {
19
+ const GET_REPO_BRANCH_COMMAND = `git rev-parse --abbrev-ref HEAD`;
20
+ const repoBranch = await execAsync(GET_REPO_BRANCH_COMMAND);
21
+ return repoBranch.stdout.toString().trim();
22
+ } catch (error) {
23
+ console.error(error);
24
+ return null;
25
+ }
26
+ };
27
+
28
+ export const getGitUser = async (): Promise<string | null> => {
29
+ try {
30
+ const GET_USER_COMMAND = `git config user.name`;
31
+ const user = await execAsync(GET_USER_COMMAND);
32
+ return user.stdout.toString().trim();
33
+ } catch (error) {
34
+ console.error(error);
35
+ return null;
36
+ }
37
+ };
38
+
39
+ export const getGitUserEmail = async (): Promise<string | null> => {
40
+ try {
41
+ const GET_USER_EMAIL_COMMAND = `git config user.email`;
42
+ const userEmail = await execAsync(GET_USER_EMAIL_COMMAND);
43
+ return userEmail.stdout.toString().trim();
44
+ } catch (error) {
45
+ console.error(error);
46
+ return null;
47
+ }
48
+ };
@@ -0,0 +1,55 @@
1
+ import express from 'express';
2
+ import cors from 'cors';
3
+ import { RPCHandler } from '@orpc/server/node';
4
+ import { onError } from '@orpc/server';
5
+ import { router } from './router.js';
6
+ import path from 'path';
7
+ import { fileURLToPath } from 'url';
8
+ import fs from 'fs';
9
+
10
+ const __filename = fileURLToPath(import.meta.url);
11
+ const __dirname = path.dirname(__filename);
12
+
13
+ const frontendDist = path.join(__dirname, '../../frontend/dist');
14
+
15
+ export function startServer(port: number = 6767) {
16
+ const app = express();
17
+
18
+ app.use(cors());
19
+
20
+ const handler = new RPCHandler(router, {
21
+ interceptors: [
22
+ onError((error) => {
23
+ console.error(error);
24
+ }),
25
+ ],
26
+ });
27
+
28
+ app.use('/rpc{/*path}', async (req, res, next) => {
29
+ const { matched } = await handler.handle(req, res, {
30
+ prefix: '/rpc',
31
+ context: {},
32
+ });
33
+
34
+ if (matched) return;
35
+
36
+ next();
37
+ });
38
+
39
+ // serve static files in frontend/dist
40
+ app.use((req, res, next) => {
41
+ if (req.url === '/') {
42
+ const html = fs.readFileSync(path.join(frontendDist, 'index.html'), 'utf8');
43
+ res.send(html.replace(': BACKEND_PORT', `: ${port.toString()}`));
44
+ } else {
45
+ express.static(frontendDist)(req, res, next);
46
+ }
47
+ });
48
+
49
+ app.get('/{*splat}', (req, res) => {
50
+ const html = fs.readFileSync(path.join(frontendDist, 'index.html'), 'utf8');
51
+ res.send(html.replace(': BACKEND_PORT', `: ${port.toString()}`));
52
+ });
53
+
54
+ app.listen(port, () => console.log(`Access your documentation here: http://localhost:${port}`));
55
+ }
@@ -0,0 +1,65 @@
1
+ import { os } from '@orpc/server';
2
+ import {
3
+ getGitRepositoryBranch,
4
+ getGitRepositoryName,
5
+ getGitUser,
6
+ getGitUserEmail,
7
+ } from '../core/git.js';
8
+ import { createDoc, getDoc, getDocs, updateDoc } from '../core/docs.js';
9
+ import { z } from 'zod';
10
+ import { getFileContents, getFolderContents } from '../core/codebase.js';
11
+ import path from 'path';
12
+
13
+ export const router = {
14
+ git: {
15
+ getRepoInfo: os.handler(async () => {
16
+ return {
17
+ repositoryName: await getGitRepositoryName(),
18
+ repositoryBranch: await getGitRepositoryBranch(),
19
+ user: await getGitUser(),
20
+ userEmail: await getGitUserEmail(),
21
+ };
22
+ }),
23
+ },
24
+ docs: {
25
+ getDocs: os.handler(async () => {
26
+ return await getDocs();
27
+ }),
28
+ getDoc: os.input(z.object({ filePath: z.string() })).handler(async ({ input }) => {
29
+ const { filePath } = input;
30
+ return await getDoc(filePath);
31
+ }),
32
+ updateDoc: os
33
+ .input(z.object({ filePath: z.string(), content: z.string() }))
34
+ .handler(async ({ input }) => {
35
+ const { filePath, content } = input;
36
+ return await updateDoc(filePath, content);
37
+ }),
38
+ createDoc: os
39
+ .input(
40
+ z.object({ filePath: z.string(), title: z.string(), icon: z.string().optional() })
41
+ )
42
+ .handler(async ({ input }) => {
43
+ const { filePath, title, icon } = input;
44
+ return await createDoc(filePath, title, icon);
45
+ }),
46
+ },
47
+ codebase: {
48
+ getFolderContents: os
49
+ .input(z.object({ filePath: z.string() }))
50
+ .handler(async ({ input }) => {
51
+ const { filePath } = input;
52
+ return await getFolderContents(filePath);
53
+ }),
54
+ getFileContents: os.input(z.object({ filePath: z.string() })).handler(async ({ input }) => {
55
+ const { filePath } = input;
56
+ const currentDir = process.cwd();
57
+ return {
58
+ contents: await getFileContents(filePath),
59
+ fullPath: path.join(currentDir, '..', filePath),
60
+ };
61
+ }),
62
+ },
63
+ };
64
+
65
+ export type Router = typeof router;
package/tsconfig.json ADDED
@@ -0,0 +1,15 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2020",
4
+ "module": "ESNext",
5
+ "moduleResolution": "bundler",
6
+ "lib": ["ES2020"],
7
+ "outDir": "dist",
8
+ "rootDir": "src",
9
+ "strict": true,
10
+ "esModuleInterop": true,
11
+ "skipLibCheck": true,
12
+ "types": ["node"]
13
+ },
14
+ "include": ["src"]
15
+ }