A super lightweight and testable library for Twitter written in Swift using Combine and async/await
- Support Twitter v1 API (OAuth1.0)
- Login with Twitter
- Multiple accounts management
- Support making an authorized
URLRequest
- Support chunked media upload
- Support Twitter v2 API (OAuth2.0)
Use swift package manager.
Use ClientCredential
, TwitterAuthorizationFlow
and TwitterAccountStore
.
- Prepare for login.
import Kotori
// Your credentials.
let clientCredential: ClientCredential = .init(consumerKey: "Your consumer key from https://developer.twitter.com",
consumerSecret: "Your consumer secret from https://developer.twitter.com",
callbackURL: URL(string: "Your app's url scheme. see https://developer.twitter.com")!)
// Login flow manager.
let twitterLoginFlow = TwitterAuthorizationFlow(clientCredential: clientCredential, urlSession: .shared)
// Account manager.
let accountStore = TwitterAccountStore(keychainAccessGroup: "Your keychain access group.")
- Authorization handshake start.
twitterLoginFlow.authorize()
.receive(on: RunLoop.main)
.sink(
receiveCompletion: { _ in },
receiveValue: { credentials in
accountStore.add(credential) // Save to the keychain.
self.credentials = accountStore.allCredentials() // Get all logged in accounts.
})
.store(in: &cancellables)
- Open the Twitter login page in your app.
Kotori request you for open the twitter login page to authorize by the user. So, add codes which open the twitter login page in your app.
NotificationCenter.default.publisher(for: TwitterAuthorizationFlow.resourceOwnerAuthorizationOpenURL)
.sink(
receiveCompletion: { _ in },
receiveValue: { output in
guard let url = output.object as? URL else { fatalError() }
// Present WebView or SFSafariViewController in SwiftUI or UIKit for the user authentication.
})
.store(in: &cancellables)
- Open your app from the Twitter login page. (URL Scheme)
After finished login in the Twitter login page, The Twitter login page redirects to the your app using url scheme including credential parameters. So, add codes which handle the url scheme and pass it to the Kotori.
// SwiftUI
// Use View Modifier.
.onOpenURL(perform: { url in
twitterLoginFlow.handleCallbackFromTwitter(url: url)
})
// UIKit
// Use AppDelegate method.
func application(_ app: UIApplication,
open url: URL,
options: [UIApplication.OpenURLOptionsKey : Any] = [:]) -> Bool {
twitterLoginFlow.handleCallbackFromTwitter(url: url)
}
- Handle login completion callback from the Kotori.
After call this, twitterLoginFlow.authorize().sink()
will be called.
Use TwitterAccountStore
.
let accountStore = TwitterAccountStore(keychainAccessGroup: "your keychain access group")
accountStore.allCredentials()
Use TwitterAPIRequest
.
Kotori does not provide the network layer, response parser. Just only URLRequest.
let twitterRequest = TwitterAPIRequest(resourceURL: Endpoint.statusUpdate,
httpMethod: .POST,
parameters: tweet.asParameters(),
credential: credentials,
clientCredential: clientCredential)
let urlRequest = twitterRequest.makeURLRequest()
Use TwitterMediaUploader
.
Kotori supports chunked upload. Chuneked upload is a very fast upload method but has many steps and difficult. So, Kotori may help you.
Single image upload sample.
let uploader: TwitterMediaUploader = .init(credential: credential,data: media.data, mimeType: media.mimeType, clientCredential: clientCredential, index: index)
try await uploader.upload()
Multiple image upload sample.
try await withThrowingTaskGroup(of: MediaUploadOutput.self, body: { group -> [MediaUploadOutput] in
var mediaUploadResults: [MediaUploadOutput] = []
for (index, media) in tweet.medias.enumerated() {
group.addTask(priority: .userInitiated) {
let uploader: TwitterMediaUploader = .init(credential: credential,
data: media.data,
mimeType: media.mimeType,
clientCredential: clientCredential,
index: index)
return try await uploader.upload()
}
for try await output in group {
mediaUploadResults.append(output)
}
return mediaUploadResults
}
})
PRs are welcome. Format is free!
Kotori is small bird
in Japanese.
Kotori is small and lightweight library for the Twitter.