Fixed a few issues uncovered when testing the shared util feature flag

Added code to handle an edge-case where an account could exist without a display name if crashing at the right time when onboarding/linking (now request a display name on start up)
Fixed a crash which could occur when downloading new avatars
Fixed a minor layout issue with the seed reminder view
pull/856/head
Morgan Pretty 2 years ago
parent 4d487f2686
commit 29ba25916c

@ -1554,8 +1554,7 @@ extension ConversationVC:
db, db,
roomToken: room, roomToken: room,
server: server, server: server,
publicKey: publicKey, publicKey: publicKey
isConfigMessage: false
) )
} }
.receive(on: DispatchQueue.main) .receive(on: DispatchQueue.main)

@ -405,10 +405,17 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
self.hasInitialRootViewController = true self.hasInitialRootViewController = true
self.window?.rootViewController = StyledNavigationController( self.window?.rootViewController = StyledNavigationController(
rootViewController: (Identity.userExists() ? rootViewController: {
HomeVC() : guard Identity.userExists() else { return LandingVC() }
LandingVC() guard !Profile.fetchOrCreateCurrentUser().name.isEmpty else {
) // If we have no display name then collect one (this can happen if the
// app crashed during onboarding which would leave the user in an invalid
// state with no display name)
return DisplayNameVC(flow: .register)
}
return HomeVC()
}()
) )
UIViewController.attemptRotationToDeviceOrientation() UIViewController.attemptRotationToDeviceOrientation()

@ -9,6 +9,15 @@ import SessionMessagingKit
enum Onboarding { enum Onboarding {
private static let profileNameRetrievalPublisher: Atomic<AnyPublisher<String?, Error>> = { private static let profileNameRetrievalPublisher: Atomic<AnyPublisher<String?, Error>> = {
// FIXME: Remove this once `useSharedUtilForUserConfig` is permanent
guard Features.useSharedUtilForUserConfig else {
return Atomic(
Just(nil)
.setFailureType(to: Error.self)
.eraseToAnyPublisher()
)
}
let userPublicKey: String = getUserHexEncodedPublicKey() let userPublicKey: String = getUserHexEncodedPublicKey()
return Atomic( return Atomic(

@ -82,10 +82,11 @@ final class SeedReminderView: UIView {
// Set up button // Set up button
let button = SessionButton(style: .bordered, size: .small) let button = SessionButton(style: .bordered, size: .small)
button.setContentCompressionResistancePriority(.required, for: .horizontal)
button.accessibilityLabel = "Continue" button.accessibilityLabel = "Continue"
button.isAccessibilityElement = true button.isAccessibilityElement = true
button.setTitle("continue_2".localized(), for: UIControl.State.normal) button.setTitle("continue_2".localized(), for: UIControl.State.normal)
button.set(.width, to: 96) button.set(.width, greaterThanOrEqualTo: 96)
button.addTarget(self, action: #selector(handleContinueButtonTapped), for: UIControl.Event.touchUpInside) button.addTarget(self, action: #selector(handleContinueButtonTapped), for: UIControl.Event.touchUpInside)
// Set up content stack view // Set up content stack view

@ -174,8 +174,7 @@ final class JoinOpenGroupVC: BaseVC, UIPageViewControllerDataSource, UIPageViewC
db, db,
roomToken: roomToken, roomToken: roomToken,
server: server, server: server,
publicKey: publicKey, publicKey: publicKey
isConfigMessage: false
) )
} }
.receive(on: DispatchQueue.main) .receive(on: DispatchQueue.main)

@ -265,11 +265,13 @@ public extension ConfigurationSyncJob {
// FIXME: Remove this once `useSharedUtilForUserConfig` is permanent // FIXME: Remove this once `useSharedUtilForUserConfig` is permanent
guard Features.useSharedUtilForUserConfig else { guard Features.useSharedUtilForUserConfig else {
// If we don't have a userKeyPair yet then there is no need to sync the configuration // If we don't have a userKeyPair (or name) yet then there is no need to sync the
// as the user doesn't exist yet (this will get triggered on the first launch of a // configuration as the user doesn't fully exist yet (this will get triggered on
// fresh install due to the migrations getting run) // the first launch of a fresh install due to the migrations getting run and a few
// times during onboarding)
guard guard
Identity.userExists(db), Identity.userExists(db),
!Profile.fetchOrCreateCurrentUser(db).name.isEmpty,
let legacyConfigMessage: Message = try? ConfigurationMessage.getCurrent(db) let legacyConfigMessage: Message = try? ConfigurationMessage.getCurrent(db)
else { return } else { return }

@ -60,6 +60,12 @@ public extension QueryInterfaceRequest where RowDecoder: FetchableRecord & Table
} }
} }
// FIXME: Remove this once `useSharedUtilForUserConfig` is permanent
guard Features.useSharedUtilForUserConfig else {
return try self.updateAndFetchAll(db, assignments)
}
// Update the config dump state where needed
switch self { switch self {
case is QueryInterfaceRequest<Contact>: case is QueryInterfaceRequest<Contact>:
return try SessionUtil.updatingContacts(db, try updateAndFetchAll(db, assignments)) return try SessionUtil.updatingContacts(db, try updateAndFetchAll(db, assignments))

@ -280,6 +280,8 @@ public enum SessionUtil {
messages: [SharedConfigMessage], messages: [SharedConfigMessage],
publicKey: String publicKey: String
) throws { ) throws {
// FIXME: Remove this once `useSharedUtilForUserConfig` is permanent
guard Features.useSharedUtilForUserConfig else { return }
guard !messages.isEmpty else { return } guard !messages.isEmpty else { return }
guard !publicKey.isEmpty else { throw MessageReceiverError.noThread } guard !publicKey.isEmpty else { throw MessageReceiverError.noThread }

@ -946,7 +946,6 @@ public enum OpenGroupAPI {
queue: OpenGroupAPI.workQueue queue: OpenGroupAPI.workQueue
) )
) -> AnyPublisher<(ResponseInfoType, Data), Error> { ) -> AnyPublisher<(ResponseInfoType, Data), Error> {
print("Download File")
return OpenGroupAPI return OpenGroupAPI
.send( .send(
db, db,
@ -957,7 +956,6 @@ public enum OpenGroupAPI {
using: dependencies using: dependencies
) )
.flatMap { responseInfo, maybeData -> AnyPublisher<(ResponseInfoType, Data), Error> in .flatMap { responseInfo, maybeData -> AnyPublisher<(ResponseInfoType, Data), Error> in
print("Download File FlatMap")
guard let data: Data = maybeData else { guard let data: Data = maybeData else {
return Fail(error: HTTPError.parsingFailed) return Fail(error: HTTPError.parsingFailed)
.eraseToAnyPublisher() .eraseToAnyPublisher()

@ -207,12 +207,12 @@ public final class OpenGroupManager {
roomToken: String, roomToken: String,
server: String, server: String,
publicKey: String, publicKey: String,
isConfigMessage: Bool, calledFromConfigHandling: Bool = false,
dependencies: OGMDependencies = OGMDependencies() dependencies: OGMDependencies = OGMDependencies()
) -> AnyPublisher<Void, Error> { ) -> AnyPublisher<Void, Error> {
// If we are currently polling for this server and already have a TSGroupThread for this room the do nothing // If we are currently polling for this server and already have a TSGroupThread for this room the do nothing
if hasExistingOpenGroup(db, roomToken: roomToken, server: server, publicKey: publicKey, dependencies: dependencies) { if hasExistingOpenGroup(db, roomToken: roomToken, server: server, publicKey: publicKey, dependencies: dependencies) {
SNLog("Ignoring join open group attempt (already joined), user initiated: \(!isConfigMessage)") SNLog("Ignoring join open group attempt (already joined), user initiated: \(!calledFromConfigHandling)")
return Just(()) return Just(())
.setFailureType(to: Error.self) .setFailureType(to: Error.self)
.eraseToAnyPublisher() .eraseToAnyPublisher()
@ -276,7 +276,9 @@ public final class OpenGroupManager {
Future<Void, Error> { resolver in Future<Void, Error> { resolver in
dependencies.storage.write { db in dependencies.storage.write { db in
// Enqueue a config sync job (have a newly added open group to sync) // Enqueue a config sync job (have a newly added open group to sync)
ConfigurationSyncJob.enqueue(db) if !calledFromConfigHandling {
ConfigurationSyncJob.enqueue(db)
}
// Store the capabilities first // Store the capabilities first
OpenGroupManager.handleCapabilities( OpenGroupManager.handleCapabilities(

@ -43,7 +43,8 @@ extension MessageReceiver {
fileName: nil fileName: nil
) )
}(), }(),
sentTimestamp: messageSentTimestamp sentTimestamp: messageSentTimestamp,
calledFromConfigHandling: true
) )
// Create a contact for the current user if needed (also force-approve the current user // Create a contact for the current user if needed (also force-approve the current user
@ -197,7 +198,13 @@ extension MessageReceiver {
for openGroupURL in message.openGroups { for openGroupURL in message.openGroups {
if let (room, server, publicKey) = OpenGroupManager.parseOpenGroup(from: openGroupURL) { if let (room, server, publicKey) = OpenGroupManager.parseOpenGroup(from: openGroupURL) {
OpenGroupManager.shared OpenGroupManager.shared
.add(db, roomToken: room, server: server, publicKey: publicKey, isConfigMessage: true) .add(
db,
roomToken: room,
server: server,
publicKey: publicKey,
calledFromConfigHandling: true
)
.sinkUntilComplete() .sinkUntilComplete()
} }
} }

@ -594,6 +594,25 @@ public struct ProfileManager {
profileChanges profileChanges
) )
} }
// FIXME: Remove this once `useSharedUtilForUserConfig` is permanent
else if !Features.useSharedUtilForUserConfig {
// If we have a contact record for the profile (ie. it's a synced profile) then
// should should send an updated config message, otherwise we should just update
// the local state (the shared util has this logic build in to it's handling)
if (try? Contact.exists(db, id: publicKey)) == true {
try Profile
.filter(id: publicKey)
.updateAllAndConfig(db, profileChanges)
}
else {
try Profile
.filter(id: publicKey)
.updateAll(
db,
profileChanges
)
}
}
else { else {
try Profile try Profile
.filter(id: publicKey) .filter(id: publicKey)
@ -606,7 +625,7 @@ public struct ProfileManager {
db.afterNextTransaction { db in db.afterNextTransaction { db in
// Need to refetch to ensure the db changes have occurred // Need to refetch to ensure the db changes have occurred
ProfileManager.downloadAvatar(for: Profile.fetchOrCreate(id: publicKey)) ProfileManager.downloadAvatar(for: Profile.fetchOrCreate(db, id: publicKey))
} }
} }
} }

@ -822,7 +822,6 @@ class OpenGroupManagerSpec: QuickSpec {
roomToken: "testRoom", roomToken: "testRoom",
server: "testServer", server: "testServer",
publicKey: TestConstants.serverPublicKey, publicKey: TestConstants.serverPublicKey,
isConfigMessage: false,
dependencies: dependencies dependencies: dependencies
) )
} }
@ -853,7 +852,6 @@ class OpenGroupManagerSpec: QuickSpec {
roomToken: "testRoom", roomToken: "testRoom",
server: "testServer", server: "testServer",
publicKey: TestConstants.serverPublicKey, publicKey: TestConstants.serverPublicKey,
isConfigMessage: false,
dependencies: dependencies dependencies: dependencies
) )
} }
@ -892,7 +890,6 @@ class OpenGroupManagerSpec: QuickSpec {
publicKey: TestConstants.serverPublicKey publicKey: TestConstants.serverPublicKey
.replacingOccurrences(of: "c3", with: "00") .replacingOccurrences(of: "c3", with: "00")
.replacingOccurrences(of: "b3", with: "00"), .replacingOccurrences(of: "b3", with: "00"),
isConfigMessage: false,
dependencies: dependencies dependencies: dependencies
) )
} }
@ -946,7 +943,6 @@ class OpenGroupManagerSpec: QuickSpec {
roomToken: "testRoom", roomToken: "testRoom",
server: "testServer", server: "testServer",
publicKey: TestConstants.serverPublicKey, publicKey: TestConstants.serverPublicKey,
isConfigMessage: false,
dependencies: dependencies dependencies: dependencies
) )
} }

Loading…
Cancel
Save