import Foundation import SwiftSignalKit import Postbox import TelegramApi //communities.exportCommunityInvite#41fe69d9 community:InputCommunity title:string peers:Vector = communities.ExportedCommunityInvite; //communities.exportedCommunityInvite#6b97a8ea filter:DialogFilter invite:ExportedCommunityInvite = communities.ExportedCommunityInvite; //exportedCommunityInvite#af7afb2f title:string url:string peers:Vector = ExportedCommunityInvite; public enum ExportChatFolderError { case generic } public struct ExportedChatFolderLink: Equatable { public var title: String public var link: String public var peerIds: [EnginePeer.Id] public var isRevoked: Bool public init( title: String, link: String, peerIds: [EnginePeer.Id], isRevoked: Bool ) { self.title = title self.link = link self.peerIds = peerIds self.isRevoked = isRevoked } } public extension ExportedChatFolderLink { var slug: String { var slug = self.link if slug.hasPrefix("https://t.me/folder/") { slug = String(slug[slug.index(slug.startIndex, offsetBy: "https://t.me/folder/".count)...]) } return slug } } func _internal_exportChatFolder(account: Account, filterId: Int32, title: String, peerIds: [PeerId]) -> Signal { return account.postbox.transaction { transaction -> [Api.InputPeer] in return peerIds.compactMap(transaction.getPeer).compactMap(apiInputPeer) } |> castError(ExportChatFolderError.self) |> mapToSignal { inputPeers -> Signal in return account.network.request(Api.functions.communities.exportCommunityInvite(community: .inputCommunityDialogFilter(filterId: filterId), title: title, peers: inputPeers)) |> mapError { _ -> ExportChatFolderError in return .generic } |> mapToSignal { result -> Signal in return account.postbox.transaction { transaction -> Signal in switch result { case let .exportedCommunityInvite(filter, invite): let parsedFilter = ChatListFilter(apiFilter: filter) let _ = updateChatListFiltersState(transaction: transaction, { state in var state = state if let index = state.filters.firstIndex(where: { $0.id == filterId }) { state.filters[index] = parsedFilter } else { state.filters.append(parsedFilter) } state.remoteFilters = state.filters return state }) switch invite { case let .exportedCommunityInvite(flags, title, url, peers): return .single(ExportedChatFolderLink( title: title, link: url, peerIds: peers.map(\.peerId), isRevoked: (flags & (1 << 0)) != 0 )) } } } |> castError(ExportChatFolderError.self) |> switchToLatest } } } func _internal_getExportedChatFolderLinks(account: Account, id: Int32) -> Signal<[ExportedChatFolderLink], NoError> { return account.network.request(Api.functions.communities.getExportedInvites(community: .inputCommunityDialogFilter(filterId: id))) |> map(Optional.init) |> `catch` { _ -> Signal in return .single(nil) } |> mapToSignal { result -> Signal<[ExportedChatFolderLink], NoError> in guard let result = result else { return .single([]) } return account.postbox.transaction { transaction -> [ExportedChatFolderLink] in switch result { case let .exportedInvites(invites, chats, users): var peers: [Peer] = [] var peerPresences: [PeerId: Api.User] = [:] for user in users { let telegramUser = TelegramUser(user: user) peers.append(telegramUser) peerPresences[telegramUser.id] = user } for chat in chats { if let peer = parseTelegramGroupOrChannel(chat: chat) { peers.append(peer) } } updatePeers(transaction: transaction, peers: peers, update: { _, updated -> Peer in return updated }) updatePeerPresences(transaction: transaction, accountPeerId: account.peerId, peerPresences: peerPresences) var result: [ExportedChatFolderLink] = [] for invite in invites { switch invite { case let .exportedCommunityInvite(flags, title, url, peers): result.append(ExportedChatFolderLink( title: title, link: url, peerIds: peers.map(\.peerId), isRevoked: (flags & (1 << 0)) != 0 )) } } return result } } } } public enum EditChatFolderLinkError { case generic } func _internal_editChatFolderLink(account: Account, filterId: Int32, link: ExportedChatFolderLink, title: String?, revoke: Bool) -> Signal { var flags: Int32 = 0 if revoke { flags |= 1 << 0 } if title != nil { flags |= 1 << 1 } return account.network.request(Api.functions.communities.editExportedInvite(flags: flags, community: .inputCommunityDialogFilter(filterId: filterId), slug: link.slug, title: title)) |> mapError { _ -> EditChatFolderLinkError in return .generic } |> ignoreValues } public enum RevokeChatFolderLinkError { case generic } func _internal_revokeChatFolderLink(account: Account, filterId: Int32, link: ExportedChatFolderLink) -> Signal { return account.network.request(Api.functions.communities.deleteExportedInvite(community: .inputCommunityDialogFilter(filterId: filterId), slug: link.slug)) |> mapError { _ -> RevokeChatFolderLinkError in return .generic } |> ignoreValues } public enum CheckChatFolderLinkError { case generic } public final class ChatFolderLinkContents { public let localFilterId: Int32? public let title: String? public let peers: [EnginePeer] public let alreadyMemberPeerIds: Set public init( localFilterId: Int32?, title: String?, peers: [EnginePeer], alreadyMemberPeerIds: Set ) { self.localFilterId = localFilterId self.title = title self.peers = peers self.alreadyMemberPeerIds = alreadyMemberPeerIds } } func _internal_checkChatFolderLink(account: Account, slug: String) -> Signal { return account.network.request(Api.functions.communities.checkCommunityInvite(slug: slug)) |> mapError { _ -> CheckChatFolderLinkError in return .generic } |> mapToSignal { result -> Signal in return account.postbox.transaction { transaction -> ChatFolderLinkContents in switch result { case let .communityInvite(title, peers, chats, users): var allPeers: [Peer] = [] var peerPresences: [PeerId: Api.User] = [:] for user in users { let telegramUser = TelegramUser(user: user) allPeers.append(telegramUser) peerPresences[telegramUser.id] = user } for chat in chats { if let peer = parseTelegramGroupOrChannel(chat: chat) { allPeers.append(peer) } } updatePeers(transaction: transaction, peers: allPeers, update: { _, updated -> Peer in return updated }) updatePeerPresences(transaction: transaction, accountPeerId: account.peerId, peerPresences: peerPresences) var resultPeers: [EnginePeer] = [] var alreadyMemberPeerIds = Set() for peer in peers { if let peerValue = transaction.getPeer(peer.peerId) { resultPeers.append(EnginePeer(peerValue)) if transaction.getPeerChatListIndex(peer.peerId) != nil { alreadyMemberPeerIds.insert(peer.peerId) } } } return ChatFolderLinkContents(localFilterId: nil, title: title, peers: resultPeers, alreadyMemberPeerIds: alreadyMemberPeerIds) case let .communityInviteAlready(filterId, missingPeers, chats, users): var allPeers: [Peer] = [] var peerPresences: [PeerId: Api.User] = [:] for user in users { let telegramUser = TelegramUser(user: user) allPeers.append(telegramUser) peerPresences[telegramUser.id] = user } for chat in chats { if let peer = parseTelegramGroupOrChannel(chat: chat) { allPeers.append(peer) } } updatePeers(transaction: transaction, peers: allPeers, update: { _, updated -> Peer in return updated }) updatePeerPresences(transaction: transaction, accountPeerId: account.peerId, peerPresences: peerPresences) let currentFilters = _internal_currentChatListFilters(transaction: transaction) var currentFilterTitle: String? if let index = currentFilters.firstIndex(where: { $0.id == filterId }) { switch currentFilters[index] { case let .filter(_, title, _, _): currentFilterTitle = title default: break } } var resultPeers: [EnginePeer] = [] var alreadyMemberPeerIds = Set() for peer in missingPeers { if let peerValue = transaction.getPeer(peer.peerId) { resultPeers.append(EnginePeer(peerValue)) if transaction.getPeerChatListIndex(peer.peerId) != nil { alreadyMemberPeerIds.insert(peer.peerId) } } } return ChatFolderLinkContents(localFilterId: filterId, title: currentFilterTitle, peers: resultPeers, alreadyMemberPeerIds: alreadyMemberPeerIds) } } |> castError(CheckChatFolderLinkError.self) } } public enum JoinChatFolderLinkError { case generic } func _internal_joinChatFolderLink(account: Account, slug: String, peerIds: [EnginePeer.Id]) -> Signal { return account.postbox.transaction { transaction -> [Api.InputPeer] in return peerIds.compactMap(transaction.getPeer).compactMap(apiInputPeer) } |> castError(JoinChatFolderLinkError.self) |> mapToSignal { inputPeers -> Signal in return account.network.request(Api.functions.communities.joinCommunityInvite(slug: slug, peers: inputPeers)) |> mapError { _ -> JoinChatFolderLinkError in return .generic } |> mapToSignal { result -> Signal in account.stateManager.addUpdates(result) return .complete() } } }