OpenAPI 3.1 tool server for iFlow Search (心流搜索) — drop-in for Open WebUI, Coze, and other platforms that consume OpenAPI tool catalogues.
Project description
iflow-search-openapi
OpenAPI 3.1 tool server for iFlow Search (心流搜索) — exposes iflow_web_search, iflow_image_search, and iflow_web_fetch over plain HTTP so platforms that consume OpenAPI tool catalogues (Open WebUI, Coze, …) can wire them in directly.
- Core SDK:
iflow-search— this package wraps it; do not skip installing it (it is a transitive dependency, sopip installhandles it). - Sibling adapters:
iflow-search-mcp(MCP stdio server),iflow-search-langchain(LangChain tools). - API docs: https://platform.iflow.cn/docs/
- Status: alpha pre-release (
0.1.0a0). Requirespip install --pre.
Install
pip install --pre iflow-search-openapi
--pre is required while the version is still a PEP 440 prerelease.
Run
export IFLOW_API_KEY="your-iflow-api-key"
iflow-search-openapi
Output (stderr):
[iflow-search-openapi] v0.1.0a0 listening on http://127.0.0.1:8787 — bearer auth DISABLED (open mode)
By default the server binds 127.0.0.1:8787 — local-only. Set IFLOW_OPENAPI_HOST=0.0.0.0 to expose it to a LAN or a container network. See Configuration for the full env list.
Endpoints
| Method | Path | Description |
|---|---|---|
GET |
/health |
Liveness probe — {"ok": true, "version": "..."} |
GET |
/openapi.json |
OpenAPI 3.1 schema (auto-generated from Pydantic models) |
POST |
/tools/iflow_web_search |
{"query": "...", "count": 3} |
POST |
/tools/iflow_image_search |
{"query": "...", "count": 3} |
POST |
/tools/iflow_web_fetch |
{"url": "https://example.com"} |
Success envelope
{
"ok": true,
"data": {
"query": "latest LLM benchmarks",
"results": [
{"title": "...", "url": "https://...", "snippet": "...", "position": 1}
],
"took_ms": 142
}
}
Field names are snake_case. The core SDK's raw (upstream envelope) is excluded by default — it bloats LLM context for fields the model cannot act on.
Error envelope
{
"ok": false,
"error": {
"code": "business_rate_limited",
"message": "Rate limit exceeded.",
"status_code": 429
}
}
code is a stable string; switch on it (not on HTTP status alone) for retry/backoff decisions. The full code table is in docs/design/python-openapi-design.md §8.3.
Use with Open WebUI
- Run
iflow-search-openapisomewhere Open WebUI can reach (http://your-host:8787). - In Open WebUI → Workspace → Tools → Add Tool (the "OpenAPI Servers" entry), paste
http://your-host:8787/openapi.json. - If you set
IFLOW_OPENAPI_AUTH_TOKEN, Open WebUI will prompt for it when importing. - Open WebUI imports the three tools; assign them to a model/conversation as usual.
If Open WebUI runs in a browser on a different origin from this server, set IFLOW_OPENAPI_CORS_ORIGIN=https://your-open-webui-host.
Use with Coze
- Run
iflow-search-openapisomewhere Coze can reach. - Coze → Plugins → Create plugin → Import from OpenAPI → URL
http://your-host:8787/openapi.json. - Provide the bearer token (
IFLOW_OPENAPI_AUTH_TOKEN) when prompted. - The three tools appear as plugin actions; attach to a bot as usual.
Authentication
Two distinct credentials:
| Credential | Direction | Source |
|---|---|---|
IFLOW_API_KEY |
this server → iFlow API | env, required |
IFLOW_OPENAPI_AUTH_TOKEN |
external client → this server | env, optional |
When IFLOW_OPENAPI_AUTH_TOKEN is unset, the server is open: any caller that can reach the socket can invoke the tools. Intended for local dev or behind a private network / reverse proxy.
When set, all routes except /health require Authorization: Bearer <token>. The compare is constant-time. /openapi.json is also gated so the schema (which advertises the server as an iFlow proxy) stays behind the bearer.
IFLOW_API_KEY is never echoed in any response, log line, OpenAPI schema, or startup banner — only its presence is implied by the server having started successfully.
Configuration
All configuration is via environment variables. No .env loader. No CLI flags. No config file.
| Variable | Required | Default | Notes |
|---|---|---|---|
IFLOW_API_KEY |
yes | — | iFlow API key. Read once at startup, never echoed. |
IFLOW_BASE_URL |
no | core default | Override iFlow API base URL (staging/proxy). |
IFLOW_TIMEOUT_MS |
no | 30000 |
Per-request timeout in milliseconds. |
IFLOW_OPENAPI_HOST |
no | 127.0.0.1 |
Bind address. Use 0.0.0.0 for LAN/container exposure. |
IFLOW_OPENAPI_PORT |
no | 8787 |
Bind port. |
IFLOW_OPENAPI_AUTH_TOKEN |
no | unset (open mode) | Bearer token required from external callers when set. |
IFLOW_OPENAPI_CORS_ORIGIN |
no | unset (no CORS) | * or an exact origin (https://host[:port]). |
IFLOW_OPENAPI_CLIENT |
no | unset | Free-form host name (e.g. open-webui). Banner only — not on the wire. |
Heroku / App Engine / fly.io — bridging PORT
This package uses IFLOW_OPENAPI_PORT instead of the generic PORT so a platform's auto-injected port can't silently take over the bind. If your platform requires PORT, bridge it in your start command:
IFLOW_OPENAPI_PORT="$PORT" IFLOW_OPENAPI_HOST=0.0.0.0 iflow-search-openapi
CORS
IFLOW_OPENAPI_CORS_ORIGIN:
- unset — no CORS headers; server is intended for same-origin or server-to-server use.
*— wildcard allow-origin.http(s)://host[:port]— exact-origin allow.
Anything with a path, query, fragment, or non-printable character is rejected at startup with ConfigError and exit code 1 — header-injection guard.
Preflight (OPTIONS) short-circuits to 204 before the bearer check, so browser-side tool importers (Open WebUI) can complete preflight without a token.
Access-Control-Allow-Headers is Content-Type, Authorization, X-Session-Id. X-Session-Id is allowed for Open WebUI compatibility; the adapter does not read it.
Attribution
The adapter does not construct any Authorization, IFlow-*, or User-Agent header itself — that's the core SDK's job. Outbound requests to iFlow carry:
| Header | Value |
|---|---|
Authorization |
Bearer <IFLOW_API_KEY> |
IFlow-Source |
openapi |
IFlow-Integration |
iflow-search-openapi |
IFlow-Integration-Version |
this package's __version__ |
User-Agent |
iflow-search-openapi/<version> |
IFLOW_OPENAPI_CLIENT is not forwarded as a wire header (that namespace belongs to MCP transports). It appears in the startup banner only.
What's not included in v0.1.0a0
- TLS termination — put a reverse proxy in front.
- Streaming / SSE / WebSocket — iFlow Search is request/response.
- Per-platform packages (no
iflow-search-open-webui, noiflow-search-coze). - A public Python embedding API (
build_app(...)is underscore-prefixed and not part of the contract). .envfile loading, a CLI flag for the API key, or any config file.- Per-request access logs — operators put a reverse proxy in front for those.
Public Python surface
from iflow_search_openapi import __version__
That is the only supported import in MVP. Internal modules (_app, _routes, _auth, etc.) are underscore-prefixed; they may be reorganised without notice. The stable contract is the HTTP surface, not the Python surface.
Local development
From packages/iflow-search-openapi/:
python -m pip install -e ".[dev]"
python -m pytest -q
python -m ruff check .
python -m mypy src/iflow_search_openapi
python -m build
Real-API smoke
cd packages/iflow-search-openapi
export IFLOW_API_KEY="your-api-key"
export IFLOW_OPENAPI_SMOKE=1
python scripts/smoke_real_api.py
The script:
- Is opt-in — without
IFLOW_OPENAPI_SMOKE=1it refuses to call the live API. - Reads
IFLOW_API_KEYfrom the environment only — never from disk. - Redacts the key in all log output.
- Does not write any file.
License
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 iflow_search_openapi-0.1.0a0.tar.gz.
File metadata
- Download URL: iflow_search_openapi-0.1.0a0.tar.gz
- Upload date:
- Size: 17.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
fc80ee3820ac3e31a39d2fde7638b9b1855cb2ebb23f81b62660595a00a8e197
|
|
| MD5 |
665e59921e83d1287b86f5a35eff4d5a
|
|
| BLAKE2b-256 |
65bd4f0697df305a0a64ea60d680f868e467535de8ab2d0db12ff0770bbd91f0
|
File details
Details for the file iflow_search_openapi-0.1.0a0-py3-none-any.whl.
File metadata
- Download URL: iflow_search_openapi-0.1.0a0-py3-none-any.whl
- Upload date:
- Size: 23.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4db210813c7e4f506ed9d94c5a22b220c663058304163260c9920bd4e7ed4d67
|
|
| MD5 |
1605e96b0784bb9d32b19521fa993e9b
|
|
| BLAKE2b-256 |
676daae3f540d21914687d8998a46ab5a47904db40dbb7d599737e278a45417f
|