The full Venice Forge security model is maintained in SECURITY.md.
Do not include exploit details, API keys, tokens, or private user data in a public issue.
Use one of these routes:
- GitHub private vulnerability reporting — If enabled, use the Security → Report a Vulnerability workflow on the repository. This keeps the report private until a fix is coordinated and is the preferred channel.
- Issue label routing — If private reporting is not available, open a
GitHub issue labeled
securityand request a private maintainer discussion in the issue body. Do not post exploit details publicly.
The maintainer will triage reports for supported versions and coordinate disclosure before any public details are posted.
Only the latest release tag is actively maintained. Older versions do not receive security patches.
If you encounter unsafe content, safety guard bypasses, or AI-generated material that inappropriately represents minors (CSAM), report it immediately:
- NCMEC CyberTipline: If the material involves child exploitation, report it directly to the National Center for Missing & Exploited Children (NCMEC).
- Venice.ai Trust & Safety: Report the incident to Venice.ai through their official support channels at venice.ai/support.
- Repository Maintainers: Report bypasses of the Venice Forge safety guard using GitHub's private vulnerability reporting feature in this repository.
Venice Forge strictly requires users to be 18 years or older. The application connects to unrestricted AI endpoints that may generate explicit or sensitive content. Due to the inherent risk of producing AI-generated images that may inappropriately represent minors (CSAM), use of this software by minors is strictly prohibited. Users assume all responsibility for the generated content.
When Family Safe Mode is enabled (the default), outgoing requests are screened by Venice Forge's local family-safe guard
(src/shared/safety/childExploitationGuard.ts) before the payload is
forwarded. The conditional pipeline runs at every enforcement boundary — the renderer transport
(src/services/veniceClient.ts), Electron IPC handlers
(electron/ipc/handlers.ts), the Express web proxy (server.ts), research
and Jina dispatch (src/components/search/SearchScrapeView.tsx,
src/research/agent/researchRunner.ts), and RP/scene prompt paths
(src/shared/safety/characterImportSafety.ts,
src/services/rp/sceneGenerationService.ts). The guard implements advanced features such as
cross-sentence context detection and negative_prompt extraction. The proxy
operates on a "fail-close" design (returning a 500 status) if the guard
encounters any extraction errors. Raw prompt text is never logged by the safety
system. When Family Safe Mode is disabled, Adult Mode skips the local rule engine entirely. This does not disable or alter Venice's provider-side moderation or the independent Venice API Safe Mode parameter.
The Jina research provider (r.jina.ai/s.jina.ai) and generic HTTP scrape
traffic route through renderer-side safety wrappers in
src/components/search/SearchScrapeView.tsx before any research dispatch,
ensuring this path is also guarded.
Web Deployment Warning: In web mode, the web proxy defaults Local Family Safe Mode to ON. The client-sent
X-Venice-Forge-Family-Safe-Modeheader is ignored unless the server-side environment variableVENICE_FORGE_ALLOW_CLIENT_SAFETY_OVERRIDE=trueis explicitly set. The authoritative server override is the environment variableVENICE_FORGE_LOCAL_FAMILY_SAFE_MODE_ENABLED. The client header is dev-only compatibility plumbing, not a production safety boundary. Use Electron/local desktop mode for owner-controlled Family Safe Mode behavior. Provider/API restrictions may still apply regardless of Adult Mode.
Safety-guard enforcement is verified by scripts/verify-safety-guard.cjs,
which checks the core enforcement boundaries (src/services/veniceClient.ts,
electron/ipc/handlers.ts, server.ts) plus current research/Jina dispatch
paths (src/components/search/SearchScrapeView.tsx,
src/research/agent/researchRunner.ts, src/research/providers/veniceResearchProvider.ts,
src/research/providers/jinaResearchProvider.ts) as a CI gate. Run it with:
npm run verify:safety-guardA comprehensive safety guard audit was conducted in May–June 2026 covering all
request paths, payload extraction coverage, logging/diagnostics behavior, static
verification script robustness, and test fixture safety. Current findings and
validation evidence are tracked in docs/summary_of_work.md.
Maintainer trigger: Update this document whenever the allowed Venice API endpoint list (
src/shared/validation.ts) or the safety guard enforcement boundaries change.
When started with --headless, the application runs an Express loopback bridge server (electron/services/bridgeServer.ts). The following safety measures are strictly enforced:
- Loopback-Only Interface: The bridge server binds strictly to the local loopback interface (
127.0.0.1). Binding to any public/LAN interface is blocked by default to prevent network-level credential reuse or SSRF attacks. - Token Authorization: Every request must provide a
Bearertoken in theAuthorizationheader. IfVENICE_BRIDGE_TOKENis not set in the environment, a cryptographically secure 32-byte hex token is generated at boot and logged to standard output. - Family Safe Mode: Prompt-carrying requests are checked in the Main process when Family Safe Mode is enabled. The headless bridge reads
safety.local_family_safe_mode_enabledfrom the validated YAML config. Adult Mode skips the local check. - Runtime snapshot is the source of truth: Every Venice-touching IPC handler routes through
performGuardedVeniceRequest/checkLocalFamilyGuardinelectron/services/guardPipeline.ts. The renderer-suppliedlocalFamilySafeModeEnabledfield is no longer trusted; the canonical toggle state lives inelectron/services/runtimeSafetySettings.tsand is initialised from the validated YAML at boot. All entry points (chat, image, audio, video, embeddings, augment, Jina, scrape, research context) emit the same canonical 451 block shape. See VERIFY-015 intests/safety/guardPipeline.test.ts. - Renderer hydration gate: Renderer-side preflight guards (
saveCharacterCard,savePersona,appendRpMessage,generateScene) route the toggle value throughgetEffectiveRendererLocalFamilySafeModeEnabled/getEffectiveRendererVeniceApiSafeModeinsrc/safetyHydration.ts. In Electron mode these helpers throwConfigNotHydratedErroruntil the main-process config snapshot has hydrated into the renderer, preventing the renderer from making a preflight decision that disagrees with the canonical main-process state. The RP Studio orchestrator disables safety-sensitive save controls while the hydration is pending and surfaces a banner explaining the wait. See VERIFY-017 intests/safety/hydrationGate.test.ts. - Returned body screening: Jina (
/api/proxy-jina) and generic HTTP scrape (/api/proxy-scrape) web-proxy endpoints now screen the response body throughscreenResponseBody()before forwarding to the renderer. Large bodies are sampled against an 8 KiB window. A detected block is converted into the same 451 shape as a request-side block.
The Developer Traffic Inspector logs request/response diagnostics to the renderer store (src/stores/inspector-store.ts).
- Secret Masking: To prevent exposing keys to local logs, the UI, or export functions, all header lists are sanitized. Header keys matching
Authorization,Cookie,x-api-key, or names containingkeyortokenare automatically replaced with******. - Sandbox Red-Team Mode: The Red-Team Mode switch disables renderer Markdown/HTML formatting to prevent template injection or rendering exploits from unsafe model outputs, and shows the raw text directly alongside local safety audit signals.
External URLs opened via shell.openExternal are validated by
electron/utils/urlSecurity.ts: only https: with public routable hostnames
is allowed. RFC 1918 and loopback addresses are blocked.
Venice Forge stores API keys using OS-provided encryption where available:
- Windows:
safeStorage(DPAPI) - macOS:
safeStorage(Keychain)
Both the Venice API key and the optional Jina API key use the same storage policy and file (secure-prefs.json).
For Windows and macOS, there is no plaintext fallback. The application will refuse to save any API key if OS-level encryption is unavailable.
For Linux and other platforms, a plaintext fallback may be permitted if the VENICE_FORGE_ALLOW_PLAINTEXT_KEY_STORAGE=true environment variable is explicitly set in the process environment (e.g., .env for web mode development, or the shell environment for Electron).
The optional config.yaml and themes.yaml files are a bootstrap mechanism, not a key store. Their security model is:
- Renderer never sees raw API keys. The IPC channel
config:getreturns onlysecrets.has_venice_api_key: booleanandsecrets.has_jina_api_key: boolean. Raw values never cross the IPC boundary. - Plaintext keys in the YAML are imported into
safeStorageon startup. They are then redacted from the file unlesssecrets.keep_plaintext_keys: trueis explicitly set. - Default generated config files never contain real keys. The shipped example templates ship with empty strings.
- Existing secure-store keys are not overwritten unless
developer.force_import_keys: trueis set. - Local secret files are gitignored.
.config/*.yamlis ignored;!.config/*.example.yamlis re-included to keep the templates tracked. - Path values must be local. Any value with a
scheme://(e.g.https://,file://) is rejected by the schema and replaced with the default. - Generic patches cannot set plaintext keys.
config:writeSanitizedalways stripssecrets.*regardless of the patch payload; UI-entered keys still flow through the dedicatedapiKey:set/jinaApiKey:setIPC channels. - No remote URLs. The config service never makes network calls; the schema blocks any URL-like path.
- Explicit local-mode control.
safety.local_family_safe_mode_enableddefaults to true. Setting it false selects Adult Mode and skips local rule evaluation.safety.venice_api_safe_modeis separate and controls only the provider request parameter.
- Jina AI: Electron sends requests from the main process using the OS-secure Jina key. Web mode sends requests through the Express proxy using only the server-side
JINA_API_KEY; renderer-supplied Jina credential headers are dropped. Keys are redacted from logs, diagnostics, and exports. A renderer-layer safety guard runs before dispatch (see Content Safety above). - Generic HTTP: Disabled by default. When enabled, it routes traffic through a backend proxy to perform DNS resolution and enforce strict SSRF blocklists on the resolved IP. Only allows
text/html,text/plain,application/xhtml+xml, andapplication/jsonresponses. - All research traffic respects the same endpoint allowlist and safety guard as Venice API calls.
The security model does not protect against the following:
- Unsigned Windows SmartScreen warnings.
- Unsigned macOS Gatekeeper warnings.
- Local malware or debuggers running under the same OS user account.
- Keychain/session compromise if the OS user is compromised.
Clarification: macOS Gatekeeper and quarantine flags are mechanisms for distribution trust and execution prevention. They are not app data encryption mechanisms.
Dependencies are audited with npm audit before each release. To run a
manual audit:
npm auditA clean audit at the moderate level or higher (npm audit --audit-level=moderate) is a release gate requirement.
npm run verify:safety-guard is also a mandatory release and commit security gate — see the Content Safety section above for details.
The tracked .github/workflows/codeql.yml workflow automatically runs CodeQL analysis on pull requests, pushes to main, and on a schedule. You can opt-out by setting the repository variable VENICE_FORGE_DISABLE_CODEQL=true. Findings appear in
Security → Code Scanning.
All CodeQL findings in server.ts and the GitHub Actions workflows are either
fixed or dismissed with justification. The two defended false positives are
annotated at the call site with // nosec:js/<rule-id> plus an inline
justification:
server.ts:299-305—js/resource-exhaustion:setTimeoutduration isMath.min(timeoutMs, 180000) || 30000(3-minute max). CodeQL does not see the clamp because it lives inside a conditional expression.server.ts:684-717—js/request-forgery: The scrape URL is parsed, restricted tohttp:/https:, rejected for private hostnames, resolved with DNS, and every returned A/AAAA record is checked for private IP ranges before the upstream request is made.
If a future CodeQL update flags these sites again, the suppressions and allowlist check should be re-verified, not removed.
All third-party Actions are pinned to a commit SHA, not a tag, to prevent supply-chain attacks via a compromised tag update. The version comment is appended after the SHA for maintainer reference:
actions/checkout@34e114876b0b11c390a56381ad16ebd13914f8d5(v4.2.2)actions/setup-node@49933ea5288caeca8642d1e84afbd3f7d6820020(v4.4.0)actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02(v4.6.2)actions/download-artifact@d3f86a106a0bac45b974a628896c90dbdf5c8093(v4.3.0)softprops/action-gh-release@b4309332981a82ec1c5618f44dd2e27cc8bfbfda(v3.0.0)github/codeql-action/*@dd903d2e4f5405488e5ef1422510ee31c8b32357(v3)actions/dependency-review-action@2031cfc080254a8a887f58cffee85186f0e49e48(v4.9.0)
When bumping any pinned action, look up the new SHA via
gh api repos/<owner>/<repo>/git/refs/tags/<tag> and update both the SHA
and the version comment. Dependabot's github-actions ecosystem entry in
.github/dependabot.yml keeps the SHAs current.