Swiftgram/submodules/TelegramUI/Sources/UpgradedAccounts.swift
Peter Iakovlev e9a4a9347a Revert "Rename directories [skip ci]"
This reverts commit 789438a27450dcbdee6065ebf096198ed3b90fec
2020-03-01 10:06:51 +00:00

322 lines
16 KiB
Swift

import Foundation
import UIKit
import TelegramCore
import SyncCore
import Postbox
import SwiftSignalKit
import TelegramUIPreferences
import MediaResources
private enum LegacyPreferencesKeyValues: Int32 {
case cacheStorageSettings = 1
case localizationSettings = 2
case proxySettings = 5
var key: ValueBoxKey {
let key = ValueBoxKey(length: 4)
key.setInt32(0, value: self.rawValue)
return key
}
}
private enum UpgradedSharedDataKeyValues: Int32 {
case cacheStorageSettings = 2
case localizationSettings = 3
case proxySettings = 4
var key: ValueBoxKey {
let key = ValueBoxKey(length: 4)
key.setInt32(0, value: self.rawValue)
return key
}
}
private enum LegacyApplicationSpecificPreferencesKeyValues: Int32 {
case inAppNotificationSettings = 0
case presentationPasscodeSettings = 1
case automaticMediaDownloadSettings = 2
case generatedMediaStoreSettings = 3
case voiceCallSettings = 4
case presentationThemeSettings = 5
case instantPagePresentationSettings = 6
case callListSettings = 7
case experimentalSettings = 8
case musicPlaybackSettings = 9
case mediaInputSettings = 10
case experimentalUISettings = 11
case contactSynchronizationSettings = 12
case stickerSettings = 13
case watchPresetSettings = 14
case webSearchSettings = 15
case voipDerivedState = 16
var key: ValueBoxKey {
return applicationSpecificPreferencesKey(self.rawValue)
}
}
private enum UpgradedApplicationSpecificSharedDataKeyValues: Int32 {
case inAppNotificationSettings = 0
case presentationPasscodeSettings = 1
case automaticMediaDownloadSettings = 2
case generatedMediaStoreSettings = 3
case voiceCallSettings = 4
case presentationThemeSettings = 5
case instantPagePresentationSettings = 6
case callListSettings = 7
case experimentalSettings = 8
case musicPlaybackSettings = 9
case mediaInputSettings = 10
case experimentalUISettings = 11
case stickerSettings = 12
case watchPresetSettings = 13
case webSearchSettings = 14
case contactSynchronizationSettings = 15
var key: ValueBoxKey {
return applicationSpecificSharedDataKey(self.rawValue)
}
}
private let preferencesKeyMapping: [LegacyPreferencesKeyValues: UpgradedSharedDataKeyValues] = [
.cacheStorageSettings: .cacheStorageSettings,
.localizationSettings: .localizationSettings,
.proxySettings: .proxySettings
]
private let applicationSpecificPreferencesKeyMapping: [LegacyApplicationSpecificPreferencesKeyValues: UpgradedApplicationSpecificSharedDataKeyValues] = [
.inAppNotificationSettings: .inAppNotificationSettings,
.presentationPasscodeSettings: .presentationPasscodeSettings,
.automaticMediaDownloadSettings: .automaticMediaDownloadSettings,
.generatedMediaStoreSettings: .generatedMediaStoreSettings,
.voiceCallSettings: .voiceCallSettings,
.presentationThemeSettings: .presentationThemeSettings,
.instantPagePresentationSettings: .instantPagePresentationSettings,
.callListSettings: .callListSettings,
.experimentalSettings: .experimentalSettings,
.musicPlaybackSettings: .musicPlaybackSettings,
.mediaInputSettings: .mediaInputSettings,
.experimentalUISettings: .experimentalUISettings,
.stickerSettings: .stickerSettings,
.watchPresetSettings: .watchPresetSettings,
.webSearchSettings: .webSearchSettings,
.contactSynchronizationSettings: .contactSynchronizationSettings
]
private func upgradedSharedDataValue(_ value: PreferencesEntry?) -> PreferencesEntry? {
if let settings = value as? AutomaticMediaDownloadSettings {
return MediaAutoDownloadSettings.upgradeLegacySettings(settings)
} else {
return value
}
}
public func upgradedAccounts(accountManager: AccountManager, rootPath: String, encryptionParameters: ValueBoxEncryptionParameters) -> Signal<Float, NoError> {
return accountManager.transaction { transaction -> (Int32?, AccountRecordId?) in
return (transaction.getVersion(), transaction.getCurrent()?.0)
}
|> mapToSignal { version, currentId -> Signal<Float, NoError> in
guard let version = version else {
return accountManager.transaction { transaction -> Void in
transaction.setVersion(4)
}
|> ignoreValues
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
}
var signal: Signal<Float, NoError> = .complete()
if version < 1 {
if let currentId = currentId {
let upgradePreferences = accountPreferenceEntries(rootPath: rootPath, id: currentId, keys: Set(preferencesKeyMapping.keys.map({ $0.key }) + applicationSpecificPreferencesKeyMapping.keys.map({ $0.key })), encryptionParameters: encryptionParameters)
|> mapToSignal { result -> Signal<Float, NoError> in
switch result {
case let .progress(progress):
return .single(progress)
case let .result(path, values):
return accountManager.transaction { transaction -> Void in
for (key, value) in values {
var upgradedKey: ValueBoxKey?
for (k, v) in preferencesKeyMapping {
if k.key == key {
upgradedKey = v.key
break
}
}
for (k, v) in applicationSpecificPreferencesKeyMapping {
if k.key == key {
upgradedKey = v.key
break
}
}
if let upgradedKey = upgradedKey {
transaction.updateSharedData(upgradedKey, { _ in
return upgradedSharedDataValue(value)
})
}
}
if let value = values[LegacyApplicationSpecificPreferencesKeyValues.presentationThemeSettings.key] as? PresentationThemeSettings {
let mediaBox = MediaBox(basePath: path + "/postbox/media")
let wallpapers = Array(value.themeSpecificChatWallpapers.values)
for wallpaper in wallpapers {
switch wallpaper {
case let .file(file):
if let path = mediaBox.completedResourcePath(file.file.resource), let data = try? Data(contentsOf: URL(fileURLWithPath: path), options: .mappedRead) {
accountManager.mediaBox.storeResourceData(file.file.resource.id, data: data)
let _ = accountManager.mediaBox.cachedResourceRepresentation(file.file.resource, representation: CachedScaledImageRepresentation(size: CGSize(width: 720.0, height: 720.0), mode: .aspectFit), complete: true, fetch: true).start()
if wallpaper.isPattern {
if let color = file.settings.color, let intensity = file.settings.intensity {
let _ = accountManager.mediaBox.cachedResourceRepresentation(file.file.resource, representation: CachedPatternWallpaperRepresentation(color: color, bottomColor: file.settings.bottomColor, intensity: intensity, rotation: file.settings.rotation), complete: true, fetch: true).start()
}
} else {
if file.settings.blur {
let _ = accountManager.mediaBox.cachedResourceRepresentation(file.file.resource, representation: CachedBlurredWallpaperRepresentation(), complete: true, fetch: true).start()
}
}
}
case let .image(representations, _):
for representation in representations {
let resource = representation.resource
if let path = mediaBox.completedResourcePath(resource), let data = try? Data(contentsOf: URL(fileURLWithPath: path), options: .mappedRead) {
accountManager.mediaBox.storeResourceData(resource.id, data: data)
let _ = mediaBox.cachedResourceRepresentation(resource, representation: CachedScaledImageRepresentation(size: CGSize(width: 720.0, height: 720.0), mode: .aspectFit), complete: true, fetch: true).start()
}
}
default:
break
}
}
}
transaction.setVersion(1)
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
}
}
signal = signal |> then(upgradePreferences)
} else {
let upgradePreferences = accountManager.transaction { transaction -> Void in
transaction.setVersion(1)
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
signal = signal |> then(upgradePreferences)
}
}
if version < 2 {
if let currentId = currentId {
let upgradeNotices = accountNoticeEntries(rootPath: rootPath, id: currentId, encryptionParameters: encryptionParameters)
|> mapToSignal { result -> Signal<Float, NoError> in
switch result {
case let .progress(progress):
return .single(progress)
case let .result(_, values):
return accountManager.transaction { transaction -> Void in
for (key, value) in values {
transaction.setNotice(NoticeEntryKey(namespace: ValueBoxKey(length: 0), key: key), value)
}
transaction.setVersion(2)
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
}
}
signal = signal |> then(upgradeNotices)
} else {
let upgradeNotices = accountManager.transaction { transaction -> Void in
transaction.setVersion(2)
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
signal = signal |> then(upgradeNotices)
}
let upgradeSortOrder = accountManager.transaction { transaction -> Void in
var index: Int32 = 0
for record in transaction.getRecords() {
transaction.updateRecord(record.id, { _ in
return AccountRecord(id: record.id, attributes: record.attributes + [AccountSortOrderAttribute(order: index)], temporarySessionId: record.temporarySessionId)
})
index += 1
}
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
signal = signal |> then(upgradeSortOrder)
}
if version < 3 {
if let currentId = currentId {
let upgradeAccessChallengeData = accountLegacyAccessChallengeData(rootPath: rootPath, id: currentId, encryptionParameters: encryptionParameters)
|> mapToSignal { result -> Signal<Float, NoError> in
switch result {
case let .progress(progress):
return .single(progress)
case let .result(accessChallengeData):
return accountManager.transaction { transaction -> Void in
if case .none = transaction.getAccessChallengeData() {
transaction.setAccessChallengeData(accessChallengeData)
}
transaction.setVersion(3)
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
}
}
signal = signal |> then(upgradeAccessChallengeData)
} else {
let upgradeAccessChallengeData = accountManager.transaction { transaction -> Void in
transaction.setVersion(3)
}
|> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
signal = signal |> then(upgradeAccessChallengeData)
}
}
if version < 4 {
let updatedContactSynchronizationSettings = accountManager.transaction { transaction -> (ContactSynchronizationSettings, [AccountRecordId]) in
return (transaction.getSharedData(ApplicationSpecificSharedDataKeys.contactSynchronizationSettings) as? ContactSynchronizationSettings ?? ContactSynchronizationSettings.defaultSettings, transaction.getRecords().map({ $0.id }))
}
|> mapToSignal { globalSettings, ids -> Signal<Never, NoError> in
var importSignal: Signal<Never, NoError> = .complete()
for id in ids {
let importInfoAccounttSignal = accountTransaction(rootPath: rootPath, id: id, encryptionParameters: encryptionParameters, transaction: { transaction -> Void in
transaction.updatePreferencesEntry(key: PreferencesKeys.contactsSettings, { current in
var settings = current as? ContactsSettings ?? ContactsSettings.defaultSettings
settings.synchronizeContacts = globalSettings._legacySynchronizeDeviceContacts
return settings
})
})
|> ignoreValues
importSignal = importSignal |> then(importInfoAccounttSignal)
}
return importSignal
}
let applyVersion = accountManager.transaction { transaction -> Void in
transaction.setVersion(4)
}
|> ignoreValues
signal = signal |> then(
(updatedContactSynchronizationSettings
|> then(
applyVersion
)) |> mapToSignal { _ -> Signal<Float, NoError> in
return .complete()
}
)
}
return signal
}
}