mcp_stdio_proxy/client/support/
args.rs1use clap::Parser;
6use std::path::PathBuf;
7
8#[derive(Parser, Debug, Clone)]
12pub struct LoggingArgs {
13 #[arg(
15 long,
16 default_value = "false",
17 help = "启用详细诊断模式,追踪连接生命周期和超时问题(默认关闭)"
18 )]
19 pub diagnostic: bool,
20
21 #[arg(long, help = "日志输出目录,将自动生成日志文件名")]
23 pub log_dir: Option<PathBuf>,
24
25 #[arg(long, conflicts_with = "log_dir", help = "日志文件完整路径")]
27 pub log_file: Option<PathBuf>,
28
29 #[arg(
34 long,
35 env = "OTEL_EXPORTER_OTLP_ENDPOINT",
36 help = "OTLP 追踪端点 (如 http://localhost:4317)"
37 )]
38 pub otlp_endpoint: Option<String>,
39
40 #[arg(
42 long,
43 default_value = "mcp-proxy",
44 help = "追踪服务名称(用于 Jaeger 等追踪后端标识)"
45 )]
46 pub service_name: String,
47}
48
49#[derive(Parser, Debug)]
51#[command(name = "mcp-proxy")]
52#[command(version = env!("CARGO_PKG_VERSION"))]
53#[command(about = "MCP 协议转换代理工具", long_about = None)]
54pub struct Cli {
55 #[command(subcommand)]
56 pub command: Option<Commands>,
57
58 #[arg(value_name = "URL", help = "MCP 服务的 URL 地址(直接模式)")]
60 pub url: Option<String>,
61
62 #[arg(short, long, global = true)]
64 pub verbose: bool,
65
66 #[arg(short, long, global = true)]
68 pub quiet: bool,
69}
70
71#[derive(clap::Subcommand, Debug)]
72pub enum Commands {
73 Convert(ConvertArgs),
75
76 Check(CheckArgs),
78
79 Detect(DetectArgs),
81
82 Proxy(crate::client::proxy_server::ProxyArgs),
84
85 Health(HealthArgs),
87}
88
89#[derive(Parser, Debug, Clone)]
91pub struct ConvertArgs {
92 #[arg(value_name = "URL", help = "MCP 服务的 URL 地址")]
94 pub url: Option<String>,
95
96 #[arg(long, conflicts_with = "config_file", help = "MCP 服务配置 JSON")]
98 pub config: Option<String>,
99
100 #[arg(long, conflicts_with = "config", help = "MCP 服务配置文件路径")]
102 pub config_file: Option<PathBuf>,
103
104 #[arg(short, long, help = "MCP 服务名称(多服务配置时必需)")]
106 pub name: Option<String>,
107
108 #[arg(long, value_enum, help = "指定远程服务协议类型(不指定则自动检测)")]
110 pub protocol: Option<crate::client::proxy_server::ProxyProtocol>,
111
112 #[arg(short, long, help = "认证 header")]
114 pub auth: Option<String>,
115
116 #[arg(short = 'H', long, value_parser = parse_key_val, help = "自定义 HTTP headers (KEY=VALUE 格式)")]
118 pub header: Vec<(String, String)>,
119
120 #[arg(long, default_value = "0", help = "重试次数,0 表示无限重试")]
122 pub retries: u32,
123
124 #[arg(
126 long,
127 value_delimiter = ',',
128 help = "工具白名单(逗号分隔),只允许指定的工具"
129 )]
130 pub allow_tools: Option<Vec<String>>,
131
132 #[arg(
134 long,
135 value_delimiter = ',',
136 help = "工具黑名单(逗号分隔),排除指定的工具"
137 )]
138 pub deny_tools: Option<Vec<String>>,
139
140 #[arg(
142 long,
143 default_value = "30",
144 help = "客户端 ping 间隔(秒),0 表示禁用"
145 )]
146 pub ping_interval: u64,
147
148 #[arg(
150 long,
151 default_value = "10",
152 help = "客户端 ping 超时(秒),超时则认为连接断开"
153 )]
154 pub ping_timeout: u64,
155
156 #[command(flatten)]
158 pub logging: LoggingArgs,
159}
160
161#[derive(Parser, Debug)]
163pub struct CheckArgs {
164 #[arg(value_name = "URL")]
166 pub url: String,
167
168 #[arg(short, long)]
170 pub auth: Option<String>,
171
172 #[arg(long, default_value = "10")]
174 pub timeout: u64,
175}
176
177#[derive(Parser, Debug)]
179pub struct DetectArgs {
180 #[arg(value_name = "URL")]
182 pub url: String,
183
184 #[arg(short, long)]
186 pub auth: Option<String>,
187}
188
189#[derive(Parser, Debug)]
191#[command(after_help = "\
192退出码:
193 0 服务健康 - MCP 连接握手成功
194 1 服务不健康 - 连接失败、超时或握手失败
195
196示例:
197 # 基本用法
198 mcp-proxy health http://localhost:8080/mcp
199
200 # 带认证
201 mcp-proxy health http://localhost:8080/mcp -a \"Bearer token123\"
202
203 # 指定协议和超时
204 mcp-proxy health http://localhost:8080/mcp --protocol sse --timeout 5
205
206 # 静默模式(仅返回退出码,适合脚本使用)
207 mcp-proxy health http://localhost:8080/mcp -q
208
209 # 在 shell 脚本中使用
210 if mcp-proxy health http://localhost:8080/mcp -q; then
211 echo \"MCP 服务正常\"
212 else
213 echo \"MCP 服务不可用\"
214 fi
215")]
216pub struct HealthArgs {
217 #[arg(value_name = "URL")]
219 pub url: String,
220
221 #[arg(short, long, help = "认证 header")]
223 pub auth: Option<String>,
224
225 #[arg(short = 'H', long, value_parser = parse_key_val, help = "自定义 HTTP headers (KEY=VALUE 格式)")]
227 pub header: Vec<(String, String)>,
228
229 #[arg(long, default_value = "10")]
231 pub timeout: u64,
232
233 #[arg(long, value_enum, help = "指定远程服务协议类型(不指定则自动检测)")]
235 pub protocol: Option<crate::client::proxy_server::ProxyProtocol>,
236}
237
238pub fn parse_key_val(s: &str) -> Result<(String, String), String> {
240 let pos = s
241 .find('=')
242 .ok_or_else(|| format!("无效的 KEY=VALUE 格式: {}", s))?;
243 Ok((s[..pos].to_string(), s[pos + 1..].to_string()))
244}