Skip to content

Instantly share code, notes, and snippets.

@doxy-ai
Created October 28, 2023 01:26
Show Gist options
  • Save doxy-ai/f605f31922051151dc044494d1d9af56 to your computer and use it in GitHub Desktop.
Save doxy-ai/f605f31922051151dc044494d1d9af56 to your computer and use it in GitHub Desktop.
ZatsuDachi Kick Plugin (readonly)
# Requirements: pip install tls_client
# Import required packages
from api import PluginBase, Message, Color
from flask import escape
import asyncio
import websockets
import json
import sys
import tls_client
import re
# Define the plugin class
# Based on: https://github.com/mumixam/kick.com-chatlogger-python
# Updated with code from: https://github.com/lukemvc/kickbot
class Plugin(PluginBase):
# Set plugin name
name = "Kick Chat Plugin"
# Set image URL for the plugin
pluginImageURL = ""
# Reference to the current websocket
_ws = None
# CONNECTION NOTE: Paste the name of your livestream channel here!
targetChannel = "pandaskills"
async def go(self):
if self._ws is not None: return
slug = self.targetChannel.replace('_', '-')
url = f"https://kick.com/api/v2/channels/{slug}"
api = tls_client.Session(
client_identifier="chrome_116",
random_tls_extension_order=True
)
response = api.get(url, headers=self._BASE_HEADERS)
status = response.status_code
match status:
case 403 | 429:
raise RuntimeError(f"Error retrieving streamer info. Blocked By cloudflare. ({status})")
case 404:
raise RuntimeError(f"Streamer info for '{self.targetChannel}' not found. (404 error) ")
data = response.json()
channelID = data["chatroom"]["id"]
# TODO: Need to register badges
self._ws = await websockets.connect("wss://ws-us2.pusher.com/app/eb1d5f283081a78b932c?protocol=7&client=js&version=7.6.0&flash=false")
connection_response = json.loads(await self._ws.recv())
socketId = json.loads(connection_response.get("data"))["socket_id"]
await self._ws.send('{"event":"pusher:subscribe","data":{"auth":"","channel":"chatrooms.' + str(channelID) + '.v2"}}')
join_response = json.loads(await self._ws.recv())
print(f"Connected to Kick channel '{self.targetChannel}'")
while self._keepAlive:
self.on_message(await self._ws.recv())
def stop(self):
"""
Stop the plugin and halt the websocket listener
"""
PluginBase.stop(self)
# Plugin._lock.acquire()
try:
if Plugin._ws is not None: Plugin._ws.close()
Plugin._ws = None
finally:
# Plugin._lock.release()
print("Kick stopped! Waiting for connection to timeout...")
def on_message(self, message):
msg = json.loads(message)
if 'ChatMessageEvent' in msg['event']:
print(msg)
data = json.loads(msg['data'])
content = data["content"]
if not content:
pass
user = data['sender']['username']
color = Color(data['sender']['identity']['color'])
badges = data['sender']['identity']['badges']
badges = [badge["type"] for badge in badges]
emotes = re.findall(self._pattern, content)
for (sub, id, text) in emotes:
self.register_emote(text, f"https://files.kick.com/emotes/{id}/fullsize")
content = content.replace(sub, text)
# Create a Message object with the constructed content, sender information, color, and plugin image URL
self.recieve_message(Message(content=content, sender=user, senderColor=color, senderBadges=badges, pluginImageUrl=self.pluginImageURL))
_BASE_HEADERS = {
"Accept": "application/json, text/plain, */*",
"Accept-Encoding": "gzip, deflate, br",
"Accept-Language": "en-US,en;q=0.9",
"Sec-Fetch-Dest": "empty",
"Sec-Fetch-Mode": "cors",
"Sec-Fetch-Site": "same-origin",
"User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 "
"(KHTML, like Gecko) Chrome/116.0.0.0 Safari/537.36",
}
_pattern = r'(\[emote:(\d*?):(\S*?)\])'
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment