Настройка хранилища медиафайлов мобильного приложения (S3/Cloud Storage)

TRUETECH занимается разработкой, поддержкой и обслуживанием мобильных приложений iOS, Android, PWA. Имеем большой опыт и экспертизу для публикации мобильных приложений в популярные маркеты Google Play, App Store, Amazon, AppGallery и другие.
Разработка и поддержка любых видов мобильных приложений:
Информационные и развлекательные мобильные приложения
Новостные приложения, игры, справочники, онлайн-каталоги, погодные, фитнес и здоровье, туристические, образовательные, социальные сети и мессенджеры, квиз, блоги и подкасты, форумы, агрегаторы
Мобильные приложения электронной коммерции
Интернет-магазины, B2B-приложения, маркетплейсы, онлайн-обменники, кэшбэк-сервисы, биржи, дропшиппинг-платформы, программы лояльности, доставка еды и товаров, платежные системы
Мобильные приложения для управления бизнес-процессами
CRM-системы, ERP-системы, управление проектами, инструменты для команды продаж, учет финансов, управление производством, логистика и доставка, управление персоналом, системы мониторинга данных
Мобильные приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, платформы предоставления электронных услуг, платформы кешбека, видеохостинги, тематические порталы, платформы онлайн-бронирования и записи, платформы онлайн-торговли

Это лишь некоторые из типы мобильных приложений, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента.

Предлагаемые услуги
Показано 1 из 1 услугВсе 1735 услуг
Настройка хранилища медиафайлов мобильного приложения (S3/Cloud Storage)
Средняя
от 1 рабочего дня до 3 рабочих дней
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_mobile-applications_feedme_467_0.webp
    Разработка мобильного приложения для компании FEEDME
    756
  • image_mobile-applications_xoomer_471_0.webp
    Разработка мобильного приложения для компании XOOMER
    624
  • image_mobile-applications_rhl_428_0.webp
    Разработка мобильного приложения для компании RHL
    1052
  • image_mobile-applications_zippy_411_0.webp
    Разработка мобильного приложения для компании ZIPPY
    947
  • image_mobile-applications_affhome_429_0.webp
    Разработка мобильного приложения для компании Affhome
    862
  • image_mobile-applications_flavors_409_0.webp
    Разработка мобильного приложения для компании FLAVORS
    445

Настройка хранилища медиафайлов мобильного приложения (S3/Cloud Storage)

Загрузка фото и видео из мобильного приложения — задача, которую легко недооценить. Один запрос на 50 МБ видео через бэкенд-прокси — это нагрузка на сервер, двойной трафик и медленная загрузка для пользователя. Правильная архитектура: presigned URL → прямая загрузка из клиента в S3 или GCS, бэкенд только выдаёт токен и получает подтверждение.

Выбор провайдера

Провайдер SDK Сильные стороны
AWS S3 aws-sdk-swift, aws-sdk-kotlin, Amplify Зрелость, богатство функций, multipart из коробки
Google Cloud Storage google-cloud-storage (Java/Kotlin), GCS REST API Хорошая интеграция с Firebase, GCP
Cloudflare R2 S3-совместимый API Нет egress-трафика, дешевле
Backblaze B2 S3-совместимый API Самый дешёвый объектный storage

R2 и B2 совместимы с S3 API — один и тот же клиентский код, только другой endpoint.

Загрузка с прогрессом

Пользователь видит progress bar — это не опционально для видео. iOS через URLSession.uploadTask с делегатом:

class MediaUploader: NSObject, URLSessionTaskDelegate {
    private lazy var session: URLSession = {
        URLSession(configuration: .default, delegate: self, delegateQueue: nil)
    }()

    func upload(fileURL: URL, presignedURL: URL,
                progress: @escaping (Double) -> Void,
                completion: @escaping (Result<Void, Error>) -> Void) {
        var request = URLRequest(url: presignedURL)
        request.httpMethod = "PUT"
        request.setValue(fileURL.mimeType, forHTTPHeaderField: "Content-Type")

        let task = session.uploadTask(with: request, fromFile: fileURL)
        task.taskDescription = fileURL.lastPathComponent

        uploadCompletionHandlers[task.taskIdentifier] = completion
        uploadProgressHandlers[task.taskIdentifier] = progress
        task.resume()
    }

    func urlSession(_ session: URLSession, task: URLSessionTask,
                    didSendBodyData bytesSent: Int64,
                    totalBytesSent: Int64, totalBytesExpectedToSend: Int64) {
        let progress = Double(totalBytesSent) / Double(totalBytesExpectedToSend)
        uploadProgressHandlers[task.taskIdentifier]?(progress)
    }
}

На Android через OkHttp с кастомным RequestBody:

class ProgressRequestBody(
    private val file: File,
    private val contentType: MediaType,
    private val onProgress: (Int) -> Unit
) : RequestBody() {
    override fun contentType() = contentType
    override fun contentLength() = file.length()

    override fun writeTo(sink: BufferedSink) {
        val buffer = ByteArray(8192)
        var uploaded = 0L
        file.inputStream().use { input ->
            var bytesRead: Int
            while (input.read(buffer).also { bytesRead = it } != -1) {
                sink.write(buffer, 0, bytesRead)
                uploaded += bytesRead
                val progress = (uploaded * 100 / file.length()).toInt()
                onProgress(progress)
            }
        }
    }
}

Multipart upload для видео

S3 требует multipart для файлов больше 5 МБ (рекомендуется от 100 МБ). Преимущества: параллельная загрузка частей, возможность возобновить после прерывания.

class S3MultipartUploader(private val s3Client: S3Client) {
    suspend fun upload(bucketName: String, key: String, file: File): String {
        // 1. Инициализируем multipart upload
        val createResponse = s3Client.createMultipartUpload {
            bucket = bucketName
            this.key = key
            contentType = file.detectMimeType()
        }
        val uploadId = createResponse.uploadId!!

        val partSize = 10 * 1024 * 1024L // 10 МБ на часть
        val parts = mutableListOf<CompletedPart>()

        try {
            file.inputStream().use { stream ->
                var partNumber = 1
                val buffer = ByteArray(partSize.toInt())
                var bytesRead: Int

                while (stream.read(buffer).also { bytesRead = it } != -1) {
                    val partData = buffer.copyOf(bytesRead)
                    val uploadPartResponse = s3Client.uploadPart {
                        bucket = bucketName
                        this.key = key
                        this.uploadId = uploadId
                        this.partNumber = partNumber
                        body = ByteStream.fromBytes(partData)
                    }
                    parts.add(CompletedPart {
                        this.partNumber = partNumber
                        eTag = uploadPartResponse.eTag
                    })
                    partNumber++
                }
            }

            // 3. Завершаем multipart upload
            s3Client.completeMultipartUpload {
                bucket = bucketName
                this.key = key
                this.uploadId = uploadId
                multipartUpload = CompletedMultipartUpload { this.parts = parts }
            }

            return "https://$bucketName.s3.amazonaws.com/$key"
        } catch (e: Exception) {
            // Очищаем незавершённый upload — иначе оплачивается
            s3Client.abortMultipartUpload {
                bucket = bucketName
                this.key = key
                this.uploadId = uploadId
            }
            throw e
        }
    }
}

abortMultipartUpload при ошибке — важно. Незавершённые multipart uploads продолжают тарифицироваться в AWS. Добавьте Lifecycle Rule на удаление незавершённых uploads через 7 дней как страховку.

Обработка медиафайлов перед загрузкой

Видео 4K 200 МБ напрямую в S3 — редко нужно. На клиенте перед загрузкой:

  • Изображения: сжатие через UIGraphicsImageRenderer (iOS) или BitmapFactory.Options.inSampleSize (Android). WebP вместо JPEG — лучшее соотношение качества/размера.
  • Видео: транскодирование через AVAssetExportSession (iOS) или MediaCodec/Transformer (Android) до 1080p/720p. Для React Native — react-native-video-processing.
// iOS: сжатие видео перед загрузкой
let exporter = AVAssetExportSession(asset: asset, presetName: AVAssetExportPreset1280x720)!
exporter.outputURL = outputURL
exporter.outputFileType = .mp4
exporter.shouldOptimizeForNetworkUse = true

await exporter.export()
// После экспорта — загружаем outputURL в S3

Клиентское сжатие — компромисс: меньше трафика, быстрее загрузка, но нагрузка на CPU устройства.

CDN для раздачи

S3 напрямую — только для приватных файлов. Публичные медиа (аватары, контент) — через CloudFront или Cloudflare CDN. Это кэширование на edge-нодах по всему миру и HTTPS без дополнительной настройки.

Настройка хранилища медиафайлов с presigned upload, multipart для видео, CDN и сжатием: 1–2 недели. Стоимость рассчитывается индивидуально.