Skip to main content
POST
/
v1
/
transfers
Create Transfer
curl --request POST \
  --url https://api.agent-drop.com/v1/transfers \
  --header 'Authorization: <authorization>' \
  --header 'Content-Type: <content-type>' \
  --data '
{
  "sender": "<string>",
  "recipient": "<string>",
  "recipient_account": "<string>",
  "mode": "<string>",
  "is_encrypted": true,
  "message": "<string>",
  "auto_delete": true,
  "max_downloads": 123,
  "expires_in": "<string>"
}
'
{
  "id": "<string>",
  "url": "<string>",
  "api_url": "<string>",
  "status": "<string>",
  "pending_reason": {},
  "sender": "<string>",
  "recipient": "<string>",
  "files": [
    {}
  ],
  "max_downloads": 123,
  "downloads": 123,
  "total_size": 123,
  "created_at": "<string>",
  "expires_at": "<string>",
  "is_encrypted": true,
  "auto_delete": true,
  "message": {}
}

Documentation Index

Fetch the complete documentation index at: https://docs.agent-drop.com/llms.txt

Use this file to discover all available pages before exploring further.

Upload one or more files and create a transfer that a recipient agent can download.
Use the SDK, not this endpoint directly. The AgentDrop Python SDK and Node.js SDK wrap this endpoint and handle X25519 key exchange, AES-256-GCM encryption, pairwise channel derivation, and Shield scanning automatically. Calling this endpoint directly skips all of that, your files will be uploaded as plaintext and downloads will not be scanned for prompt injection or malware. The endpoint is documented here for transparency, not because raw HTTP is a supported integration path.
This endpoint accepts multipart/form-data, not JSON. Files are uploaded directly in the request body.

Request

Headers

Authorization
string
required
Bearer token. Example: Bearer agd_live_xxxxxxxxxxxxxxxxxxxx
Content-Type
string
required
Must be multipart/form-data

Body Parameters

sender
string
required
Identifier for the sending agent. Free-form string used for tracking and filtering.
recipient
string
required
Identifier for the intended recipient agent.
recipient_account
string
Disambiguator when the same agent_id exists on multiple paired accounts. Accepts the recipient’s account email, account UUID, or account display name. Required only when /v1/agents/resolve returns AMBIGUOUS_RECIPIENT for the recipient. See the Ambiguous Recipients section below.
mode
string
required
Transfer mode. One of: agent-to-agent, agent-to-human, human-to-agent.
files
file
required
One or more files to upload. Include multiple files fields for multiple files.
is_encrypted
boolean
default:"false"
Whether the uploaded files are already encrypted with AES-256-GCM (done client-side before upload). The SDK always sets this to true and encrypts before calling the endpoint. If you call the API directly with is_encrypted=false, your files are stored on our servers as plaintext, do not do this for any file that matters.
message
string
Optional message to attach to the transfer.
auto_delete
boolean
default:"true"
Whether to automatically delete the transfer after first download.
max_downloads
integer
default:"10"
Maximum number of times the transfer can be downloaded before it locks.
expires_in
string
default:"24h"
How long the transfer stays active. Examples: 1h, 12h, 24h, 7d, 30d. Maximum depends on your plan.

Response

id
string
Unique transfer ID. Example: tr_abc123
url
string
Human-readable URL for the transfer.
api_url
string
Direct API URL for programmatic access.
status
string
Transfer status. One of: active, pending_recipient, expired, deleted.pending_recipient means the recipient doesn’t have an AgentDrop account or hasn’t set a receiving password yet. They’ll be notified by email. The transfer activates automatically when they sign up and set their receiving password.
pending_reason
string|null
Present when status is pending_recipient. Explains why the transfer is pending and what the recipient needs to do.
sender
string
The sender identifier provided in the request.
recipient
string
The recipient identifier provided in the request.
files
array
Array of uploaded file objects, each containing name, size, and type.
max_downloads
integer
Maximum allowed downloads.
downloads
integer
Current download count (starts at 0).
total_size
integer
Total size of all uploaded files in bytes.
created_at
string
ISO 8601 creation timestamp.
expires_at
string
ISO 8601 timestamp when the transfer expires.
is_encrypted
boolean
Whether the files are end-to-end encrypted.
auto_delete
boolean
Whether the transfer will be automatically deleted after expiry.
message
string|null
The message attached to the transfer, or null if none was provided.

Examples

curl -X POST https://api.agent-drop.com/v1/transfers \
  -H "Authorization: Bearer agd_live_xxxxxxxxxxxxxxxxxxxx" \
  -F "sender=data-pipeline" \
  -F "recipient=analysis-agent" \
  -F "mode=agent-to-agent" \
  -F "files=@./report.pdf" \
  -F "files=@./data.csv" \
  -F "is_encrypted=true" \
  -F "message=Weekly report files" \
  -F "auto_delete=true" \
  -F "max_downloads=3" \
  -F "expires_in=12h"

Response

{
  "id": "tr_abc123",
  "url": "https://agent-drop.com/t/tr_abc123",
  "api_url": "https://api.agent-drop.com/v1/transfers/tr_abc123/download",
  "status": "active",
  "mode": "agent-to-agent",
  "sender": "data-pipeline",
  "recipient": "analysis-agent",
  "files": [
    { "name": "report.pdf", "size": 1048576, "type": "application/pdf" },
    { "name": "data.csv", "size": 245760, "type": "text/csv" }
  ],
  "max_downloads": 3,
  "downloads": 0,
  "total_size": 1294336,
  "created_at": "2026-03-22T12:00:00Z",
  "expires_at": "2026-03-22T00:00:00Z",
  "is_encrypted": true,
  "auto_delete": false,
  "message": "Weekly report files"
}

Errors

StatusCodeDescription
400VALIDATION_ERRORMissing required fields or invalid values
400AMBIGUOUS_RECIPIENTThe recipient agent_id matches multiple paired accounts. Retry with recipient_account set to an email, account UUID, or account display name from the returned candidates list.
401UNAUTHORIZEDInvalid or missing API key
413FILE_TOO_LARGEFile exceeds your plan’s max file size
429RATE_LIMITEDToo many requests. Back off and retry.
429INVITE_RATE_LIMITDaily limit (5) reached for transfers to non-registered recipients.

Ambiguous Recipients

An agent_id is unique within an account but can collide across accounts you’re paired with. If two paired accounts both use, say, claude-code-agent, the server refuses to pick one and returns:
{
  "error": {
    "code": "AMBIGUOUS_RECIPIENT",
    "message": "Multiple agents with this agent_id exist across paired accounts. Pass `recipient_account` (email or account UUID) to disambiguate.",
    "status": 400,
    "candidates": [
      {
        "account_id": "ee9b1017-4d9d-457d-9006-6f70daa6bfd6",
        "account_name": "Alex Morgan",
        "email_masked": "et***********@gmail.com"
      },
      {
        "account_id": "1f2c2188-e1dc-45fc-900f-75f5c68b8e50",
        "account_name": "Jamie Chen",
        "email_masked": "as********@gmail.com"
      }
    ]
  }
}
Retry with recipient_account in the multipart body:
curl -X POST https://api.agent-drop.com/v1/transfers \
  -H "Authorization: Bearer agd_live_xxxxxxxxxxxxxxxxxxxx" \
  -F "sender=my-agent" \
  -F "recipient=claude-code-agent" \
  -F "[email protected]" \
  -F "mode=agent-to-agent" \
  -F "files=@./report.pdf" \
  -F "is_encrypted=true"
Emails in candidates are masked; the full address is never disclosed to the sender. If the recipient’s agent_id only exists on one paired account, recipient_account is ignored.

Agent-to-Human Transfers

When sending files to a human email (mode: agent-to-human), the behavior depends on the recipient’s account status:
Recipient StatusWhat Happens
Registered + receiving password setFiles encrypted with recipient’s X25519 public key (E2E). Status: active.
Registered, no receiving passwordFiles encrypted with server key. Recipient gets a nudge email to set their password. Status: pending_recipient.
Not registeredFiles encrypted with server key. Recipient gets an invite email to create an account. Status: pending_recipient.
When a pending_recipient transfer activates (recipient signs up + sets password), files are re-encrypted with their public key and the server key is deleted. The recipient gets a “files ready” notification.
Transfers to non-registered emails are rate-limited to 5 per account per day to prevent abuse.