{"id":1536,"date":"2026-05-20T07:58:57","date_gmt":"2026-05-20T14:58:57","guid":{"rendered":"https:\/\/www.kenwalger.com\/blog\/?p=1536"},"modified":"2026-04-28T08:26:17","modified_gmt":"2026-04-28T15:26:17","slug":"the-feature-store-consistency-and-latency-are-both-non-negotiable","status":"publish","type":"post","link":"https:\/\/www.kenwalger.com\/blog\/ai\/the-feature-store-consistency-and-latency-are-both-non-negotiable\/","title":{"rendered":"The Feature Store: Consistency and Latency Are Both Non-Negotiable"},"content":{"rendered":"<p>In <a href=\"https:\/\/www.kenwalger.com\/blog\/ai\/feature-freshness-designing-pipelines-that-keep-up-with-the-world\/\">the previous post<\/a> in this series, we worked through the pipeline architecture that gets features from raw events to a computed state. Now we need to talk about where those features live once they&#8217;re computed \u2014 and how they get from storage to your model at inference time.<\/p>\n<p>That&#8217;s the feature store&#8217;s job.<\/p>\n<p>The feature store is the operational center of a real-time ML system. It sits between the pipeline that produces features and the model that consumes them. Get it right, and you have a foundation for every model you&#8217;ll build. Get it wrong, and you&#8217;ll spend years firefighting problems that trace back to a design decision made early on.<\/p>\n<p>The central tension in feature store design is this: <strong>you need consistency and low latency simultaneously, at scale.<\/strong> Those goals pull in different directions. Understanding why \u2014 and what architectural patterns resolve the tension \u2014 is what this post is about.<\/p>\n<hr \/>\n<h2>What a Feature Store Actually Does<\/h2>\n<p>Before getting into design, it&#8217;s worth being precise about what a feature store is responsible for, because the term gets used loosely.<\/p>\n<p>A feature store has four core responsibilities:<\/p>\n<ol>\n<li><strong>Storage<\/strong>: Persisting feature values in a form that can be retrieved efficiently for both model training (batch reads over large historical windows) and model inference (point reads of current values with sub-millisecond latency requirements).<\/p>\n<\/li>\n<li><strong>Serving<\/strong>: Delivering feature values to the model at inference time. This includes fetching features for a given entity, handling missing values, and assembling a complete feature vector from potentially many feature groups.<\/p>\n<\/li>\n<li><strong>Registry<\/strong>: Maintaining a catalog of what features exist, how they&#8217;re defined, who owns them, and what version is currently in production. This is the governance layer.<\/p>\n<\/li>\n<li><strong>Consistency enforcement<\/strong>: Ensuring that the features used to train a model are computed the same way as the features served at inference time. This is where most feature store implementations have gaps.<\/p>\n<\/li>\n<\/ol>\n<p>Systems that call themselves feature stores but only address one or two of these responsibilities create hidden risk. The gaps don&#8217;t show up in demos. They show up in production.<\/p>\n<hr \/>\n<h2>The Dual-Store Architecture<\/h2>\n<p>The fundamental design pattern for production feature stores is the dual-store architecture. It separates storage into two distinct layers, each optimized for a different access pattern.<\/p>\n<p><strong>The online store<\/strong> serves inference. It holds the current feature values for every entity your models need to reason about \u2014 users, products, accounts, transactions. Reads must be extremely fast: in a low-latency serving path, the feature retrieval step often has a budget of 5-20ms for fetching dozens of feature values simultaneously. This demands in-memory or SSD-backed storage with O(1) key-value access patterns.<\/p>\n<p><strong>The offline store<\/strong> serves training and analysis. It holds the full history of feature values, queryable by entity and time. Reads are slower \u2014 seconds to minutes \u2014 but the storage cost is dramatically lower than the online store. Columnar formats like Parquet on object storage, or purpose-built analytical databases, are typical choices here.<\/p>\n<p>A <strong>write path<\/strong> keeps both stores synchronized. When a new feature value is computed \u2014 by a batch job or a streaming pipeline \u2014 it&#8217;s written to both stores. The online store gets the current value; the offline store appends to the historical record.<\/p>\n<pre><code class=\"language-mermaid\">flowchart TD\n    FP[\"Feature Pipeline\\n(Batch or Streaming)\"]\n    WP[\"Write Path\\n(Synchronization Layer)\"]\n    OS[\"Online Store\\nLow latency \u00b7 Current values\\nKey-value \/ In-memory\"]\n    OF[\"Offline Store\\nFull history \u00b7 Batch reads\\nColumnar \/ Object storage\"]\n    MI[\"Model Inference\\n(Real-time serving)\"]\n    MT[\"Model Training\\n(Historical datasets)\"]\n\n    FP --&gt; WP\n    WP --&gt; OS\n    WP --&gt; OF\n    OS --&gt; MI\n    OF --&gt; MT\n\n    style OS fill:#d4edda,stroke:#28a745,color:#000\n    style OF fill:#d1ecf1,stroke:#17a2b8,color:#000\n    style WP fill:#fff3cd,stroke:#ffc107,color:#000\n<\/code><\/pre>\n<p>This pattern is well-established and widely implemented. The execution details \u2014 which technologies you use for each store, how you handle the write path, how you synchronize \u2014 vary considerably and matter a great deal. But the conceptual split is the right starting point for almost every production ML system.<\/p>\n<hr \/>\n<h2>The Consistency Problem (And Why It&#8217;s Harder Than It Looks)<\/h2>\n<p>The dual-store architecture introduces a consistency challenge that&#8217;s easy to underestimate: <strong>the online store and offline store must agree on how features are defined.<\/strong><\/p>\n<p>If a feature is computed differently in the batch pipeline that writes to the offline store and the streaming pipeline that writes to the online store, your model is trained on data that doesn&#8217;t match what it sees in production. We touched on this as training-serving skew in the previous post. Here we&#8217;re looking at the structural causes.<\/p>\n<p>The most common source of inconsistency is <strong>transformation logic duplication<\/strong>. Consider a feature defined as &#8220;the number of purchases a user has made in the last 7 days.&#8221; The batch pipeline computes this as a SQL aggregation over a historical table. The streaming pipeline computes it by maintaining a rolling count in memory. Both produce a number with the same name. But if there&#8217;s any difference in how they handle timezone boundaries, null transactions, cancelled orders, or edge cases in the event data \u2014 and there almost always is \u2014 the values will diverge.<\/p>\n<p>The model trained on the batch-computed values will behave differently at inference time than its offline metrics predicted.<\/p>\n<p><strong>Feature definitions as code<\/strong> is the architectural response to this. Rather than implementing transformation logic separately in the batch and streaming systems, you define a feature once \u2014 as a versioned, named computation \u2014 and a shared transformation layer executes that definition in both contexts.<\/p>\n<pre><code class=\"language-python\"># Define once\n@feature(name=\"user_purchases_7d\", version=1)\ndef user_purchases_7d(events: EventStream, window: Window) -&gt; int:\n    return events\n        .filter(type=\"purchase\", status=\"completed\")\n        .window(days=7)\n        .count()\n\n# The feature store executes this definition in both:\n# - batch context (for offline store \/ training data)\n# - streaming context (for online store \/ inference)\n<\/code><\/pre>\n<p>The implementation varies by framework, but the principle is consistent: the definition is the source of truth, not the pipeline code that executes it. When the definition changes, both paths update together.<\/p>\n<hr \/>\n<h2>Access Pattern Design: What Inference Actually Looks Like<\/h2>\n<p>One of the most consequential decisions in feature store design is one that rarely gets explicit attention: <strong>how does the model actually retrieve features at serving time?<\/strong><\/p>\n<p>The naive assumption is that inference retrieval looks like a database lookup \u2014 give me the features for user 12345. That&#8217;s partially true, but the reality is more demanding.<\/p>\n<p>A single inference request typically requires:<\/p>\n<ul>\n<li>Features from multiple feature groups (user features, item features, context features, cross features)<\/li>\n<li>Multiple entities resolved simultaneously (the requesting user <em>and<\/em> the item being scored <em>and<\/em> the user-item interaction history)<\/li>\n<li>Values that must arrive within a strict latency budget, because feature retrieval is one step in a larger serving pipeline that also includes model execution, pre\/post-processing, and network overhead<\/li>\n<\/ul>\n<p>This means feature retrieval for inference is almost always a <strong>batch point lookup<\/strong> \u2014 fetching many feature values for multiple entities in a single operation \u2014 rather than a sequence of individual reads.<\/p>\n<p>The difference matters enormously for performance. A feature store that executes N separate reads to assemble a feature vector will be N times slower than one that batches those reads into a single round-trip. At a P99 latency budget of 20ms, the difference between one network round-trip and five is the difference between a system that meets its SLA and one that doesn&#8217;t.<\/p>\n<p>Design your feature store&#8217;s serving API \u2014 and choose your online store technology \u2014 around this access pattern, not around the pattern that&#8217;s easiest to implement.<\/p>\n<hr \/>\n<h2>Schema Versioning and Governance<\/h2>\n<p>Features change. A feature that was defined one way last quarter may need to be redefined this quarter \u2014 a new data source becomes available, a bug is found in the transformation logic, a business definition shifts. Managing this change without breaking production systems is the feature store&#8217;s schema governance problem.<\/p>\n<p>The failure mode without explicit governance is silent: a feature&#8217;s definition changes, the new version is deployed to the pipeline, the online store starts serving new values \u2014 and models trained on the old definition are now receiving inputs that don&#8217;t match their training distribution. No error is thrown. Prediction quality degrades. Debugging is expensive.<\/p>\n<p>A versioned feature registry addresses this by making each change to a feature definition explicit and tracked:<\/p>\n<pre><code>feature: user_purchases_7d\n  version: 1  \u2192  \"completed\" purchases only, UTC timezone\n  version: 2  \u2192  adds \"refunded\" status exclusion, user's local timezone\n  version: 3  \u2192  changes window to rolling 7 days vs. calendar week\n<\/code><\/pre>\n<p>Models are pinned to specific feature versions. A new model version can be trained against a new feature version while the old model continues to run against the old version in production. Rollbacks are clean and auditable.<\/p>\n<p>The registry also enables <strong>discoverability<\/strong>: a data scientist looking for a user engagement feature can search the registry rather than building a new pipeline that computes the same thing differently. This is the organizational leverage point we discussed in the previous post \u2014 reuse only happens when features are visible and well-documented.<\/p>\n<p>Minimum viable governance includes: feature name, version, owner, description, transformation definition, schema of outputs, and the models currently consuming each version. Teams that invest in this infrastructure early save significant operational cost later.<\/p>\n<hr \/>\n<h2>Cold Start: The Edge Case That Isn&#8217;t<\/h2>\n<p>Every feature store encounters the cold start problem: what feature values does the model receive for a new entity that has no history in the system?<\/p>\n<p>This is often treated as an edge case and handled hastily \u2014 a null value, a zero, a global average imputed at serving time. In practice, cold start is not an edge case. Every user&#8217;s first session is a cold start. Every new product listing is a cold start. Every new account is a cold start.<\/p>\n<p>For some models and some features, the imputation strategy doesn&#8217;t matter much. For others, it matters enormously. A fraud model that sees a null for &#8220;number of purchases in the last 30 days&#8221; when a new account is created may behave very differently than one that sees a zero, or a global average, or a segment-specific prior.<\/p>\n<p><strong>Cold start strategy belongs in the feature definition, not in the serving layer.<\/strong> The definition should specify:<\/p>\n<ul>\n<li>What value to serve when no history exists<\/li>\n<li>Whether to use a global default, a segment-specific prior, or a model-specific override<\/li>\n<li>How long an entity must have existed before it graduates from cold-start handling<\/li>\n<\/ul>\n<p>Treating cold start as a serving-layer afterthought means the strategy is invisible to the model training process \u2014 the model was trained without cold-start examples, so it&#8217;s never learned to handle them appropriately.<\/p>\n<hr \/>\n<h2>Monitoring: What Does &#8220;Healthy&#8221; Look Like?<\/h2>\n<p>A feature store has no natural test suite. You can verify that a feature pipeline runs without errors and that values are being written to both stores. But correctness \u2014 whether the values are actually right \u2014 requires a monitoring strategy.<\/p>\n<p>The key signals worth tracking:<\/p>\n<p><strong>Feature freshness<\/strong>: How old is the most recent value in the online store for each feature? This should be an active alert, not a metric you look at retrospectively. A feature that hasn&#8217;t been updated in twice its expected refresh interval is probably broken.<\/p>\n<p><strong>Value distribution drift<\/strong>: The statistical distribution of feature values in the online store should be approximately stable over time (or change in expected ways as your user base grows). Sudden shifts in mean, variance, or cardinality are early warning signals of upstream pipeline problems \u2014 a schema change in source data, a filtering bug introduced in a new pipeline version, a data source going stale.<\/p>\n<p><strong>Training-serving distribution comparison<\/strong>: Periodically compare the distribution of feature values logged at serving time against the distribution in your training dataset. Systematic divergence is evidence of training-serving skew accumulating over time.<\/p>\n<p><strong>Serving latency by feature group<\/strong>: Not all features are equally expensive to retrieve. Tracking retrieval latency at the feature group level surfaces which groups are contributing to serving SLA violations.<\/p>\n<pre><code class=\"language-python\"># A minimal freshness check\ndef check_feature_freshness(feature_name, max_age_seconds):\n    last_updated = feature_store.get_last_updated(feature_name)\n    age = time.now() - last_updated\n    if age &gt; max_age_seconds:\n        alert(f\"{feature_name} is {age}s old, threshold is {max_age_seconds}s\")\n<\/code><\/pre>\n<p>Teams that treat feature monitoring as an afterthought discover problems the way they discover most production ML problems: through user-facing degradation that&#8217;s difficult to attribute. Teams that build monitoring into the feature store from the start catch the same problems in minutes.<\/p>\n<hr \/>\n<h2>Putting It Together: Feature Store Design Checklist<\/h2>\n<p>Before committing to a feature store architecture, validate against these questions:<\/p>\n<p><strong>Storage and serving:<\/strong><br \/>\n&#8211; Is your online store optimized for batch point lookups, not sequential reads?<br \/>\n&#8211; Can you retrieve a complete feature vector for inference in a single round-trip?<br \/>\n&#8211; Is your offline store capable of point-in-time correct reads for model training?<\/p>\n<p><strong>Consistency:<\/strong><br \/>\n&#8211; Is transformation logic defined once and executed in both batch and streaming contexts?<br \/>\n&#8211; Do you have a process for detecting training-serving skew before it affects production models?<\/p>\n<p><strong>Governance:<\/strong><br \/>\n&#8211; Are features versioned, named, and documented in a central registry?<br \/>\n&#8211; Are models pinned to specific feature versions?<br \/>\n&#8211; Can a data scientist discover existing features before building a new pipeline?<\/p>\n<p><strong>Operational:<\/strong><br \/>\n&#8211; Is feature freshness actively monitored with alerts?<br \/>\n&#8211; Is value distribution drift tracked for each feature?<br \/>\n&#8211; Is there an explicit cold start strategy in each feature definition?<\/p>\n<p>The goal isn&#8217;t to answer all of these perfectly from day one. The goal is to know which questions you&#8217;ve answered and which you&#8217;ve deferred \u2014 because the deferred ones will eventually surface as production incidents.<\/p>\n<hr \/>\n<p>In the next post, we move to the third pillar: vector search at scale, where index degradation, hybrid filtering, and recall monitoring introduce a different class of production challenges.<\/p>\n<h3>When Your AI Pipeline Grows Up Series<\/h3>\n<ul>\n<li><a href=\"https:\/\/www.kenwalger.com\/blog\/ai\/when-your-ai-pipeline-grows-up-infrastructure-thinking-for-real-time-inference-at-scale\">Real Time AI at Scale<\/a><\/li>\n<li><a href=\"https:\/\/www.kenwalger.com\/blog\/ai\/feature-freshness-designing-pipelines-that-keep-up-with-the-world\">Feature Freshness<\/a><\/li>\n<li>Feature Store &#8211; <em>This Post.<\/em><\/li>\n<li>Vector Search &#8211; <em>Coming Soon.<\/em><\/li>\n<\/ul>\n<a class=\"synved-social-button synved-social-button-share synved-social-size-48 synved-social-resolution-single synved-social-provider-facebook nolightbox\" data-provider=\"facebook\" target=\"_blank\" rel=\"nofollow\" title=\"Share on Facebook\" href=\"https:\/\/www.facebook.com\/sharer.php?u=https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-json%2Fwp%2Fv2%2Fposts%2F1536&#038;t=The%20Feature%20Store%3A%20Consistency%20and%20Latency%20Are%20Both%20Non-Negotiable&#038;s=100&#038;p&#091;url&#093;=https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-json%2Fwp%2Fv2%2Fposts%2F1536&#038;p&#091;images&#093;&#091;0&#093;=https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-content%2Fuploads%2F2026%2F04%2Fblog-of-ken-w.-alger-69f0d0fb4a0f1.png&#038;p&#091;title&#093;=The%20Feature%20Store%3A%20Consistency%20and%20Latency%20Are%20Both%20Non-Negotiable\" style=\"font-size: 0px;width:48px;height:48px;margin:0;margin-bottom:5px;margin-right:5px\"><img loading=\"lazy\" decoding=\"async\" alt=\"Facebook\" title=\"Share on Facebook\" class=\"synved-share-image synved-social-image synved-social-image-share\" width=\"48\" height=\"48\" style=\"display: inline;width:48px;height:48px;margin: 0;padding: 0;border: none;box-shadow: none\" src=\"https:\/\/www.kenwalger.com\/blog\/wp-content\/plugins\/social-media-feather\/synved-social\/image\/social\/regular\/96x96\/facebook.png\" \/><\/a><a class=\"synved-social-button synved-social-button-share synved-social-size-48 synved-social-resolution-single synved-social-provider-twitter nolightbox\" data-provider=\"twitter\" target=\"_blank\" rel=\"nofollow\" title=\"Share on Twitter\" href=\"https:\/\/twitter.com\/intent\/tweet?url=https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-json%2Fwp%2Fv2%2Fposts%2F1536&#038;text=Hey%20check%20this%20out\" style=\"font-size: 0px;width:48px;height:48px;margin:0;margin-bottom:5px;margin-right:5px\"><img loading=\"lazy\" decoding=\"async\" alt=\"twitter\" title=\"Share on Twitter\" class=\"synved-share-image synved-social-image synved-social-image-share\" width=\"48\" height=\"48\" style=\"display: inline;width:48px;height:48px;margin: 0;padding: 0;border: none;box-shadow: none\" src=\"https:\/\/www.kenwalger.com\/blog\/wp-content\/plugins\/social-media-feather\/synved-social\/image\/social\/regular\/96x96\/twitter.png\" \/><\/a><a class=\"synved-social-button synved-social-button-share synved-social-size-48 synved-social-resolution-single synved-social-provider-reddit nolightbox\" data-provider=\"reddit\" target=\"_blank\" rel=\"nofollow\" title=\"Share on Reddit\" href=\"https:\/\/www.reddit.com\/submit?url=https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-json%2Fwp%2Fv2%2Fposts%2F1536&#038;title=The%20Feature%20Store%3A%20Consistency%20and%20Latency%20Are%20Both%20Non-Negotiable\" style=\"font-size: 0px;width:48px;height:48px;margin:0;margin-bottom:5px;margin-right:5px\"><img loading=\"lazy\" decoding=\"async\" alt=\"reddit\" title=\"Share on Reddit\" class=\"synved-share-image synved-social-image synved-social-image-share\" width=\"48\" height=\"48\" style=\"display: inline;width:48px;height:48px;margin: 0;padding: 0;border: none;box-shadow: none\" src=\"https:\/\/www.kenwalger.com\/blog\/wp-content\/plugins\/social-media-feather\/synved-social\/image\/social\/regular\/96x96\/reddit.png\" \/><\/a><a class=\"synved-social-button synved-social-button-share synved-social-size-48 synved-social-resolution-single synved-social-provider-linkedin nolightbox\" data-provider=\"linkedin\" target=\"_blank\" rel=\"nofollow\" title=\"Share on Linkedin\" href=\"https:\/\/www.linkedin.com\/shareArticle?mini=true&#038;url=https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-json%2Fwp%2Fv2%2Fposts%2F1536&#038;title=The%20Feature%20Store%3A%20Consistency%20and%20Latency%20Are%20Both%20Non-Negotiable\" style=\"font-size: 0px;width:48px;height:48px;margin:0;margin-bottom:5px;margin-right:5px\"><img loading=\"lazy\" decoding=\"async\" alt=\"linkedin\" title=\"Share on Linkedin\" class=\"synved-share-image synved-social-image synved-social-image-share\" width=\"48\" height=\"48\" style=\"display: inline;width:48px;height:48px;margin: 0;padding: 0;border: none;box-shadow: none\" src=\"https:\/\/www.kenwalger.com\/blog\/wp-content\/plugins\/social-media-feather\/synved-social\/image\/social\/regular\/96x96\/linkedin.png\" \/><\/a><a class=\"synved-social-button synved-social-button-share synved-social-size-48 synved-social-resolution-single synved-social-provider-mail nolightbox\" data-provider=\"mail\" rel=\"nofollow\" title=\"Share by email\" href=\"mailto:?subject=The%20Feature%20Store%3A%20Consistency%20and%20Latency%20Are%20Both%20Non-Negotiable&#038;body=Hey%20check%20this%20out:%20https%3A%2F%2Fwww.kenwalger.com%2Fblog%2Fwp-json%2Fwp%2Fv2%2Fposts%2F1536\" style=\"font-size: 0px;width:48px;height:48px;margin:0;margin-bottom:5px\"><img loading=\"lazy\" decoding=\"async\" alt=\"mail\" title=\"Share by email\" class=\"synved-share-image synved-social-image synved-social-image-share\" width=\"48\" height=\"48\" style=\"display: inline;width:48px;height:48px;margin: 0;padding: 0;border: none;box-shadow: none\" src=\"https:\/\/www.kenwalger.com\/blog\/wp-content\/plugins\/social-media-feather\/synved-social\/image\/social\/regular\/96x96\/mail.png\" \/><\/a>","protected":false},"excerpt":{"rendered":"<p>In the previous post in this series, we worked through the pipeline architecture that gets features from raw events to a computed state. Now we need to talk about where those features live once they&#8217;re computed \u2014 and how they get from storage to your model at inference time. That&#8217;s the feature store&#8217;s job. The &hellip; <a href=\"https:\/\/www.kenwalger.com\/blog\/ai\/the-feature-store-consistency-and-latency-are-both-non-negotiable\/\" class=\"more-link\">Continue reading<span class=\"screen-reader-text\"> &#8220;The Feature Store: Consistency and Latency Are Both Non-Negotiable&#8221;<\/span><\/a><\/p>\n","protected":false},"author":1,"featured_media":1559,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"pmpro_default_level":"","_monsterinsights_skip_tracking":false,"_monsterinsights_sitenote_active":false,"_monsterinsights_sitenote_note":"","_monsterinsights_sitenote_category":0,"_jetpack_newsletter_access":"","_jetpack_dont_email_post_to_subs":false,"_jetpack_newsletter_tier_id":0,"_jetpack_memberships_contains_paywalled_content":false,"_jetpack_memberships_contains_paid_content":false,"footnotes":"","jetpack_post_was_ever_published":false},"categories":[1669],"tags":[],"yst_prominent_words":[1061,90,290,420,715,625],"class_list":["post-1536","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-ai","pmpro-has-access"],"jetpack_featured_media_url":"https:\/\/www.kenwalger.com\/blog\/wp-content\/uploads\/2026\/04\/blog-of-ken-w.-alger-69f0d0fb4a0f1.png","jetpack_sharing_enabled":true,"jetpack_shortlink":"https:\/\/wp.me\/p8lx70-oM","jetpack-related-posts":[],"_links":{"self":[{"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/posts\/1536","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/comments?post=1536"}],"version-history":[{"count":7,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/posts\/1536\/revisions"}],"predecessor-version":[{"id":1552,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/posts\/1536\/revisions\/1552"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/media\/1559"}],"wp:attachment":[{"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/media?parent=1536"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/categories?post=1536"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/tags?post=1536"},{"taxonomy":"yst_prominent_words","embeddable":true,"href":"https:\/\/www.kenwalger.com\/blog\/wp-json\/wp\/v2\/yst_prominent_words?post=1536"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}