Shared scanner, schema extraction, and output toolkit for apcore framework adapters
Project description
apcore-toolkit-python
Python implementation of the apcore-toolkit.
Extracts ~1,400 lines of duplicated framework-agnostic logic from django-apcore and flask-apcore into a standalone Python package.
Installation
pip install apcore-toolkit
Core Modules
| Module | Description |
|---|---|
ScannedModule |
Canonical dataclass representing a scanned endpoint |
BaseScanner |
Abstract base class for framework scanners with filtering and deduplication |
YAMLWriter |
Generates .binding.yaml files for apcore.BindingLoader |
BindingLoader |
Parses .binding.yaml files back into ScannedModule objects (pure-data inverse of YAMLWriter, with loose/strict modes) |
BindingLoadError |
Exception raised when binding parsing fails; carries file_path, module_id, missing_fields, reason |
PythonWriter |
Generates @module-decorated Python wrapper files |
RegistryWriter |
Registers modules directly into an apcore.Registry |
HTTPProxyRegistryWriter |
Registers HTTP proxy modules that forward requests to a running API |
Enhancer |
Pluggable protocol for metadata enhancement |
AIEnhancer |
SLM-based metadata enhancement for scanned modules |
WriteResult |
Structured result type for all writer operations |
WriteError |
Error class for I/O failures during write |
Verifier |
Pluggable protocol for validating written artifacts |
VerifyResult |
Result type for verification operations |
YAMLVerifier |
Verifies YAML files parse correctly with required fields |
SyntaxVerifier |
Verifies source files are non-empty and readable |
RegistryVerifier |
Verifies modules are registered and retrievable |
MagicBytesVerifier |
Verifies file headers match expected magic bytes |
JSONVerifier |
Verifies JSON files parse correctly |
to_markdown |
Converts arbitrary dicts to Markdown with depth control and table heuristics |
flatten_pydantic_params |
Converts Pydantic model parameters to flat kwargs |
resolve_target |
Resolves "module.path:function_name" to callable |
enrich_schema_descriptions |
Merges descriptions into JSON Schema properties |
get_writer |
Factory function for writer instances |
DisplayResolver |
Sparse binding.yaml display overlay — resolves surface-facing alias, description, guidance, tags into metadata["display"] (§5.13) |
ConventionScanner |
Scans a commands/ directory of plain Python files for public functions and converts them to ScannedModule instances with schema inferred from type annotations (§5.14) |
extract_input_schema |
Merges OpenAPI query, path, and request body params into a single JSON Schema |
extract_output_schema |
Extracts response schema from OpenAPI operation objects |
resolve_ref |
Resolves a single internal $ref JSON pointer |
resolve_schema |
Resolves a top-level $ref in a schema |
deep_resolve_refs |
Recursively resolves all $ref pointers, depth-limited to 16 levels |
annotations_to_dict |
Converts ModuleAnnotations to a plain dict |
module_to_dict |
Converts a ScannedModule to a dict for JSON/YAML serialization |
modules_to_dicts |
Batch version of module_to_dict |
run_verifier_chain |
Runs multiple verifiers in sequence, stopping on first failure |
Usage
Scanning and Writing
from apcore_toolkit import BaseScanner, ScannedModule, YAMLWriter
class MyScanner(BaseScanner):
def scan(self, **kwargs):
# Scan your framework endpoints and return ScannedModule instances
return [
ScannedModule(
module_id="users.get_user",
description="Get a user by ID",
input_schema={"type": "object", "properties": {"id": {"type": "integer"}}, "required": ["id"]},
output_schema={"type": "object", "properties": {"name": {"type": "string"}}},
tags=["users"],
target="myapp.views:get_user",
)
]
def get_source_name(self):
return "my-framework"
scanner = MyScanner()
modules = scanner.scan()
# Filter and deduplicate
modules = scanner.filter_modules(modules, include=r"^users\.")
modules = scanner.deduplicate_ids(modules)
# Write YAML binding files
writer = YAMLWriter()
writer.write(modules, output_dir="./bindings")
Direct Registry Registration
from apcore import Registry
from apcore_toolkit import RegistryWriter
registry = Registry()
writer = RegistryWriter()
writer.write(modules, registry)
Output Format Factory
from apcore_toolkit.output import get_writer
writer = get_writer("yaml") # YAMLWriter
writer = get_writer("python") # PythonWriter
writer = get_writer("registry") # RegistryWriter
Pydantic Model Flattening
from apcore_toolkit import flatten_pydantic_params, resolve_target
# Resolve a target string to a callable
func = resolve_target("myapp.views:create_task")
# Flatten Pydantic model params into scalar kwargs for MCP tools
wrapped = flatten_pydantic_params(func)
OpenAPI Schema Extraction
from apcore_toolkit.openapi import extract_input_schema, extract_output_schema
input_schema = extract_input_schema(operation, openapi_doc)
output_schema = extract_output_schema(operation, openapi_doc)
Schema Enrichment
from apcore_toolkit import enrich_schema_descriptions
enriched = enrich_schema_descriptions(schema, {"user_id": "The user ID"})
Markdown Formatting
from apcore_toolkit import to_markdown
md = to_markdown({"name": "Alice", "role": "admin"}, title="User Info")
Display Overlay (§5.13)
DisplayResolver applies a sparse binding.yaml display overlay to a list of ScannedModule instances, populating metadata["display"] with surface-facing presentation fields (alias, description, guidance, tags) for CLI, MCP, and A2A surfaces.
from apcore_toolkit.display import DisplayResolver
resolver = DisplayResolver()
# Apply overlay from a directory of *.binding.yaml files
modules = resolver.resolve(scanned_modules, binding_path="bindings/")
# Or from a pre-parsed dict
modules = resolver.resolve(
scanned_modules,
binding_data={
"bindings": [
{
"module_id": "product.get",
"display": {
"alias": "product-get",
"description": "Get a product by ID",
"cli": {"alias": "get-product"},
"mcp": {"alias": "get_product"},
},
}
]
},
)
# Resolved fields are in metadata["display"]
mod = modules[0]
print(mod.metadata["display"]["cli"]["alias"]) # "get-product"
print(mod.metadata["display"]["mcp"]["alias"]) # "get_product"
print(mod.metadata["display"]["a2a"]["alias"]) # "product-get"
Resolution chain (per field): surface-specific override > display default > binding-level field > scanner value.
MCP alias constraints: automatically sanitized (non-[a-zA-Z0-9_-] chars replaced with _; leading digit prefixed with _); raises ValueError if result exceeds 64 characters.
CLI alias validation: warns and falls back to display.alias when a user-explicitly-set alias does not match ^[a-z][a-z0-9_-]*$.
Convention Module Discovery (§5.14)
ConventionScanner scans a directory of plain Python files for public functions and converts them to ScannedModule instances. No decorators, no base classes, no imports from apcore -- just functions with type hints.
from apcore_toolkit import ConventionScanner
scanner = ConventionScanner()
modules = scanner.scan(commands_dir="commands/")
# Each public function becomes a module:
# commands/deploy.py -> deploy.deploy
# commands/deploy.py -> deploy.rollback (if rollback() exists)
Module-level constants customize behavior:
# commands/deploy.py
MODULE_PREFIX = "ops" # override file-based prefix -> ops.deploy
CLI_GROUP = "operations" # group hint for CLI surface
TAGS = ["infra", "deploy"] # tags stored in metadata
def deploy(env: str, tag: str = "latest") -> dict:
"""Deploy the app to the given environment."""
return {"status": "deployed", "env": env}
Input and output schemas are inferred from PEP 484 type annotations. Use include / exclude regex filters to control which module IDs are registered.
Documentation
Full documentation is available at https://github.com/aiperceivable/apcore-toolkit.
License
Apache-2.0
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file apcore_toolkit-0.6.0.tar.gz.
File metadata
- Download URL: apcore_toolkit-0.6.0.tar.gz
- Upload date:
- Size: 100.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.10
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a109a4c284f5dffb69108196bb0590d95773b1885929cbbba834191e7e4194f7
|
|
| MD5 |
1475b91c8caf9a01f662e5ac7ed103bc
|
|
| BLAKE2b-256 |
f297cb03a9a4d3e0e710064ce0039099392c2a80a8f313eea08a7c56a19646fd
|
File details
Details for the file apcore_toolkit-0.6.0-py3-none-any.whl.
File metadata
- Download URL: apcore_toolkit-0.6.0-py3-none-any.whl
- Upload date:
- Size: 66.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.10
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6886ac8404b95516d715b5f4fbeadf889d57b460715b81641b357f4cbeb8bbfc
|
|
| MD5 |
7bad8d2135c8c900a0020f3cfb0e5436
|
|
| BLAKE2b-256 |
ca061f4417681a71dd8cbb9e39eaf8ee8668c100ba756a5b4d39272a7d6be178
|