José Alvarez
José Alvarez7mo ago

Why is it that file names only appear when uploading through the console?

When I upload files using a script equal to the one in the documentation, the file name does not seem to be stored, as you can see from the screen shot. But when I upload the file manually through the console, it does seem to get the file name store (and with the proper extension). I don't see a parameter in the documentation to specify the file name. I also want to note that when I log the file information in the console, it does seem to have the file name in it, so I wonder why the information doesn't get to the Convex cloud. React logic
const [arrayBuffer, setArrayBuffer] = useState<ArrayBuffer | null>(null);

// Convex mutations
const generateUploadUrl = useMutation(api.stlFiles.generateUploadUrl);
const sendFile = useMutation(api.stlFiles.sendFile);

// File handling and upload
async function handleFile(file: File) {
console.log("File information", file);

// Get a short-lived upload URL
const postUrl = await generateUploadUrl();

// Is there a way to specify the file name??? (e.g., "aneurysm.stl")
const init = {
method: "POST",
headers: {
// note: I don't think the Content-Type
// has anything to do with the issue
// I've tried multiple things, from "" (gives bad header error)
// to "application/octect-stream" (uploads the file to the Convex cloud but,
// again, withou the name)
"Content-Type": "application/sla",
},
// As shown in the screenshots, the file name property (among others) is actually
// in the object when logged in the browser, but it doesn't get to the Convex
// cloud :(
body: file,
};

// POST the file to the URL
const response = await fetch(postUrl, init);
const result = await response.json();

// Save the newly allocated storage id to the database
const { storageId } = result;
await sendFile({ storageId, author: "me" });

setArrayBuffer(await file.arrayBuffer());
}
const [arrayBuffer, setArrayBuffer] = useState<ArrayBuffer | null>(null);

// Convex mutations
const generateUploadUrl = useMutation(api.stlFiles.generateUploadUrl);
const sendFile = useMutation(api.stlFiles.sendFile);

// File handling and upload
async function handleFile(file: File) {
console.log("File information", file);

// Get a short-lived upload URL
const postUrl = await generateUploadUrl();

// Is there a way to specify the file name??? (e.g., "aneurysm.stl")
const init = {
method: "POST",
headers: {
// note: I don't think the Content-Type
// has anything to do with the issue
// I've tried multiple things, from "" (gives bad header error)
// to "application/octect-stream" (uploads the file to the Convex cloud but,
// again, withou the name)
"Content-Type": "application/sla",
},
// As shown in the screenshots, the file name property (among others) is actually
// in the object when logged in the browser, but it doesn't get to the Convex
// cloud :(
body: file,
};

// POST the file to the URL
const response = await fetch(postUrl, init);
const result = await response.json();

// Save the newly allocated storage id to the database
const { storageId } = result;
await sendFile({ storageId, author: "me" });

setArrayBuffer(await file.arrayBuffer());
}
No description
No description
10 Replies
Convex Bot
Convex Bot7mo ago
Thanks for posting in <#1088161997662724167>. Reminder: If you have a Convex Pro account, use the Convex Dashboard to file support tickets. - Provide context: What are you trying to achieve, what is the end-user interaction, what are you seeing? (full error message, command output, etc.) - Use search.convex.dev to search Docs, Stack, and Discord all at once. - Additionally, you can post your questions in the Convex Community's <#1228095053885476985> channel to receive a response from AI. - Avoid tagging staff unless specifically instructed. Thank you!
José Alvarez
José AlvarezOP7mo ago
Convex logic (it's basically the same as the docs https://docs.convex.dev/file-storage/upload-files)
import { v } from "convex/values";
import { mutation } from "./_generated/server";

export const generateUploadUrl = mutation(async (ctx) => {
return await ctx.storage.generateUploadUrl();
});

export const sendFile = mutation({
args: { storageId: v.id("_storage"), author: v.string() },
handler: async (ctx, args) => {
await ctx.db.insert("stlFiles", {
body: args.storageId,
author: args.author,
format: "stl",
});
},
});
import { v } from "convex/values";
import { mutation } from "./_generated/server";

export const generateUploadUrl = mutation(async (ctx) => {
return await ctx.storage.generateUploadUrl();
});

export const sendFile = mutation({
args: { storageId: v.id("_storage"), author: v.string() },
handler: async (ctx, args) => {
await ctx.db.insert("stlFiles", {
body: args.storageId,
author: args.author,
format: "stl",
});
},
});
ari
ari7mo ago
Hiya, the UI shows the filename as a convenience, but it still isn't stored. Once you refresh the page, the filename will no longer be rendered
ian
ian7mo ago
You can upload to a custom http action where you can store the file and save any other metadata at the same time. Having a table with file names and storage ids and other metadata is a good approach, as you’ll likely want to store other things too, like the user id of the uploader, the size of the image, etc.
José Alvarez
José AlvarezOP7mo ago
Thanks, I understand now. Wondering if you guys have considered the creation of buckets in the feature roadmap?
ian
ian7mo ago
I think buckets would be a great Component / library. Any reason it'd need to be built in? Anything you can think of that can't be done by storing metadata in a table?
Convex: The Software-Defined Database
Which to choose, the expressive power of code, or the robustness of built-in database features? With Convex, you can have both. By eliminating the bou...
José Alvarez
José AlvarezOP7mo ago
Well, if you put it that way, I guess it's not an absolute necessity. I come from a background of using Google Cloud Storage a lot, and even S3, so it's what I'm accustomed. I've even used Vercel's storage version and it works similar. But I'm guessing that if Convex wants to keep it simple, it's really not needed.
ian
ian7mo ago
Yeah it'd be great to have a few options folks can choose from that have different ergonomics - one that simulates a file system, another that works with S3 bucket mentality (possibly including access rules), another backed by GCS possibly (maybe plugging into existing buckets)... keeping the core of Convex simple will hopefully keep all of the options open, but does require building that higher level abstraction. wdyt?
José Alvarez
José AlvarezOP7mo ago
What I appreciated the most was the feeling that I was uploading stuff directly to Convex "through the front-end". Now this is obviously not the case, but Convex makes it so easy to set up the backend and handles it all that it was the feeling I had from a developer experience. That is a huge plus because it lets me focus on my app and not on integrating the libraries or REST APIs of other cloud providers on my backend routes I may have more feedback in 1 or 2 weeks as I finish building the app I am working on, and I plan to use convex all the way for cloud storage
ian
ian7mo ago
Looking forward to it, thanks! And by "through the front-end" you mean your app, not the dashboard, right?

Did you find this page helpful?