Merge branch 'feature/7565_Migration_user_info' into 'develop'

Feature/7565 migration user info

See merge request bwmessenger/bundesmessenger/bundesmessenger-ios!439
This commit is contained in:
Frank Rotermund
2025-09-26 04:54:57 +00:00
7 changed files with 191 additions and 2 deletions

View File

@@ -1894,6 +1894,22 @@ NSString *const kRecentsDataSourceTapOnDirectoryServerChange = @"kRecentsDataSou
MXSession* session = [self mxSession];
if(!session)
return;
// bwi #7565 as we have a session here and its a migration step -> use the migrationAssistant here
// this is called everytime when migration level is >= 1 we should find a better place when level 2 and 3 are implemented
MigrationAssistant *migrationAssistant = [[MigrationAssistant alloc] init];
BOOL retVal = [migrationAssistant storeUsernameWithSession:session];
if ( retVal == false) {
MXLogError(@"[RecentsDataSource] shouldShowFeatureBanner could not store displaname for migration")
}
retVal = [migrationAssistant storeHomeserverWithSession:session];
if ( retVal == false) {
MXLogError(@"[RecentsDataSource] shouldShowFeatureBanner could not store homeserver for migration")
}
FeatureBannerVisibilityService *featureBannerService = [FeatureBannerVisibilityService alloc];
[featureBannerService isUnreadWithCompletion: ^(BOOL unread) {
if (unread) {

View File

@@ -622,8 +622,9 @@ class AllChatsViewController: HomeViewController {
// bwi: feature banner cell
if sectionType == .featureBanner {
var username = ""
if let mainSession = self.mainSession {
username = mainSession.myUser.displayname
if let myUsername = self.mainSession.myUser.username() {
// bwi 7565 change to userID
username = myUsername
}
guard let cell = tableView.dequeueReusableCell(withIdentifier: "featureBanner", for: indexPath) as? FeatureBannerViewCell<FeatureBannerView> else {
return UITableViewCell()

View File

@@ -149,5 +149,7 @@
<string>$(BASE_BUNDLE_IDENTIFIER)</string>
<key>keychainAccessGroup</key>
<string>$(KEYCHAIN_ACCESS_GROUP)</string>
<key>KeychainMigrationSharedGroup</key>
<string>$(AppIdentifierPrefix)de.bwi.messenger.shared.migration</string>
</dict>
</plist>

View File

@@ -13,6 +13,7 @@
<key>keychain-access-groups</key>
<array>
<string>$(KEYCHAIN_ACCESS_GROUP)</string>
<string>$(AppIdentifierPrefix)de.bwi.messenger.shared.migration</string>
</array>
</dict>
</plist>

View File

@@ -0,0 +1,34 @@
//
/*
* Copyright (c) 2025 BWI GmbH
*
* 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 MatrixSDK
// bwi #7576 username is used in several places, create a helper
// we need only the username part of the matrix id
extension MXUser {
func username () -> String? {
if let fullID = self.userId {
let withoutAt = fullID.hasPrefix("@") ? String(fullID.dropFirst()) : fullID
if let username = withoutAt.split(separator: ":").first {
return String(username)
}
}
return nil
}
}

View File

@@ -0,0 +1,46 @@
//
/*
* Copyright (c) 2025 BWI GmbH
*
* 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 MatrixSDK
// bwi #7565 has to be a class to be called by Objective C
// reads user name and displayName from session and stores in shared keychain
@objcMembers class MigrationAssistant : NSObject {
func storeUsername( session: MXSession ) -> Bool {
var returnValue = false
if let username = session.myUser.username() {
let ossStatus = SharedKeychain.save(account: "migration_username", value: String(username))
returnValue = ossStatus == errSecSuccess
}
return returnValue
}
func storeHomeserver( session: MXSession ) -> Bool {
var returnValue = false
if let homeserver = session.matrixRestClient.homeserver {
returnValue = SharedKeychain.save(account: "migration_homeserver", value: homeserver) == errSecSuccess
}
return returnValue
}
}

View File

@@ -0,0 +1,89 @@
//
/*
* Copyright (c) 2025 BWI GmbH
*
* 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 Security
import Foundation
enum SharedKeychainConfig {
static var sharedGroup: String {
Bundle.main.object(forInfoDictionaryKey: "KeychainMigrationSharedGroup") as? String
?? ""
}
static var service = "de.bwi.sharedkeychain"
}
// bwi #7565 do some keychain magic, needs to have entitlement for SharedKeychainConfig.sharedGroup and needs to store it in info.plist to hide team id.
enum SharedKeychain {
@discardableResult
static func save(
account: String,
value: String
) -> OSStatus {
let data = value.data(using: .utf8)!
// Delete existing item first
SharedKeychain.delete(account: account)
// Add new value
let addQuery: [String: Any] = [
kSecClass as String: kSecClassGenericPassword,
kSecAttrAccount as String: account,
kSecAttrService as String: SharedKeychainConfig.service,
kSecAttrAccessGroup as String: SharedKeychainConfig.sharedGroup,
kSecValueData as String: data
]
return SecItemAdd(addQuery as CFDictionary, nil)
}
static func load(
account: String
) -> String? {
let query: [String: Any] = [
kSecClass as String: kSecClassGenericPassword,
kSecAttrAccount as String: account,
kSecAttrService as String: SharedKeychainConfig.service,
kSecAttrAccessGroup as String: SharedKeychainConfig.sharedGroup,
kSecMatchLimit as String: kSecMatchLimitOne,
kSecReturnData as String: true
]
var result: AnyObject?
let status = SecItemCopyMatching(query as CFDictionary, &result)
guard status == errSecSuccess,
let data = result as? Data,
let value = String(data: data, encoding: .utf8) else {
return nil
}
return value
}
@discardableResult
static func delete(
account: String
) -> OSStatus {
let query: [String: Any] = [
kSecClass as String: kSecClassGenericPassword,
kSecAttrAccount as String: account,
kSecAttrService as String: SharedKeychainConfig.service,
kSecAttrAccessGroup as String: SharedKeychainConfig.sharedGroup
]
return SecItemDelete(query as CFDictionary)
}
}