2022-04-19 15:12:16 +02:00

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!),
});
}