Come up with sillier names for the binaries
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.
This commit is contained in:
		
							
								
								
									
										111
									
								
								plugins/xB/youtube
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										111
									
								
								plugins/xB/youtube
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,111 @@
 | 
			
		||||
#!/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 ()
 | 
			
		||||
		Reference in New Issue
	
	Block a user