mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-15 13:35:19 +00:00
867 lines
36 KiB
Swift
867 lines
36 KiB
Swift
import Foundation
|
|
import Postbox
|
|
import SwiftSignalKit
|
|
import TelegramApi
|
|
import MtProtoKit
|
|
|
|
import SyncCore
|
|
|
|
public enum CreatePeerExportedInvitationError {
|
|
case generic
|
|
}
|
|
|
|
public func createPeerExportedInvitation(account: Account, peerId: PeerId, expireDate: Int32?, usageLimit: Int32?) -> Signal<ExportedInvitation?, CreatePeerExportedInvitationError> {
|
|
return account.postbox.transaction { transaction -> Signal<ExportedInvitation?, CreatePeerExportedInvitationError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) {
|
|
var flags: Int32 = 0
|
|
if let _ = expireDate {
|
|
flags |= (1 << 0)
|
|
}
|
|
if let _ = usageLimit {
|
|
flags |= (1 << 1)
|
|
}
|
|
return account.network.request(Api.functions.messages.exportChatInvite(flags: flags, peer: inputPeer, expireDate: expireDate, usageLimit: usageLimit))
|
|
|> mapError { _ in return CreatePeerExportedInvitationError.generic }
|
|
|> map { result -> ExportedInvitation? in
|
|
return ExportedInvitation(apiExportedInvite: result)
|
|
}
|
|
} else {
|
|
return .complete()
|
|
}
|
|
}
|
|
|> castError(CreatePeerExportedInvitationError.self)
|
|
|> switchToLatest
|
|
}
|
|
|
|
public enum EditPeerExportedInvitationError {
|
|
case generic
|
|
}
|
|
|
|
public func editPeerExportedInvitation(account: Account, peerId: PeerId, link: String, expireDate: Int32?, usageLimit: Int32?) -> Signal<ExportedInvitation?, EditPeerExportedInvitationError> {
|
|
return account.postbox.transaction { transaction -> Signal<ExportedInvitation?, EditPeerExportedInvitationError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) {
|
|
var flags: Int32 = 0
|
|
if let _ = expireDate {
|
|
flags |= (1 << 0)
|
|
}
|
|
if let _ = usageLimit {
|
|
flags |= (1 << 1)
|
|
}
|
|
return account.network.request(Api.functions.messages.editExportedChatInvite(flags: flags, peer: inputPeer, link: link, expireDate: expireDate, usageLimit: usageLimit))
|
|
|> mapError { _ in return EditPeerExportedInvitationError.generic }
|
|
|> mapToSignal { result -> Signal<ExportedInvitation?, EditPeerExportedInvitationError> in
|
|
return account.postbox.transaction { transaction in
|
|
if case let .exportedChatInvite(invite, users) = result {
|
|
var peers: [Peer] = []
|
|
for user in users {
|
|
let telegramUser = TelegramUser(user: user)
|
|
peers.append(telegramUser)
|
|
}
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in
|
|
return updated
|
|
})
|
|
return ExportedInvitation(apiExportedInvite: invite)
|
|
} else {
|
|
return nil
|
|
}
|
|
} |> mapError { _ in .generic }
|
|
}
|
|
} else {
|
|
return .complete()
|
|
}
|
|
}
|
|
|> castError(EditPeerExportedInvitationError.self)
|
|
|> switchToLatest
|
|
}
|
|
|
|
public enum RevokePeerExportedInvitationError {
|
|
case generic
|
|
}
|
|
|
|
public enum RevokeExportedInvitationResult {
|
|
case update(ExportedInvitation)
|
|
case replace(ExportedInvitation, ExportedInvitation)
|
|
}
|
|
|
|
public func revokePeerExportedInvitation(account: Account, peerId: PeerId, link: String) -> Signal<RevokeExportedInvitationResult?, RevokePeerExportedInvitationError> {
|
|
return account.postbox.transaction { transaction -> Signal<RevokeExportedInvitationResult?, RevokePeerExportedInvitationError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) {
|
|
let flags: Int32 = (1 << 2)
|
|
return account.network.request(Api.functions.messages.editExportedChatInvite(flags: flags, peer: inputPeer, link: link, expireDate: nil, usageLimit: nil))
|
|
|> mapError { _ in return RevokePeerExportedInvitationError.generic }
|
|
|> mapToSignal { result -> Signal<RevokeExportedInvitationResult?, RevokePeerExportedInvitationError> in
|
|
return account.postbox.transaction { transaction in
|
|
if case let .exportedChatInvite(invite, users) = result {
|
|
var peers: [Peer] = []
|
|
for user in users {
|
|
let telegramUser = TelegramUser(user: user)
|
|
peers.append(telegramUser)
|
|
}
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in
|
|
return updated
|
|
})
|
|
return .update(ExportedInvitation(apiExportedInvite: invite))
|
|
} else if case let .exportedChatInviteReplaced(invite, newInvite, users) = result {
|
|
var peers: [Peer] = []
|
|
for user in users {
|
|
let telegramUser = TelegramUser(user: user)
|
|
peers.append(telegramUser)
|
|
}
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in
|
|
return updated
|
|
})
|
|
return .replace(ExportedInvitation(apiExportedInvite: invite), ExportedInvitation(apiExportedInvite: newInvite))
|
|
} else {
|
|
return nil
|
|
}
|
|
} |> mapError { _ in .generic }
|
|
}
|
|
} else {
|
|
return .complete()
|
|
}
|
|
}
|
|
|> castError(RevokePeerExportedInvitationError.self)
|
|
|> switchToLatest
|
|
}
|
|
|
|
public struct ExportedInvitations : Equatable {
|
|
public let list: [ExportedInvitation]?
|
|
public let totalCount: Int32
|
|
}
|
|
|
|
public func peerExportedInvitations(account: Account, peerId: PeerId, revoked: Bool, adminId: PeerId? = nil, offsetLink: ExportedInvitation? = nil) -> Signal<ExportedInvitations?, NoError> {
|
|
return account.postbox.transaction { transaction -> Signal<ExportedInvitations?, NoError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer), let adminPeer = transaction.getPeer(adminId ?? account.peerId), let adminId = apiInputUser(adminPeer) {
|
|
var flags: Int32 = 0
|
|
if let _ = offsetLink {
|
|
flags |= (1 << 2)
|
|
}
|
|
if revoked {
|
|
flags |= (1 << 3)
|
|
}
|
|
return account.network.request(Api.functions.messages.getExportedChatInvites(flags: flags, peer: inputPeer, adminId: adminId, offsetDate: offsetLink?.date, offsetLink: offsetLink?.link, limit: 50))
|
|
|> map(Optional.init)
|
|
|> `catch` { _ -> Signal<Api.messages.ExportedChatInvites?, NoError> in
|
|
return .single(nil)
|
|
}
|
|
|> mapToSignal { result -> Signal<ExportedInvitations?, NoError> in
|
|
return account.postbox.transaction { transaction -> ExportedInvitations? in
|
|
if let result = result, case let .exportedChatInvites(count, apiInvites, users) = result {
|
|
var peers: [Peer] = []
|
|
var peersMap: [PeerId: Peer] = [:]
|
|
for user in users {
|
|
let telegramUser = TelegramUser(user: user)
|
|
peers.append(telegramUser)
|
|
peersMap[telegramUser.id] = telegramUser
|
|
}
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in
|
|
return updated
|
|
})
|
|
|
|
let invites = apiInvites.map { ExportedInvitation(apiExportedInvite: $0) }
|
|
return ExportedInvitations(list: invites, totalCount: count)
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
return .single(nil)
|
|
}
|
|
} |> switchToLatest
|
|
}
|
|
|
|
|
|
public enum DeletePeerExportedInvitationError {
|
|
case generic
|
|
}
|
|
|
|
public func deletePeerExportedInvitation(account: Account, peerId: PeerId, link: String) -> Signal<Never, DeletePeerExportedInvitationError> {
|
|
return account.postbox.transaction { transaction -> Signal<Never, DeletePeerExportedInvitationError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) {
|
|
return account.network.request(Api.functions.messages.deleteExportedChatInvite(peer: inputPeer, link: link))
|
|
|> mapError { _ in return DeletePeerExportedInvitationError.generic }
|
|
|> ignoreValues
|
|
} else {
|
|
return .complete()
|
|
}
|
|
}
|
|
|> castError(DeletePeerExportedInvitationError.self)
|
|
|> switchToLatest
|
|
}
|
|
|
|
public func deleteAllRevokedPeerExportedInvitations(account: Account, peerId: PeerId, adminId: PeerId) -> Signal<Never, NoError> {
|
|
return account.postbox.transaction { transaction -> Signal<Never, NoError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer), let adminPeer = transaction.getPeer(adminId), let inputAdminId = apiInputUser(adminPeer) {
|
|
return account.network.request(Api.functions.messages.deleteRevokedExportedChatInvites(peer: inputPeer, adminId: inputAdminId))
|
|
|> `catch` { _ -> Signal<Api.Bool, NoError> in
|
|
return .single(.boolFalse)
|
|
}
|
|
|> ignoreValues
|
|
} else {
|
|
return .complete()
|
|
}
|
|
}
|
|
|> switchToLatest
|
|
}
|
|
|
|
private let cachedPeerExportedInvitationsCollectionSpec = ItemCacheCollectionSpec(lowWaterItemCount: 10, highWaterItemCount: 20)
|
|
|
|
public struct PeerExportedInvitationsState: Equatable {
|
|
public var invitations: [ExportedInvitation]
|
|
public var isLoadingMore: Bool
|
|
public var hasLoadedOnce: Bool
|
|
public var canLoadMore: Bool
|
|
public var count: Int32
|
|
|
|
public init() {
|
|
self.invitations = []
|
|
self.isLoadingMore = false
|
|
self.hasLoadedOnce = false
|
|
self.canLoadMore = false
|
|
self.count = 0
|
|
}
|
|
|
|
public init(invitations: [ExportedInvitation], isLoadingMore: Bool, hasLoadedOnce: Bool, canLoadMore: Bool, count: Int32) {
|
|
self.invitations = invitations
|
|
self.isLoadingMore = isLoadingMore
|
|
self.hasLoadedOnce = hasLoadedOnce
|
|
self.canLoadMore = canLoadMore
|
|
self.count = count
|
|
}
|
|
}
|
|
|
|
final class CachedPeerExportedInvitations: PostboxCoding {
|
|
let invitations: [ExportedInvitation]
|
|
let canLoadMore: Bool
|
|
let count: Int32
|
|
|
|
public static func key(peerId: PeerId, revoked: Bool) -> ValueBoxKey {
|
|
let key = ValueBoxKey(length: 8 + 4)
|
|
key.setInt64(0, value: peerId.toInt64())
|
|
key.setInt32(8, value: revoked ? 1 : 0)
|
|
return key
|
|
}
|
|
|
|
init(invitations: [ExportedInvitation], canLoadMore: Bool, count: Int32) {
|
|
self.invitations = invitations
|
|
self.canLoadMore = canLoadMore
|
|
self.count = count
|
|
}
|
|
|
|
public init(decoder: PostboxDecoder) {
|
|
self.invitations = decoder.decodeObjectArrayForKey("invitations")
|
|
self.canLoadMore = decoder.decodeBoolForKey("canLoadMore", orElse: false)
|
|
self.count = decoder.decodeInt32ForKey("count", orElse: 0)
|
|
}
|
|
|
|
public func encode(_ encoder: PostboxEncoder) {
|
|
encoder.encodeObjectArray(self.invitations, forKey: "invitations")
|
|
encoder.encodeBool(self.canLoadMore, forKey: "canLoadMore")
|
|
encoder.encodeInt32(self.count, forKey: "count")
|
|
}
|
|
}
|
|
|
|
private final class PeerExportedInvitationsContextImpl {
|
|
private let queue: Queue
|
|
private let account: Account
|
|
private let peerId: PeerId
|
|
private let adminId: PeerId
|
|
private let revoked: Bool
|
|
private var forceUpdate: Bool
|
|
private let disposable = MetaDisposable()
|
|
private let updateDisposable = MetaDisposable()
|
|
private var isLoadingMore: Bool = false
|
|
private var hasLoadedOnce: Bool = false
|
|
private var canLoadMore: Bool = true
|
|
private var loadedFromCache: Bool = false
|
|
private var results: [ExportedInvitation] = []
|
|
private var count: Int32
|
|
private var populateCache: Bool = true
|
|
private var isMainList: Bool
|
|
|
|
let state = Promise<PeerExportedInvitationsState>()
|
|
|
|
init(queue: Queue, account: Account, peerId: PeerId, adminId: PeerId?, revoked: Bool, forceUpdate: Bool) {
|
|
self.queue = queue
|
|
self.account = account
|
|
self.peerId = peerId
|
|
self.adminId = adminId ?? account.peerId
|
|
self.revoked = revoked
|
|
self.forceUpdate = forceUpdate
|
|
self.isMainList = adminId == nil
|
|
|
|
self.count = 0
|
|
|
|
if adminId == nil {
|
|
self.isLoadingMore = true
|
|
self.disposable.set((account.postbox.transaction { transaction -> CachedPeerExportedInvitations? in
|
|
return transaction.retrieveItemCacheEntry(id: ItemCacheEntryId(collectionId: Namespaces.CachedItemCollection.cachedPeerExportedInvitations, key: CachedPeerExportedInvitations.key(peerId: peerId, revoked: revoked))) as? CachedPeerExportedInvitations
|
|
}
|
|
|> deliverOn(self.queue)).start(next: { [weak self] cachedResult in
|
|
guard let strongSelf = self else {
|
|
return
|
|
}
|
|
strongSelf.isLoadingMore = false
|
|
if let cachedResult = cachedResult {
|
|
strongSelf.results = cachedResult.invitations
|
|
strongSelf.count = cachedResult.count
|
|
strongSelf.hasLoadedOnce = true
|
|
strongSelf.canLoadMore = cachedResult.canLoadMore
|
|
strongSelf.loadedFromCache = true
|
|
}
|
|
strongSelf.loadMore()
|
|
}))
|
|
}
|
|
|
|
self.loadMore()
|
|
}
|
|
|
|
deinit {
|
|
self.disposable.dispose()
|
|
self.updateDisposable.dispose()
|
|
}
|
|
|
|
func reload() {
|
|
self.forceUpdate = true
|
|
self.loadMore()
|
|
}
|
|
|
|
func loadMore() {
|
|
if self.isLoadingMore {
|
|
return
|
|
}
|
|
self.isLoadingMore = true
|
|
let account = self.account
|
|
let peerId = self.peerId
|
|
let adminId = self.adminId
|
|
let revoked = self.revoked
|
|
var lastResult = self.results.last
|
|
|
|
if self.forceUpdate {
|
|
self.populateCache = self.isMainList
|
|
self.forceUpdate = false
|
|
lastResult = nil
|
|
} else if self.loadedFromCache {
|
|
self.populateCache = false
|
|
self.loadedFromCache = false
|
|
}
|
|
let populateCache = self.populateCache
|
|
|
|
self.disposable.set((self.account.postbox.transaction { transaction -> (peerId: Api.InputPeer?, adminId: Api.InputUser?) in
|
|
return (transaction.getPeer(peerId).flatMap(apiInputPeer), transaction.getPeer(adminId).flatMap(apiInputUser))
|
|
}
|
|
|> mapToSignal { inputPeer, adminId -> Signal<([ExportedInvitation], Int32), NoError> in
|
|
if let inputPeer = inputPeer, let adminId = adminId {
|
|
let offsetLink = lastResult?.link
|
|
let offsetDate = lastResult?.date
|
|
var flags: Int32 = 0
|
|
if let _ = offsetLink {
|
|
flags |= (1 << 2)
|
|
}
|
|
if revoked {
|
|
flags |= (1 << 3)
|
|
}
|
|
let signal = account.network.request(Api.functions.messages.getExportedChatInvites(flags: flags, peer: inputPeer, adminId: adminId, offsetDate: offsetDate, offsetLink: offsetLink, limit: lastResult == nil ? 50 : 100))
|
|
|> map(Optional.init)
|
|
|> `catch` { _ -> Signal<Api.messages.ExportedChatInvites?, NoError> in
|
|
return .single(nil)
|
|
}
|
|
|> mapToSignal { result -> Signal<([ExportedInvitation], Int32), NoError> in
|
|
return account.postbox.transaction { transaction -> ([ExportedInvitation], Int32) in
|
|
guard let result = result else {
|
|
return ([], 0)
|
|
}
|
|
switch result {
|
|
case let .exportedChatInvites(count, invites, users):
|
|
var peers: [Peer] = []
|
|
for apiUser in users {
|
|
peers.append(TelegramUser(user: apiUser))
|
|
}
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated in
|
|
return updated
|
|
})
|
|
let invitations: [ExportedInvitation] = invites.compactMap { ExportedInvitation(apiExportedInvite: $0) }
|
|
if populateCache {
|
|
transaction.putItemCacheEntry(id: ItemCacheEntryId(collectionId: Namespaces.CachedItemCollection.cachedPeerExportedInvitations, key: CachedPeerExportedInvitations.key(peerId: peerId, revoked: revoked)), entry: CachedPeerExportedInvitations(invitations: invitations, canLoadMore: count >= 50, count: count), collectionSpec: cachedPeerExportedInvitationsCollectionSpec)
|
|
}
|
|
return (invitations, count)
|
|
}
|
|
}
|
|
}
|
|
return signal
|
|
} else {
|
|
return .single(([], 0))
|
|
}
|
|
}
|
|
|> deliverOn(self.queue)).start(next: { [weak self] invitations, updatedCount in
|
|
guard let strongSelf = self else {
|
|
return
|
|
}
|
|
if strongSelf.populateCache {
|
|
strongSelf.populateCache = false
|
|
strongSelf.results.removeAll()
|
|
}
|
|
var existingLinks = Set(strongSelf.results.map { $0.link })
|
|
for invitation in invitations {
|
|
if !existingLinks.contains(invitation.link) {
|
|
strongSelf.results.append(invitation)
|
|
existingLinks.insert(invitation.link)
|
|
}
|
|
}
|
|
strongSelf.isLoadingMore = false
|
|
strongSelf.hasLoadedOnce = true
|
|
strongSelf.canLoadMore = !invitations.isEmpty
|
|
if strongSelf.canLoadMore {
|
|
strongSelf.count = max(updatedCount, Int32(strongSelf.results.count))
|
|
} else {
|
|
strongSelf.count = Int32(strongSelf.results.count)
|
|
}
|
|
strongSelf.updateState()
|
|
|
|
if strongSelf.forceUpdate {
|
|
strongSelf.loadMore()
|
|
}
|
|
}))
|
|
self.updateState()
|
|
}
|
|
|
|
public func add(_ invite: ExportedInvitation) {
|
|
var results = self.results
|
|
results.removeAll(where: { $0.link == invite.link})
|
|
results.insert(invite, at: 0)
|
|
self.results = results
|
|
self.updateState()
|
|
self.updateCache()
|
|
}
|
|
|
|
public func update(_ invite: ExportedInvitation) {
|
|
var results = self.results
|
|
if let index = self.results.firstIndex(where: { $0.link == invite.link }) {
|
|
results[index] = invite
|
|
}
|
|
self.results = results
|
|
self.updateState()
|
|
self.updateCache()
|
|
}
|
|
|
|
public func remove(_ invite: ExportedInvitation) {
|
|
var results = self.results
|
|
results.removeAll(where: { $0.link == invite.link})
|
|
self.results = results
|
|
self.updateState()
|
|
self.updateCache()
|
|
}
|
|
|
|
public func clear() {
|
|
self.results = []
|
|
self.count = 0
|
|
self.updateState()
|
|
self.updateCache()
|
|
}
|
|
|
|
private func updateCache() {
|
|
guard self.hasLoadedOnce && !self.isLoadingMore else {
|
|
return
|
|
}
|
|
|
|
let peerId = self.peerId
|
|
let revoked = self.revoked
|
|
let invitations = Array(self.results.prefix(50))
|
|
let canLoadMore = self.canLoadMore
|
|
let count = self.count
|
|
self.updateDisposable.set(self.account.postbox.transaction({ transaction in
|
|
transaction.putItemCacheEntry(id: ItemCacheEntryId(collectionId: Namespaces.CachedItemCollection.cachedPeerExportedInvitations, key: CachedPeerExportedInvitations.key(peerId: peerId, revoked: revoked)), entry: CachedPeerExportedInvitations(invitations: invitations, canLoadMore: canLoadMore, count: count), collectionSpec: cachedPeerExportedInvitationsCollectionSpec)
|
|
}).start())
|
|
}
|
|
|
|
private func updateState() {
|
|
self.state.set(.single(PeerExportedInvitationsState(invitations: self.results, isLoadingMore: self.isLoadingMore, hasLoadedOnce: self.hasLoadedOnce, canLoadMore: self.canLoadMore, count: self.count)))
|
|
}
|
|
}
|
|
|
|
public final class PeerExportedInvitationsContext {
|
|
private let queue: Queue = Queue()
|
|
private let impl: QueueLocalObject<PeerExportedInvitationsContextImpl>
|
|
|
|
public var state: Signal<PeerExportedInvitationsState, NoError> {
|
|
return Signal { subscriber in
|
|
let disposable = MetaDisposable()
|
|
self.impl.with { impl in
|
|
disposable.set(impl.state.get().start(next: { value in
|
|
subscriber.putNext(value)
|
|
}))
|
|
}
|
|
return disposable
|
|
}
|
|
}
|
|
|
|
public init(account: Account, peerId: PeerId, adminId: PeerId?, revoked: Bool, forceUpdate: Bool) {
|
|
let queue = self.queue
|
|
self.impl = QueueLocalObject(queue: queue, generate: {
|
|
return PeerExportedInvitationsContextImpl(queue: queue, account: account, peerId: peerId, adminId: adminId, revoked: revoked, forceUpdate: forceUpdate)
|
|
})
|
|
}
|
|
|
|
public func reload() {
|
|
self.impl.with { impl in
|
|
impl.reload()
|
|
}
|
|
}
|
|
|
|
public func loadMore() {
|
|
self.impl.with { impl in
|
|
impl.loadMore()
|
|
}
|
|
}
|
|
|
|
public func add(_ invite: ExportedInvitation) {
|
|
self.impl.with { impl in
|
|
impl.add(invite)
|
|
}
|
|
}
|
|
|
|
public func update(_ invite: ExportedInvitation) {
|
|
self.impl.with { impl in
|
|
impl.update(invite)
|
|
}
|
|
}
|
|
|
|
public func remove(_ invite: ExportedInvitation) {
|
|
self.impl.with { impl in
|
|
impl.remove(invite)
|
|
}
|
|
}
|
|
|
|
public func clear() {
|
|
self.impl.with { impl in
|
|
impl.clear()
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
private let cachedPeerInvitationImportersCollectionSpec = ItemCacheCollectionSpec(lowWaterItemCount: 10, highWaterItemCount: 20)
|
|
|
|
public struct PeerInvitationImportersState: Equatable {
|
|
public struct Importer: Equatable {
|
|
public var peer: RenderedPeer
|
|
public var date: Int32
|
|
}
|
|
public var importers: [Importer]
|
|
public var isLoadingMore: Bool
|
|
public var hasLoadedOnce: Bool
|
|
public var canLoadMore: Bool
|
|
public var count: Int32
|
|
}
|
|
|
|
final class CachedPeerInvitationImporters: PostboxCoding {
|
|
let peerIds: [PeerId]
|
|
let dates: [PeerId: Int32]
|
|
let count: Int32
|
|
|
|
public static func key(peerId: PeerId, link: String) -> ValueBoxKey {
|
|
let key = ValueBoxKey(length: 8 + 4)
|
|
key.setInt64(0, value: peerId.toInt64())
|
|
key.setInt32(8, value: Int32(HashFunctions.murMurHash32(link)))
|
|
return key
|
|
}
|
|
|
|
init(importers: [PeerInvitationImportersState.Importer], count: Int32) {
|
|
self.peerIds = importers.map { $0.peer.peerId }
|
|
self.dates = importers.reduce(into: [PeerId: Int32]()) {
|
|
$0[$1.peer.peerId] = $1.date
|
|
}
|
|
self.count = count
|
|
}
|
|
|
|
public init(peerIds: [PeerId], dates: [PeerId: Int32], count: Int32) {
|
|
self.peerIds = peerIds
|
|
self.dates = dates
|
|
self.count = count
|
|
}
|
|
|
|
public init(decoder: PostboxDecoder) {
|
|
self.peerIds = decoder.decodeInt64ArrayForKey("peerIds").map(PeerId.init)
|
|
|
|
var dates: [PeerId: Int32] = [:]
|
|
let datesArray = decoder.decodeInt32ArrayForKey("dates")
|
|
for index in stride(from: 0, to: datesArray.endIndex, by: 2) {
|
|
let userId = datesArray[index]
|
|
let date = datesArray[index + 1]
|
|
let peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: userId)
|
|
dates[peerId] = date
|
|
}
|
|
self.dates = dates
|
|
|
|
self.count = decoder.decodeInt32ForKey("count", orElse: 0)
|
|
}
|
|
|
|
public func encode(_ encoder: PostboxEncoder) {
|
|
encoder.encodeInt64Array(self.peerIds.map { $0.toInt64() }, forKey: "peerIds")
|
|
|
|
var dates: [Int32] = []
|
|
for (peerId, date) in self.dates {
|
|
dates.append(peerId.id)
|
|
dates.append(date)
|
|
}
|
|
encoder.encodeInt32Array(dates, forKey: "dates")
|
|
|
|
encoder.encodeInt32(self.count, forKey: "count")
|
|
}
|
|
}
|
|
|
|
private final class PeerInvitationImportersContextImpl {
|
|
private let queue: Queue
|
|
private let account: Account
|
|
private let peerId: PeerId
|
|
private let link: String
|
|
private let disposable = MetaDisposable()
|
|
private var isLoadingMore: Bool = false
|
|
private var hasLoadedOnce: Bool = false
|
|
private var canLoadMore: Bool = true
|
|
private var loadedFromCache = false
|
|
private var results: [PeerInvitationImportersState.Importer] = []
|
|
private var count: Int32
|
|
private var populateCache: Bool = true
|
|
|
|
let state = Promise<PeerInvitationImportersState>()
|
|
|
|
init(queue: Queue, account: Account, peerId: PeerId, invite: ExportedInvitation) {
|
|
self.queue = queue
|
|
self.account = account
|
|
self.peerId = peerId
|
|
self.link = invite.link
|
|
|
|
let count = invite.count ?? 0
|
|
self.count = count
|
|
|
|
self.isLoadingMore = true
|
|
self.disposable.set((account.postbox.transaction { transaction -> (peers: [PeerInvitationImportersState.Importer], canLoadMore: Bool)? in
|
|
let cachedResult = transaction.retrieveItemCacheEntry(id: ItemCacheEntryId(collectionId: Namespaces.CachedItemCollection.cachedPeerInvitationImporters, key: CachedPeerInvitationImporters.key(peerId: peerId, link: invite.link))) as? CachedPeerInvitationImporters
|
|
if let cachedResult = cachedResult, Int(cachedResult.count) == count {
|
|
var result: [PeerInvitationImportersState.Importer] = []
|
|
for peerId in cachedResult.peerIds {
|
|
if let peer = transaction.getPeer(peerId), let date = cachedResult.dates[peerId] {
|
|
result.append(PeerInvitationImportersState.Importer(peer: RenderedPeer(peer: peer), date: date))
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
return (result, Int(cachedResult.count) > result.count)
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
|> deliverOn(self.queue)).start(next: { [weak self] cachedPeersAndCanLoadMore in
|
|
guard let strongSelf = self else {
|
|
return
|
|
}
|
|
strongSelf.isLoadingMore = false
|
|
if let (cachedPeers, canLoadMore) = cachedPeersAndCanLoadMore {
|
|
strongSelf.results = cachedPeers
|
|
strongSelf.hasLoadedOnce = true
|
|
strongSelf.canLoadMore = canLoadMore
|
|
strongSelf.loadedFromCache = true
|
|
}
|
|
strongSelf.loadMore()
|
|
}))
|
|
|
|
self.loadMore()
|
|
}
|
|
|
|
deinit {
|
|
self.disposable.dispose()
|
|
}
|
|
|
|
func loadMore() {
|
|
if self.isLoadingMore {
|
|
return
|
|
}
|
|
self.isLoadingMore = true
|
|
let account = self.account
|
|
let peerId = self.peerId
|
|
let link = self.link
|
|
let populateCache = self.populateCache
|
|
|
|
var lastResult = self.results.last
|
|
if self.loadedFromCache {
|
|
self.loadedFromCache = false
|
|
lastResult = nil
|
|
}
|
|
|
|
self.disposable.set((self.account.postbox.transaction { transaction -> Api.InputPeer? in
|
|
return transaction.getPeer(peerId).flatMap(apiInputPeer)
|
|
}
|
|
|> mapToSignal { inputPeer -> Signal<([PeerInvitationImportersState.Importer], Int32), NoError> in
|
|
if let inputPeer = inputPeer {
|
|
let offsetUser = lastResult?.peer.peer.flatMap { apiInputUser($0) } ?? .inputUserEmpty
|
|
let offsetDate = lastResult?.date ?? 0
|
|
let signal = account.network.request(Api.functions.messages.getChatInviteImporters(peer: inputPeer, link: link, offsetDate: offsetDate, offsetUser: offsetUser, limit: lastResult == nil ? 10 : 50))
|
|
|> map(Optional.init)
|
|
|> `catch` { _ -> Signal<Api.messages.ChatInviteImporters?, NoError> in
|
|
return .single(nil)
|
|
}
|
|
|> mapToSignal { result -> Signal<([PeerInvitationImportersState.Importer], Int32), NoError> in
|
|
return account.postbox.transaction { transaction -> ([PeerInvitationImportersState.Importer], Int32) in
|
|
guard let result = result else {
|
|
return ([], 0)
|
|
}
|
|
switch result {
|
|
case let .chatInviteImporters(count, importers, users):
|
|
var peers: [Peer] = []
|
|
for apiUser in users {
|
|
peers.append(TelegramUser(user: apiUser))
|
|
}
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated in
|
|
return updated
|
|
})
|
|
var resultImporters: [PeerInvitationImportersState.Importer] = []
|
|
for importer in importers {
|
|
let peerId: PeerId
|
|
let date: Int32
|
|
switch importer {
|
|
case let .chatInviteImporter(userId, dateValue):
|
|
peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: userId)
|
|
date = dateValue
|
|
}
|
|
if let peer = transaction.getPeer(peerId) {
|
|
resultImporters.append(PeerInvitationImportersState.Importer(peer: RenderedPeer(peer: peer), date: date))
|
|
}
|
|
}
|
|
if populateCache {
|
|
transaction.putItemCacheEntry(id: ItemCacheEntryId(collectionId: Namespaces.CachedItemCollection.cachedPeerInvitationImporters, key: CachedPeerInvitationImporters.key(peerId: peerId, link: link)), entry: CachedPeerInvitationImporters(importers: resultImporters, count: count), collectionSpec: cachedPeerInvitationImportersCollectionSpec)
|
|
}
|
|
return (resultImporters, count)
|
|
}
|
|
}
|
|
}
|
|
return signal
|
|
} else {
|
|
return .single(([], 0))
|
|
}
|
|
}
|
|
|> deliverOn(self.queue)).start(next: { [weak self] importers, updatedCount in
|
|
guard let strongSelf = self else {
|
|
return
|
|
}
|
|
if strongSelf.populateCache {
|
|
strongSelf.populateCache = false
|
|
strongSelf.results.removeAll()
|
|
}
|
|
var existingIds = Set(strongSelf.results.map { $0.peer.peerId })
|
|
for importer in importers {
|
|
if !existingIds.contains(importer.peer.peerId) {
|
|
strongSelf.results.append(importer)
|
|
existingIds.insert(importer.peer.peerId)
|
|
}
|
|
}
|
|
strongSelf.isLoadingMore = false
|
|
strongSelf.hasLoadedOnce = true
|
|
strongSelf.canLoadMore = !importers.isEmpty
|
|
if strongSelf.canLoadMore {
|
|
strongSelf.count = max(updatedCount, Int32(strongSelf.results.count))
|
|
} else {
|
|
strongSelf.count = Int32(strongSelf.results.count)
|
|
}
|
|
strongSelf.updateState()
|
|
}))
|
|
self.updateState()
|
|
}
|
|
|
|
private func updateState() {
|
|
self.state.set(.single(PeerInvitationImportersState(importers: self.results, isLoadingMore: self.isLoadingMore, hasLoadedOnce: self.hasLoadedOnce, canLoadMore: self.canLoadMore, count: self.count)))
|
|
}
|
|
}
|
|
|
|
public final class PeerInvitationImportersContext {
|
|
private let queue: Queue = Queue()
|
|
private let impl: QueueLocalObject<PeerInvitationImportersContextImpl>
|
|
|
|
public var state: Signal<PeerInvitationImportersState, NoError> {
|
|
return Signal { subscriber in
|
|
let disposable = MetaDisposable()
|
|
self.impl.with { impl in
|
|
disposable.set(impl.state.get().start(next: { value in
|
|
subscriber.putNext(value)
|
|
}))
|
|
}
|
|
return disposable
|
|
}
|
|
}
|
|
|
|
public init(account: Account, peerId: PeerId, invite: ExportedInvitation) {
|
|
let queue = self.queue
|
|
self.impl = QueueLocalObject(queue: queue, generate: {
|
|
return PeerInvitationImportersContextImpl(queue: queue, account: account, peerId: peerId, invite: invite)
|
|
})
|
|
}
|
|
|
|
public func loadMore() {
|
|
self.impl.with { impl in
|
|
impl.loadMore()
|
|
}
|
|
}
|
|
}
|
|
|
|
public struct ExportedInvitationCreator : Equatable {
|
|
public let peer: RenderedPeer
|
|
public let count: Int32
|
|
public let revokedCount: Int32
|
|
}
|
|
|
|
public func peerExportedInvitationsCreators(account: Account, peerId: PeerId) -> Signal<[ExportedInvitationCreator], NoError> {
|
|
return account.postbox.transaction { transaction -> Signal<[ExportedInvitationCreator], NoError> in
|
|
if let peer = transaction.getPeer(peerId), let inputPeer = apiInputPeer(peer) {
|
|
var isCreator = false
|
|
if let peer = peer as? TelegramGroup, case .creator = peer.role {
|
|
isCreator = true
|
|
} else if let peer = peer as? TelegramChannel, peer.flags.contains(.isCreator) {
|
|
isCreator = true
|
|
}
|
|
if !isCreator {
|
|
return .single([])
|
|
} else {
|
|
return account.network.request(Api.functions.messages.getAdminsWithInvites(peer: inputPeer))
|
|
|> map(Optional.init)
|
|
|> `catch` { _ -> Signal<Api.messages.ChatAdminsWithInvites?, NoError> in
|
|
return .single(nil)
|
|
}
|
|
|> mapToSignal { result -> Signal<[ExportedInvitationCreator], NoError> in
|
|
return account.postbox.transaction { transaction -> [ExportedInvitationCreator] in
|
|
if let result = result, case let .chatAdminsWithInvites(admins, users) = result {
|
|
var creators: [ExportedInvitationCreator] = []
|
|
var peers: [Peer] = []
|
|
var peersMap: [PeerId: Peer] = [:]
|
|
for user in users {
|
|
let telegramUser = TelegramUser(user: user)
|
|
peers.append(telegramUser)
|
|
peersMap[telegramUser.id] = telegramUser
|
|
}
|
|
|
|
for case let .chatAdminWithInvites(adminId, invitesCount, revokedInvitesCount) in admins {
|
|
let peerId = PeerId(namespace: Namespaces.Peer.CloudUser, id: adminId)
|
|
if let peer = peersMap[peerId], peerId != account.peerId {
|
|
creators.append(ExportedInvitationCreator(peer: RenderedPeer(peer: peer), count: invitesCount, revokedCount: revokedInvitesCount))
|
|
}
|
|
}
|
|
|
|
updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in
|
|
return updated
|
|
})
|
|
|
|
return creators
|
|
} else {
|
|
return []
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
return .single([])
|
|
}
|
|
} |> switchToLatest
|
|
}
|
|
|