Sandboxing
When anyquery is exposed to clients — as a MySQL server, or as an LLM endpoint (gpt/mcp) — those clients can run arbitrary SQL. Anyquery’s built-in features make arbitrary SQL powerful: the read_* table functions read files, several of them fetch remote URLs, ATTACH DATABASE writes files, and a few scalar functions read files or delete cache directories. On a server, that turns into local file read, server-side request forgery (SSRF), and arbitrary file write for anyone who can reach the port.
The sandbox closes those doors. It is a policy attached to the database namespace that confines file access to an explicit set of directories, blocks remote fetches, and denies the dangerous SQL statements and functions.
When is the sandbox active?
Section titled “When is the sandbox active?”The default depends on the command, because the exposure differs:
| Command | Sandbox by default | How to change it |
|---|---|---|
anyquery server | On | --no-sandbox to disable |
anyquery gpt | On (exposes an internet tunnel by default) | --no-sandbox to disable |
anyquery mcp | Auto — on when network-exposed (--tunnel, or a non-loopback --host); off for plain localhost/--stdio | --sandbox to force on, --sandbox=false to force off |
anyquery query / interactive shell | Off (local use is trusted) | --sandbox to opt in |
CLI mode is meant for local data analysis and is not an attack surface, so it is unrestricted by default. You can still opt in with --sandbox to mirror the server’s behaviour.
What the sandbox restricts
Section titled “What the sandbox restricts”When active, the sandbox enforces the following. The default is deny everything, then you relax it with the flags below.
- File reads — the
read_*table functions (read_csv,read_json,read_parquet,read_yaml,read_toml,read_jsonl,read_html,read_log) may only read files inside the directories you list with--allow-dirs. Symlinks are resolved before the check, so a link inside an allowed directory cannot point out of it. - Remote fetches — fetching
http,https,s3,gcs,git, … URLs is disabled. Only local files are reachable unless you pass--allow-remote. - Database readers — the
duckdb_reader,postgres_reader,mysql_reader,clickhouse_readerandcassandra_readermodules are not registered at all (they take arbitrary connection strings, and DuckDB can itself read local files and load extensions).CREATE VIRTUAL TABLE … USING duckdb_reader(...)fails withno such moduleunless you pass--allow-db-connections. ATTACH DATABASE/VACUUM … INTO— both are arbitrary-file-write primitives. In-memory databases (:memory:,mode=memory) are always allowed; writing to disk is denied unless you pass--allow-attach, and even then it is confined to--allow-dirs.- Blocked SQL functions — see below.
- Restricted PRAGMAs — see below.
Relaxing the restrictions
Section titled “Relaxing the restrictions”The default configuration is intentionally strict: no readable directories, no remote access, no database connections. Open up only what you need.
anyquery server --allow-dirs /var/data,/srv/exportsanyquery server --allow-dirs /var/data --allow-remoteanyquery server --allow-dirs /var/data --allow-attachanyquery server --allow-db-connections| Flag | Effect |
|---|---|
--allow-dirs <dir,dir> | Directories the read_* tables (and on-disk ATTACH) may access. Repeatable / comma-separated. Empty by default. |
--allow-remote | Allow read_* tables to fetch remote URLs. |
--allow-attach | Allow ATTACH DATABASE / VACUUM … INTO to on-disk paths within --allow-dirs. |
--allow-db-connections | Register the duckdb_reader/postgres_reader/… modules. |
Blocked SQL functions
Section titled “Blocked SQL functions”A handful of scalar functions read files or delete directories on disk. When the sandbox is active they are denied outright by the SQLite authorizer — they cannot be relaxed with --allow-dirs:
| Function | Why it is blocked |
|---|---|
load_file, load_file_bytes | Read an arbitrary file into a value (a local-file-read bypass of the read_* confinement). |
clear_plugin_cache, clear_file_cache | Delete cache directories on disk (cache management is an operator action, not a client one). |
load_extension | Loading a SQLite extension is remote code execution. The SQL function is disabled by the driver, and the sandbox denies it explicitly as defence in depth. |
SELECT load_file('/etc/passwd'); -- error: not authorizedTo read a file inside an allowed directory, use a read_* table function instead of load_file — those are permitted within --allow-dirs:
SELECT * FROM read_csv('/var/data/report.csv');Restricted PRAGMAs
Section titled “Restricted PRAGMAs”PRAGMA is gated to a read-only allowlist. Only schema-introspection pragmas that the engine, the MySQL protocol handler and the information_schema/SHOW emulation rely on are permitted:
table_info, table_xinfo, table_list, index_info, index_xinfo, index_list,foreign_key_list, database_list, collation_list, function_list, module_list,pragma_list, compile_optionsEvery other PRAGMA is denied. This blocks schema-corruption vectors (PRAGMA writable_schema=ON followed by UPDATE sqlite_master) and memory-inflation pragmas (PRAGMA cache_size, PRAGMA mmap_size).
PRAGMA writable_schema=ON; -- error: not authorizedDisabling the sandbox
Section titled “Disabling the sandbox”The function deny-list and the PRAGMA allowlist are part of the sandbox and cannot be relaxed individually. If you genuinely need load_file, an arbitrary PRAGMA, or the database readers without restriction, you must turn the sandbox off entirely — only do this on a trusted, non-exposed deployment:
anyquery server --no-sandboxanyquery mcp --tunnel --sandbox=falseEnabling the sandbox in CLI mode
Section titled “Enabling the sandbox in CLI mode”CLI mode is unrestricted by default. Pass --sandbox to apply the same policy — useful when running untrusted SQL locally, or to reproduce the server’s behaviour:
anyquery query --sandbox --allow-dirs /var/data -q "SELECT * FROM read_csv('/var/data/report.csv')"Without --allow-dirs, a sandboxed query cannot read any file:
anyquery query --sandbox -q "SELECT * FROM read_csv('/etc/passwd')"# error: sandbox: access to "/etc/passwd" is not allowed; permitted directories: []