Skip to content

Fix dev apps form: union types, textarea support, JSON parsing#3597

Merged
jlowin merged 7 commits intomainfrom
fix/dev-apps-textarea-and-unions
Mar 23, 2026
Merged

Fix dev apps form: union types, textarea support, JSON parsing#3597
jlowin merged 7 commits intomainfrom
fix/dev-apps-textarea-and-unions

Conversation

@jlowin
Copy link
Copy Markdown
Member

@jlowin jlowin commented Mar 23, 2026

The fastmcp dev apps picker form had several issues that made it hard to work with tools that accept multiline strings, union-typed parameters, or complex arguments.

Union types fell back to plain string inputs. Parameters like data: str | dict[str, Any] | None produce an anyOf schema with no top-level type. _model_from_schema was doing prop.get("type", "string") which silently defaulted everything to str. Now it resolves anyOf/oneOf schemas by picking the most specific non-null type — object/array types get textarea rendering for JSON editing, and string is preferred over scalars so int | str unions still get a usable text input.

The json_schema_extra textarea check was broken. Pydantic merges json_schema_extra flat into the property schema, so {"ui": {"type": "textarea"}} appears at prop["ui"], not nested under prop["json_schema_extra"].

Form values were always strings. api_launch now tries to parse string values that look like JSON objects/arrays so dict/list arguments arrive as their intended types.

This PR also introduces fastmcp.types.Textarea — a type alias for annotating tool parameters that should render as multiline inputs:

from fastmcp.types import Textarea

@mcp.tool()
async def execute_ui(code: Textarea) -> str:
    ...

Picker UI improvements:

  • Arguments section with "Edit as JSON" toggle for pasting raw JSON args
  • First paragraph of tool description shown inline (no wall of text)
  • "← Back" button in the log panel to return to the picker after launching

@marvin-context-protocol marvin-context-protocol Bot added bug Something isn't working. Reports of errors, unexpected behavior, or broken functionality. cli Related to FastMCP CLI commands (run, dev, install) or CLI functionality. labels Mar 23, 2026
Copy link
Copy Markdown

@chatgpt-codex-connector chatgpt-codex-connector Bot left a comment

Choose a reason for hiding this comment

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

💡 Codex Review

Here are some automated review suggestions for this pull request.

Reviewed commit: 0cd95719b6

ℹ️ About Codex in GitHub

Your team has set up Codex to review pull requests in this repo. Reviews are triggered when you

  • Open a pull request for review
  • Mark a draft as ready
  • Comment "@codex review".

If Codex has suggestions, it will comment; otherwise it will react with 👍.

Codex can also answer questions or update the PR. Try commenting "@codex address that feedback".

Comment thread src/fastmcp/cli/apps_dev.py Outdated
Comment on lines +1266 to +1270
if stripped and stripped[0] in ("{", "["):
try:
parsed = json.loads(stripped)
if isinstance(parsed, (dict, list)):
v = parsed
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P2 Badge Preserve JSON-looking strings when the schema says str

/api/launch now parses every value that starts with { or [ into a dict/list without consulting the tool's inputSchema. That breaks any dev-app tool that intentionally accepts raw JSON text as a string (for example, passthrough request bodies, templates, or a str | dict union using the string arm), because entering {"k":1} now changes the argument type before the tool is called.

Useful? React with 👍 / 👎.

Comment on lines +893 to +895
if candidate in types:
json_type = candidate
break
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P2 Badge Avoid collapsing multi-branch unions to one widget type

When an anyOf/oneOf schema has several non-null branches, this loop picks a single preferred type and throws the rest away. _model_from_schema() then gives Form.from_model() a one-type field, so unions like int | str | None or bool | str | None now render as number/checkbox controls and the string branch is no longer enterable from fastmcp dev apps.

Useful? React with 👍 / 👎.

Copy link
Copy Markdown

@chatgpt-codex-connector chatgpt-codex-connector Bot left a comment

Choose a reason for hiding this comment

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

💡 Codex Review

Here are some automated review suggestions for this pull request.

Reviewed commit: 1d20fc5221

ℹ️ About Codex in GitHub

Your team has set up Codex to review pull requests in this repo. Reviews are triggered when you

  • Open a pull request for review
  • Mark a draft as ready
  • Comment "@codex review".

If Codex has suggestions, it will comment; otherwise it will react with 👍.

Codex can also answer questions or update the PR. Try commenting "@codex address that feedback".

Comment thread src/fastmcp/cli/apps_dev.py Outdated
Comment on lines +1325 to +1327
raw_json_args = data.pop("_json_args", None)
if raw_json_args is not None:
tool_args = json.loads(raw_json_args) if raw_json_args.strip() else {}
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P2 Badge Stop reserving _json_args as an implicit tool parameter

Any UI tool that legitimately exposes a parameter named _json_args now breaks in form mode. api_launch() always pops that key and switches into the JSON-tab path, so the real argument is stripped and the rest of the submitted form is ignored. FastMCP already supports leading-underscore tool parameters elsewhere, so this introduces a new reserved-name regression for existing tools.

Useful? React with 👍 / 👎.

Comment thread src/fastmcp/cli/apps_dev.py Outdated
Comment on lines +1326 to +1327
if raw_json_args is not None:
tool_args = json.loads(raw_json_args) if raw_json_args.strip() else {}
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P2 Badge Validate JSON-tab payloads before building the launch URL

The new JSON tab assumes _json_args is a JSON object and lets json.loads() failures propagate. A malformed payload like {"x":,} now turns /api/launch into a 500, and valid non-object JSON such as [] or "x" is forwarded into client.callTool(..., arguments: toolArgs) even though MCP tool arguments are object-shaped. Because this tab is the escape hatch for complex inputs, normal editing mistakes now fail as server errors instead of a user-facing validation error.

Useful? React with 👍 / 👎.

Copy link
Copy Markdown

@chatgpt-codex-connector chatgpt-codex-connector Bot left a comment

Choose a reason for hiding this comment

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

💡 Codex Review

Here are some automated review suggestions for this pull request.

Reviewed commit: e363dc2ab0

ℹ️ About Codex in GitHub

Your team has set up Codex to review pull requests in this repo. Reviews are triggered when you

  • Open a pull request for review
  • Mark a draft as ready
  • Comment "@codex review".

If Codex has suggestions, it will comment; otherwise it will react with 👍.

Codex can also answer questions or update the PR. Try commenting "@codex address that feedback".

Comment on lines +886 to +889
types = [t.get("type") for t in non_null if "type" in t]
# Prefer object/array (need textarea for JSON editing),
# then string (most versatile text input), then scalars.
for candidate in (
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P2 Badge Preserve textarea metadata on optional Textarea params

When a tool argument is declared as Textarea | None = None, FastMCP emits an anyOf where the string branch carries "format": "textarea". This branch-flattening keeps only the branch type, and the later textarea check only inspects prop["format"], so _model_from_schema() rebuilds the field as a plain str without the multiline UI hint. In fastmcp dev apps, the new fastmcp.types.Textarea helper therefore stops working for the common optional/defaulted case and falls back to a single-line input.

Useful? React with 👍 / 👎.

@jlowin jlowin merged commit d23f553 into main Mar 23, 2026
7 checks passed
@jlowin jlowin deleted the fix/dev-apps-textarea-and-unions branch March 23, 2026 19:53
Copy link
Copy Markdown

@chatgpt-codex-connector chatgpt-codex-connector Bot left a comment

Choose a reason for hiding this comment

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

💡 Codex Review

Here are some automated review suggestions for this pull request.

Reviewed commit: 106b08e654

ℹ️ About Codex in GitHub

Your team has set up Codex to review pull requests in this repo. Reviews are triggered when you

  • Open a pull request for review
  • Mark a draft as ready
  • Comment "@codex review".

If Codex has suggestions, it will comment; otherwise it will react with 👍.

Codex can also answer questions or update the PR. Try commenting "@codex address that feedback".

Comment on lines +1347 to +1353
if stripped and stripped[0] in ("{", "["):
try:
parsed = json.loads(stripped)
if isinstance(parsed, (dict, list)):
v = parsed
except (json.JSONDecodeError, TypeError):
pass
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P2 Badge Reject malformed JSON in form-mode object/array fields

For parameters whose schema resolves to object or array, _model_from_schema() now intentionally renders them as textareas and relies on api_launch() to turn the submitted string back into a structured value. In form mode, though, this block silently swallows json.loads() failures and leaves the raw string in place, so inputs like {foo: 1} still launch the app with a str where the tool schema requires a dict/list. That makes the new form support for complex arguments fail at invocation time instead of surfacing a validation error to the user.

Useful? React with 👍 / 👎.

jlowin added a commit that referenced this pull request Mar 30, 2026
Co-authored-by: Claude Opus 4.6 <[email protected]>
Co-authored-by: Jeremiah Lowin <[email protected]>
Co-authored-by: Marvin Context Protocol <41898282+Marvin Context [email protected]>
Co-authored-by: voidborne-d <[email protected]>
Co-authored-by: marvin-context-protocol[bot] <225465937+marvin-context-protocol[bot]@users.noreply.github.com>
Co-authored-by: Claude <[email protected]>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: d 🔹 <[email protected]>
Co-authored-by: Jeremiah Lowin <[email protected]>
Co-authored-by: nightcityblade <[email protected]>
Co-authored-by: Claude Opus 4.6 (1M context) <[email protected]>
Co-authored-by: Claude Sonnet 4.6 <[email protected]>
Co-authored-by: Bill Easton <[email protected]>
Co-authored-by: Sumanshu Nankana <[email protected]>
Co-authored-by: Eric Robinson <[email protected]>
Co-authored-by: Martim Santos <[email protected]>
Co-authored-by: d 🔹 <[email protected]>
Co-authored-by: Matthieu B <[email protected]>
Co-authored-by: Sascha Buehrle <[email protected]>
Co-authored-by: Hakancan <[email protected]>
Co-authored-by: nightcityblade <[email protected]>
Co-authored-by: Matt Hallowell <[email protected]>
Co-authored-by: nate nowack <[email protected]>
Co-authored-by: Bill Easton <[email protected]>
Co-authored-by: Marcus Shu <[email protected]>
Co-authored-by: Rushabh Doshi <[email protected]>
Co-authored-by: AIKAWA Shigechika <[email protected]>
Co-authored-by: Jeremy Simon <[email protected]>
Co-authored-by: Miguel Miranda Dias <[email protected]>
Co-authored-by: Anthony James Padavano <[email protected]>
Co-authored-by: Mostafa Kamal <[email protected]>
Fix auto-close MRE script posting comment without closing (#3386)
Fix WorkOS token scope verification bypass 🤖 Generated with Codex (#3407)
Fix initialize McpError fallthrough 🤖 Generated with Codex (#3413)
Fix transform arg collisions with passthrough params (#3431)
Fix get_* returning None when latest version is disabled (#3439)
Fix get_* returning None when latest version is disabled (#3421)
Fix server lifespan overlap teardown (#3415)
Fix $ref output schema object detection regression (#3420)
resolved annotations (#3429)
Fix async partial callables rejected by iscoroutinefunction (#3438)
Fix async partial callables rejected by iscoroutinefunction (#3423)
fix: add version to components (#3458)
fix: use intent-based flag for OIDC scope patch in load_access_token (#3465)
Fixes #3461
fix: normalize Google scope shorthands and surface valid_scopes (#3477)
fix: resolve ty 0.0.23 type-checking errors and bump pin (#3481)
fix: shield lifespan teardown from cancellation (#3480)
fix: forward custom_route endpoints from mounted servers (#3462)
fix updates _get_additional_http_routes() to traverse providers,
Fixes #3457
fix: remove hardcoded version from CLI help text (#3456)
fix: monty 0.0.8 compatibility, drop external_functions from constructor (#3468)
fix: task test teardown hanging 5s per test (#3499)
Closes #3498
fix: validate workspace path is a directory before cursor install (#3440)
Fixes #3426
fix: handle re.error from malformed URI templates in build_regex (#3501)
fix: reject empty/OIDC-only required_scopes in AzureProvider (#3503)
fix: restrict $ref resolution to local refs only (SSRF/LFI) (#3502)
fix warnings and timeouts (#3504)
close upgrade check issue when build passes (#3505)
Closes #3484
fix: URL-encode path params to prevent SSRF/path traversal (GHSA-vv7q-7jx5-f767) (#3507)
fix: prevent path traversal in skill download (#3493)
fix: prefer IdP-granted scopes over client-requested scopes in OAuthProxy (#3492)
fix: remove unrelated transform and http.py changes from PR scope
fix: remove forced follow_redirects from httpx_client_factory calls (#3496)
fix: stop passing follow_redirects to httpx_client_factory
fix: restore follow_redirects=True for custom httpx client factories
Closes #3509
fix: CSRF double-submit cookie check in consent flow (#3519)
fix: validate server names in install commands (#3522)
fix: use raw strings for regex in pytest.raises match (#3523)
fix: reject refresh tokens used as Bearer access tokens (#3524)
fix: route ResourcesAsTools/PromptsAsTools through server middleware (#3495)
fix: resolve Pyright "Module is not callable" on @tool, @resource, @prompt decorators (#3540)
fix: filter warnings by message in KEY_PREFIX test (#3549)
fix: suppress output schema for ToolResult subclass annotations (#3548)
fix: increase sleep duration in proxy cache tests (#3567)
fix: store absolute token expiry to prevent stale expires_in on reload (#3572)
fix: preserve tool properties named 'title' during schema compression (#3582)
Fix loopback redirect URI port matching per RFC 8252 §7.3 (#3589)
Fix app tool routing: visibility check and middleware propagation (#3591)
Fix query parameter serialization to respect OpenAPI explode/style settings (#3595)
Fix dev apps form: union types, textarea support, JSON parsing (#3597)
fix(google): replace deprecated /oauth2/v1/tokeninfo with /oauth2/v3/userinfo (#3603)
fix: resolve EntraOBOToken dependency injection through MultiAuth (#3609)
fix(docs): correct misleading stateless_http header (#3622)
fix: filesystem provider import machinery (#3626)
Closes #3625 (issues 2, 3, 6)
fix: recover StdioTransport after subprocess exits (#3630)
fix(server): preserve mounted tool task metadata (#3632)
fix: scope deprecation warning filter to FastMCPDeprecationWarning (#3649)
fix imports, add PrefabAppConfig (#3650)
fix: resolve CurrentFastMCP/ctx.fastmcp to child server in mounted background tasks (#3651)
Fix blocking docs issues: chart imports, Select API, Rx consistency (#3652)
closed by default (#3657)
Fix prompt caching middleware missing wrap/unwrap round-trip (#3666)
fix: serialize object query params per OpenAPI style/explode rules (#3662)
Fixes #2857
fix: HTTP request headers not accessible in background task workers (#3631)
fix: restore HTTP headers in worker execution path for background tasks (#3681)
fix: strip discriminator after dereferencing schemas (#3682)
fix: remove stale ty:ignore directives for ty 0.0.26 (#3684)
Fix docs gaps in app provider pages (#3690)
fix: dev apps log panel UX improvements (#3698)
fix dev server empty string args (#3700)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

bug Something isn't working. Reports of errors, unexpected behavior, or broken functionality. cli Related to FastMCP CLI commands (run, dev, install) or CLI functionality.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

1 participant