From e3dfbb4f406dde0a0a1a1c907435a70da7445d59 Mon Sep 17 00:00:00 2001 From: Jan Date: Wed, 21 Aug 2024 16:42:20 +0200 Subject: [PATCH 01/13] Add support for file metadata, `info` and `exists` --- .../github/jan/supabase/storage/BucketApi.kt | 46 +++++++--- .../jan/supabase/storage/BucketApiImpl.kt | 88 ++++++++++++++----- .../github/jan/supabase/storage/BucketItem.kt | 45 +++++++++- .../jan/supabase/storage/FileOptionBuilder.kt | 23 +++++ .../io/github/jan/supabase/storage/FlowExt.kt | 21 ++--- .../io/github/jan/supabase/storage/Storage.kt | 14 ++- 6 files changed, 189 insertions(+), 48 deletions(-) create mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt index a4c7da748..d0b7a8db3 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt @@ -41,9 +41,9 @@ sealed interface BucketApi { * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun upload(path: String, data: ByteArray, upsert: Boolean = false): FileUploadResponse { + suspend fun upload(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } - return upload(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert) + return upload(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) } /** @@ -56,7 +56,7 @@ sealed interface BucketApi { * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun upload(path: String, data: UploadData, upsert: Boolean = false): FileUploadResponse + suspend fun upload(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Uploads a file in [bucketId] under [path] using a presigned url @@ -67,10 +67,15 @@ sealed interface BucketApi { * @return the key of the uploaded file * @throws IllegalArgumentException if data to upload is empty */ - suspend fun uploadToSignedUrl(path: String, token: String, data: ByteArray, upsert: Boolean = false + suspend fun uploadToSignedUrl( + path: String, + token: String, + data: ByteArray, + upsert: Boolean = false, + options: FileOptionBuilder.() -> Unit = {} ): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } - return uploadToSignedUrl(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), upsert) + return uploadToSignedUrl(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) } /** @@ -85,7 +90,7 @@ sealed interface BucketApi { * @throws HttpRequestException on network related issues * @throws HttpRequestException on network related issues */ - suspend fun uploadToSignedUrl(path: String, token: String, data: UploadData, upsert: Boolean = false): FileUploadResponse + suspend fun uploadToSignedUrl(path: String, token: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Updates a file in [bucketId] under [path] @@ -98,9 +103,9 @@ sealed interface BucketApi { * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun update(path: String, data: ByteArray, upsert: Boolean = false): FileUploadResponse { + suspend fun update(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } - return update(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert) + return update(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) } /** @@ -113,7 +118,7 @@ sealed interface BucketApi { * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun update(path: String, data: UploadData, upsert: Boolean = false): FileUploadResponse + suspend fun update(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Deletes all files in [bucketId] with in [paths] @@ -242,8 +247,8 @@ sealed interface BucketApi { /** - * Searches for buckets with the given [prefix] and [filter] - * @return The filtered buckets + * Searches for files with the given [prefix] and [filter] + * @return The filtered bucket items * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues @@ -253,6 +258,25 @@ sealed interface BucketApi { filter: BucketListFilter.() -> Unit = {} ): List + /** + * Returns information about the file under [path] + * @param path The path to get information about + * @return The file object + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ + suspend fun info(path: String): FileObjectV2 + + /** + * Checks if a file exists under [path] + * @return true if the file exists, false otherwise + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ + suspend fun exists(path: String): Boolean + /** * Changes the bucket's public status to [public] * @throws RestException or one of its subclasses if receiving an error response diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index 4d664684c..8ebb2694c 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -1,5 +1,6 @@ package io.github.jan.supabase.storage +import io.github.jan.supabase.exceptions.RestException import io.github.jan.supabase.putJsonObject import io.github.jan.supabase.safeBody import io.github.jan.supabase.storage.BucketApi.Companion.UPSERT_HEADER @@ -29,6 +30,8 @@ import kotlinx.serialization.json.jsonPrimitive import kotlinx.serialization.json.put import kotlinx.serialization.json.putJsonArray import kotlinx.serialization.json.putJsonObject +import kotlin.io.encoding.Base64 +import kotlin.io.encoding.ExperimentalEncodingApi import kotlin.time.Duration internal class BucketApiImpl(override val bucketId: String, val storage: StorageImpl, resumableCache: ResumableCache) : BucketApi { @@ -37,18 +40,24 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override val resumable = ResumableClientImpl(this, resumableCache) - override suspend fun update(path: String, data: UploadData, upsert: Boolean): FileUploadResponse = + override suspend fun update( + path: String, + data: UploadData, + upsert: Boolean, + options: FileOptionBuilder.() -> Unit + ): FileUploadResponse = uploadOrUpdate( - HttpMethod.Put, bucketId, path, data, upsert + HttpMethod.Put, bucketId, path, data, upsert, options ) override suspend fun uploadToSignedUrl( path: String, token: String, data: UploadData, - upsert: Boolean + upsert: Boolean, + options: FileOptionBuilder.() -> Unit ): FileUploadResponse { - return uploadToSignedUrl(path, token, data, upsert) {} + return uploadToSignedUrl(path, token, data, upsert, options) {} } override suspend fun createSignedUploadUrl(path: String): UploadSignedUrl { @@ -64,9 +73,14 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage ) } - override suspend fun upload(path: String, data: UploadData, upsert: Boolean): FileUploadResponse = + override suspend fun upload( + path: String, + data: UploadData, + upsert: Boolean, + options: FileOptionBuilder.() -> Unit + ): FileUploadResponse = uploadOrUpdate( - HttpMethod.Post, bucketId, path, data, upsert + HttpMethod.Post, bucketId, path, data, upsert, options ) override suspend fun delete(paths: Collection) { @@ -210,24 +224,37 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage }).safeBody() } + override suspend fun info(path: String): FileObjectV2 { + val response = storage.api.get("object/info/public/$bucketId/$path") + return response.safeBody() + } + + override suspend fun exists(path: String): Boolean { + try { + storage.api.request("object/$bucketId/$path") { + method = HttpMethod.Head + } + return true + } catch (e: RestException) { + if (e.statusCode in listOf(400, 404)) return false + throw e + } + } + + @OptIn(ExperimentalEncodingApi::class) internal suspend fun uploadOrUpdate( method: HttpMethod, bucket: String, path: String, data: UploadData, upsert: Boolean, + options: FileOptionBuilder.() -> Unit, extra: HttpRequestBuilder.() -> Unit = {} ): FileUploadResponse { + val optionBuilder = FileOptionBuilder(storage.serializer).apply(options) val response = storage.api.request("object/$bucket/$path") { this.method = method - setBody(object : OutgoingContent.ReadChannelContent() { - override val contentType: ContentType = ContentType.defaultForFilePath(path) - override val contentLength: Long = data.size - override fun readFrom(): ByteReadChannel = data.stream - }) - header(HttpHeaders.ContentType, ContentType.defaultForFilePath(path)) - header(UPSERT_HEADER, upsert.toString()) - extra() + defaultUploadRequest(path, data, upsert, optionBuilder, extra) }.body() val key = response["Key"]?.jsonPrimitive?.content ?: error("Expected a key in a upload response") @@ -236,23 +263,19 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage return FileUploadResponse(id, path, key) } + @OptIn(ExperimentalEncodingApi::class) internal suspend fun uploadToSignedUrl( path: String, token: String, data: UploadData, upsert: Boolean, + options: FileOptionBuilder.() -> Unit, extra: HttpRequestBuilder.() -> Unit = {} ): FileUploadResponse { + val optionBuilder = FileOptionBuilder(storage.serializer).apply(options) val response = storage.api.put("object/upload/sign/$bucketId/$path") { parameter("token", token) - setBody(object : OutgoingContent.ReadChannelContent() { - override val contentType: ContentType = ContentType.defaultForFilePath(path) - override val contentLength: Long = data.size - override fun readFrom(): ByteReadChannel = data.stream - }) - header(HttpHeaders.ContentType, ContentType.defaultForFilePath(path)) - header("x-upsert", upsert.toString()) - extra() + defaultUploadRequest(path, data, upsert, optionBuilder, extra) }.body() val key = response["Key"]?.jsonPrimitive?.content ?: error("Expected a key in a upload response") @@ -260,6 +283,27 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage return FileUploadResponse(id, path, key) } + @OptIn(ExperimentalEncodingApi::class) + private fun HttpRequestBuilder.defaultUploadRequest( + path: String, + data: UploadData, + upsert: Boolean, + optionBuilder: FileOptionBuilder, + extra: HttpRequestBuilder.() -> Unit + ) { + setBody(object : OutgoingContent.ReadChannelContent() { + override val contentType: ContentType = ContentType.defaultForFilePath(path) + override val contentLength: Long = data.size + override fun readFrom(): ByteReadChannel = data.stream + }) + header(HttpHeaders.ContentType, ContentType.defaultForFilePath(path)) + header(UPSERT_HEADER, upsert.toString()) + optionBuilder.userMetadata?.let { + header("x-metadata", Base64.Default.encode(it.toString().encodeToByteArray()).also((::println))) + } + extra() + } + override suspend fun changePublicStatusTo(public: Boolean) = storage.updateBucket(bucketId) { this@updateBucket.public = public } diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt index 595eb8d96..922f251f5 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt @@ -1,8 +1,12 @@ package io.github.jan.supabase.storage +import io.github.jan.supabase.SupabaseSerializer +import io.github.jan.supabase.decode +import io.github.jan.supabase.serializer.KotlinXSerializer import kotlinx.datetime.Instant import kotlinx.serialization.SerialName import kotlinx.serialization.Serializable +import kotlinx.serialization.Transient import kotlinx.serialization.json.JsonObject /** @@ -14,6 +18,7 @@ import kotlinx.serialization.json.JsonObject * @param lastAccessedAt The last access date of the item * @param metadata The metadata of the item */ +//TODO: Rename to FileObject @Serializable data class BucketItem( val name: String, @@ -25,4 +30,42 @@ data class BucketItem( @SerialName("last_accessed_at") val lastAccessedAt: Instant?, val metadata: JsonObject? -) \ No newline at end of file +) + +/** + * Represents a file or a folder in a bucket. If the item is a folder, everything except [name] is null. + * @param name The name of the item + * @param id The id of the item + * @param updatedAt The last update date of the item + * @param createdAt The creation date of the item + * @param lastAccessedAt The last access date of the item + * @param metadata The metadata of the item + */ +@Serializable +data class FileObjectV2( + val name: String, + val id: String?, + val version: String, + @SerialName("bucket_id") + val bucketId: String? = null, + @SerialName("updated_at") + val updatedAt: Instant? = null, + @SerialName("created_at") + val createdAt: Instant?, + @SerialName("last_accessed_at") + val lastAccessedAt: Instant? = null, + val metadata: JsonObject?, + val size: Long, + @SerialName("content_type") + val contentType: String, + val etag: String?, + @SerialName("last_modified") + val lastModified: Instant?, + @SerialName("cache_control") + val cacheControl: String?, + @Transient @PublishedApi internal val serializer: SupabaseSerializer = KotlinXSerializer() +) { + + inline fun decodeMetadata(): T? = metadata?.let { serializer.decode(it.toString()) } + +} \ No newline at end of file diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt new file mode 100644 index 000000000..e3f9f3070 --- /dev/null +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt @@ -0,0 +1,23 @@ +package io.github.jan.supabase.storage + +import io.github.jan.supabase.SupabaseSerializer +import io.github.jan.supabase.encodeToJsonElement +import kotlinx.serialization.json.JsonObject +import kotlinx.serialization.json.JsonObjectBuilder +import kotlinx.serialization.json.buildJsonObject +import kotlinx.serialization.json.jsonObject + +class FileOptionBuilder( + @PublishedApi internal val serializer: SupabaseSerializer, + var userMetadata: JsonObject? = null, +) { + + inline fun userMetadata(data: T) { + userMetadata = serializer.encodeToJsonElement(data).jsonObject + } + + inline fun userMetadata(builder: JsonObjectBuilder.() -> Unit) { + userMetadata = buildJsonObject(builder) + } + +} diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt index d473c0972..dc3c2d284 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt @@ -22,9 +22,9 @@ import kotlinx.coroutines.flow.callbackFlow * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ -fun BucketApi.updateAsFlow(path: String, data: UploadData, upsert: Boolean = false): Flow = callbackFlow { +fun BucketApi.updateAsFlow(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = callbackFlow { this@updateAsFlow as BucketApiImpl - val key = uploadOrUpdate(HttpMethod.Put, bucketId, path, data, upsert) { + val key = uploadOrUpdate(HttpMethod.Put, bucketId, path, data, upsert, options) { onUpload { bytesSentTotal, contentLength -> trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) } @@ -43,8 +43,8 @@ fun BucketApi.updateAsFlow(path: String, data: UploadData, upsert: Boolean = fal * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ -fun BucketApi.uploadAsFlow(path: String, data: ByteArray, upsert: Boolean = false): Flow = uploadAsFlow(path, UploadData( - ByteReadChannel(data), data.size.toLong()), upsert) +fun BucketApi.uploadAsFlow(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = uploadAsFlow(path, UploadData( + ByteReadChannel(data), data.size.toLong()), upsert, options) /** * Uploads a file in [bucketId] under [path] using a presigned url @@ -61,11 +61,12 @@ fun BucketApi.uploadToSignedUrlAsFlow( path: String, token: String, data: UploadData, - upsert: Boolean = false + upsert: Boolean = false, + options: FileOptionBuilder.() -> Unit = {} ): Flow { return callbackFlow { this@uploadToSignedUrlAsFlow as BucketApiImpl - val key = uploadToSignedUrl(path, token, data, upsert) { + val key = uploadToSignedUrl(path, token, data, upsert, options) { onUpload { bytesSentTotal, contentLength -> trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) } @@ -83,7 +84,7 @@ fun BucketApi.uploadToSignedUrlAsFlow( * @param upsert Whether to overwrite an existing file * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, data: ByteArray, upsert: Boolean = false): Flow = uploadToSignedUrlAsFlow(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), upsert) +fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = uploadToSignedUrlAsFlow(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) /** * Updates a file in [bucketId] under [path] @@ -95,10 +96,10 @@ fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, data: ByteArr * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ -fun BucketApi.uploadAsFlow(path: String, data: UploadData, upsert: Boolean = false): Flow { +fun BucketApi.uploadAsFlow(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow { return callbackFlow { this@uploadAsFlow as BucketApiImpl - val key = uploadOrUpdate(HttpMethod.Post, bucketId, path, data, upsert) { + val key = uploadOrUpdate(HttpMethod.Post, bucketId, path, data, upsert, options) { onUpload { bytesSentTotal, contentLength -> trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) } @@ -118,7 +119,7 @@ fun BucketApi.uploadAsFlow(path: String, data: UploadData, upsert: Boolean = fal * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ -fun BucketApi.updateAsFlow(path: String, data: ByteArray, upsert: Boolean = false): Flow = updateAsFlow(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert) +fun BucketApi.updateAsFlow(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = updateAsFlow(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) /** * Downloads a file from [bucketId] under [path] diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt index 64eea8f3c..7749c6906 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt @@ -1,6 +1,7 @@ package io.github.jan.supabase.storage import io.github.jan.supabase.SupabaseClient +import io.github.jan.supabase.SupabaseSerializer import io.github.jan.supabase.annotations.SupabaseInternal import io.github.jan.supabase.bodyOrNull import io.github.jan.supabase.collections.AtomicMutableMap @@ -13,6 +14,8 @@ import io.github.jan.supabase.exceptions.UnknownRestException import io.github.jan.supabase.gotrue.authenticatedSupabaseApi import io.github.jan.supabase.logging.SupabaseLogger import io.github.jan.supabase.logging.w +import io.github.jan.supabase.plugins.CustomSerializationConfig +import io.github.jan.supabase.plugins.CustomSerializationPlugin import io.github.jan.supabase.plugins.MainConfig import io.github.jan.supabase.plugins.MainPlugin import io.github.jan.supabase.plugins.SupabasePluginProvider @@ -46,7 +49,7 @@ import kotlin.time.Duration.Companion.seconds * val bytes = bucket.downloadAuthenticated("icon.png") * ``` */ -sealed interface Storage : MainPlugin { +sealed interface Storage : MainPlugin, CustomSerializationPlugin { /** * Creates a new bucket in the storage @@ -116,8 +119,9 @@ sealed interface Storage : MainPlugin { */ data class Config( var transferTimeout: Duration = 120.seconds, - @PublishedApi internal var resumable: Resumable = Resumable() - ) : MainConfig() { + @PublishedApi internal var resumable: Resumable = Resumable(), + override var serializer: SupabaseSerializer? = null + ) : MainConfig(), CustomSerializationConfig { /** * @param cache the cache for caching resumable upload urls @@ -127,7 +131,7 @@ sealed interface Storage : MainPlugin { data class Resumable( var cache: ResumableCache? = null, var retryTimeout: Duration = 5.seconds, - var onlyUpdateStateAfterChunk: Boolean = false + var onlyUpdateStateAfterChunk: Boolean = false, ) { /** @@ -185,6 +189,8 @@ internal class StorageImpl(override val supabaseClient: SupabaseClient, override override val apiVersion: Int get() = Storage.API_VERSION + override val serializer: SupabaseSerializer = config.serializer ?: supabaseClient.defaultSerializer + @OptIn(SupabaseInternal::class) internal val api = supabaseClient.authenticatedSupabaseApi(this) { timeout { From 1d272d86fb1d74d06ba2714cbcc6c9d3611d812a Mon Sep 17 00:00:00 2001 From: Jan Date: Sun, 25 Aug 2024 14:03:25 +0200 Subject: [PATCH 02/13] Finish up `info` method and rename `BucketItem` to `FileObject` --- .../io/github/jan/supabase/storage/BucketApi.kt | 2 +- .../github/jan/supabase/storage/BucketApiImpl.kt | 6 +++--- .../storage/{BucketItem.kt => FileObject.kt} | 16 +++++++++++++--- 3 files changed, 17 insertions(+), 7 deletions(-) rename Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/{BucketItem.kt => FileObject.kt} (82%) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt index d0b7a8db3..0895820fc 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt @@ -256,7 +256,7 @@ sealed interface BucketApi { suspend fun list( prefix: String = "", filter: BucketListFilter.() -> Unit = {} - ): List + ): List /** * Returns information about the file under [path] diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index adfd81259..3f3513652 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -217,7 +217,7 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override suspend fun list( prefix: String, filter: BucketListFilter.() -> Unit - ): List { + ): List { return storage.api.postJson("object/list/$bucketId", buildJsonObject { put("prefix", prefix) putJsonObject(BucketListFilter().apply(filter).build()) @@ -225,8 +225,8 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage } override suspend fun info(path: String): FileObjectV2 { - val response = storage.api.get("object/info/public/$bucketId/$path") - return response.safeBody() + val response = storage.api.get("object/info/$bucketId/$path") + return response.safeBody().copy(serializer = storage.serializer) } override suspend fun exists(path: String): Boolean { diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt similarity index 82% rename from Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt rename to Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt index 922f251f5..1f49bfe29 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketItem.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt @@ -3,6 +3,7 @@ package io.github.jan.supabase.storage import io.github.jan.supabase.SupabaseSerializer import io.github.jan.supabase.decode import io.github.jan.supabase.serializer.KotlinXSerializer +import io.ktor.http.ContentType import kotlinx.datetime.Instant import kotlinx.serialization.SerialName import kotlinx.serialization.Serializable @@ -18,9 +19,8 @@ import kotlinx.serialization.json.JsonObject * @param lastAccessedAt The last access date of the item * @param metadata The metadata of the item */ -//TODO: Rename to FileObject @Serializable -data class BucketItem( +data class FileObject( val name: String, val id: String?, @SerialName("updated_at") @@ -40,6 +40,12 @@ data class BucketItem( * @param createdAt The creation date of the item * @param lastAccessedAt The last access date of the item * @param metadata The metadata of the item + * @param size The size of the item + * @param contentType The content type of the item + * @param etag The etag of the item + * @param lastModified The last modified date of the item + * @param cacheControl The cache control of the item + * @param serializer The serializer to use for decoding the metadata */ @Serializable data class FileObjectV2( @@ -57,7 +63,7 @@ data class FileObjectV2( val metadata: JsonObject?, val size: Long, @SerialName("content_type") - val contentType: String, + val rawContentType: String, val etag: String?, @SerialName("last_modified") val lastModified: Instant?, @@ -66,6 +72,10 @@ data class FileObjectV2( @Transient @PublishedApi internal val serializer: SupabaseSerializer = KotlinXSerializer() ) { + val contentType by lazy { + ContentType.parse(rawContentType) + } + inline fun decodeMetadata(): T? = metadata?.let { serializer.decode(it.toString()) } } \ No newline at end of file From d928abedacb06390de66fcdb341ef3bb8317c9ca Mon Sep 17 00:00:00 2001 From: Jan Date: Sun, 25 Aug 2024 14:20:49 +0200 Subject: [PATCH 03/13] Add some missing docs and tests --- .../jan/supabase/storage/BucketApiImpl.kt | 3 +- .../github/jan/supabase/storage/FileObject.kt | 10 +- .../jan/supabase/storage/FileOptionBuilder.kt | 13 ++ .../src/commonTest/kotlin/BucketApiTest.kt | 122 ++++++++++++++++-- 4 files changed, 132 insertions(+), 16 deletions(-) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index 3f3513652..df2625c56 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -16,6 +16,7 @@ import io.ktor.client.statement.bodyAsChannel import io.ktor.http.ContentType import io.ktor.http.HttpHeaders import io.ktor.http.HttpMethod +import io.ktor.http.HttpStatusCode import io.ktor.http.Url import io.ktor.http.content.OutgoingContent import io.ktor.http.defaultForFilePath @@ -236,7 +237,7 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage } return true } catch (e: RestException) { - if (e.statusCode in listOf(400, 404)) return false + if (e.statusCode in listOf(HttpStatusCode.NotFound.value, HttpStatusCode.BadRequest.value)) return false throw e } } diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt index 1f49bfe29..9ae862e59 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt @@ -36,12 +36,14 @@ data class FileObject( * Represents a file or a folder in a bucket. If the item is a folder, everything except [name] is null. * @param name The name of the item * @param id The id of the item + * @param version The version of the item + * @param bucketId The bucket id of the item * @param updatedAt The last update date of the item * @param createdAt The creation date of the item * @param lastAccessedAt The last access date of the item * @param metadata The metadata of the item * @param size The size of the item - * @param contentType The content type of the item + * @param rawContentType The content type of the item * @param etag The etag of the item * @param lastModified The last modified date of the item * @param cacheControl The cache control of the item @@ -72,10 +74,16 @@ data class FileObjectV2( @Transient @PublishedApi internal val serializer: SupabaseSerializer = KotlinXSerializer() ) { + /** + * The content type of the file + */ val contentType by lazy { ContentType.parse(rawContentType) } + /** + * Decodes the metadata using the [serializer] + */ inline fun decodeMetadata(): T? = metadata?.let { serializer.decode(it.toString()) } } \ No newline at end of file diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt index e3f9f3070..d59c30e58 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt @@ -7,15 +7,28 @@ import kotlinx.serialization.json.JsonObjectBuilder import kotlinx.serialization.json.buildJsonObject import kotlinx.serialization.json.jsonObject +/** + * Builder for uploading files with additional options + * @param serializer The serializer to use for encoding the metadata + * @param userMetadata The user metadata to upload with the file + */ class FileOptionBuilder( @PublishedApi internal val serializer: SupabaseSerializer, var userMetadata: JsonObject? = null, ) { + /** + * Sets the user metadata to upload with the file + * @param data The data to upload. Must be serializable by the [serializer] + */ inline fun userMetadata(data: T) { userMetadata = serializer.encodeToJsonElement(data).jsonObject } + /** + * Sets the user metadata to upload with the file + * @param builder The builder for the metadata + */ inline fun userMetadata(builder: JsonObjectBuilder.() -> Unit) { userMetadata = buildJsonObject(builder) } diff --git a/Storage/src/commonTest/kotlin/BucketApiTest.kt b/Storage/src/commonTest/kotlin/BucketApiTest.kt index b5e56849e..2dc3640cf 100644 --- a/Storage/src/commonTest/kotlin/BucketApiTest.kt +++ b/Storage/src/commonTest/kotlin/BucketApiTest.kt @@ -1,6 +1,7 @@ import io.github.jan.supabase.SupabaseClient import io.github.jan.supabase.SupabaseClientBuilder import io.github.jan.supabase.storage.BucketApi +import io.github.jan.supabase.storage.FileObjectV2 import io.github.jan.supabase.storage.FileUploadResponse import io.github.jan.supabase.storage.ImageTransformation import io.github.jan.supabase.storage.Storage @@ -13,23 +14,35 @@ import io.github.jan.supabase.testing.pathAfterVersion import io.github.jan.supabase.testing.toJsonElement import io.ktor.client.engine.mock.MockRequestHandleScope import io.ktor.client.engine.mock.respond +import io.ktor.client.engine.mock.respondError +import io.ktor.client.engine.mock.respondOk import io.ktor.client.engine.mock.toByteArray import io.ktor.client.request.HttpRequestData import io.ktor.http.ContentType import io.ktor.http.HttpHeaders import io.ktor.http.HttpMethod +import io.ktor.http.HttpStatusCode import io.ktor.http.headersOf import kotlinx.coroutines.test.runTest import kotlinx.datetime.Clock +import kotlinx.serialization.encodeToString +import kotlinx.serialization.json.Json +import kotlinx.serialization.json.JsonObject +import kotlinx.serialization.json.buildJsonObject import kotlinx.serialization.json.int import kotlinx.serialization.json.jsonArray import kotlinx.serialization.json.jsonObject import kotlinx.serialization.json.jsonPrimitive import kotlinx.serialization.json.long +import kotlinx.serialization.json.put +import kotlin.io.encoding.Base64 +import kotlin.io.encoding.ExperimentalEncodingApi import kotlin.test.Test import kotlin.test.assertContentEquals import kotlin.test.assertEquals +import kotlin.test.assertFalse import kotlin.test.assertNotNull +import kotlin.test.assertTrue import kotlin.time.Duration.Companion.seconds class BucketApiTest { @@ -48,8 +61,10 @@ class BucketApiTest { testUploadMethod( method = HttpMethod.Post, urlPath = "/object/$bucketId/data.png", - request = { client, expectedPath, data -> - client.storage[bucketId].upload(expectedPath, data) + request = { client, expectedPath, data, meta -> + client.storage[bucketId].upload(expectedPath, data) { + userMetadata = meta + } }, extra = { assertEquals( @@ -73,8 +88,10 @@ class BucketApiTest { "Upsert header should be true" ) }, - request = { client, expectedPath, data -> - client.storage[bucketId].upload(expectedPath, data, upsert = true) + request = { client, expectedPath, data, meta -> + client.storage[bucketId].upload(expectedPath, data, upsert = true) { + userMetadata = meta + } } ) } @@ -97,8 +114,10 @@ class BucketApiTest { "Token should be $expectedToken" ) }, - request = { client, expectedPath, data -> - client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data, upsert = false) + request = { client, expectedPath, data, meta -> + client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data, upsert = false) { + userMetadata = meta + } } ) } @@ -121,8 +140,10 @@ class BucketApiTest { "Token should be $expectedToken" ) }, - request = { client, expectedPath, data -> - client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data, upsert = true) + request = { client, expectedPath, data, meta -> + client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data, upsert = true) { + userMetadata = meta + } } ) } @@ -132,8 +153,10 @@ class BucketApiTest { testUploadMethod( method = HttpMethod.Put, urlPath = "/object/$bucketId/data.png", - request = { client, expectedPath, data -> - client.storage[bucketId].update(expectedPath, data) + request = { client, expectedPath, data, meta -> + client.storage[bucketId].update(expectedPath, data) { + userMetadata = meta + } }, extra = { assertEquals( @@ -157,8 +180,10 @@ class BucketApiTest { "Upsert header should be true" ) }, - request = { client, expectedPath, data -> - client.storage[bucketId].update(expectedPath, data, upsert = true) + request = { client, expectedPath, data, meta -> + client.storage[bucketId].update(expectedPath, data, upsert = true) { + userMetadata = meta + } } ) } @@ -429,6 +454,69 @@ class BucketApiTest { } } + @Test + fun testInfo() { + runTest { + val expectedPath = "data.png" + val file = FileObjectV2( + "data.png", + "id", + "version", + createdAt = Clock.System.now(), + metadata = null, + size = 0, + rawContentType = "image/png", + etag = null, + lastModified = null, + cacheControl = null + ) + val client = createMockedSupabaseClient(configuration = configureClient) { + assertMethodIs(HttpMethod.Get, it.method) + assertPathIs("/object/info/$bucketId/$expectedPath", it.url.pathAfterVersion()) + respond( + content = Json.encodeToString(file), + headers = headersOf( + HttpHeaders.ContentType, + ContentType.Application.Json.toString() + ) + ) + } + val data = client.storage[bucketId].info(expectedPath) + assertEquals(file.copy(serializer = client.storage.serializer), data, "Data should be $file") + } + } + + @Test + fun testExistsWithExistingFile() { + runTest { + val expectedPath = "data.png" + val client = createMockedSupabaseClient(configuration = configureClient) { + assertMethodIs(HttpMethod.Head, it.method) + assertPathIs("/object/$bucketId/$expectedPath", it.url.pathAfterVersion()) + respondOk() + } + val exists = client.storage[bucketId].exists(expectedPath) + assertTrue { exists } + } + } + + @Test + fun testExistsWithNonExistingFile() { + val statusCodes = listOf(404, 400) + for(code in statusCodes) { + runTest { + val expectedPath = "data.png" + val client = createMockedSupabaseClient(configuration = configureClient) { + assertMethodIs(HttpMethod.Head, it.method) + assertPathIs("/object/$bucketId/$expectedPath", it.url.pathAfterVersion()) + respondError(HttpStatusCode(code, "Not Found")) + } + val exists = client.storage[bucketId].exists(expectedPath) + assertFalse { exists } + } + } + } + private fun testDownloadWithTransform( authenticated: Boolean ) { @@ -463,18 +551,24 @@ class BucketApiTest { } } + @OptIn(ExperimentalEncodingApi::class) private fun testUploadMethod( method: HttpMethod, urlPath: String, expectedPath: String = "data.png", extra: suspend MockRequestHandleScope.(HttpRequestData) -> Unit, - request: suspend (client: SupabaseClient, expectedPath: String, data: ByteArray) -> FileUploadResponse + request: suspend (client: SupabaseClient, expectedPath: String, data: ByteArray, metadata: JsonObject) -> FileUploadResponse ) { runTest { val expectedData = byteArrayOf(1, 2, 3) + val expectedMetadata = buildJsonObject { + put("key", "value") + } val client = createMockedSupabaseClient(configuration = configureClient) { val data = it.body.toByteArray() assertMethodIs(method, it.method) + val metadata = Json.decodeFromString(Base64.decode(it.headers["x-metadata"] ?: error("Metadata should not be null")).decodeToString()) + assertEquals(expectedMetadata, metadata, "Metadata should be $expectedMetadata") assertPathIs(urlPath, it.url.pathAfterVersion()) assertContentEquals(expectedData, data, "Data should be [1, 2, 3]") assertEquals(ContentType.Image.PNG, it.body.contentType, "Content type should be image/png") @@ -492,7 +586,7 @@ class BucketApiTest { ) ) } - val response = request(client, expectedPath, expectedData) + val response = request(client, expectedPath, expectedData, expectedMetadata) assertEquals("someBucket/$expectedPath", response.key, "Key should be $expectedPath") assertEquals("someId", response.id, "Id should be someId") assertEquals(expectedPath, response.path, "Path should be $expectedPath") From 1f48846b5b1a05fc9d50f3e6ab20242782283f2c Mon Sep 17 00:00:00 2001 From: Jan Date: Sun, 25 Aug 2024 14:22:00 +0200 Subject: [PATCH 04/13] fix docs --- .../kotlin/io/github/jan/supabase/storage/BucketApi.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt index 0895820fc..aa164c974 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt @@ -248,7 +248,6 @@ sealed interface BucketApi { /** * Searches for files with the given [prefix] and [filter] - * @return The filtered bucket items * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues From 88ceb2f309f77ff0f362ddddc5094728990e6e68 Mon Sep 17 00:00:00 2001 From: Jan Date: Sun, 25 Aug 2024 14:43:26 +0200 Subject: [PATCH 05/13] suppress warning --- .../kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt | 1 + 1 file changed, 1 insertion(+) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index df2625c56..2b3a41306 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -285,6 +285,7 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage return FileUploadResponse(id, path, key) } + @Suppress("LongParameterList") //TODO: maybe refactor @OptIn(ExperimentalEncodingApi::class) private fun HttpRequestBuilder.defaultUploadRequest( path: String, From 4ca681d2e9b329b85e969334791dbc9e5ccf85cf Mon Sep 17 00:00:00 2001 From: Jan Date: Sun, 25 Aug 2024 18:51:26 +0200 Subject: [PATCH 06/13] suppress warning for signed urls --- .../kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt | 1 + 1 file changed, 1 insertion(+) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index 2b3a41306..6051ccf32 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -266,6 +266,7 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage } @OptIn(ExperimentalEncodingApi::class) + @Suppress("LongParameterList") //TODO: maybe refactor internal suspend fun uploadToSignedUrl( path: String, token: String, From b442961e00538779bb8e8951bf8bfca1002b1d48 Mon Sep 17 00:00:00 2001 From: Jan Date: Mon, 26 Aug 2024 19:04:33 +0200 Subject: [PATCH 07/13] Move upsert parameter to new FileOptionBuilder --- .../github/jan/supabase/storage/BucketApi.kt | 25 +++++++------------ .../jan/supabase/storage/BucketApiImpl.kt | 22 ++++++---------- .../jan/supabase/storage/FileOptionBuilder.kt | 5 ++++ .../src/commonTest/kotlin/BucketApiTest.kt | 12 ++++++--- 4 files changed, 30 insertions(+), 34 deletions(-) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt index aa164c974..5c76d4958 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt @@ -34,36 +34,33 @@ sealed interface BucketApi { * Uploads a file in [bucketId] under [path] * @param path The path to upload the file to * @param data The data to upload - * @param upsert Whether to overwrite an existing file * @return the key to the uploaded file * @throws IllegalArgumentException if data to upload is empty * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun upload(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { + suspend fun upload(path: String, data: ByteArray, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } - return upload(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) + return upload(path, UploadData(ByteReadChannel(data), data.size.toLong()), options) } /** * Uploads a file in [bucketId] under [path] * @param path The path to upload the file to * @param data The data to upload - * @param upsert Whether to overwrite an existing file * @return the key to the uploaded file * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun upload(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse + suspend fun upload(path: String, data: UploadData, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Uploads a file in [bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param data The data to upload - * @param upsert Whether to overwrite an existing file * @return the key of the uploaded file * @throws IllegalArgumentException if data to upload is empty */ @@ -71,11 +68,10 @@ sealed interface BucketApi { path: String, token: String, data: ByteArray, - upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {} ): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } - return uploadToSignedUrl(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) + return uploadToSignedUrl(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), options) } /** @@ -83,42 +79,39 @@ sealed interface BucketApi { * @param path The path to upload the file to * @param token The presigned url token * @param data The data to upload - * @param upsert Whether to overwrite an existing file * @return the key of the uploaded file * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues * @throws HttpRequestException on network related issues */ - suspend fun uploadToSignedUrl(path: String, token: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse + suspend fun uploadToSignedUrl(path: String, token: String, data: UploadData, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Updates a file in [bucketId] under [path] * @param path The path to update the file to * @param data The new data - * @param upsert Whether to overwrite an existing file * @return the key to the updated file * @throws IllegalArgumentException if data to upload is empty * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun update(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { + suspend fun update(path: String, data: ByteArray, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } - return update(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) + return update(path, UploadData(ByteReadChannel(data), data.size.toLong()), options) } /** * Updates a file in [bucketId] under [path] * @param path The path to update the file to * @param data The new data - * @param upsert Whether to overwrite an existing file * @return the key to the updated file * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun update(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse + suspend fun update(path: String, data: UploadData, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Deletes all files in [bucketId] with in [paths] diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index 6051ccf32..76c83d165 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -44,21 +44,19 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override suspend fun update( path: String, data: UploadData, - upsert: Boolean, options: FileOptionBuilder.() -> Unit ): FileUploadResponse = uploadOrUpdate( - HttpMethod.Put, bucketId, path, data, upsert, options + HttpMethod.Put, bucketId, path, data, options ) override suspend fun uploadToSignedUrl( path: String, token: String, data: UploadData, - upsert: Boolean, options: FileOptionBuilder.() -> Unit ): FileUploadResponse { - return uploadToSignedUrl(path, token, data, upsert, options) {} + return uploadToSignedUrl(path, token, data, options) {} } override suspend fun createSignedUploadUrl(path: String): UploadSignedUrl { @@ -77,11 +75,10 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override suspend fun upload( path: String, data: UploadData, - upsert: Boolean, options: FileOptionBuilder.() -> Unit ): FileUploadResponse = uploadOrUpdate( - HttpMethod.Post, bucketId, path, data, upsert, options + HttpMethod.Post, bucketId, path, data, options ) override suspend fun delete(paths: Collection) { @@ -249,14 +246,13 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage bucket: String, path: String, data: UploadData, - upsert: Boolean, options: FileOptionBuilder.() -> Unit, extra: HttpRequestBuilder.() -> Unit = {} ): FileUploadResponse { val optionBuilder = FileOptionBuilder(storage.serializer).apply(options) val response = storage.api.request("object/$bucket/$path") { this.method = method - defaultUploadRequest(path, data, upsert, optionBuilder, extra) + defaultUploadRequest(path, data, optionBuilder, extra) }.body() val key = response["Key"]?.jsonPrimitive?.content ?: error("Expected a key in a upload response") @@ -271,14 +267,13 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage path: String, token: String, data: UploadData, - upsert: Boolean, options: FileOptionBuilder.() -> Unit, extra: HttpRequestBuilder.() -> Unit = {} ): FileUploadResponse { val optionBuilder = FileOptionBuilder(storage.serializer).apply(options) val response = storage.api.put("object/upload/sign/$bucketId/$path") { parameter("token", token) - defaultUploadRequest(path, data, upsert, optionBuilder, extra) + defaultUploadRequest(path, data, optionBuilder, extra) }.body() val key = response["Key"]?.jsonPrimitive?.content ?: error("Expected a key in a upload response") @@ -291,17 +286,16 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage private fun HttpRequestBuilder.defaultUploadRequest( path: String, data: UploadData, - upsert: Boolean, optionBuilder: FileOptionBuilder, extra: HttpRequestBuilder.() -> Unit ) { setBody(object : OutgoingContent.ReadChannelContent() { - override val contentType: ContentType = ContentType.defaultForFilePath(path) + override val contentType: ContentType = optionBuilder.contentType ?: ContentType.defaultForFilePath(path) override val contentLength: Long = data.size override fun readFrom(): ByteReadChannel = data.stream }) - header(HttpHeaders.ContentType, ContentType.defaultForFilePath(path)) - header(UPSERT_HEADER, upsert.toString()) + header(HttpHeaders.ContentType, optionBuilder.contentType ?: ContentType.defaultForFilePath(path)) + header(UPSERT_HEADER, optionBuilder.upsert.toString()) optionBuilder.userMetadata?.let { header("x-metadata", Base64.Default.encode(it.toString().encodeToByteArray()).also((::println))) } diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt index d59c30e58..67b998ca2 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt @@ -2,6 +2,7 @@ package io.github.jan.supabase.storage import io.github.jan.supabase.SupabaseSerializer import io.github.jan.supabase.encodeToJsonElement +import io.ktor.http.ContentType import kotlinx.serialization.json.JsonObject import kotlinx.serialization.json.JsonObjectBuilder import kotlinx.serialization.json.buildJsonObject @@ -11,10 +12,14 @@ import kotlinx.serialization.json.jsonObject * Builder for uploading files with additional options * @param serializer The serializer to use for encoding the metadata * @param userMetadata The user metadata to upload with the file + * @param upsert Whether to update the file if it already exists + * @param contentType The content type of the file. If null, the content type will be inferred from the file extension */ class FileOptionBuilder( @PublishedApi internal val serializer: SupabaseSerializer, var userMetadata: JsonObject? = null, + var upsert: Boolean = false, + var contentType: ContentType? = null, ) { /** diff --git a/Storage/src/commonTest/kotlin/BucketApiTest.kt b/Storage/src/commonTest/kotlin/BucketApiTest.kt index 2dc3640cf..ac09886ad 100644 --- a/Storage/src/commonTest/kotlin/BucketApiTest.kt +++ b/Storage/src/commonTest/kotlin/BucketApiTest.kt @@ -89,8 +89,9 @@ class BucketApiTest { ) }, request = { client, expectedPath, data, meta -> - client.storage[bucketId].upload(expectedPath, data, upsert = true) { + client.storage[bucketId].upload(expectedPath, data) { userMetadata = meta + upsert = true } } ) @@ -115,8 +116,9 @@ class BucketApiTest { ) }, request = { client, expectedPath, data, meta -> - client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data, upsert = false) { + client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data) { userMetadata = meta + upsert = false } } ) @@ -141,8 +143,9 @@ class BucketApiTest { ) }, request = { client, expectedPath, data, meta -> - client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data, upsert = true) { + client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data) { userMetadata = meta + upsert = true } } ) @@ -181,8 +184,9 @@ class BucketApiTest { ) }, request = { client, expectedPath, data, meta -> - client.storage[bucketId].update(expectedPath, data, upsert = true) { + client.storage[bucketId].update(expectedPath, data) { userMetadata = meta + upsert = true } } ) From 98556c9cce9f04e03ad4f258b28c1b45e60c672a Mon Sep 17 00:00:00 2001 From: Jan Date: Mon, 26 Aug 2024 19:09:46 +0200 Subject: [PATCH 08/13] remove comma --- .../commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt index 7749c6906..8e106f4d1 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/Storage.kt @@ -131,7 +131,7 @@ sealed interface Storage : MainPlugin, CustomSerializationPlugin data class Resumable( var cache: ResumableCache? = null, var retryTimeout: Duration = 5.seconds, - var onlyUpdateStateAfterChunk: Boolean = false, + var onlyUpdateStateAfterChunk: Boolean = false ) { /** From a97dce6b470e49766aff4ac94c7c871643946cd9 Mon Sep 17 00:00:00 2001 From: Jan Date: Mon, 26 Aug 2024 19:11:50 +0200 Subject: [PATCH 09/13] remove println --- .../kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index 76c83d165..4757c0ab5 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -297,7 +297,7 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage header(HttpHeaders.ContentType, optionBuilder.contentType ?: ContentType.defaultForFilePath(path)) header(UPSERT_HEADER, optionBuilder.upsert.toString()) optionBuilder.userMetadata?.let { - header("x-metadata", Base64.Default.encode(it.toString().encodeToByteArray()).also((::println))) + header("x-metadata", Base64.Default.encode(it.toString().encodeToByteArray())) } extra() } From cc395ed6d1c028c907933aaf74a355844256c0f8 Mon Sep 17 00:00:00 2001 From: Jan Tennert Date: Mon, 16 Sep 2024 19:52:18 +0200 Subject: [PATCH 10/13] Refactor Storage download & upload (flow) implementation --- .../github/jan/supabase/storage/JvmUtils.kt | 82 ++++--- .../jan/supabase/storage/AndroidUtils.kt | 26 +- .../github/jan/supabase/storage/BucketApi.kt | 55 ++--- .../jan/supabase/storage/BucketApiImpl.kt | 127 ++++------ .../supabase/storage/DownloadOptionBuilder.kt | 29 +++ .../jan/supabase/storage/FileOptionBuilder.kt | 41 ---- .../io/github/jan/supabase/storage/FlowExt.kt | 226 ----------------- .../jan/supabase/storage/FlowExtension.kt | 229 ++++++++++++++++++ .../supabase/storage/UploadOptionBuilder.kt | 27 +++ .../commonTest/kotlin/BucketApiFlowTest.kt | 24 +- .../src/commonTest/kotlin/BucketApiTest.kt | 105 +------- .../network/KtorSupabaseHttpClient.kt | 5 + settings.gradle.kts | 1 + 13 files changed, 441 insertions(+), 536 deletions(-) create mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/DownloadOptionBuilder.kt delete mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt delete mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt create mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt create mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/UploadOptionBuilder.kt diff --git a/Storage/src/androidAndJvmMain/kotlin/io/github/jan/supabase/storage/JvmUtils.kt b/Storage/src/androidAndJvmMain/kotlin/io/github/jan/supabase/storage/JvmUtils.kt index 2586f13d5..213e319fa 100644 --- a/Storage/src/androidAndJvmMain/kotlin/io/github/jan/supabase/storage/JvmUtils.kt +++ b/Storage/src/androidAndJvmMain/kotlin/io/github/jan/supabase/storage/JvmUtils.kt @@ -11,163 +11,170 @@ import kotlin.io.path.writeBytes * Uploads a file in [BucketApi.bucketId] under [path] * @param path The path to upload the file to * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return the key to the uploaded file */ -suspend fun BucketApi.upload(path: String, file: File, upsert: Boolean = false) = upload(path, UploadData(file.readChannel(), file.length()), upsert) +suspend fun BucketApi.upload(path: String, file: File, options: UploadOptionBuilder.() -> Unit = {}) = upload(path, UploadData(file.readChannel(), file.length()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] * @param path The path to upload the file to * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.uploadAsFlow(path: String, file: File, upsert: Boolean = false) = uploadAsFlow(path, UploadData(file.readChannel(), file.length()), upsert) +fun BucketApi.uploadAsFlow(path: String, file: File, options: UploadOptionBuilder.() -> Unit = {}) = uploadAsFlow(path, UploadData(file.readChannel(), file.length()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] * @param path The path to upload the file to * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return the key to the uploaded file */ -suspend fun BucketApi.upload(path: String, file: Path, upsert: Boolean = false) = upload(path, UploadData(file.readChannel(), file.fileSize()), upsert) +suspend fun BucketApi.upload(path: String, file: Path, options: UploadOptionBuilder.() -> Unit = {}) = upload(path, UploadData(file.readChannel(), file.fileSize()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] * @param path The path to upload the file to * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.uploadAsFlow(path: String, file: Path, upsert: Boolean = false) = uploadAsFlow(path, UploadData(file.readChannel(), file.fileSize()), upsert) +fun BucketApi.uploadAsFlow(path: String, file: Path, options: UploadOptionBuilder.() -> Unit = {}) = uploadAsFlow(path, UploadData(file.readChannel(), file.fileSize()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return the key to the uploaded file */ -suspend fun BucketApi.uploadToSignedUrl(path: String, token: String, file: File, upsert: Boolean = false) = uploadToSignedUrl(path, token, UploadData(file.readChannel(), file.length()), upsert) +suspend fun BucketApi.uploadToSignedUrl(path: String, token: String, file: File, options: UploadOptionBuilder.() -> Unit = {}) = uploadToSignedUrl(path, token, UploadData(file.readChannel(), file.length()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, file: File, upsert: Boolean = false) = uploadToSignedUrlAsFlow(path, token, UploadData(file.readChannel(), file.length()), upsert) +fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, file: File, options: UploadOptionBuilder.() -> Unit = {}) = uploadToSignedUrlAsFlow(path, token, UploadData(file.readChannel(), file.length()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload */ -suspend fun BucketApi.uploadToSignedUrl(path: String, token: String, file: Path, upsert: Boolean = false) = uploadToSignedUrl(path, token, UploadData(file.readChannel(), file.fileSize()), upsert) +suspend fun BucketApi.uploadToSignedUrl(path: String, token: String, file: Path, options: UploadOptionBuilder.() -> Unit = {}) = uploadToSignedUrl(path, token, UploadData(file.readChannel(), file.fileSize()), options) /** * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param file The file to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, file: Path, upsert: Boolean = false) = uploadToSignedUrlAsFlow(path, token, UploadData(file.readChannel(), file.fileSize()), upsert) +fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, file: Path, options: UploadOptionBuilder.() -> Unit = {}) = uploadToSignedUrlAsFlow(path, token, UploadData(file.readChannel(), file.fileSize()), options) /** * Updates a file in [BucketApi.bucketId] under [path] * @param path The path to be updated * @param file The new file - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload */ -suspend fun BucketApi.update(path: String, file: Path, upsert: Boolean = false) = update(path, UploadData(file.readChannel(), file.fileSize()), upsert) +suspend fun BucketApi.update(path: String, file: Path, options: UploadOptionBuilder.() -> Unit = {}) = update(path, UploadData(file.readChannel(), file.fileSize()), options) /** * Updates a file in [BucketApi.bucketId] under [path] * @param path The path to be updated * @param file The new file - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.updateAsFlow(path: String, file: Path, upsert: Boolean = false) = updateAsFlow(path, UploadData(file.readChannel(), file.fileSize()), upsert) +fun BucketApi.updateAsFlow(path: String, file: Path, options: UploadOptionBuilder.() -> Unit = {}) = updateAsFlow(path, UploadData(file.readChannel(), file.fileSize()), options) /** * Updates a file in [BucketApi.bucketId] under [path] * @param path The path to be updated * @param file The new file - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload */ -suspend fun BucketApi.update(path: String, file: File, upsert: Boolean = false) = update(path, UploadData(file.readChannel(), file.length()), upsert) +suspend fun BucketApi.update(path: String, file: File, options: UploadOptionBuilder.() -> Unit = {}) = update(path, UploadData(file.readChannel(), file.length()), options) /** * Updates a file in [BucketApi.bucketId] under [path] * @param path The path to be updated * @param file The new file - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the uploaded file */ -fun BucketApi.updateAsFlow(path: String, file: File, upsert: Boolean = false) = updateAsFlow(path, UploadData(file.readChannel(), file.length()), upsert) +fun BucketApi.updateAsFlow(path: String, file: File, options: UploadOptionBuilder.() -> Unit = {}) = updateAsFlow(path, UploadData(file.readChannel(), file.length()), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download */ -suspend fun BucketApi.downloadAuthenticatedTo(path: String, file: File, transform: ImageTransformation.() -> Unit = {}) = downloadAuthenticated(path, file.writeChannel(), transform) +suspend fun BucketApi.downloadAuthenticatedTo(path: String, file: File, options: DownloadOptionBuilder.() -> Unit = {}) = downloadAuthenticated(path, file.writeChannel(), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download * @return A flow that emits the download progress and at last the key to the downloaded file */ -fun BucketApi.downloadAuthenticatedToAsFlow(path: String, file: File, transform: ImageTransformation.() -> Unit = {}) = downloadAuthenticatedAsFlow(path, file.writeChannel(), transform) +fun BucketApi.downloadAuthenticatedToAsFlow(path: String, file: File, options: DownloadOptionBuilder.() -> Unit = {}) = downloadAuthenticatedAsFlow(path, file.writeChannel(), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download */ -suspend fun BucketApi.downloadAuthenticatedTo(path: String, file: Path, transform: ImageTransformation.() -> Unit = {}) = downloadAuthenticated(path, file.toFile().writeChannel(), transform) +suspend fun BucketApi.downloadAuthenticatedTo(path: String, file: Path, options: DownloadOptionBuilder.() -> Unit = {}) = downloadAuthenticated(path, file.toFile().writeChannel(), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download * @return A flow that emits the download progress and at last the key to the downloaded file */ -fun BucketApi.downloadAuthenticatedToAsFlow(path: String, file: Path, transform: ImageTransformation.() -> Unit = {}) = downloadAuthenticatedAsFlow(path, file.toFile().writeChannel(), transform) +fun BucketApi.downloadAuthenticatedToAsFlow(path: String, file: Path, options: DownloadOptionBuilder.() -> Unit = {}) = downloadAuthenticatedAsFlow(path, file.toFile().writeChannel(), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download */ -suspend fun BucketApi.downloadPublicTo(path: String, file: File, transform: ImageTransformation.() -> Unit = {}) = downloadPublic(path, file.writeChannel(), transform) +suspend fun BucketApi.downloadPublicTo(path: String, file: File, options: DownloadOptionBuilder.() -> Unit = {}) = downloadPublic(path, file.writeChannel(), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download * @return A flow that emits the download progress and at last the key to the downloaded file */ -fun BucketApi.downloadPublicToAsFlow(path: String, file: File, transform: ImageTransformation.() -> Unit = {}) = downloadPublicAsFlow(path, file.writeChannel(), transform) +fun BucketApi.downloadPublicToAsFlow(path: String, file: File, options: DownloadOptionBuilder.() -> Unit = {}) = downloadPublicAsFlow(path, file.writeChannel(), options) /** * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download */ -suspend fun BucketApi.downloadPublicTo(path: String, file: Path, transform: ImageTransformation.() -> Unit = {}) { - val bytes = downloadPublic(path, transform) +suspend fun BucketApi.downloadPublicTo(path: String, file: Path, options: DownloadOptionBuilder.() -> Unit = {}) { + val bytes = downloadPublic(path, options) file.writeBytes(bytes) } @@ -175,6 +182,7 @@ suspend fun BucketApi.downloadPublicTo(path: String, file: Path, transform: Imag * Downloads a file from [BucketApi.bucketId] under [path] and saves it to [file] * @param path The path to download the file from * @param file The file to save the data to + * @param options Additional options for the download * @return A flow that emits the download progress and at last the key to the downloaded file */ -fun BucketApi.downloadPublicToAsFlow(path: String, file: Path, transform: ImageTransformation.() -> Unit = {}) = downloadPublicAsFlow(path, file.toFile().writeChannel(), transform) \ No newline at end of file +fun BucketApi.downloadPublicToAsFlow(path: String, file: Path, options: DownloadOptionBuilder.() -> Unit = {}) = downloadPublicAsFlow(path, file.toFile().writeChannel(), options) \ No newline at end of file diff --git a/Storage/src/androidMain/kotlin/io/github/jan/supabase/storage/AndroidUtils.kt b/Storage/src/androidMain/kotlin/io/github/jan/supabase/storage/AndroidUtils.kt index 48f883f99..e30c50706 100644 --- a/Storage/src/androidMain/kotlin/io/github/jan/supabase/storage/AndroidUtils.kt +++ b/Storage/src/androidMain/kotlin/io/github/jan/supabase/storage/AndroidUtils.kt @@ -9,56 +9,56 @@ import io.ktor.utils.io.jvm.javaio.toByteReadChannel * Uploads a file in [BucketApi.bucketId] under [path] * @param path The path to upload the file to * @param uri The uri to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return the key to the updated file */ -suspend fun BucketApi.upload(path: String, uri: Uri, upsert: Boolean = false) = upload(path, UploadData(uri.readChannel(), uri.contentSize), upsert) +suspend fun BucketApi.upload(path: String, uri: Uri, options: UploadOptionBuilder.() -> Unit = {}) = upload(path, UploadData(uri.readChannel(), uri.contentSize), options) /** * Uploads a file in [BucketApi.bucketId] under [path] * @param path The path to upload the file to * @param uri The uri to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the updated file */ -fun BucketApi.uploadAsFlow(path: String, uri: Uri, upsert: Boolean = false) = uploadAsFlow(path, UploadData(uri.readChannel(), uri.contentSize), upsert) +fun BucketApi.uploadAsFlow(path: String, uri: Uri, options: UploadOptionBuilder.() -> Unit = {}) = uploadAsFlow(path, UploadData(uri.readChannel(), uri.contentSize), options) /** * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param uri The uri to upload * @return the key to the updated file */ -suspend fun BucketApi.uploadToSignedUrl(path: String, token: String, uri: Uri, upsert: Boolean = false) = uploadToSignedUrl(path, token, UploadData(uri.readChannel(), uri.contentSize), upsert) +suspend fun BucketApi.uploadToSignedUrl(path: String, token: String, uri: Uri, options: UploadOptionBuilder.() -> Unit = {}) = uploadToSignedUrl(path, token, UploadData(uri.readChannel(), uri.contentSize), options) /** * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url * @param path The path to upload the file to - * @param token The presigned url token + * @param token The pre-signed url token * @param uri The uri to upload - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the updated file */ -fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, uri: Uri, upsert: Boolean = false) = uploadToSignedUrlAsFlow(path, token, UploadData(uri.readChannel(), uri.contentSize), upsert) +fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, uri: Uri, options: UploadOptionBuilder.() -> Unit = {}) = uploadToSignedUrlAsFlow(path, token, UploadData(uri.readChannel(), uri.contentSize), options) /** * Updates a file in [BucketApi.bucketId] under [path] * @param path The path to update the file to * @param uri The uri to update - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return the key to the updated file */ -suspend fun BucketApi.update(path: String, uri: Uri, upsert: Boolean = false) = update(path, UploadData(uri.readChannel(), uri.contentSize), upsert) +suspend fun BucketApi.update(path: String, uri: Uri, options: UploadOptionBuilder.() -> Unit = {}) = update(path, UploadData(uri.readChannel(), uri.contentSize), options) /** * Updates a file in [BucketApi.bucketId] under [path] * @param path The path to update the file to * @param uri The uri to update - * @param upsert Whether to overwrite an existing file + * @param options Additional options for the upload * @return A flow that emits the upload progress and at last the key to the updated file */ -fun BucketApi.updateAsFlow(path: String, uri: Uri, upsert: Boolean = false) = updateAsFlow(path, UploadData(uri.readChannel(), uri.contentSize), upsert) +fun BucketApi.updateAsFlow(path: String, uri: Uri, options: UploadOptionBuilder.() -> Unit = {}) = updateAsFlow(path, UploadData(uri.readChannel(), uri.contentSize), options) @SuppressLint("Recycle") //toByteReadChannel closes the input stream automatically private fun Uri.readChannel(): ByteReadChannel { diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt index 5c76d4958..f9f943992 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApi.kt @@ -34,13 +34,14 @@ sealed interface BucketApi { * Uploads a file in [bucketId] under [path] * @param path The path to upload the file to * @param data The data to upload + * @param options Additional options for the upload * @return the key to the uploaded file * @throws IllegalArgumentException if data to upload is empty * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun upload(path: String, data: ByteArray, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { + suspend fun upload(path: String, data: ByteArray, options: UploadOptionBuilder.() -> Unit = {}): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } return upload(path, UploadData(ByteReadChannel(data), data.size.toLong()), options) } @@ -49,18 +50,20 @@ sealed interface BucketApi { * Uploads a file in [bucketId] under [path] * @param path The path to upload the file to * @param data The data to upload + * @param options Additional options for the upload * @return the key to the uploaded file * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun upload(path: String, data: UploadData, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse + suspend fun upload(path: String, data: UploadData, options: UploadOptionBuilder.() -> Unit = {}): FileUploadResponse /** - * Uploads a file in [bucketId] under [path] using a presigned url + * Uploads a file in [bucketId] under [path] using a pre-signed url * @param path The path to upload the file to * @param token The pre-signed url token * @param data The data to upload + * @param options Additional options for the upload * @return the key of the uploaded file * @throws IllegalArgumentException if data to upload is empty */ @@ -68,7 +71,7 @@ sealed interface BucketApi { path: String, token: String, data: ByteArray, - options: FileOptionBuilder.() -> Unit = {} + options: UploadOptionBuilder.() -> Unit = {} ): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } return uploadToSignedUrl(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), options) @@ -79,25 +82,27 @@ sealed interface BucketApi { * @param path The path to upload the file to * @param token The presigned url token * @param data The data to upload + * @param options Additional options for the upload * @return the key of the uploaded file * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues * @throws HttpRequestException on network related issues */ - suspend fun uploadToSignedUrl(path: String, token: String, data: UploadData, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse + suspend fun uploadToSignedUrl(path: String, token: String, data: UploadData, options: UploadOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Updates a file in [bucketId] under [path] * @param path The path to update the file to * @param data The new data + * @param options Additional options for the upload * @return the key to the updated file * @throws IllegalArgumentException if data to upload is empty * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun update(path: String, data: ByteArray, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse { + suspend fun update(path: String, data: ByteArray, options: UploadOptionBuilder.() -> Unit = {}): FileUploadResponse { require(data.isNotEmpty()) { "The data to upload should not be empty" } return update(path, UploadData(ByteReadChannel(data), data.size.toLong()), options) } @@ -106,12 +111,13 @@ sealed interface BucketApi { * Updates a file in [bucketId] under [path] * @param path The path to update the file to * @param data The new data + * @param options Additional options for the upload * @return the key to the updated file * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun update(path: String, data: UploadData, options: FileOptionBuilder.() -> Unit = {}): FileUploadResponse + suspend fun update(path: String, data: UploadData, options: UploadOptionBuilder.() -> Unit = {}): FileUploadResponse /** * Deletes all files in [bucketId] with in [paths] @@ -197,46 +203,46 @@ sealed interface BucketApi { /** * Downloads a file from [bucketId] under [path] * @param path The path to download - * @param transform The transformation to apply to the image + * @param options Additional options for the download * @return The file as a byte array * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun downloadAuthenticated(path: String, transform: ImageTransformation.() -> Unit = {}): ByteArray + suspend fun downloadAuthenticated(path: String, options: DownloadOptionBuilder.() -> Unit = {}): ByteArray /** * Downloads a file from [bucketId] under [path] * @param path The path to download * @param channel The channel to write the data to - * @param transform The transformation to apply to the image + * @param options Additional options for the download * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun downloadAuthenticated(path: String, channel: ByteWriteChannel, transform: ImageTransformation.() -> Unit = {}) + suspend fun downloadAuthenticated(path: String, channel: ByteWriteChannel, options: DownloadOptionBuilder.() -> Unit = {}) /** * Downloads a file from [bucketId] under [path] using the public url * @param path The path to download - * @param transform The transformation to apply to the image + * @param options Additional options for the download * @return The file as a byte array * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun downloadPublic(path: String, transform: ImageTransformation.() -> Unit = {}): ByteArray + suspend fun downloadPublic(path: String, options: DownloadOptionBuilder.() -> Unit = {}): ByteArray /** * Downloads a file from [bucketId] under [path] using the public url * @param path The path to download * @param channel The channel to write the data to - * @param transform The transformation to apply to the image + * @param options Additional options for the download * @throws RestException or one of its subclasses if receiving an error response * @throws HttpRequestTimeoutException if the request timed out * @throws HttpRequestException on network related issues */ - suspend fun downloadPublic(path: String, channel: ByteWriteChannel, transform: ImageTransformation.() -> Unit = {}) + suspend fun downloadPublic(path: String, channel: ByteWriteChannel, options: DownloadOptionBuilder.() -> Unit = {}) /** @@ -250,25 +256,6 @@ sealed interface BucketApi { filter: BucketListFilter.() -> Unit = {} ): List - /** - * Returns information about the file under [path] - * @param path The path to get information about - * @return The file object - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ - suspend fun info(path: String): FileObjectV2 - - /** - * Checks if a file exists under [path] - * @return true if the file exists, false otherwise - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ - suspend fun exists(path: String): Boolean - /** * Changes the bucket's public status to [public] * @throws RestException or one of its subclasses if receiving an error response diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt index 4757c0ab5..ea31d0162 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/BucketApiImpl.kt @@ -1,6 +1,5 @@ package io.github.jan.supabase.storage -import io.github.jan.supabase.exceptions.RestException import io.github.jan.supabase.putJsonObject import io.github.jan.supabase.safeBody import io.github.jan.supabase.storage.BucketApi.Companion.UPSERT_HEADER @@ -9,14 +8,12 @@ import io.github.jan.supabase.storage.resumable.ResumableClientImpl import io.ktor.client.call.body import io.ktor.client.request.HttpRequestBuilder import io.ktor.client.request.header -import io.ktor.client.request.parameter import io.ktor.client.request.setBody import io.ktor.client.request.url import io.ktor.client.statement.bodyAsChannel import io.ktor.http.ContentType import io.ktor.http.HttpHeaders import io.ktor.http.HttpMethod -import io.ktor.http.HttpStatusCode import io.ktor.http.Url import io.ktor.http.content.OutgoingContent import io.ktor.http.defaultForFilePath @@ -31,8 +28,6 @@ import kotlinx.serialization.json.jsonPrimitive import kotlinx.serialization.json.put import kotlinx.serialization.json.putJsonArray import kotlinx.serialization.json.putJsonObject -import kotlin.io.encoding.Base64 -import kotlin.io.encoding.ExperimentalEncodingApi import kotlin.time.Duration internal class BucketApiImpl(override val bucketId: String, val storage: StorageImpl, resumableCache: ResumableCache) : BucketApi { @@ -44,19 +39,21 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override suspend fun update( path: String, data: UploadData, - options: FileOptionBuilder.() -> Unit + options: UploadOptionBuilder.() -> Unit ): FileUploadResponse = uploadOrUpdate( - HttpMethod.Put, bucketId, path, data, options + HttpMethod.Put, defaultUploadUrl(path), data, options ) override suspend fun uploadToSignedUrl( path: String, token: String, data: UploadData, - options: FileOptionBuilder.() -> Unit + options: UploadOptionBuilder.() -> Unit ): FileUploadResponse { - return uploadToSignedUrl(path, token, data, options) {} + return uploadOrUpdate( + HttpMethod.Put, uploadToSignedUrlUrl(path, token), data, options + ) } override suspend fun createSignedUploadUrl(path: String): UploadSignedUrl { @@ -75,10 +72,10 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override suspend fun upload( path: String, data: UploadData, - options: FileOptionBuilder.() -> Unit + options: UploadOptionBuilder.() -> Unit ): FileUploadResponse = uploadOrUpdate( - HttpMethod.Post, bucketId, path, data, options + HttpMethod.Post, defaultUploadUrl(path), data, options ) override suspend fun delete(paths: Collection) { @@ -140,51 +137,54 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage override suspend fun downloadAuthenticated( path: String, - transform: ImageTransformation.() -> Unit - ): ByteArray { - return storage.api.rawRequest { - prepareDownloadRequest(path, false, transform) - }.body() - } + options: DownloadOptionBuilder.() -> Unit + ): ByteArray = normalDownloadRequest(path, false, options) override suspend fun downloadPublic( path: String, - transform: ImageTransformation.() -> Unit + options: DownloadOptionBuilder.() -> Unit + ): ByteArray = normalDownloadRequest(path, true, options) + + private suspend fun normalDownloadRequest( + path: String, + public: Boolean, + options: DownloadOptionBuilder.() -> Unit ): ByteArray { + val downloadOptions = DownloadOptionBuilder().apply(options) return storage.api.rawRequest { - prepareDownloadRequest(path, true, transform) + prepareDownloadRequest(path, public, downloadOptions) + downloadOptions.httpRequestOverrides.forEach { it() } }.body() } - override suspend fun downloadAuthenticated( path: String, channel: ByteWriteChannel, - transform: ImageTransformation.() -> Unit + options: DownloadOptionBuilder.() -> Unit ) { - channelDownloadRequest(path, channel, false, transform) + channelDownloadRequest(path, channel, false, options) } override suspend fun downloadPublic( path: String, channel: ByteWriteChannel, - transform: ImageTransformation.() -> Unit + options: DownloadOptionBuilder.() -> Unit ) { - channelDownloadRequest(path, channel, true, transform) + channelDownloadRequest(path, channel, true, options) } internal suspend fun channelDownloadRequest( path: String, channel: ByteWriteChannel, public: Boolean, - transform: ImageTransformation.() -> Unit, - extra: HttpRequestBuilder.() -> Unit = {} + options: DownloadOptionBuilder.() -> Unit, ) { + val downloadOptions = DownloadOptionBuilder().apply(options) storage.api.prepareRequest { - prepareDownloadRequest(path, public, transform) - extra() + prepareDownloadRequest(path, public, downloadOptions) + downloadOptions.httpRequestOverrides.forEach { it() } }.execute { it.bodyAsChannel().copyTo(channel) } @@ -194,18 +194,17 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage internal fun HttpRequestBuilder.prepareDownloadRequest( path: String, public: Boolean, - transform: ImageTransformation.() -> Unit + options: DownloadOptionBuilder ) { - val transformation = ImageTransformation().apply(transform).queryString() + val transformation = ImageTransformation().apply(options.transform).queryString() val url = when (public) { true -> if (transformation.isBlank()) publicUrl(path) else publicRenderUrl( path, - transform + options.transform ) - false -> if (transformation.isBlank()) authenticatedUrl(path) else authenticatedRenderUrl( path, - transform + options.transform ) } method = HttpMethod.Get @@ -222,37 +221,22 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage }).safeBody() } - override suspend fun info(path: String): FileObjectV2 { - val response = storage.api.get("object/info/$bucketId/$path") - return response.safeBody().copy(serializer = storage.serializer) - } + private fun defaultUploadUrl(path: String) = "object/$bucketId/$path" - override suspend fun exists(path: String): Boolean { - try { - storage.api.request("object/$bucketId/$path") { - method = HttpMethod.Head - } - return true - } catch (e: RestException) { - if (e.statusCode in listOf(HttpStatusCode.NotFound.value, HttpStatusCode.BadRequest.value)) return false - throw e - } - } + private fun uploadToSignedUrlUrl(path: String, token: String) = "object/upload/sign/$bucketId/$path?token=$token" - @OptIn(ExperimentalEncodingApi::class) - @Suppress("LongParameterList") //TODO: maybe refactor internal suspend fun uploadOrUpdate( method: HttpMethod, - bucket: String, - path: String, + url: String, data: UploadData, - options: FileOptionBuilder.() -> Unit, - extra: HttpRequestBuilder.() -> Unit = {} + options: UploadOptionBuilder.() -> Unit, ): FileUploadResponse { - val optionBuilder = FileOptionBuilder(storage.serializer).apply(options) - val response = storage.api.request("object/$bucket/$path") { + val path = url.substringAfterLast('/').substringBeforeLast("?") + val optionBuilder = UploadOptionBuilder(storage.serializer).apply(options) + val response = storage.api.request(url) { this.method = method - defaultUploadRequest(path, data, optionBuilder, extra) + defaultUploadRequest(path, data, optionBuilder) + optionBuilder.httpRequestOverrides.forEach { it() } }.body() val key = response["Key"]?.jsonPrimitive?.content ?: error("Expected a key in a upload response") @@ -261,33 +245,10 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage return FileUploadResponse(id, path, key) } - @OptIn(ExperimentalEncodingApi::class) - @Suppress("LongParameterList") //TODO: maybe refactor - internal suspend fun uploadToSignedUrl( - path: String, - token: String, - data: UploadData, - options: FileOptionBuilder.() -> Unit, - extra: HttpRequestBuilder.() -> Unit = {} - ): FileUploadResponse { - val optionBuilder = FileOptionBuilder(storage.serializer).apply(options) - val response = storage.api.put("object/upload/sign/$bucketId/$path") { - parameter("token", token) - defaultUploadRequest(path, data, optionBuilder, extra) - }.body() - val key = response["Key"]?.jsonPrimitive?.content - ?: error("Expected a key in a upload response") - val id = response["Id"]?.jsonPrimitive?.content ?: error("Expected an id in a upload response") - return FileUploadResponse(id, path, key) - } - - @Suppress("LongParameterList") //TODO: maybe refactor - @OptIn(ExperimentalEncodingApi::class) private fun HttpRequestBuilder.defaultUploadRequest( path: String, data: UploadData, - optionBuilder: FileOptionBuilder, - extra: HttpRequestBuilder.() -> Unit + optionBuilder: UploadOptionBuilder, ) { setBody(object : OutgoingContent.ReadChannelContent() { override val contentType: ContentType = optionBuilder.contentType ?: ContentType.defaultForFilePath(path) @@ -296,10 +257,6 @@ internal class BucketApiImpl(override val bucketId: String, val storage: Storage }) header(HttpHeaders.ContentType, optionBuilder.contentType ?: ContentType.defaultForFilePath(path)) header(UPSERT_HEADER, optionBuilder.upsert.toString()) - optionBuilder.userMetadata?.let { - header("x-metadata", Base64.Default.encode(it.toString().encodeToByteArray())) - } - extra() } override suspend fun changePublicStatusTo(public: Boolean) = storage.updateBucket(bucketId) { diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/DownloadOptionBuilder.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/DownloadOptionBuilder.kt new file mode 100644 index 000000000..9110f7d7b --- /dev/null +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/DownloadOptionBuilder.kt @@ -0,0 +1,29 @@ +package io.github.jan.supabase.storage + +import io.github.jan.supabase.network.HttpRequestOverride + +/** + * Builder for downloading files with additional options + */ +class DownloadOptionBuilder( + internal var transform: ImageTransformation.() -> Unit = {}, + internal val httpRequestOverrides: MutableList = mutableListOf() +) { + + /** + * Transforms the image before downloading + * @param transform The transformation to apply + */ + fun transform(transform: ImageTransformation.() -> Unit) { + this.transform = transform + } + + /** + * Overrides the HTTP request + * @param override The override to apply + */ + fun httpOverride(override: HttpRequestOverride) { + httpRequestOverrides.add(override) + } + +} \ No newline at end of file diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt deleted file mode 100644 index 67b998ca2..000000000 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileOptionBuilder.kt +++ /dev/null @@ -1,41 +0,0 @@ -package io.github.jan.supabase.storage - -import io.github.jan.supabase.SupabaseSerializer -import io.github.jan.supabase.encodeToJsonElement -import io.ktor.http.ContentType -import kotlinx.serialization.json.JsonObject -import kotlinx.serialization.json.JsonObjectBuilder -import kotlinx.serialization.json.buildJsonObject -import kotlinx.serialization.json.jsonObject - -/** - * Builder for uploading files with additional options - * @param serializer The serializer to use for encoding the metadata - * @param userMetadata The user metadata to upload with the file - * @param upsert Whether to update the file if it already exists - * @param contentType The content type of the file. If null, the content type will be inferred from the file extension - */ -class FileOptionBuilder( - @PublishedApi internal val serializer: SupabaseSerializer, - var userMetadata: JsonObject? = null, - var upsert: Boolean = false, - var contentType: ContentType? = null, -) { - - /** - * Sets the user metadata to upload with the file - * @param data The data to upload. Must be serializable by the [serializer] - */ - inline fun userMetadata(data: T) { - userMetadata = serializer.encodeToJsonElement(data).jsonObject - } - - /** - * Sets the user metadata to upload with the file - * @param builder The builder for the metadata - */ - inline fun userMetadata(builder: JsonObjectBuilder.() -> Unit) { - userMetadata = buildJsonObject(builder) - } - -} diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt deleted file mode 100644 index dc3c2d284..000000000 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt +++ /dev/null @@ -1,226 +0,0 @@ -package io.github.jan.supabase.storage - -import io.github.jan.supabase.exceptions.HttpRequestException -import io.github.jan.supabase.exceptions.RestException -import io.ktor.client.call.body -import io.ktor.client.plugins.HttpRequestTimeoutException -import io.ktor.client.plugins.onDownload -import io.ktor.client.plugins.onUpload -import io.ktor.http.HttpMethod -import io.ktor.utils.io.ByteReadChannel -import io.ktor.utils.io.ByteWriteChannel -import kotlinx.coroutines.flow.Flow -import kotlinx.coroutines.flow.callbackFlow - -/** - * Uploads a file in [bucketId] under [path] - * @param path The path to upload the file to - * @param upsert Whether to overwrite an existing file - * @param data The data to upload - * @return A flow that emits the upload progress and at last the key to the uploaded file - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.updateAsFlow(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = callbackFlow { - this@updateAsFlow as BucketApiImpl - val key = uploadOrUpdate(HttpMethod.Put, bucketId, path, data, upsert, options) { - onUpload { bytesSentTotal, contentLength -> - trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) - } - } - trySend(UploadStatus.Success(key)) - close() -} - -/** - * Uploads a file in [bucketId] under [path] - * @param path The path to upload the file to - * @param upsert Whether to overwrite an existing file - * @param data The data to upload - * @return A flow that emits the upload progress and at last the key to the uploaded file - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.uploadAsFlow(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = uploadAsFlow(path, UploadData( - ByteReadChannel(data), data.size.toLong()), upsert, options) - -/** - * Uploads a file in [bucketId] under [path] using a presigned url - * @param path The path to upload the file to - * @param token The presigned url token - * @param data The data to upload - * @param upsert Whether to overwrite an existing file - * @return A flow that emits the upload progress and at last the key to the uploaded file - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.uploadToSignedUrlAsFlow( - path: String, - token: String, - data: UploadData, - upsert: Boolean = false, - options: FileOptionBuilder.() -> Unit = {} -): Flow { - return callbackFlow { - this@uploadToSignedUrlAsFlow as BucketApiImpl - val key = uploadToSignedUrl(path, token, data, upsert, options) { - onUpload { bytesSentTotal, contentLength -> - trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) - } - } - trySend(UploadStatus.Success(key)) - close() - } -} - -/** - * Uploads a file in [bucketId] under [path] using a presigned url - * @param path The path to upload the file to - * @param token The presigned url token - * @param data The data to upload - * @param upsert Whether to overwrite an existing file - * @return A flow that emits the upload progress and at last the key to the uploaded file - */ -fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = uploadToSignedUrlAsFlow(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) - -/** - * Updates a file in [bucketId] under [path] - * @param path The path to update the file to - * @param data The new data - * @param upsert Whether to overwrite an existing file - * @return A flow that emits the upload progress and at last the key to the uploaded file - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.uploadAsFlow(path: String, data: UploadData, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow { - return callbackFlow { - this@uploadAsFlow as BucketApiImpl - val key = uploadOrUpdate(HttpMethod.Post, bucketId, path, data, upsert, options) { - onUpload { bytesSentTotal, contentLength -> - trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) - } - } - trySend(UploadStatus.Success(key)) - close() - } -} - -/** - * Updates a file in [bucketId] under [path] - * @param path The path to update the file to - * @param data The new data - * @param upsert Whether to overwrite an existing file - * @return A flow that emits the upload progress and at last the key to the uploaded file - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.updateAsFlow(path: String, data: ByteArray, upsert: Boolean = false, options: FileOptionBuilder.() -> Unit = {}): Flow = updateAsFlow(path, UploadData(ByteReadChannel(data), data.size.toLong()), upsert, options) - -/** - * Downloads a file from [bucketId] under [path] - * @param path The path to download - * @param transform The transformation to apply to the image - * @return A flow that emits the download progress and at last the data as a byte array - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.downloadAuthenticatedAsFlow( - path: String, - transform: ImageTransformation.() -> Unit = {} -): Flow { - return callbackFlow { - this@downloadAuthenticatedAsFlow as BucketApiImpl - val data = storage.api.rawRequest { - prepareDownloadRequest(path, false, transform) - onDownload { bytesSentTotal, contentLength -> - trySend(DownloadStatus.Progress(bytesSentTotal, contentLength)) - } - }.body() - trySend(DownloadStatus.Success) - trySend(DownloadStatus.ByteData(data)) - close() - } -} - -/** - * Downloads a file from [bucketId] under [path] - * @param path The path to download - * @param channel The channel to write the data to - * @param transform The transformation to apply to the image - * @return A flow that emits the download progress and at last the data as a byte array - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.downloadAuthenticatedAsFlow( - path: String, - channel: ByteWriteChannel, - transform: ImageTransformation.() -> Unit = {} -): Flow { - this as BucketApiImpl - return flowChannelDownloadRequest(path, channel, false, transform) -} - -/** - * Downloads a file from [bucketId] under [path] using the public url - * @param path The path to download - * @param transform The transformation to apply to the image - * @return A flow that emits the download progress and at last the data as a byte array - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.downloadPublicAsFlow(path: String, transform: ImageTransformation.() -> Unit = {}): Flow { - return callbackFlow { - this@downloadPublicAsFlow as BucketApiImpl - val data = storage.api.rawRequest { - prepareDownloadRequest(path, true, transform) - onDownload { bytesSentTotal, contentLength -> - trySend(DownloadStatus.Progress(bytesSentTotal, contentLength)) - } - }.body() - trySend(DownloadStatus.Success) - trySend(DownloadStatus.ByteData(data)) - close() - } -} - -/** - * Downloads a file from [bucketId] under [path] using the public url - * @param path The path to download - * @param channel The channel to write the data to - * @param transform The transformation to apply to the image - * @return A flow that emits the download progress and at last the data as a byte array - * @throws RestException or one of its subclasses if receiving an error response - * @throws HttpRequestTimeoutException if the request timed out - * @throws HttpRequestException on network related issues - */ -fun BucketApi.downloadPublicAsFlow( - path: String, - channel: ByteWriteChannel, - transform: ImageTransformation.() -> Unit = {} -): Flow { - this as BucketApiImpl - return flowChannelDownloadRequest(path, channel, true, transform) -} - -private fun BucketApiImpl.flowChannelDownloadRequest( - path: String, - channel: ByteWriteChannel, - public: Boolean, - transform: ImageTransformation.() -> Unit -): Flow = callbackFlow { - channelDownloadRequest(path, channel, public, transform) { - onDownload { bytesSentTotal, contentLength -> - trySend(DownloadStatus.Progress(bytesSentTotal, contentLength)) - } - } - trySend(DownloadStatus.Success) - close() -} \ No newline at end of file diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt new file mode 100644 index 000000000..b2f34a4d2 --- /dev/null +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt @@ -0,0 +1,229 @@ +package io.github.jan.supabase.storage + +import io.github.jan.supabase.exceptions.HttpRequestException +import io.github.jan.supabase.exceptions.RestException +import io.github.jan.supabase.network.HttpRequestOverride +import io.ktor.client.plugins.HttpRequestTimeoutException +import io.ktor.client.plugins.onDownload +import io.ktor.client.plugins.onUpload +import io.ktor.utils.io.ByteReadChannel +import io.ktor.utils.io.ByteWriteChannel +import kotlinx.coroutines.channels.ProducerScope +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.callbackFlow + +private fun downloadOverride(flowProducer: ProducerScope): HttpRequestOverride = { + onDownload { bytesSentTotal, contentLength -> + flowProducer.trySend(DownloadStatus.Progress(bytesSentTotal, contentLength)) + } +} + +private fun uploadOverride(flowProducer: ProducerScope): HttpRequestOverride = { + onUpload { bytesSentTotal, contentLength -> + flowProducer.trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) + } +} + +/** + * Uploads a file in [BucketApi.bucketId] under [path] + * @param path The path to upload the file to + * @param options Additional options for the upload + * @param data The data to upload + * @return A flow that emits the upload progress and at last the key to the uploaded file + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.updateAsFlow( + path: String, + data: UploadData, + options: UploadOptionBuilder.() -> Unit = {} +): Flow = uploadAsFlowRequest { + update(path, data) { + options() + httpOverride(it) + } +} + +/** + * Uploads a file in [BucketApi.bucketId] under [path] + * @param path The path to upload the file to + * @param options Additional options for the upload + * @param data The data to upload + * @return A flow that emits the upload progress and at last the key to the uploaded file + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.uploadAsFlow(path: String, data: ByteArray, options: UploadOptionBuilder.() -> Unit = {}): Flow = uploadAsFlow(path, UploadData( + ByteReadChannel(data), data.size.toLong()), options) + +/** + * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url + * @param path The path to upload the file to + * @param token The presigned url token + * @param data The data to upload + * @param options Additional options for the upload + * @return A flow that emits the upload progress and at last the key to the uploaded file + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.uploadToSignedUrlAsFlow( + path: String, + token: String, + data: UploadData, + options: UploadOptionBuilder.() -> Unit = {} +): Flow = uploadAsFlowRequest { + uploadToSignedUrl(path, token, data) { + options() + httpOverride(it) + } +} + +/** + * Uploads a file in [BucketApi.bucketId] under [path] using a presigned url + * @param path The path to upload the file to + * @param token The presigned url token + * @param data The data to upload + * @param options Additional options for the upload + * @return A flow that emits the upload progress and at last the key to the uploaded file + */ +fun BucketApi.uploadToSignedUrlAsFlow(path: String, token: String, data: ByteArray, options: UploadOptionBuilder.() -> Unit = {}): Flow = uploadToSignedUrlAsFlow(path, token, UploadData(ByteReadChannel(data), data.size.toLong()), options) + +/** + * Updates a file in [BucketApi.bucketId] under [path] + * @param path The path to update the file to + * @param data The new data + * @param options Additional options for the upload + * @return A flow that emits the upload progress and at last the key to the uploaded file + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.uploadAsFlow( + path: String, + data: UploadData, + options: UploadOptionBuilder.() -> Unit = {} +): Flow = uploadAsFlowRequest { + upload(path, data) { + options() + httpOverride(it) + } +} + +/** + * Updates a file in [BucketApi.bucketId] under [path] + * @param path The path to update the file to + * @param data The new data + * @param options Additional options for the upload + * @return A flow that emits the upload progress and at last the key to the uploaded file + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.updateAsFlow(path: String, data: ByteArray, options: UploadOptionBuilder.() -> Unit = {}): Flow = updateAsFlow(path, UploadData(ByteReadChannel(data), data.size.toLong()), options) + +private fun BucketApi.uploadAsFlowRequest( + producer: suspend (HttpRequestOverride) -> FileUploadResponse +) = callbackFlow { + val key = producer(uploadOverride(this@callbackFlow)) + trySend(UploadStatus.Success(key)) + close() +} + +/** + * Downloads a file from [BucketApi.bucketId] under [path] + * @param path The path to download + * @param options Additional options for the download + * @return A flow that emits the download progress and at last the data as a byte array + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.downloadAuthenticatedAsFlow( + path: String, + options: DownloadOptionBuilder.() -> Unit = {} +): Flow = downloadAsFlowRequest { + downloadAuthenticated(path) { + options() + httpOverride(it) + } +} + +/** + * Downloads a file from [BucketApi.bucketId] under [path] using the public url + * @param path The path to download + * @param options Additional options for the download + * @return A flow that emits the download progress and at last the data as a byte array + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.downloadPublicAsFlow( + path: String, + options: DownloadOptionBuilder.() -> Unit = {} +): Flow = + downloadAsFlowRequest { + downloadPublic(path) { + options() + httpOverride(it) + } + } + + +/** + * Downloads a file from [BucketApi.bucketId] under [path] + * @param path The path to download + * @param channel The channel to write the data to + * @param options Additional options for the download + * @return A flow that emits the download progress and at last the data as a byte array + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.downloadAuthenticatedAsFlow( + path: String, + channel: ByteWriteChannel, + options: DownloadOptionBuilder.() -> Unit = {} +): Flow = downloadAsFlowRequest { + downloadAuthenticated(path, channel) { + options() + httpOverride(it) + } + null +} + +/** + * Downloads a file from [BucketApi.bucketId] under [path] using the public url + * @param path The path to download + * @param channel The channel to write the data to + * @param options Additional options for the download + * @return A flow that emits the download progress and at last the data as a byte array + * @throws RestException or one of its subclasses if receiving an error response + * @throws HttpRequestTimeoutException if the request timed out + * @throws HttpRequestException on network related issues + */ +fun BucketApi.downloadPublicAsFlow( + path: String, + channel: ByteWriteChannel, + options: DownloadOptionBuilder.() -> Unit = {} +): Flow = downloadAsFlowRequest { + downloadPublic(path, channel) { + options() + httpOverride(it) + } + null +} + +private fun BucketApi.downloadAsFlowRequest( + producer: suspend (HttpRequestOverride) -> ByteArray? +) = callbackFlow { + //If null, the data gets streamed to a channel, so we don't need to emit it later + val data = producer(downloadOverride(this@callbackFlow)) + trySend(DownloadStatus.Success) + if (data != null) { + trySend(DownloadStatus.ByteData(data)) + } + close() +} diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/UploadOptionBuilder.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/UploadOptionBuilder.kt new file mode 100644 index 000000000..61bfa326e --- /dev/null +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/UploadOptionBuilder.kt @@ -0,0 +1,27 @@ +package io.github.jan.supabase.storage + +import io.github.jan.supabase.SupabaseSerializer +import io.github.jan.supabase.network.HttpRequestOverride +import io.ktor.http.ContentType + +/** + * Builder for uploading files with additional options + * @param serializer The serializer to use for encoding the metadata + * @param upsert Whether to update the file if it already exists + * @param contentType The content type of the file. If null, the content type will be inferred from the file extension + */ +class UploadOptionBuilder( + @PublishedApi internal val serializer: SupabaseSerializer, + var upsert: Boolean = false, + var contentType: ContentType? = null, + internal val httpRequestOverrides: MutableList = mutableListOf() +) { + + /** + * Overrides the HTTP request + */ + fun httpOverride(override: HttpRequestOverride) { + httpRequestOverrides.add(override) + } + +} diff --git a/Storage/src/commonTest/kotlin/BucketApiFlowTest.kt b/Storage/src/commonTest/kotlin/BucketApiFlowTest.kt index c0eb57575..939a73065 100644 --- a/Storage/src/commonTest/kotlin/BucketApiFlowTest.kt +++ b/Storage/src/commonTest/kotlin/BucketApiFlowTest.kt @@ -73,7 +73,9 @@ class BucketApiFlowTest { @Test fun testUpsertAsFlowMethodWithByteArray() { testUploadAsFlow(upsert = true) { client, expectedPath, data -> - val flow = client.storage[bucketId].uploadAsFlow(expectedPath, data, upsert = true) + val flow = client.storage[bucketId].uploadAsFlow(expectedPath, data) { + upsert = true + } testUploadFlowWithByteArray(flow, expectedPath, data) } } @@ -95,7 +97,9 @@ class BucketApiFlowTest { val channel = ByteReadChannel(bytes) val expectedSize = bytes.size.toLong() val data = UploadData(channel, expectedSize) - val flow = client.storage[bucketId].uploadAsFlow(expectedPath, data, upsert = true) + val flow = client.storage[bucketId].uploadAsFlow(expectedPath, data) { + upsert = true + } testUploadFlowWithByteArray(flow, expectedPath, bytes) } } @@ -131,7 +135,9 @@ class BucketApiFlowTest { @Test fun testUpdateUpsertAsFlowMethodWithByteArray() { testUpdateAsFlow(upsert = true) { client, expectedPath, data -> - val flow = client.storage[bucketId].updateAsFlow(expectedPath, data, upsert = true) + val flow = client.storage[bucketId].updateAsFlow(expectedPath, data) { + upsert = true + } testUploadFlowWithByteArray(flow, expectedPath, data) } } @@ -153,7 +159,9 @@ class BucketApiFlowTest { val channel = ByteReadChannel(bytes) val expectedSize = bytes.size.toLong() val data = UploadData(channel, expectedSize) - val flow = client.storage[bucketId].updateAsFlow(expectedPath, data, upsert = true) + val flow = client.storage[bucketId].updateAsFlow(expectedPath, data) { + upsert = true + } testUploadFlowWithByteArray(flow, expectedPath, bytes) } } @@ -206,7 +214,9 @@ class BucketApiFlowTest { val channel = ByteReadChannel(bytes) val expectedSize = bytes.size.toLong() val data = UploadData(channel, expectedSize) - val flow = client.storage[bucketId].uploadToSignedUrlAsFlow(expectedPath, expectedToken, data, true) + val flow = client.storage[bucketId].uploadToSignedUrlAsFlow(expectedPath, expectedToken, data) { + upsert = true + } testUploadFlowWithByteArray(flow, expectedPath, bytes) } } @@ -214,7 +224,9 @@ class BucketApiFlowTest { @Test fun testUpsertToSignedUrlAsFlowMethodByteArray() { testUploadToSignedUrl(upsert = true) { client, expectedPath, bytes -> - val flow = client.storage[bucketId].uploadToSignedUrlAsFlow(expectedPath, expectedToken, bytes, true) + val flow = client.storage[bucketId].uploadToSignedUrlAsFlow(expectedPath, expectedToken, bytes) { + upsert = true + } testUploadFlowWithByteArray(flow, expectedPath, bytes) } } diff --git a/Storage/src/commonTest/kotlin/BucketApiTest.kt b/Storage/src/commonTest/kotlin/BucketApiTest.kt index ac09886ad..ece065241 100644 --- a/Storage/src/commonTest/kotlin/BucketApiTest.kt +++ b/Storage/src/commonTest/kotlin/BucketApiTest.kt @@ -1,7 +1,6 @@ import io.github.jan.supabase.SupabaseClient import io.github.jan.supabase.SupabaseClientBuilder import io.github.jan.supabase.storage.BucketApi -import io.github.jan.supabase.storage.FileObjectV2 import io.github.jan.supabase.storage.FileUploadResponse import io.github.jan.supabase.storage.ImageTransformation import io.github.jan.supabase.storage.Storage @@ -14,20 +13,14 @@ import io.github.jan.supabase.testing.pathAfterVersion import io.github.jan.supabase.testing.toJsonElement import io.ktor.client.engine.mock.MockRequestHandleScope import io.ktor.client.engine.mock.respond -import io.ktor.client.engine.mock.respondError -import io.ktor.client.engine.mock.respondOk import io.ktor.client.engine.mock.toByteArray import io.ktor.client.request.HttpRequestData import io.ktor.http.ContentType import io.ktor.http.HttpHeaders import io.ktor.http.HttpMethod -import io.ktor.http.HttpStatusCode import io.ktor.http.headersOf import kotlinx.coroutines.test.runTest import kotlinx.datetime.Clock -import kotlinx.serialization.encodeToString -import kotlinx.serialization.json.Json -import kotlinx.serialization.json.JsonObject import kotlinx.serialization.json.buildJsonObject import kotlinx.serialization.json.int import kotlinx.serialization.json.jsonArray @@ -35,14 +28,11 @@ import kotlinx.serialization.json.jsonObject import kotlinx.serialization.json.jsonPrimitive import kotlinx.serialization.json.long import kotlinx.serialization.json.put -import kotlin.io.encoding.Base64 import kotlin.io.encoding.ExperimentalEncodingApi import kotlin.test.Test import kotlin.test.assertContentEquals import kotlin.test.assertEquals -import kotlin.test.assertFalse import kotlin.test.assertNotNull -import kotlin.test.assertTrue import kotlin.time.Duration.Companion.seconds class BucketApiTest { @@ -61,10 +51,8 @@ class BucketApiTest { testUploadMethod( method = HttpMethod.Post, urlPath = "/object/$bucketId/data.png", - request = { client, expectedPath, data, meta -> - client.storage[bucketId].upload(expectedPath, data) { - userMetadata = meta - } + request = { client, expectedPath, data -> + client.storage[bucketId].upload(expectedPath, data) }, extra = { assertEquals( @@ -88,9 +76,8 @@ class BucketApiTest { "Upsert header should be true" ) }, - request = { client, expectedPath, data, meta -> + request = { client, expectedPath, data -> client.storage[bucketId].upload(expectedPath, data) { - userMetadata = meta upsert = true } } @@ -115,9 +102,8 @@ class BucketApiTest { "Token should be $expectedToken" ) }, - request = { client, expectedPath, data, meta -> + request = { client, expectedPath, data -> client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data) { - userMetadata = meta upsert = false } } @@ -142,9 +128,8 @@ class BucketApiTest { "Token should be $expectedToken" ) }, - request = { client, expectedPath, data, meta -> + request = { client, expectedPath, data -> client.storage[bucketId].uploadToSignedUrl(path = expectedPath, token = expectedToken, data = data) { - userMetadata = meta upsert = true } } @@ -156,10 +141,8 @@ class BucketApiTest { testUploadMethod( method = HttpMethod.Put, urlPath = "/object/$bucketId/data.png", - request = { client, expectedPath, data, meta -> - client.storage[bucketId].update(expectedPath, data) { - userMetadata = meta - } + request = { client, expectedPath, data -> + client.storage[bucketId].update(expectedPath, data) }, extra = { assertEquals( @@ -183,9 +166,8 @@ class BucketApiTest { "Upsert header should be true" ) }, - request = { client, expectedPath, data, meta -> + request = { client, expectedPath, data -> client.storage[bucketId].update(expectedPath, data) { - userMetadata = meta upsert = true } } @@ -458,69 +440,6 @@ class BucketApiTest { } } - @Test - fun testInfo() { - runTest { - val expectedPath = "data.png" - val file = FileObjectV2( - "data.png", - "id", - "version", - createdAt = Clock.System.now(), - metadata = null, - size = 0, - rawContentType = "image/png", - etag = null, - lastModified = null, - cacheControl = null - ) - val client = createMockedSupabaseClient(configuration = configureClient) { - assertMethodIs(HttpMethod.Get, it.method) - assertPathIs("/object/info/$bucketId/$expectedPath", it.url.pathAfterVersion()) - respond( - content = Json.encodeToString(file), - headers = headersOf( - HttpHeaders.ContentType, - ContentType.Application.Json.toString() - ) - ) - } - val data = client.storage[bucketId].info(expectedPath) - assertEquals(file.copy(serializer = client.storage.serializer), data, "Data should be $file") - } - } - - @Test - fun testExistsWithExistingFile() { - runTest { - val expectedPath = "data.png" - val client = createMockedSupabaseClient(configuration = configureClient) { - assertMethodIs(HttpMethod.Head, it.method) - assertPathIs("/object/$bucketId/$expectedPath", it.url.pathAfterVersion()) - respondOk() - } - val exists = client.storage[bucketId].exists(expectedPath) - assertTrue { exists } - } - } - - @Test - fun testExistsWithNonExistingFile() { - val statusCodes = listOf(404, 400) - for(code in statusCodes) { - runTest { - val expectedPath = "data.png" - val client = createMockedSupabaseClient(configuration = configureClient) { - assertMethodIs(HttpMethod.Head, it.method) - assertPathIs("/object/$bucketId/$expectedPath", it.url.pathAfterVersion()) - respondError(HttpStatusCode(code, "Not Found")) - } - val exists = client.storage[bucketId].exists(expectedPath) - assertFalse { exists } - } - } - } - private fun testDownloadWithTransform( authenticated: Boolean ) { @@ -550,7 +469,7 @@ class BucketApiTest { quality = expectedQuality resize = expectedResize } - val data = if(authenticated) client.storage[bucketId].downloadAuthenticated(expectedPath, transform) else client.storage[bucketId].downloadPublic(expectedPath, transform) + val data = if(authenticated) client.storage[bucketId].downloadAuthenticated(expectedPath) { transform(transform) } else client.storage[bucketId].downloadPublic(expectedPath) { transform(transform) } assertContentEquals(expectedData, data, "Data should be [1, 2, 3]") } } @@ -561,7 +480,7 @@ class BucketApiTest { urlPath: String, expectedPath: String = "data.png", extra: suspend MockRequestHandleScope.(HttpRequestData) -> Unit, - request: suspend (client: SupabaseClient, expectedPath: String, data: ByteArray, metadata: JsonObject) -> FileUploadResponse + request: suspend (client: SupabaseClient, expectedPath: String, data: ByteArray) -> FileUploadResponse ) { runTest { val expectedData = byteArrayOf(1, 2, 3) @@ -571,8 +490,6 @@ class BucketApiTest { val client = createMockedSupabaseClient(configuration = configureClient) { val data = it.body.toByteArray() assertMethodIs(method, it.method) - val metadata = Json.decodeFromString(Base64.decode(it.headers["x-metadata"] ?: error("Metadata should not be null")).decodeToString()) - assertEquals(expectedMetadata, metadata, "Metadata should be $expectedMetadata") assertPathIs(urlPath, it.url.pathAfterVersion()) assertContentEquals(expectedData, data, "Data should be [1, 2, 3]") assertEquals(ContentType.Image.PNG, it.body.contentType, "Content type should be image/png") @@ -590,7 +507,7 @@ class BucketApiTest { ) ) } - val response = request(client, expectedPath, expectedData, expectedMetadata) + val response = request(client, expectedPath, expectedData) assertEquals("someBucket/$expectedPath", response.key, "Key should be $expectedPath") assertEquals("someId", response.id, "Id should be someId") assertEquals(expectedPath, response.path, "Path should be $expectedPath") diff --git a/Supabase/src/commonMain/kotlin/io/github/jan/supabase/network/KtorSupabaseHttpClient.kt b/Supabase/src/commonMain/kotlin/io/github/jan/supabase/network/KtorSupabaseHttpClient.kt index a46079bf3..b86c9bf2f 100644 --- a/Supabase/src/commonMain/kotlin/io/github/jan/supabase/network/KtorSupabaseHttpClient.kt +++ b/Supabase/src/commonMain/kotlin/io/github/jan/supabase/network/KtorSupabaseHttpClient.kt @@ -29,6 +29,11 @@ import kotlin.time.Duration.Companion.milliseconds private const val HTTPS_PORT = 443 +/** + * A function that can be used to override the default request configuration + */ +typealias HttpRequestOverride = HttpRequestBuilder.() -> Unit + /** * A [SupabaseHttpClient] that uses ktor to send requests */ diff --git a/settings.gradle.kts b/settings.gradle.kts index 1af111188..f3cc0eec1 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -21,6 +21,7 @@ include("bom") // Test module include("test-common") +include("test") // Serializers include(":serializers:Moshi") From 01f2e7e77a05a38cbac50acb983cc5f469203a90 Mon Sep 17 00:00:00 2001 From: Jan Tennert Date: Mon, 16 Sep 2024 19:52:41 +0200 Subject: [PATCH 11/13] Revert accidental change --- settings.gradle.kts | 1 - 1 file changed, 1 deletion(-) diff --git a/settings.gradle.kts b/settings.gradle.kts index f3cc0eec1..1af111188 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -21,7 +21,6 @@ include("bom") // Test module include("test-common") -include("test") // Serializers include(":serializers:Moshi") From 60992cc755f494c5f72026988606a3079902d50a Mon Sep 17 00:00:00 2001 From: Jan Tennert Date: Mon, 16 Sep 2024 19:55:07 +0200 Subject: [PATCH 12/13] Remove file object V2 --- .../github/jan/supabase/storage/FileObject.kt | 63 +------------------ .../io/github/jan/supabase/storage/FlowExt.kt | 0 2 files changed, 1 insertion(+), 62 deletions(-) delete mode 100644 Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt index 9ae862e59..8caf02a8a 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FileObject.kt @@ -1,13 +1,8 @@ package io.github.jan.supabase.storage -import io.github.jan.supabase.SupabaseSerializer -import io.github.jan.supabase.decode -import io.github.jan.supabase.serializer.KotlinXSerializer -import io.ktor.http.ContentType import kotlinx.datetime.Instant import kotlinx.serialization.SerialName import kotlinx.serialization.Serializable -import kotlinx.serialization.Transient import kotlinx.serialization.json.JsonObject /** @@ -30,60 +25,4 @@ data class FileObject( @SerialName("last_accessed_at") val lastAccessedAt: Instant?, val metadata: JsonObject? -) - -/** - * Represents a file or a folder in a bucket. If the item is a folder, everything except [name] is null. - * @param name The name of the item - * @param id The id of the item - * @param version The version of the item - * @param bucketId The bucket id of the item - * @param updatedAt The last update date of the item - * @param createdAt The creation date of the item - * @param lastAccessedAt The last access date of the item - * @param metadata The metadata of the item - * @param size The size of the item - * @param rawContentType The content type of the item - * @param etag The etag of the item - * @param lastModified The last modified date of the item - * @param cacheControl The cache control of the item - * @param serializer The serializer to use for decoding the metadata - */ -@Serializable -data class FileObjectV2( - val name: String, - val id: String?, - val version: String, - @SerialName("bucket_id") - val bucketId: String? = null, - @SerialName("updated_at") - val updatedAt: Instant? = null, - @SerialName("created_at") - val createdAt: Instant?, - @SerialName("last_accessed_at") - val lastAccessedAt: Instant? = null, - val metadata: JsonObject?, - val size: Long, - @SerialName("content_type") - val rawContentType: String, - val etag: String?, - @SerialName("last_modified") - val lastModified: Instant?, - @SerialName("cache_control") - val cacheControl: String?, - @Transient @PublishedApi internal val serializer: SupabaseSerializer = KotlinXSerializer() -) { - - /** - * The content type of the file - */ - val contentType by lazy { - ContentType.parse(rawContentType) - } - - /** - * Decodes the metadata using the [serializer] - */ - inline fun decodeMetadata(): T? = metadata?.let { serializer.decode(it.toString()) } - -} \ No newline at end of file +) \ No newline at end of file diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExt.kt deleted file mode 100644 index e69de29bb..000000000 From 1e638e51af71b766b1fec61d9c5854668ac34913 Mon Sep 17 00:00:00 2001 From: Jan Tennert Date: Mon, 16 Sep 2024 20:14:37 +0200 Subject: [PATCH 13/13] Fix type mismatch --- .../kotlin/io/github/jan/supabase/storage/FlowExtension.kt | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt index b2f34a4d2..146b0bbb1 100644 --- a/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt +++ b/Storage/src/commonMain/kotlin/io/github/jan/supabase/storage/FlowExtension.kt @@ -14,13 +14,13 @@ import kotlinx.coroutines.flow.callbackFlow private fun downloadOverride(flowProducer: ProducerScope): HttpRequestOverride = { onDownload { bytesSentTotal, contentLength -> - flowProducer.trySend(DownloadStatus.Progress(bytesSentTotal, contentLength)) + flowProducer.trySend(DownloadStatus.Progress(bytesSentTotal, contentLength ?: 0)) } } private fun uploadOverride(flowProducer: ProducerScope): HttpRequestOverride = { onUpload { bytesSentTotal, contentLength -> - flowProducer.trySend(UploadStatus.Progress(bytesSentTotal, contentLength)) + flowProducer.trySend(UploadStatus.Progress(bytesSentTotal, contentLength ?: 0)) } }