* Replace Rust's Authentication service by building a Client. * Drop the Proxy from AuthenticationServiceProxy[Protocol]. * Make sure to call abortOIDCLogin. * Centralise common ClientBuilder code. * Update the SDK.
166 lines
7.0 KiB
Swift
166 lines
7.0 KiB
Swift
//
|
|
// Copyright 2022 New Vector Ltd
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
import Foundation
|
|
import Kingfisher
|
|
import MatrixRustSDK
|
|
|
|
class UserSessionStore: UserSessionStoreProtocol {
|
|
private let keychainController: KeychainControllerProtocol
|
|
private let matrixSDKStateKey = "matrix-sdk-state"
|
|
|
|
/// Whether or not there are sessions in the store.
|
|
var hasSessions: Bool { !keychainController.restorationTokens().isEmpty }
|
|
/// All the user IDs managed by the store.
|
|
var userIDs: [String] { keychainController.restorationTokens().map(\.userID) }
|
|
|
|
var clientSessionDelegate: ClientSessionDelegate { keychainController }
|
|
|
|
init(keychainController: KeychainControllerProtocol) {
|
|
self.keychainController = keychainController
|
|
}
|
|
|
|
/// Deletes all data stored in the shared container and keychain
|
|
func reset() {
|
|
MXLog.warning("Resetting the UserSessionStore. All accounts will be affected.")
|
|
try? FileManager.default.removeItem(at: .sessionsBaseDirectory)
|
|
keychainController.removeAllRestorationTokens()
|
|
}
|
|
|
|
func restoreUserSession() async -> Result<UserSessionProtocol, UserSessionStoreError> {
|
|
let availableCredentials = keychainController.restorationTokens()
|
|
|
|
guard let credentials = availableCredentials.first else {
|
|
return .failure(.missingCredentials)
|
|
}
|
|
|
|
switch await restorePreviousLogin(credentials) {
|
|
case .success(let clientProxy):
|
|
return .success(buildUserSessionWithClient(clientProxy))
|
|
case .failure(let error):
|
|
MXLog.error("Failed restoring login with error: \(error)")
|
|
|
|
// On any restoration failure reset the token and restart
|
|
keychainController.removeRestorationTokenForUsername(credentials.userID)
|
|
deleteSessionDirectory(for: credentials)
|
|
|
|
return .failure(error)
|
|
}
|
|
}
|
|
|
|
func userSession(for client: Client, sessionDirectory: URL, passphrase: String?) async -> Result<UserSessionProtocol, UserSessionStoreError> {
|
|
do {
|
|
let session = try client.session()
|
|
let userID = try client.userId()
|
|
let clientProxy = await setupProxyForClient(client)
|
|
|
|
keychainController.setRestorationToken(RestorationToken(session: session,
|
|
sessionDirectory: sessionDirectory,
|
|
passphrase: passphrase,
|
|
pusherNotificationClientIdentifier: clientProxy.pusherNotificationClientIdentifier),
|
|
forUsername: userID)
|
|
|
|
return .success(buildUserSessionWithClient(clientProxy))
|
|
} catch {
|
|
MXLog.error("Failed creating user session with error: \(error)")
|
|
return .failure(.failedSettingUpSession)
|
|
}
|
|
}
|
|
|
|
func logout(userSession: UserSessionProtocol) {
|
|
let userID = userSession.clientProxy.userID
|
|
let credentials = keychainController.restorationTokens().first { $0.userID == userID }
|
|
keychainController.removeRestorationTokenForUsername(userID)
|
|
|
|
if let credentials {
|
|
deleteSessionDirectory(for: credentials)
|
|
}
|
|
}
|
|
|
|
func clearCache(for userID: String) {
|
|
guard let credentials = keychainController.restorationTokens().first(where: { $0.userID == userID }) else {
|
|
MXLog.error("Failed to clearing caches: Credentials missing")
|
|
return
|
|
}
|
|
deleteCaches(for: credentials)
|
|
}
|
|
|
|
// MARK: - Private
|
|
|
|
private func buildUserSessionWithClient(_ clientProxy: ClientProxyProtocol) -> UserSessionProtocol {
|
|
let mediaProvider = MediaProvider(mediaLoader: clientProxy,
|
|
imageCache: .onlyInMemory)
|
|
|
|
let voiceMessageMediaManager = VoiceMessageMediaManager(mediaProvider: mediaProvider)
|
|
|
|
return UserSession(clientProxy: clientProxy,
|
|
mediaProvider: mediaProvider,
|
|
voiceMessageMediaManager: voiceMessageMediaManager)
|
|
}
|
|
|
|
private func restorePreviousLogin(_ credentials: KeychainCredentials) async -> Result<ClientProxyProtocol, UserSessionStoreError> {
|
|
if credentials.restorationToken.passphrase != nil {
|
|
MXLog.info("Restoring client with encrypted store.")
|
|
}
|
|
|
|
let homeserverURL = credentials.restorationToken.session.homeserverUrl
|
|
|
|
let builder = ClientBuilder
|
|
.baseBuilder(httpProxy: URL(string: homeserverURL)?.globalProxy,
|
|
sessionDelegate: keychainController)
|
|
.sessionPath(path: credentials.restorationToken.sessionDirectory.path(percentEncoded: false))
|
|
.username(username: credentials.userID)
|
|
.homeserverUrl(url: homeserverURL)
|
|
.passphrase(passphrase: credentials.restorationToken.passphrase)
|
|
|
|
do {
|
|
let client = try await builder.build()
|
|
|
|
try await client.restoreSession(session: credentials.restorationToken.session)
|
|
|
|
return await .success(setupProxyForClient(client))
|
|
} catch {
|
|
MXLog.error("Failed restoring login with error: \(error)")
|
|
return .failure(.failedRestoringLogin)
|
|
}
|
|
}
|
|
|
|
private func setupProxyForClient(_ client: Client) async -> ClientProxyProtocol {
|
|
await ClientProxy(client: client,
|
|
networkMonitor: ServiceLocator.shared.networkMonitor,
|
|
appSettings: ServiceLocator.shared.settings)
|
|
}
|
|
|
|
private func deleteSessionDirectory(for credentials: KeychainCredentials) {
|
|
do {
|
|
try FileManager.default.removeItem(at: credentials.restorationToken.sessionDirectory)
|
|
} catch {
|
|
MXLog.failure("Failed deleting the session data: \(error)")
|
|
}
|
|
}
|
|
|
|
private func deleteCaches(for credentials: KeychainCredentials) {
|
|
do {
|
|
let sessionDirectoryContents = try FileManager.default.contentsOfDirectory(at: credentials.restorationToken.sessionDirectory, includingPropertiesForKeys: nil)
|
|
for url in sessionDirectoryContents where url.path.contains(matrixSDKStateKey) {
|
|
try FileManager.default.removeItem(at: url)
|
|
}
|
|
} catch {
|
|
MXLog.failure("Failed clearing caches: \(error)")
|
|
}
|
|
}
|
|
}
|