opensidian 1.0.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 (137) hide show
  1. package/.eslintrc.json +1 -0
  2. package/.github/ISSUE_TEMPLATE/bug_report.md +49 -0
  3. package/.github/ISSUE_TEMPLATE/feature_request.md +35 -0
  4. package/.github/ISSUE_TEMPLATE/question.md +23 -0
  5. package/.github/PULL_REQUEST_TEMPLATE.md +45 -0
  6. package/.github/README.md +5 -0
  7. package/.github/workflows/cd.yml +44 -0
  8. package/.github/workflows/ci.yml +128 -0
  9. package/.github/workflows/qa.yml +45 -0
  10. package/.planning/PROJECT.md +96 -0
  11. package/.planning/REQUIREMENTS.md +66 -0
  12. package/.planning/ROADMAP.md +129 -0
  13. package/.planning/STATE.md +47 -0
  14. package/.planning/config.json +14 -0
  15. package/CONTRIBUTING.md +232 -0
  16. package/LICENSE +21 -0
  17. package/README.md +244 -0
  18. package/dist/api/auth.d.ts +5 -0
  19. package/dist/api/auth.d.ts.map +1 -0
  20. package/dist/api/auth.js +112 -0
  21. package/dist/api/auth.js.map +1 -0
  22. package/dist/api/routes.d.ts +3 -0
  23. package/dist/api/routes.d.ts.map +1 -0
  24. package/dist/api/routes.js +119 -0
  25. package/dist/api/routes.js.map +1 -0
  26. package/dist/api/themes.d.ts +3 -0
  27. package/dist/api/themes.d.ts.map +1 -0
  28. package/dist/api/themes.js +48 -0
  29. package/dist/api/themes.js.map +1 -0
  30. package/dist/core/graph.d.ts +16 -0
  31. package/dist/core/graph.d.ts.map +1 -0
  32. package/dist/core/graph.js +115 -0
  33. package/dist/core/graph.js.map +1 -0
  34. package/dist/core/markdown.d.ts +21 -0
  35. package/dist/core/markdown.d.ts.map +1 -0
  36. package/dist/core/markdown.js +77 -0
  37. package/dist/core/markdown.js.map +1 -0
  38. package/dist/core/search.d.ts +34 -0
  39. package/dist/core/search.d.ts.map +1 -0
  40. package/dist/core/search.js +159 -0
  41. package/dist/core/search.js.map +1 -0
  42. package/dist/core/sync.d.ts +30 -0
  43. package/dist/core/sync.d.ts.map +1 -0
  44. package/dist/core/sync.js +121 -0
  45. package/dist/core/sync.js.map +1 -0
  46. package/dist/core/vault.d.ts +28 -0
  47. package/dist/core/vault.d.ts.map +1 -0
  48. package/dist/core/vault.js +235 -0
  49. package/dist/core/vault.js.map +1 -0
  50. package/dist/index.d.ts +2 -0
  51. package/dist/index.d.ts.map +1 -0
  52. package/dist/index.js +32 -0
  53. package/dist/index.js.map +1 -0
  54. package/dist/mcp/cli.d.ts +3 -0
  55. package/dist/mcp/cli.d.ts.map +1 -0
  56. package/dist/mcp/cli.js +7 -0
  57. package/dist/mcp/cli.js.map +1 -0
  58. package/dist/mcp/server.d.ts +18 -0
  59. package/dist/mcp/server.d.ts.map +1 -0
  60. package/dist/mcp/server.js +272 -0
  61. package/dist/mcp/server.js.map +1 -0
  62. package/dist/plugins/host.d.ts +23 -0
  63. package/dist/plugins/host.d.ts.map +1 -0
  64. package/dist/plugins/host.js +104 -0
  65. package/dist/plugins/host.js.map +1 -0
  66. package/dist/plugins/sample-plugin.d.ts +10 -0
  67. package/dist/plugins/sample-plugin.d.ts.map +1 -0
  68. package/dist/plugins/sample-plugin.js +23 -0
  69. package/dist/plugins/sample-plugin.js.map +1 -0
  70. package/dist/server.d.ts +15 -0
  71. package/dist/server.d.ts.map +1 -0
  72. package/dist/server.js +77 -0
  73. package/dist/server.js.map +1 -0
  74. package/dist/shared/types.d.ts +86 -0
  75. package/dist/shared/types.d.ts.map +1 -0
  76. package/dist/shared/types.js +2 -0
  77. package/dist/shared/types.js.map +1 -0
  78. package/docker/Dockerfile +37 -0
  79. package/docker/docker-compose.yml +46 -0
  80. package/docs/ARCHITECTURE.md +321 -0
  81. package/futuras_implementacoes.md +0 -0
  82. package/package.json +65 -0
  83. package/scripts/fix-gitignore.ps1 +5 -0
  84. package/scripts/seed-notes.mjs +60 -0
  85. package/src/api/auth.ts +130 -0
  86. package/src/api/routes.ts +133 -0
  87. package/src/api/themes.ts +60 -0
  88. package/src/core/graph.ts +145 -0
  89. package/src/core/markdown.ts +92 -0
  90. package/src/core/search.ts +208 -0
  91. package/src/core/sync.ts +157 -0
  92. package/src/core/vault.ts +286 -0
  93. package/src/index.ts +37 -0
  94. package/src/mcp/cli.ts +7 -0
  95. package/src/mcp/server.ts +296 -0
  96. package/src/plugins/host.ts +120 -0
  97. package/src/plugins/sample-plugin.ts +29 -0
  98. package/src/server.ts +90 -0
  99. package/src/shared/types.ts +92 -0
  100. package/tests/api/routes.test.ts +167 -0
  101. package/tests/core/graph.test.ts +236 -0
  102. package/tests/core/markdown.test.ts +157 -0
  103. package/tests/core/search.test.ts +132 -0
  104. package/tests/core/sync.test.ts +62 -0
  105. package/tests/core/vault.test.ts +162 -0
  106. package/tests/mcp/server.test.ts +118 -0
  107. package/tests/plugins/host.test.ts +165 -0
  108. package/tests/plugins/sample-plugin.test.ts +35 -0
  109. package/tests/server.test.ts +76 -0
  110. package/tsconfig.json +27 -0
  111. package/vite.config.ts +27 -0
  112. package/vitest.config.ts +33 -0
  113. package/web/index.html +13 -0
  114. package/web/package.json +26 -0
  115. package/web/public/favicon.svg +4 -0
  116. package/web/src/App.tsx +63 -0
  117. package/web/src/api/auth.ts +65 -0
  118. package/web/src/api/client.ts +117 -0
  119. package/web/src/api/themes.ts +78 -0
  120. package/web/src/components/GraphView.tsx +139 -0
  121. package/web/src/components/Layout.tsx +74 -0
  122. package/web/src/components/LoginPage.tsx +52 -0
  123. package/web/src/components/NoteEditor.tsx +114 -0
  124. package/web/src/components/NoteList.tsx +95 -0
  125. package/web/src/components/RegisterPage.tsx +58 -0
  126. package/web/src/components/SearchBar.tsx +71 -0
  127. package/web/src/components/SearchPanel.tsx +152 -0
  128. package/web/src/components/ThemeEditor.tsx +129 -0
  129. package/web/src/components/ThemeSelector.tsx +41 -0
  130. package/web/src/components/VaultList.tsx +89 -0
  131. package/web/src/hooks/AuthContext.tsx +57 -0
  132. package/web/src/hooks/ThemeContext.tsx +77 -0
  133. package/web/src/hooks/useWebSocket.ts +34 -0
  134. package/web/src/main.tsx +10 -0
  135. package/web/src/styles/global.css +449 -0
  136. package/web/tsconfig.json +21 -0
  137. package/web/vite.config.ts +19 -0
@@ -0,0 +1,119 @@
1
+ import express from 'express';
2
+ import { VaultManager } from '../core/vault.js';
3
+ import { GraphEngine } from '../core/graph.js';
4
+ const router = express.Router();
5
+ const vaultManager = new VaultManager();
6
+ const graphEngine = new GraphEngine();
7
+ router.get('/vaults', (_req, res) => {
8
+ const vaults = vaultManager.listVaults();
9
+ return res.json({ vaults });
10
+ });
11
+ router.post('/vaults', (req, res) => {
12
+ const { name, path } = req.body;
13
+ if (!name || !path) {
14
+ return res.status(400).json({ error: 'name and path are required' });
15
+ }
16
+ const vault = vaultManager.createVault(name, path);
17
+ return res.status(201).json({ vault });
18
+ });
19
+ router.get('/vaults/:id', (req, res) => {
20
+ const vault = vaultManager.openVault(req.params.id);
21
+ if (!vault) {
22
+ return res.status(404).json({ error: 'Vault not found' });
23
+ }
24
+ return res.json({ vault });
25
+ });
26
+ router.delete('/vaults/:id', (_req, res) => {
27
+ return res.status(501).json({ error: 'Not implemented' });
28
+ });
29
+ router.get('/notes', (req, res) => {
30
+ const vaultPath = req.query.vault;
31
+ if (!vaultPath) {
32
+ return res.status(400).json({ error: 'vault query parameter is required' });
33
+ }
34
+ const vault = vaultManager.openVault(vaultPath);
35
+ if (!vault) {
36
+ return res.status(404).json({ error: 'Vault not found' });
37
+ }
38
+ return res.json({ notes: vault.notes });
39
+ });
40
+ router.post('/notes', async (req, res) => {
41
+ const { vaultPath, filename, content } = req.body;
42
+ if (!vaultPath || !filename || !content) {
43
+ return res.status(400).json({ error: 'vaultPath, filename, and content are required' });
44
+ }
45
+ const note = await vaultManager.createNote(vaultPath, filename, content);
46
+ graphEngine.indexNote(vaultPath, note);
47
+ return res.status(201).json({ note });
48
+ });
49
+ router.get('/notes/search', async (req, res) => {
50
+ const vaultPath = req.query.vault;
51
+ const q = req.query.q;
52
+ if (!vaultPath || !q) {
53
+ return res.status(400).json({ error: 'vault and q parameters are required' });
54
+ }
55
+ const tag = req.query.tag;
56
+ const after = req.query.after;
57
+ const before = req.query.before;
58
+ const hasBacklinks = req.query.hasBacklinks === 'true' ? true : req.query.hasBacklinks === 'false' ? false : undefined;
59
+ const results = await vaultManager.fullTextSearch(vaultPath, q, { tag, after, before, hasBacklinks });
60
+ return res.json({ results });
61
+ });
62
+ router.get('/notes/:path(*)', async (req, res) => {
63
+ const vaultPath = req.query.vault;
64
+ if (!vaultPath) {
65
+ return res.status(400).json({ error: 'vault query parameter is required' });
66
+ }
67
+ const note = await vaultManager.readNote(vaultPath, req.params.path);
68
+ if (!note) {
69
+ return res.status(404).json({ error: 'Note not found' });
70
+ }
71
+ return res.json({ note });
72
+ });
73
+ router.put('/notes/:path(*)', async (req, res) => {
74
+ const vaultPath = req.query.vault;
75
+ const { content } = req.body;
76
+ if (!vaultPath) {
77
+ return res.status(400).json({ error: 'vault query parameter is required' });
78
+ }
79
+ if (!content) {
80
+ return res.status(400).json({ error: 'content is required' });
81
+ }
82
+ const note = await vaultManager.updateNote(vaultPath, req.params.path, content);
83
+ graphEngine.indexNote(vaultPath, note);
84
+ return res.json({ note });
85
+ });
86
+ router.delete('/notes/:path(*)', async (req, res) => {
87
+ const vaultPath = req.query.vault;
88
+ if (!vaultPath) {
89
+ return res.status(400).json({ error: 'vault query parameter is required' });
90
+ }
91
+ await vaultManager.deleteNote(vaultPath, req.params.path);
92
+ graphEngine.removeNote(vaultPath, req.params.path);
93
+ return res.status(204).send();
94
+ });
95
+ router.get('/graph', (req, res) => {
96
+ const vaultPath = req.query.vault;
97
+ if (!vaultPath) {
98
+ return res.status(400).json({ error: 'vault query parameter is required' });
99
+ }
100
+ const graph = graphEngine.getGraph(vaultPath);
101
+ if (!graph) {
102
+ return res.status(404).json({ error: 'Graph not found' });
103
+ }
104
+ return res.json({ graph });
105
+ });
106
+ router.get('/graph/neighbors/:path(*)', (req, res) => {
107
+ const vaultPath = req.query.vault;
108
+ if (!vaultPath) {
109
+ return res.status(400).json({ error: 'vault query parameter is required' });
110
+ }
111
+ const neighbors = graphEngine.getNeighbors(vaultPath, req.params.path);
112
+ return res.json({ neighbors });
113
+ });
114
+ router.use((err, _req, res, _next) => {
115
+ console.error('API Error:', err);
116
+ return res.status(500).json({ error: err.message || 'Internal server error' });
117
+ });
118
+ export default router;
119
+ //# sourceMappingURL=routes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes.js","sourceRoot":"","sources":["../../src/api/routes.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ declare const router: import("express-serve-static-core").Router;
2
+ export default router;
3
+ //# sourceMappingURL=themes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"themes.d.ts","sourceRoot":"","sources":["../../src/api/themes.ts"],"names":[],"mappings":"AAeA,QAAA,MAAM,MAAM,4CAAmB,CAAC;AA4ChC,eAAe,MAAM,CAAC"}
@@ -0,0 +1,48 @@
1
+ import express from 'express';
2
+ import fs from 'fs/promises';
3
+ import path from 'path';
4
+ const THEMES_DIR = path.resolve(process.cwd(), 'data', 'themes');
5
+ async function ensureDir() {
6
+ await fs.mkdir(THEMES_DIR, { recursive: true });
7
+ }
8
+ const router = express.Router();
9
+ router.get('/', async (_req, res) => {
10
+ try {
11
+ await ensureDir();
12
+ const files = await fs.readdir(THEMES_DIR);
13
+ const themes = [];
14
+ for (const f of files) {
15
+ if (f.endsWith('.json')) {
16
+ const data = await fs.readFile(path.join(THEMES_DIR, f), 'utf-8');
17
+ themes.push(JSON.parse(data));
18
+ }
19
+ }
20
+ res.json({ themes });
21
+ }
22
+ catch {
23
+ res.json({ themes: [] });
24
+ }
25
+ });
26
+ router.post('/', async (req, res) => {
27
+ const { name, variables } = req.body;
28
+ if (!name || !variables) {
29
+ res.status(400).json({ error: 'name e variables são obrigatórios' });
30
+ return;
31
+ }
32
+ await ensureDir();
33
+ const filename = name.toLowerCase().replace(/\s+/g, '-') + '.json';
34
+ await fs.writeFile(path.join(THEMES_DIR, filename), JSON.stringify({ name, variables }, null, 2));
35
+ res.status(201).json({ theme: { name, variables } });
36
+ });
37
+ router.delete('/:name', async (req, res) => {
38
+ try {
39
+ const filename = req.params.name.toLowerCase().replace(/\s+/g, '-') + '.json';
40
+ await fs.unlink(path.join(THEMES_DIR, filename));
41
+ res.status(204).send();
42
+ }
43
+ catch {
44
+ res.status(404).json({ error: 'Tema não encontrado' });
45
+ }
46
+ });
47
+ export default router;
48
+ //# sourceMappingURL=themes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"themes.js","sourceRoot":"","sources":["../../src/api/themes.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,16 @@
1
+ import { Note, Graph, GraphNode } from '../shared/types.js';
2
+ export declare class GraphEngine {
3
+ private graphs;
4
+ indexNote(vaultPath: string, note: Note): void;
5
+ removeNote(vaultPath: string, notePath: string): void;
6
+ getGraph(vaultPath: string): Graph | null;
7
+ getNeighbors(vaultPath: string, notePath: string): GraphNode[];
8
+ getBacklinks(vaultPath: string, notePath: string): GraphNode[];
9
+ searchByTag(vaultPath: string, tag: string): GraphNode[];
10
+ private extractTags;
11
+ computeStats(vaultPath: string): {
12
+ mostConnected: GraphNode[];
13
+ orphanNotes: GraphNode[];
14
+ };
15
+ }
16
+ //# sourceMappingURL=graph.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graph.d.ts","sourceRoot":"","sources":["../../src/core/graph.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAE5D,qBAAa,WAAW;IACtB,OAAO,CAAC,MAAM,CAAiC;IAE/C,SAAS,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,GAAG,IAAI;IA0C9C,UAAU,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,IAAI;IAYrD,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,KAAK,GAAG,IAAI;IAIzC,YAAY,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,SAAS,EAAE;IAmB9D,YAAY,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,SAAS,EAAE;IAU9D,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,SAAS,EAAE;IAQxD,OAAO,CAAC,WAAW;IAiBnB,YAAY,CAAC,SAAS,EAAE,MAAM,GAAG;QAAE,aAAa,EAAE,SAAS,EAAE,CAAC;QAAC,WAAW,EAAE,SAAS,EAAE,CAAA;KAAE;CA2B1F"}
@@ -0,0 +1,115 @@
1
+ export class GraphEngine {
2
+ graphs = new Map();
3
+ indexNote(vaultPath, note) {
4
+ let graph = this.graphs.get(vaultPath);
5
+ if (!graph) {
6
+ graph = { nodes: [], edges: [], metadata: { totalNotes: 0, totalLinks: 0 } };
7
+ this.graphs.set(vaultPath, graph);
8
+ }
9
+ const existingNodeIndex = graph.nodes.findIndex(n => n.id === note.path);
10
+ const node = {
11
+ id: note.path,
12
+ label: note.filename,
13
+ tags: this.extractTags(note.frontmatter),
14
+ connections: note.links.length,
15
+ };
16
+ if (existingNodeIndex >= 0) {
17
+ graph.nodes[existingNodeIndex] = node;
18
+ }
19
+ else {
20
+ graph.nodes.push(node);
21
+ }
22
+ const edgeSet = new Set(graph.edges.map(e => `${e.source}-${e.target}`));
23
+ for (const link of note.links) {
24
+ const targetPath = link.endsWith('.md') ? link : `${link}.md`;
25
+ const edgeId = `${note.path}-${targetPath}`;
26
+ if (!edgeSet.has(edgeId)) {
27
+ graph.edges.push({
28
+ id: edgeId,
29
+ source: note.path,
30
+ target: targetPath,
31
+ type: 'wikilink',
32
+ });
33
+ edgeSet.add(edgeId);
34
+ }
35
+ }
36
+ graph.metadata.totalNotes = graph.nodes.length;
37
+ graph.metadata.totalLinks = graph.edges.length;
38
+ }
39
+ removeNote(vaultPath, notePath) {
40
+ const graph = this.graphs.get(vaultPath);
41
+ if (!graph)
42
+ return;
43
+ graph.nodes = graph.nodes.filter(n => n.id !== notePath);
44
+ graph.edges = graph.edges.filter(e => e.source !== notePath && e.target !== notePath);
45
+ graph.metadata.totalNotes = graph.nodes.length;
46
+ graph.metadata.totalLinks = graph.edges.length;
47
+ }
48
+ getGraph(vaultPath) {
49
+ return this.graphs.get(vaultPath) || null;
50
+ }
51
+ getNeighbors(vaultPath, notePath) {
52
+ const graph = this.graphs.get(vaultPath);
53
+ if (!graph)
54
+ return [];
55
+ const neighborIds = new Set();
56
+ for (const edge of graph.edges) {
57
+ if (edge.source === notePath) {
58
+ neighborIds.add(edge.target);
59
+ }
60
+ if (edge.target === notePath) {
61
+ neighborIds.add(edge.source);
62
+ }
63
+ }
64
+ return graph.nodes.filter(n => neighborIds.has(n.id));
65
+ }
66
+ getBacklinks(vaultPath, notePath) {
67
+ const graph = this.graphs.get(vaultPath);
68
+ if (!graph)
69
+ return [];
70
+ return graph.nodes.filter(n => {
71
+ return graph.edges.some(e => e.target === notePath && e.source === n.id);
72
+ });
73
+ }
74
+ searchByTag(vaultPath, tag) {
75
+ const graph = this.graphs.get(vaultPath);
76
+ if (!graph)
77
+ return [];
78
+ return graph.nodes.filter(n => n.tags?.includes(tag));
79
+ }
80
+ extractTags(frontmatter) {
81
+ if (!frontmatter)
82
+ return [];
83
+ const tags = frontmatter.tags;
84
+ if (!tags)
85
+ return [];
86
+ if (Array.isArray(tags)) {
87
+ return tags.map(t => String(t));
88
+ }
89
+ if (typeof tags === 'string') {
90
+ return tags.split(',').map(t => t.trim());
91
+ }
92
+ return [];
93
+ }
94
+ computeStats(vaultPath) {
95
+ const graph = this.graphs.get(vaultPath);
96
+ if (!graph) {
97
+ return { mostConnected: [], orphanNotes: [] };
98
+ }
99
+ const connectionCounts = new Map();
100
+ for (const edge of graph.edges) {
101
+ connectionCounts.set(edge.source, (connectionCounts.get(edge.source) || 0) + 1);
102
+ connectionCounts.set(edge.target, (connectionCounts.get(edge.target) || 0) + 1);
103
+ }
104
+ const nodesWithCount = graph.nodes.map(n => ({
105
+ ...n,
106
+ connectionCount: connectionCounts.get(n.id) || 0,
107
+ }));
108
+ const mostConnected = nodesWithCount
109
+ .sort((a, b) => b.connectionCount - a.connectionCount)
110
+ .slice(0, 10);
111
+ const orphanNotes = nodesWithCount.filter(n => n.connectionCount === 0);
112
+ return { mostConnected, orphanNotes };
113
+ }
114
+ }
115
+ //# sourceMappingURL=graph.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graph.js","sourceRoot":"","sources":["../../src/core/graph.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ import { ParsedNote } from '../shared/types.js';
2
+ export declare class MarkdownParser {
3
+ constructor();
4
+ parse<T extends Record<string, unknown> = Record<string, unknown>>(content: string): ParsedNote<T>;
5
+ parseContent(content: string): {
6
+ frontmatter: Record<string, unknown>;
7
+ body: string;
8
+ };
9
+ extractLinks(content: string): string[];
10
+ extractHeadings(content: string): Array<{
11
+ level: number;
12
+ text: string;
13
+ id: string;
14
+ }>;
15
+ slugify(text: string): string;
16
+ toMarkdown(note: {
17
+ frontmatter?: Record<string, unknown>;
18
+ body: string;
19
+ }): string;
20
+ }
21
+ //# sourceMappingURL=markdown.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown.d.ts","sourceRoot":"","sources":["../../src/core/markdown.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAEhD,qBAAa,cAAc;;IAQzB,KAAK,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,EAAE,MAAM,GAAG,UAAU,CAAC,CAAC,CAAC;IAYlG,YAAY,CAAC,OAAO,EAAE,MAAM,GAAG;QAAE,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE;IAQrF,YAAY,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,EAAE;IAmBvC,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,KAAK,CAAC;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,EAAE,EAAE,MAAM,CAAA;KAAE,CAAC;IAepF,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAW7B,UAAU,CAAC,IAAI,EAAE;QAAE,WAAW,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,MAAM;CAclF"}
@@ -0,0 +1,77 @@
1
+ import { marked } from 'marked';
2
+ import fm from 'front-matter';
3
+ export class MarkdownParser {
4
+ constructor() {
5
+ marked.setOptions({
6
+ gfm: true,
7
+ breaks: true,
8
+ });
9
+ }
10
+ parse(content) {
11
+ const result = fm(content);
12
+ return {
13
+ frontmatter: result.attributes ?? {},
14
+ body: result.body,
15
+ html: marked.parse(result.body),
16
+ links: this.extractLinks(result.body),
17
+ headings: this.extractHeadings(result.body),
18
+ };
19
+ }
20
+ parseContent(content) {
21
+ const result = fm(content);
22
+ return {
23
+ frontmatter: result.attributes || {},
24
+ body: result.body,
25
+ };
26
+ }
27
+ extractLinks(content) {
28
+ const links = [];
29
+ const wikiLinkRegex = /\[\[([^\]|]+)(?:\|([^\]]+))?\]\]/g;
30
+ const mdLinkRegex = /\[([^\]]+)\]\(([^)]+)\)/g;
31
+ let match;
32
+ while ((match = wikiLinkRegex.exec(content)) !== null) {
33
+ links.push(match[1]);
34
+ }
35
+ while ((match = mdLinkRegex.exec(content)) !== null) {
36
+ if (!match[2].startsWith('http')) {
37
+ links.push(match[2].replace(/\.md$/, ''));
38
+ }
39
+ }
40
+ return [...new Set(links)];
41
+ }
42
+ extractHeadings(content) {
43
+ const headings = [];
44
+ const headingRegex = /^(#{1,6})\s+(.+)$/gm;
45
+ let match;
46
+ while ((match = headingRegex.exec(content)) !== null) {
47
+ const level = match[1].length;
48
+ const text = match[2].trim();
49
+ const id = this.slugify(text);
50
+ headings.push({ level, text, id });
51
+ }
52
+ return headings;
53
+ }
54
+ slugify(text) {
55
+ return text
56
+ .normalize('NFKD')
57
+ .replace(/[\u0300-\u036f]/g, '')
58
+ .toLowerCase()
59
+ .replace(/[^\w\s-]/g, '')
60
+ .replace(/\s+/g, '-')
61
+ .replace(/-+/g, '-')
62
+ .trim();
63
+ }
64
+ toMarkdown(note) {
65
+ let result = '';
66
+ if (note.frontmatter && Object.keys(note.frontmatter).length > 0) {
67
+ result += '---\n';
68
+ for (const [key, value] of Object.entries(note.frontmatter)) {
69
+ result += `${key}: ${value}\n`;
70
+ }
71
+ result += '---\n\n';
72
+ }
73
+ result += note.body;
74
+ return result;
75
+ }
76
+ }
77
+ //# sourceMappingURL=markdown.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown.js","sourceRoot":"","sources":["../../src/core/markdown.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAChC,OAAO,EAAE,MAAM,cAAc,CAAC;AAG9B,MAAM,OAAO,cAAc;IACzB;QACE,MAAM,CAAC,UAAU,CAAC;YAChB,GAAG,EAAE,IAAI;YACT,MAAM,EAAE,IAAI;SACb,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAA8D,OAAe;QAChF,MAAM,MAAM,GAAG,EAAE,CAAI,OAAO,CAAC,CAAC;QAE9B,OAAO;YACL,WAAW,EAAE,MAAM,CAAC,UAAU,IAAK,EAAQ;YAC3C,IAAI,EAAE,MAAM,CAAC,IAAI;YACjB,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAW;YACzC,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC;YACrC,QAAQ,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC;SAC5C,CAAC;IACJ,CAAC;IAED,YAAY,CAAC,OAAe;QAC1B,MAAM,MAAM,GAAG,EAAE,CAA0B,OAAO,CAAC,CAAC;QACpD,OAAO;YACL,WAAW,EAAE,MAAM,CAAC,UAAU,IAAI,EAAE;YACpC,IAAI,EAAE,MAAM,CAAC,IAAI;SAClB,CAAC;IACJ,CAAC;IAED,YAAY,CAAC,OAAe;QAC1B,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,MAAM,aAAa,GAAG,mCAAmC,CAAC;QAC1D,MAAM,WAAW,GAAG,0BAA0B,CAAC;QAE/C,IAAI,KAAK,CAAC;QACV,OAAO,CAAC,KAAK,GAAG,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;YACtD,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACvB,CAAC;QACD,OAAO,CAAC,KAAK,GAAG,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;YACpD,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;gBACjC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;YAC5C,CAAC;QACH,CAAC;QAED,OAAO,CAAC,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC;IAC7B,CAAC;IAED,eAAe,CAAC,OAAe;QAC7B,MAAM,QAAQ,GAAuD,EAAE,CAAC;QACxE,MAAM,YAAY,GAAG,qBAAqB,CAAC;QAE3C,IAAI,KAAK,CAAC;QACV,OAAO,CAAC,KAAK,GAAG,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;YACrD,MAAM,KAAK,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;YAC9B,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAC7B,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAC9B,QAAQ,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,OAAO,CAAC,IAAY;QAClB,OAAO,IAAI;aACR,SAAS,CAAC,MAAM,CAAC;aACjB,OAAO,CAAC,kBAAkB,EAAE,EAAE,CAAC;aAC/B,WAAW,EAAE;aACb,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC;aACxB,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;aACpB,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;aACnB,IAAI,EAAE,CAAC;IACZ,CAAC;IAED,UAAU,CAAC,IAA6D;QACtE,IAAI,MAAM,GAAG,EAAE,CAAC;QAEhB,IAAI,IAAI,CAAC,WAAW,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACjE,MAAM,IAAI,OAAO,CAAC;YAClB,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;gBAC5D,MAAM,IAAI,GAAG,GAAG,KAAK,KAAK,IAAI,CAAC;YACjC,CAAC;YACD,MAAM,IAAI,SAAS,CAAC;QACtB,CAAC;QAED,MAAM,IAAI,IAAI,CAAC,IAAI,CAAC;QACpB,OAAO,MAAM,CAAC;IAChB,CAAC;CACF"}
@@ -0,0 +1,34 @@
1
+ export interface SearchDoc {
2
+ path: string;
3
+ filename: string;
4
+ content: string;
5
+ tags: string[];
6
+ modified: string;
7
+ backlinks: number;
8
+ }
9
+ export interface SearchResult {
10
+ path: string;
11
+ filename: string;
12
+ snippet: string;
13
+ score: number;
14
+ tags: string[];
15
+ modified: string;
16
+ backlinks: number;
17
+ }
18
+ export declare class FullTextIndexer {
19
+ private index;
20
+ private docCount;
21
+ private indexDir;
22
+ constructor(indexDir?: string);
23
+ load(): Promise<void>;
24
+ save(): Promise<void>;
25
+ indexNote(note: SearchDoc): void;
26
+ removeNote(notePath: string): void;
27
+ search(query: string, allNotes: SearchDoc[], filters?: {
28
+ tag?: string;
29
+ after?: string;
30
+ before?: string;
31
+ hasBacklinks?: boolean;
32
+ }): SearchResult[];
33
+ }
34
+ //# sourceMappingURL=search.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.d.ts","sourceRoot":"","sources":["../../src/core/search.ts"],"names":[],"mappings":"AAUA,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AA4ED,qBAAa,eAAe;IAC1B,OAAO,CAAC,KAAK,CAAqB;IAClC,OAAO,CAAC,QAAQ,CAAK;IACrB,OAAO,CAAC,QAAQ,CAAS;gBAEb,QAAQ,SAAwB;IAItC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAarB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAQ3B,SAAS,CAAC,IAAI,EAAE,SAAS,GAAG,IAAI;IAiBhC,UAAU,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAOlC,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,SAAS,EAAE,EAAE,OAAO,CAAC,EAAE;QACrD,GAAG,CAAC,EAAE,MAAM,CAAC;QACb,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,YAAY,CAAC,EAAE,OAAO,CAAC;KACxB,GAAG,YAAY,EAAE;CA6CnB"}
@@ -0,0 +1,159 @@
1
+ import fs from 'fs/promises';
2
+ import path from 'path';
3
+ import { existsSync } from 'fs';
4
+ const STOPWORDS = new Set([
5
+ 'de', 'da', 'do', 'das', 'dos', 'a', 'o', 'e', 'em', 'para', 'com',
6
+ 'um', 'uma', 'uns', 'umas', 'no', 'na', 'nos', 'nas', 'ao', 'aos',
7
+ 'à', 'às', 'pelo', 'pela', 'pelos', 'pelas', 'que', 'se', 'por',
8
+ 'como', 'mais', 'mas', 'ou', 'entre', 'sem', 'sua', 'seu', 'seus',
9
+ 'suas', 'meu', 'minha', 'teu', 'tuas', 'nosso', 'nossa', 'isto',
10
+ 'isso', 'aquilo', 'este', 'esta', 'esse', 'essa', 'aquele', 'aquela',
11
+ 'the', 'and', 'for', 'are', 'but', 'not', 'you', 'all', 'can',
12
+ 'had', 'her', 'was', 'one', 'our', 'out', 'has', 'have', 'been',
13
+ 'some', 'them', 'than', 'that', 'this', 'very', 'were', 'will',
14
+ 'with', 'from', 'they', 'what', 'when', 'where', 'which', 'who',
15
+ 'how', 'its', 'also', 'into', 'over', 'then', 'many', 'each',
16
+ 'would', 'could', 'should', 'about', 'there', 'their', 'other',
17
+ ]);
18
+ const PUNCTUATION = /[^\w\s]/g;
19
+ const WHITESPACE = /\s+/g;
20
+ function normalize(text) {
21
+ return text
22
+ .normalize('NFD')
23
+ .replace(/[\u0300-\u036f]/g, '')
24
+ .toLowerCase();
25
+ }
26
+ function tokenize(text) {
27
+ const cleaned = normalize(text).replace(PUNCTUATION, ' ').replace(WHITESPACE, ' ');
28
+ return cleaned.split(' ').filter(t => t.length > 1 && !STOPWORDS.has(t));
29
+ }
30
+ function highlightSnippet(text, queryTerms, maxLen = 160) {
31
+ const lower = normalize(text);
32
+ let bestIdx = 0;
33
+ let bestScore = 0;
34
+ for (const term of queryTerms) {
35
+ const idx = lower.indexOf(term);
36
+ if (idx >= 0) {
37
+ const proximity = queryTerms.reduce((acc, t) => acc + (lower.indexOf(t, Math.max(0, idx - 40)) >= 0 ? 1 : 0), 0);
38
+ if (proximity > bestScore) {
39
+ bestScore = proximity;
40
+ bestIdx = idx;
41
+ }
42
+ }
43
+ }
44
+ let start = Math.max(0, bestIdx - 60);
45
+ let end = Math.min(text.length, start + maxLen);
46
+ if (start > 0)
47
+ start = text.indexOf(' ', start - 20) + 1 || start;
48
+ if (end < text.length)
49
+ end = text.lastIndexOf(' ', end) || end;
50
+ let snippet = text.slice(start, end);
51
+ for (const term of queryTerms) {
52
+ const regex = new RegExp(`(${escapeRegex(term)})`, 'gi');
53
+ snippet = snippet.replace(regex, '<mark>$1</mark>');
54
+ }
55
+ return snippet;
56
+ }
57
+ function escapeRegex(str) {
58
+ return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
59
+ }
60
+ function computeTF(terms) {
61
+ const freq = new Map();
62
+ for (const t of terms)
63
+ freq.set(t, (freq.get(t) || 0) + 1);
64
+ const maxFreq = Math.max(...freq.values(), 1);
65
+ for (const [k, v] of freq)
66
+ freq.set(k, v / maxFreq);
67
+ return freq;
68
+ }
69
+ export class FullTextIndexer {
70
+ index = {};
71
+ docCount = 0;
72
+ indexDir;
73
+ constructor(indexDir = './data/search-index') {
74
+ this.indexDir = indexDir;
75
+ }
76
+ async load() {
77
+ try {
78
+ const filePath = path.resolve(this.indexDir, 'index.json');
79
+ if (!existsSync(filePath))
80
+ return;
81
+ const data = JSON.parse(await fs.readFile(filePath, 'utf-8'));
82
+ this.index = data.index || {};
83
+ this.docCount = data.docCount || 0;
84
+ }
85
+ catch {
86
+ this.index = {};
87
+ this.docCount = 0;
88
+ }
89
+ }
90
+ async save() {
91
+ await fs.mkdir(this.indexDir, { recursive: true });
92
+ await fs.writeFile(path.join(this.indexDir, 'index.json'), JSON.stringify({ index: this.index, docCount: this.docCount }, null, 2));
93
+ }
94
+ indexNote(note) {
95
+ const text = `${note.filename} ${note.content} ${(note.tags || []).join(' ')}`;
96
+ const terms = tokenize(text);
97
+ const tf = computeTF(terms);
98
+ for (const [term, freq] of tf) {
99
+ if (!this.index[term])
100
+ this.index[term] = {};
101
+ this.index[term][note.path] = freq;
102
+ }
103
+ this.docCount = Object.keys(Object.values(this.index).reduce((acc, t) => { Object.keys(t).forEach(k => { acc[k] = true; }); return acc; }, {})).length;
104
+ }
105
+ removeNote(notePath) {
106
+ for (const term of Object.keys(this.index)) {
107
+ delete this.index[term][notePath];
108
+ if (Object.keys(this.index[term]).length === 0)
109
+ delete this.index[term];
110
+ }
111
+ }
112
+ search(query, allNotes, filters) {
113
+ const queryTerms = tokenize(query);
114
+ if (queryTerms.length === 0)
115
+ return [];
116
+ const idf = (term) => {
117
+ const df = Object.keys(this.index[term] || {}).length;
118
+ return df > 0 ? Math.log(1 + (this.docCount - df + 0.5) / (df + 0.5)) + 1 : 0;
119
+ };
120
+ const docScores = new Map();
121
+ for (const term of queryTerms) {
122
+ const postings = this.index[term];
123
+ if (!postings)
124
+ continue;
125
+ const weight = idf(term);
126
+ for (const [docPath, tf] of Object.entries(postings)) {
127
+ docScores.set(docPath, (docScores.get(docPath) || 0) + tf * weight);
128
+ }
129
+ }
130
+ const noteMap = new Map(allNotes.map(n => [n.path, n]));
131
+ const results = [];
132
+ for (const [docPath, score] of docScores) {
133
+ const note = noteMap.get(docPath);
134
+ if (!note)
135
+ continue;
136
+ if (filters?.tag && !note.tags.includes(filters.tag))
137
+ continue;
138
+ if (filters?.after && note.modified < filters.after)
139
+ continue;
140
+ if (filters?.before && note.modified > filters.before)
141
+ continue;
142
+ if (filters?.hasBacklinks === true && note.backlinks === 0)
143
+ continue;
144
+ if (filters?.hasBacklinks === false && note.backlinks > 0)
145
+ continue;
146
+ results.push({
147
+ path: note.path,
148
+ filename: note.filename,
149
+ snippet: highlightSnippet(note.content, queryTerms),
150
+ score,
151
+ tags: note.tags,
152
+ modified: note.modified,
153
+ backlinks: note.backlinks,
154
+ });
155
+ }
156
+ return results.sort((a, b) => b.score - a.score);
157
+ }
158
+ }
159
+ //# sourceMappingURL=search.js.map