#!/usr/bin/env python3 # # ZyklonB YouTube plugin, displaying info about YouTube links # # Copyright 2014 - 2015, Přemysl Janouch # See the file LICENSE for licensing information. # import sys import io import re import json import urllib.request class Plugin: re_msg = re.compile ('(?::([^! ]*)(?:!([^@]*)@([^ ]*))? +)?' '([^ ]+)(?: +(.*))?\r\n$') re_args = re.compile (':?((?<=:).*|[^ ]+) *') def parse (self, line): m = self.re_msg.match (line) if m is None: return None (nick, user, host, command, args) = m.groups () args = [] if args is None else self.re_args.findall (args) return (nick, user, host, command, args) def get_config (self, key): print ("ZYKLONB get_config :%s" % key) (_, _, _, _, args) = self.parse (sys.stdin.readline ()) return args[0] def bot_print (self, what): print ('ZYKLONB print :%s' % what) class YouTube (Plugin): re_videos = [re.compile (x) for x in [ r'youtube\.[a-z]+/[^ ]*[&?]v=([-\w]+)', r'youtube\.[a-z]+/v/([-\w]+)', r'youtu\.be/([-\w]+)' ]] re_playlists = [re.compile (x) for x in [ r'youtube\.[a-z]+/playlist[&?][^ ]*(?<=&|\?)list=([-\w]+)', ]] def print_info (self, channel, url, cb): try: data = json.loads (urllib.request.urlopen (url, None, 30).read ().decode ('utf-8')) for line in map (lambda x: "YouTube: " + cb (x), data['items']): print ("PRIVMSG %s :%s" % (channel, line.encode ('utf-8').decode ('iso8859-1'))) except Exception as err: self.bot_print ('youtube: %s' % (err)) def print_video_info (self, channel, video_id): url = 'https://www.googleapis.com/youtube/v3/' \ + 'videos?id=%s&key=%s&part=snippet,contentDetails,statistics' \ % (video_id, self.youtube_api_key) self.print_info (channel, url, lambda x: "%s | %s | %sx" % ( x['snippet']['title'], x['contentDetails']['duration'][2:].lower (), x['statistics']['viewCount'])) def print_playlist_info (self, channel, playlist_id): url = 'https://www.googleapis.com/youtube/v3/' \ + 'playlists?id=%s&key=%s&part=snippet,contentDetails' \ % (playlist_id, self.youtube_api_key) self.print_info (channel, url, lambda x: "%s | %d videos" % ( x['snippet']['title'], x['contentDetails']['itemCount'])) def process_line (self, line): msg = self.parse (line) if msg is None: return (nick, user, host, command, args) = msg if command != 'PRIVMSG' or len (args) < 2: return ctx = args[0] if not ctx.startswith (('#', '+', '&', '!')): ctx = nick for regex in self.re_videos: for i in regex.findall (args[1]): self.print_video_info (ctx, i) for regex in self.re_playlists: for i in regex.findall (args[1]): self.print_playlist_info (ctx, i) def run (self): self.youtube_api_key = self.get_config ('youtube_api_key') if self.youtube_api_key == "": self.bot_print ("youtube: missing `youtube_api_key'") print ("ZYKLONB register") for line in sys.stdin: self.process_line (line) sys.stdin = io.TextIOWrapper (sys.__stdin__.buffer, encoding = 'iso8859-1', newline = '\r\n', line_buffering = True) sys.stdout = io.TextIOWrapper (sys.__stdout__.buffer, encoding = 'iso8859-1', newline = '\r\n', line_buffering = True) YouTube ().run ()