update tg_user_bot

This commit is contained in:
2021-12-14 23:57:50 +08:00
parent e1693f3c63
commit cfa4b7e2d0
3 changed files with 168 additions and 13 deletions

View File

@@ -16,5 +16,10 @@
"secret": "1234567890abcdef" // Optional. The proxy's secret in hexadecimal encoding. Supported if @type is proxyTypeMtproto. "secret": "1234567890abcdef" // Optional. The proxy's secret in hexadecimal encoding. Supported if @type is proxyTypeMtproto.
} }
}, },
"phone_number": "+11234567890" // Optional. User's phone number "phone_number": "+11234567890", // Optional. User's phone number
"BotTdlibParameters": { // Optional. Same as TdlibParameters, but when creating a bot session, these parameters will override parameters in TdlibParameters
"database_directory": "/path/to/botdatabase"
},
"bot_encryption_key": "Base64 encoded's key", // Optional. Encryption key to check or set up for bot. Need base64 encoded.
"bot_token": "12345678:235zoHFG" // Optional. Bot Token, which can be obtained with @BotFather
} }

View File

@@ -298,6 +298,10 @@ def json_object_hook(value):
return parse_message_forward_orgin(value) return parse_message_forward_orgin(value)
elif value['@type'] == 'message': elif value['@type'] == 'message':
value['media_album_id'] = int(value['media_album_id']) value['media_album_id'] = int(value['media_album_id'])
elif value['@type'] == 'getStickerSet':
value['set_id'] = int(value['set_id'])
elif value['@type'] == 'stickerSet':
value['id'] = int(value['id'])
except Exception: except Exception:
print_exc() print_exc()
return value return value
@@ -320,6 +324,12 @@ class TdLibJSONEncoder(JSONEncoder):
if o['@type'] == 'message': if o['@type'] == 'message':
if 'media_album_id' in o: if 'media_album_id' in o:
o['media_album_id'] = str(o['media_album_id']) o['media_album_id'] = str(o['media_album_id'])
elif o['@type'] == 'getStickerSet':
if 'set_id' in o:
o['set_id'] = str(o['set_id'])
elif o['@type'] == 'stickerSet':
if 'id' in o:
o['id'] = str(o['id'])
return super().default(o) return super().default(o)
@@ -328,6 +338,7 @@ class TdLib:
self._initalized = False self._initalized = False
self._destoried = False self._destoried = False
self._db_initalized = False self._db_initalized = False
self._logined = False
self._v = verbose self._v = verbose
self._maxCache = 1000 self._maxCache = 1000
if maxCache is not None: if maxCache is not None:
@@ -407,6 +418,16 @@ class TdLib:
"port": port, "enable": enable, "port": port, "enable": enable,
"type": type}) "type": type})
async def addStickerToSet(self, user_id: int, name: str, sticker):
re = await self._send({"@type": "addStickerToSet", "user_id": user_id,
"name": name, "sticker": sticker})
if re['@type'] == 'stickerSet':
return re
else:
if re['@type'] == 'error':
print(f"{re['code']} {re['message']}")
return None
async def checkAuthenticationCode(self, code: str): async def checkAuthenticationCode(self, code: str):
re = await self._send({"@type": "checkAuthenticationCode", re = await self._send({"@type": "checkAuthenticationCode",
"code": code}) "code": code})
@@ -437,11 +458,23 @@ class TdLib:
print(f"{re['code']} {re['message']}") print(f"{re['code']} {re['message']}")
return False return False
async def checkAuthenticationBotToken(self, token: str):
re = await self._send({"@type": "checkAuthenticationBotToken",
"token": token})
if re['@type'] == 'ok':
return True
else:
if re['@type'] == 'error':
print(f"{re['code']} {re['message']}")
return False
async def createNewStickerSet(self, title: str, name: str, stickers, async def createNewStickerSet(self, title: str, name: str, stickers,
is_masks: bool = False, source: str = ''): is_masks: bool = False, source: str = '',
user_id: int = 0):
re = await self._send({"@type": "createNewStickerSet", "title": title, re = await self._send({"@type": "createNewStickerSet", "title": title,
"name": name, "stickers": stickers, "name": name, "stickers": stickers,
"is_masks": is_masks, "source": source}) "is_masks": is_masks, "source": source,
"user_id": user_id})
if re['@type'] == 'stickerSet': if re['@type'] == 'stickerSet':
return re return re
else: else:
@@ -602,6 +635,15 @@ class TdLib:
await asyncio.sleep(0.1) await asyncio.sleep(0.1)
return await self._send({"@type": "getProxies"}) return await self._send({"@type": "getProxies"})
async def getStickerSet(self, set_id: int):
re = await self._send({"@type": "getStickerSet", "set_id": set_id})
if re['@type'] == 'stickerSet':
return re
else:
if re['@type'] == 'error':
print(f"{re['code']} {re['message']}")
return None
async def getTextEntities(self, text: str): async def getTextEntities(self, text: str):
re = await self._send({"@type": "getTextEntities", "text": text}) re = await self._send({"@type": "getTextEntities", "text": text})
if re['@type'] == 'textEntities': if re['@type'] == 'textEntities':
@@ -616,7 +658,13 @@ class TdLib:
self._uid = (await self.getMe())['id'] self._uid = (await self.getMe())['id']
return self._uid return self._uid
async def login(self, parameters, encryption_key, proxy, phone_number): async def getUsername(self) -> str:
if not hasattr(self, "_username"):
self._username = (await self.getMe())['username']
return self._username
async def login(self, parameters, encryption_key, proxy, phone_number=None,
bot_token=None):
while True: while True:
re = await self.receive('updateAuthorizationState') re = await self.receive('updateAuthorizationState')
state = re['authorization_state'] state = re['authorization_state']
@@ -638,6 +686,10 @@ class TdLib:
if not await self.setProxy(proxy['server'], proxy['port'], proxy['type']): # noqa: E501 if not await self.setProxy(proxy['server'], proxy['port'], proxy['type']): # noqa: E501
raise ValueError('Can not set proxy.') raise ValueError('Can not set proxy.')
elif state['@type'] == 'authorizationStateWaitPhoneNumber': elif state['@type'] == 'authorizationStateWaitPhoneNumber':
if bot_token is not None:
if not await self.checkAuthenticationBotToken(bot_token):
raise ValueError('Invalid bot token')
continue
if phone_number is None: if phone_number is None:
phone_number = input('Please input your phone number:') phone_number = input('Please input your phone number:')
if not await self.setAuthenticationPhoneNumber(phone_number): if not await self.setAuthenticationPhoneNumber(phone_number):
@@ -651,6 +703,7 @@ class TdLib:
if not await self.checkAuthenticationPassword(paw): if not await self.checkAuthenticationPassword(paw):
raise ValueError('Incorrect passwrod.') raise ValueError('Incorrect passwrod.')
elif state['@type'] == 'authorizationStateReady': elif state['@type'] == 'authorizationStateReady':
self._logined = True
return True return True
else: else:
raise ValueError("Unknown authorization_state", state) raise ValueError("Unknown authorization_state", state)
@@ -748,6 +801,15 @@ class TdLib:
print(f"{re['code']} {re['message']}") print(f"{re['code']} {re['message']}")
return None return None
async def searchStickerSet(self, name: str):
re = await self._send({"@type": "searchStickerSet", "name": name})
if re['@type'] == 'stickerSet':
return re
else:
if re['@type'] == 'error':
print(f"{re['code']} {re['message']}")
return None
async def setAuthenticationPhoneNumber(self, phone_number, settings=None): async def setAuthenticationPhoneNumber(self, phone_number, settings=None):
sett = {"@type": "phoneNumberAuthenticationSettings", sett = {"@type": "phoneNumberAuthenticationSettings",
"allow_flash_call": False, "allow_flash_call": False,

View File

@@ -43,10 +43,17 @@ class MyArgParser(ArgumentParser):
cnss = MyArgParser('-createnewstickerset', add_help=False, description="Create a new sticker set.", epilog="Note: Reply to a sticker message is needed.") # noqa: E501 cnss = MyArgParser('-createnewstickerset', add_help=False, description="Create a new sticker set.", epilog="Note: Reply to a sticker message is needed.") # noqa: E501
cnss.add_argument('name', help="Sticker set name. Can contain only English letters, digits and underscores. 1-64 characters", type=validate_sticker_set_name) # noqa: E501 cnss.add_argument('name', help="Sticker set name. Can contain only English letters, digits and underscores. 1-64 characters. Must ended with _by_<botname> if create with a bot account", type=validate_sticker_set_name) # noqa: E501
cnss.add_argument('title', help="Sticker set title. 1-64 characters", type=validate_sticker_set_title) # noqa: E501 cnss.add_argument('title', help="Sticker set title. 1-64 characters", type=validate_sticker_set_title) # noqa: E501
cnss.add_argument('source', help='Source of the sticker set', nargs='?', default='') # noqa: E501 cnss.add_argument('source', help='Source of the sticker set', nargs='?', default='') # noqa: E501
cnss.add_argument('-e', '--emoji', help='Emojis corresponding to the sticker', metavar='EMOJI', dest='emoji') # noqa: E501 cnss.add_argument('-e', '--emoji', help='Emojis corresponding to the sticker', metavar='EMOJI', dest='emoji') # noqa: E501
cnss.add_argument('-u', '--user', help='Create with user account. Will have issue when try add sticker to set', action='store_true', dest='user') # noqa: E501
cnss.add_argument('-a', '--add-suffix', help='Add _by_<botname> automatically', action='store_true', dest='add_suffix') # noqa: E501
asts = MyArgParser('-addstickertoset', add_help=False, description="Add a sticker to a exist sticker set.", epilog="Note: Reply to a sticker message is needed.") # noqa: E501
asts.add_argument('name', help="Sticker set name. Must ended with _by_<botname>", type=validate_sticker_set_name, nargs='?') # noqa: E501
asts.add_argument('--id', help='Sticker set ID. Needed if name not specified.', type=int, metavar='ID', dest='id') # noqa: E501
asts.add_argument('-e', '--emoji', help='Emojis corresponding to the sticker', metavar='EMOJI', dest='emoji') # noqa: E501
asts.add_argument('-a', '--add-suffix', help='Add _by_<botname> automatically', action='store_true', dest='add_suffix') # noqa: E501
def generateFileInfo(f: dict) -> str: def generateFileInfo(f: dict) -> str:
@@ -149,7 +156,7 @@ async def handle_message_info(lib: TdLib, mes):
print('Can not edit message') print('Can not edit message')
async def handle_create_new_sticker_set(lib: TdLib, mes: dict, async def handle_create_new_sticker_set(lib: TdLib, robot: TdLib, mes: dict,
argv: List[str]): argv: List[str]):
if len(argv) == 1: if len(argv) == 1:
re = await lib.editMessageText( re = await lib.editMessageText(
@@ -158,6 +165,11 @@ async def handle_create_new_sticker_set(lib: TdLib, mes: dict,
else: else:
try: try:
r = cnss.parse_intermixed_args(argv[1:]) r = cnss.parse_intermixed_args(argv[1:])
if r.add_suffix and not r.user:
un = await robot.getUsername()
if not r.name.endswith(f'_by_{un}'):
r.name += f'_by_{un}'
validate_sticker_set_name(r.name)
if mes['reply_to_message_id'] == 0: if mes['reply_to_message_id'] == 0:
raise ValueError('Reply a sticker message is needed.') raise ValueError('Reply a sticker message is needed.')
nmes = await lib.getMessage(mes['reply_in_chat_id'], nmes = await lib.getMessage(mes['reply_in_chat_id'],
@@ -175,7 +187,15 @@ async def handle_create_new_sticker_set(lib: TdLib, mes: dict,
if fid == '': if fid == '':
raise ValueError("The target sticker's file id is empty.") raise ValueError("The target sticker's file id is empty.")
st = [{"@type": "inputStickerStatic", "sticker": {"@type": "inputFileRemote", "id": fid}, "emojis": emoji}] # noqa: E501 st = [{"@type": "inputStickerStatic", "sticker": {"@type": "inputFileRemote", "id": fid}, "emojis": emoji}] # noqa: E501
r = await lib.createNewStickerSet(r.title, r.name, st, source=r.source) # noqa: E501 if not r.user and not robot._logined:
raise ValueError('A bot account is needed. Or add -u to create with user account.') # noqa: E501
if r.user:
r = await lib.createNewStickerSet(r.title, r.name, st, source=r.source) # noqa: E501
else:
uid = await lib.getUid()
if uid is None:
raise ValueError('Can not get User ID.')
r = await robot.createNewStickerSet(r.title, r.name, st, source=r.source, user_id=uid) # noqa: E501
if r is None: if r is None:
raise ValueError('Can not create new sticker set.') raise ValueError('Can not create new sticker set.')
re = await lib.editMessageText(mes['chat_id'], mes['id'], f"Created successfully.\nSticker Set ID: {r['id']}\nhttps://t.me/addstickers/{r['name']}") # noqa: E501 re = await lib.editMessageText(mes['chat_id'], mes['id'], f"Created successfully.\nSticker Set ID: {r['id']}\nhttps://t.me/addstickers/{r['name']}") # noqa: E501
@@ -183,19 +203,84 @@ async def handle_create_new_sticker_set(lib: TdLib, mes: dict,
if len(e.args) == 0: if len(e.args) == 0:
re = await lib.editMessageText(mes['chat_id'], mes['id'], "Unknown error.") # noqa: E501 re = await lib.editMessageText(mes['chat_id'], mes['id'], "Unknown error.") # noqa: E501
else: else:
re = await lib.editMessageText(mes['chat_id'], mes['id'], f"```\n{cnss.format_usage()}\n{cnss.prog}: error: {e.args[0] if len(e.args) == 1 else e.args}\n```", TextParseMode.MarkDown) # noqa: E501 re = await lib.editMessageText(mes['chat_id'], mes['id'], f"```\n{cnss.format_usage()}{cnss.prog}: error: {e.args[0] if len(e.args) == 1 else e.args}\n```", TextParseMode.MarkDown) # noqa: E501
except Exception: except Exception:
re = await lib.editMessageText(mes['chat_id'], mes['id'], f"```\n{format_exc()}\n```", TextParseMode.MarkDown) # noqa: E501 re = await lib.editMessageText(mes['chat_id'], mes['id'], f"```\n{format_exc()}\n```", TextParseMode.MarkDown) # noqa: E501
if re is None: if re is None:
print('Can not edit message') print('Can not edit message')
async def main(lib: TdLib): async def handle_add_sticker_to_set(lib: TdLib, robot: TdLib, mes: dict,
argv: List[str]):
if len(argv) == 1:
re = await lib.editMessageText(
mes['chat_id'], mes['id'], f"```\n{asts.format_help()}\n```",
TextParseMode.MarkDown)
else:
try:
if not robot._logined:
raise ValueError('bot_token is not set. This future need a bot. Please contact @BotFather to get a valid bot_token.') # noqa: E501
r = asts.parse_intermixed_args(argv[1:])
if r.add_suffix and r.name:
un = await robot.getUsername()
if not r.name.endswith(f'_by_{un}'):
r.name += f'_by_{un}'
validate_sticker_set_name(r.name)
if r.id is None and r.name is None:
raise ValueError('name or ID is needed.')
if mes['reply_to_message_id'] == 0:
raise ValueError('Reply a sticker message is needed.')
nmes = await lib.getMessage(mes['reply_in_chat_id'],
mes['reply_to_message_id'])
if nmes is None:
raise ValueError('Can not get the replied message.')
if nmes['content']['@type'] != 'messageSticker':
raise ValueError('Reply a sticker message is needed.')
width = nmes['content']['sticker']['width']
height = nmes['content']['sticker']['height']
if max(width, height) != 512:
raise ValueError('Invalid width or height')
fid = nmes['content']['sticker']['sticker']['remote']['id']
emoji = nmes['content']['sticker']['emoji'] if r.emoji is None else r.emoji # noqa: E501
if fid == '':
raise ValueError("The target sticker's file id is empty.")
st = {"@type": "inputStickerStatic", "sticker": {"@type": "inputFileRemote", "id": fid}, "emojis": emoji} # noqa: E501
if r.name is None:
si = await lib.getStickerSet(r.id)
if si is None:
raise ValueError('Can not get sticker set from ID.')
r.name = si['name']
uid = await lib.getUid()
if uid is None:
raise ValueError('Can not get user ID.')
r = await robot.addStickerToSet(uid, r.name, st)
if r is None:
raise ValueError('Failed to add sticker to set')
re = await lib.editMessageText(mes['chat_id'], mes['id'], "Added sticker to set successfully.") # noqa: E501
except ValueError as e:
if len(e.args) == 0:
re = await lib.editMessageText(mes['chat_id'], mes['id'], "Unknown error.") # noqa: E501
else:
re = await lib.editMessageText(mes['chat_id'], mes['id'], f"```\n{asts.format_usage()}{asts.prog}: error: {e.args[0] if len(e.args) == 1 else e.args}\n```", TextParseMode.MarkDown) # noqa: E501
except Exception:
re = await lib.editMessageText(mes['chat_id'], mes['id'], f"```\n{format_exc()}\n```", TextParseMode.MarkDown) # noqa: E501
if re is None:
print('Can not edit message.')
async def main(lib: TdLib, robot: TdLib):
with open('tdlib.json', 'r', encoding='UTF-8') as f: with open('tdlib.json', 'r', encoding='UTF-8') as f:
se = load(f) se = load(f)
if not await lib.login(se['TdlibParameters'], se['encryption_key'], if not await lib.login(se['TdlibParameters'], se['encryption_key'],
se.get("proxy"), se.get("phone_number")): se.get("proxy"), se.get("phone_number")):
raise ValueError('Can not login') raise ValueError('Can not login')
if se.get("bot_token") is not None:
paras = se['TdlibParameters'].copy()
if se.get("BotTdlibParameters"):
paras.update(se["BotTdlibParameters"])
ek = se['bot_encryption_key'] if se.get("bot_encryption_key") else se['encryption_key'] # noqa: E501
if not await robot.login(paras, ek, se.get("proxy"), bot_token=se["bot_token"]): # noqa: E501
raise ValueError('Can not login as bot')
uid = await lib.getUid() uid = await lib.getUid()
while True: while True:
mes = await lib.receive('updateNewMessage') mes = await lib.receive('updateNewMessage')
@@ -212,7 +297,6 @@ async def main(lib: TdLib):
e = e.result() e = e.result()
if e is None: if e is None:
print('Can not edit message.') print('Can not edit message.')
print(e)
if mes['content']['text']['text'] == '-hello': if mes['content']['text']['text'] == '-hello':
re = lib.editMessageText(mes['chat_id'], mes['id'], re = lib.editMessageText(mes['chat_id'], mes['id'],
'Hello World!') 'Hello World!')
@@ -223,9 +307,13 @@ async def main(lib: TdLib):
elif mes['content']['text']['text'].startswith('-'): elif mes['content']['text']['text'].startswith('-'):
argv = commandLineToArgv(mes['content']['text']['text']) argv = commandLineToArgv(mes['content']['text']['text'])
if argv[0] in ['-createnewstickerset', '-cnss']: if argv[0] in ['-createnewstickerset', '-cnss']:
re = handle_create_new_sticker_set(lib, mes, argv) re = handle_create_new_sticker_set(lib, robot, mes, argv)
asyncio.create_task(re)
elif argv[0] in ['-addstickertoset', '-asts']:
re = handle_add_sticker_to_set(lib, robot, mes, argv)
asyncio.create_task(re) asyncio.create_task(re)
with TdLib(True) as lib: with TdLib() as lib:
asyncio.run(main(lib)) with TdLib() as robot:
asyncio.run(main(lib, robot))