mn-docs-mcp 0.2.0 → 0.2.1

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.
package/mcp/cli.mjs CHANGED
@@ -10,6 +10,7 @@ let mode = 'stdio';
10
10
  let port;
11
11
  let prebuild = false;
12
12
  let silent = false;
13
+ let root;
13
14
 
14
15
  for (let i = 0; i < args.length; i += 1) {
15
16
  const arg = args[i];
@@ -38,10 +39,19 @@ for (let i = 0; i < args.length; i += 1) {
38
39
  i += 1;
39
40
  continue;
40
41
  }
42
+ if (arg === '--root' && args[i + 1]) {
43
+ root = args[i + 1];
44
+ i += 1;
45
+ continue;
46
+ }
41
47
  if (arg?.startsWith('--port=')) {
42
48
  port = arg.split('=')[1];
43
49
  continue;
44
50
  }
51
+ if (arg?.startsWith('--root=')) {
52
+ root = arg.split('=')[1];
53
+ continue;
54
+ }
45
55
  if (arg === '--help' || arg === '-h') {
46
56
  process.stderr.write(
47
57
  [
@@ -51,6 +61,7 @@ for (let i = 0; i < args.length; i += 1) {
51
61
  ' --prebuild 启动后后台预构建索引',
52
62
  ' --verbose 输出日志(覆盖--silent)',
53
63
  ' --silent 关闭开屏与日志',
64
+ ' --root <path> 指定文档仓库根目录',
54
65
  ' --stdio 显式使用stdio模式',
55
66
  ].join('\n') + '\n'
56
67
  );
@@ -72,6 +83,7 @@ if (mode === 'stdio') {
72
83
  process.env.MCP_SILENT = silent ? '1' : '0';
73
84
  process.env.MN_DOCS_VERSION = version;
74
85
  process.env.MN_DOCS_MODE = 'stdio';
86
+ process.env.MN_DOCS_ROOT = root || process.env.MN_DOCS_ROOT || process.cwd();
75
87
  if (prebuild) process.env.MCP_PREBUILD = '1';
76
88
  await import('./server.mjs');
77
89
  } else {
@@ -79,6 +91,7 @@ if (mode === 'stdio') {
79
91
  process.env.MN_DOCS_VERSION = version;
80
92
  process.env.MN_DOCS_MODE = 'http';
81
93
  if (port) process.env.MN_DOCS_PORT = String(port);
94
+ process.env.MN_DOCS_ROOT = root || process.env.MN_DOCS_ROOT || process.cwd();
82
95
  if (prebuild) process.env.MCP_PREBUILD = '1';
83
96
  await import('./server-http.mjs');
84
97
  }
package/mcp/lib.mjs CHANGED
@@ -1,4 +1,5 @@
1
1
  import fs from 'node:fs/promises';
2
+ import fsSync from 'node:fs';
2
3
  import path from 'node:path';
3
4
  import { fileURLToPath } from 'node:url';
4
5
  import matter from 'gray-matter';
@@ -9,7 +10,28 @@ import dotenv from 'dotenv';
9
10
  const __filename = fileURLToPath(import.meta.url);
10
11
  const __dirname = path.dirname(__filename);
11
12
 
12
- const ROOT_DIR = path.resolve(__dirname, '../..');
13
+ const DEFAULT_ROOT = path.resolve(__dirname, '..');
14
+
15
+ function resolveRootDir() {
16
+ const envRoot = (process.env.MN_DOCS_ROOT || '').trim();
17
+ if (envRoot) return envRoot;
18
+ const cwd = process.cwd();
19
+ const docsFromCwd = path.join(cwd, 'src', 'content', 'docs');
20
+ try {
21
+ if (fsSyncExists(docsFromCwd)) return cwd;
22
+ } catch {}
23
+ return DEFAULT_ROOT;
24
+ }
25
+
26
+ function fsSyncExists(p) {
27
+ try {
28
+ return fsSync.statSync(p).isDirectory();
29
+ } catch {
30
+ return false;
31
+ }
32
+ }
33
+
34
+ const ROOT_DIR = resolveRootDir();
13
35
  const DOCS_DIR = path.join(ROOT_DIR, 'src', 'content', 'docs');
14
36
  const MCP_DIR = path.join(ROOT_DIR, '.mcp');
15
37
  const INDEX_PATH = path.join(MCP_DIR, 'index.json');
@@ -12,20 +12,33 @@ function color(text, code) {
12
12
  return `\x1b[${code}m${text}\x1b[0m`;
13
13
  }
14
14
 
15
+ function stripAnsi(text) {
16
+ return text.replace(/\x1b\[[0-9;]*m/g, '');
17
+ }
18
+
19
+ function padLine(line, width) {
20
+ const len = stripAnsi(line).length;
21
+ const padding = width - len;
22
+ return line + (padding > 0 ? ' '.repeat(padding) : '');
23
+ }
24
+
15
25
  function renderSplash() {
16
26
  if (IS_SILENT) return;
17
27
  const version = process.env.MN_DOCS_VERSION || '0.0.0';
18
28
  const mode = process.env.MN_DOCS_MODE || 'http';
19
29
  const port = process.env.MN_DOCS_PORT || String(PORT);
20
- const lines = [
21
- color('╭──────────────────────────────────────────────────────────────╮', '38;5;45'),
22
- color('│ mn-docs-mcp', '38;5;45') + color(` v${version}`, '38;5;214') + color(' (Charm风格启动) │', '38;5;45'),
23
- color('│ │', '38;5;45'),
24
- color(`│ 模式: ${mode.padEnd(12)} 端口: ${port.padEnd(6)} 状态: 已启动 │`, '38;5;39'),
25
- color('│ │', '38;5;45'),
26
- color('╰──────────────────────────────────────────────────────────────╯', '38;5;45'),
30
+ const contentLines = [
31
+ color(`mn-docs-mcp v${version}`, '38;5;45'),
32
+ color(`模式: ${mode} 端口: ${port}`, '38;5;39'),
27
33
  ];
28
- process.stdout.write(lines.join('\n') + '\n');
34
+ const maxWidth = Math.max(...contentLines.map((line) => stripAnsi(line).length)) + 4;
35
+ const top = color('╭' + '─'.repeat(maxWidth) + '╮', '38;5;45');
36
+ const bottom = color('╰' + '─'.repeat(maxWidth) + '╯', '38;5;45');
37
+ const body = contentLines.map((line) => {
38
+ const padded = padLine(line, maxWidth - 4);
39
+ return color('│', '38;5;45') + ' ' + padded + ' ' + color('│', '38;5;45');
40
+ });
41
+ process.stdout.write([top, ...body, bottom].join('\n') + '\n');
29
42
  }
30
43
 
31
44
  async function ensureIndex() {
package/mcp/server.mjs CHANGED
@@ -16,19 +16,32 @@ function color(text, code) {
16
16
  return `\x1b[${code}m${text}\x1b[0m`;
17
17
  }
18
18
 
19
+ function stripAnsi(text) {
20
+ return text.replace(/\x1b\[[0-9;]*m/g, '');
21
+ }
22
+
23
+ function padLine(line, width) {
24
+ const len = stripAnsi(line).length;
25
+ const padding = width - len;
26
+ return line + (padding > 0 ? ' '.repeat(padding) : '');
27
+ }
28
+
19
29
  function renderSplash() {
20
30
  if (IS_SILENT) return;
21
31
  const version = process.env.MN_DOCS_VERSION || '0.0.0';
22
32
  const mode = process.env.MN_DOCS_MODE || 'stdio';
23
- const lines = [
24
- color('╭──────────────────────────────────────────────────────────────╮', '38;5;45'),
25
- color('│ mn-docs-mcp', '38;5;45') + color(` v${version}`, '38;5;214') + color(' (Charm风格启动) │', '38;5;45'),
26
- color('│ │', '38;5;45'),
27
- color(`│ 模式: ${mode.padEnd(12)} 状态: 已启动 │`, '38;5;39'),
28
- color('│ │', '38;5;45'),
29
- color('╰──────────────────────────────────────────────────────────────╯', '38;5;45'),
33
+ const contentLines = [
34
+ color(`mn-docs-mcp v${version}`, '38;5;45'),
35
+ color(`模式: ${mode}`, '38;5;39'),
30
36
  ];
31
- process.stderr.write(lines.join('\n') + '\n');
37
+ const maxWidth = Math.max(...contentLines.map((line) => stripAnsi(line).length)) + 4;
38
+ const top = color('╭' + '─'.repeat(maxWidth) + '╮', '38;5;45');
39
+ const bottom = color('╰' + '─'.repeat(maxWidth) + '╯', '38;5;45');
40
+ const body = contentLines.map((line) => {
41
+ const padded = padLine(line, maxWidth - 4);
42
+ return color('│', '38;5;45') + ' ' + padded + ' ' + color('│', '38;5;45');
43
+ });
44
+ process.stderr.write([top, ...body, bottom].join('\n') + '\n');
32
45
  }
33
46
 
34
47
  async function ensureIndex() {
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "mn-docs-mcp",
3
3
  "type": "module",
4
- "version": "0.2.0",
4
+ "version": "0.2.1",
5
5
  "repository": {
6
6
  "type": "git",
7
7
  "url": "https://github.com/Temsys-Shen/marginnote-addon-docs.git"