mirror of
https://github.com/TelegramMessenger/Telegram-iOS.git
synced 2026-05-21 18:20:41 +00:00
d1aa0db537
Squash of 63 commits spanning waves 46-93 (plus interspersed docs commits) of the gradual Postbox->TelegramEngine consumer-side migration. Scope: 139 files changed, 2123 insertions(+), 452 deletions(-). ## Themes by wave-block **Waves 46-58 — Peer field migrations + facade additions** Foundational EnginePeer convenience init additions (PeerReference, RenderedPeer, SelectivePrivacyPeer). Multiple `peer: Peer` field migrations across PeerInfo, ChatList, and SettingsUI components. **Waves 59-73 — peer field cascade + EnginePeer wrap drops** Series of single- to two-file peer-field migrations; consumer-side wrap removal (`EnginePeer(peer)` -> direct EnginePeer use); `as? TelegramUser` cast conversion to `case let .user(...)` enum match. Wave 64: RenderedPeer convenience init. Wave 68: SelectivePrivacyPeer convenience init. **Waves 74-83 — controller-Node bridge cleanup + small migrations** Wave-71 shadow-pattern cleanup at controller->Node bridges. Migrations of ChatRecentActionsController.peer (74), PeerInfoMember (75), MentionChatInputPanelItem (76), PassportUI SecureIdAuthController (77), AccountWithInfo + ShareController (78), peerInputActivitiesPromise (79), InactiveChannel (80), BlockedPeers (81), openHashtag resolveSignal (82), NotificationExceptionsList (83). **Waves 84-90 — TelegramEngine.Resources facade migrations** Per-method Shape-A/B sweeps converting `<ctx>.account.postbox.mediaBox.X(...)` to `<ctx>.engine.resources.X(...)`. Wave 90 was a single-commit big sweep: 40 fetchedMediaResource sites in 25 files migrated to engine.resources.fetch facade in one atomic pass with first-pass-clean build. Methods covered: storeResourceData, completedResourcePath, cancelInteractiveResourceFetch, resourceRangesStatus, resourceStatus, fetch (fetchedMediaResource). **Waves 91-92 — additional type migrations** Wave 91: ItemListWebsiteItem.peer + RecentSessionsController enum-case payload + openWebSession callback Peer? -> EnginePeer?. Wave 92: ChatListController StateHolder.EntryContext status type MediaResourceStatus -> EngineMediaResource.FetchStatus. **Wave 93 — speculative `import Postbox` drop sweep** Drop import from 7 wave-touched files where it became unused; restore in 5 files where bare PeerId/Message/MediaId/StoryId references escaped the pre-flight regex. Includes one MediaId(...) -> EngineMedia.Id(...) swap in InAppPurchaseManager to unlock its import drop. ## Build state Final state at squash: clean Telegram/Telegram build at debug_sim_arm64. ## Persistent-state notes - Pre-existing WIP unchanged across the squashed range: - build-system/bazel-rules/sourcekit-bazel-bsp submodule marker - Untracked: build-system/tulsi/, submodules/TgVoip/, third-party/libx264/ Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
124 lines
5.0 KiB
Swift
124 lines
5.0 KiB
Swift
import Foundation
|
|
import UIKit
|
|
import Display
|
|
import SwiftSignalKit
|
|
import TelegramCore
|
|
import AccountContext
|
|
import FileMediaResourceStatus
|
|
import ZipArchive
|
|
|
|
private let queue = Queue()
|
|
|
|
public enum AgeVerificationAvailability {
|
|
case available(String, Bool)
|
|
case progress(Float)
|
|
case unavailable
|
|
}
|
|
|
|
private var forceCoreMLVariant: Bool {
|
|
#if targetEnvironment(simulator)
|
|
return true
|
|
#else
|
|
return false
|
|
#endif
|
|
}
|
|
|
|
private func modelPath() -> String {
|
|
return NSTemporaryDirectory() + "AgeNet.mlmodelc"
|
|
}
|
|
private let modelPeer = "agecomputation"
|
|
|
|
private func legacyModelPath() -> String {
|
|
return NSTemporaryDirectory() + "AgeNetLegacy.mlmodelc"
|
|
}
|
|
private let legacyModelPeer = "agelegacycomputation"
|
|
|
|
public func ageVerificationAvailability(context: AccountContext) -> Signal<AgeVerificationAvailability, NoError> {
|
|
let compiledModelPath: String
|
|
let modelPeerName: String
|
|
let isLegacy: Bool
|
|
if #available(iOS 15.0, *) {
|
|
compiledModelPath = modelPath()
|
|
modelPeerName = modelPeer
|
|
isLegacy = false
|
|
} else {
|
|
compiledModelPath = legacyModelPath()
|
|
modelPeerName = legacyModelPeer
|
|
isLegacy = true
|
|
}
|
|
if FileManager.default.fileExists(atPath: compiledModelPath) {
|
|
return .single(.available(compiledModelPath, isLegacy))
|
|
}
|
|
return context.engine.peers.resolvePeerByName(name: modelPeerName, referrer: nil)
|
|
|> mapToSignal { result -> Signal<AgeVerificationAvailability, NoError> in
|
|
guard case let .result(maybePeer) = result else {
|
|
return .complete()
|
|
}
|
|
guard let peer = maybePeer else {
|
|
return .single(.unavailable)
|
|
}
|
|
|
|
return context.account.viewTracker.aroundMessageHistoryViewForLocation(.peer(peerId: peer.id, threadId: nil), index: .lowerBound, anchorIndex: .lowerBound, count: 5, fixedCombinedReadStates: nil)
|
|
|> mapToSignal { view -> Signal<(TelegramMediaFile, EngineMessage)?, NoError> in
|
|
if !view.0.isLoading {
|
|
if let message = view.0.entries.last?.message, let file = message.media.first(where: { $0 is TelegramMediaFile }) as? TelegramMediaFile {
|
|
return .single((file, EngineMessage(message)))
|
|
} else {
|
|
return .single(nil)
|
|
}
|
|
} else {
|
|
return .complete()
|
|
}
|
|
}
|
|
|> take(1)
|
|
|> mapToSignal { maybeFileAndMessage -> Signal<AgeVerificationAvailability, NoError> in
|
|
if let (file, message) = maybeFileAndMessage {
|
|
let fetchedData = context.engine.resources.fetch(reference: FileMediaReference.message(message: MessageReference(message._asMessage()), media: file).resourceReference(file.resource), userLocation: .other, userContentType: .file)
|
|
|
|
enum FetchStatus {
|
|
case completed(String)
|
|
case progress(Float)
|
|
case failed
|
|
}
|
|
|
|
let fetchStatus = Signal<FetchStatus, NoError> { subscriber in
|
|
let fetchedDisposable = fetchedData.start()
|
|
let resourceDataDisposable = context.engine.resources.data(resource: EngineMediaResource(file.resource)).start(next: { next in
|
|
if next.isComplete {
|
|
SSZipArchive.unzipFile(atPath: next.path, toDestination: NSTemporaryDirectory())
|
|
subscriber.putNext(.completed(compiledModelPath))
|
|
subscriber.putCompletion()
|
|
}
|
|
}, error: subscriber.putError, completed: subscriber.putCompletion)
|
|
let progressDisposable = messageFileMediaResourceStatus(context: context, file: file, message: message, isRecentActions: false).start(next: { status in
|
|
switch status.fetchStatus {
|
|
case let .Remote(progress), let .Fetching(_, progress), let .Paused(progress):
|
|
subscriber.putNext(.progress(progress))
|
|
default:
|
|
break
|
|
}
|
|
})
|
|
return ActionDisposable {
|
|
fetchedDisposable.dispose()
|
|
resourceDataDisposable.dispose()
|
|
progressDisposable.dispose()
|
|
}
|
|
}
|
|
return fetchStatus
|
|
|> mapToSignal { status -> Signal<AgeVerificationAvailability, NoError> in
|
|
switch status {
|
|
case .completed:
|
|
return .single(.available(compiledModelPath, isLegacy))
|
|
case let .progress(progress):
|
|
return .single(.progress(progress))
|
|
case .failed:
|
|
return .single(.unavailable)
|
|
}
|
|
}
|
|
} else {
|
|
return .single(.unavailable)
|
|
}
|
|
}
|
|
}
|
|
}
|