From 80a5ba92ac84342ec72833943d1f2e50ab7412f4 Mon Sep 17 00:00:00 2001 From: Trey Pendragon Date: Tue, 9 Jul 2024 13:49:15 -0700 Subject: [PATCH] Add first draft of indexing ADR. --- .../0001-document-architecture-decisions.md | 20 +++++ architecture-decisions/0002-indexing.md | 86 +++++++++++++++++++ 2 files changed, 106 insertions(+) create mode 100644 architecture-decisions/0001-document-architecture-decisions.md create mode 100644 architecture-decisions/0002-indexing.md diff --git a/architecture-decisions/0001-document-architecture-decisions.md b/architecture-decisions/0001-document-architecture-decisions.md new file mode 100644 index 00000000..d7569c84 --- /dev/null +++ b/architecture-decisions/0001-document-architecture-decisions.md @@ -0,0 +1,20 @@ +# 1. Record architecture decisions + +Date: 2024-07-09 + +## Status + +Accepted + +## Context + +We need to record the architectural decisions made on this project. + +## Decision + +We will use Architecture Decision Records, as described by Michael Nygard in this article: http://thinkrelevance.com/blog/2011/11/15/documenting-architecture-decisions + +## Consequences + +* See Michael Nygard's article, linked above. For a lightweight ADR toolset, see Nat Pryce's _adr-tools_ at https://github.com/npryce/adr-tools. +* Pull requests that change configuration settings, either in this repository, or in https://github.com/pulibrary/princeton_ansible, should be accompanied by an ADR. diff --git a/architecture-decisions/0002-indexing.md b/architecture-decisions/0002-indexing.md new file mode 100644 index 00000000..01b02186 --- /dev/null +++ b/architecture-decisions/0002-indexing.md @@ -0,0 +1,86 @@ +# 2. Indexing Architecture + +Date: 2024-07-09 + +## Status + +Accepted + +## Context + +DPUL-Collections must have a resilient indexing pipeline that can quickly harvest, transform, and index millions of records. We foresee needing to index millions of records, regularly change weighting algorithms, and accept records from external institutions which may not be stable in the long term. + +There must be a verifiable method of ensuring that 100% of Figgy's relevant records are indexed into DPUL-Collections, to prevent us from constantly scrambling and diagnosing indexing issues as we do now with our spotlight-powered DPUL. + +We will be starting with indexing from Figgy, so that's where our initial performance requirements will be based upon. + +Often times systems like this use event streaming platforms such as Kafka, but we'd like to prevent adding new technology to our stack. We think we can use Postgres tables as a compact event log. + +## Decision + +Our indexing pipeline will consist of three steps - Hydration, Transformation, and Indexing. + +Each step has a performance requirement - the lower bound is the point at which we stop optimizing in the case of running that full process, the upper bound is the maximum we'll allow it to take before re-architecting. + +For newly added records (not a full pipeline run of all records), we expect to see changes within five minutes of persistence in Figgy, as our stakeholders often do patron requests by "Completing" a record in Figgy and then sending a resource to a patron. They shouldn't have to wait more than 5 minutes to do that. + +```mermaid +flowchart LR + A[Figgy Postgres] -->|Hydrate| B[DPUL-C Figgy Record Cache] + B -->|Transform| C[DPUL-C Solr Record Cache] + C -->|Index| D[DPUL-C Solr] +``` + +### Hydration + +Hydration will copy records from Figgy and place them into a cache in DPUL-Collections. This pattern will allow us to do the following steps no matter the uptime or performance characteristics of our source repository. + +Every minute DPUL-C will poll Figgy's `orm_resources` table for newly updated records and copy them into a local postgres cache that has the following structure: + +| id | data | log_order | figgy_modified_date | +|------|-------|-----------|---------------------| +| UUID | JSONB | INT | DATETIME | + +We'll pull records as well as DeletionMarkers so we'll know and record when records have been deleted from Figgy. + +#### Performance Requirements + +1 Hour - 2 Days + +##### Performance Reasoning + +The faster we can do a full re-harvest, the faster we can pull in broad metadata changes from upstream (such as new Figgy or Bibdata data.) We want these kinds of tickets to have at most two days of delay. + +### Transformation + +Every minute Transformation will poll the records cached by the Hydration step, convert them to a Solr document, and store that solr document in a local postgres cache with the following structure: + +| id | data | log_order | +|------|-------|-----------| +| UUID | JSONB | INT | + +#### Performance Requirements + +30 minutes - 2 hours + +##### Performance Reasoning + +We will need to do a re-transformation when we add new fields to the index, which we expect to do often. The faster we can do that, the more of those tickets we can do. With a two hour transformation stage we can do more than one such transformation a day, significantly improving our productivity. + +### Reindex + +Every minute Reindex will poll the records cached by the Transformation step and index them into Solr as a batch. + +#### Performance Requirements + +10 minutes - 1 hour + +##### Performance Reasoning + +We expect reindexing to need to happen often - either because of changing weights in Solr, migrating Solr machines, or testing new configurations. By tightening up this time as much as possible we can try many different weights in a day, supporting our vision of being able to create a joyful discovery experience. We believe this performance estimate is reasonable given that there won't be any transformation necessary - it will go as fast as Solr can accept documents. + +## Consequences + +We need to find a way to validate that we're indexing 100% of the documents that we pull from Figgy. + +Keeping track of three different tables may be complicated. However, we expect to be able to scale this architecture out to allow for multiple harvest sources and transformation steps in the future.