X-Git-Url: https://git.ladys.computer/x_status_git/blobdiff_plain/b07270ecbc9ca480c69b7085cf445a0358e22df4..e2db8c2a3219ec8e97cef6529fe4be9d07f8b392:/post-receive?ds=inline

diff --git a/post-receive b/post-receive
index f631236..f6289da 100755
--- a/post-receive
+++ b/post-receive
@@ -1,4 +1,5 @@
 #!/usr/bin/env python3
+from datetime import datetime as dt, timezone
 from glob import iglob
 from itertools import starmap
 import json
@@ -11,7 +12,7 @@ from subprocess import run
 from sys import stdin
 from warnings import warn
 from xml.dom import XHTML_NAMESPACE
-from xml.dom.minidom import getDOMImplementation
+from xml.dom.minidom import getDOMImplementation, parseString
 
 GIT_DIRECTORY = "/home/USERNAME/Status.git"
 BUILD_DIRECTORY = "/home/USERNAME/status.site.example/.build"
@@ -20,6 +21,10 @@ PUBLIC_URL = "https://status.site.example"
 LANG = "en"
 LIVE_BRANCH = "live"
 
+UTC = timezone.utc
+CURRENT_DATETIME = f"{dt.now(UTC).replace(tzinfo=None).isoformat(timespec='seconds')}Z"
+ATOM_NAMESPACE = "http://www.w3.org/2005/Atom"
+
 if stdin.read().split()[-1] == f"refs/heads/{LIVE_BRANCH}":
 
 	print(f"This is an update to the '{LIVE_BRANCH}' branch; regenerating site…")
@@ -27,7 +32,10 @@ if stdin.read().split()[-1] == f"refs/heads/{LIVE_BRANCH}":
 	# Set up the build directory.
 	if exists(BUILD_DIRECTORY):
 		rmtree(BUILD_DIRECTORY)
-	run(["git", "clone", "--local", "--branch", "live", GIT_DIRECTORY, BUILD_DIRECTORY], capture_output=True, encoding="utf-8")
+	cloneresult = run(["git", "clone", "--local", "--branch", LIVE_BRANCH, GIT_DIRECTORY, BUILD_DIRECTORY], capture_output=True, encoding="utf-8")
+	# if cloneresult.stderr:
+		# print(cloneresult.stderr)
+	cloneresult.check_returncode()
 
 	# Set up various containers.
 	months = {}
@@ -98,6 +106,69 @@ if stdin.read().split()[-1] == f"refs/heads/{LIVE_BRANCH}":
 			status["content"] = statusxml(text.read().strip())
 		return (datetime, identifier, status)
 
+	def atomForLD (ld):
+		doc = getDOMImplementation().createDocument(None, "feed", None)
+		atomElt = doc.documentElement
+		atomElt.setAttribute("xmlns", ATOM_NAMESPACE)
+		atomElt.setAttribute("xml:lang", LANG)
+		subject = ld["subject"] if "subject" in ld else "Statuses"
+		titleElt = atomElt.appendChild(doc.createElement("title"))
+		titleElt.appendChild(doc.createTextNode(f"{subject} @ {PUBLIC_URL}"))
+		updatedElt = atomElt.appendChild(doc.createElement("updated"))
+		updatedElt.appendChild(doc.createTextNode(CURRENT_DATETIME))
+		generatorElt = atomElt.appendChild(doc.createElement("generator"))
+		generatorElt.appendChild(doc.createTextNode("x_status_git"))
+		generatorElt.setAttribute("uri", "https://git.ladys.computer/x_status_git")
+		atomLinks = {}
+		if "OrderedCollectionPage" in ld["@type"]:
+			idElt = atomElt.appendChild(doc.createElement("id"))
+			idElt.appendChild(doc.createTextNode(f"{PUBLIC_URL}/statuses"))
+			atomLinks["alternate"] = f"{PUBLIC_URL}/statuses"
+			atomLinks["current"] = f"{PUBLIC_URL}/statuses.atom"
+			atomLinks["self"] = atomLinks["current"] if ld["@id"] == ld["current"] else f"{ld['@id']}.atom"
+			if "prev" in ld:
+				atomLinks["prev-archive"] = f"{ld['prev']}.atom"
+			if "next" in ld and ld["next"] != ld["current"]:
+				atomLinks["next-archive"] = f"{ld['next']}.atom"
+		else:
+			idElt = atomElt.appendChild(doc.createElement("id"))
+			idElt.appendChild(doc.createTextNode(ld["@id"]))
+			atomLinks["alternate"] = ld["@id"]
+			atomLinks["self"] = f"{ld['@id']}.atom"
+		for (rel, href) in atomLinks.items():
+			linkElt = atomElt.appendChild(doc.createElement("link"))
+			linkElt.setAttribute("rel", rel)
+			linkElt.setAttribute("href", href)
+		for item in ld["items"]:
+			entryElt = atomElt.appendChild(doc.createElement("entry"))
+			title = item["title"] if "title" in item else item["content"][0:27] + "…"
+			titleElt = entryElt.appendChild(doc.createElement("title"))
+			titleElt.appendChild(doc.createTextNode(title))
+			idElt = entryElt.appendChild(doc.createElement("id"))
+			idElt.appendChild(doc.createTextNode(item["@id"]))
+			updatedElt = entryElt.appendChild(doc.createElement("updated"))
+			updatedElt.appendChild(doc.createTextNode(CURRENT_DATETIME))
+			if "created" in item:
+				publishedElt = entryElt.appendChild(doc.createElement("published"))
+				publishedElt.appendChild(doc.createTextNode(item["created"]))
+			authorElt = entryElt.appendChild(doc.createElement("author"))
+			if "author" in item:
+				nameElt = authorElt.appendChild(doc.createElement("name"))
+				nameElt.appendChild(doc.createTextNode(item["author"]["name"]))
+				uriElt = authorElt.appendChild(doc.createElement("uri"))
+				uriElt.appendChild(doc.createTextNode(item["author"]["@id"]))
+			else:
+				nameElt = authorElt.appendChild(doc.createElement("name"))
+				nameElt.appendChild(doc.createTextNode("Anonymous"))
+			contentElt = entryElt.appendChild(doc.createElement("content"))
+			contentElt.setAttribute("type", "xhtml")
+			contentDiv = contentElt.appendChild(doc.createElement("div"))
+			contentDiv.setAttribute("xmlns", XHTML_NAMESPACE)
+			contentDiv.setAttribute("lang", LANG)
+			for child in list(parseString(item["content"]).documentElement.childNodes):
+				contentDiv.appendChild(child)
+		return (atomLinks["self"], atomElt.toxml())
+
 	# Get status paths.
 	status_paths = []
 	for yearpath in Path(f"{BUILD_DIRECTORY}/").glob("[0-9][0-9][0-9][0-9]"):
@@ -160,6 +231,9 @@ if stdin.read().split()[-1] == f"refs/heads/{LIVE_BRANCH}":
 			ld["next"] = f"{PUBLIC_URL}/statuses/{statuspairs[index + 1][1][0]}"
 		with open(f"{PUBLIC_DIRECTORY}/statuses/{yyyy_mm}/index.jsonld", "w", encoding="utf-8") as f:
 			json.dump(ld, f, ensure_ascii=False, allow_nan=False)
+		atomlink, atomxml = atomForLD(ld)
+		with open(f"{PUBLIC_DIRECTORY}/{atomlink[len(PUBLIC_URL):-5]}/index.atom", "w", encoding="utf-8") as f:
+			f.write(atomxml)
 	with open(f"{PUBLIC_DIRECTORY}/statuses/index.jsonld", "w", encoding="utf-8") as f:
 		json.dump({ "@context": { "@language": LANG, "activity": "https://www.w3.org/ns/activitystreams#", "sioc": "http://rdfs.org/sioc/ns#", "OrderedCollection": "activity:OrderedCollection", "Thread": "sioc:Thread", "current": { "@id": "activity:current", "@type": "@id" }, "first": { "@id": "activity:first", "@type": "@id" }, "has_parent": { "@id": "sioc:has_parent", "@type": "id" } }, "@id": f"{PUBLIC_URL}/statuses", "@type": ["OrderedCollection", "Thread"], "first": f"{PUBLIC_URL}/statuses/{statuspairs[0][1][0]}", "current": f"{PUBLIC_URL}/statuses/{statuspairs[-1][1][0]}", "has_parent": f"{PUBLIC_URL}" }, f, ensure_ascii=False, allow_nan=False)
 
@@ -171,6 +245,9 @@ if stdin.read().split()[-1] == f"refs/heads/{LIVE_BRANCH}":
 			mkdir(f"{PUBLIC_DIRECTORY}/topics/{topic}")
 		with open(f"{PUBLIC_DIRECTORY}/topics/{topic}/index.jsonld", "w", encoding="utf-8") as f:
 			json.dump(ld, f, ensure_ascii=False, allow_nan=False)
+		atomlink, atomxml = atomForLD(ld)
+		with open(f"{PUBLIC_DIRECTORY}/{atomlink[len(PUBLIC_URL):-5]}/index.atom", "w", encoding="utf-8") as f:
+			f.write(atomxml)
 	with open(f"{PUBLIC_DIRECTORY}/topics/index.jsonld", "w", encoding="utf-8") as f:
 		json.dump({ "@context": { "@language": LANG, "activity": "https://www.w3.org/ns/activitystreams#", "dct": "http://purl.org/dc/terms/", "sioc": "http://rdfs.org/sioc/ns#", "Collection": "activity:Collection", "Forum": "sioc:Forum", "items": { "@id": "activity:items", "@type": "@id" }, "has_parent": { "@id": "sioc:has_parent", "@type": "id" }, "subject": "dct:subject" }, "@id": f"{PUBLIC_URL}/topics", "@type": ["Collection", "Forum"], "items": list(map(lambda a: { "@id": a["@id"], "subject": a["subject"] }, topics.values())), "has_parent": f"{PUBLIC_URL}" }, f, ensure_ascii=False, allow_nan=False)