// In order to upload a file or image, you can use the `StreamAttachmentUploader`.
// The StreamAttachmentUploader is a thread-safe, asynchronous uploader for `AnyStreamAttachment` objects.
// It wraps a lower-level CDNClient implementation to simplify file uploads and provide progress updates and completion callbacks.
import StreamCore
import StreamFeeds
func uploadAttachmentPayloads(_ attachments: [AnyAttachmentPayload], in feed: FeedId) async throws -> [Attachment] {
let dataAttachments: [StreamAttachment<Data>] = try attachments
.filter { $0.localFileURL != nil }
.enumerated()
.compactMap { index, attachment in
guard let localFileURL = attachment.localFileURL else { return nil }
let attachmentFile = try AttachmentFile(url: localFileURL)
let payloadData = try JSONEncoder().encode(attachment.payload)
try Task.checkCancellation()
return StreamAttachment<Data>(
id: AttachmentId(
fid: feed.rawValue,
activityId: UUID().uuidString,
index: index
),
type: attachment.type,
payload: payloadData,
downloadingState: nil,
uploadingState: .init(
localFileURL: localFileURL,
state: .pendingUpload,
file: attachmentFile
)
)
}
return try await attachmentUploader.upload(dataAttachments, progress: nil)
.map { uploadedAttachment in
Attachment(
assetUrl: uploadedAttachment.remoteURL.absoluteString,
custom: [:],
imageUrl: uploadedAttachment.remoteURL.absoluteString
)
}
}
// You can then use these uploads as attachments when creating activity:
let activity = try await feed.addActivity(
request: .init(
attachments: uploadedAttachments,
text: "Hi",
type: "activity"
)
)
// Alternatively, you can directly pass local attachments and they will be automatically uploaded:
let payloads: [AnyAttachmentPayload] = [] // set payloads
let activity = try await feed.addActivity(
request: .init(
attachmentUploads: payloads,
text: "Hi",
type: "activity"
)
)File Uploads
Stream allows you to easily upload files to our CDN and use them as attachments for activities and comments.
This functionality defaults to using the Stream CDN. If you would like, you can easily change the logic to upload to your own CDN of choice.
How to upload a file or image
import (
"context"
"os"
"github.com/GetStream/getstream-go/v3"
)
// Upload an image file
tmpFile, err := os.CreateTemp("", "test-image-*.jpg")
if err != nil {
log.Fatal(err)
}
defer os.Remove(tmpFile.Name())
// Write some test image content (in real usage, use actual image data)
tmpFile.WriteString("fake-image-data")
tmpFile.Close()
uploadReq := &getstream.UploadImageRequest{
File: getstream.PtrTo(tmpFile.Name()),
UploadSizes: []getstream.ImageSize{
{Width: getstream.PtrTo(100), Height: getstream.PtrTo(100), Resize: getstream.PtrTo("scale"), Crop: getstream.PtrTo("center")},
},
User: &getstream.OnlyUserID{
ID: "john",
},
}
imageResponse, err := client.UploadImage(context.Background(), uploadReq)
if err != nil {
log.Fatal("Error uploading image:", err)
}
// Upload a regular file
tmpFile2, err := os.CreateTemp("", "test-file-*.pdf")
if err != nil {
log.Fatal(err)
}
defer os.Remove(tmpFile2.Name())
tmpFile2.WriteString("fake-file-data")
tmpFile2.Close()
fileUploadReq := &getstream.UploadFileRequest{
File: getstream.PtrTo(tmpFile2.Name()),
User: &getstream.OnlyUserID{
ID: "john",
},
}
fileResponse, err := client.UploadFile(context.Background(), fileUploadReq)
if err != nil {
log.Fatal("Error uploading file:", err)
}
// Use the uploaded file URLs in activity attachments
feedsClient := client.Feeds()
response, err := feedsClient.AddActivity(context.Background(), &getstream.AddActivityRequest{
Type: "post",
Feeds: []string{"user:john"},
Text: getstream.PtrTo("Look at this amazing view of NYC!"),
UserID: getstream.PtrTo("john"),
Attachments: []getstream.Attachment{
{
ImageUrl: imageResponse.Data.File,
Type: getstream.PtrTo("image"),
Title: getstream.PtrTo("NYC Skyline"),
},
},
Custom: map[string]any{
"location": "New York City",
"camera": "iPhone 15 Pro",
},
})
if err != nil {
log.Fatal("Error adding activity:", err)
}The upload_sizes array lets you create specific versions of the uploaded image. It's also possible to resize images on the fly (see Image resizing section), but if you know what versions you'll need beforehand, it's faster to create them in advance.
Deleting Files and Images
We expose two methods for deleting files and images, client.deleteImage and client.deleteFile
_, err = client.DeleteImage(context.Background(), &getstream.DeleteImageRequest{
Url: getstream.PtrTo("<image URL>"),
})
_, err = client.DeleteFile(context.Background(), &getstream.DeleteFileRequest{
Url: getstream.PtrTo("<file URL>"),
})Requirements for Images
-
Stream supported image types are: image/bmp, image/gif, image/jpeg, image/png, image/webp, image/heic, image/heic-sequence, image/heif, image/heif-sequence, image/svg+xml.
-
You can set a more restrictive list for your application if needed.
-
The maximum file size is 100MB.
Requirements for Files
-
Stream will not block any file types from uploading, however, different clients may handle different types differently or not at all.
-
You can set a more restrictive list for your application if needed.
-
The maximum file size is 100MB.
How to Allow/Block file extensions
Stream will allow any file extension. If you want to be more restrictive for an application, this is can be set via API or by logging into your dashboard.
-
To update via the dashboard, login and go to the Chat Overview page >> Upload Configuration.
-
API updates are made using UpdateAppSettings.
Image resizing
You can automatically resize an image appending query parameters to a valid image link stored on the Stream CDN.
An image can only be resized if the total pixel count of the source image is 16.800.000 or less. Attempting to resize an image with more pixels will result in an API error. An image of 4000 by 4000 would be accepted, but an image of 4100 by 4100 would pass the upper treshold for resizing.
There are four supported params - all of them are optional and can be used interchangeably:
| Parameter | Type | Values | Description |
|---|---|---|---|
| w | number | Width | |
| h | number | Height | |
| resize | string | clip , crop , scale , fill | The resizing mode |
| crop | string | center , top , bottom , left , right | The cropping direction during resize |
Resized images will count against your stored files quota.
Access control and link expiration
The Stream CDN URL returned during the upload contains a signature that validates the access to the file it points to. Only the members of a channel a file was uploaded to can see the attachment and its unique, signed link. Links can only be accessed with a valid signature, which also protects against enumeration attacks.
Whenever messages containing your attachments are retrieved (i.e., when querying a channel), the attachment links will contain a new, fresh signature.
A single Stream CDN URL expires after 14 days, after which its signature will stop working and the link won't be valid anymore. You can check when a link will expire by comparing the current time with the Unix timestamp in the Expires parameter of the link's query: https://us-east.stream-io-cdn.com/0000/images/foo.png?...& Expires=1602666347 &...
Using Your Own CDN
All our SDKs make it easy to use your own CDN for uploads. The code examples below show how to change where files are uploaded:
// No need to set anything, just call your own CDN instead of client.uploadFile / client.uploadImage