marionette_logger 0.4.0 copy "marionette_logger: ^0.4.0" to clipboard
marionette_logger: ^0.4.0 copied to clipboard

Flutter extensions for AI agent interaction via MCP — lets Claude, Copilot, and Cursor tap, scroll, type, and screenshot your app. Log collector integration with logger package.

marionette_mcp

Marionette MCP #

License marionette_mcp pub.dev badge

"Playwright MCP/Cursor Browser, but for Flutter apps"

Marionette MCP enables AI agents (Claude Code, Copilot, Cursor, Gemini CLI, and more) to inspect and interact with running Flutter applications. It connects your AI agent directly to a running app via the Model Context Protocol (MCP), so it can see the widget tree, tap elements, enter text, scroll, and capture screenshots for automated AI-driven smoke testing and interaction.

Marionette MCP keeps the surface area intentionally small. It exposes only a handful of high-signal actions and returns the minimum actionable data, which helps keep prompts focused and context sizes under control.

Marionette MCP vs Flutter MCP #

The official Dart & Flutter MCP server focuses on development-time tasks: searching pub.dev, managing dependencies, analyzing code, and inspecting runtime errors. It can also drive the UI, but it does so through Flutter Driver, which introduces extra instrumentation in your app. Marionette MCP focuses solely (and in an opinionated way) on runtime interaction: tapping buttons, entering text, scrolling, and taking screenshots, while requiring minimal changes to your app. Use Flutter MCP to build your app, use Marionette MCP to test and interact with it with minimal code changes.

Quick Start #

Note: Your Flutter app must be prepared to be compatible with this MCP.

  1. Prepare your Flutter app - Add the marionette_flutter package and initialize MarionetteBinding in your main.dart.
  2. Install the MCP server - Add marionette_mcp to your projects dev_dependencies.
  3. Configure your AI tool - Add the MCP server command (dart run marionette_mcp) to your tool's configuration (Claude Code, Copilot, Cursor, Gemini CLI).
  4. Run your app in debug mode - Look for the VM service URI in the console (e.g., ws://127.0.0.1:12345/ws).
  5. Connect and interact - Ask your AI agent (Claude, Copilot, or any MCP-compatible assistant) to connect to your app using the URI and start interacting.

Installation #

1. Add MCP Server Package #

Run the following command to activate the marionette_mcp global tool:

dart pub global activate marionette_mcp

Note

You can also install the package as a dev-dependency using

dart pub add dev:marionette_mcp

Then invoke the MCP server as dart run marionette_mcp. It might be necessary to change the working directory, so that dart run is able to find marionette_mcp. You can do it like so: cd ${workspaceFolder}/packages/mypackage && dart run marionette_mcp (it will vary between tooling).

If it does not work, we suggest using the global tool method.

2. Add Flutter Package #

Run the following command in your Flutter app directory:

flutter pub add marionette_flutter

Flutter App Integration #

You need to initialize the MarionetteBinding in your app. This binding registers the necessary VM service extensions that the MCP server communicates with.

Basic Setup #

If your app uses standard Flutter widgets (like ElevatedButton, TextField, Text, etc.), the default configuration works out of the box.

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:marionette_flutter/marionette_flutter.dart';

void main() {
  // Initialize Marionette only in debug mode
  if (kDebugMode) {
    MarionetteBinding.ensureInitialized();
  } else {
    WidgetsFlutterBinding.ensureInitialized();
  }

  runApp(const MyApp());
}

Important

MarionetteBinding must be the only binding initialized in the process. Flutter allows only one WidgetsBinding per app. If another binding (e.g. AutomatedTestWidgetsFlutterBinding from flutter test, or IntegrationTestWidgetsFlutterBinding) is already initialized, calling MarionetteBinding.ensureInitialized() will throw a binding assertion error.

This commonly happens when your test calls main() and kDebugMode is true during tests. You can work around it in several ways:

Option A – Check the FLUTTER_TEST environment variable

import 'dart:io' show Platform;

final bool isFlutterTest = Platform.environment.containsKey('FLUTTER_TEST');
if (kDebugMode && !isFlutterTest) {
MarionetteBinding.ensureInitialized();
} else {
WidgetsFlutterBinding.ensureInitialized();
}

Option B – Use a separate entrypoint for tests

Keep MarionetteBinding in your production main() (lib/main.dart) and create a different entrypoint for tests (e.g. lib/main_test.dart or test/app_test.dart) that does not initialize MarionetteBinding.

Log Collection (get_logs) #

Marionette supports flexible log collection through the LogCollector interface. You can choose from several options depending on your logging setup:

Option 1: Using the logging package

If your app uses Dart's logging package:

flutter pub add marionette_logging
import 'package:flutter/foundation.dart';
import 'package:logging/logging.dart';
import 'package:marionette_flutter/marionette_flutter.dart';
import 'package:marionette_logging/marionette_logging.dart';

void main() {
  if (kDebugMode) {
    MarionetteBinding.ensureInitialized(
      MarionetteConfiguration(logCollector: LoggingLogCollector()),
    );
  } else {
    WidgetsFlutterBinding.ensureInitialized();
  }

  Logger.root.level = Level.ALL;
  runApp(const MyApp());
}

Option 2: Using the logger package

If your app uses the logger package:

flutter pub add marionette_logger
import 'package:flutter/foundation.dart';
import 'package:logger/logger.dart';
import 'package:marionette_flutter/marionette_flutter.dart';
import 'package:marionette_logger/marionette_logger.dart';

void main() {
  final logCollector = LoggerLogCollector();

  if (kDebugMode) {
    MarionetteBinding.ensureInitialized(
      MarionetteConfiguration(logCollector: logCollector),
    );
  } else {
    WidgetsFlutterBinding.ensureInitialized();
  }

  final logger = Logger(
    output: MultiOutput([ConsoleOutput(), logCollector]),
  );

  runApp(const MyApp());
}

Option 3: Custom logging with PrintLogCollector

For other logging solutions or custom setups, use PrintLogCollector:

import 'package:flutter/foundation.dart';
import 'package:marionette_flutter/marionette_flutter.dart';

void main() {
  final collector = PrintLogCollector();

  if (kDebugMode) {
    MarionetteBinding.ensureInitialized(
      MarionetteConfiguration(logCollector: collector),
    );
  } else {
    WidgetsFlutterBinding.ensureInitialized();
  }

  // Hook into your logging system
  myLogger.onLog((message) => collector.addLog(message));

  runApp(const MyApp());
}

No logging

If you don't need log collection, simply omit the logCollector parameter. The get_logs tool will return a helpful message explaining how to enable it.

Custom Design System #

If you use custom widgets in your design system, you can configure Marionette to recognize them as interactive elements or extract text from them.

Why isInteractiveWidget? A typical Flutter screen has hundreds of widgets in its tree - Padding, Container, Column, SizedBox, etc. When the AI agent calls get_interactive_elements, Marionette filters this down to only actionable targets: buttons, text fields, switches, sliders, etc. This gives the agent a concise, manageable list instead of an overwhelming dump of layout widgets.

By default, Marionette recognizes standard Flutter widgets like ElevatedButton, TextField, and Switch. If your app uses custom widgets (e.g., MyPrimaryButton that wraps styling around a GestureDetector), Marionette won't know they're tappable unless you tell it. The isInteractiveWidget callback lets you mark your custom widget types as interactive, so they appear in the element list and can be targeted by tap and other tools.

Why extractText? The extractText callback serves two purposes:

  1. Element discovery: Widgets with extractable text are automatically included in the interactive elements tree returned by get_interactive_elements, even if they are not explicitly interactive. The extracted text appears in the element's text field, helping the AI agent understand what each element displays.

  2. Text-based matching: The tap, scroll_to, and other interaction tools can match elements by their text content using the text parameter (e.g., tap(text: "Submit")).

By default, Marionette extracts text from standard Flutter widgets (Text, RichText, EditableText, TextField, TextFormField). Use extractText to add support for your custom text widgets.

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:marionette_flutter/marionette_flutter.dart';
import 'package:my_app/design_system/buttons.dart';
import 'package:my_app/design_system/inputs.dart';

void main() {
  if (kDebugMode) {
    MarionetteBinding.ensureInitialized(
      MarionetteConfiguration(
        // Identify your custom interactive widgets
        isInteractiveWidget: (type) =>
            type == MyPrimaryButton ||
            type == MyTextField ||
            type == MyCheckbox,

        // Extract text from your custom widgets
        extractText: (widget) {
          if (widget is MyText) return widget.data;
          if (widget is MyTextField) return widget.controller?.text;
          return null;
        },
      ),
    );
  } else {
    WidgetsFlutterBinding.ensureInitialized();
  }

  runApp(const MyApp());
}

Screenshot sizing

By default, Marionette will downscale screenshots to fit within 2000×2000 physical pixels. You can override this via maxScreenshotSize in MarionetteConfiguration (set it to null to disable resizing).

Tool Configuration #

Add the MCP server to your AI coding assistant's configuration.

Cursor #

Install MCP Server

Or manually add to your project's .cursor/mcp.json or your global ~/.cursor/mcp.json:

{
  "mcpServers": {
    "marionette": {
      "command": "marionette_mcp",
      "args": []
    }
  }
}

Google Antigravity #

Open the MCP store, click “Manage MCP Servers”, then “View raw config” and add to the opened mcp_config.json:

{
  "mcpServers": {
    "marionette": {
      "command": "marionette_mcp",
      "args": []
    }
  }
}

Gemini CLI #

Add to your ~/.gemini/settings.json:

{
  "mcpServers": {
    "marionette": {
      "command": "marionette_mcp",
      "args": []
    }
  }
}

Claude Code #

You can run the following command to add it:

claude mcp add --transport stdio marionette -- marionette_mcp

Copilot #

Add to your mcp.json:

{
  "servers": {
    "marionette": {
      "command": "marionette_mcp",
      "args": []
    }
  }
}

Available Tools #

Once connected, the AI agent has access to these tools:

Tool Description
connect Connect to a Flutter app via its VM service URI (e.g., ws://127.0.0.1:54321/ws).
disconnect Disconnect from the currently connected app.
get_interactive_elements Returns a list of all interactive UI elements (buttons, inputs, etc.) visible on screen.
tap Taps an element matching a specific key or visible text.
enter_text Enters text into a text field matching a key.
scroll_to Scrolls the view until an element matching a key or text becomes visible.
get_logs Retrieves application logs collected since app start or the last hot reload (requires a LogCollector to be configured).
take_screenshots Captures screenshots of all active views and returns them as base64 images.
hot_reload Performs a hot reload of the Flutter app, applying code changes without losing state.

Example Scenarios #

Marionette MCP shines when used by coding agents to verify their work or explore the app. Here are some real-world scenarios:

1. Verify a New Feature #

Context: You just asked the agent to implement a "Forgot Password" flow. Prompt:

"Now that you've implemented the Forgot Password screen, let's verify it. Connect to the app, navigate to the login screen, tap 'Forgot Password', enter a valid email, and submit. Check the logs to ensure the API call was made successfully."

2. Post-Refactor Smoke Test #

Context: You performed a large refactor on the navigation logic. Prompt:

"I've refactored the routing. Please run a quick smoke test: connect to the app, cycle through all tabs in the bottom navigation bar, and verify that each screen loads without throwing exceptions in the logs."

3. Debugging UI Issues #

Context: Users reported a button is unresponsive on the Settings page. Prompt:

"Investigate the 'Clear Cache' button on the Settings page. Connect to the app, navigate there, find the button using get_interactive_elements, tap it, and analyze the logs to see if an error is occurring or if the tap is being ignored."

How It Works #

  1. Initialization: Your Flutter app initializes MarionetteBinding, which registers custom VM service extensions (ext.flutter.marionette.*).
  2. Connection: The MCP server connects to your app's VM Service URL.
  3. Interaction: When an AI agent calls a tool (like tap), the MCP server translates this into a call to the corresponding VM service extension in your app.
  4. Execution: The Flutter app executes the action (e.g., simulates a tap gesture) and returns the result.

Assumptions & Limitations #

  • Prefer pasting the VM Service URI manually: While some tooling can sometimes discover or infer the VM Service endpoint, the most reliable workflow is to copy the ws://.../ws URI from your flutter run output (or DevTools link) and paste it to the agent when calling connect.

  • The agent may not know your app: Marionette can “see” the widget tree and interact with UI elements, but it doesn’t automatically understand your product’s flows, naming conventions, or edge cases. If you want reliable navigation and assertions, provide extra context in the prompt (what screen to reach, expected labels/keys, preconditions, and the goal of the interaction).

  • “Your mileage may vary” interactions: Some actions are implemented via best-effort simulation of user behavior (gestures, focus, text entry, scrolling). Depending on platform, custom widgets, overlays, or app-specific gesture handling, results may vary. If a flow is flaky, consider exposing clearer widget keys, simplifying hit targets, or adding custom MarionetteConfiguration hooks for your design system. And if you hit something that consistently doesn’t behave as expected, a small repro in an issue helps us improve it.

Marionette CLI #

Why a CLI? #

The MCP server works great with tools that support MCP natively (Cursor, Claude Code, etc.), but many enterprise environments have restrictions on which AI models can be used and which protocols are allowed. Not every team can run an MCP server.

The CLI bridges this gap. Any AI agent that can execute shell commands — even smaller, less capable models — can drive a Flutter app through Marionette if given a clear reference document. A well-structured .md file describing each command's syntax, expected outputs, and exit codes is often all a constrained agent needs to work autonomously. This makes the CLI the most portable and universally compatible way to integrate Marionette into AI workflows.

Installation #

Install the CLI globally from pub.dev:

dart pub global activate marionette_cli

This adds the marionette executable to your PATH (ensure ~/.pub-cache/bin is on your PATH).

Teaching AI Agents to Use the CLI #

For an AI agent to use the CLI effectively, it needs a reference describing every command, its arguments, expected outputs, and exit codes. The help-ai command prints exactly that — a comprehensive, machine-readable reference designed for AI consumption:

marionette help-ai

Have the agent run this once at the start of a session, capture the output, and use it as a guide for all subsequent interactions. You can also pipe it to a file and include it in your project as a Cursor rule, Agent Skill (SKILL.md) or system prompt:

marionette help-ai > .cursor/rules/marionette-cli.md

Direct URI Mode (Stateless) #

Pass the VM service URI directly with --uri — no registration, no cleanup, no files on disk:

marionette --uri ws://127.0.0.1:8181/ws get-interactive-elements
marionette --uri ws://127.0.0.1:8181/ws tap --key submit_button
marionette --uri ws://127.0.0.1:8181/ws take-screenshots --output ./screenshot.png

--uri and --instance are mutually exclusive. Use --uri for one-off interactions and --instance when targeting the same app repeatedly.

Named Instance Mode (Stateful) #

For repeated interactions with the same app, register it as a named instance to avoid passing the URI every time:

# Register Flutter app instances (use the VM service URI from flutter run output)
marionette register my-app ws://127.0.0.1:8181/ws
marionette register other-app ws://127.0.0.1:9090/ws

# Interact with a specific instance
marionette -i my-app get-interactive-elements
marionette -i my-app tap --key submit_button
marionette -i my-app tap --text "Submit"
marionette -i my-app enter-text --key email_field --input "test@example.com"
marionette -i my-app scroll-to --text "Bottom Item"
marionette -i my-app take-screenshots --output ./screenshot.png
marionette -i my-app get-logs
marionette -i my-app hot-reload

# Instance management
marionette list
marionette unregister my-app
marionette doctor              # Check connectivity of all instances

Troubleshooting #

  • "Not connected to any app": Ensure the AI agent has called connect with the valid VM Service URI before using other tools.
  • Finding the URI: Run your Flutter app in debug mode (flutter run). Look for a line like: The Flutter DevTools debugger and profiler on iPhone 15 Pro is available at: http://127.0.0.1:9101?uri=ws://127.0.0.1:9101/ws. Use the ws://... part.
  • Release Mode: Marionette only works in debug (and profile) mode because it relies on the VM Service. It will not work in release builds.
  • Elements not found: Ensure your widgets are visible. If using custom widgets, make sure they are configured in MarionetteConfiguration.

🛠️ Maintained by LeanCode #

This package is built with 💙 by LeanCode. We are top-tier experts focused on Flutter Enterprise solutions.

Why LeanCode? #

  • Creators of Patrol – the next-gen testing framework for Flutter.

  • Production-Ready – We use this package in apps with millions of users.

  • Full-Cycle Product Development – We take your product from scratch to long-term maintenance.


Need help with your Flutter project?

👉 Hire our team   •   Check our other packages

0
likes
140
points
--
downloads

Documentation

API reference

Publisher

verified publisherleancode.co

Flutter extensions for AI agent interaction via MCP — lets Claude, Copilot, and Cursor tap, scroll, type, and screenshot your app. Log collector integration with logger package.

Repository (GitHub)
View/report issues

Topics

#mcp #ai #claude #agent #copilot

License

Apache-2.0 (license)

Dependencies

flutter, logger, marionette_flutter

More

Packages that depend on marionette_logger