MCP-DB-Client
Health Pass
- License — License: MIT
- Description — Repository has a description
- Active repo — Last push 0 days ago
- Community trust — 27 GitHub stars
Code Fail
- rm -rf — Recursive force deletion command in .github/workflows/ci.yml
- rm -rf — Recursive force deletion command in .github/workflows/release.yml
- rm -rf — Recursive force deletion command in build/build-http1c-dll-debug.sh
- rm -rf — Recursive force deletion command in build/build-http1c-dll-release.sh
- rm -rf — Recursive force deletion command in build/ensure-tools.sh
- rm -rf — Recursive force deletion command in build/package-http1c-addin.sh
- rm -rf — Recursive force deletion command in build/run-tests.sh
Permissions Pass
- Permissions — No dangerous permissions requested
No AI report is available for this listing yet.
MCP layer for 1C
http1c — MCP Server Framework for 1C:Enterprise
http1c is a framework for building Model Context Protocol (MCP) servers from 1C:Enterprise. It provides a native component (DLL) that handles the MCP transport layer and a reference 1C data processor that demonstrates how to implement tools, resources, and prompts.
Use it as a template to expose any 1C business logic — catalogs, documents, reports, calculations — to AI applications like VS Code Copilot, Claude Desktop, and other MCP-compatible clients.
Core Concept
The project is intentionally split into two layers with different responsibilities.
Native component responsibilities
The DLL is the MCP engine. It handles protocol and transport details that should not be reimplemented in 1C business code:
- HTTP/SSE transport
- JSON-RPC request/response lifecycle
- MCP session management
- authentication, origin validation, and rate limiting
- pagination, notifications, and progress streaming
- converting 1C responses into valid MCP replies
1C responsibilities
The 1C side owns business logic. A 1C developer should work at the level of tools, resources, and prompts, not at the level of MCP internals:
- describe a tool/resource/prompt in BSL
- register it in the component
- handle the incoming call in 1C
- run any required client-side or server-side 1C logic
- return the result or send progress updates
Why the project is designed this way
The goal is to let a 1C developer publish almost any 1C functionality through MCP without having to understand HTTP, JSON-RPC, SSE, session handling, or MCP message formatting.
In other words:
- the component knows how to be an MCP server
- 1C knows what the tool actually does
This keeps the integration point simple. To add new functionality, a 1C developer does not need to change the native transport layer. They only add or update 1C definitions and handlers.
Mental model for a 1C developer
From the 1C side, the workflow is intentionally simple:
- Define the MCP object in BSL.
- Register it in the component.
- Receive the request through
ExternalEvent. - Execute arbitrary 1C logic.
- Return the final result, and optionally send progress while the operation is running.
This means the project is not a fixed set of built-in utilities. It is an MCP transport and protocol layer for 1C, with the actual application behavior defined in 1C code.
Key Features
- Full MCP protocol support — tools, resources, prompts with
listChangednotifications - Streamable HTTP transport —
POST /mcpfor requests,GET /mcpfor SSE notification stream - Session management —
Mcp-Session-Idheader, UUID v4 sessions per spec - Security — Origin validation (DNS rebinding protection), Bearer token auth, rate limiting
- Progress streaming — SSE-based progress notifications for long-running operations
- Pagination — cursor-based pagination for
tools/list,resources/list,prompts/list - Tool annotations —
readOnlyHint,destructiveHint,idempotentHint,openWorldHint - Output schemas — typed response contracts for tool results
- Dynamic registration — register/update tools, resources, prompts at runtime from 1C
- Built-in semantic search (RAG) — optional
search/grep/get_segment/list_collectionstools backed by a Rust search core (rcore) with dense/keyword/hybrid retrieval (see Search Subsystem)
Architecture
┌─────────────────┐ HTTP/SSE ┌──────────────────┐ ExternalEvent ┌──────────────────┐
│ MCP Client │◄─────────────────►│ Native DLL │◄──────────────────►│ 1C:Enterprise │
│ (VS Code, etc) │ POST/GET /mcp │ (HttpServer) │ ToolCall, etc. │ (BSL Module) │
└─────────────────┘ └──────────────────┘ └──────────────────┘
- A 1C form loads the native add-in and starts the HTTP server.
- An MCP client connects to
http://localhost:PORT/mcp. - The native component handles protocol-level messages (initialize, tools/list, etc.).
- Business logic requests (tools/call, resources/read, prompts/get) are forwarded to 1C via
ExternalEvent. - The 1C module processes the request and sends results back through
SendResponse. - The DLL wraps the result in a JSON-RPC response and returns it to the client.
Quick Start
1. Build the DLL
build/build-http1c-dll-release.sh
Requires Visual Studio Build Tools 2019+ with C++ support.
2. Package the add-in
Packaging is done automatically at the end of the build script. To run it standalone:
build/package-http1c-addin.sh
3. Compile the EPF (requires OneScript)
build/compile-http1c-epf.sh
4. Open in 1C
- Open the
http1c.epfdata processor in your 1C infobase. - Click Connect — the MCP server starts on the configured port.
- Configure your MCP client to connect to
http://localhost:PORT/mcp.
5. VS Code configuration
Add to your .vscode/mcp.json:
Without authentication:
{
"servers": {
"1c-mcp-server": {
"type": "sse",
"url": "http://localhost:8888/mcp"
}
}
}
With Bearer token authentication:
{
"servers": {
"1c-mcp-server": {
"type": "sse",
"url": "http://localhost:8888/mcp",
"headers": {
"Authorization": "Bearer ${input:mcpToken}"
}
}
},
"inputs": [
{
"id": "mcpToken",
"type": "promptString",
"description": "Bearer token for the 1C MCP server",
"password": true
}
]
}
When using the ${input:...} syntax, VS Code will prompt for the token each time the MCP server connection starts. The entered value is masked as a password.
Important: The token in VS Code must match the value set on the 1C side. If the server has no token configured (empty string), authentication is disabled and no
headersare needed. If a token is set on the server but VS Code sends noAuthorizationheader, the server responds with HTTP 401 and VS Code may try to start an OAuth flow — this is not supported; use theheadersapproach above instead.
How to Build Your Own MCP Server from 1C
The reference data processor (http-1c-dp) is a working example. Use it as a starting point:
Registering Tools
Tools are executable functions that AI clients can invoke. Define them as JSON structures and register with the component:
// Create a tool definition
Tool = NewTool("myTool", "Description of what this tool does");
AddToolParam(Tool, "paramName", "string", "Parameter description");
AddToolAnnotations(Tool, True); // readOnly, safe
// Define output schema (optional, helps clients validate responses)
Schema = NewOutputSchema();
AddOutputProperty(Schema, "result", "string", "Result description");
SetToolOutputSchema(Tool, Schema);
// Register all tools
Tools = New Array;
Tools.Add(Tool);
Await Component.RegisterToolsAsync(SerializeToJson(Tools));
Handle the tool call in the ExternalEvent handler:
&AtClient
Async Procedure ExternalEvent(Source, Event, Data)
If Source <> "HttpServer" Then Return; EndIf;
If Event = "ToolCall" Then ProcessToolCall(Data); EndIf;
EndProcedure
Registering Resources
Resources provide contextual data to AI clients (metadata, file contents, etc.):
Resource = New Structure;
Resource.Insert("uri", "1c://metadata/catalogs");
Resource.Insert("name", "1C Catalogs");
Resource.Insert("description", "List of all catalog metadata objects");
Resource.Insert("mimeType", "application/json");
Resources = New Array;
Resources.Add(Resource);
Await Component.RegisterResourcesAsync(SerializeToJson(Resources));
Handle resource reads via "ResourceRead" events.
Registering Prompts
Prompts are reusable interaction templates:
Prompt = New Structure;
Prompt.Insert("name", "analyzeData");
Prompt.Insert("description", "Prompt for analyzing 1C data");
PromptArgs = New Array;
Arg = New Structure("name,description,required", "topic", "Analysis topic", False);
PromptArgs.Add(Arg);
Prompt.Insert("arguments", PromptArgs);
Prompts = New Array;
Prompts.Add(Prompt);
Await Component.RegisterPromptsAsync(SerializeToJson(Prompts));
Handle prompt gets via "PromptGet" events.
Dynamic Updates
Call RegisterToolsAsync() / RegisterResourcesAsync() / RegisterPromptsAsync() again at any time with an updated list. The component will automatically send notifications/tools/list_changed (or equivalent) to all connected MCP clients.
Security Configuration
The component supports optional Bearer token authentication. When a token is set, every HTTP request must include the Authorization: Bearer <token> header or it will be rejected with HTTP 401.
// Enable authentication — all requests must include Authorization: Bearer my-secret-token
Component.AuthToken = "my-secret-token";
// Disable authentication — any request is accepted
Component.AuthToken = "";
How it works:
| Server token | Client header | Result |
|---|---|---|
| Empty (default) | None needed | All requests accepted |
"my-secret" |
Authorization: Bearer my-secret |
Request accepted |
"my-secret" |
Missing or wrong token | HTTP 401 Unauthorized |
Changing the token at runtime: You can set or clear AuthToken while the server is running. The change takes effect immediately for all new requests — no restart needed.
VS Code note: If the server returns 401, VS Code may attempt an OAuth 2.0 PKCE authorization flow (redirecting to /authorize). This is not supported by the component. Always configure the token in .vscode/mcp.json via the headers field (see VS Code configuration above).
The component also enforces:
- Origin validation — only requests from
localhost/127.0.0.1/ VS Code origins are accepted - Rate limiting — token-bucket algorithm (60 burst, 20/sec)
- Session management —
Mcp-Session-Idassigned on initialize, validated on subsequent requests. A request presenting an unknown session id is not rejected with 404 — the session is transparently resurrected under the presented id, so a server restart does not strand connected MCP clients that never re-initialize (logged asMCP: unknown session ... resurrected (server restart?))
Native Component API
Methods exposed to 1C (English / Russian names). All of these are callable
asynchronously from BSL via the platform-generated BeginCalling<Name> wrappers
(BeginCallingStartListen, …):
| Method | Description |
|---|---|
StartListen(port) / НачатьПрослушивание |
Start the HTTP server on the given port |
StopListen() / ОстановитьПрослушивание |
Stop the server and unblock all pending requests |
SendResponse(json) / ОтправитьОтвет |
Send the final response for a pending request |
SendProgress(id, progress, total, message) / ОтправитьПрогресс |
Send a progress notification for a pending request |
ApplyConfig(json) / ПрименитьНастройки |
Async-safe config sink (see below). Applies any subset of logging_enabled, log_path, timeout, tools_json, resources_json, prompts_json, auth_token in one call |
GetStatus() / ПолучитьСтатус |
Async-safe read of the status JSON (same payload as the Status property) |
RagDispatch(method, payload) / RagВыполнить |
Drive the Rust search core (rcore) — see Search Subsystem |
TakeScreenshot(pid, format, quality, grayscale) / СделатьСкриншот |
Capture windows of a process as base64 images |
GetProcessId() / ПолучитьИдентификаторПроцесса |
Return the current 1C process id |
Configuration: async methods vs. synchronous properties
The component also exposes its configuration as properties (LoggingEnabled,LogPath, Timeout, Tools, Resources, Prompts, AuthToken, Status,Version). These still exist, but reading or writing an AddIn property is a
synchronous platform call.
In an async-only infobase — i.e. one where "synchronous extension and
add-in calls" are disabled (the modern 1C default) — everyComponent.Prop = …
assignment and every… = Component.Propread throwsCannot call synchronous methods on the client!. 1C provides no async
property accessors (there is noBeginSetProperty), so configuration cannot
go through properties at all in that mode.
That is why all configuration is funneled through the async ApplyConfig
method and all status reads through the async GetStatus function. The
reference form (Module.bsl) uses only these — it never touches a property on
the client. The properties are retained for backward compatibility and for
infobases that still allow synchronous calls.
Config field (in ApplyConfig JSON) |
Equivalent property | Notes |
|---|---|---|
logging_enabled (bool) |
LoggingEnabled |
runtime logging on/off |
log_path (string) |
LogPath |
log file path (empty → default) |
timeout (int) |
Timeout |
response timeout, seconds |
tools_json (string) |
Tools |
pre-serialized tool-list JSON array |
resources_json (string) |
Resources |
pre-serialized resource-list JSON array |
prompts_json (string) |
Prompts |
pre-serialized prompt-list JSON array |
auth_token (string) |
AuthToken |
Bearer token (empty = no auth) |
(read via GetStatus) |
Status / Version |
status JSON includes version |
Every field is optional — ApplyConfig only touches the keys you send, so it
doubles as a "set just the logging flag" call or a full one-shot configuration.
ExternalEvent Types
Events sent from the native component to 1C:
| Event | Description | Data |
|---|---|---|
ToolCall |
MCP tools/call request |
{id, type, tool, arguments, progressToken} |
ResourceRead |
MCP resources/read request |
{id, type, uri} |
PromptGet |
MCP prompts/get request |
{id, type, name, arguments} |
Request |
Legacy HTTP request (non-MCP) | {id, method, path, body, params} |
MCP Protocol Support
Implemented Methods
| Method | Handler |
|---|---|
initialize |
Native — returns capabilities, creates session |
notifications/initialized |
Native — accepted silently |
ping |
Native — returns empty result |
tools/list |
Native — paginated, from cache |
tools/call |
search / grep / get_segment / list_collections handled natively by the search core; all other tools delegated to 1C via ExternalEvent |
resources/list |
Native — paginated, from cache |
resources/read |
Delegated to 1C via ExternalEvent |
prompts/list |
Native — paginated, from cache |
prompts/get |
Delegated to 1C via ExternalEvent |
Capabilities Advertised
{
"tools": { "listChanged": true },
"resources": { "listChanged": true },
"prompts": { "listChanged": true }
}
HTTP Endpoints
| Endpoint | Description |
|---|---|
POST /mcp |
MCP JSON-RPC messages |
GET /mcp |
SSE notification stream (list_changed events) |
DELETE /mcp |
Session termination |
GET /health |
Health check |
OPTIONS * |
CORS preflight |
Reference Tools (in the demo processor)
| Tool | Purpose | Annotations |
|---|---|---|
getStatus |
Component + runtime status | readOnly |
openForm |
Open a 1C form by path | idempotent |
execute |
Execute arbitrary 1C code | destructive |
evaluate |
Evaluate a 1C expression | readOnly, idempotent |
runLongTask |
Test progress notification | readOnly |
Reference Resources
| URI | Description |
|---|---|
1c://metadata/catalogs |
JSON list of catalog metadata objects |
1c://metadata/documents |
JSON list of document metadata objects |
Reference Prompts
| Prompt | Arguments | Description |
|---|---|---|
analyze1CData |
topic (optional) |
System prompt for data analysis with metadata context |
generate1CCode |
task (required) |
System prompt for BSL code generation with conventions |
Search Subsystem (RAG)
The component ships with an optional semantic-search subsystem so a 1C MCP server can index its own content (documentation, code, catalog descriptions, logs, QA scenarios, master data) and let AI clients search it by meaning. The retrieval engine is a small Rust core, crate rcore, that runs entirely in-process inside the 1C session.
Why this exists
An MCP server already lets an AI client act inside a specific 1C installation (run a query, call a method, execute a test). What it lacked was the ability to find by meaning the thing to act on. That is retrieval — the "R" in RAG — and it is what this subsystem adds. Generation stays with the client model; we only retrieve and ground.
Concretely, it closes gaps like: an AI writing a Vanessa BDD test needs the catalog of available steps and the existing scenarios so it reuses real, runnable phrases instead of hallucinating step text; or finding a product/client by intent ("that construction contractor from Kyiv", "red winter jacket") rather than by exact code.
Design decisions (and why)
The shape of the subsystem follows directly from its constraints — they are part of the problem, not incidental:
| Decision | Rationale |
|---|---|
| Content-agnostic core; domain logic lives in outside adapters | The core never knows about Gherkin, SKUs, or INNs. Adapters (in 1C/BSL) parse the domain and feed the core generic records. One engine serves QA, products, clients, docs — instead of N bespoke searches. |
| In-process, in-memory, no disk persistence | Data is corporate and session-bound (Vanessa runs in the client session); the index lives inside the 1C process and dies with it. A stale cache is worse than none for grounding, so the index is rebuilt from the live source on load. |
| Flat brute-force vector index (no ANN/Qdrant) | Target scale is ~5–10k records per session. Search is microseconds and RAM is tens of MB — an ANN index would be pure complexity. |
| Hybrid retrieval (dense + keyword), not pure semantic | Half the queries are exact identifiers (SKU, INN, step syntax) where embeddings fail; half are intent where exact match fails. Both channels are needed; keyword is a full scan (cheap at this scale). |
| Asynchronous ingest (push from 1C) | Embedding 5–10k records takes 15–60 s and must never freeze the thin-client UI. Ingest calls return immediately; a background worker embeds off-thread. Progress is observed by polling stats / list_collections. |
| Async-only config/status methods | 1C forbids synchronous AddIn property access in async-only infobases, so configuration and status go through the async ApplyConfig/GetStatus methods (see Native Component API). |
text vs. embed_text split |
The single biggest quality lever — the adapter decides what gets embedded (an enriched composite) separately from what is stored/returned verbatim, without polluting the core with domain fields. |
MCP tools
Four search tools are served natively by the component — they are handled inside the DLL and are not forwarded to 1C via ExternalEvent:
| Tool | Purpose |
|---|---|
search |
Semantic / keyword / hybrid ranked search over indexed segments. Returns hits with doc_id, name, collection, score, segment line ranges, (optionally) text, and the hit's effective meta — document meta overlaid by segment meta (segment wins on collision), the same view the meta filters match against. |
grep |
RE2 regex search (linear time, no backreferences/lookaround) over the stored document text. Works the instant a document is ingested — no vectors needed. |
get_segment |
O(1) line-range slice of a raw document by doc_id via an offset table. Out-of-range requests are clamped and the actual range is returned. |
list_collections |
The collection registry — AI-facing discovery of what is searchable before querying (see Collections). Takes no arguments. |
search supports three retrieval modes via the mode argument:
- dense (default) — cosine/dot-product similarity over normalized embedding vectors.
- keyword — lexical scoring, no vectors required.
- hybrid — Reciprocal Rank Fusion (RRF) of the dense and keyword result lists.
Both search and grep accept metadata filters (all/any clauses) and a collection to scope the query.
Ingest
Documents are pushed into the core through rcore's JSON ABI. Ingest is asynchronous: the call is accepted under a short lock and returns immediately, while a background worker embeds the segments off-thread. Text-based tools (grep, get_segment) work right after acceptance; dense search becomes available once embedding finishes (stats exposes per-collection progress).
index_segments— ingest a document as a list of pre-chunked segments (each with optionalembed_text, line range, and per-segmentmeta).doc_idis required so segments can be upserted/deleted.index_raw— ingest a raw multi-line document; the core normalizes line endings, builds a line-offset table, stores the full text, and chunks it (line-snapped, by token budget, with overlap).doc_idis optional (auto-assigned and returned in the ack). Accepts the same optionalcollection_descriptionasindex_segments(last non-empty wins).
Each segment carries two distinct texts:
text— stored and returned verbatim (the canonical record).embed_text— the enriched composite that actually goes into the vector (name + tags + steps for a scenario; name + brand + category for a product). If omitted,textis embedded. This is where the adapter spends its quality budget.
Collections and the collection registry
Every document belongs to a collection (a named bucket: qa_steps, products, clients, …). Collections are created implicitly on first ingest and can carry a free-text description.
list_collectionsreturns the registry: for each collection itsname,description,n_docs,n_segments,vector_status(empty/building/ready/error), andtext_ready. This is how an AI client discovers what is searchable before querying — it is exposed both as aRagDispatchmethod and as the fourth native MCP tool (no arguments):{ "collections": [ { "name": "qa_steps", "description": "Vanessa BDD step catalog", "n_docs": 2, "n_segments": 780, "text_ready": true, "vector_status": "ready" } ] }searchscopes bycollection: omit it to search all collections, pass one name, or pass a comma-separated list (or acollectionsarray) to search a chosen subset. So an agent can read the registry, pick the relevant collections by description, and search just those.
Embedding worker pool
The embedder runs on a background pool whose size is set by embed_workers in configure (settable from the 1C form). One worker (default) keeps a single ONNX session and serializes embedding; multiple workers run several sessions concurrently. On CPU this trades memory for throughput (roughly memory-bandwidth bound, not linear in workers); multi-worker forces CPU because concurrent DirectML sessions are not supported. Query embeddings are prioritized so a bulk reindex does not starve live search latency.
Core method contract (RagDispatch)
All of the following are invoked from 1C via the async RagDispatch(method, payload) method (the C++ component forwards the JSON envelope to rcore verbatim and does not interpret the method name). Each returns {"ok":true,"result":…} or {"ok":false,"error":{"code","message"}}:
| Method | Purpose |
|---|---|
configure |
Load the model once — model (a built-in name from the whitelist: multilingual-e5-small (default) / multilingual-e5-base / multilingual-e5-large; an unknown name fails with a structural bad_model error listing the supported names), cache_dir (directory the built-in model is downloaded into / cached in), model_path (offline local files, bypasses the whitelist), device, normalize, max_seq_len, intra_threads, embed_workers. dim is fixed by the model. Idempotent; re-configuring with a different model/dim after indexing implies reset. |
index_segments |
Async ingest of pre-chunked segments (with embed_text, line ranges, per-segment meta, optional collection description). |
index_raw |
Async ingest of a raw document; the core chunks it (line-snapped, token-budgeted, overlapping) and builds the offset table. Same optional collection_description as index_segments (last non-empty wins). |
search |
Ranked retrieval — mode dense/keyword/hybrid, collection(s), meta filters, k, min_score, max_per_doc, include_text. Hits echo the effective meta (doc overlaid by segment, segment wins) — the same view the filters match. |
get_segment |
O(1) line-range slice of a raw document by doc_id. |
grep |
RE2 regex / substring scan over stored text. |
stats |
Global + per-collection counters, model, dim, memory estimate, collection statuses. |
list_collections |
The collection registry (names + descriptions + counts + status). |
list_models |
The built-in model whitelist — {default, models:[…]}. Works before configure and in the lite/mock build, so a client can always render a model picker. |
delete_document / delete_collection |
Remove a document (by doc_id) or a whole collection. |
reset |
Clear the entire index. |
Working with the subsystem — for the 1C developer
You write adapters: thin BSL that turns a domain object into generic records and pushes them in. The core stays domain-free.
- Attach + configure once. Attach the full component, then
configureviaRagDispatchwith a built-inmodelname (or a localmodel_path, plus optionallycache_dir) andembed_workers. (Component-level settings — logging/timeout — go throughApplyConfig, never through synchronous properties.) - Index asynchronously. For each domain object build segments — set
text(verbatim) andembed_text(the enriched composite), attachmeta({sku, inn, type, tags, …}) and adoc_idfor upsert — and callindex_segments. The call returns immediately; embedding happens in the background. - Watch progress by polling
list_collections/statsuntilvector_status = ready. Text tools (grep,get_segment) work the instant a document is accepted; densesearchlights up when embedding finishes. - Keep it fresh within the session. When one object changes, re-
index_segmentsjust thatdoc_id(atomic upsert) ordelete_document— never rebuild the whole corpus. - All file/content reads stay client-side (
&AtClient) in the reference form, so source files are read from the operator's machine, not the server.
The reference end-to-end flow is exercised headlessly by the RAG self-test (stub QA/products/clients adapters → index → embed → search, asserting pass/fail). See the onec-rag-selftest skill.
Working with the subsystem — for the AI agent / MCP client
From the client side it is just four MCP tools, but used in a deliberate order:
- Discover. Call
list_collectionsfirst to see what this specific 1C base has indexed and read each collection's description — don't assume; the corpus is per-installation and per-session. - Search by meaning, scoped. Call
searchwithmode: hybridfor anything involving exact identifiers (SKU/INN/article/step syntax) anddensefor pure intent. Scope to the relevantcollection(s) from step 1, or omit to search everything. Use meta filters (all/any) to narrow. - Ground, don't hallucinate. Treat hits as the source of truth for this base. For QA, prefer canonical step phrases and existing scenarios returned by search over inventing them; a reverse step→scenario lookup (a QA-adapter convenience over the same
search) surfaces real usages with real parameters. - Zoom in. Use
get_segmentto pull the exact line range of a hit verbatim, andgrepfor exact-string / regex confirmation (it works even before vectors are ready). Abuildingcollection returns partial results flagged as incomplete — retry once it isready.
The contract is: the client retrieves and grounds on what actually exists in this base right now; it never swallows the whole corpus into context and never relies on exact match alone.
Lite vs full
The same libhttp1cWin.dll ships in two distributions. The component is pure C++ built /MT and does not link the Rust core; instead it loads an optional rcore.dll at runtime (LoadLibrary + GetProcAddress, resolved next to the component DLL — see http-1c-dll/src/RustCore.h):
| Distribution | Contents | Search behavior |
|---|---|---|
| lite | libhttp1cWin.dll only |
search / grep / get_segment / list_collections are advertised in tools/list but return a structured error (code: rag_not_installed) telling the caller to install the RAG package. |
| full | libhttp1cWin.dll + rcore.dll + DirectML.dll |
Real fastembed/onnxruntime search core; the four tools run for real. |
Detection is automatic at runtime: if rcore.dll is present next to the component and all four ABI entry points (rcore_version / rcore_dispatch / rcore_free_string / rcore_shutdown) resolve, the component runs the full search path. A missing, partial, or version-mismatched rcore.dll degrades cleanly to the lite path (an "install RAG" error) — never a crash. The tool schemas are identical in both distributions, so a client sees the same tools/list either way.
GPU / CPU (DirectML)
The full search core uses onnxruntime's DirectML execution provider for GPU acceleration, with automatic CPU fallback when no compatible GPU is available. The device is selected via the configure method's device field:
device |
Behavior |
|---|---|
auto (default) |
DirectML with onnxruntime's automatic CPU fallback. |
dml |
Force the DirectML (GPU) execution provider. |
cpu |
Force CPU execution. |
DirectML.dllis a hard dependency of the full package. ort's prebuilt onnxruntime bundles the DirectML provider, sorcore.dllhard-importsDirectML.dll. The full bundle must shipDirectML.dllalongsidercore.dll, otherwisercore.dllfails to load and the component silently falls back to the lite "install RAG" behavior. On Windows it lives atC:\Windows\System32\DirectML.dll. (Version-coupling caveat: thisDirectML.dllis paired with the bundled onnxruntime; a future CPU-only onnxruntime build would remove the import and this DLL could be dropped.)
The embedding model itself is fetched at runtime, not at build time — nothing model-related is downloaded during the build or in CI.
Vanessa Automation Integration
The repository ships a ready-made integration for Vanessa Automation (1C BDD testing): the MCPRagSearch plugin in vanessa-plugin/. It runs the http1c component inside the VA session and serves Vanessa's entire MCP contract through it — all VA tools are collected from MCPVA via a small facade (a micro-hook already present in the VA fork), while search / grep / get_segment / list_collections run natively over the indexed VA step library (vanessa_steps), the VA knowledge base (vanessa_kb), and arbitrary file collections. See vanessa-plugin/README.md (Russian) for setup, the collections, the search-guide prompt / get_search_guide tool, and the auto-reindex watcher.
Repository Layout
├── build/
│ ├── build-http1c-dll-debug.sh # Build DLL in debug mode
│ ├── build-http1c-dll-release.sh # Build DLL in release mode
│ ├── compile-http1c-epf.sh # Compile EPF from XML
│ ├── package-http1c-addin.sh # Package DLL as 1C add-in ZIP
│ └── onescript/
│ └── compile-external-processor.os
├── http-1c-dll/
│ ├── CMakeLists.txt # CMake build configuration
│ ├── version.h # Version management
│ ├── include/ # 1C API headers + vendored libraries
│ │ ├── httplib.h # cpp-httplib (HTTP server)
│ │ ├── json.hpp # nlohmann/json (JSON parser)
│ │ └── AddInDefBase.h, ... # 1C Native API headers
│ └── src/
│ ├── AddInNative.cpp/h # Generic 1C add-in framework
│ ├── HttpServerComponent.cpp/h # MCP server implementation
│ ├── RustCore.h # Runtime loader for rcore.dll (search)
│ └── AddInNative.def # DLL export definitions
├── rust-core/ # `rcore` — Rust search core (rcore.dll)
│ ├── Cargo.toml # cdylib crate, `fastembed` feature
│ └── src/ # dispatch, embed, grep, filter, core
├── vanessa-plugin/
│ └── MCPRagSearch/ # Vanessa Automation plugin (MCP server + RAG)
├── http-1c-dp/
│ ├── http1c.xml # 1C data processor XML source
│ └── http1c/
│ └── Forms/Form/Ext/Form/
│ └── Module.bsl # Reference MCP server implementation
└── http1c.epf # Compiled 1C external data processor
Technology Stack
- C++17 — native component
- CMake + MSVC — build system
- cpp-httplib — embedded HTTP server
- nlohmann/json — JSON parser
- 1C Native API — integration with 1C:Enterprise
- OneScript — EPF compilation from XML
- Rust (
rcorecrate) + fastembed / onnxruntime (ort) — optional search core inrcore.dll(full distribution only)
Based on lintest/AddinTemplate for the native add-in layer.
License
See LICENSE.
Build Requirements
The checked-in build scripts target Windows.
Required tools:
- Microsoft Visual Studio Build Tools 2019+ with MSVC (C++ workload) — must be installed manually (system-level, requires admin)
- CMake and Ninja — downloaded automatically to
build/tools/on first run if not found in PATH
Setting Up a New Developer Machine
Full Visual Studio is not required — the free standalone Build Tools are sufficient, and they are the only thing you need to install manually.
Install via winget (run once, requires admin):
winget install Microsoft.VisualStudio.2022.BuildTools --override "--quiet --add Microsoft.VisualStudio.Workload.VCTools --includeRecommended"
Or via Chocolatey:
choco install visualstudio2022buildtools --package-parameters "--add Microsoft.VisualStudio.Workload.VCTools --includeRecommended" -y
Or download the installer manually from https://visualstudio.microsoft.com/downloads/#build-tools-for-visual-studio-2022 and select the "Desktop development with C++" workload.
After that, just run a build script — CMake and Ninja will be downloaded automatically to build/tools/ on first use.
Building the DLL
Release build:
build/build-http1c-dll-release.sh
Debug build:
build/build-http1c-dll-debug.sh
The DLL output is written to:
http-1c-dll/bin/libhttp1cWin.dll
Debug symbols are also written into http-1c-dll/bin/.
After a successful DLL build, the top-level build scripts also package the native add-in bundle with MANIFEST.XML metadata and update the embedded 1C template here:
http-1c-dp/http1c/Templates/http1c/Ext/Template.bin
Building the search core (full distribution)
By default the build produces the lite component only — pure C++, no Rust/cargo involved. To also build the full search core (rcore.dll), configure CMake with -DRCORE_FASTEMBED=ON:
cd http-1c-dll && mkdir -p build && cd build
# lite — libhttp1cWin.dll alone (no cargo):
cmake .. -G Ninja -DCMAKE_BUILD_TYPE=Release
cmake --build .
# full — also builds rcore.dll via cargo:
cmake .. -G Ninja -DCMAKE_BUILD_TYPE=Release -DRCORE_FASTEMBED=ON
cmake --build .
The full build invokes cargo build --release --features fastembed (handled inside CMake) and drops rcore.dll next to the component in http-1c-dll/bin/. The embedding model is fetched at runtime, so no model is downloaded at build time.
Why two binaries (/MT vs /MD). The C++ component is built with the static CRT (/MT) — it cannot compile under /MD because MSVC's char16_t stream code hits error C2491. onnxruntime (via ort), however, requires the dynamic CRT (/MD). Linking the two together is impossible, so the search core is a separate rcore.dll (a cdylib) that the /MT component loads at runtime instead of linking. CMake builds the cdylib with the dynamic CRT by removing the +crt-static default (RUSTFLAGS=-C target-feature=-crt-static); this is independent of the component's /MT and they never link together.
Packaging the add-in (lite / full)
build/package-http1c-addin.sh assembles a 1C add-in ZIP (and updates Template.bin). The variant is selected via the VARIANT env var:
# lite (default) — libhttp1cWin.dll only:
VARIANT=lite bash build/package-http1c-addin.sh
# full — also bundles rcore.dll + DirectML.dll:
VARIANT=full bash build/package-http1c-addin.sh
The full packaging step copies DirectML.dll from C:\Windows\System32\DirectML.dll (override with DIRECTML_SRC) into the bundle — it is a hard dependency of rcore.dll (see GPU / CPU) and the script fails clearly if it is missing. Set SKIP_TEMPLATE=1 to produce only the ZIP without overwriting the committed Template.bin (which tracks the lite bundle).
The release workflow (.github/workflows/release.yml) builds both variants and attaches both ZIPs to the GitHub release:
http1c-addin-lite-v<version>.zip # libhttp1cWin.dll
http1c-addin-full-v<version>.zip # libhttp1cWin.dll + rcore.dll + DirectML.dll
Running the 1C Side
The repository includes an external data processor:
http-1c-dp/http1c.epf
Its form module is responsible for:
- attaching the native add-in,
- starting the HTTP listener,
- registering MCP tools,
- handling requests forwarded from the DLL,
- sending responses and progress messages back to the DLL.
The add-in is attached from the embedded 1C template instead of a machine-specific DLL path.
Logging
The default log file is no longer a hardcoded absolute path.
http_debug.log
On the native side, when no explicit log path is provided from 1C, the component falls back to the system temporary directory.
%TEMP%\http1c.log
You can still override the log path through the form field before connecting.
VS Code MCP Configuration
The workspace already contains a sample MCP client configuration:
{
"servers": {
"ConnectionTo1C": {
"type": "sse",
"url": "http://localhost:8888/mcp"
}
}
}
This matches the default listener port used by the 1C form when no custom port is set.
Typical Local Workflow
- Build
http-1c-dll. - Open
http-1c-dp/http1c.epfin 1C:Enterprise. - Optionally adjust the port and log path in the form.
- Attach the add-in from the embedded template.
- Start the local listener.
- Use the MCP server from VS Code or another MCP-capable client.
EPF compilation from XML sources is now launched from the top-level build folder:
build/compile-http1c-epf.sh
The compiled processor artifact is written to the repository root:
http1c.epf
Limitations and Caveats
- Windows-focused setup. The build scripts target MSVC and NMake via bash.
- The build still depends on Windows toolchain availability through
PATHor an initialized Visual Studio build environment. - The HTTP server is intentionally local-only and binds to
127.0.0.1. executeandevaluateexpose powerful server-side capabilities and should only be used in trusted local environments.- The current tool set mixes practical helpers with demo functionality such as
runLongTask. - There is no packaging or deployment flow yet for distributing the add-in as a polished product.
Testing & Deployment Notes
When updating the native add-in DLL during development:
- Clear the component cache. 1C caches native add-ins in a temporary directory. Delete the cache folder before loading a new version:
%APPDATA%\1C\1cv8\ExtCompT - Restart the 1C session. Close the 1C application completely and reopen it — a running session keeps the old DLL locked.
- Disable dangerous action protection. In the 1C user settings, uncheck "Protection from dangerous actions" (
Защита от опасных действий). Otherwise the platform will block add-in attachment.
Without these steps the platform may silently load an outdated DLL or refuse to attach the add-in.
Version
The native component version defined in the source is:
1.5.0
Recent changes:
- 1.5.0 — Async-only config/status. Added the async
ApplyConfigandGetStatusmethods so the component is fully usable from async-only infobases (where synchronous AddIn property access throws "Cannot call synchronous methods on the client"). The reference form no longer touches a property on the client. Search subsystem gained the collection registry (list_collections, per-collection descriptions, multi-collectionsearch) and a configurable embedding worker pool (embed_workers).
License
This project is licensed under the MIT License. See LICENSE.
Reviews (0)
Sign in to leave a review.
Leave a reviewNo results found