diff --git a/examples/cache-warmer/README.md b/examples/cache-warmer/README.md
new file mode 100644
index 0000000..0aba46a
--- /dev/null
+++ b/examples/cache-warmer/README.md
@@ -0,0 +1,118 @@
+# Cache Warmer
+
+Drupal's internal page cache system is wiped whenever `drush cr` is called. This causes deployments to slow down the site for anonymous viewers.
+
+## Install
+
+Update your docker-compose.yml with
+
+```
+ cache-warmer-prod: &cache-warmer-prod
+ <<: [*prod, *common]
+ image: us-docker.pkg.dev/lehigh-lts-images/public/scyllaridae-cache-warmer:main
+ environment:
+ DRUPAL_URL: "https://${DOMAIN}"
+ # when warming the cache, how many nodes to process at once
+ NODE_PARALLEL_EXECUTIONS: 5
+ # when warming the cache, how many IIIF manifests to process at once
+ IIIF_PARALLEL_EXECUTIONS: 3
+ depends_on:
+ activemq-prod:
+ condition: service_started
+```
+
+You'd also need an accompanying Drupal module to handle disk caching/invalidation on the Drupal side, which can eventually be broken out of [lehigh-university-libraries/lehigh_islandora](https://github.com/lehigh-university-libraries/lehigh_islandora)
+
+## Decision
+
+Keep the three slowest responses cached on disk to persist beyond cache flushes.
+
+- IIIF book manifests
+- Node canonical pages
+- Search results
+
+### Deployment
+
+When deployments happen, send an event to cache the entire site
+
+```mermaid
+sequenceDiagram
+ actor Developer
+ Developer->>GitLab: git push
+ GitLab->>Islandora: deploy
+ Islandora->>activemq: queue:islandora-cache-warmer
+ scyllaridae->>activemq: read message
+ note over scyllaridae,Islandora: get all nodes
+ scyllaridae->>Islandora: cURL /sitemap.xml
+ loop for URL in sitemap.xml
+ scyllaridae->>Islandora: curl $URL
+ end
+ note over scyllaridae,Islandora: all nodes cache are now warm
+
+ note over scyllaridae,Islandora: get all nodes with children
+ scyllaridae->>Islandora: cURL /paged-content-nids
+ Islandora->>scyllaridae: PAGED_CONTENT_NIDS
+ loop for NID in PAGED_CONTENT_NIDS
+ scyllaridae->>Islandora: curl /node/$NID/book-manifest
+ end
+ note over scyllaridae,Islandora: all IIIF manifests are now warm
+```
+
+### Node CUD
+
+When a node is CUD, its filesystem cache is wiped from disk and an event is emitted to warm the cache for that node and its parent.
+
+```mermaid
+sequenceDiagram
+ actor Content editor
+ Content editor->>Islandora: save node
+ note over Islandora: hook_cud()
flush filesystem cache
for node and parents
+ Islandora->>activemq: queue:islandora-cache-warmer
+ scyllaridae->>activemq: read message
+ scyllaridae->>Islandora: curl /node/$NID
+ Islandora-->>scyllaridae: HTTP/1.1 200
+ note over scyllaridae,Islandora: node view cache warm
+ scyllaridae->>Islandora: curl /node/$PARENT_NID/book-manifest
+ Islandora-->>scyllaridae: HTTP/1.1 200
+ note over scyllaridae,Islandora: parent book manifest cache warm
+
+```
+
+## Rationale
+
+### Traditional
+
+Out of the box Drupal/Islandora's internal page cache is wiped on deployment. This means the site will be slow between the time the deployment happens until the node cache can get warm again. Traditionally, the only way the cache could get warm would be a site visitor visiting the page, which would mean potentially a very poor experience if the page takes a long time to load. Or the cache might get filled by a custom crawler, but to avoid overloading the system that can take several hours to traverse a large site. Which means a larger window for site visitors to have poor experiences.
+
+Traditionally individual item pages are also slow for the initial site visitor when node/media are CUD by a content editor, which flushes the cache for that given entity view after, e.g. an item record has its metadata updated.
+
+### What this service solves
+
+If instead we hold the cache on disk instead of in Drupal's database cache we can invalidate the cache on a per node basis after deployment. This means after a deployment until the cache is invalidated // warmed site visitors will still enjoy a speedy site. Old cached entries will be served until the deployment script can invalidate them and fill the cache with their new version.
+
+We can also emit events on node/media CUD events to have the anonymous view kept warm after entity updates using this same mechanism.
+
+## Consequences
+
+Positive:
+
+- Fast page loads for site visitors
+- A significantly faster site means resources can be better utilized by other processes
+- With our cache invalidation strategy, only the cache warmer service should ever run into a slow page load during the cache invalidation / warming process
+- Having the cache handled internally by Drupal avoids external configurations like a CDN or Varnish which would introduce additional cache purging configuration. We're instead solving with a custom event subscriber and `hook_entity_CUD` implementations
+- Having the cache logic inside Drupal also allows for holding the cache and only invalidating on a given signal, with that logic written in php/Drupal so keeps a bit more consistent tech stack
+
+Negative:
+
+- This is a novel caching implementation with unknown side effects. Though given we're hooking into Drupal's caching system to invalidate cache during CUD events this risk seems fairly low
+- Search results could still have slow load times for different faceted/search results. We need to look into other performance improvements in code to speed these up. though less load on the system overall, which this caching strategy affords, should help
+
+## Conclusion
+
+This will be implemented by:
+
+- This scyllaridae microservice
+- A Drupal module that handles
+ - hook_node/media_insert/update/delete implementations to wipe the cached responses from the filesystem
+ - A custom route to get paged content nids
+ - An event subscriber to handle caching the responses on disk and invalidating when receiving requests from the microservice
diff --git a/examples/cache-warmer/cmd.sh b/examples/cache-warmer/cmd.sh
index 121506c..6652d34 100755
--- a/examples/cache-warmer/cmd.sh
+++ b/examples/cache-warmer/cmd.sh
@@ -4,6 +4,17 @@ set -eou pipefail
export LOCK_FILE="/tmp/scyllaridae-cache.lock"
+# how many cURL commands to run in parallel for /node/\d+
+if [ ! -v NODE_PARALLEL_EXECUTIONS ] || [ "$NODE_PARALLEL_EXECUTIONS" = "" ]; then
+ NODE_PARALLEL_EXECUTIONS=5
+fi
+
+# how many cURL commands to run in parallel for IIIF manifests
+if [ ! -v IIIF_PARALLEL_EXECUTIONS ] || [ "$IIIF_PARALLEL_EXECUTIONS" = "" ]; then
+ IIIF_PARALLEL_EXECUTIONS=3
+fi
+
+
handle_error() {
rm -f "$LOCK_FILE"
exit 1
@@ -45,9 +56,6 @@ fi
touch "$LOCK_FILE"
-# how many cURL commands to run in parallel
-PARALLEL_EXECUTIONS=3
-
# Warm everything in the sitemap
BASE_URL="$DRUPAL_URL/sitemap.xml"
PAGE=1
@@ -62,7 +70,7 @@ while true; do
if [ "${STATUS}" -eq 200 ]; then
mapfile -t URLS < <(grep -oP '\K[^<]+' links.xml)
while [ "${#URLS[@]}" -gt 0 ]; do
- for ((i = 0; i < PARALLEL_EXECUTIONS; i++)); do
+ for ((i = 0; i < NODE_PARALLEL_EXECUTIONS; i++)); do
array_length=${#URLS[@]}
if [ "$array_length" -gt 0 ]; then
URL="${URLS[$((array_length-1))]}"
@@ -94,7 +102,7 @@ rm -f links.xml
curl -s "$DRUPAL_URL/api/v1/paged-content" > pc.json
mapfile -t NIDS < <(jq -r '.[]' pc.json)
for NID in "${NIDS[@]}"; do
- for ((i = 0; i < PARALLEL_EXECUTIONS; i++)); do
+ for ((i = 0; i < IIIF_PARALLEL_EXECUTIONS; i++)); do
array_length=${#NIDS[@]}
if [ "$array_length" -gt 0 ]; then
NID="${NIDS[$((array_length-1))]}"