2026-05-07 07:02:05 -07:00
|
|
|
"""Schema-shape tests for the built-in memory tool.
|
|
|
|
|
|
|
|
|
|
The memory tool previously used ``allOf: [{if: ..., then: {required: ...}}]``
|
|
|
|
|
at the top level of ``parameters`` to hint per-action required fields. That
|
|
|
|
|
form was:
|
|
|
|
|
|
|
|
|
|
1. Ignored by every provider (Chat Completions doesn't honour ``if/then``
|
|
|
|
|
on function schemas), so it never actually enforced anything.
|
|
|
|
|
2. **Rejected outright by strict backends** — OpenAI's Codex endpoint
|
|
|
|
|
(``chatgpt.com/backend-api/codex``, gpt-5.x) returns
|
|
|
|
|
``Invalid schema for function 'memory': schema must have type 'object'
|
|
|
|
|
and not have 'oneOf'/'anyOf'/'allOf'/'enum'/'not' at the top level``.
|
2026-05-03 18:38:23 +08:00
|
|
|
|
2026-05-07 07:02:05 -07:00
|
|
|
We now rely on the runtime handler (``memory_tool()`` in ``tools/memory_tool.py``)
|
|
|
|
|
to validate required fields per action and return actionable error messages.
|
|
|
|
|
These tests guard the schema against regressing back to a shape strict
|
|
|
|
|
backends reject.
|
|
|
|
|
"""
|
2026-05-03 18:38:23 +08:00
|
|
|
|
2026-05-07 07:02:05 -07:00
|
|
|
import json
|
|
|
|
|
|
|
|
|
|
from tools.memory_tool import MEMORY_SCHEMA
|
2026-05-03 18:38:23 +08:00
|
|
|
|
|
|
|
|
|
2026-05-07 07:02:05 -07:00
|
|
|
_FORBIDDEN_TOP_LEVEL_KEYS = ("allOf", "anyOf", "oneOf", "enum", "not")
|
2026-05-03 18:38:23 +08:00
|
|
|
|
|
|
|
|
|
2026-05-07 07:02:05 -07:00
|
|
|
def test_memory_schema_has_no_forbidden_top_level_combinators():
|
|
|
|
|
"""OpenAI's Codex backend rejects these at the top level of parameters."""
|
|
|
|
|
params = MEMORY_SCHEMA["parameters"]
|
|
|
|
|
for key in _FORBIDDEN_TOP_LEVEL_KEYS:
|
|
|
|
|
assert key not in params, (
|
|
|
|
|
f"top-level {key!r} in memory tool parameters will break the "
|
|
|
|
|
"Codex backend (chatgpt.com/backend-api/codex). Per-action "
|
|
|
|
|
"required-field checks belong in the runtime handler, not the schema."
|
|
|
|
|
)
|
2026-05-03 18:38:23 +08:00
|
|
|
|
|
|
|
|
|
2026-05-07 07:02:05 -07:00
|
|
|
def test_memory_schema_is_well_formed():
|
|
|
|
|
params = MEMORY_SCHEMA["parameters"]
|
|
|
|
|
assert params["type"] == "object"
|
|
|
|
|
assert params["required"] == ["action", "target"]
|
|
|
|
|
# Nested ``enum`` on property values is fine — only top-level is forbidden.
|
|
|
|
|
assert params["properties"]["action"]["enum"] == ["add", "replace", "remove"]
|
|
|
|
|
assert params["properties"]["target"]["enum"] == ["memory", "user"]
|
2026-05-03 18:38:23 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_memory_schema_is_json_serializable():
|
|
|
|
|
json.dumps(MEMORY_SCHEMA)
|