11.7 C
Canberra
Thursday, April 3, 2025

Introducing SwiftMCP | Cocoanetics


I’m thrilled to announce SwiftMCP, a Swift macro-based framework that elegantly exposes your Swift features as highly effective Mannequin Context Protocol (MCP) instruments for AI assistants. After months of cautious refinement, SwiftMCP now delivers the expertise I’ve at all times dreamed of: turning customary Swift documentation immediately into AI-integrable instruments—effortlessly.

Behind the Scenes

For a very long time, I watched with envy as builders in languages like Python effortlessly created instruments for AI brokers by merely including decorators to their features, together with documentation enclosed in triple quotes. One thing related felt painfully out of attain for Swift builders—till I spotted the unimaginable potential of Swift Macros.

Macros in Swift have full entry to the syntax tree of your supply code, together with each documentation remark, parameter sort, and extra. This opens up astonishing potentialities:

  • Extracting detailed metadata immediately out of your present Swift documentation feedback.
  • Robotically producing extra supply code that captures and shops this metadata.
  • Dynamically creating operate wrappers that simplify invocation with versatile arguments.

For example, think about this easy Swift operate:

/// Provides two integers and returns their sum
/// - Parameter a: First quantity so as to add
/// - Parameter b: Second quantity so as to add
/// - Returns: The sum of a and b
@MCPTool
func add(a: Int, b: Int) -> Int {
    return a + b
}

Utilizing SwiftMCP’s @MCPTool macro, the above operate mechanically generates metadata like this:

/// autogenerated
let __mcpMetadata_add = MCPToolMetadata(
	identify: "add",
	description: "Provides two integers and returns their sum",
	parameters: [MCPToolParameterInfo(name: "a", label: "a", type: "Int", description: "First number to add", defaultValue: nil), MCPToolParameterInfo(name: "b", label: "b", type: "Int", description: "Second number to add", defaultValue: nil)],
	returnType: "Int",
	returnTypeDescription: "The sum of a and b",
	isAsync: false,
	isThrowing: false
)

Moreover, SwiftMCP generates a versatile invocation wrapper, very similar to Goal-C’s NSInvocation, enabling your features to simply accept parameters as dictionaries:

/// Autogenerated wrapper for add that takes a dictionary of parameters
func __mcpCall_add(_ params: [String: Sendable]) async throws -> (Codable & Sendable) {
    let a = attempt params.extractInt(named: "a")
    let b = attempt params.extractInt(named: "b")
    return add(a: a, b: b)
}

This wrapper intelligently validates and parses incoming parameters, mechanically dealing with conversions and offering informative error messages if something goes flawed.

The ultimate magic occurs on the server degree with @MCPServer, which features a common tool-invocation methodology:

public func callTool(_ identify: String, arguments: [String: Sendable]) async throws -> (Codable & Sendable) {
   guard let software = mcpTools.first(the place: { $0.identify == identify }) else {
      throw MCPToolError.unknownTool(identify: identify)
   }

   let enrichedArguments = attempt software.enrichArguments(arguments, forObject: self)

   change identify {
      case "add":
         return attempt await __mcpCall_add(enrichedArguments)
      default:
         throw MCPToolError.unknownTool(identify: identify)
   }
}

With this plumbing in place, SwiftMCP effortlessly helps features which are async, throwing, returning void, or returning any Codable sort. You may serve your MCP instruments both through customary IO or as an HTTP+SSE server, permitting seamless integration into numerous workflows.

The less complicated methodology of serving – through customary IO – is includes the consumer really launching your app after which speaking it’s sending single-line JSONRPC requests to stdin and receiving JSONRPC responses through stdout. If you wish to ship an error, then you need to ship that to stderr.

let transport = StdioTransport(server: calculator)
attempt await transport.run()

The opposite – extra subtle manner of serving – is through a HTTP-SSE server. Right here the consumer makes a GET request to /sse and retains the connection open. It’s knowledgeable of an endpoint to POST JSONRPC requests to. When a message is shipped to the endpoint, the reply for it (with matching id) shall be despatched through the SSE channel.

let transport = HTTPSSETransport(server: calculator, port: 8080)
attempt await transport.run()

SwiftMCP helps each strategies. On the HTTP+SSE transport it has a couple of further bells and whistles, like for instance you’ll be able to restrict entry to purchasers sending a selected bearer token.

If in case you have your native server working you’ll be able to expose it through an OpenAPI scheme to customized GPTs, in order that even these can work together together with your native instruments.

What’s Subsequent?

My speedy aim is to combine SwiftMCP with my present libraries resembling SwiftMail, enabling my brokers to work together with e-mail through IMAP and SMTP, deal with notifications, and manipulate calendar occasions through EventKit. There are numerous extra features you can thus make out there to agentic coders like Cursor.

I’ve began a mission to do exactly that. For now it simply lets me begin an MCP Server and authorize location notifications.

This exposes the operate to ship native notifications to my laptop:

/// Sends a notification with the desired title and physique
/// - Parameters:
///   - title: The title of the notification
///   - physique: The physique textual content of the notification
///   - subtitle: Optionally available subtitle for the notification
@MCPTool
func sendNotification(title: String,
                      physique: String,
                      subtitle: String? = nil
                     ) async throws

One other concept is to have some fundamental workflow issues that Cursor is lacking however Xcode supplies, like launching an app in Simulator, or a wrapper for xcode-build and swift for constructing and testing.

Conclusion

Shortly after beginning SwiftMCP, I observed the NSHipster article about iMCP showing in my feed, highlighting the broader group’s rising curiosity in MCP. Shortly thereafter, I additionally discovered one other Swift MCP implementation by Compiler-Inc. It’s thrilling to see others exploring related options!

SwiftMCP is open-source, actively maintained, and keen in your suggestions and contributions. I’m glad to listen to from you the way you propose to make use of it or to assist including lacking options.

Test it out on GitHub: SwiftMCP, Swift Bundle Index is internet hosting the documentation.


Classes: Administrative

Related Articles

LEAVE A REPLY

Please enter your comment!
Please enter your name here

[td_block_social_counter facebook="tagdiv" twitter="tagdivofficial" youtube="tagdiv" style="style8 td-social-boxed td-social-font-icons" tdc_css="eyJhbGwiOnsibWFyZ2luLWJvdHRvbSI6IjM4IiwiZGlzcGxheSI6IiJ9LCJwb3J0cmFpdCI6eyJtYXJnaW4tYm90dG9tIjoiMzAiLCJkaXNwbGF5IjoiIn0sInBvcnRyYWl0X21heF93aWR0aCI6MTAxOCwicG9ydHJhaXRfbWluX3dpZHRoIjo3Njh9" custom_title="Stay Connected" block_template_id="td_block_template_8" f_header_font_family="712" f_header_font_transform="uppercase" f_header_font_weight="500" f_header_font_size="17" border_color="#dd3333"]
- Advertisement -spot_img

Latest Articles