const response1 = await channel.sendImage(
file1,
file1.name,
file1.type,
undefined,
// Optional
{
onUploadProgress: (e) => {
const pct = e.total ? Math.round((100 * e.loaded) / e.total) : 0;
console.log(`Image upload: ${pct}%`);
},
},
);
const response2 = await channel.sendFile(
file2,
file2.name,
file2.type,
undefined,
// Optional
{
onUploadProgress: (e) => {
const pct = e.total ? Math.round((100 * e.loaded) / e.total) : 0;
console.log(`File upload: ${pct}%`);
},
},
);
await channel.sendMessage({
text: "Check these out",
attachments: [
{
type: "image",
asset_url: response1.file,
thumb_url: response1.file,
},
{
type: "file",
asset_url: response2.file,
title: file2.name,
},
],
});Image and File Uploads
Stream Chat allows you to upload images, videos, and other files to the Stream CDN or your own CDN. Uploaded files can be used as message attachments, user avatars, or channel images.
Stream's UI SDKs (React, React Native, Flutter, SwiftUI, Jetpack Compose, etc.) handle file uploads automatically through their message composer components. The upload process, progress tracking, and attachment handling are built into these components. Use the methods described on this page only if you need custom upload behavior or are building a custom UI.
Uploading Files to a Channel
Files uploaded to a channel can be attached to messages. You can either upload a file first and then attach it to a message, or let the SDK handle the upload when sending a message with attachments.
// Option 1: Send message with local attachments (SDK handles upload)
final message = Message(text: "Hello", attachments: [
Attachment(
type: "image",
file: AttachmentFile(path: "imagePath/imageName.png"),
),
]);
await channel.sendMessage(message);
// Option 2: Upload first, then attach to message
await client.sendImage(
image,
channelId,
channelType,
onSendProgress: (sent, total) {
// Update progress UI
},
).then((response) {
final imageUrl = response.file;
final message = Message(attachments: [
Attachment(type: "image", imageUrl: imageUrl),
]);
client.sendMessage(message, channelId, channelType);
});Uploading Standalone Files
Files not tied to a specific channel can be used for user avatars, channel images, or other application needs.
final client = StreamChatClient("api-key");
// Upload an image
final image = AttachmentFile(path: "imagePath/image.png");
final response = await client.uploadImage(
image,
onUploadProgress: (count, total) {
// Update progress UI
},
);
final imageUrl = response.file;
// Use the URL (e.g., update user avatar)
final user = User(id: "user-id", image: imageUrl);
await client.updateUser(user);Deleting Files
Delete uploaded files to free storage space. Deleting a file from the CDN does not remove it from message attachments that reference it.
// Delete from channel
await channel.deleteFile("file-url");
await channel.deleteImage("image-url");
// Delete standalone file
await client.removeFile("file-url");
await client.removeImage("image-url");File Requirements
Images
| Requirement | Value |
|---|---|
| Supported formats | BMP, GIF, JPEG, PNG, WebP, HEIC, HEIC-sequence, HEIF, HEIF-sequence, SVG+XML |
| Maximum file size | 100 MB |
Other Files
| Requirement | Value |
|---|---|
| Supported formats | All file types are allowed by default. Different clients may handle certain types differently. |
| Maximum file size | 100 MB |
You can configure a more restrictive list of allowed file types for your application.
Configuring Allowed File Types
Stream allows all file extensions by default. To restrict allowed file types:
- Dashboard: Go to Chat Overview > Upload Configuration
- API: Use the App Settings endpoint
Access Control and Link Expiration
Stream CDN URLs include a signature that validates access to the file. Only channel members can access files uploaded to that channel.
| Behavior | Description |
|---|---|
| Access control | URLs are signed and only accessible by channel members |
| Link expiration | URLs expire after 14 days |
| Automatic refresh | Links are refreshed automatically when messages are retrieved (e.g., when querying a channel) |
| Manual refresh | Call getMessage to retrieve fresh URLs for expired attachments |
To check when a link expires, examine the Expires query parameter in the URL (Unix timestamp).
Image Resizing
Append query parameters to Stream CDN image URLs to resize images on the fly.
| Parameter | Type | Values | Description |
|---|---|---|---|
| w | number | Width in pixels | |
| h | number | Height in pixels | |
| resize | string | clip, crop, scale, fill | Resizing mode |
| crop | string | center, top, bottom, left, right | Crop anchor position |
Images can only be resized if the source image has 16,800,000 pixels or fewer. An image of 4000x4000 pixels (16,000,000) would be accepted, but 4100x4100 (16,810,000) would fail.
Resized images count against your storage quota.
Using Your Own CDN
All SDKs support custom CDN implementations. Implement a custom file uploader to use your own storage solution.
final client = StreamChatClient(
"api-key",
attachmentFileUploader: MyCustomFileUploader(),
);