Mikkel Munch Mortensen
3230a077d1
Add a new one for favourite dishes and introduce the possibitly to have replaceable tokens in introductions.
304 lines
9.5 KiB
Python
304 lines
9.5 KiB
Python
"""
|
|
Post today's menu to Slack.
|
|
|
|
This is done by parsing the JSON data from SharePoint.
|
|
"""
|
|
|
|
import datetime
|
|
import json
|
|
import os
|
|
import sys
|
|
import random
|
|
from urllib import error, parse, request
|
|
|
|
import browser_cookie3
|
|
from googletrans import Translator
|
|
import requests
|
|
from requests.exceptions import JSONDecodeError, ReadTimeout
|
|
|
|
|
|
# Load configuration.
|
|
try:
|
|
IS_LIVE = bool(os.environ.get("IS_LIVE"))
|
|
SLACK_ERROR_HOOK = os.environ["SLACK_ERROR_HOOK"]
|
|
SLACK_SUCCESS_HOOK = (
|
|
os.environ["SLACK_SUCCESS_HOOK"] if IS_LIVE else SLACK_ERROR_HOOK
|
|
)
|
|
SHAREPOINT_URL = os.environ["SHAREPOINT_URL"]
|
|
MENU_URL = os.environ["MENU_URL"]
|
|
except KeyError as e:
|
|
sys.stderr.write(f"Unable to load configuration for {e}.\n")
|
|
sys.exit(1)
|
|
sys.stdout.write("Configuration loaded.\n")
|
|
|
|
|
|
def get_photo_url(menu: str):
|
|
"""Retrieve a photo of the menu."""
|
|
# Translate menu from Danish to English.
|
|
try:
|
|
translation = Translator().translate(menu, src="da", dest="en").text
|
|
except TypeError:
|
|
sys.stderr.write("Unable to translate menu :(\n")
|
|
return None
|
|
sys.stdout.write(f"English translation of the menu is: {translation}\n")
|
|
|
|
# Ask Crayion to be creative.
|
|
cookies = browser_cookie3.firefox(domain_name="craiyon.com")
|
|
cookie = None
|
|
for cookie in cookies:
|
|
if cookie.name == "supabase-auth-token":
|
|
cookie = parse.unquote(cookie.value)
|
|
break
|
|
if cookie is None:
|
|
sys.stderr.write("Unable to extract Craiyon cookie.\n")
|
|
sys.exit(1)
|
|
token = json.loads(cookie)[0]
|
|
headers = {
|
|
"User-Agent": "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:109.0) Gecko/20100101 Firefox/111.0",
|
|
"Accept": "application/json",
|
|
"Accept-Language": "en-US,en;q=0.5",
|
|
"Accept-Encoding": "gzip, deflate, br",
|
|
"Content-Type": "application/json",
|
|
"Origin": "https://www.craiyon.com",
|
|
"Connection": "keep-alive",
|
|
"Sec-Fetch-Dest": "empty",
|
|
"Sec-Fetch-Mode": "cors",
|
|
"Sec-Fetch-Site": "same-site",
|
|
"Sec-GPC": "1",
|
|
"Pragma": "no-cache",
|
|
"Cache-Control": "no-cache",
|
|
"TE": "trailers",
|
|
}
|
|
|
|
request_data = {
|
|
"model": "photo",
|
|
"negative_prompt": "",
|
|
"prompt": translation,
|
|
"token": token,
|
|
"version": "35s5hfwn9n78gb06",
|
|
}
|
|
sys.stdout.write("Asking Craiyon to generate images for the menu...\n")
|
|
craiyon_start = datetime.datetime.now()
|
|
try:
|
|
response = requests.post(
|
|
"https://api.craiyon.com/v3",
|
|
headers=headers,
|
|
json=request_data,
|
|
timeout=90,
|
|
)
|
|
except ReadTimeout:
|
|
sys.stderr.write("Timeout while waiting for Craiyon :(\n")
|
|
return None
|
|
|
|
try:
|
|
image = random.choice(response.json()["images"])
|
|
except JSONDecodeError:
|
|
sys.stderr.write("Unable to parse JSON from Craiyon response:\n")
|
|
sys.stderr.write(f"{response.text}\n")
|
|
sys.stderr.write("Waited for {datetime.datetime.now() - craiyon_start}.")
|
|
return None
|
|
|
|
return (f"https://img.craiyon.com/{image}", translation)
|
|
|
|
|
|
# Set up and execute request to SharePoint.
|
|
cookies = browser_cookie3.firefox(domain_name="sharepoint.com")
|
|
headers = {
|
|
"Accept": "application/json;odata=verbose",
|
|
"Content-Type": "application/json;odata=verbose",
|
|
}
|
|
request_data = {
|
|
"parameters": {
|
|
"__metadata": {"type": "SP.RenderListDataParameters"},
|
|
"AddRequiredFields": True,
|
|
"AllowMultipleValueFilterForTaxonomyFields": True,
|
|
"FilterOutChannelFoldersInDefaultDocLib": True,
|
|
"RenderOptions": 5707271,
|
|
}
|
|
}
|
|
response = requests.post(
|
|
SHAREPOINT_URL,
|
|
headers=headers,
|
|
cookies=cookies,
|
|
json=request_data,
|
|
)
|
|
data = response.json()
|
|
sys.stdout.write("Data retrieved from SharePoint.\n")
|
|
if response.status_code != 200:
|
|
hook_url = SLACK_ERROR_HOOK
|
|
photo = None
|
|
message = (
|
|
plain_message
|
|
) = f"SharePoint responded with status {response.status_code} :("
|
|
else:
|
|
# Extract today's menu from the SharePoint data.
|
|
today = datetime.date.today().strftime("%d-%m-%Y")
|
|
menu = None
|
|
if "error" in data:
|
|
message = plain_message = f"Error: {data['error']['message']['value']}"
|
|
photo = None
|
|
hook_url = SLACK_ERROR_HOOK
|
|
sys.stderr.write(f"SharePoint {message}\n")
|
|
else:
|
|
hook_url = SLACK_SUCCESS_HOOK
|
|
for entry in data["ListData"]["Row"]:
|
|
if entry.get("DAto") == today:
|
|
menu = entry
|
|
break
|
|
|
|
if menu is None:
|
|
sys.stderr.write("Unable to find the menu of today :(\n")
|
|
sys.exit(1)
|
|
menu = menu["Menutekst"]
|
|
sys.stdout.write(f"The menu of today ({today}) is: {menu}\n")
|
|
|
|
# Determine appropriate emojis for the menu.
|
|
emojis = []
|
|
for emoji, keywords in {
|
|
"flag-in": ["indisk", "indien"],
|
|
"flag-gr": ["græsk", "grækenland"],
|
|
"flag-kr": ["korea"],
|
|
"hushed": ["surprise", "surprice"],
|
|
"fish": ["fisk", "laks", "rødspætte", "sej ", "kulmule"],
|
|
"shrimp": ["reje"],
|
|
"pig2": ["skinke", "gris", "nakkefilet", "flæsk"],
|
|
"cow": ["hakkebøf"],
|
|
"cow2": ["kalv", "okse"],
|
|
"chicken": ["kylling", "chicken"],
|
|
"turkey": ["kalkun"],
|
|
"rabbit2": [" hare"],
|
|
"rooster": ["coq au vin"],
|
|
"egg": [" æg ", " ægge"],
|
|
"butter": ["smør"],
|
|
"falafel": ["falafel"],
|
|
"hot_pepper": ["chili", "hot sauce"],
|
|
"onion": ["løg"],
|
|
"mango": ["mango"],
|
|
"lemon": ["citron"],
|
|
"mushroom": ["svampe", "kantarel", "champignon"],
|
|
"eggplant": ["moussaka", "mousakka"],
|
|
"cheese_wedge": [" ost"],
|
|
"beans": ["bønne"],
|
|
"olive": ["oliven"],
|
|
"spaghetti": ["spaghetti"],
|
|
"pizza": ["pizza"],
|
|
"hamburger": ["burger"],
|
|
"sandwich": ["sandwich"],
|
|
"stuffed_flatbread": ["pita"],
|
|
"flatbread": ["naanbrød", " naan"],
|
|
"pie": ["tærte"],
|
|
"hotdog": ["hotdog", "pølse"],
|
|
"fries": ["fritter "],
|
|
"wine_glass": ["coq au vin"],
|
|
"bowl_with_spoon": ["suppe"],
|
|
"fire": ["grill", "brændt", "bbq"],
|
|
"stew": ["gryde", "gullasch"],
|
|
"rice": [" ris "],
|
|
"ramen": ["nudler"],
|
|
"cloud": ["sky"],
|
|
"potato": ["kartoffel", "kartofler"],
|
|
"tomato": ["tomat"],
|
|
"apple": ["æble"],
|
|
"peanuts": ["peanut"],
|
|
"baguette_bread": ["flute"],
|
|
"birthday": ["tillykke", "fødselsdag"],
|
|
"wave": ["farvel"],
|
|
"gift_heart": ["valentines"],
|
|
}.items():
|
|
for keyword in keywords:
|
|
if keyword in f" {menu.lower()} ":
|
|
emojis.append(emoji)
|
|
break
|
|
if emojis:
|
|
emojis = f":{': :'.join(emojis)}:"
|
|
sys.stdout.write(f"Emojis determined: {emojis}\n")
|
|
else:
|
|
emojis = ""
|
|
sys.stdout.write("No emojis determined.\n")
|
|
|
|
# Pick an introduction for the menu.
|
|
tokens = {
|
|
"persons": random.choice([
|
|
"Abrahims",
|
|
"Elias'",
|
|
"Martins",
|
|
"Benjamins",
|
|
"Karstens",
|
|
"Josephines",
|
|
"Christians",
|
|
"Davids",
|
|
"Pias",
|
|
"Minas",
|
|
"Mariannes",
|
|
"Alexanders",
|
|
"Mettes",
|
|
"Jellings",
|
|
"Imers",
|
|
"Djinnies",
|
|
]),
|
|
}
|
|
introduction = random.choice(
|
|
[
|
|
"Dagens menu er",
|
|
"I dag forkæler kantinen os med",
|
|
"Du kan godt glæde dig til senere! For vi skal have",
|
|
"Der bliver knoklet i køkkenet for at blive klar til at servere",
|
|
"Klokken 11:30 har kantinen fremtryllet en lækker omgang",
|
|
"I dag skal vi have {persons} livret:",
|
|
]
|
|
)
|
|
introduction = introduction.format(**tokens)
|
|
sys.stdout.write(f"Introduction picked: {introduction}\n")
|
|
|
|
# Retrieve a photo of the menu from Craiyon.
|
|
photo = get_photo_url(menu)
|
|
|
|
# Compose message for Slack.
|
|
plain_message = f"{introduction} {menu}"
|
|
menu_url = MENU_URL
|
|
message = f"{introduction} {emojis} <{menu_url}|*{menu}*>".strip()
|
|
|
|
payload = {
|
|
"text": plain_message,
|
|
"blocks": [
|
|
{
|
|
"type": "section",
|
|
"text": {
|
|
"type": "mrkdwn",
|
|
"text": message,
|
|
},
|
|
},
|
|
],
|
|
}
|
|
|
|
if photo is not None:
|
|
photo_text = "Et ukvalificeret bud på hvordan dagens menu kunne se ud"
|
|
payload["blocks"].insert(
|
|
0,
|
|
{
|
|
"type": "image",
|
|
"image_url": photo[0],
|
|
"alt_text": photo_text,
|
|
"title": {
|
|
"type": "plain_text",
|
|
"text": photo_text,
|
|
},
|
|
},
|
|
)
|
|
|
|
sys.stdout.write("Posting menu to Slack...\n")
|
|
hook = request.Request(
|
|
hook_url,
|
|
data=json.dumps(payload).encode("utf-8"),
|
|
headers={
|
|
"Content-Type": "application/json",
|
|
},
|
|
method="POST",
|
|
)
|
|
try:
|
|
response = request.urlopen(hook)
|
|
except error.HTTPError as e:
|
|
sys.stderr.write(f"{e}\n")
|
|
sys.exit(1)
|
|
sys.stdout.write("Done :)\n")
|