Refactor: upgrade to latest MCP and docx-rs; add Router, fonts CLI, and builder-based DOCX edits

- Integrate mcp-server Router with mcp-spec and expose tools
- Add fonts subcommands (download/verify) with pinned sources + checksums
- Replace deprecated docx-rs APIs; rebuild DOCX via ops (paragraphs/headings/tables/lists/page breaks/headers/footers)
- Implement proper numbered lists via docx-rs numbering
- Gate advanced features behind `advanced-docx` for future porting
- Resolve lopdf and image import ambiguities; adapt search and responses
This commit is contained in:
Andy
2025-08-11 19:19:04 +08:00
parent f75a47fe76
commit ad8909d749
7 changed files with 247 additions and 173 deletions
+56 -3
View File
@@ -13,7 +13,7 @@ mod docx_handler;
mod converter;
#[cfg(feature = "runtime-server")]
mod pure_converter;
#[cfg(feature = "runtime-server")]
#[cfg(all(feature = "runtime-server", feature = "advanced-docx"))]
mod advanced_docx;
mod security;
@@ -55,11 +55,64 @@ async fn main() -> Result<()> {
#[cfg(feature = "runtime-server")]
{
use mcp_server::{Router, Server};
use mcp_server::router::RouterService;
use mcp_server::router::CapabilitiesBuilder;
use mcp_spec::{prompt::Prompt, resource::Resource};
use mcp_spec::protocol::ServerCapabilities;
use mcp_spec::content::Content;
use mcp_spec::tool::Tool as SpecTool;
use serde_json::Value as JsonValue;
use std::pin::Pin;
use std::future::Future;
use tokio::io::{stdin, stdout};
let security_config = security::SecurityConfig::from_args(args);
info!("Starting DOCX MCP Server - Security: {}", security_config.get_summary());
// TODO: Integrate with mcp-server Router here. For now, just exit successfully.
info!("Server integration pending refactor; exiting.");
#[derive(Clone)]
struct DocxRouter(docx_tools::DocxToolsProvider);
impl Router for DocxRouter {
fn name(&self) -> String { "docx-mcp-server".to_string() }
fn instructions(&self) -> String { "DOCX tools for reading and exporting".to_string() }
fn capabilities(&self) -> ServerCapabilities {
CapabilitiesBuilder::new().with_tools(true).build()
}
fn list_tools(&self) -> Vec<SpecTool> {
// DocxToolsProvider::list_tools is async; block briefly with tokio runtime handle
let rt = tokio::runtime::Handle::current();
let tools = rt.block_on(self.0.list_tools());
tools.into_iter().map(|t| SpecTool{ name: t.name, description: t.description.unwrap_or_default(), input_schema: t.input_schema }).collect()
}
fn call_tool(&self, tool_name: &str, arguments: JsonValue) -> Pin<Box<dyn Future<Output = Result<Vec<Content>, mcp_spec::handler::ToolError>> + Send + 'static>> {
let provider = self.0.clone();
let name = tool_name.to_string();
Box::pin(async move {
let resp = provider.call_tool(&name, arguments).await;
// Convert our CallToolResponse (text JSON) to Content::text
let text = match resp.content.get(0) {
Some(mcp_core::types::ToolResponseContent::Text(t)) => t.text.clone(),
_ => serde_json::to_string(&resp).unwrap_or_else(|_| "{}".to_string()),
};
Ok(vec![Content::text(text)])
})
}
fn list_resources(&self) -> Vec<Resource> { vec![] }
fn read_resource(&self, _uri: &str) -> Pin<Box<dyn Future<Output = Result<String, mcp_spec::handler::ResourceError>> + Send + 'static>> {
Box::pin(async { Ok(String::new()) })
}
fn list_prompts(&self) -> Vec<Prompt> { vec![] }
fn get_prompt(&self, _prompt_name: &str) -> Pin<Box<dyn Future<Output = Result<String, mcp_spec::handler::PromptError>> + Send + 'static>> {
Box::pin(async { Ok(String::new()) })
}
}
let router = DocxRouter(DocxToolsProvider::new_with_security(security_config));
let service = RouterService(router);
let server = Server::new(service);
let transport = mcp_server::ByteTransport::new(stdin(), stdout());
server.run(transport).await?;
}
#[cfg(not(feature = "runtime-server"))]