Přemysl Eric Janouch
50057d5149
I'm not entirely sure, but it looks like some people might not like jokes about the Holocaust. On a more serious note, the project has become more serious over the 7 or so years of its existence.
112 lines
3.1 KiB
Python
Executable File
112 lines
3.1 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
#
|
|
# xB YouTube plugin, displaying info about YouTube links
|
|
#
|
|
# Copyright 2014 - 2015, Přemysl Eric Janouch <p@janouch.name>
|
|
# 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 ()
|