Files
weekly-calendar/server.py
Nathan Woodburn a546da0191
All checks were successful
Build Docker / BuildImage (push) Successful in 48s
feat: Get data from google sheets
2025-09-29 21:32:49 +10:00

197 lines
5.7 KiB
Python

from functools import cache
import json
from flask import (
Flask,
make_response,
redirect,
request,
jsonify,
render_template,
send_from_directory,
send_file,
)
import os
import json
import requests
from datetime import datetime
import dotenv
import csv
from io import StringIO
dotenv.load_dotenv()
def load_schedule_data():
"""Load schedule data from Google Sheets or fallback to JSON file"""
# Try to load from Google Sheets first
google_sheet_url = os.getenv('GOOGLE_SHEET_URL')
if google_sheet_url:
try:
# Convert Google Sheets URL to CSV export URL
if '/edit' in google_sheet_url:
csv_url = google_sheet_url.replace('/edit#gid=', '/export?format=csv&gid=')
csv_url = csv_url.replace('/edit', '/export?format=csv')
else:
csv_url = google_sheet_url
print(f"Fetching schedule from Google Sheets: {csv_url}")
response = requests.get(csv_url, timeout=10)
response.raise_for_status()
# Parse CSV data
csv_data = StringIO(response.text)
reader = csv.DictReader(csv_data)
schedule = []
for row in reader:
# Map CSV columns to our expected format
# Expected columns: Date, Primary Leader, Secondary Leader, Topic
schedule_item = {
"date": row.get('Date', '').strip(),
"primary_leader": row.get('Primary Leader', '').strip(),
"secondary_leader": row.get('Secondary Leader', '').strip(),
"topic": row.get('Topic', '').strip()
}
# Only add rows that have at least a date and topic
if schedule_item['date'] and schedule_item['topic']:
schedule.append(schedule_item)
print(f"Successfully loaded {len(schedule)} items from Google Sheets")
return schedule
except requests.RequestException as e:
print(f"Error fetching from Google Sheets: {e}")
except Exception as e:
print(f"Error parsing Google Sheets data: {e}")
# Fallback to JSON file
try:
with open('schedule_data.json', 'r') as f:
data = json.load(f)
print("Loaded schedule from local JSON file")
return data.get('schedule', [])
except FileNotFoundError:
print("Warning: schedule_data.json not found. Using empty schedule.")
return []
except json.JSONDecodeError:
print("Warning: Invalid JSON in schedule_data.json. Using empty schedule.")
return []
# Load schedule data from JSON file
SCHEDULE_DATA = load_schedule_data()
app = Flask(__name__)
def find(name, path):
for root, dirs, files in os.walk(path):
if name in files:
return os.path.join(root, name)
# Assets routes
@app.route("/assets/<path:path>")
def send_assets(path):
if path.endswith(".json"):
return send_from_directory(
"templates/assets", path, mimetype="application/json"
)
if os.path.isfile("templates/assets/" + path):
return send_from_directory("templates/assets", path)
# Try looking in one of the directories
filename: str = path.split("/")[-1]
if (
filename.endswith(".png")
or filename.endswith(".jpg")
or filename.endswith(".jpeg")
or filename.endswith(".svg")
):
if os.path.isfile("templates/assets/img/" + filename):
return send_from_directory("templates/assets/img", filename)
if os.path.isfile("templates/assets/img/favicon/" + filename):
return send_from_directory("templates/assets/img/favicon", filename)
return render_template("404.html"), 404
# region Special routes
@app.route("/favicon.png")
def faviconPNG():
return send_from_directory("templates/assets/img", "favicon.png")
@app.route("/.well-known/<path:path>")
def wellknown(path):
# Try to proxy to https://nathan.woodburn.au/.well-known/
req = requests.get(f"https://nathan.woodburn.au/.well-known/{path}")
return make_response(
req.content, 200, {"Content-Type": req.headers["Content-Type"]}
)
# endregion
# region Main routes
@app.route("/")
def index():
return render_template("schedule.html", schedule=SCHEDULE_DATA)
@app.route("/<path:path>")
def catch_all(path: str):
if os.path.isfile("templates/" + path):
return render_template(path)
# Try with .html
if os.path.isfile("templates/" + path + ".html"):
return render_template(path + ".html")
if os.path.isfile("templates/" + path.strip("/") + ".html"):
return render_template(path.strip("/") + ".html")
# Try to find a file matching
if path.count("/") < 1:
# Try to find a file matching
filename = find(path, "templates")
if filename:
return send_file(filename)
return render_template("404.html"), 404
# endregion
# region API routes
@app.route("/api/v1/schedule", methods=["GET"])
def api_schedule():
"""
API endpoint that returns the weekly schedule data.
"""
# Always reload data in case Google Sheet has been updated
current_schedule = load_schedule_data()
# Set the global variable to the latest data
global SCHEDULE_DATA
SCHEDULE_DATA = current_schedule
return jsonify({"schedule": current_schedule})
# endregion
# region Error Catching
# 404 catch all
@app.errorhandler(404)
def not_found(e):
return render_template("404.html"), 404
# endregion
if __name__ == "__main__":
app.run(debug=True, port=5000, host="0.0.0.0")