.agents/skills/bknd-file-upload/SKILL.md
Use when uploading files to Bknd storage. Covers MediaApi SDK methods (upload, uploadToEntity), REST endpoints, React integration with file inputs, progress tracking with XHR, browser upload patterns, and entity field attachments.
npx skillsauth add cameronapak/cultivate-fellowship bknd-file-uploadInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Upload files to Bknd storage using the MediaApi SDK or REST endpoints.
bknd package installedmedia() field definedNavigate to http://localhost:7654 (or your Bknd URL).
Click "Media" in sidebar to access file management.
Files appear in the list with name, size, type, and date.
import { Api } from "bknd";
const api = new Api({ host: "http://localhost:7654" });
// From File object (browser)
const input = document.querySelector('input[type="file"]');
const file = input.files[0];
const { ok, data, error } = await api.media.upload(file);
if (ok) {
console.log("Uploaded:", data.name);
console.log("Size:", data.meta.size);
console.log("Type:", data.meta.type);
}
const { ok, data } = await api.media.upload(file, {
filename: "custom-name.png",
});
// Direct URL string
const { ok, data } = await api.media.upload("https://example.com/image.png");
// Or from fetch Response
const response = await fetch("https://example.com/image.png");
const { ok, data } = await api.media.upload(response);
Attach file directly to an entity record:
// Assumes "posts" entity has a media() field called "cover_image"
const { ok, data } = await api.media.uploadToEntity(
"posts", // entity name
123, // record ID
"cover_image", // field name
file,
{ overwrite: true } // replace existing file
);
if (ok) {
console.log("File attached:", data.result.cover_image);
}
function FileUpload({ onUploaded }) {
const { api } = useApp(); // or your Api instance
const [uploading, setUploading] = useState(false);
const [error, setError] = useState(null);
const handleChange = async (e) => {
const file = e.target.files?.[0];
if (!file) return;
setUploading(true);
setError(null);
const { ok, data, error } = await api.media.upload(file);
setUploading(false);
if (ok) {
onUploaded(data);
} else {
setError(error?.message || "Upload failed");
}
};
return (
<div>
<input
type="file"
onChange={handleChange}
disabled={uploading}
/>
{uploading && <span>Uploading...</span>}
{error && <span style={{ color: "red" }}>{error}</span>}
</div>
);
}
function ImageUpload({ value, onChange }) {
const { api } = useApp();
const [preview, setPreview] = useState(value);
const [uploading, setUploading] = useState(false);
const handleChange = async (e) => {
const file = e.target.files?.[0];
if (!file) return;
// Show local preview immediately
const localUrl = URL.createObjectURL(file);
setPreview(localUrl);
setUploading(true);
const { ok, data } = await api.media.upload(file);
setUploading(false);
if (ok) {
// Clean up local preview
URL.revokeObjectURL(localUrl);
// Use server URL
onChange(data.name);
}
};
return (
<div>
{preview && (
<img
src={preview}
alt="Preview"
style={{ maxWidth: 200, opacity: uploading ? 0.5 : 1 }}
/>
)}
<input type="file" accept="image/*" onChange={handleChange} />
{uploading && <span>Uploading...</span>}
</div>
);
}
For large files, use XHR for progress:
function ProgressUpload({ onUploaded }) {
const { api } = useApp();
const [progress, setProgress] = useState(0);
const [uploading, setUploading] = useState(false);
const uploadWithProgress = (file) => {
return new Promise((resolve, reject) => {
const xhr = new XMLHttpRequest();
const url = api.media.getFileUploadUrl({ path: file.name });
xhr.upload.addEventListener("progress", (e) => {
if (e.lengthComputable) {
setProgress(Math.round((e.loaded / e.total) * 100));
}
});
xhr.addEventListener("load", () => {
if (xhr.status >= 200 && xhr.status < 300) {
resolve(JSON.parse(xhr.responseText));
} else {
reject(new Error(xhr.statusText));
}
});
xhr.addEventListener("error", () => reject(new Error("Upload failed")));
xhr.open("POST", url);
xhr.setRequestHeader("Content-Type", file.type);
// Add auth if using header transport
const token = api.getAuthState().token;
if (token) {
xhr.setRequestHeader("Authorization", `Bearer ${token}`);
}
xhr.send(file);
});
};
const handleChange = async (e) => {
const file = e.target.files?.[0];
if (!file) return;
setUploading(true);
setProgress(0);
try {
const result = await uploadWithProgress(file);
onUploaded(result);
} catch (err) {
console.error("Upload failed:", err);
} finally {
setUploading(false);
}
};
return (
<div>
<input type="file" onChange={handleChange} disabled={uploading} />
{uploading && (
<div>
<progress value={progress} max={100} />
<span>{progress}%</span>
</div>
)}
</div>
);
}
Complete avatar upload with entity attachment:
function AvatarUpload({ userId, currentAvatar }) {
const { api } = useApp();
const [preview, setPreview] = useState(currentAvatar);
const [uploading, setUploading] = useState(false);
const handleChange = async (e) => {
const file = e.target.files?.[0];
if (!file) return;
// Validate image
if (!file.type.startsWith("image/")) {
alert("Please select an image file");
return;
}
if (file.size > 5 * 1024 * 1024) {
alert("Image must be under 5MB");
return;
}
setPreview(URL.createObjectURL(file));
setUploading(true);
const { ok, data } = await api.media.uploadToEntity(
"users",
userId,
"avatar",
file,
{ overwrite: true }
);
setUploading(false);
if (ok) {
setPreview(data.result.avatar);
}
};
return (
<div>
<img
src={preview || "/default-avatar.png"}
alt="Avatar"
style={{
width: 100,
height: 100,
borderRadius: "50%",
opacity: uploading ? 0.5 : 1
}}
/>
<label>
<input
type="file"
accept="image/*"
onChange={handleChange}
disabled={uploading}
style={{ display: "none" }}
/>
<button type="button" disabled={uploading}>
{uploading ? "Uploading..." : "Change Avatar"}
</button>
</label>
</div>
);
}
# Basic upload (filename in path)
curl -X POST \
-H "Authorization: Bearer $TOKEN" \
-H "Content-Type: image/png" \
--data-binary @image.png \
http://localhost:7654/api/media/upload/image.png
# Upload to entity field
curl -X POST \
-H "Authorization: Bearer $TOKEN" \
-H "Content-Type: image/jpeg" \
--data-binary @photo.jpg \
"http://localhost:7654/api/media/entity/users/123/avatar?overwrite=true"
{
"name": "image.png",
"meta": {
"type": "image/png",
"size": 24680,
"width": 800,
"height": 600
},
"etag": "abc123...",
"state": {
"name": "image.png",
"path": "image.png"
}
}
Define a media field to link files to records:
import { em, entity, text, media } from "bknd";
const schema = em({
posts: entity("posts", {
title: text(),
cover_image: media(), // Stores file reference
}),
users: entity("users", {
name: text(),
avatar: media(),
}),
});
Media fields store the filename, not the file content.
Upload files in Bknd server context (seeds, flows):
// In seed function
export default defineConfig({
options: {
seed: async (ctx) => {
const media = ctx.server?.modules.media;
if (!media) return;
// Upload from URL
const response = await fetch("https://example.com/default-avatar.png");
const buffer = await response.arrayBuffer();
await media.adapter.putObject(
"default-avatar.png",
new Uint8Array(buffer)
);
},
},
});
const { ok, data, error } = await api.media.upload(file);
if (!ok) {
// Handle error
if (error?.status === 413) {
console.error("File too large");
} else if (error?.status === 401) {
console.error("Not authenticated");
} else if (error?.status === 403) {
console.error("No upload permission");
} else {
console.error("Upload failed:", error?.message);
}
return;
}
// Success - use data
console.log("Filename:", data.name);
console.log("MIME type:", data.meta.type);
console.log("Size (bytes):", data.meta.size);
// For images
if (data.meta.width && data.meta.height) {
console.log("Dimensions:", data.meta.width, "x", data.meta.height);
}
Problem: Upload fails with 413 status.
Fix: Increase body_max_size in config:
export default defineConfig({
media: {
enabled: true,
body_max_size: 50 * 1024 * 1024, // 50MB
adapter: { ... },
},
});
Problem: File uploaded with wrong MIME type.
Fix: Always set Content-Type in REST uploads:
# WRONG - no content type
curl -X POST --data-binary @image.png .../upload/image.png
# CORRECT
curl -X POST \
-H "Content-Type: image/png" \
--data-binary @image.png \
.../upload/image.png
SDK handles this automatically from File object.
Problem: Browser blocks upload to S3.
Fix: Configure CORS on storage bucket (not Bknd):
{
"CORSRules": [{
"AllowedOrigins": ["https://yourapp.com"],
"AllowedMethods": ["GET", "PUT", "POST"],
"AllowedHeaders": ["*"]
}]
}
Problem: 401 error when using XHR progress upload.
Fix: Add Authorization header:
const token = api.getAuthState().token;
if (token) {
xhr.setRequestHeader("Authorization", `Bearer ${token}`);
}
Problem: uploadToEntity fails with "field not found".
Fix: Ensure entity has media() field:
// WRONG - no media field
posts: entity("posts", {
title: text(),
cover_image: text(), // This is just a string
}),
// CORRECT
posts: entity("posts", {
title: text(),
cover_image: media(), // Proper media field
}),
Problem: Browser crashes on large file upload.
Fix: Use streaming or chunked upload:
// For very large files, consider chunked upload
// Bknd doesn't have native chunking, so use S3 presigned URLs
// or implement custom chunking endpoint
Test upload works correctly:
async function testUpload() {
// 1. Check media enabled
const { ok: listOk } = await api.media.listFiles();
console.log("Media module enabled:", listOk);
// 2. Test file upload
const testFile = new File(["test"], "test.txt", { type: "text/plain" });
const { ok, data, error } = await api.media.upload(testFile);
console.log("Upload result:", ok ? data.name : error);
// 3. Verify file exists
const { data: files } = await api.media.listFiles();
const exists = files.some(f => f.key === "test.txt");
console.log("File in list:", exists);
// 4. Clean up
if (exists) {
await api.media.deleteFile("test.txt");
console.log("Test file deleted");
}
}
DO:
uploadToEntity for entity attachmentsbody_max_size limitDON'T:
development
Use btca (Better Context App) to efficiently query and learn from the bknd backend framework. Use when working with bknd for (1) Understanding data module and schema definitions, (2) Implementing authentication and authorization, (3) Setting up media file handling, (4) Configuring adapters (Node, Cloudflare, etc.), (5) Learning from bknd source code and examples, (6) Debugging bknd-specific issues
development
Use when configuring webhook integrations in Bknd. Covers receiving incoming webhooks via HTTP triggers, sending outgoing webhooks with FetchTask, event-triggered webhooks on data changes, signature verification, retry patterns, and async processing.
development
Use when encountering Bknd errors, getting error messages, something not working, or needing quick fixes. Covers error code reference, quick solutions, and common mistake patterns.
tools
Use when writing tests for Bknd applications, setting up test infrastructure, creating unit/integration tests, or testing API endpoints. Covers in-memory database setup, test helpers, mocking, and test patterns.