2026-03-31 11:28:30 +02:00

2042 lines
54 KiB
TypeScript

import { Api as GramJs } from '../../../lib/gramjs';
import { PasswordFreshError, RPCError, SessionFreshError } from '../../../lib/gramjs/errors';
import type { ChatListType, ThreadReadState } from '../../../types';
import {
type ApiChat,
type ApiChatAdminRights,
type ApiChatBannedRights,
type ApiChatFolder,
type ApiChatFullInfo,
type ApiChatReactions,
type ApiDraft,
type ApiGroupCall,
type ApiMessage,
type ApiMissingInvitedUser,
type ApiPeer,
type ApiPeerNotifySettings,
type ApiPhoto,
type ApiProfileTab,
type ApiThreadInfo,
type ApiUser,
type ApiUserStatus,
MAIN_THREAD_ID,
} from '../../types';
import {
ALL_FOLDER_ID,
ARCHIVED_FOLDER_ID,
DEBUG,
MEMBERS_LOAD_SLICE,
SERVICE_NOTIFICATIONS_USER_ID,
} from '../../../config';
import { buildCollectionByKey, omitUndefined } from '../../../util/iteratees';
import { GLOBAL_SEARCH_CONTACTS_LIMIT } from '../../../limits';
import {
buildApiChatBotCommands,
buildApiChatFolder,
buildApiChatFolderFromSuggested,
buildApiChatFromDialog,
buildApiChatFromPreview,
buildApiChatFromSavedDialog,
buildApiChatInviteInfo,
buildApiChatlistExportedInvite,
buildApiChatlistInvite,
buildApiChatReactions,
buildApiMissingInvitedUser,
buildApiSponsoredPeer,
buildApiThreadInfoFromDialog,
buildChatMember,
buildChatMembers,
buildThreadReadState,
getPeerKey,
} from '../apiBuilders/chats';
import { buildApiPhoto } from '../apiBuilders/common';
import { buildApiMessage, buildMessageDraft } from '../apiBuilders/messages';
import {
buildApiBotVerification,
buildApiPeerId,
buildApiPeerNotifySettings,
buildApiProfileTab,
getApiChatIdFromMtpPeer,
} from '../apiBuilders/peers';
import { buildStickerSet } from '../apiBuilders/symbols';
import { buildApiPeerSettings, buildApiUser, buildApiUserStatuses } from '../apiBuilders/users';
import {
buildChatAdminRights,
buildChatBannedRights,
buildFilterFromApiFolder,
buildInputChannel,
buildInputChat,
buildInputChatReactions,
buildInputPeer,
buildInputPhoto,
buildInputProfileTab,
buildInputReplyTo,
buildInputSuggestedPost,
buildInputUser,
buildMtpMessageEntity,
DEFAULT_PRIMITIVES,
getEntityTypeById,
} from '../gramjsBuilders';
import {
addPhotoToLocalDb,
} from '../helpers/localDb';
import { checkErrorType, isChatFolder, wrapError } from '../helpers/misc';
import { scheduleMutedChatUpdate } from '../scheduleUnmute';
import { sendApiUpdate } from '../updates/apiUpdateEmitter';
import {
applyState, updateChannelState,
} from '../updates/updateManager';
import { handleGramJsUpdate, invokeRequest, uploadFile } from './client';
import { getPassword } from './twoFaSettings';
type FullChatData = {
fullInfo: ApiChatFullInfo;
chats: ApiChat[];
userStatusesById: Record<string, ApiUserStatus>;
groupCall?: Partial<ApiGroupCall>;
membersCount?: number;
isForumAsMessages?: true;
};
type ChatListData = {
chatIds: string[];
chats: ApiChat[];
users: ApiUser[];
userStatusesById: Record<string, ApiUserStatus>;
draftsById: Record<string, ApiDraft>;
threadReadStatesById?: Record<string, ThreadReadState>;
threadInfos: ApiThreadInfo[];
orderedPinnedIds: string[] | undefined;
totalChatCount: number;
messages: ApiMessage[];
notifyExceptionById: Record<string, ApiPeerNotifySettings>;
lastMessageByChatId: Record<string, number>;
nextOffsetId?: number;
nextOffsetPeerId?: string;
nextOffsetDate?: number;
};
export async function fetchChats({
limit,
offsetDate,
offsetPeer,
offsetId,
archived,
withPinned,
lastLocalServiceMessageId,
}: {
limit: number;
offsetDate?: number;
offsetPeer?: ApiPeer;
offsetId?: number;
archived?: boolean;
withPinned?: boolean;
lastLocalServiceMessageId?: number;
}): Promise<ChatListData | undefined> {
const peer = (offsetPeer && buildInputPeer(offsetPeer.id, offsetPeer.accessHash)) || new GramJs.InputPeerEmpty();
const result = await invokeRequest(new GramJs.messages.GetDialogs({
hash: DEFAULT_PRIMITIVES.BIGINT,
offsetPeer: peer,
offsetId: offsetId ?? DEFAULT_PRIMITIVES.INT,
limit,
offsetDate: offsetDate ?? DEFAULT_PRIMITIVES.INT,
folderId: archived ? ARCHIVED_FOLDER_ID : undefined,
...(withPinned && { excludePinned: true }),
}));
const resultPinned = withPinned
? await invokeRequest(new GramJs.messages.GetPinnedDialogs({
folderId: archived ? ARCHIVED_FOLDER_ID : ALL_FOLDER_ID,
}))
: undefined;
if (!result || result instanceof GramJs.messages.DialogsNotModified) {
return undefined;
}
const messages = (resultPinned ? resultPinned.messages : [])
.concat(result.messages)
.map(buildApiMessage)
.filter(Boolean);
const peersByKey = preparePeers(result);
if (resultPinned) {
Object.assign(peersByKey, preparePeers(resultPinned, peersByKey));
}
const chats: ApiChat[] = [];
const draftsById: Record<string, ApiDraft> = {};
const notifyExceptionById: Record<string, ApiPeerNotifySettings> = {};
const threadReadStatesById: Record<string, ThreadReadState> = {};
const threadInfos: ApiThreadInfo[] = [];
const dialogs = (resultPinned?.dialogs || []).concat(result.dialogs);
const orderedPinnedIds: string[] = [];
const lastMessageByChatId: Record<string, number> = {};
dialogs.forEach((dialog) => {
if (
!(dialog instanceof GramJs.Dialog)
// This request can return dialogs not belonging to specified folder
|| (!archived && dialog.folderId === ARCHIVED_FOLDER_ID)
|| (archived && dialog.folderId !== ARCHIVED_FOLDER_ID)
) {
return;
}
const peerEntity = peersByKey[getPeerKey(dialog.peer)];
const chat = buildApiChatFromDialog(dialog, peerEntity);
lastMessageByChatId[chat.id] = dialog.topMessage;
if (dialog.pts) {
updateChannelState(chat.id, dialog.pts);
}
if (
chat.id === SERVICE_NOTIFICATIONS_USER_ID
&& lastLocalServiceMessageId
&& (lastLocalServiceMessageId > dialog.topMessage)
) {
lastMessageByChatId[chat.id] = lastLocalServiceMessageId;
}
chat.isListed = true;
chats.push(chat);
const notifySettings = buildApiPeerNotifySettings(dialog.notifySettings);
if (Object.values(omitUndefined(notifySettings)).length) {
notifyExceptionById[chat.id] = notifySettings;
if (notifySettings.mutedUntil) {
scheduleMutedChatUpdate(chat.id, notifySettings.mutedUntil, sendApiUpdate);
}
}
if (withPinned && dialog.pinned) {
orderedPinnedIds.push(chat.id);
}
if (dialog.draft) {
const draft = buildMessageDraft(dialog.draft);
if (draft) {
draftsById[chat.id] = draft;
}
}
const readState = buildThreadReadState(dialog);
threadReadStatesById[chat.id] = readState;
const threadInfo = buildApiThreadInfoFromDialog(chat.id, dialog);
threadInfos.push(threadInfo);
});
const chatIds = chats.map((chat) => chat.id);
const users = result.users.map(buildApiUser).filter(Boolean);
const userStatusesById = buildApiUserStatuses((resultPinned?.users || []).concat(result.users));
let totalChatCount: number;
if (result instanceof GramJs.messages.DialogsSlice) {
totalChatCount = result.count;
} else {
totalChatCount = chatIds.length;
}
const lastDialog = chats[chats.length - 1];
const lastMessageId = lastMessageByChatId[lastDialog?.id];
const nextOffsetId = lastMessageId;
const nextOffsetPeerId = lastDialog?.id;
const nextOffsetDate = messages.reverse()
.find((message) => message.chatId === lastDialog?.id && message.id === lastMessageId)?.date;
return {
chatIds,
chats,
users,
userStatusesById,
draftsById,
orderedPinnedIds: withPinned ? orderedPinnedIds : undefined,
totalChatCount,
lastMessageByChatId,
messages,
notifyExceptionById,
nextOffsetId,
nextOffsetPeerId,
nextOffsetDate,
threadReadStatesById,
threadInfos,
};
}
export async function fetchSavedChats({
parentPeer,
limit,
offsetDate,
offsetPeer,
offsetId,
withPinned,
}: {
parentPeer: ApiPeer;
limit: number;
offsetDate?: number;
offsetPeer?: ApiPeer;
offsetId?: number;
withPinned?: boolean;
}): Promise<ChatListData | undefined> {
const isMonoforum = 'title' in parentPeer;
const inputParentPeer = isMonoforum ? buildInputPeer(parentPeer.id, parentPeer.accessHash) : undefined;
const peer = (offsetPeer && buildInputPeer(offsetPeer.id, offsetPeer.accessHash)) || new GramJs.InputPeerEmpty();
const result = await invokeRequest(new GramJs.messages.GetSavedDialogs({
offsetPeer: peer,
offsetId: offsetId ?? DEFAULT_PRIMITIVES.INT,
parentPeer: inputParentPeer,
limit,
offsetDate: offsetDate ?? DEFAULT_PRIMITIVES.INT,
hash: DEFAULT_PRIMITIVES.BIGINT,
...(withPinned && { excludePinned: true }),
}));
const resultPinned = withPinned
? await invokeRequest(new GramJs.messages.GetPinnedSavedDialogs())
: undefined;
if (!result || result instanceof GramJs.messages.SavedDialogsNotModified) {
return undefined;
}
const hasPinned = resultPinned && !(resultPinned instanceof GramJs.messages.SavedDialogsNotModified);
const messages = (hasPinned ? resultPinned.messages : [])
.concat(result.messages)
.map(buildApiMessage)
.filter(Boolean);
const peersByKey = preparePeers(result);
if (hasPinned) {
Object.assign(peersByKey, preparePeers(resultPinned, peersByKey));
}
const dialogs = (hasPinned ? resultPinned.dialogs : []).concat(result.dialogs);
const chatIds: string[] = [];
const orderedPinnedIds: string[] = [];
const lastMessageByChatId: Record<string, number> = {};
const threadInfos: ApiThreadInfo[] = [];
const chats: ApiChat[] = [];
dialogs.forEach((dialog) => {
if (dialog instanceof GramJs.MonoForumDialog) {
return;
}
const peerEntity = peersByKey[getPeerKey(dialog.peer)];
const chat = buildApiChatFromSavedDialog(dialog, peerEntity);
const chatId = getApiChatIdFromMtpPeer(dialog.peer);
chatIds.push(chatId);
if (withPinned && dialog.pinned) {
orderedPinnedIds.push(chatId);
}
lastMessageByChatId[chatId] = dialog.topMessage;
chats.push(chat);
const threadInfo = buildApiThreadInfoFromDialog(parentPeer.id, dialog);
threadInfos.push(threadInfo);
});
const users = result.users.map(buildApiUser).filter(Boolean);
const userStatusesById = buildApiUserStatuses((hasPinned ? resultPinned.users : [])
.concat(result.users));
let totalChatCount: number;
if (result instanceof GramJs.messages.SavedDialogsSlice) {
totalChatCount = result.count;
} else {
totalChatCount = chatIds.length;
}
const lastDialog = chats[chats.length - 1];
const lastMessageId = lastMessageByChatId[lastDialog?.id];
const nextOffsetId = lastMessageId;
const nextOffsetPeerId = lastDialog?.id;
const nextOffsetDate = messages.reverse()
.find((message) => message.chatId === lastDialog?.id && message.id === lastMessageId)?.date;
return {
chatIds,
chats,
users,
userStatusesById,
orderedPinnedIds: withPinned ? orderedPinnedIds : undefined,
totalChatCount,
lastMessageByChatId,
messages,
draftsById: {},
notifyExceptionById: {},
nextOffsetId,
nextOffsetPeerId,
nextOffsetDate,
threadInfos,
};
}
const fullChatRequestDedupe = new Map<string, Promise<FullChatData | undefined>>();
export async function fetchFullChat(chat: ApiChat) {
const { id } = chat;
if (fullChatRequestDedupe.has(id)) {
return fullChatRequestDedupe.get(id);
}
const type = getEntityTypeById(chat.id);
const promise = type === 'channel'
? getFullChannelInfo(chat)
: getFullChatInfo(id);
fullChatRequestDedupe.set(id, promise);
promise.finally(() => {
fullChatRequestDedupe.delete(id);
});
return promise;
}
export async function fetchPeerSettings(peer: ApiPeer) {
const { id, accessHash } = peer;
const result = await invokeRequest(new GramJs.messages.GetPeerSettings({
peer: buildInputPeer(id, accessHash),
}), {
abortControllerChatId: id,
});
if (!result) {
return undefined;
}
return {
settings: buildApiPeerSettings(result.settings),
};
}
export async function searchChats({ query }: { query: string }) {
const result = await invokeRequest(new GramJs.contacts.Search({ q: query, limit: GLOBAL_SEARCH_CONTACTS_LIMIT }));
if (!result) {
return undefined;
}
const accountPeerIds = result.myResults.map(getApiChatIdFromMtpPeer);
const globalPeerIds = result.results.map(getApiChatIdFromMtpPeer)
.filter((id) => !accountPeerIds.includes(id));
return {
accountResultIds: accountPeerIds,
globalResultIds: globalPeerIds,
};
}
export async function fetchChat({
type, user,
}: {
type: 'user' | 'self' | 'support'; user?: ApiUser;
}) {
let mtpUser: GramJs.TypeUser | undefined;
if (type === 'self' || type === 'user') {
const result = await invokeRequest(new GramJs.users.GetUsers({
id: [
type === 'user' && user
? buildInputUser(user.id, user.accessHash)
: new GramJs.InputUserSelf(),
],
}));
if (!result || !result.length) {
return undefined;
}
[mtpUser] = result;
} else if (type === 'support') {
const result = await invokeRequest(new GramJs.help.GetSupport());
if (!result || !result.user) {
return undefined;
}
mtpUser = result.user;
}
const chat = buildApiChatFromPreview(mtpUser!, type === 'support');
if (!chat) {
return undefined;
}
sendApiUpdate({
'@type': 'updateChat',
id: chat.id,
chat,
});
return { chatId: chat.id };
}
export async function requestChatUpdate({
chat,
lastLocalMessage,
noLastMessage,
}: {
chat: ApiChat; lastLocalMessage?: ApiMessage; noLastMessage?: boolean;
}) {
const { id, accessHash } = chat;
const result = await invokeRequest(new GramJs.messages.GetPeerDialogs({
peers: [new GramJs.InputDialogPeer({
peer: buildInputPeer(id, accessHash),
})],
}));
if (!result) {
return;
}
const dialog = result.dialogs[0];
if (!dialog || !(dialog instanceof GramJs.Dialog)) {
return;
}
const peersByKey = preparePeers(result);
const peerEntity = peersByKey[getPeerKey(dialog.peer)];
if (!peerEntity) {
return;
}
const lastRemoteMessage = buildApiMessage(result.messages[0]);
const lastMessage = lastLocalMessage && (!lastRemoteMessage || (lastLocalMessage.date > lastRemoteMessage.date))
? lastLocalMessage
: lastRemoteMessage;
const chatUpdate = buildApiChatFromDialog(dialog, peerEntity);
const readState = buildThreadReadState(dialog);
const threadInfo = buildApiThreadInfoFromDialog(chat.id, dialog);
sendApiUpdate({
'@type': 'updateThreadReadState',
chatId: id,
threadId: MAIN_THREAD_ID,
readState,
});
sendApiUpdate({
'@type': 'updateThreadInfo',
threadInfo,
});
sendApiUpdate({
'@type': 'updateChat',
id,
chat: chatUpdate,
});
if (!noLastMessage && lastMessage) {
sendApiUpdate({
'@type': 'updateChatLastMessage',
id,
lastMessage,
});
}
applyState(result.state);
const notifySettings = buildApiPeerNotifySettings(dialog.notifySettings);
if (notifySettings.mutedUntil) {
scheduleMutedChatUpdate(chatUpdate.id, notifySettings.mutedUntil, sendApiUpdate);
}
}
export function saveDraft({
chat,
draft,
}: {
chat: ApiChat;
draft?: ApiDraft;
}) {
return invokeRequest(new GramJs.messages.SaveDraft({
peer: buildInputPeer(chat.id, chat.accessHash),
message: draft?.text?.text || DEFAULT_PRIMITIVES.STRING,
entities: draft?.text?.entities?.map(buildMtpMessageEntity),
replyTo: draft?.replyInfo && buildInputReplyTo(draft.replyInfo),
suggestedPost: draft?.suggestedPostInfo && buildInputSuggestedPost(draft.suggestedPostInfo),
}));
}
async function getFullChatInfo(chatId: string): Promise<FullChatData | undefined> {
const result = await invokeRequest(new GramJs.messages.GetFullChat({
chatId: buildInputChat(chatId),
}));
if (!result || !(result.fullChat instanceof GramJs.ChatFull)) {
return undefined;
}
const {
about,
participants,
exportedInvite,
botInfo,
call,
availableReactions,
recentRequesters,
requestsPending,
chatPhoto,
translationsDisabled,
reactionsLimit,
hasScheduled,
} = result.fullChat;
if (chatPhoto) {
addPhotoToLocalDb(chatPhoto);
}
const members = buildChatMembers(participants);
const adminMembers = members ? members.filter(({ isAdmin, isOwner }) => isAdmin || isOwner) : undefined;
const botCommands = botInfo ? buildApiChatBotCommands(botInfo) : undefined;
const inviteLink = exportedInvite instanceof GramJs.ChatInviteExported ? exportedInvite.link : undefined;
const userStatusesById = buildApiUserStatuses(result.users);
const chats = result.chats.map((chat) => buildApiChatFromPreview(chat)).filter(Boolean);
const groupCall = call instanceof GramJs.InputGroupCall ? call : undefined;
return {
fullInfo: {
...(chatPhoto instanceof GramJs.Photo && { profilePhoto: buildApiPhoto(chatPhoto) }),
about,
members,
adminMembersById: adminMembers ? buildCollectionByKey(adminMembers, 'userId') : undefined,
canViewMembers: true,
botCommands,
inviteLink,
groupCallId: groupCall?.id.toString(),
enabledReactions: buildApiChatReactions(availableReactions),
reactionsLimit,
requestsPending,
recentRequesterIds: recentRequesters?.map((userId) => buildApiPeerId(userId, 'user')),
isTranslationDisabled: translationsDisabled,
isPreHistoryHidden: true,
hasScheduledMessages: hasScheduled,
},
chats,
userStatusesById,
groupCall: groupCall ? {
chatId,
isLoaded: false,
id: groupCall.id.toString(),
accessHash: groupCall.accessHash.toString(),
connectionState: 'disconnected',
participantsCount: 0,
version: 0,
participants: {},
} : undefined,
membersCount: members?.length,
};
}
async function getFullChannelInfo(
chat: ApiChat,
): Promise<FullChatData | undefined> {
const { id, adminRights, isMonoforum } = chat;
const accessHash = chat.accessHash!;
const result = await invokeRequest(new GramJs.channels.GetFullChannel({
channel: buildInputChannel(id, accessHash),
}));
if (!result || !(result.fullChat instanceof GramJs.ChannelFull)) {
return undefined;
}
const {
about,
onlineCount,
exportedInvite,
slowmodeSeconds,
slowmodeNextSendDate,
migratedFromChatId,
migratedFromMaxId,
canViewParticipants,
canViewStats,
linkedChatId,
hiddenPrehistory,
call,
botInfo,
availableReactions,
reactionsLimit,
defaultSendAs,
requestsPending,
recentRequesters,
statsDc,
participantsCount,
stickerset,
chatPhoto,
participantsHidden,
translationsDisabled,
storiesPinnedAvailable,
viewForumAsMessages,
emojiset,
boostsApplied,
boostsUnrestrict,
botVerification,
canViewRevenue: canViewMonetization,
paidReactionsAvailable,
hasScheduled,
stargiftsCount,
stargiftsAvailable,
paidMessagesAvailable,
mainTab,
} = result.fullChat;
if (chatPhoto) {
addPhotoToLocalDb(chatPhoto);
}
const inviteLink = exportedInvite instanceof GramJs.ChatInviteExported
? exportedInvite.link
: undefined;
const canLoadParticipants = canViewParticipants && !isMonoforum;
const { members, userStatusesById } = (canLoadParticipants && await fetchMembers({ chat })) || {};
const { members: kickedMembers, userStatusesById: bannedStatusesById } = (
canLoadParticipants && adminRights && await fetchMembers({ chat, memberFilter: 'kicked' })
) || {};
const { members: adminMembers, userStatusesById: adminStatusesById } = (
canLoadParticipants && await fetchMembers({ chat, memberFilter: 'admin' })
) || {};
const botCommands = botInfo ? buildApiChatBotCommands(botInfo) : undefined;
const memberInfoRequest = !chat.isNotJoined && chat.type === 'chatTypeChannel'
? await fetchMember({ chat }) : undefined;
const memberInfo = memberInfoRequest?.member;
const joinInfo = memberInfo?.joinedDate ? {
joinedDate: memberInfo.joinedDate,
inviterId: memberInfo.inviterId,
isViaRequest: memberInfo.isViaRequest,
} : undefined;
const chats = result.chats.map((c) => buildApiChatFromPreview(c)).filter(Boolean);
if (result?.chats?.length > 1) {
const [, mtpLinkedChat] = result.chats;
const linkedChat = buildApiChatFromPreview(mtpLinkedChat);
if (linkedChat) {
sendApiUpdate({
'@type': 'updateChat',
id: linkedChat.id,
chat: linkedChat,
});
}
}
if (result.fullChat.pts) {
updateChannelState(chat.id, result.fullChat.pts);
}
const statusesById = {
...userStatusesById,
...bannedStatusesById,
...adminStatusesById,
};
const groupCall = call instanceof GramJs.InputGroupCall ? call : undefined;
return {
fullInfo: {
...(chatPhoto instanceof GramJs.Photo && { profilePhoto: buildApiPhoto(chatPhoto) }),
about,
onlineCount,
inviteLink,
slowMode: slowmodeSeconds ? {
seconds: slowmodeSeconds,
nextSendDate: slowmodeNextSendDate,
} : undefined,
migratedFrom: migratedFromChatId ? {
chatId: buildApiPeerId(migratedFromChatId, 'chat'),
maxMessageId: migratedFromMaxId,
} : undefined,
canViewMembers: canLoadParticipants,
canViewStatistics: canViewStats,
canViewMonetization,
isPreHistoryHidden: hiddenPrehistory,
joinInfo,
members,
kickedMembers,
adminMembersById: adminMembers ? buildCollectionByKey(adminMembers, 'userId') : undefined,
groupCallId: groupCall ? String(groupCall.id) : undefined,
linkedChatId: linkedChatId ? buildApiPeerId(linkedChatId, 'channel') : undefined,
botCommands,
enabledReactions: buildApiChatReactions(availableReactions),
reactionsLimit,
sendAsId: defaultSendAs ? getApiChatIdFromMtpPeer(defaultSendAs) : undefined,
requestsPending,
recentRequesterIds: recentRequesters?.map((userId) => buildApiPeerId(userId, 'user')),
statisticsDcId: statsDc,
stickerSet: stickerset ? buildStickerSet(stickerset) : undefined,
emojiSet: emojiset ? buildStickerSet(emojiset) : undefined,
areParticipantsHidden: participantsHidden,
isTranslationDisabled: translationsDisabled,
hasPinnedStories: Boolean(storiesPinnedAvailable),
boostsApplied,
boostsToUnrestrict: boostsUnrestrict,
botVerification: botVerification && buildApiBotVerification(botVerification),
isPaidReactionAvailable: paidReactionsAvailable,
hasScheduledMessages: hasScheduled,
starGiftCount: stargiftsCount,
areStarGiftsAvailable: Boolean(stargiftsAvailable),
arePaidMessagesAvailable: paidMessagesAvailable,
mainTab: mainTab && buildApiProfileTab(mainTab),
},
chats,
userStatusesById: statusesById,
groupCall: groupCall ? {
chatId: id,
isLoaded: false,
id: groupCall.id.toString(),
accessHash: groupCall?.accessHash.toString(),
participants: {},
version: 0,
participantsCount: 0,
connectionState: 'disconnected',
} : undefined,
membersCount: participantsCount,
...(viewForumAsMessages && { isForumAsMessages: true }),
};
}
export function updateChatNotifySettings({
chat, settings,
}: {
chat: ApiChat; settings: Partial<ApiPeerNotifySettings>;
}) {
invokeRequest(new GramJs.account.UpdateNotifySettings({
peer: new GramJs.InputNotifyPeer({
peer: buildInputPeer(chat.id, chat.accessHash),
}),
settings: new GramJs.InputPeerNotifySettings({
muteUntil: settings.mutedUntil,
showPreviews: settings.shouldShowPreviews,
silent: settings.isSilentPosting,
}),
}));
sendApiUpdate({
'@type': 'updateChatNotifySettings',
chatId: chat.id,
settings,
});
void requestChatUpdate({
chat,
noLastMessage: true,
});
}
export function updateTopicMutedState({
chat, topicId, mutedUntil,
}: {
chat: ApiChat; topicId: number; mutedUntil: number;
}) {
invokeRequest(new GramJs.account.UpdateNotifySettings({
peer: new GramJs.InputNotifyForumTopic({
peer: buildInputPeer(chat.id, chat.accessHash),
topMsgId: topicId,
}),
settings: new GramJs.InputPeerNotifySettings({ muteUntil: mutedUntil }),
}));
sendApiUpdate({
'@type': 'updateTopicNotifySettings',
chatId: chat.id,
topicId,
settings: {
mutedUntil,
},
});
// TODO[forums] Request forum topic thread update
}
export async function createChannel({
title, about = DEFAULT_PRIMITIVES.STRING, users, isBroadcast, isMegagroup,
}: {
title: string; about?: string; users?: ApiUser[]; isBroadcast?: true; isMegagroup?: true;
}) {
const result = await invokeRequest(new GramJs.channels.CreateChannel({
broadcast: isBroadcast,
title,
about,
megagroup: isMegagroup,
}), {
shouldThrow: true,
});
// `createChannel` can return a lot of different update types according to docs,
// but currently channel creation returns only `Updates` type.
// Errors are added to catch unexpected cases in future testing
if (!(result instanceof GramJs.Updates)) {
if (DEBUG) {
// eslint-disable-next-line no-console
console.error('Unexpected channel creation update', result);
}
return undefined;
}
const newChannel = result.chats[0];
if (!newChannel || !(newChannel instanceof GramJs.Channel)) {
if (DEBUG) {
// eslint-disable-next-line no-console
console.error('Created channel not found', result);
}
return undefined;
}
const channel = buildApiChatFromPreview(newChannel)!;
let missingUsers: ApiMissingInvitedUser[] | undefined;
if (users?.length) {
const invitedUsers = await invokeRequest(new GramJs.channels.InviteToChannel({
channel: buildInputChannel(channel.id, channel.accessHash),
users: users.map(({ id, accessHash }) => buildInputUser(id, accessHash)),
}));
if (!invitedUsers) return undefined;
handleGramJsUpdate(invitedUsers.updates);
missingUsers = invitedUsers.missingInvitees.map(buildApiMissingInvitedUser);
}
return { channel, missingUsers };
}
export function joinChannel({
channelId, accessHash,
}: {
channelId: string; accessHash: string;
}) {
return invokeRequest(new GramJs.channels.JoinChannel({
channel: buildInputChannel(channelId, accessHash),
}), {
shouldReturnTrue: true,
shouldThrow: true,
});
}
export function deleteChatUser({
chat, user, shouldRevokeHistory,
}: {
chat: ApiChat; user: ApiUser; shouldRevokeHistory?: boolean;
}) {
if (chat.type !== 'chatTypeBasicGroup') return undefined;
return invokeRequest(new GramJs.messages.DeleteChatUser({
chatId: buildInputChat(chat.id),
userId: buildInputUser(user.id, user.accessHash),
revokeHistory: shouldRevokeHistory || undefined,
}), {
shouldReturnTrue: true,
});
}
export function deleteChat({
chatId,
}: {
chatId: string;
}) {
return invokeRequest(new GramJs.messages.DeleteChat({
chatId: buildInputChat(chatId),
}), {
shouldReturnTrue: true,
});
}
export function leaveChannel({ chat }: { chat: ApiChat }) {
return invokeRequest(new GramJs.channels.LeaveChannel({
channel: buildInputChannel(chat.id, chat.accessHash),
}), {
shouldReturnTrue: true,
});
}
export async function fetchFutureCreatorAfterLeave({ chat }: { chat: ApiChat }) {
const result = await invokeRequest(new GramJs.messages.GetFutureChatCreatorAfterLeave({
peer: buildInputPeer(chat.id, chat.accessHash),
}));
if (!result) {
return undefined;
}
return buildApiUser(result);
}
export async function verifyTransferOwnership({
chat, user,
}: {
chat: ApiChat;
user: ApiUser;
}) {
try {
await invokeRequest(new GramJs.messages.EditChatCreator({
peer: buildInputPeer(chat.id, chat.accessHash),
userId: buildInputUser(user.id, user.accessHash),
password: new GramJs.InputCheckPasswordEmpty(),
}), {
shouldReturnTrue: true,
shouldThrow: true,
});
return { canTransfer: true };
} catch (err: any) {
if (!checkErrorType(err)) return undefined;
if (err instanceof RPCError && err.errorMessage === 'PASSWORD_HASH_INVALID') return { canTransfer: true };
if (err instanceof PasswordFreshError) return { errorMessage: 'PASSWORD_TOO_FRESH' };
if (err instanceof SessionFreshError) return { errorMessage: 'SESSION_TOO_FRESH' };
if (err instanceof RPCError && err.errorMessage === 'PASSWORD_MISSING') return { errorMessage: 'PASSWORD_MISSING' };
return wrapError(err);
}
}
export async function editChatCreator({
chat, user, password,
}: {
chat: ApiChat;
user: ApiUser;
password: string;
}) {
try {
const passwordCheck = await getPassword(password);
if (!passwordCheck) {
return undefined;
}
if ('error' in passwordCheck) {
return passwordCheck;
}
return invokeRequest(new GramJs.messages.EditChatCreator({
peer: buildInputPeer(chat.id, chat.accessHash),
userId: buildInputUser(user.id, user.accessHash),
password: passwordCheck,
}), {
shouldReturnTrue: true,
shouldThrow: true,
});
} catch (err) {
if (!checkErrorType(err)) return undefined;
return wrapError(err);
}
}
export function deleteChannel({
channelId, accessHash,
}: {
channelId: string; accessHash: string;
}) {
return invokeRequest(new GramJs.channels.DeleteChannel({
channel: buildInputChannel(channelId, accessHash),
}), {
shouldReturnTrue: true,
});
}
export async function createGroupChat({
title, users,
}: {
title: string; users: ApiUser[];
}) {
const invitedUsers = await invokeRequest(new GramJs.messages.CreateChat({
title,
users: users.map(({ id, accessHash }) => buildInputUser(id, accessHash)),
}));
if (!invitedUsers) return undefined;
handleGramJsUpdate(invitedUsers.updates);
const missingUsers = invitedUsers.missingInvitees.map(buildApiMissingInvitedUser);
const newChat = (invitedUsers.updates as GramJs.Updates).chats[0];
if (!newChat || !(newChat instanceof GramJs.Chat)) {
if (DEBUG) {
// eslint-disable-next-line no-console
console.error('Created chat not found', invitedUsers.updates);
}
return undefined;
}
return { chat: buildApiChatFromPreview(newChat), missingUsers };
}
export async function editChatPhoto({
chatId, accessHash, photo,
}: {
chatId: string; accessHash?: string; photo?: File | ApiPhoto;
}) {
const chatType = getEntityTypeById(chatId);
let inputPhoto: GramJs.TypeInputChatPhoto;
if (photo instanceof File) {
const uploadedPhoto = await uploadFile(photo);
inputPhoto = new GramJs.InputChatUploadedPhoto({
file: uploadedPhoto,
});
} else if (photo) {
const photoId = buildInputPhoto(photo);
if (!photoId) return false;
inputPhoto = new GramJs.InputChatPhoto({
id: photoId,
});
} else {
inputPhoto = new GramJs.InputChatPhotoEmpty();
}
return invokeRequest(
chatType === 'channel'
? new GramJs.channels.EditPhoto({
channel: buildInputChannel(chatId, accessHash),
photo: inputPhoto,
})
: new GramJs.messages.EditChatPhoto({
chatId: buildInputChat(chatId),
photo: inputPhoto,
}),
{
shouldReturnTrue: true,
},
);
}
export async function toggleChatPinned({
chat,
shouldBePinned,
}: {
chat: ApiChat;
shouldBePinned: boolean;
}) {
const { id, accessHash } = chat;
const isActionSuccessful = await invokeRequest(new GramJs.messages.ToggleDialogPin({
peer: new GramJs.InputDialogPeer({
peer: buildInputPeer(id, accessHash),
}),
pinned: shouldBePinned || undefined,
}));
if (isActionSuccessful) {
sendApiUpdate({
'@type': 'updateChatPinned',
id: chat.id,
isPinned: shouldBePinned,
});
}
}
export async function toggleSavedDialogPinned({
chat, shouldBePinned,
}: {
chat: ApiChat;
shouldBePinned: boolean;
}) {
const { id, accessHash } = chat;
const isActionSuccessful = await invokeRequest(new GramJs.messages.ToggleSavedDialogPin({
peer: new GramJs.InputDialogPeer({
peer: buildInputPeer(id, accessHash),
}),
pinned: shouldBePinned || undefined,
}));
if (isActionSuccessful) {
sendApiUpdate({
'@type': 'updateSavedDialogPinned',
id: chat.id,
isPinned: shouldBePinned,
});
}
}
export function toggleChatArchived({
chat, folderId,
}: {
chat: ApiChat; folderId: number;
}) {
const { id, accessHash } = chat;
return invokeRequest(new GramJs.folders.EditPeerFolders({
folderPeers: [new GramJs.InputFolderPeer({
peer: buildInputPeer(id, accessHash),
folderId,
})],
}), {
shouldReturnTrue: true,
});
}
export async function fetchChatFolders() {
const result = await invokeRequest(new GramJs.messages.GetDialogFilters());
if (!result) {
return undefined;
}
const { filters, tagsEnabled: areTagsEnabled } = result;
const defaultFolderPosition = filters.findIndex((folder) => folder instanceof GramJs.DialogFilterDefault);
const dialogFilters = filters.filter(isChatFolder);
const orderedIds = dialogFilters.map(({ id }) => id);
if (defaultFolderPosition !== -1) {
orderedIds.splice(defaultFolderPosition, 0, ALL_FOLDER_ID);
}
return {
byId: buildCollectionByKey(
dialogFilters
.map(buildApiChatFolder), 'id',
) as Record<number, ApiChatFolder>,
orderedIds,
areTagsEnabled,
};
}
export async function fetchPinnedDialogs({
listType,
}: {
listType: ChatListType;
}) {
const result = await invokeRequest(new GramJs.messages.GetPinnedDialogs({
folderId: listType === 'archived' ? ARCHIVED_FOLDER_ID : ALL_FOLDER_ID,
}));
if (!result) {
return undefined;
}
const { dialogs, messages, chats, users } = result;
return {
dialogIds: dialogs.map((dialog) => getApiChatIdFromMtpPeer(dialog.peer)),
messages: messages.map((message) => buildApiMessage(message)).filter(Boolean),
chats: chats.map((chat) => buildApiChatFromPreview(chat)).filter(Boolean),
users: users.map((user) => buildApiUser(user)).filter(Boolean),
};
}
export async function fetchRecommendedChatFolders() {
const results = await invokeRequest(new GramJs.messages.GetSuggestedDialogFilters());
if (!results) {
return undefined;
}
return results.map(buildApiChatFolderFromSuggested).filter(Boolean);
}
export async function editChatFolder({
id,
folderUpdate,
}: {
id: number;
folderUpdate: ApiChatFolder;
}) {
// Telegram ignores excluded chats if they also present in the included list
folderUpdate.excludedChatIds = folderUpdate.excludedChatIds.filter((chatId) => {
return !folderUpdate.includedChatIds.includes(chatId);
});
const filter = buildFilterFromApiFolder(folderUpdate);
const isActionSuccessful = await invokeRequest(new GramJs.messages.UpdateDialogFilter({
id,
filter,
}));
if (isActionSuccessful) {
sendApiUpdate({
'@type': 'updateChatFolder',
id,
folder: folderUpdate,
});
}
}
export async function deleteChatFolder(id: number) {
const isActionSuccessful = await invokeRequest(new GramJs.messages.UpdateDialogFilter({
id,
filter: undefined,
}));
const recommendedChatFolders = await fetchRecommendedChatFolders();
if (isActionSuccessful) {
sendApiUpdate({
'@type': 'updateChatFolder',
id,
folder: undefined,
});
}
if (recommendedChatFolders) {
sendApiUpdate({
'@type': 'updateRecommendedChatFolders',
folders: recommendedChatFolders,
});
}
}
export function sortChatFolders(ids: number[]) {
return invokeRequest(new GramJs.messages.UpdateDialogFiltersOrder({
order: ids,
}));
}
export function toggleDialogFilterTags(isEnabled: boolean) {
return invokeRequest(new GramJs.messages.ToggleDialogFilterTags({
enabled: isEnabled,
}));
}
export async function toggleDialogUnread({
chat, hasUnreadMark,
}: {
chat: ApiChat; hasUnreadMark?: true;
}) {
const { id, accessHash } = chat;
const isActionSuccessful = await invokeRequest(new GramJs.messages.MarkDialogUnread({
peer: new GramJs.InputDialogPeer({
peer: buildInputPeer(id, accessHash),
}),
unread: hasUnreadMark,
}));
if (isActionSuccessful) {
sendApiUpdate({
'@type': 'updateThreadReadState',
chatId: chat.id,
threadId: MAIN_THREAD_ID,
readState: {
hasUnreadMark,
},
});
}
}
export async function getChatByPhoneNumber(phoneNumber: string) {
const result = await invokeRequest(new GramJs.contacts.ResolvePhone({
phone: phoneNumber,
}));
return processResolvedPeer(result);
}
export async function getChatByUsername(username: string, referrer?: string) {
const result = await invokeRequest(new GramJs.contacts.ResolveUsername({
username,
referer: referrer,
}));
return processResolvedPeer(result);
}
function processResolvedPeer(result?: GramJs.contacts.TypeResolvedPeer) {
if (!result) {
return undefined;
}
const { users, chats } = result;
const chat = chats.length
? buildApiChatFromPreview(chats[0])
: buildApiChatFromPreview(users[0]);
if (!chat) {
return undefined;
}
return {
chat,
user: buildApiUser(users[0]),
};
}
export function togglePreHistoryHidden({
chat, isEnabled,
}: { chat: ApiChat; isEnabled: boolean }) {
const { id, accessHash } = chat;
const channel = buildInputChannel(id, accessHash);
return invokeRequest(new GramJs.channels.TogglePreHistoryHidden({
channel,
enabled: isEnabled,
}), {
shouldReturnTrue: true,
});
}
export function updateChatDefaultBannedRights({
chat, bannedRights,
}: { chat: ApiChat; bannedRights: ApiChatBannedRights }) {
const { id, accessHash } = chat;
const peer = buildInputPeer(id, accessHash);
return invokeRequest(new GramJs.messages.EditChatDefaultBannedRights({
peer,
bannedRights: buildChatBannedRights(bannedRights),
}), {
shouldReturnTrue: true,
});
}
export function updateChatMemberBannedRights({
chat, user, bannedRights, untilDate,
}: { chat: ApiChat; user: ApiUser; bannedRights: ApiChatBannedRights; untilDate?: number }) {
const channel = buildInputChannel(chat.id, chat.accessHash);
const participant = buildInputPeer(user.id, user.accessHash);
return invokeRequest(new GramJs.channels.EditBanned({
channel,
participant,
bannedRights: buildChatBannedRights(bannedRights, untilDate),
}), {
shouldReturnTrue: true,
});
}
export function updateChatAdmin({
chat, user, adminRights, customTitle = DEFAULT_PRIMITIVES.STRING,
}: { chat: ApiChat; user: ApiUser; adminRights: ApiChatAdminRights; customTitle?: string }) {
const channel = buildInputChannel(chat.id, chat.accessHash);
const userId = buildInputUser(user.id, user.accessHash);
return invokeRequest(new GramJs.channels.EditAdmin({
channel,
userId,
adminRights: buildChatAdminRights(adminRights),
rank: customTitle,
}), {
shouldReturnTrue: true,
});
}
export async function updateChatTitle(chat: ApiChat, title: string) {
const type = getEntityTypeById(chat.id);
await invokeRequest(
type === 'channel'
? new GramJs.channels.EditTitle({
channel: buildInputChannel(chat.id, chat.accessHash),
title,
}) : new GramJs.messages.EditChatTitle({
chatId: buildInputChat(chat.id),
title,
}),
{
shouldReturnTrue: true,
},
);
}
export async function updateChatAbout(chat: ApiChat, about: string) {
const result = await invokeRequest(new GramJs.messages.EditChatAbout({
peer: buildInputPeer(chat.id, chat.accessHash),
about,
}));
if (!result) {
return;
}
sendApiUpdate({
'@type': 'updateChatFullInfo',
id: chat.id,
fullInfo: {
about,
},
});
}
export function toggleSignatures({
chat, areSignaturesEnabled, areProfilesEnabled,
}: {
chat: ApiChat;
areSignaturesEnabled: boolean;
areProfilesEnabled: boolean;
}) {
const { id, accessHash } = chat;
const channel = buildInputChannel(id, accessHash);
return invokeRequest(new GramJs.channels.ToggleSignatures({
channel,
signaturesEnabled: areSignaturesEnabled || undefined,
profilesEnabled: areProfilesEnabled || undefined,
}), {
shouldReturnTrue: true,
});
}
type ChannelMembersFilter =
'kicked'
| 'admin'
| 'recent'
| 'search';
export async function fetchMembers({
chat,
memberFilter = 'recent',
offset,
query = DEFAULT_PRIMITIVES.STRING,
}: {
chat: ApiChat;
memberFilter?: ChannelMembersFilter;
offset?: number;
query?: string;
}) {
let filter: GramJs.TypeChannelParticipantsFilter;
switch (memberFilter) {
case 'kicked':
filter = new GramJs.ChannelParticipantsKicked({ q: query });
break;
case 'admin':
filter = new GramJs.ChannelParticipantsAdmins();
break;
case 'search':
filter = new GramJs.ChannelParticipantsSearch({ q: query });
break;
default:
filter = new GramJs.ChannelParticipantsRecent();
break;
}
const result = await invokeRequest(new GramJs.channels.GetParticipants({
channel: buildInputChannel(chat.id, chat.accessHash),
filter,
offset: offset ?? DEFAULT_PRIMITIVES.INT,
hash: DEFAULT_PRIMITIVES.BIGINT,
limit: MEMBERS_LOAD_SLICE,
}), {
abortControllerChatId: chat.id,
});
if (!result || result instanceof GramJs.channels.ChannelParticipantsNotModified) {
return undefined;
}
const userStatusesById = buildApiUserStatuses(result.users);
return {
members: buildChatMembers(result),
userStatusesById,
};
}
export async function fetchMember({
chat,
peer,
}: {
chat: ApiChat;
peer?: ApiPeer;
}) {
const participant = peer ? buildInputPeer(peer.id, peer.accessHash) : new GramJs.InputPeerSelf();
const result = await invokeRequest(new GramJs.channels.GetParticipant({
channel: buildInputChannel(chat.id, chat.accessHash),
participant,
}), {
abortControllerChatId: chat.id,
});
if (!result) {
return undefined;
}
const userStatusesById = buildApiUserStatuses(result.users);
const member = buildChatMember(result.participant);
if (!member) {
return undefined;
}
return {
member,
userStatusesById,
};
}
export async function fetchGroupsForDiscussion() {
const result = await invokeRequest(new GramJs.channels.GetGroupsForDiscussion());
if (!result) {
return undefined;
}
return result.chats.map((chat) => buildApiChatFromPreview(chat));
}
export function setDiscussionGroup({
channel,
chat,
}: {
channel: ApiChat;
chat?: ApiChat;
}) {
return invokeRequest(new GramJs.channels.SetDiscussionGroup({
broadcast: buildInputChannel(channel.id, channel.accessHash),
group: chat ? buildInputChannel(chat.id, chat.accessHash) : new GramJs.InputChannelEmpty(),
}), {
shouldReturnTrue: true,
});
}
export async function migrateChat(chat: ApiChat) {
const result = await invokeRequest(
new GramJs.messages.MigrateChat({ chatId: buildInputChat(chat.id) }),
{
shouldThrow: true,
},
);
// `migrateChat` can return a lot of different update types according to docs,
// but currently chat migrations returns only `Updates` type.
// Errors are added to catch unexpected cases in future testing
if (!result || !(result instanceof GramJs.Updates)) {
if (DEBUG) {
// eslint-disable-next-line no-console
console.error('Unexpected channel creation update', result);
}
return undefined;
}
const newChannelId = result.updates
.find((update): update is GramJs.UpdateChannel => update instanceof GramJs.UpdateChannel)!.channelId;
const newChannel = result.chats.find((c) => (
c instanceof GramJs.Channel && c.id.toString() === newChannelId.toString()
))!;
return buildApiChatFromPreview(newChannel);
}
export async function checkChatInvite(hash: string) {
const result = await invokeRequest(new GramJs.messages.CheckChatInvite({ hash }));
if (!result) {
return undefined;
}
if (result instanceof GramJs.ChatInvite) {
return {
chat: undefined,
invite: buildApiChatInviteInfo(result),
users: result.participants?.map(buildApiUser).filter(Boolean),
};
}
const chat = buildApiChatFromPreview(result.chat);
if (!chat) {
return undefined;
}
return { chat, invite: undefined, users: undefined };
}
export async function addChatMembers(chat: ApiChat, users: ApiUser[]) {
try {
if (chat.type === 'chatTypeChannel' || chat.type === 'chatTypeSuperGroup') {
const invitedUsers = await invokeRequest(new GramJs.channels.InviteToChannel({
channel: buildInputChannel(chat.id, chat.accessHash),
users: users.map((user) => buildInputUser(user.id, user.accessHash)),
}));
if (!invitedUsers) return undefined;
handleGramJsUpdate(invitedUsers.updates);
return invitedUsers.missingInvitees.map(buildApiMissingInvitedUser);
}
const addChatUsersResult = await Promise.all(
users.map(async (user) => {
const invitedUsers = await invokeRequest(new GramJs.messages.AddChatUser({
chatId: buildInputChat(chat.id),
userId: buildInputUser(user.id, user.accessHash),
fwdLimit: DEFAULT_PRIMITIVES.INT,
}));
if (!invitedUsers) return undefined;
handleGramJsUpdate(invitedUsers.updates);
return invitedUsers.missingInvitees.map(buildApiMissingInvitedUser);
}),
);
if (addChatUsersResult) {
return addChatUsersResult.flat().filter(Boolean);
}
} catch (err: unknown) {
const message = err instanceof RPCError ? err.errorMessage : (err as Error).message;
sendApiUpdate({
'@type': 'error',
error: {
message,
},
});
}
return undefined;
}
export function deleteChatMember(chat: ApiChat, user: ApiUser) {
return invokeRequest(new GramJs.messages.DeleteChatUser({
chatId: buildInputChat(chat.id),
userId: buildInputUser(user.id, user.accessHash),
}), {
shouldReturnTrue: true,
});
}
export function toggleJoinToSend(chat: ApiChat, isEnabled: boolean) {
return invokeRequest(new GramJs.channels.ToggleJoinToSend({
channel: buildInputChannel(chat.id, chat.accessHash),
enabled: isEnabled,
}), {
shouldReturnTrue: true,
});
}
export function toggleJoinRequest(chat: ApiChat, isEnabled: boolean) {
return invokeRequest(new GramJs.channels.ToggleJoinRequest({
channel: buildInputChannel(chat.id, chat.accessHash),
enabled: isEnabled,
}), {
shouldReturnTrue: true,
});
}
function preparePeers(
result: GramJs.messages.Dialogs | GramJs.messages.DialogsSlice | GramJs.messages.PeerDialogs |
GramJs.messages.SavedDialogs | GramJs.messages.SavedDialogsSlice,
currentStore?: Record<string, GramJs.TypeChat | GramJs.TypeUser>,
) {
const store: Record<string, GramJs.TypeChat | GramJs.TypeUser> = {};
result.chats?.forEach((chat) => {
const key = `chat${chat.id.toString()}`;
if (currentStore?.[key] && 'min' in chat && chat.min) {
return;
}
store[key] = chat;
});
result.users?.forEach((user) => {
const key = `user${user.id.toString()}`;
if (currentStore?.[key] && 'min' in user && user.min) {
return;
}
store[key] = user;
});
return store;
}
export async function importChatInvite({ hash }: { hash: string }) {
const updates = await invokeRequest(new GramJs.messages.ImportChatInvite({ hash }));
if (!(updates instanceof GramJs.Updates) || !updates.chats.length) {
return undefined;
}
return buildApiChatFromPreview(updates.chats[0]);
}
export function setChatEnabledReactions({
chat, enabledReactions, reactionsLimit,
}: {
chat: ApiChat; enabledReactions?: ApiChatReactions; reactionsLimit?: number;
}) {
return invokeRequest(new GramJs.messages.SetChatAvailableReactions({
peer: buildInputPeer(chat.id, chat.accessHash),
availableReactions: buildInputChatReactions(enabledReactions),
reactionsLimit,
}), {
shouldReturnTrue: true,
});
}
export function toggleIsProtected({
chat, isProtected,
}: { chat: ApiChat; isProtected: boolean }) {
const { id, accessHash } = chat;
return invokeRequest(new GramJs.messages.ToggleNoForwards({
peer: buildInputPeer(id, accessHash),
enabled: isProtected,
}), {
shouldReturnTrue: true,
});
}
export function toggleParticipantsHidden({
chat, isEnabled,
}: { chat: ApiChat; isEnabled: boolean }) {
const { id, accessHash } = chat;
return invokeRequest(new GramJs.channels.ToggleParticipantsHidden({
channel: buildInputChannel(id, accessHash),
enabled: isEnabled,
}), {
shouldReturnTrue: true,
});
}
export function toggleForum({
chat, isEnabled,
}: { chat: ApiChat; isEnabled: boolean }) {
const { id, accessHash } = chat;
return invokeRequest(new GramJs.channels.ToggleForum({
channel: buildInputChannel(id, accessHash),
enabled: isEnabled,
tabs: Boolean(chat.withForumTabs),
}), {
shouldReturnTrue: true,
shouldThrow: true,
});
}
export async function checkChatlistInvite({
slug,
}: {
slug: string;
}) {
const result = await invokeRequest(new GramJs.chatlists.CheckChatlistInvite({
slug,
}));
const invite = buildApiChatlistInvite(result, slug);
if (!result || !invite) return undefined;
return {
invite,
};
}
export function joinChatlistInvite({
slug,
peers,
}: {
slug: string;
peers: ApiChat[];
}) {
return invokeRequest(new GramJs.chatlists.JoinChatlistInvite({
slug,
peers: peers.map((peer) => buildInputPeer(peer.id, peer.accessHash)),
}), {
shouldReturnTrue: true,
shouldThrow: true,
});
}
export async function fetchLeaveChatlistSuggestions({
folderId,
}: {
folderId: number;
}) {
const result = await invokeRequest(new GramJs.chatlists.GetLeaveChatlistSuggestions({
chatlist: new GramJs.InputChatlistDialogFilter({
filterId: folderId,
}),
}));
if (!result) return undefined;
return result.map(getApiChatIdFromMtpPeer);
}
export function leaveChatlist({
folderId,
peers,
}: {
folderId: number;
peers: ApiChat[];
}) {
return invokeRequest(new GramJs.chatlists.LeaveChatlist({
chatlist: new GramJs.InputChatlistDialogFilter({
filterId: folderId,
}),
peers: peers.map((peer) => buildInputPeer(peer.id, peer.accessHash)),
}), {
shouldReturnTrue: true,
});
}
export async function createChalistInvite({
folderId,
title = DEFAULT_PRIMITIVES.STRING,
peers,
}: {
folderId: number;
title?: string;
peers: ApiPeer[];
}) {
const result = await invokeRequest(new GramJs.chatlists.ExportChatlistInvite({
chatlist: new GramJs.InputChatlistDialogFilter({
filterId: folderId,
}),
title,
peers: peers.map((peer) => buildInputPeer(peer.id, peer.accessHash)),
}), {
shouldThrow: true,
});
if (!result || result.filter instanceof GramJs.DialogFilterDefault) return undefined;
return {
filter: buildApiChatFolder(result.filter),
invite: buildApiChatlistExportedInvite(result.invite),
};
}
export function deleteChatlistInvite({
folderId, slug,
}: {
folderId: number;
slug: string;
}) {
return invokeRequest(new GramJs.chatlists.DeleteExportedInvite({
chatlist: new GramJs.InputChatlistDialogFilter({
filterId: folderId,
}),
slug,
}));
}
export async function editChatlistInvite({
folderId, slug, title, peers,
}: {
folderId: number;
slug: string;
title?: string;
peers: ApiPeer[];
}) {
const result = await invokeRequest(new GramJs.chatlists.EditExportedInvite({
chatlist: new GramJs.InputChatlistDialogFilter({
filterId: folderId,
}),
slug,
title,
peers: peers.map((peer) => buildInputPeer(peer.id, peer.accessHash)),
}), {
shouldThrow: true,
});
if (!result) return undefined;
return buildApiChatlistExportedInvite(result);
}
export async function fetchChatlistInvites({
folderId,
}: {
folderId: number;
}) {
const result = await invokeRequest(new GramJs.chatlists.GetExportedInvites({
chatlist: new GramJs.InputChatlistDialogFilter({
filterId: folderId,
}),
}));
if (!result) return undefined;
return {
invites: result.invites.map(buildApiChatlistExportedInvite).filter(Boolean),
};
}
export function togglePeerTranslations({
chat, isEnabled,
}: {
chat: ApiChat;
isEnabled: boolean;
}) {
return invokeRequest(new GramJs.messages.TogglePeerTranslations({
disabled: isEnabled ? undefined : true,
peer: buildInputPeer(chat.id, chat.accessHash),
}));
}
export function setViewForumAsMessages({ chat, isEnabled }: { chat: ApiChat; isEnabled: boolean }) {
const { id, accessHash } = chat;
const channel = buildInputChannel(id, accessHash);
return invokeRequest(new GramJs.channels.ToggleViewForumAsMessages({
channel,
enabled: Boolean(isEnabled),
}), {
shouldReturnTrue: true,
});
}
export async function fetchChannelRecommendations({ chat }: { chat?: ApiChat }) {
const result = await invokeRequest(new GramJs.channels.GetChannelRecommendations({
channel: chat && buildInputChannel(chat.id, chat.accessHash),
}));
if (!result) {
return undefined;
}
const similarChannels = result?.chats
.map((c) => buildApiChatFromPreview(c))
.filter(Boolean);
return {
similarChannels,
count: result instanceof GramJs.messages.ChatsSlice ? result.count : similarChannels.length,
};
}
export function updatePaidMessagesPrice({
chat, paidMessagesStars,
}: {
chat?: ApiChat; paidMessagesStars: number;
}) {
return invokeRequest(new GramJs.channels.UpdatePaidMessagesPrice({
channel: chat ? buildInputChannel(chat.id, chat.accessHash) : new GramJs.InputChannelEmpty(),
sendPaidMessagesStars: BigInt(paidMessagesStars),
}), {
shouldReturnTrue: true,
});
}
export async function fetchSponsoredPeer({ query }: { query: string }) {
const result = await invokeRequest(new GramJs.contacts.GetSponsoredPeers({ q: query }));
if (!result || result instanceof GramJs.contacts.SponsoredPeersEmpty) return undefined;
return buildApiSponsoredPeer(result.peers[0]);
}
export function toggleAutoTranslation({
chat, isEnabled,
}: {
chat: ApiChat; isEnabled: boolean;
}) {
return invokeRequest(new GramJs.channels.ToggleAutotranslation({
channel: buildInputChannel(chat.id, chat.accessHash),
enabled: isEnabled,
}), {
shouldReturnTrue: true,
});
}
export function setChannelMainProfileTab({ chat, tab }: { chat: ApiChat; tab: ApiProfileTab }) {
return invokeRequest(new GramJs.channels.SetMainProfileTab({
channel: buildInputChannel(chat.id, chat.accessHash),
tab: buildInputProfileTab(tab),
}), {
shouldReturnTrue: true,
});
}