Compare commits
2 commits
91deeca19b
...
65f9e8494a
Author | SHA1 | Date | |
---|---|---|---|
dece | 65f9e8494a | ||
dece | 404f5bb673 |
|
@ -148,7 +148,8 @@
|
|||
"negative": ["I don't like it."]
|
||||
},
|
||||
"playlistoftheday": {
|
||||
"commands": ["playlist of the day"]
|
||||
"commands": ["playlist of the day"],
|
||||
"not_fresh_reply": "It's not from today but here it is: {url}"
|
||||
},
|
||||
"plus": {
|
||||
"commands": ["plus"],
|
||||
|
|
|
@ -162,7 +162,10 @@ class Bot(irc.client.SimpleIRCClient, Logger):
|
|||
|
||||
def get_plugin(self, name: str) -> Optional[Plugin]:
|
||||
"""Get a loaded plugin by its name (e.g. 'mood'), or None."""
|
||||
matching_plugins = filter(lambda plugin: plugin.name == name, self.plugins)
|
||||
matching_plugins = filter(
|
||||
lambda plugin: plugin.name == name,
|
||||
self.plugins,
|
||||
)
|
||||
return next(matching_plugins, None)
|
||||
|
||||
def say(self, target: str, message: str) -> None:
|
||||
|
|
|
@ -1,11 +1,25 @@
|
|||
import datetime
|
||||
import re
|
||||
from typing import cast, Optional
|
||||
|
||||
from edmond.plugin import Plugin
|
||||
from edmond.plugins.shrlok import ShrlokPlugin
|
||||
|
||||
|
||||
HTML_TEMPLATE = """\
|
||||
<html>
|
||||
<head> <meta charset="utf-8"> <title>🎶</title> </head>
|
||||
<body> {} </body>
|
||||
</html>
|
||||
"""
|
||||
LINK_RE = re.compile(r"(https?://\S+)")
|
||||
|
||||
|
||||
class PlaylistOfTheDayPlugin(Plugin):
|
||||
"""Collect music links from other platforms.
|
||||
|
||||
This plugin requires a working Shrlok instance to post the playlist.
|
||||
|
||||
Plugins that want to feed the playlist must do so by calling the add_link method of
|
||||
the plugin. Later this plugin may feed its playlist using links unhandled by other,
|
||||
more specialized plugins (bandcamp, soundcloud, …).
|
||||
|
@ -27,12 +41,26 @@ class PlaylistOfTheDayPlugin(Plugin):
|
|||
|
||||
def __init__(self, bot):
|
||||
super().__init__(bot)
|
||||
self._shrlok_plugin = None
|
||||
|
||||
@property
|
||||
def shrlok_plugin(self) -> Optional[ShrlokPlugin]:
|
||||
if self._shrlok_plugin is None:
|
||||
self._shrlok_plugin = cast(
|
||||
ShrlokPlugin,
|
||||
self.bot.get_plugin("shrlok"),
|
||||
)
|
||||
return self._shrlok_plugin
|
||||
|
||||
def on_welcome(self, _):
|
||||
if not (self.shrlok_plugin and self.shrlok_plugin.is_ready):
|
||||
self.bot.log_w("Shrlok plugin is not available.")
|
||||
self.is_ready = False
|
||||
|
||||
def on_pubmsg(self, event):
|
||||
if not self.should_handle_command(event.arguments[0], no_content=True):
|
||||
return False
|
||||
|
||||
# self.post_playlist(event.target)
|
||||
self.post_playlist(event.target)
|
||||
return True
|
||||
|
||||
def add_line(self, line: str) -> None:
|
||||
|
@ -62,5 +90,29 @@ class PlaylistOfTheDayPlugin(Plugin):
|
|||
today.isoformat(),
|
||||
)
|
||||
|
||||
# def post_playlist(self):
|
||||
# pass
|
||||
def post_playlist(self, target):
|
||||
playlist: list[str] = self.get_storage_value(self.PLAYLIST_KEY, [])
|
||||
if not playlist:
|
||||
self.bot.log_e("Playlist empty.")
|
||||
self.signal_failure(target)
|
||||
|
||||
linkified_items = map(PlaylistOfTheDayPlugin.linkify, playlist)
|
||||
html_items = map(lambda item: f"<li>{item}</li>", linkified_items)
|
||||
html_list = "<ol>" + "".join(html_items) + "</ol>"
|
||||
data = HTML_TEMPLATE.format(html_list).encode()
|
||||
url = self.shrlok_plugin.post({"type": "raw"}, data)
|
||||
if not url:
|
||||
self.bot.log_e("Shrlok returned None.")
|
||||
self.signal_failure(target)
|
||||
|
||||
date = self.get_storage_value(self.DATE_KEY, "")
|
||||
if date != datetime.date.today().isoformat():
|
||||
reply = self.config["not_fresh_reply"].format(url=url)
|
||||
else:
|
||||
reply = url
|
||||
self.bot.say(target, reply)
|
||||
|
||||
@staticmethod
|
||||
def linkify(text: str) -> str:
|
||||
"""Put links in A tags."""
|
||||
return LINK_RE.sub(r'<a href="\1">\1</a>', text)
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
import json
|
||||
import socket
|
||||
from typing import Optional
|
||||
|
||||
from edmond.bot import Bot
|
||||
from edmond.plugin import Plugin
|
||||
|
@ -31,7 +32,7 @@ class ShrlokPlugin(Plugin):
|
|||
self.bot.log_d("No socket path specified, shrlok plugin disabled.")
|
||||
self.is_ready = False
|
||||
|
||||
def post(self, header: dict, data: bytes):
|
||||
def post(self, header: dict, data: bytes) -> Optional[str]:
|
||||
encoded_header = json.dumps(header).encode()
|
||||
try:
|
||||
with socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) as sock:
|
||||
|
@ -45,4 +46,4 @@ class ShrlokPlugin(Plugin):
|
|||
self.bot.log_e(f"Can't post data: {exc}")
|
||||
return None
|
||||
url = response.decode().replace(self.file_root, self.url_root, 1)
|
||||
return url
|
||||
return url or None # returning empty strings could cause confusion
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
from typing import cast, Optional
|
||||
|
||||
try:
|
||||
from googleapiclient.discovery import build as gapi_discovery_build
|
||||
from googleapiclient.errors import Error as GoogleApiError
|
||||
|
@ -7,6 +9,7 @@ except ImportError:
|
|||
DEPENDENCIES_FOUND = False
|
||||
|
||||
from edmond.plugin import Plugin
|
||||
from edmond.plugins.playlist_of_the_day import PlaylistOfTheDayPlugin
|
||||
|
||||
|
||||
class YoutubePlugin(Plugin):
|
||||
|
@ -19,21 +22,33 @@ class YoutubePlugin(Plugin):
|
|||
def __init__(self, bot):
|
||||
super().__init__(bot)
|
||||
self._youtube = None
|
||||
self._playlist_of_the_day_plugin = None
|
||||
|
||||
@property
|
||||
def youtube(self):
|
||||
if self._youtube is None:
|
||||
self._youtube = gapi_discovery_build(
|
||||
"youtube", "v3", developerKey=self.config["api_key"]
|
||||
"youtube",
|
||||
"v3",
|
||||
developerKey=self.config["api_key"],
|
||||
)
|
||||
return self._youtube
|
||||
|
||||
def has_api_key(self):
|
||||
return self.config["api_key"] != ""
|
||||
@property
|
||||
def playlist_of_the_day_plugin(self) -> Optional[PlaylistOfTheDayPlugin]:
|
||||
if self._playlist_of_the_day_plugin is None:
|
||||
self._playlist_of_the_day_plugin = cast(
|
||||
PlaylistOfTheDayPlugin,
|
||||
self.bot.get_plugin("playlistoftheday"),
|
||||
)
|
||||
return self._playlist_of_the_day_plugin
|
||||
|
||||
def on_welcome(self, _):
|
||||
if not self.config["api_key"]:
|
||||
self.bot.log_w("API key unavailable.")
|
||||
self.is_ready = False
|
||||
|
||||
def on_pubmsg(self, event):
|
||||
if not self.has_api_key():
|
||||
return False
|
||||
if self.should_handle_command(event.arguments[0]):
|
||||
self.handle_commands(event.target)
|
||||
return True
|
||||
|
@ -80,3 +95,6 @@ class YoutubePlugin(Plugin):
|
|||
self.signal_failure(target)
|
||||
return
|
||||
self.bot.say(target, f"{icon} {link} {title}")
|
||||
|
||||
if self.playlist_of_the_day_plugin:
|
||||
self.playlist_of_the_day_plugin.add_line(f"{link} {title}")
|
||||
|
|
Loading…
Reference in a new issue