mtcute/packages/core/src/utils/peer-utils.ts

224 lines
6.1 KiB
TypeScript
Raw Normal View History

2021-08-05 20:38:24 +03:00
import { tl } from '@mtcute/tl'
import { BasicPeerType } from '../types'
2021-04-08 12:19:38 +03:00
// src: https://github.com/tdlib/td/blob/master/td/telegram/DialogId.h
const ZERO_CHANNEL_ID = -1000000000000
const ZERO_SECRET_CHAT_ID = -2000000000000
const MAX_SECRET_CHAT_ID = -1997852516353 // ZERO_SECRET_CHAT_ID + 0x7fffffff
// src: https://github.com/tdlib/td/blob/master/td/telegram/ChatId.h
// const MAX_CHAT_ID = 999999999999
const MIN_MARKED_CHAT_ID = -999999999999 // -MAX_CHAT_ID
// src: https://github.com/tdlib/td/blob/master/td/telegram/UserId.h
// MAX_USER_ID = (1ll << 40) - 1
const MAX_USER_ID = 1099511627775
// src: https://github.com/tdlib/td/blob/master/td/telegram/ChannelId.h
// the last (1 << 31) - 1 identifiers will be used for secret chat dialog identifiers
// MAX_CHANNEL_ID = 1000000000000ll - (1ll << 31)
// const MAX_CHANNEL_ID = 997852516352
const MIN_MARKED_CHANNEL_ID = -1997852516352 // ZERO_CHANNEL_ID - MAX_CHANNEL_ID
2022-08-29 16:22:57 +03:00
/**
* Add or remove channel marker from ID
*/
export function toggleChannelIdMark(id: number): number {
return ZERO_CHANNEL_ID - id
}
2021-04-08 12:19:38 +03:00
/**
* Get the bare (non-marked) ID from a {@link tl.TypePeer}
*/
export function getBarePeerId(peer: tl.TypePeer): number {
switch (peer._) {
case 'peerUser':
return peer.userId
case 'peerChat':
return peer.chatId
case 'peerChannel':
return peer.channelId
}
2021-04-08 12:19:38 +03:00
}
// src: https://github.com/tdlib/td/blob/master/td/telegram/DialogId.cpp
2021-04-08 12:19:38 +03:00
/**
* Get the marked (non-bare) ID from a {@link tl.TypePeer}
*
* *Mark* is bot API compatible, which is:
* - ID stays the same for users
* - ID is negated for chats
* - ID is negated and `-1e12` is subtracted for channels
2021-04-08 12:19:38 +03:00
*/
export function getMarkedPeerId(peerId: number, peerType: BasicPeerType): number
2022-08-29 16:22:57 +03:00
export function getMarkedPeerId(peer: tl.TypePeer | tl.TypeInputPeer | tl.TypeInputUser | tl.TypeInputChannel): number
2021-04-08 12:19:38 +03:00
export function getMarkedPeerId(
2022-08-29 16:22:57 +03:00
peer: tl.TypePeer | tl.TypeInputPeer | tl.TypeInputUser | tl.TypeInputChannel | number,
peerType?: BasicPeerType,
2021-04-08 12:19:38 +03:00
): number {
if (typeof peer === 'number') {
switch (peerType) {
case 'user':
return peer
case 'chat':
return -peer
case 'channel':
return ZERO_CHANNEL_ID - peer
}
2021-04-08 12:19:38 +03:00
throw new Error('Invalid peer type')
}
switch (peer._) {
case 'peerUser':
case 'inputPeerUser':
2022-08-29 16:22:57 +03:00
case 'inputUser':
return peer.userId
case 'peerChat':
case 'inputPeerChat':
return -peer.chatId
case 'peerChannel':
case 'inputPeerChannel':
2022-08-29 16:22:57 +03:00
case 'inputChannel':
return ZERO_CHANNEL_ID - peer.channelId
}
2021-04-08 12:19:38 +03:00
throw new Error('Invalid peer')
}
/**
* Extract basic peer type from {@link tl.TypePeer} or its marked ID.
*/
export function getBasicPeerType(peer: tl.TypePeer | number): BasicPeerType {
if (typeof peer !== 'number') {
switch (peer._) {
case 'peerUser':
return 'user'
case 'peerChat':
return 'chat'
case 'peerChannel':
return 'channel'
}
2021-04-08 12:19:38 +03:00
}
if (peer < 0) {
if (MIN_MARKED_CHAT_ID <= peer) {
return 'chat'
}
if (MIN_MARKED_CHANNEL_ID <= peer && peer !== ZERO_CHANNEL_ID) { return 'channel' }
if (MAX_SECRET_CHAT_ID <= peer && peer !== ZERO_SECRET_CHAT_ID) {
// return 'secret'
throw new Error('Unsupported')
}
} else if (peer > 0 && peer <= MAX_USER_ID) {
2021-04-08 12:19:38 +03:00
return 'user'
}
throw new Error(`Invalid marked peer id: ${peer}`)
}
2022-08-29 16:22:57 +03:00
/**
* Extract bare peer ID from marked ID.
*
* @param peerId Marked peer ID
*/
2021-04-08 12:19:38 +03:00
export function markedPeerIdToBare(peerId: number): number {
const type = getBasicPeerType(peerId)
switch (type) {
case 'user':
return peerId
case 'chat':
return -peerId
case 'channel':
return toggleChannelIdMark(peerId)
}
2021-04-08 12:19:38 +03:00
throw new Error('Invalid marked peer id')
}
/**
* Extracts all (cacheable) entities from a TlObject or a list of them.
* Only checks `.user`, `.chat`, `.channel`, `.users` and `.chats` properties
*/
export function* getAllPeersFrom(
obj: tl.TlObject | tl.TlObject[],
): Iterable<tl.TypeUser | tl.TypeChat> {
if (typeof obj !== 'object') return
if (Array.isArray(obj)) {
for (const it of obj) {
yield* getAllPeersFrom(it)
}
return
}
2021-04-08 12:19:38 +03:00
switch (obj._) {
case 'user':
case 'chat':
case 'channel':
case 'chatForbidden':
case 'channelForbidden':
yield obj
return
}
2021-04-08 12:19:38 +03:00
if (
'user' in obj &&
typeof obj.user === 'object' &&
obj.user._ === 'user'
) {
yield obj.user
}
2021-04-08 12:19:38 +03:00
if ('chat' in obj && typeof obj.chat === 'object') {
switch (obj.chat._) {
case 'chat':
case 'channel':
case 'chatForbidden':
case 'channelForbidden':
yield obj.chat
break
}
}
2021-04-08 12:19:38 +03:00
if ('channel' in obj && typeof obj.channel === 'object') {
switch (obj.channel._) {
case 'chat':
case 'channel':
case 'chatForbidden':
case 'channelForbidden':
yield obj.channel
break
}
}
if ('users' in obj && Array.isArray(obj.users) && obj.users.length) {
for (const user of obj.users) {
// .users is sometimes number[]
if (typeof user === 'object' && user._ === 'user') {
yield user
2021-04-08 12:19:38 +03:00
}
}
}
if ('chats' in obj && Array.isArray(obj.chats) && obj.chats.length) {
for (const chat of obj.chats) {
// .chats is sometimes number[]
if (typeof chat === 'object') {
switch (chat._) {
case 'chat':
case 'channel':
case 'chatForbidden':
case 'channelForbidden':
yield chat
break
}
2021-04-08 12:19:38 +03:00
}
}
}
}