Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for running a local MCP server via the dev CLI command #1785

Merged
merged 12 commits into from
Mar 12, 2025

Conversation

myftija
Copy link
Contributor

@myftija myftija commented Mar 11, 2025

Changes in this PR:

  • added support to run a local MCP server via the CLI.
  • added the --mcp flag to the dev command to control the behavior. By default the MCP server does not run.
  • added the --mcp-port flag to configure the MCP server. By default it runs on 3333.
  • added main components needed for the MCP server using the official typescript SDK.
  • opted in for the SSE transport type by using a simple polka server (very lightweight version of express).
  • added an API endpoint to expose task run events/logs in JSON format, similar to the download logs endpoint. This needs some more refinement maybe in a future PR, namely: pagination and a proper response schema.
  • added an MCP tools that enable:
    • triggering tasks
    • listing task runs
    • canceling runs
    • getting run logs
    • listing all tasks
  • added MCP config file for Cursor, so that the MCP server is configured out of the box if you're using this editor.
  • briefly described the MCP server feature in the CLI readme.

More flows can be supported in the future, such as task payload parsing in the MCP tools, custom prompts to support typical workflows, more actions, etc...

This PR already enables a lot of interesting agentic LLM flows, such as:

  • triggering tasks via natural language and looking up the resulting runs.
  • checking for failures across task runs.
  • using task logs and definitions to debug issues with failed runs.

...and more. Check out the screenshots below for a preview :)

Refs #1780

✅ Checklist

  • I have followed every step in the contributing guide
  • The PR title follows the convention.
  • I ran and tested the code works

Testing

Tested only manually so far.


Changelog


Screenshots

image Screenshot 2025-03-11 at 19 41 47

Summary by CodeRabbit

  • New Features

    • Added a new API endpoint and client method to retrieve run events, improving visibility into task execution.
    • Enhanced development workflows by ensuring background services are properly shut down.
    • Introduced a tool for listing available task identifiers to streamline task management.
  • Documentation

    • Expanded the CLI documentation with a detailed MCP server section, including guidance on starting and configuring the server.

…LI command

This just adds the main components for the MCP server. It hooks on the
existing `dev` command and can be enabled by passing the `--mcp` flag.

Currently only the `trigger-task` tool is exposed, which enables users
trigger tasks via MCP and see the resulting run.

Still WIP :)
Copy link

changeset-bot bot commented Mar 11, 2025

⚠️ No Changeset found

Latest commit: 7d65aa9

Merging this PR will not cause a version bump for any packages. If these changes should not result in a new version, you're good to go. If these changes should result in a version bump, you need to add a changeset.

This PR includes no changesets

When changesets are added to this PR, you'll see the packages that this PR includes changesets for and the associated semver types

Click here to learn what changesets are, and how to add one.

Click here if you're a maintainer who wants to add a changeset to this PR

Copy link
Contributor

coderabbitai bot commented Mar 11, 2025

Walkthrough

This change introduces new dependencies and type definitions into the CLI package, along with modifications to the API client. Updates include the addition of a new import and comment in the CLI’s API client, removal of an extraneous comment in the development command, and an enhanced shutdown process for the MCP server. Further, the MCP server functionality is extended with task listing and event listening, a new API method and route are added for run events, and documentation plus a configuration file for MCP are incorporated.

Changes

Files Change Summary
packages/cli-v3/package.json Added new dependencies: @types/polka (dev), polka, and @modelcontextprotocol/sdk (prod).
packages/cli-v3/src/apiClient.ts, packages/cli-v3/src/commands/dev.ts In apiClient.ts: added import for ListRunResponse and a constructor comment; in dev.ts: removed a TODO comment regarding the --mcp option.
packages/cli-v3/src/dev/devSession.ts, packages/cli-v3/src/dev/mcpServer.ts In devSession.ts: invoked stopMcpServer() during session stop; in mcpServer.ts: declared allTaskIds, added a tool (list-all-tasks), and inserted an event listener for worker initialization.
packages/core/.../apiClient/index.ts, apps/webapp/.../api.v1.runs.$runId.events.ts Added API functionality: new method listRunEvents in ApiClient and a corresponding API route for retrieving run events with JSON BigInt conversion.
.cursor/mcp.json New JSON configuration file defining MCP server endpoints.
packages/cli-v3/README.md Updated documentation by adding a "MCP Server" section with usage instructions and configuration options.

Sequence Diagram(s)

sequenceDiagram
    participant DevSession
    participant MCPServer
    DevSession->>MCPServer: stopMcpServer()
    MCPServer-->>DevSession: Acknowledge shutdown
Loading
sequenceDiagram
    participant Client
    participant ApiClient
    participant Backend
    Client->>ApiClient: listRunEvents(runId)
    ApiClient->>Backend: GET /api/v1/runs/{runId}/events
    Backend-->>ApiClient: Return event data (BigInt as string)
    ApiClient-->>Client: Send run events JSON
Loading

Possibly related PRs

Suggested reviewers

  • matt-aitken

Poem

I'm just a bunny, my code hops with glee,
Dependencies added, now the CLI's fancy and free.
MCP servers now listen and gracefully stop,
API events are gathered with a gentle hop.
With a twitch of my whiskers and a joyful leap,
I celebrate these changes — a treat to keep!

Warning

There were issues while running some tools. Please review the errors and either fix the tool’s configuration or disable the tool if it’s a critical failure.

🔧 ESLint

If the error stems from missing dependencies, add them to the package.json file. For unrecoverable errors (e.g., due to private dependencies), disable the tool in the CodeRabbit configuration.

Scope: all 2 workspace projects
 ERR_PNPM_OPTIONAL_DEPS_REQUIRE_PROD_DEPS  Optional dependencies cannot be installed without production dependencies

✨ Finishing Touches
  • 📝 Generate Docstrings

Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

@myftija myftija marked this pull request as ready for review March 11, 2025 19:01
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🔭 Outside diff range comments (1)
packages/cli-v3/src/dev/mcpServer.ts (1)

136-226: ⚠️ Potential issue

Code duplication in the file.

There appear to be duplicate code blocks in the file:

  1. The app = polka() declaration appears twice (lines 136 and 223)
  2. The /sse route handler is duplicated
  3. There are fragments of code outside proper function definitions (lines 140-152)

This looks like a merge conflict or incomplete refactoring. The duplicate code should be removed to avoid confusion and potential runtime errors.

-const app = polka();
-app.get("/sse", (_req, res) => {
-  mcpTransport = new SSEServerTransport("/messages", res);
-  },
-  async ({ runId }) => {
-    const result = await sdkApiClient.retrieveRun(runId);
-
-    return {
-      content: [
-        {
-          type: "text",
-          text: JSON.stringify(result, null, 2),
-        },
-      ],
-    };
-  }
-);

Keep only the complete implementation starting at line 223.

🧹 Nitpick comments (5)
packages/cli-v3/README.md (1)

22-52: Comprehensive documentation of the MCP server feature.

The documentation clearly explains what the MCP server is, how to use it, and its integration with the Trigger.dev CLI. This will help users understand and utilize this new feature.

There are two minor grammatical issues to fix:

-The Trigger.dev CLI can expose an MCP server and enable you interact with Trigger.dev in agentic LLM workflows.
+The Trigger.dev CLI can expose an MCP server and enable you to interact with Trigger.dev in agentic LLM workflows.

-By default it runs on port `3333`. You can change this by passing the `--mcp-port` flag:
+By default, it runs on port `3333`. You can change this by passing the `--mcp-port` flag:
🧰 Tools
🪛 LanguageTool

[uncategorized] ~27-~27: Possible missing preposition found.
Context: ...can expose an MCP server and enable you interact with Trigger.dev in agentic LLM workflo...

(AI_HYDRA_LEO_MISSING_TO)


[uncategorized] ~37-~37: Did you mean: “By default,”?
Context: ...ommand: bash trigger dev --mcp By default it runs on port 3333. You can change ...

(BY_DEFAULT_COMMA)

packages/cli-v3/src/apiClient.ts (1)

52-53: Consider making accessToken required as suggested.

The TODO comment recommends making the accessToken parameter required. This is a good suggestion for future improvement as most API methods require it and check for its presence.

When implementing this change in the future, you would need to:

  1. Remove the optional marker from the parameter
  2. Update all callers to ensure they provide an access token
  3. Remove the null checks before API calls since the parameter would be guaranteed
-  constructor(
-    public readonly apiURL: string,
-    // TODO: consider making this required
-    public readonly accessToken?: string
-  ) {
+  constructor(
+    public readonly apiURL: string,
+    public readonly accessToken: string
+  ) {
apps/webapp/app/routes/api.v1.runs.$runId.events.ts (3)

41-41: Consider implementing field filtering.

The TODO comment correctly identifies that returning the whole events might not be optimal. Consider:

  1. Defining a specific response schema with only the necessary fields
  2. Implementing field selection parameters (similar to GraphQL)
  3. Creating a dedicated DTO (Data Transfer Object) for the response

44-51: Good handling of BigInt serialization, but consider a utility function.

The JSON serialization with BigInt handling is implemented correctly, but this pattern might be needed in multiple places.

Consider extracting this to a reusable utility function:

// Utility function in a shared location
export function serializeWithBigInt(data: any): any {
  return JSON.parse(
    JSON.stringify(data, (_, value) =>
      typeof value === "bigint" ? value.toString() : value
    )
  );
}

// Then in your code
return json(
  {
    events: runEvents.map(event => serializeWithBigInt(event)),
  },
  { status: 200 }
);

52-54: Consider adding error handling for the response.

While the current implementation returns a 200 status for successful responses, consider adding explicit error handling for cases where event retrieval might fail.

A try/catch block around the event retrieval logic could help handle unexpected errors gracefully:

try {
  const runEvents = await eventRepository.getRunEvents(
    getTaskEventStoreTableForRun(run),
    run.friendlyId,
    run.createdAt,
    run.completedAt ?? undefined
  );
  
  // Process and return events...
} catch (error) {
  console.error("Failed to retrieve run events:", error);
  return json({ error: "Failed to retrieve run events" }, { status: 500 });
}
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 67115ca and 7d65aa9.

⛔ Files ignored due to path filters (1)
  • pnpm-lock.yaml is excluded by !**/pnpm-lock.yaml
📒 Files selected for processing (22)
  • packages/cli-v3/package.json (3 hunks)
  • packages/cli-v3/src/apiClient.ts (2 hunks)
  • packages/cli-v3/src/commands/dev.ts (2 hunks)
  • packages/cli-v3/src/dev/devSession.ts (3 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (1 hunks)
  • packages/core/src/v3/apiClient/index.ts (1 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (1 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (3 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (3 hunks)
  • packages/cli-v3/src/commands/dev.ts (2 hunks)
  • packages/cli-v3/src/dev/devSession.ts (1 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (2 hunks)
  • packages/cli-v3/src/dev/devSession.ts (2 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (1 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (1 hunks)
  • apps/webapp/app/routes/api.v1.runs.$runId.events.ts (1 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (1 hunks)
  • packages/core/src/v3/apiClient/index.ts (1 hunks)
  • packages/cli-v3/src/dev/mcpServer.ts (3 hunks)
  • .cursor/mcp.json (1 hunks)
  • packages/cli-v3/README.md (1 hunks)
  • packages/cli-v3/src/commands/dev.ts (0 hunks)
🧰 Additional context used
🪛 LanguageTool
packages/cli-v3/README.md

[uncategorized] ~27-~27: Possible missing preposition found.
Context: ...can expose an MCP server and enable you interact with Trigger.dev in agentic LLM workflo...

(AI_HYDRA_LEO_MISSING_TO)


[uncategorized] ~37-~37: Did you mean: “By default,”?
Context: ...ommand: bash trigger dev --mcp By default it runs on port 3333. You can change ...

(BY_DEFAULT_COMMA)

🔇 Additional comments (18)
.cursor/mcp.json (1)

1-7: Well-structured configuration file for the MCP server.

This configuration file correctly sets up the MCP server configuration for the Cursor editor, specifying the SSE endpoint URL for Trigger.dev at the default port 3333.

packages/core/src/v3/apiClient/index.ts (1)

407-417:

❓ Verification inconclusive

API client method added for fetching run events.

The listRunEvents method implementation follows the existing patterns in the API client, but uses a placeholder schema.

The TODO comment indicates a need for a proper schema definition for the response. Consider adding a specific type for run events in a future update:


🏁 Script executed:

#!/bin/bash
# Look for any existing run event schema definitions that could be reused
echo "Searching for potential run event schemas..."
rg -t ts "RunEvent" --glob "packages/core/src/v3/schemas/**/*.ts"

Length of output: 164


Action Required: Verify Run Event Schema Definition

The listRunEvents method implementation is consistent with our existing API client patterns, but it still uses a placeholder schema (z.any()). Our search for a dedicated RunEvent schema in the packages/core/src/v3/schemas/ directory did not return any results. Please manually verify if a dedicated schema for run events can be added or reused in the future.

  • The function follows the established API client patterns.
  • No existing reusable RunEvent schema was found, so replacing z.any() remains a future improvement.
packages/cli-v3/package.json (3)

52-52: Added type definitions for polka.

The type definition for polka is appropriately added to support TypeScript development with the lightweight server library.


79-79: Added MCP SDK dependency.

The Model Context Protocol SDK is added as a dependency to support the integration of the MCP server functionality.


119-119: Added polka server dependency.

The lightweight polka server library is added to support the SSE transport for the MCP server.

packages/cli-v3/src/dev/mcpServer.ts (3)

3-3: Updated zod import to include additional utility functions.

The import for zod has been updated to explicitly include optional and union functions that are used in the schema definitions.


64-64: Enhanced description for the list-runs tool.

The description has been improved to provide more details about what information is returned by the tool.


116-135: Added new tool for retrieving run details.

The new get-run tool follows the established pattern and provides a convenient way to fetch detailed information about a specific task run.

packages/cli-v3/src/dev/devSession.ts (2)

64-65: Good implementation of port configuration for MCP server.

The change correctly passes the configured port from command line arguments to the MCP server. This aligns well with the CLI options defined in the dev.ts file.


205-205: Good cleanup of MCP server resources.

The stopMcpServer function is properly called during the cleanup process, ensuring that the MCP server is shut down when the dev session ends.

packages/cli-v3/src/commands/dev.ts (2)

24-24: Well-defined port configuration with sensible defaults.

Using z.coerce.number() ensures type safety while allowing string inputs from the command line to be properly converted to numbers. The default value of 3333 is consistent with the documentation.


54-55: Clear CLI options for MCP server configuration.

The CLI options have clear descriptions and consistent default values. The --mcp flag enables the server, and --mcp-port allows customization of the port.

packages/cli-v3/src/apiClient.ts (1)

34-34: Added import for ListRunResponse.

The addition of this import suggests preparation for MCP server functionality to list task runs, which aligns with the PR objectives.

apps/webapp/app/routes/api.v1.runs.$runId.events.ts (5)

33-39: The resource retrieval logic looks good.

The implementation properly retrieves run events using the event repository with appropriate parameters, including handling the case where completedAt might be null.


18-22: Good use of the authorization model with proper resource definition.

The authorization configuration is well-structured with appropriate scopes and resource definitions. The inclusion of shouldRetryNotFound is a good practice for handling potential race conditions.


1-7: Imports are appropriately structured.

The imports are well-organized and include all necessary dependencies for the functionality provided by this route.


8-10: Parameter validation is correctly implemented.

The Zod schema correctly validates the runId parameter, and the comment provides useful context about what the ID represents.


17-17:

❓ Verification inconclusive

Review CORS strategy setting.

The "all" CORS strategy allows access from any origin. Consider if this level of access is necessary or if a more restrictive policy would be more appropriate for this endpoint.


🏁 Script executed:

#!/bin/bash
# Check how CORS is configured in other API routes

echo "Checking CORS configuration patterns in other API routes:"
rg "corsStrategy" --type ts "apps/webapp/app/routes" -A 1 -B 1 | head -n 20

Length of output: 1444


CORS Strategy Consistency and Security Check

The corsStrategy: "all" setting on line 17 in apps/webapp/app/routes/api.v1.runs.$runId.events.ts is consistent with similar configurations in other API routes (e.g., in apps/webapp/app/routes/api.v1.runs.ts and various realtime routes). Please verify that allowing all origins is intentional for this endpoint and aligns with your security requirements. If a more restrictive CORS policy is appropriate for handling sensitive data or operations, consider updating the setting accordingly.

runId: z.string(), // This is the run friendly ID
});

// TODO: paginate the results
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Don't forget to implement pagination.

The TODO comment correctly identifies the need for pagination. Without it, large result sets could cause performance issues or timeout errors, especially if runs can generate many events.

Consider implementing standard pagination parameters (limit/offset or cursor-based) and returning metadata about total count and next page.

@ericallam ericallam merged commit 3a04759 into triggerdotdev:main Mar 12, 2025
6 of 8 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants