498 lines
13 KiB
TypeScript
498 lines
13 KiB
TypeScript
import BigInt from 'big-integer';
|
|
import { Api as GramJs } from '../../../lib/gramjs';
|
|
|
|
import { ApiPrivacyKey } from '../../../types';
|
|
|
|
import { generateRandomBytes, readBigIntFromBuffer } from '../../../lib/gramjs/Helpers';
|
|
import {
|
|
ApiChatAdminRights,
|
|
ApiChatBannedRights,
|
|
ApiChatFolder,
|
|
ApiGroupCall,
|
|
ApiMessageEntity,
|
|
ApiMessageEntityTypes,
|
|
ApiNewPoll,
|
|
ApiPhoto,
|
|
ApiPhoneCall,
|
|
ApiReportReason,
|
|
ApiSendMessageAction,
|
|
ApiSticker,
|
|
ApiVideo,
|
|
ApiThemeParameters,
|
|
} from '../../types';
|
|
import localDb from '../localDb';
|
|
import { pick } from '../../../util/iteratees';
|
|
import { deserializeBytes } from '../helpers';
|
|
|
|
const CHANNEL_ID_MIN_LENGTH = 11; // Example: -1000000000
|
|
|
|
export function getEntityTypeById(chatOrUserId: string) {
|
|
if (typeof chatOrUserId === 'number') {
|
|
return getEntityTypeByDeprecatedId(chatOrUserId);
|
|
}
|
|
|
|
if (!chatOrUserId.startsWith('-')) {
|
|
return 'user';
|
|
} else if (chatOrUserId.length >= CHANNEL_ID_MIN_LENGTH) {
|
|
return 'channel';
|
|
} else {
|
|
return 'chat';
|
|
}
|
|
}
|
|
|
|
// Workaround for old-fashioned IDs stored locally
|
|
export function getEntityTypeByDeprecatedId(chatOrUserId: number) {
|
|
if (chatOrUserId > 0) {
|
|
return 'user';
|
|
} else if (chatOrUserId <= -1000000000) {
|
|
return 'channel';
|
|
} else {
|
|
return 'chat';
|
|
}
|
|
}
|
|
|
|
export function buildPeer(chatOrUserId: string): GramJs.TypePeer {
|
|
const type = getEntityTypeById(chatOrUserId);
|
|
|
|
if (type === 'user') {
|
|
return new GramJs.PeerUser({
|
|
userId: buildMtpPeerId(chatOrUserId, 'user'),
|
|
});
|
|
} else if (type === 'channel') {
|
|
return new GramJs.PeerChannel({
|
|
channelId: buildMtpPeerId(chatOrUserId, 'channel'),
|
|
});
|
|
} else {
|
|
return new GramJs.PeerChat({
|
|
chatId: buildMtpPeerId(chatOrUserId, 'chat'),
|
|
});
|
|
}
|
|
}
|
|
|
|
export function buildInputPeer(chatOrUserId: string, accessHash?: string): GramJs.TypeInputPeer {
|
|
const type = getEntityTypeById(chatOrUserId);
|
|
|
|
if (type === 'user') {
|
|
return new GramJs.InputPeerUser({
|
|
userId: buildMtpPeerId(chatOrUserId, 'user'),
|
|
accessHash: BigInt(accessHash!),
|
|
});
|
|
} else if (type === 'channel') {
|
|
return new GramJs.InputPeerChannel({
|
|
channelId: buildMtpPeerId(chatOrUserId, 'channel'),
|
|
accessHash: BigInt(accessHash!),
|
|
});
|
|
} else {
|
|
return new GramJs.InputPeerChat({
|
|
chatId: buildMtpPeerId(chatOrUserId, 'chat'),
|
|
});
|
|
}
|
|
}
|
|
|
|
export function buildInputPeerFromLocalDb(chatOrUserId: string): GramJs.TypeInputPeer | undefined {
|
|
const type = getEntityTypeById(chatOrUserId);
|
|
let accessHash: BigInt.BigInteger | undefined;
|
|
|
|
if (type === 'user') {
|
|
accessHash = localDb.users[chatOrUserId]?.accessHash;
|
|
if (!accessHash) {
|
|
return undefined;
|
|
}
|
|
} else if (type === 'channel') {
|
|
accessHash = (localDb.chats[chatOrUserId] as GramJs.Channel)?.accessHash;
|
|
if (!accessHash) {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
return buildInputPeer(chatOrUserId, String(accessHash));
|
|
}
|
|
|
|
export function buildInputEntity(chatOrUserId: string, accessHash?: string) {
|
|
const type = getEntityTypeById(chatOrUserId);
|
|
|
|
if (type === 'user') {
|
|
return new GramJs.InputUser({
|
|
userId: buildMtpPeerId(chatOrUserId, 'user'),
|
|
accessHash: BigInt(accessHash!),
|
|
});
|
|
} else if (type === 'channel') {
|
|
return new GramJs.InputChannel({
|
|
channelId: buildMtpPeerId(chatOrUserId, 'channel'),
|
|
accessHash: BigInt(accessHash!),
|
|
});
|
|
} else {
|
|
return buildMtpPeerId(chatOrUserId, 'chat');
|
|
}
|
|
}
|
|
|
|
export function buildInputStickerSet(id: string, accessHash: string) {
|
|
return new GramJs.InputStickerSetID({
|
|
id: BigInt(id),
|
|
accessHash: BigInt(accessHash),
|
|
});
|
|
}
|
|
|
|
export function buildInputStickerSetShortName(shortName: string) {
|
|
return new GramJs.InputStickerSetShortName({
|
|
shortName,
|
|
});
|
|
}
|
|
|
|
export function buildInputDocument(media: ApiSticker | ApiVideo) {
|
|
const document = localDb.documents[media.id];
|
|
|
|
if (!document) {
|
|
return undefined;
|
|
}
|
|
|
|
return new GramJs.InputDocument(pick(document, [
|
|
'id',
|
|
'accessHash',
|
|
'fileReference',
|
|
]));
|
|
}
|
|
|
|
export function buildInputMediaDocument(media: ApiSticker | ApiVideo) {
|
|
const inputDocument = buildInputDocument(media);
|
|
|
|
if (!inputDocument) {
|
|
return undefined;
|
|
}
|
|
|
|
return new GramJs.InputMediaDocument({ id: inputDocument });
|
|
}
|
|
|
|
export function buildInputPoll(pollParams: ApiNewPoll, randomId: BigInt.BigInteger) {
|
|
const { summary, quiz } = pollParams;
|
|
|
|
const poll = new GramJs.Poll({
|
|
id: randomId,
|
|
publicVoters: summary.isPublic,
|
|
question: summary.question,
|
|
answers: summary.answers.map(({ text, option }) => {
|
|
return new GramJs.PollAnswer({ text, option: deserializeBytes(option) });
|
|
}),
|
|
quiz: summary.quiz,
|
|
multipleChoice: summary.multipleChoice,
|
|
});
|
|
|
|
if (!quiz) {
|
|
return new GramJs.InputMediaPoll({ poll });
|
|
}
|
|
|
|
const correctAnswers = quiz.correctAnswers.map(deserializeBytes);
|
|
const { solution } = quiz;
|
|
const solutionEntities = quiz.solutionEntities ? quiz.solutionEntities.map(buildMtpMessageEntity) : [];
|
|
|
|
return new GramJs.InputMediaPoll({
|
|
poll,
|
|
correctAnswers,
|
|
...(solution && {
|
|
solution,
|
|
solutionEntities,
|
|
}),
|
|
});
|
|
}
|
|
|
|
export function buildFilterFromApiFolder(folder: ApiChatFolder): GramJs.DialogFilter {
|
|
const {
|
|
emoticon,
|
|
contacts,
|
|
nonContacts,
|
|
groups,
|
|
channels,
|
|
bots,
|
|
excludeArchived,
|
|
excludeMuted,
|
|
excludeRead,
|
|
pinnedChatIds,
|
|
includedChatIds,
|
|
excludedChatIds,
|
|
} = folder;
|
|
|
|
const pinnedPeers = pinnedChatIds
|
|
? pinnedChatIds.map(buildInputPeerFromLocalDb).filter<GramJs.TypeInputPeer>(Boolean as any)
|
|
: [];
|
|
|
|
const includePeers = includedChatIds
|
|
? includedChatIds.map(buildInputPeerFromLocalDb).filter<GramJs.TypeInputPeer>(Boolean as any)
|
|
: [];
|
|
|
|
const excludePeers = excludedChatIds
|
|
? excludedChatIds.map(buildInputPeerFromLocalDb).filter<GramJs.TypeInputPeer>(Boolean as any)
|
|
: [];
|
|
|
|
return new GramJs.DialogFilter({
|
|
id: folder.id,
|
|
title: folder.title,
|
|
emoticon: emoticon || undefined,
|
|
contacts: contacts || undefined,
|
|
nonContacts: nonContacts || undefined,
|
|
groups: groups || undefined,
|
|
bots: bots || undefined,
|
|
excludeArchived: excludeArchived || undefined,
|
|
excludeMuted: excludeMuted || undefined,
|
|
excludeRead: excludeRead || undefined,
|
|
broadcasts: channels || undefined,
|
|
pinnedPeers,
|
|
includePeers,
|
|
excludePeers,
|
|
});
|
|
}
|
|
|
|
export function generateRandomBigInt() {
|
|
return readBigIntFromBuffer(generateRandomBytes(8), true, true);
|
|
}
|
|
|
|
export function generateRandomInt() {
|
|
return readBigIntFromBuffer(generateRandomBytes(4), true, true).toJSNumber();
|
|
}
|
|
|
|
export function buildMessageFromUpdate(
|
|
id: number,
|
|
chatId: string,
|
|
update: GramJs.UpdateShortSentMessage | GramJs.UpdateServiceNotification,
|
|
) {
|
|
// This is not a proper message, but we only need these fields for downloading media through `localDb`.
|
|
return new GramJs.Message({
|
|
id,
|
|
peerId: buildPeer(chatId),
|
|
fromId: buildPeer(chatId),
|
|
media: update.media,
|
|
} as GramJs.Message);
|
|
}
|
|
|
|
export function buildMtpMessageEntity(entity: ApiMessageEntity): GramJs.TypeMessageEntity {
|
|
const {
|
|
type, offset, length, url, userId,
|
|
} = entity;
|
|
|
|
const user = userId ? localDb.users[userId] : undefined;
|
|
|
|
switch (type) {
|
|
case ApiMessageEntityTypes.Bold:
|
|
return new GramJs.MessageEntityBold({ offset, length });
|
|
case ApiMessageEntityTypes.Italic:
|
|
return new GramJs.MessageEntityItalic({ offset, length });
|
|
case ApiMessageEntityTypes.Underline:
|
|
return new GramJs.MessageEntityUnderline({ offset, length });
|
|
case ApiMessageEntityTypes.Strike:
|
|
return new GramJs.MessageEntityStrike({ offset, length });
|
|
case ApiMessageEntityTypes.Code:
|
|
return new GramJs.MessageEntityCode({ offset, length });
|
|
case ApiMessageEntityTypes.Pre:
|
|
return new GramJs.MessageEntityPre({ offset, length, language: '' });
|
|
case ApiMessageEntityTypes.Blockquote:
|
|
return new GramJs.MessageEntityBlockquote({ offset, length });
|
|
case ApiMessageEntityTypes.TextUrl:
|
|
return new GramJs.MessageEntityTextUrl({ offset, length, url: url! });
|
|
case ApiMessageEntityTypes.Url:
|
|
return new GramJs.MessageEntityUrl({ offset, length });
|
|
case ApiMessageEntityTypes.Hashtag:
|
|
return new GramJs.MessageEntityHashtag({ offset, length });
|
|
case ApiMessageEntityTypes.MentionName:
|
|
return new GramJs.InputMessageEntityMentionName({
|
|
offset,
|
|
length,
|
|
userId: new GramJs.InputUser({ userId: BigInt(userId!), accessHash: user!.accessHash! }),
|
|
});
|
|
case ApiMessageEntityTypes.Spoiler:
|
|
return new GramJs.MessageEntitySpoiler({ offset, length });
|
|
default:
|
|
return new GramJs.MessageEntityUnknown({ offset, length });
|
|
}
|
|
}
|
|
|
|
export function isMessageWithMedia(message: GramJs.Message | GramJs.UpdateServiceNotification) {
|
|
const { media } = message;
|
|
if (!media) {
|
|
return false;
|
|
}
|
|
|
|
return (
|
|
media instanceof GramJs.MessageMediaPhoto
|
|
|| media instanceof GramJs.MessageMediaDocument
|
|
|| (
|
|
media instanceof GramJs.MessageMediaWebPage
|
|
&& media.webpage instanceof GramJs.WebPage
|
|
&& (
|
|
media.webpage.photo instanceof GramJs.Photo || (
|
|
media.webpage.document instanceof GramJs.Document
|
|
&& media.webpage.document.mimeType.startsWith('video')
|
|
)
|
|
)
|
|
) || (
|
|
media instanceof GramJs.MessageMediaGame
|
|
&& (media.game.document instanceof GramJs.Document || media.game.photo instanceof GramJs.Photo)
|
|
)
|
|
);
|
|
}
|
|
|
|
export function isServiceMessageWithMedia(message: GramJs.MessageService) {
|
|
return 'photo' in message.action && message.action.photo instanceof GramJs.Photo;
|
|
}
|
|
|
|
export function buildChatPhotoForLocalDb(photo: GramJs.TypePhoto) {
|
|
if (photo instanceof GramJs.PhotoEmpty) {
|
|
return new GramJs.ChatPhotoEmpty();
|
|
}
|
|
|
|
const { dcId, id: photoId } = photo;
|
|
|
|
return new GramJs.ChatPhoto({
|
|
dcId,
|
|
photoId,
|
|
});
|
|
}
|
|
|
|
export function buildInputPhoto(photo: ApiPhoto) {
|
|
const localPhoto = localDb.photos[photo?.id];
|
|
|
|
if (!localPhoto) {
|
|
return undefined;
|
|
}
|
|
|
|
return new GramJs.InputPhoto(pick(localPhoto, [
|
|
'id',
|
|
'accessHash',
|
|
'fileReference',
|
|
]));
|
|
}
|
|
|
|
export function buildInputContact({
|
|
phone,
|
|
firstName,
|
|
lastName,
|
|
}: {
|
|
phone: string;
|
|
firstName: string;
|
|
lastName: string;
|
|
}) {
|
|
return new GramJs.InputPhoneContact({
|
|
clientId: BigInt(1),
|
|
phone,
|
|
firstName,
|
|
lastName,
|
|
});
|
|
}
|
|
|
|
export function buildChatBannedRights(
|
|
bannedRights: ApiChatBannedRights,
|
|
untilDate = 0,
|
|
) {
|
|
return new GramJs.ChatBannedRights({
|
|
...bannedRights,
|
|
untilDate,
|
|
});
|
|
}
|
|
|
|
export function buildChatAdminRights(
|
|
adminRights: ApiChatAdminRights,
|
|
) {
|
|
return new GramJs.ChatAdminRights(adminRights);
|
|
}
|
|
|
|
export function buildShippingInfo(info: GramJs.PaymentRequestedInfo) {
|
|
const { shippingAddress } = info;
|
|
return new GramJs.PaymentRequestedInfo({
|
|
...info,
|
|
shippingAddress: shippingAddress
|
|
? new GramJs.PostAddress(shippingAddress)
|
|
: undefined,
|
|
});
|
|
}
|
|
|
|
export function buildInputPrivacyKey(privacyKey: ApiPrivacyKey) {
|
|
switch (privacyKey) {
|
|
case 'phoneNumber':
|
|
return new GramJs.InputPrivacyKeyPhoneNumber();
|
|
|
|
case 'lastSeen':
|
|
return new GramJs.InputPrivacyKeyStatusTimestamp();
|
|
|
|
case 'profilePhoto':
|
|
return new GramJs.InputPrivacyKeyProfilePhoto();
|
|
|
|
case 'forwards':
|
|
return new GramJs.InputPrivacyKeyForwards();
|
|
|
|
case 'chatInvite':
|
|
return new GramJs.InputPrivacyKeyChatInvite();
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
export function buildInputReportReason(reason: ApiReportReason) {
|
|
switch (reason) {
|
|
case 'spam':
|
|
return new GramJs.InputReportReasonSpam();
|
|
case 'violence':
|
|
return new GramJs.InputReportReasonViolence();
|
|
case 'childAbuse':
|
|
return new GramJs.InputReportReasonChildAbuse();
|
|
case 'pornography':
|
|
return new GramJs.InputReportReasonPornography();
|
|
case 'copyright':
|
|
return new GramJs.InputReportReasonCopyright();
|
|
case 'fake':
|
|
return new GramJs.InputReportReasonFake();
|
|
case 'geoIrrelevant':
|
|
return new GramJs.InputReportReasonGeoIrrelevant();
|
|
case 'illegalDrugs':
|
|
return new GramJs.InputReportReasonIllegalDrugs();
|
|
case 'personalDetails':
|
|
return new GramJs.InputReportReasonPersonalDetails();
|
|
case 'other':
|
|
return new GramJs.InputReportReasonOther();
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
export function buildSendMessageAction(action: ApiSendMessageAction) {
|
|
switch (action.type) {
|
|
case 'cancel':
|
|
return new GramJs.SendMessageCancelAction();
|
|
case 'typing':
|
|
return new GramJs.SendMessageTypingAction();
|
|
case 'recordAudio':
|
|
return new GramJs.SendMessageRecordAudioAction();
|
|
case 'chooseSticker':
|
|
return new GramJs.SendMessageChooseStickerAction();
|
|
case 'playingGame':
|
|
return new GramJs.SendMessageGamePlayAction();
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
export function buildInputThemeParams(params: ApiThemeParameters) {
|
|
return new GramJs.DataJSON({
|
|
data: JSON.stringify(params),
|
|
});
|
|
}
|
|
|
|
export function buildMtpPeerId(id: string, type: 'user' | 'chat' | 'channel') {
|
|
// Workaround for old-fashioned IDs stored locally
|
|
if (typeof id === 'number') {
|
|
return BigInt(Math.abs(id));
|
|
}
|
|
|
|
return type === 'user' ? BigInt(id) : BigInt(id.slice(1));
|
|
}
|
|
|
|
export function buildInputGroupCall(groupCall: Partial<ApiGroupCall>) {
|
|
return new GramJs.InputGroupCall({
|
|
id: BigInt(groupCall.id!),
|
|
accessHash: BigInt(groupCall.accessHash!),
|
|
});
|
|
}
|
|
|
|
export function buildInputPhoneCall({ id, accessHash }: ApiPhoneCall) {
|
|
return new GramJs.InputPhoneCall({
|
|
id: BigInt(id),
|
|
accessHash: BigInt(accessHash!),
|
|
});
|
|
}
|