File Uploads

The channel.sendImage and channel.sendFile methods make it easy to upload files.

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

  1. Send image or file to Stream using the sendImage or sendFile methods. It is recommended to also include attachment type: to ensure proper handling by front end. (e.g. image, video, url)

  2. This will return a signed URL that points to your file on the Stream CDN. If the file is an image, it will be served inline with HTTP requests, or as an attachment otherwise.

  3. Send message with URL as attachment.

// Android SDK
ChannelClient channelClient = client.channel("messaging", "general");

// Upload an image without detailed progress
channelClient.sendImage(imageFile).enqueue(result -> {
  if (result.isSuccess()) {
    // Successful upload, you can now attach this image
    // to a message that you then send to a channel
    String imageUrl = result.data().getFile();

    Attachment attachment = new Attachment();
    attachment.setType("image");
    attachment.setImageUrl(imageUrl);

    Message message = new Message();
    message.getAttachments().add(attachment);

    channelClient.sendMessage(message).enqueue(res -> { /* ... */ });
  }
});

// Upload a file, monitoring for progress with a ProgressCallback
channelClient.sendFile(anyOtherFile, new ProgressCallback() {
  @Override
  public void onSuccess(@NotNull String file) {
    String fileUrl = file;
  }

  @Override
  public void onError(@NotNull ChatError error) {
    // Handle error
  }

  @Override
  public void onProgress(long bytesUploaded, long totalBytes) {
    // You can render the uploading progress here
  }
}).enqueue(); // No callback passed to enqueue, as we'll get notified above anyway

// Backend SDK
String pngFileUrl =
  Message.uploadImage("messaging", "general", userId, "image/png")
    .file(new File("./helloworld.png"))
    .request()
    .getFile();
Message.send("messaging", "general")
  .message(
    MessageRequestObject.builder()
      .text("Check out what I have uploaded in parallel")
      .attachment(
        AttachmentRequestObject.builder()
          .type("image")
          .assetURL(pngFileUrl)
          .thumbURL(pngFileUrl)
          .build())
      .userId(userId)
      .build())
  .request();

Deleting Files and Images

We expose two methods for deleting files and images, channel.deleteImage and channel.deleteFile

// Android SDK
ChannelClient channelClient = client.channel("messaging", "general");

// Deletes the image
channelClient.deleteImage("{{ url of uploaded image }}").enqueue();

// Deletes the file
channelClient.deleteFile("{{ url of uploaded file }}").enqueue();

// Backend SDK
Message.deleteImage(channelType, channelId, url).request();
Message.deleteFile(channelType, channelId, url).request();

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, see the App Settings page

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:

ParameterTypeValuesDescription
wnumberWidth
hnumberHeight
resizestringclip, crop, scale, fillThe resizing mode
cropstringcenter, top, bottom, left, rightThe cropping direction during resize

Resized images will count against your stored files quota.

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&…

As explained above, all operations that return messages will refresh links automatically, but if you are storing a message containing an expired link you can perform a getMessage call to retrieve fresh links for its attachments.

Using Your Own CDN

All 5 SDKs make it easy to use your own CDN for uploads. The code examples below show how to change where files are uploaded:

// Android SDK

// Set a custom FileUploader implementation when building your client
ChatClient client = new ChatClient.Builder("{{ api_key }}", context)
    .fileUploader(new MyFileUploader())
    .build();

// Backend SDK
Message.uploadFile(
    testChannel.getType(),
    testChannel.getId(),
    testUserRequestObject.getId(),
    "text/plain")
  .file(new File(testFileUrl.getFile()))
  .withFileHandler(fileHandler) // Set custom implementation here
  .request();
© Getstream.io, Inc. All Rights Reserved.