vnsh

Ephemeral encrypted file sharing for AI. Client-side AES-256 encryption, 24h auto-vaporization.


What is vnsh?

Stop pasting walls of text into Claude. Pipe your logs, diffs, and images into a secure, host-blind URL. The server sees nothing. The data vaporizes in 24 hours.

# Pipe anything to vnsh, get a secure link
git diff | vn
# https://vnsh.dev/v/a1b2c3d4...#k=...&iv=...

Handles any context your AI needs:

  • πŸ–ΌοΈ Screenshots β€” UI bugs, error dialogs, terminal output
  • πŸ“œ Logs β€” 5000+ lines of server errors (too long for copy-paste)
  • πŸ”„ Git Diffs β€” Complex PR reviews, multi-file changes
  • πŸ“¦ Binaries β€” PDFs, CSVs, config files, database dumps
  • πŸ”§ Debug Context β€” Stack traces, environment dumps, crash reports

Philosophy

"Built for the ephemeral nature of AI workflows. Once your session is done, the data should be too."

Unlike Dropbox or pastebins, vnsh implements a Zero-Access Architecture with automatic vaporization:

LayerWhat Happens
EncryptionAES-256-CBC encryption happens entirely on your device
TransportDecryption keys travel only in the URL fragment (#k=...) β€” never sent to servers
StorageServer stores encrypted binary blobs with zero knowledge of contents
VaporizationData auto-destructs after 24 hours. No history. No leaks.

Quick Start

Option 1: Web Upload

Visit vnsh.dev, drag & drop a file, or paste text. Get an encrypted link instantly.

Option 2: CLI Installation

NPM (recommended for Node.js users):

npm install -g vnsh-cli

Homebrew (macOS/Linux):

brew tap raullenchai/vnsh
brew install vnsh

Shell script (cross-platform: macOS, Linux, WSL, Git Bash):

curl -sL https://vnsh.dev/i | sh

CLI Usage

# Upload a file
vn secrets.env

# Pipe from stdin
cat crash.log | vn
docker logs app | vn
git diff HEAD~5 | vn

# Read/decrypt a URL
vn read "https://vnsh.dev/v/abc123#k=...&iv=..."

# Custom expiry (1-168 hours)
vn --ttl 1 temp-file.txt   # expires in 1 hour

# Show version and help
vn --version
vn --help

Option 3: Claude Code (MCP Integration)

Native to Claude Code. Unlike Dropbox, vnsh has a first-party MCP server. Claude can "see" inside your encrypted links without leaving the terminal.

Create .mcp.json in your project root:

{
  "mcpServers": {
    "vnsh": {
      "command": "npx",
      "args": ["-y", "vnsh-mcp"]
    }
  }
}

Restart Claude Code after adding the config. Now Claude can:

  • Read vnsh links automatically when you paste them
  • Share large outputs via vnsh_share tool

Option 4: GitHub Action (CI/CD)

Debug CI failures with Claude in one click. When your CI fails, automatically upload logs and post a secure link to your PR.

- name: Debug with vnsh
  if: failure()
  uses: raullenchai/upload-to-vnsh@v1
  with:
    file: test.log
  env:
    GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

The action will post a comment to your PR:

πŸ” Debug with Claude

CI logs uploaded securely. View Logs | Paste link to Claude for instant analysis

See upload-to-vnsh for full documentation.

How It Works

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚                              YOUR DEVICE                                  β”‚
β”‚  β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”    β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”    β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”  β”‚
β”‚  β”‚  Data   │───▢│ AES-256-CBC  │───▢│  Encrypted Blob + URL Fragment  β”‚  β”‚
β”‚  β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜    β”‚  Encryption  β”‚    β”‚  https://vnsh.dev/v/id#k=...    β”‚  β”‚
β”‚                 β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜    β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜  β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
                                           β”‚
                           Only encrypted blob sent to server
                           (key stays in URL fragment, never transmitted)
                                           β–Ό
β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚                           VNSH SERVER (BLIND)                             β”‚
β”‚                                                                           β”‚
β”‚   Receives: [encrypted binary blob]                                       β”‚
β”‚   Stores:   [encrypted binary blob]                                       β”‚
β”‚   Knows:    upload time, size, expiry                                     β”‚
β”‚   Cannot:   decrypt, identify content type, read keys                     β”‚
β”‚                                                                           β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

URL Structure

https://vnsh.dev/v/a1b2c3d4-5678-90ab-cdef-1234567890ab#k=<64-char-key>&iv=<32-char-iv>
                  β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜ β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
                            Sent to server                    Never sent to server
                            (blob identifier)                 (decryption material)

Self-Hosting

vnsh runs on Cloudflare Workers with R2 storage. Deploy your own instance:

Prerequisites

Deploy

# Clone the repository
git clone https://github.com/raullenchai/vnsh.git
cd vnsh/worker

# Install dependencies
npm install

# Create R2 bucket
wrangler r2 bucket create vnsh-store

# Deploy
wrangler deploy

Configuration

Edit wrangler.toml to customize:

name = "vnsh"

[[r2_buckets]]
binding = "VNSH_STORE"
bucket_name = "vnsh-store"  # Your R2 bucket name

[[kv_namespaces]]
binding = "VNSH_META"
id = "your-kv-namespace-id"  # Create with: wrangler kv namespace create VNSH_META

API Reference

POST /api/drop

Upload an encrypted blob.

curl -X POST https://vnsh.dev/api/drop \
  -H "Content-Type: application/octet-stream" \
  --data-binary @encrypted.bin

Query Parameters:

ParameterTypeDescription
ttlnumberTime-to-live in hours (1-168, default: 24)
pricenumberPayment required to access (x402 protocol)

Response:

{
  "id": "a1b2c3d4-5678-90ab-cdef-1234567890ab",
  "expires": "2025-01-25T00:00:00.000Z"
}

GET /api/blob/:id

Download an encrypted blob.

curl https://vnsh.dev/api/blob/a1b2c3d4-5678-90ab-cdef-1234567890ab

Response Codes:

CodeDescription
200Success β€” returns encrypted blob
402Payment required
404Not found
410Expired

GET /v/:id

Web viewer (serves HTML directly to preserve URL fragment with encryption keys).

GET /i

CLI installation script.

Security Model

What vnsh Protects Against

βœ… Server Compromise β€” Even with full server access, attackers cannot decrypt blobs βœ… Database Leaks β€” Stored data is indistinguishable from random noise βœ… Traffic Analysis β€” No content-type information stored βœ… Subpoenas β€” Server operator cannot produce plaintext (doesn't have keys)

What vnsh Does NOT Protect Against

❌ URL Sharing β€” Anyone with the full URL (including #fragment) can decrypt ❌ Client Compromise β€” Malware on your device can intercept before encryption ❌ MITM on Upload Page β€” An attacker serving malicious JavaScript could intercept

Recommendations

  • Use vnsh over HTTPS only
  • Don't share full URLs in public channels (Slack, Discord, Twitter)
  • For maximum security, self-host the worker

Project Structure

vnsh/
β”œβ”€β”€ worker/          # Cloudflare Worker (storage API)
β”‚   β”œβ”€β”€ src/
β”‚   β”‚   └── index.ts # Main worker code
β”‚   └── test/
β”‚       └── api.test.ts
β”œβ”€β”€ mcp/             # MCP Server (Claude Code integration)
β”‚   β”œβ”€β”€ src/
β”‚   β”‚   β”œβ”€β”€ index.ts # MCP tool handlers
β”‚   β”‚   └── crypto.ts # Encryption utilities
β”‚   └── package.json
β”œβ”€β”€ cli/
β”‚   β”œβ”€β”€ vn           # Bash CLI script
β”‚   β”œβ”€β”€ npm/         # NPM package (vnsh-cli)
β”‚   β”‚   β”œβ”€β”€ src/
β”‚   β”‚   β”‚   β”œβ”€β”€ cli.ts
β”‚   β”‚   β”‚   └── crypto.ts
β”‚   β”‚   └── package.json
β”‚   └── install.sh   # Shell installer
β”œβ”€β”€ homebrew-tap/    # Homebrew formula
β”‚   └── Formula/
β”‚       └── vnsh.rb
└── docs/            # Documentation

Packages

PackageDescriptionInstall
vnsh-cliCLI toolnpm i -g vnsh-cli
vnsh-mcpMCP server for Claudenpx vnsh-mcp
upload-to-vnshGitHub Action for CI/CDuses: raullenchai/upload-to-vnsh@v1
homebrew-vnshHomebrew tapbrew install raullenchai/vnsh/vnsh

Development

# Clone
git clone https://github.com/raullenchai/vnsh.git
cd vnsh

# Install dependencies
npm install
cd worker && npm install
cd ../mcp && npm install

# Run tests (143 tests, 82%+ coverage)
npm test

# Start local worker
cd worker && npm run dev

# Build MCP server
cd mcp && npm run build

Contributing

Contributions are welcome! Please read our Contributing Guide before submitting a PR.

Development Workflow

  1. Fork the repository
  2. Create a feature branch (git checkout -b feature/amazing-feature)
  3. Run tests (npm test)
  4. Commit changes (git commit -m 'Add amazing feature')
  5. Push to branch (git push origin feature/amazing-feature)
  6. Open a Pull Request

License

MIT License β€” see LICENSE for details.


Related Servers