pulseengine-mcp-macros 0.13.0

Procedural macros for PulseEngine MCP Framework - simplified server and tool development
Documentation
# PulseEngine MCP Macros

Procedural macros for the PulseEngine MCP Framework that dramatically simplify server and tool development.

## Quick Start

Add to your `Cargo.toml`:

```toml
[dependencies]
# Core macros
pulseengine-mcp-macros = "0.7.1"

# Required dependencies for generated code
pulseengine-mcp-protocol = "0.7.1"
pulseengine-mcp-server = "0.7.1"
pulseengine-mcp-transport = "0.7.1"
async-trait = "0.1"
thiserror = "1.0"
tokio = { version = "1.0", features = ["full"] }

# For STDIO transport (recommended)
tracing = "0.1"
tracing-subscriber = "0.3"
```

## Basic Server Example

```rust
use pulseengine_mcp_macros::{mcp_server, mcp_tools};

#[mcp_server(name = "Hello Server")]
#[derive(Default, Clone)]
pub struct HelloServer;

#[mcp_tools]
impl HelloServer {
    /// Say hello to someone
    pub async fn hello(&self, name: Option<String>) -> anyhow::Result<String> {
        Ok(format!("Hello, {}!", name.unwrap_or_else(|| "World".to_string())))
    }
}

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    // CRITICAL for STDIO: Configure logging to stderr
    HelloServer::configure_stdio_logging();

    let server = HelloServer::with_defaults().serve_stdio().await?;
    server.run().await?;
    Ok(())
}
```

## Required Dependencies

### Always Required

These dependencies are **always required** when using `#[mcp_server]`:

```toml
pulseengine-mcp-protocol = "0.7.1"
pulseengine-mcp-server = "0.7.1"
pulseengine-mcp-transport = "0.7.1"
async-trait = "0.1"
thiserror = "1.0"
tokio = { version = "1.0", features = ["full"] }
```

### Feature-Based Dependencies

#### STDIO Logging (`stdio-logging` feature)

For STDIO transport compatibility:

```toml
pulseengine-mcp-macros = { version = "0.7.1", features = ["stdio-logging"] }
tracing-subscriber = "0.3"
```

#### Authentication (`auth` feature + auth parameter)

**Important**: Authentication is **opt-in**. By default, servers have no authentication.

When using `auth = "memory"`, `auth = "file"`, etc.:

```toml
pulseengine-mcp-macros = { version = "0.7.1", features = ["auth"] }
pulseengine-mcp-auth = "0.7.1"
```

## Macros

### `#[mcp_server]`

Generates a complete MCP server from a struct:

```rust
#[mcp_server(
    name = "My Server",
    version = "1.0.0",
    description = "Server description",
    auth = "memory"  // Optional: "memory", "file", "disabled"
)]
#[derive(Default, Clone)]
pub struct MyServer;
```

**Generated methods:**

- `MyServer::with_defaults()` - Create instance
- `MyServer::serve_stdio()` - Start STDIO transport
- `MyServer::serve_http(port)` - Start HTTP transport
- `MyServer::configure_stdio_logging()` - Fix STDIO logging

### `#[mcp_tools]`

Automatically discovers public methods as MCP tools:

```rust
#[mcp_tools]
impl MyServer {
    /// Tool description from doc comment
    pub async fn my_tool(&self, param: String) -> anyhow::Result<String> {
        Ok(format!("Result: {}", param))
    }
}
```

**Note**: Currently passthrough - full implementation coming soon.

## Transport Types

### STDIO (Default)

For MCP clients like Claude Desktop:

```rust
// CRITICAL: Configure logging first!
MyServer::configure_stdio_logging();
let server = MyServer::with_defaults().serve_stdio().await?;
```

### HTTP

For web-based clients:

```rust
let server = MyServer::with_defaults().serve_http(8080).await?;
```

### WebSocket

For real-time clients:

```rust
let server = MyServer::with_defaults().serve_websocket(8080).await?;
```

## Authentication

**By default, servers have no authentication**. To enable authentication, add the `auth` parameter:

```rust
// Memory-based (development)
#[mcp_server(name = "Dev Server", auth = "memory")]

// File-based (production)
#[mcp_server(name = "Prod Server", auth = "file", app_name = "my-app")]

// Explicitly disabled (same as default)
#[mcp_server(name = "Test Server", auth = "disabled")]

// No auth parameter = no authentication (default)
#[mcp_server(name = "Simple Server")]
```

**Requires**: `pulseengine-mcp-auth` dependency and `auth` feature only when `auth` parameter is specified.

## STDIO Transport Critical Note

⚠️ **CRITICAL**: For STDIO transport, you **MUST** configure logging to go to stderr, not stdout. Stdout is reserved for JSON-RPC messages only.

```rust
// Always call this before serve_stdio()
MyServer::configure_stdio_logging();
```

Failure to do this will break MCP client compatibility with errors like:

```
Error from MCP server: SyntaxError: Unexpected token '2025-0"... is not valid JSON
```

## Complete Example

See [`examples/hello-world-stdio-fixed/`](../examples/hello-world-stdio-fixed/) for a working example with proper STDIO logging configuration.

## License

Licensed under either of

- Apache License, Version 2.0, ([LICENSE-APACHE]../LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
- MIT license ([LICENSE-MIT]../LICENSE-MIT or http://opensource.org/licenses/MIT)

at your option.