Initial Commit
XiguaLiveDanmakuHelper
This commit is contained in:
commit
6719164b4e
336
api.py
Normal file
336
api.py
Normal file
@ -0,0 +1,336 @@
|
||||
import sys
|
||||
|
||||
import requests
|
||||
import time
|
||||
import ctypes
|
||||
import os
|
||||
|
||||
|
||||
class UserStruct:
|
||||
ID = 0
|
||||
name = ""
|
||||
brand= ""
|
||||
level= 0
|
||||
type = 0
|
||||
|
||||
def __init__(self, json=None):
|
||||
if json:
|
||||
self.parse(json)
|
||||
|
||||
def parse(self, json):
|
||||
if "Msg" in json:
|
||||
if "user" in json["Msg"]:
|
||||
self.ID = json["Msg"]['user']['user_id']
|
||||
self.name = json["Msg"]['user']['name']
|
||||
self.type = json["Msg"]['user']['user_type']
|
||||
if "discipulus_info" in json["Msg"]:
|
||||
self.level = json["Msg"]["discipulus_info"]["level"]
|
||||
self.brand = json["Msg"]["discipulus_info"]["discipulus_group_title"]
|
||||
if self.type is None:
|
||||
self.type = 0
|
||||
|
||||
def __str__(self):
|
||||
if self.level == 0:
|
||||
if self.type != 0:
|
||||
return "[]{}".format(self.name)
|
||||
return "{}".format(self.name)
|
||||
else:
|
||||
if self.type != 0:
|
||||
return "[{}{}]{}".format(self.brand, self.level, self.name)
|
||||
return "<{}{}>{}".format(self.brand,self.level,self.name)
|
||||
|
||||
|
||||
def __unicode__(self):
|
||||
return self.__str__()
|
||||
|
||||
|
||||
class GiftStruct:
|
||||
ID = 0
|
||||
count = 0
|
||||
|
||||
def __init__(self, json=None):
|
||||
if json:
|
||||
self.parse(json)
|
||||
|
||||
def parse(self, json):
|
||||
if "Msg" in json:
|
||||
if "present_end_info" in json["Msg"]:
|
||||
self.ID = json["Msg"]['present_end_info']['id']
|
||||
self.count = json["Msg"]['present_end_info']['count']
|
||||
elif "present_info" in json["Msg"]:
|
||||
self.ID = json["Msg"]['present_info']['id']
|
||||
self.count = json["Msg"]['present_info']['repeat_count']
|
||||
|
||||
|
||||
def readInput(caption, default, timeout=5):
|
||||
start_time = time.time()
|
||||
print('{}({})\r\n>'.format(caption,default), end="")
|
||||
input = ''
|
||||
while True:
|
||||
if msvcrt.kbhit():
|
||||
chr = msvcrt.getche()
|
||||
if ord(chr) == 13: # enter_key
|
||||
break
|
||||
elif ord(chr) == 27:
|
||||
break
|
||||
elif ord(chr) >= 32: # space_char
|
||||
input += str(chr)
|
||||
if len(input) == 0 and (time.time() - start_time) > timeout:
|
||||
break
|
||||
|
||||
if len(input) > 0:
|
||||
print()
|
||||
return input
|
||||
else:
|
||||
print("使用默认值")
|
||||
return default
|
||||
|
||||
|
||||
STD_INPUT_HANDLE = -10
|
||||
STD_OUTPUT_HANDLE = -11
|
||||
STD_ERROR_HANDLE = -12
|
||||
std_out_handle = ctypes.windll.kernel32.GetStdHandle(STD_OUTPUT_HANDLE)
|
||||
# 字体颜色定义 ,关键在于颜色编码,由2位十六进制组成,分别取0~f,前一位指的是背景色,后一位指的是字体色
|
||||
#由于该函数的限制,应该是只有这16种,可以前景色与背景色组合。也可以几种颜色通过或运算组合,组合后还是在这16种颜色中
|
||||
|
||||
# Windows CMD命令行 字体颜色定义 text colors
|
||||
FOREGROUND_BLACK = 0x00 # black.
|
||||
FOREGROUND_DARKBLUE = 0x01 # dark blue.
|
||||
FOREGROUND_DARKGREEN = 0x02 # dark green.
|
||||
FOREGROUND_DARKSKYBLUE = 0x03 # dark skyblue.
|
||||
FOREGROUND_DARKRED = 0x04 # dark red.
|
||||
FOREGROUND_DARKPINK = 0x05 # dark pink.
|
||||
FOREGROUND_DARKYELLOW = 0x06 # dark yellow.
|
||||
FOREGROUND_DARKWHITE = 0x07 # dark white.
|
||||
FOREGROUND_DARKGRAY = 0x08 # dark gray.
|
||||
FOREGROUND_BLUE = 0x09 # blue.
|
||||
FOREGROUND_GREEN = 0x0a # green.
|
||||
FOREGROUND_SKYBLUE = 0x0b # skyblue.
|
||||
FOREGROUND_RED = 0x0c # red.
|
||||
FOREGROUND_PINK = 0x0d # pink.
|
||||
FOREGROUND_YELLOW = 0x0e # yellow.
|
||||
FOREGROUND_WHITE = 0x0f # white.
|
||||
|
||||
|
||||
# Windows CMD命令行 背景颜色定义 background colors
|
||||
BACKGROUND_BLUE = 0x10 # dark blue.
|
||||
BACKGROUND_GREEN = 0x20 # dark green.
|
||||
BACKGROUND_DARKSKYBLUE = 0x30 # dark skyblue.
|
||||
BACKGROUND_DARKRED = 0x40 # dark red.
|
||||
BACKGROUND_DARKPINK = 0x50 # dark pink.
|
||||
BACKGROUND_DARKYELLOW = 0x60 # dark yellow.
|
||||
BACKGROUND_DARKWHITE = 0x70 # dark white.
|
||||
BACKGROUND_DARKGRAY = 0x80 # dark gray.
|
||||
BACKGROUND_BLUE = 0x90 # blue.
|
||||
BACKGROUND_GREEN = 0xa0 # green.
|
||||
BACKGROUND_SKYBLUE = 0xb0 # skyblue.
|
||||
BACKGROUND_RED = 0xc0 # red.
|
||||
BACKGROUND_PINK = 0xd0 # pink.
|
||||
BACKGROUND_YELLOW = 0xe0 # yellow.
|
||||
BACKGROUND_WHITE = 0xf0 # white.
|
||||
|
||||
|
||||
s = requests.Session()
|
||||
|
||||
def set_cmd_text_color(color, handle=std_out_handle):
|
||||
Bool = ctypes.windll.kernel32.SetConsoleTextAttribute(handle, color)
|
||||
return Bool
|
||||
|
||||
def resetColor():
|
||||
set_cmd_text_color(FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE)
|
||||
|
||||
class XiGuaLiveApi:
|
||||
isLive = False
|
||||
roomInfo = {}
|
||||
roomID = 0
|
||||
cursor = ""
|
||||
giftList = {10001:"西瓜"}
|
||||
|
||||
def __init__(self, room: int):
|
||||
self.room = room
|
||||
self.updRoomInfo()
|
||||
self.updGiftList()
|
||||
|
||||
def notLiveError(self):
|
||||
print("主播未开播")
|
||||
|
||||
def apiChangedError(self, msg:str):
|
||||
print(msg)
|
||||
|
||||
def onPresent(self, user:UserStruct, gift:GiftStruct):
|
||||
if gift.ID not in self.giftList:
|
||||
giftN = "未知礼物:{}".format(gift.ID)
|
||||
else:
|
||||
giftN = self.giftList[gift.ID]
|
||||
return
|
||||
print("礼物连击:", user, giftN, "x", gift.count)
|
||||
|
||||
def onPresentEnd(self, user:UserStruct, gift:GiftStruct):
|
||||
if gift.ID not in self.giftList:
|
||||
self.updGiftList()
|
||||
giftN = "未知礼物:{}".format(gift.ID)
|
||||
else:
|
||||
giftN = self.giftList[gift.ID]
|
||||
set_cmd_text_color(BACKGROUND_WHITE | FOREGROUND_BLACK)
|
||||
print("感谢", user, "送出的", giftN, "x", gift.count)
|
||||
resetColor()
|
||||
|
||||
def onAd(self, i):
|
||||
# print(i)
|
||||
pass
|
||||
|
||||
def onChat(self, user:UserStruct, content:str):
|
||||
print(user, ":", content)
|
||||
# pass
|
||||
|
||||
def onEnter(self, user:UserStruct, content:str == ""):
|
||||
if content == "":
|
||||
if user.name == "三国空白" or user.name == "四维v":
|
||||
set_cmd_text_color(FOREGROUND_DARKGRAY)
|
||||
print("消息:", user, "进入直播间")
|
||||
resetColor()
|
||||
else:
|
||||
set_cmd_text_color(FOREGROUND_DARKGRAY)
|
||||
print("消息:", content.format(user))
|
||||
resetColor()
|
||||
|
||||
def onSubscribe(self, user:UserStruct):
|
||||
if user.level >= 6 and user.brand == "永恒":
|
||||
set_cmd_text_color(FOREGROUND_DARKGRAY)
|
||||
print("消息:", user, "关注了主播")
|
||||
resetColor()
|
||||
|
||||
def onJoin(self, user:UserStruct):
|
||||
set_cmd_text_color(BACKGROUND_WHITE | FOREGROUND_BLACK)
|
||||
print("感谢", user, "加入了粉丝团")
|
||||
resetColor()
|
||||
|
||||
def onMessage(self, msg:str):
|
||||
set_cmd_text_color(FOREGROUND_DARKGRAY)
|
||||
print("消息:", msg)
|
||||
resetColor()
|
||||
|
||||
def onLike(self, user:UserStruct):
|
||||
return
|
||||
# set_cmd_text_color(FOREGROUND_DARKGRAY)
|
||||
# print("用户", user, "点了喜欢")
|
||||
# resetColor()
|
||||
|
||||
def onLeave(self, json:any):
|
||||
print("消息:", "主播离开一小会")
|
||||
self.debug(json)
|
||||
return
|
||||
|
||||
def updGiftList(self):
|
||||
p = s.get("https://live.ixigua.com/api/gifts/{roomID}".format(roomID=self.roomID))
|
||||
d = p.json()
|
||||
self.debug(d)
|
||||
if "data" not in d:
|
||||
self.warning("Warning: Api Has Changed")
|
||||
return
|
||||
for i in d["data"]:
|
||||
self.debug(i["ID"], i["Name"])
|
||||
self.giftList[i["ID"]] = i["Name"]
|
||||
|
||||
def warning(self, *args):
|
||||
print(args)
|
||||
|
||||
def debug(self, *args):
|
||||
# print(args)
|
||||
pass
|
||||
|
||||
def enterRoom(self):
|
||||
p = s.post("https://live.ixigua.com/api/room/enter/{roomID}".format(roomID=self.roomID))
|
||||
self.debug(p.json())
|
||||
|
||||
def updRoomInfo(self):
|
||||
p = s.get("https://live.ixigua.com/api/room/{room}".format(room=self.room))
|
||||
d = p.json()
|
||||
self.debug(d)
|
||||
if "data" not in d:
|
||||
self.apiChangedError("数据结构改变,请与我联系")
|
||||
self.debug(d)
|
||||
return
|
||||
self.roomInfo = d["data"]
|
||||
print("进入", self.roomInfo["anchorInfo"]["name"], "的直播间")
|
||||
if "Id" in d["data"]:
|
||||
self.roomID = d["data"]["Id"]
|
||||
else:
|
||||
self.warning("无法获取RoomID,请与我联系")
|
||||
if "FinishTime" in d["data"]:
|
||||
self.isLive = False
|
||||
self.notLiveError()
|
||||
else:
|
||||
self.isLive = True
|
||||
|
||||
def getDanmaku(self):
|
||||
p = s.get("https://live.ixigua.com/api/msg/list/{roomID}?AnchorID={room}&Cursor={cursor}".format(
|
||||
roomID=self.roomID,
|
||||
room=self.room,
|
||||
cursor=self.cursor
|
||||
))
|
||||
d = p.json()
|
||||
self.debug(d)
|
||||
if "data" not in d:
|
||||
self.apiChangedError("数据结构改变,请与我联系")
|
||||
self.debug(d)
|
||||
return
|
||||
if "Extra" not in d["data"]:
|
||||
self.apiChangedError("数据结构改变,请与我联系")
|
||||
self.debug(d["data"])
|
||||
return
|
||||
if "Cursor" not in d["data"]["Extra"]:
|
||||
self.apiChangedError("数据结构改变,请与我联系")
|
||||
self.debug(d["data"])
|
||||
return
|
||||
else:
|
||||
self.cursor = d["data"]["Extra"]["Cursor"]
|
||||
if "LiveMsgs" not in d["data"]:
|
||||
return
|
||||
for i in d['data']['LiveMsgs']:
|
||||
if i['Method'] == "VideoLivePresentMessage":
|
||||
self.onPresent(UserStruct(i), GiftStruct(i))
|
||||
elif i['Method'] == "VideoLivePresentEndTipMessage":
|
||||
self.onPresentEnd(UserStruct(i), GiftStruct(i))
|
||||
elif i['Method'] == "VideoLiveRoomAdMessage":
|
||||
self.onAd(i)
|
||||
elif i['Method'] == "VideoLiveChatMessage":
|
||||
self.onChat(UserStruct(i), i["Msg"]['content'])
|
||||
elif i['Method'] == "VideoLiveMemberMessage":
|
||||
self.onEnter(UserStruct(i), i["Msg"]["content"])
|
||||
elif i['Method'] == "VideoLiveSocialMessage":
|
||||
self.onSubscribe(UserStruct(i))
|
||||
elif i['Method'] == "VideoLiveJoinDiscipulusMessage":
|
||||
self.onJoin(UserStruct(i))
|
||||
elif i['Method'] == "VideoLiveControlMessage":
|
||||
print("消息:", "主播离开一小会")
|
||||
elif i['Method'] == "VideoLiveDiggMessage":
|
||||
self.onLike(UserStruct(i))
|
||||
else:
|
||||
self.debug(i)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
room = 97621754276 #永恒
|
||||
# room = 75366565294
|
||||
# room = 83940182312 #Dae
|
||||
if len(sys.argv)>1:
|
||||
room = int(sys.argv[1])
|
||||
resetColor()
|
||||
print("西瓜直播弹幕助手 by JerryYan")
|
||||
print("正在进入房间", room)
|
||||
api = XiGuaLiveApi(room)
|
||||
os.system("title {} {}".format(api.roomInfo["anchorInfo"]["name"],"的直播间 --西瓜直播弹幕助手 by JerryYan"))
|
||||
api.enterRoom()
|
||||
print("="*30)
|
||||
while True:
|
||||
if api.isLive:
|
||||
try:
|
||||
api.getDanmaku()
|
||||
except Exception as e:
|
||||
api.warning(e)
|
||||
time.sleep(1)
|
||||
else:
|
||||
print("主播未开播,等待1分钟后重试")
|
||||
time.sleep(60)
|
||||
api.updRoomInfo()
|
Reference in New Issue
Block a user