Skip to content

supabase-community/storage-swift

Repository files navigation

storage-swift

Swift Client library to interact with Supabase Storage.

Prelude: RLS

A lot of bug reports get filed towards all the Supabase Storage libraries because RLS is easy to get wrong for the un-initiated.

Before you can use any client service, you should follow the RLS guides on Supabase's website. You will need to create policies on each bucket individually or use global ones on the storage/object and storage/bucket sections.

For image/video uploads from users, there is a key example on how to setup basic RLS for all CRUD operations here Pay particular attention to the difference between USING and WITH CHECK keyword for POSTGRES SQL.

Once you have RLS setup for ALL CRUD operations you should be able to use the bucket with the clients. There are some issues with uploading or downloading with buckets that change their permission levels from public to private and vice versa, so use caution when playing around with those settings if you're experimenting. If you experience a bunch of 404's with uploads and downloads, try deleting the bucket and re-initializing it.

Instantiating the client

If you're using the full supabase-swift experience, you can create a reference to the storage client by accessing the following:

import Supabase
import SupabaseStorage

lazy var supabaseClient: SupabaseClient = {
    return SupabaseClient(supabaseURL: supabaseUrl, supabaseKey: apiKey)
}()

lazy var supabaseStorageClient: SupabaseStorageClient = {
    return supabaseClient.storage
}()

If you're experiencing errors using this instance, or would just like to construct your own, you may do so via:

func storageClient(bucketName: String = "photos") async -> StorageFileApi? {
    guard let jwt = try? await supabaseClient.auth.session.accessToken else { return nil}
    return SupabaseStorageClient(
        url: "\(supabaseUrl)/storage/v1",
        headers: [
            "Authorization": "Bearer \(jwt)",
            "apikey": apiKey,
        ]
    ).from(id: bucketName)
}

Architecturally, a pattern you can follow is holding these values and functions in a provider class using a simple Singleton pattern or inject it with something like Resolver.

class SupabaseProvider {
    
    private let apiDictionaryKey = "supabase-key"
    private let supabaseUrlKey = "supabase-url"
    private let discordUrlKey = "discord-callback-url"
    
    private init() {}
    
    static let shared = SupabaseProvider()
    
    lazy var supabaseClient: SupabaseClient = {
        return SupabaseClient(supabaseURL: supabaseUrl, supabaseKey: apiKey)
    }()
    
    func loggedInUserId() async -> String? {
        return try? await SupabaseProvider.shared.supabaseClient.auth.session.user.id.uuidString
    }
    
    private var keysPlist: NSDictionary {
        if
            let path = Bundle.main.path(forResource: "Keys", ofType: "plist"),
            let dictionary = NSDictionary(contentsOfFile: path) {
            return dictionary
        }
        fatalError("You must have a Keys.plist file in your application codebase.")
    }
    
    private var apiKey: String {
        guard let apiKey = keysPlist[apiDictionaryKey] as? String else {
            fatalError("Your Keys.plist must have a key of: \(apiDictionaryKey) and a corresponding value of type String.")
        }
        return apiKey
    }
    
    var supabaseUrl: URL {
        guard let url = keysPlist[supabaseUrlKey] as? String else {
            fatalError("Your Keys.plist must have a key of: \(supabaseUrlKey) and a corresponding value of type String.")
        }
        return URL(string: url)!
    }
    
    var discordCallbackUrl: String {
        guard let url = keysPlist[discordUrlKey] as? String else {
            fatalError("Your Keys.plist must have a key of: \(discordUrlKey) and a corresponding value of type String.")
        }
        return url
    }
    
    // Storage
    
    func storageClient(bucketName: String = "photos") async -> StorageFileApi? {
        guard let jwt = try? await supabaseClient.auth.session.accessToken else { return nil}
        return SupabaseStorageClient(
            url: "\(supabaseUrl)/storage/v1",
            headers: [
                "Authorization": "Bearer \(jwt)",
                "apikey": apiKey,
            ]
        ).from(id: bucketName)
    }
}

Uploading and Downloading

With your client of choice, you can download via the following given some RLS and naming conventions illustrated above.

Example of converting a .png image download to a UIImage for a UIImageView:

if let data = try? await SupabaseProvider.shared.storageClient()?.download(
    path: profilePhotoUrl
) {
    imageView.image = UIImage(data: data)
}

Example of uploading an image via a UIImageView's current UIImage and saving it to a user's bucket folder:

func loggedInUserId() async -> String? {
    return try? await SupabaseProvider.shared.client.auth.session.user.id.uuidString
}

guard let image = imageView.image?.pngData() else { return }

// Note that Swift has UUID's all capitalized, but Supabase will always lowercase
// them.

guard let userId = loggedInUserId().lowercased() else { return }

let fileId = UUID().uuidString
let filename = "\(fileId).png"
let storageClient = await SupabaseProvider.shared.storageClient()
guard let uploadResponseData = try? await storageClient?.upload(
        path: "\(userId)/\(filename)", 
        file: File(
            name: filename, 
            data: image, 
            fileName: filename, 
            contentType: "image/png"), 
            fileOptions: FileOptions(cacheControl: "3600")
        )
    ) else { return }

URL Creation

Signed URL creation is fairly straightforward and is the recommended way to grab URLs from storage devices. For larger files, you should incorporate CoreData to keep them on device, probably with the help of an LRU Cache.

let storageClient = await SupabaseProvider.shared.storageClient(bucketName: "bucket_name")
// URL will expire in 1 hour (3600 seconds)
guard let url = try? await storageClient?.createSignedURL(path: imageUrl, expiresIn: 3600) else {
    return
}
if let image = getThumbnailImage(forUrl: url) {
    DispatchQueue.main.async {
        self.userPostImageView.image = image
    }
}

Sponsors

We are building the features of Firebase using enterprise-grade, open source products. We support existing communities wherever possible, and if the products don’t exist we build them and open source them ourselves. Thanks to these sponsors who are making the OSS ecosystem better for everyone.

New Sponsor

About

Swift client library to interact with Supabase Storage

Resources

License

Code of conduct

Stars

Watchers

Forks

Packages

No packages published