<?xml version="1.0" encoding="utf-8"?>
<feed xmlns="http://www.w3.org/2005/Atom">
  <title>Dynoxide changelog</title>
  <subtitle>The fastest local DynamoDB emulator. A native binary with ~2ms startup, a 3 MB download, and no Docker or JVM. A drop-in alternative to DynamoDB Local.</subtitle>
  <link href="https://dynoxide.dev/changelog.xml" rel="self"/>
  <link href="https://dynoxide.dev/changelog"/>
  <id>https://dynoxide.dev/changelog</id>
  <updated>2026-06-26T00:00:00Z</updated>
  <author>
    <name>Martin Hicks</name>
    <uri>https://martinhicks.dev</uri>
  </author>
  <entry>
    <title>0.11.1</title>
    <link href="https://dynoxide.dev/changelog#0.11.1"/>
    <id>https://dynoxide.dev/changelog#0.11.1</id>
    <updated>2026-06-26T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;A &lt;code&gt;ConditionExpression&lt;/code&gt; comparing a Map (&lt;code&gt;M&lt;/code&gt;) or List (&lt;code&gt;L&lt;/code&gt;) attribute for equality now works, where &lt;code&gt;=&lt;/code&gt; always reported not-equal and &lt;code&gt;&amp;lt;&amp;gt;&lt;/code&gt; always equal regardless of the values. &lt;code&gt;compare_values&lt;/code&gt; had no arm for document types, so every map or list comparison fell through to the not-equal default; it now compares them deeply - maps order-independently, lists element-wise in order - with nested numbers normalised as elsewhere. The same path backs &lt;code&gt;IN&lt;/code&gt;, &lt;code&gt;BETWEEN&lt;/code&gt;, and &lt;code&gt;contains&lt;/code&gt; over document operands, so those are fixed too (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/103&quot;&gt;#103&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;ExpressionAttributeValues&lt;/code&gt; nested beyond DynamoDB&#39;s 32-level document limit are now rejected up front with the same &lt;code&gt;ValidationException&lt;/code&gt; AWS returns, where before they were accepted and evaluated. The check runs on every path that takes expression values - PutItem, UpdateItem, DeleteItem, Query, Scan, and TransactWriteItems. The stored-item nesting check was also one level too lenient (it accepted a value AWS rejects) and carried a non-AWS message; both now match DynamoDB&#39;s limit and wording, confirmed against real AWS in eu-west-2 (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/110&quot;&gt;#110&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;Number-set equality in a condition or filter expression now compares at full precision, where it parsed each member to &lt;code&gt;f64&lt;/code&gt; and so reported two sets differing only beyond ~15 significant digits as equal. It now uses the canonical numeric form, matching DynamoDB and the way number-set duplicates are already detected on write; the fix also covers number sets nested inside a map or list (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/111&quot;&gt;#111&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;A &lt;code&gt;Number&lt;/code&gt; with a leading &lt;code&gt;+&lt;/code&gt; on the mantissa (&lt;code&gt;+5&lt;/code&gt;, &lt;code&gt;+1.5&lt;/code&gt;, &lt;code&gt;+1e2&lt;/code&gt;) is now accepted and stored normalised (&lt;code&gt;+5&lt;/code&gt; reads back as &lt;code&gt;5&lt;/code&gt;), matching real DynamoDB, where dynoxide rejected it with a &lt;code&gt;ValidationException&lt;/code&gt;. The validator was reworked to accept exactly DynamoDB&#39;s numeric grammar, which also closes two pre-existing gaps in the same direction: malformed forms such as &lt;code&gt;1+2&lt;/code&gt;, &lt;code&gt;1.2.3&lt;/code&gt;, &lt;code&gt;+e2&lt;/code&gt;, and a digitless exponent are now rejected, as is any surrounding or internal whitespace (&lt;code&gt;&amp;quot; 5&amp;quot;&lt;/code&gt; was previously trimmed and accepted). The accept and reject boundary was verified against real DynamoDB (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/109&quot;&gt;#109&lt;/a&gt;).&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.11.0</title>
    <link href="https://dynoxide.dev/changelog#0.11.0"/>
    <id>https://dynoxide.dev/changelog#0.11.0</id>
    <updated>2026-06-24T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;UpdateTable&lt;/code&gt; on the wasm preview engine: add or delete a global secondary index, with existing rows backfilled into a newly added index, and change the simple table settings (provisioned throughput, billing mode, table class, on-demand throughput, deletion protection). A stream-specification change through &lt;code&gt;UpdateTable&lt;/code&gt; stays unsupported, since streams remain a preview gap, and a newly added GSI is reported immediately &lt;code&gt;ACTIVE&lt;/code&gt; rather than transitioning through &lt;code&gt;CREATING&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;The wasm engine gained an operation-level &lt;code&gt;execute&lt;/code&gt; API, and a new npm package, &lt;code&gt;@dynoxide/wasm-engine&lt;/code&gt;, that ships it. The Worker answers a small versioned RPC - &lt;code&gt;open&lt;/code&gt;, &lt;code&gt;execute&lt;/code&gt;, &lt;code&gt;capabilities&lt;/code&gt;, &lt;code&gt;contractVersion&lt;/code&gt; - with &lt;code&gt;{id, op, payload}&lt;/code&gt; in and &lt;code&gt;{id, ok, result|error}&lt;/code&gt; out, and a bundled &lt;code&gt;EngineClient&lt;/code&gt; owns the round trip so you deal in objects instead of hand-building postMessage envelopes. &lt;code&gt;npm run build:wasm&lt;/code&gt; assembles the package: the Worker, the two &lt;code&gt;.wasm&lt;/code&gt;, the &lt;code&gt;EngineClient&lt;/code&gt;, and a &lt;code&gt;manifest.json&lt;/code&gt; stamped with the engine and contract versions. Depend on that built package, not this repo&#39;s source. The client checks its &lt;code&gt;CONTRACT_VERSION&lt;/code&gt; against the engine on boot and fails loudly if they differ, so a stale embed can&#39;t quietly mis-read a newer one. The package ships TypeScript types for the client. Still a preview: the wasm path isn&#39;t run against the conformance suite.&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;On the wasm backend, the per-write and per-delete secondary-index fan-out now crosses the JS bridge once per index type rather than once per index operation. Keeping a table&#39;s GSIs and LSIs in step with a write is a delete and a re-insert per index, each previously its own bridge crossing; a new &lt;code&gt;exec_script&lt;/code&gt; primitive carries the whole ordered batch over in a single crossing, so an indexed &lt;code&gt;PutItem&lt;/code&gt; or &lt;code&gt;DeleteItem&lt;/code&gt; on a table with K GSIs and L LSIs drops from order K+L crossings to a constant two. Index contents and native behaviour are unchanged (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/85&quot;&gt;#85&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;The browser backend moved from &lt;code&gt;wa-sqlite&lt;/code&gt; to the official &lt;a href=&quot;https://github.com/sqlite/sqlite-wasm&quot;&gt;&lt;code&gt;@sqlite.org/sqlite-wasm&lt;/code&gt;&lt;/a&gt; engine, maintained by the SQLite team and versioned to track SQLite releases. The bridge now runs through the &lt;code&gt;sqlite3.oo1&lt;/code&gt; API over the OPFS SAHPool VFS, which keeps the no-COOP/COEP guarantee that motivated the original VFS choice (it needs no &lt;code&gt;SharedArrayBuffer&lt;/code&gt;). The &lt;code&gt;open&lt;/code&gt;/&lt;code&gt;exec&lt;/code&gt;/&lt;code&gt;query&lt;/code&gt;/&lt;code&gt;close&lt;/code&gt; contract is unchanged, so consumers of &lt;code&gt;@dynoxide/wasm-engine&lt;/code&gt; need no code change. A busy database now recovers once the holder releases it rather than staying busy until reload, and the full 64-bit integer round-trip and the &lt;code&gt;fnv1a_hash&lt;/code&gt; scalar are re-proven on the new engine (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/61&quot;&gt;#61&lt;/a&gt;). The shipped SQLite &lt;code&gt;.wasm&lt;/code&gt; is larger than before (~845 KB against wa-sqlite&#39;s ~545 KB).&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;An empty-binary key value now surfaces as a top-level &lt;code&gt;ValidationException&lt;/code&gt; on every path, matching DynamoDB. Previously the lookup path (&lt;code&gt;GetItem&lt;/code&gt;/&lt;code&gt;DeleteItem&lt;/code&gt;/&lt;code&gt;UpdateItem&lt;/code&gt;, batch, and a transact &lt;code&gt;Update&lt;/code&gt;/&lt;code&gt;Delete&lt;/code&gt;/&lt;code&gt;ConditionCheck&lt;/code&gt; &lt;code&gt;Key&lt;/code&gt;) returned the older &lt;code&gt;...were invalid:...&lt;/code&gt; wording and, inside a transaction, a &lt;code&gt;ValidationError&lt;/code&gt; cancellation reason rather than hoisting; the same cancellation-instead-of-hoist gap also affected an empty-binary table item key and an empty-binary secondary-index key in a transaction. This is the binary counterpart to the empty-string key fix &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/98&quot;&gt;#98&lt;/a&gt;; real DynamoDB returns the same top-level &lt;code&gt;...are not valid. ... empty binary value...&lt;/code&gt; messages (table keys, and the put and update forms for secondary-index keys), confirmed identical across four regions.&lt;/li&gt;
&lt;li&gt;Inside a &lt;code&gt;TransactWriteItems&lt;/code&gt;, an empty-string value in the lookup &lt;code&gt;Key&lt;/code&gt; of an &lt;code&gt;Update&lt;/code&gt;, &lt;code&gt;Delete&lt;/code&gt;, or &lt;code&gt;ConditionCheck&lt;/code&gt; was wrapped in a &lt;code&gt;TransactionCanceledException&lt;/code&gt;; it now surfaces as a top-level &lt;code&gt;ValidationException&lt;/code&gt;, matching DynamoDB and completing the empty-string key fix &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/95&quot;&gt;#95&lt;/a&gt; made for the &lt;code&gt;Put&lt;/code&gt; item key. Wrong-type and non-scalar lookup keys still cancel with a &lt;code&gt;ValidationError&lt;/code&gt; reason, and the corrected empty-string message now also matches DynamoDB on the single-action &lt;code&gt;GetItem&lt;/code&gt;/&lt;code&gt;DeleteItem&lt;/code&gt;/&lt;code&gt;UpdateItem&lt;/code&gt; and batch lookup paths (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/98&quot;&gt;#98&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;BatchWriteItem&lt;/code&gt; now reports a wrong-type or non-scalar table key in a put request with DynamoDB&#39;s generic &lt;code&gt;The provided key element does not match the schema&lt;/code&gt;, rather than borrowing &lt;code&gt;PutItem&lt;/code&gt;&#39;s &lt;code&gt;Type mismatch for key ...&lt;/code&gt; wording. Real DynamoDB collapses both cases to the schema error inside a batch. The empty-string table-key message and the secondary-index key messages already matched and are unchanged, and &lt;code&gt;PutItem&lt;/code&gt; and the other put-shaped paths keep the specific type-mismatch message (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/97&quot;&gt;#97&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;Query&lt;/code&gt; and &lt;code&gt;Scan&lt;/code&gt; now reject two &lt;code&gt;Select&lt;/code&gt;/&lt;code&gt;ProjectionExpression&lt;/code&gt; combinations that real DynamoDB rejects before reading any item, where dynoxide previously returned results: a &lt;code&gt;ProjectionExpression&lt;/code&gt; with any &lt;code&gt;Select&lt;/code&gt; other than &lt;code&gt;SPECIFIC_ATTRIBUTES&lt;/code&gt; (such as &lt;code&gt;ALL_ATTRIBUTES&lt;/code&gt;), and &lt;code&gt;Select: ALL_PROJECTED_ATTRIBUTES&lt;/code&gt; without an &lt;code&gt;IndexName&lt;/code&gt;. Both now return a &lt;code&gt;ValidationException&lt;/code&gt; with DynamoDB&#39;s message (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/96&quot;&gt;#96&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;Inside a &lt;code&gt;TransactWriteItems&lt;/code&gt;, a key (table or secondary index) carrying an empty string was wrapped in a &lt;code&gt;TransactionCanceledException&lt;/code&gt;; it now surfaces as a top-level &lt;code&gt;ValidationException&lt;/code&gt;, matching DynamoDB. Wrong-type and non-scalar key values still cancel with a &lt;code&gt;ValidationError&lt;/code&gt; reason, so only the empty-string case changes. A non-scalar table key no longer fails as an internal error before the transaction runs, and an update that sets a secondary-index key to an empty string now returns DynamoDB&#39;s distinct update-path message rather than the put-shaped one (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/95&quot;&gt;#95&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;A write whose secondary-index (GSI or LSI) key attribute is the wrong type, a non-scalar, or an empty string is now rejected with a &lt;code&gt;ValidationException&lt;/code&gt; matching DynamoDB&#39;s exact message, where before it was silently accepted (kept out of the index but still written to the base table). Validation runs on every write path - put, update, batch, transactional, PartiQL, and import. An update only re-checks an index key it actually changes, so an unrelated update to a row holding a pre-existing bad value still succeeds (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/92&quot;&gt;#92&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;A &lt;code&gt;Scan&lt;/code&gt; or &lt;code&gt;Query&lt;/code&gt; on a composite global secondary index no longer returns items that are missing the index sort key; they are now excluded from the index (sparse-index behaviour), matching DynamoDB. Index membership was gated on the partition key alone, so an item carrying the partition key but no sort key was written into the index at an empty sort-key position. Membership is now a single shared rule across both global and local secondary indexes, applied on every write path - put, update, batch, transactional, PartiQL, import, and GSI backfill - and it also excludes an item whose index key attribute is present but not a scalar. In-memory databases start fresh each run and are unaffected; only a file-backed database, or a snapshot taken from one, written by an older build carries stray index rows. They clear as each affected item is next written, and a persisted store can rebuild an index by dropping and re-adding it (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/91&quot;&gt;#91&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;PutItem&lt;/code&gt; and the other write paths now accept a &lt;code&gt;{&amp;quot;NULL&amp;quot;: false}&lt;/code&gt; attribute value and read it back as &lt;code&gt;{&amp;quot;NULL&amp;quot;: true}&lt;/code&gt;, where before they rejected it with &lt;code&gt;One or more parameter values were invalid: Null attribute value types must have the value of true&lt;/code&gt;. The NULL member is typed as a plain boolean in the model, so &lt;code&gt;false&lt;/code&gt; was valid input all along; AWS has dropped the server-side true-only rule and normalises &lt;code&gt;false&lt;/code&gt; to &lt;code&gt;true&lt;/code&gt; on read, and dynoxide now matches. A non-boolean NULL such as &lt;code&gt;{&amp;quot;NULL&amp;quot;: &amp;quot;no&amp;quot;}&lt;/code&gt; is still rejected as a type error (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/62&quot;&gt;#62&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;Hardened the wasm engine preview ahead of a stable &lt;code&gt;@dynoxide/wasm-engine&lt;/code&gt; publish. A body-less operation such as &lt;code&gt;ListTables&lt;/code&gt; now round-trips instead of failing as a &lt;code&gt;SerializationException&lt;/code&gt; (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/65&quot;&gt;#65&lt;/a&gt;). OPFS open tells a busy database (another tab holding its lock) apart from one that is genuinely unavailable: the busy case surfaces a stable &lt;code&gt;com.dynoxide.wasm#OpfsUnavailable&lt;/code&gt; error rather than silently forking to a separate in-memory store, while a private window or quota error still degrades to an ephemeral session. Re-opening opens the new database before closing the old, so a failed re-open leaves the working session intact, and closing a database releases its OPFS handles so the name is free for another tab (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/64&quot;&gt;#64&lt;/a&gt;). The bridge round-trips full 64-bit integers, and a cross-backend test pins the &lt;code&gt;fnv1a_hash&lt;/code&gt; scalar the wasm and native backends share (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/61&quot;&gt;#61&lt;/a&gt;). A headless-browser CI job exercises the shipped bundle against the real wasm engine and OPFS on every PR (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/68&quot;&gt;#68&lt;/a&gt;).&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.10.0</title>
    <link href="https://dynoxide.dev/changelog#0.10.0"/>
    <id>https://dynoxide.dev/changelog#0.10.0</id>
    <updated>2026-05-29T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;A &lt;code&gt;StorageBackend&lt;/code&gt; trait in the new &lt;code&gt;dynoxide::storage_backend&lt;/code&gt; module, decoupling the data layer from a specific SQLite binding. The native rusqlite-backed &lt;code&gt;Storage&lt;/code&gt; implements the trait, and the action handlers and &lt;code&gt;Database&lt;/code&gt; now consume it (see Changed). The trait surface also carries a &lt;code&gt;clock()&lt;/code&gt; accessor for the stream and TTL paths and batch-shaped &lt;code&gt;put_base_items&lt;/code&gt; / &lt;code&gt;insert_gsi_items&lt;/code&gt; methods that replaced the last two raw &lt;code&gt;Storage::conn()&lt;/code&gt; escape hatches in the handlers.&lt;/li&gt;
&lt;li&gt;A &lt;code&gt;BackendError&lt;/code&gt; enum returned by the trait surface, with an explicit &lt;code&gt;rusqlite::Error -&amp;gt; BackendError&lt;/code&gt; mapping for the common failure modes (&lt;code&gt;NotADatabase&lt;/code&gt;, locked / busy, constraint violations, I/O failures), plus an &lt;code&gt;Unsupported { capability }&lt;/code&gt; variant for a capability a backend cannot serve (the wasm preview uses it for TTL). It is &lt;code&gt;#[non_exhaustive]&lt;/code&gt; so future backends can add failure modes without a breaking change.&lt;/li&gt;
&lt;li&gt;A &lt;code&gt;Clock&lt;/code&gt; capability on &lt;code&gt;Storage&lt;/code&gt; so the trait surface does not assume &lt;code&gt;std::time&lt;/code&gt;. Stream and TTL paths route their &lt;code&gt;created_at&lt;/code&gt; and sweep timestamps through the clock; &lt;code&gt;SystemClock&lt;/code&gt; is the default and &lt;code&gt;ManualClock&lt;/code&gt; ships as a deterministic test helper. Other &lt;code&gt;std::time&lt;/code&gt; call sites (idempotency cache, action-handler timestamps, snapshots) remain native-only and are unchanged.&lt;/li&gt;
&lt;li&gt;A &lt;code&gt;wasm-sqlite&lt;/code&gt; cargo feature and a working WebAssembly backend. dynoxide compiles to &lt;code&gt;wasm32-unknown-unknown&lt;/code&gt; and runs in the browser against &lt;a href=&quot;https://github.com/rhashimoto/wa-sqlite&quot;&gt;wa-sqlite&lt;/a&gt; (a WASM build of SQLite) over a wasm-bindgen bridge, persisting to OPFS. &lt;code&gt;WasmBridgeBackend&lt;/code&gt; implements &lt;code&gt;StorageBackend&lt;/code&gt;, and &lt;code&gt;WasmDatabase&lt;/code&gt; (&lt;code&gt;Database&amp;lt;WasmBridgeBackend&amp;gt;&lt;/code&gt;) exposes the handlers as &lt;code&gt;async fn&lt;/code&gt; with no &lt;code&gt;block_on&lt;/code&gt;. It covers create-table, put, get, delete, query, and scan over base tables and both index types (GSI and LSI), with index fan-out atomic with the base write. TTL returns &lt;code&gt;BackendError::Unsupported&lt;/code&gt;; streams return a preview &amp;quot;not yet implemented&amp;quot; error pending a delivery design; &lt;code&gt;TransactWriteItems&lt;/code&gt;, tags, table-setting updates, stats, and bulk import are preview placeholders. The native and wasm backends share one set of SQL builders (&lt;code&gt;storage_backend::sql_builders&lt;/code&gt;), so both issue identical SQL.&lt;/li&gt;
&lt;li&gt;A self-contained browser build: &lt;code&gt;npm run build:wasm&lt;/code&gt; (wasm-pack + esbuild) emits a &lt;code&gt;dist/&lt;/code&gt; of three files - a bundled Web Worker plus the two &lt;code&gt;.wasm&lt;/code&gt; assets (dynoxide ~550 KB, wa-sqlite ~545 KB; ~1.2 MB total). The engine runs in a Web Worker because wa-sqlite&#39;s OPFS persistence uses synchronous access handles, which browsers expose only in a Worker; pairing wa-sqlite&#39;s synchronous VFS (&lt;code&gt;AccessHandlePoolVFS&lt;/code&gt;) with its non-async build needs no &lt;code&gt;SharedArrayBuffer&lt;/code&gt;, and so &lt;strong&gt;no cross-origin isolation (COOP/COEP)&lt;/strong&gt; - it drops onto ordinary static hosting. A build-visible &lt;code&gt;WASM_PREVIEW&lt;/code&gt; constant (&lt;code&gt;true&lt;/code&gt; under &lt;code&gt;wasm-sqlite&lt;/code&gt;) marks the preview. The harness under &lt;code&gt;harness/&lt;/code&gt; loads the same bundled Worker that ships, so a green harness means the shipping artefact works; it exercises CRUD, GSI query/scan, and error-envelope fidelity on OPFS. CI builds the &lt;code&gt;wasm32-unknown-unknown&lt;/code&gt; target for both the &lt;code&gt;wasm-sqlite&lt;/code&gt; and &lt;code&gt;wasm-harness&lt;/code&gt; features on every PR, so the harness&#39;s use of &lt;code&gt;WasmDatabase&lt;/code&gt; and the action types is type-checked too.&lt;/li&gt;
&lt;li&gt;Official Docker image. &lt;code&gt;docker run -p 8000:8000 ghcr.io/nubo-db/dynoxide&lt;/code&gt; is a ~5 MB drop-in for &lt;code&gt;amazon/dynamodb-local&lt;/code&gt; in containerised test suites: multi-arch (&lt;code&gt;linux/amd64&lt;/code&gt; and &lt;code&gt;linux/arm64&lt;/code&gt;), &lt;code&gt;FROM scratch&lt;/code&gt;, published to GHCR on each release with Docker Hub and ECR Public mirrors pushed best-effort. The image ships a &lt;code&gt;HEALTHCHECK&lt;/code&gt; backed by a new &lt;code&gt;dynoxide healthcheck&lt;/code&gt; subcommand, so &lt;code&gt;docker ps&lt;/code&gt; and Compose health gates report status without extra tooling (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/3&quot;&gt;#3&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;SECURITY.md&lt;/code&gt;, documenting the MCP HTTP transport&#39;s threat model: the bearer-token authentication it now requires, plus the Host and Origin allowlists that back it (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/27&quot;&gt;#27&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;MCP HTTP transport options: &lt;code&gt;--mcp-host&lt;/code&gt;/&lt;code&gt;--host&lt;/code&gt; to bind beyond loopback, &lt;code&gt;--mcp-allowed-host&lt;/code&gt;/&lt;code&gt;--allowed-host&lt;/code&gt; to accept additional &lt;code&gt;Host&lt;/code&gt; headers by name, and &lt;code&gt;--mcp-no-auth&lt;/code&gt;/&lt;code&gt;--no-auth&lt;/code&gt; to disable authentication on loopback binds only. With a token set, these make the transport reachable from outside a container, unblocking the Docker MCP path (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/24&quot;&gt;#24&lt;/a&gt;).&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;Database&lt;/code&gt; is now generic over its storage backend: &lt;code&gt;Database&amp;lt;S&amp;gt;&lt;/code&gt;, monomorphised, no &lt;code&gt;dyn&lt;/code&gt;. The parameter defaults to the native rusqlite backend, so existing code that names &lt;code&gt;Database&lt;/code&gt; is unaffected, and a new &lt;code&gt;NativeDatabase&lt;/code&gt; alias names that default explicitly. The action handlers are now &lt;code&gt;async&lt;/code&gt; and route through the &lt;code&gt;StorageBackend&lt;/code&gt; trait. &lt;code&gt;NativeDatabase&lt;/code&gt; keeps the historical synchronous public API: each method drives the handler future to completion with &lt;code&gt;block_on&lt;/code&gt; (via &lt;code&gt;pollster&lt;/code&gt;), and because the native backend&#39;s futures never suspend, that &lt;code&gt;block_on&lt;/code&gt; never parks the thread, so it stays safe inside the tokio-based HTTP and MCP servers.&lt;/li&gt;
&lt;li&gt;&lt;code&gt;DynoxideError&lt;/code&gt; is now &lt;code&gt;#[non_exhaustive]&lt;/code&gt;. Match arms in downstream code must include a wildcard. Done now, while 0.10.0 is already a breaking release, so later variant additions stay non-breaking.&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Breaking:&lt;/strong&gt; the MCP HTTP transport (&lt;code&gt;dynoxide mcp --http&lt;/code&gt;, &lt;code&gt;dynoxide serve --mcp&lt;/code&gt;) now requires bearer-token authentication on every request. On a loopback bind, dynoxide generates a token on first run, persists it to a per-user config file, and prints a client-config snippet; later runs reuse it silently. &lt;strong&gt;Existing clients break until updated&lt;/strong&gt;: add &lt;code&gt;&amp;quot;headers&amp;quot;: { &amp;quot;Authorization&amp;quot;: &amp;quot;Bearer &amp;lt;token&amp;gt;&amp;quot; }&lt;/code&gt; to your MCP client config. A non-loopback bind requires an explicit token via &lt;code&gt;--mcp-token&lt;/code&gt;/&lt;code&gt;--token&lt;/code&gt; or &lt;code&gt;DYNOXIDE_MCP_AUTH_TOKEN&lt;/code&gt; and will not start without one. The stdio transport is unaffected (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/27&quot;&gt;#27&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Breaking (library API):&lt;/strong&gt; &lt;code&gt;dynoxide::mcp::serve_http&lt;/code&gt; and &lt;code&gt;serve_http_with_shutdown&lt;/code&gt; now take an &lt;code&gt;HttpOptions&lt;/code&gt; struct (bind host, &lt;code&gt;AuthMode&lt;/code&gt;, extra allowed hosts) in place of a bare &lt;code&gt;port: u16&lt;/code&gt;. Embedders constructing the MCP HTTP server must build &lt;code&gt;HttpOptions&lt;/code&gt; and choose an &lt;code&gt;AuthMode&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;&lt;code&gt;rusqlite&lt;/code&gt; is now an optional dependency behind the &lt;code&gt;native-sqlite&lt;/code&gt; feature (on by default, so native builds are unchanged). The crate type-checks with rusqlite absent, which is the precondition for the wasm build. Cross-platform wall-clock paths (the idempotency cache, &lt;code&gt;created_at&lt;/code&gt; stamps, and &lt;code&gt;SystemClock&lt;/code&gt;) moved to &lt;code&gt;web-time&lt;/code&gt; - &lt;code&gt;std::time&lt;/code&gt; on native, the browser clock on wasm. The native binary now builds behind a &lt;code&gt;cli&lt;/code&gt; marker feature (pulled in by &lt;code&gt;http-server&lt;/code&gt;, &lt;code&gt;mcp-server&lt;/code&gt;, and &lt;code&gt;import&lt;/code&gt;), so it is skipped in backend-neutral builds such as &lt;code&gt;--features wasm-sqlite&lt;/code&gt;. The &lt;code&gt;DynoxideError::SqliteError&lt;/code&gt; variant is consequently &lt;code&gt;native-sqlite&lt;/code&gt;-gated and absent on backend-neutral builds, which matters only for code that matches it by name on a wasm target.&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;PartiQL &lt;code&gt;DELETE&lt;/code&gt; and &lt;code&gt;UPDATE&lt;/code&gt; now evaluate the non-key predicates in a &lt;code&gt;WHERE&lt;/code&gt; clause instead of acting on the key alone. Before, the executor pulled the primary key out of the &lt;code&gt;WHERE&lt;/code&gt; and ignored the rest, so &lt;code&gt;DELETE FROM &amp;quot;t&amp;quot; WHERE pk = &#39;a&#39; AND NOT begins_with(name, &#39;x&#39;)&lt;/code&gt; deleted the row even when &lt;code&gt;name&lt;/code&gt; began with &lt;code&gt;x&lt;/code&gt;, mutating a row the filter should have excluded (a data-correctness bug predating v0.9.5). The write paths now run the full condition against the fetched item, the same &lt;code&gt;matches_where&lt;/code&gt; pass &lt;code&gt;SELECT&lt;/code&gt; already uses: a present item whose non-key predicate is false raises &lt;code&gt;ConditionalCheckFailedException&lt;/code&gt;, matching how AWS treats a PartiQL write whose condition fails, and a missing item stays a silent no-op (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/54&quot;&gt;#54&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;DescribeTable&lt;/code&gt; now returns a stable &lt;code&gt;TableId&lt;/code&gt; instead of a freshly generated UUID on every call. The id is a random UUID assigned once at create time and persisted (a new &lt;code&gt;table_id&lt;/code&gt; column, added to existing databases through the versioned schema migration and backfilled), so it stays the same across calls, &lt;code&gt;CreateTable&lt;/code&gt; returns the same value, and a dropped-and-recreated table gets a new one, matching AWS (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/55&quot;&gt;#55&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;UpdateItem&lt;/code&gt; evaluates an &lt;code&gt;UpdateExpression&lt;/code&gt; against the pre-update item image and accepts parenthesised arithmetic. &lt;code&gt;SET a = :v, b = a&lt;/code&gt; now gives &lt;code&gt;b&lt;/code&gt; the old value of &lt;code&gt;a&lt;/code&gt; rather than the value assigned earlier in the same call, and &lt;code&gt;SET c = (c - :v)&lt;/code&gt; parses and applies on the &lt;code&gt;BigDecimal&lt;/code&gt; path instead of being rejected with &lt;code&gt;Expected operand in SET, got (&lt;/code&gt; (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/35&quot;&gt;#35&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;UpdateItem&lt;/code&gt; &lt;code&gt;ReturnValues: UPDATED_NEW&lt;/code&gt; matches AWS granularity. A nested &lt;code&gt;SET parent.child = :v&lt;/code&gt; returns only the changed fragment &lt;code&gt;{parent: {M: {child}}}&lt;/code&gt; instead of the whole &lt;code&gt;parent&lt;/code&gt; map, and a REMOVE-only update omits &lt;code&gt;Attributes&lt;/code&gt; entirely rather than returning an empty map (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/36&quot;&gt;#36&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;Paginating a &lt;code&gt;Query&lt;/code&gt; over a GSI no longer drops items when several entries share the same index key and the base table has only a partition key. On a hash-only base table the continuation cursor lost its base-key component and stalled after the first page, the same defect &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/38&quot;&gt;#38&lt;/a&gt; fixed for &lt;code&gt;Scan&lt;/code&gt;; the &lt;code&gt;Query&lt;/code&gt; path now carries the base partition key, so every tied item is returned across the paged walk (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/52&quot;&gt;#52&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;TransactWriteItems&lt;/code&gt;, &lt;code&gt;TransactGetItems&lt;/code&gt; and PartiQL &lt;code&gt;ExecuteStatement&lt;/code&gt; now report &lt;code&gt;ConsumedCapacity&lt;/code&gt; the way AWS does. A transactional write charges 2 WCU per item and a transactional read 2 RCU per item including a missing one (each item rounded up before the 2x factor); the &lt;code&gt;TransactGetItems&lt;/code&gt; &lt;code&gt;INDEXES&lt;/code&gt; breakdown carries &lt;code&gt;Table.ReadCapacityUnits&lt;/code&gt;; and &lt;code&gt;ExecuteStatement&lt;/code&gt; returns the &lt;code&gt;ConsumedCapacity&lt;/code&gt; block whenever &lt;code&gt;ReturnConsumedCapacity&lt;/code&gt; is requested instead of omitting it (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/37&quot;&gt;#37&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;PartiQL &lt;code&gt;ExecuteStatement&lt;/code&gt; accepts the bracket &lt;code&gt;IN [...]&lt;/code&gt; list form, not just &lt;code&gt;IN (...)&lt;/code&gt;, and evaluates &lt;code&gt;NOT begins_with(...)&lt;/code&gt; as a negated predicate. &lt;code&gt;IS NOT MISSING&lt;/code&gt; already evaluated; the gaps were the bracket list and the &lt;code&gt;NOT&lt;/code&gt; function arm, which the same statement bundles together (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/40&quot;&gt;#40&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;DescribeTable&lt;/code&gt; now round-trips &lt;code&gt;OnDemandThroughput&lt;/code&gt; and reports the full &lt;code&gt;SSEDescription&lt;/code&gt; shape. A table created with &lt;code&gt;OnDemandThroughput&lt;/code&gt; reports its &lt;code&gt;MaxReadRequestUnits&lt;/code&gt; and &lt;code&gt;MaxWriteRequestUnits&lt;/code&gt; back; the value lives in a new &lt;code&gt;on_demand_throughput&lt;/code&gt; column added through the versioned schema migration, so existing on-disk databases pick it up on open. Server-side encryption enabled with the AWS-managed key now reports &lt;code&gt;SSEType: KMS&lt;/code&gt; and a &lt;code&gt;KMSMasterKeyArn&lt;/code&gt; alongside &lt;code&gt;Status: ENABLED&lt;/code&gt;, where before it returned the status alone (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/44&quot;&gt;#44&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;UpdateTable&lt;/code&gt; now accepts a lone &lt;code&gt;TableClass&lt;/code&gt; or &lt;code&gt;OnDemandThroughput&lt;/code&gt; change instead of rejecting it with &lt;code&gt;At least one of ProvisionedThroughput, BillingMode, ... is required&lt;/code&gt;. Both fields are validated (an unknown &lt;code&gt;TableClass&lt;/code&gt; is a &lt;code&gt;ValidationException&lt;/code&gt;) and persisted, so the change shows up on the next &lt;code&gt;DescribeTable&lt;/code&gt; (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/45&quot;&gt;#45&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;DeleteTable&lt;/code&gt; on a table with deletion protection enabled now returns the exact AWS message, &lt;code&gt;Resource cannot be deleted as it is currently protected against deletion. Disable deletion protection first.&lt;/code&gt;, in place of the ARN-prefixed wording dynoxide used before (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/46&quot;&gt;#46&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;TransactGetItems&lt;/code&gt; now omits &lt;code&gt;Item&lt;/code&gt; from a response entry when a &lt;code&gt;ProjectionExpression&lt;/code&gt; matches no attribute on an otherwise-present item, matching AWS. The projection always re-injects the table key, so the entry previously came back as a key-only object instead of being omitted (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/39&quot;&gt;#39&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;BatchWriteItem&lt;/code&gt; now rejects a &lt;code&gt;PutRequest&lt;/code&gt; whose item is missing the table key with a 400 &lt;code&gt;ValidationException&lt;/code&gt; rather than a 500 &lt;code&gt;InternalServerError&lt;/code&gt;. The duplicate-key detection pass extracted keys before validating them; it now validates first, the same ordering the single-item write paths already use (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/39&quot;&gt;#39&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;Paginating a &lt;code&gt;Scan&lt;/code&gt; over a GSI no longer drops items when several entries share the same index key and the base table has only a partition key. On a hash-only base table the continuation cursor lost its base-key component and stalled after the first page; it now carries the base partition key, so every tied item is returned across the paged walk (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/38&quot;&gt;#38&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;A single-item write (&lt;code&gt;PutItem&lt;/code&gt;, &lt;code&gt;DeleteItem&lt;/code&gt;, &lt;code&gt;UpdateItem&lt;/code&gt;) and its GSI/LSI index fan-out now run in a single transaction. A failure partway through the fan-out rolls the whole write back rather than leaving a base row with a half-applied (torn) index. The same per-item atomicity now also covers &lt;code&gt;BatchWriteItem&lt;/code&gt; (each write request) and the TTL sweep (each expired-item delete). This matches DynamoDB, where a single-item write does not half-apply to its indexes.&lt;/li&gt;
&lt;li&gt;Write paths now roll back on a failed &lt;code&gt;COMMIT&lt;/code&gt; and surface a failed &lt;code&gt;ROLLBACK&lt;/code&gt; rather than leaving the connection stuck mid-transaction, which would make the next write fail. Every write path shares one transaction helper for this.&lt;/li&gt;
&lt;li&gt;A client-facing &lt;code&gt;ValidationException&lt;/code&gt; raised inside a backend method (the 50-tag limit in &lt;code&gt;set_tags&lt;/code&gt;) keeps its 400 status across the &lt;code&gt;StorageBackend&lt;/code&gt; boundary instead of collapsing to a 500.&lt;/li&gt;
&lt;li&gt;Tighter expression and scan validation, to match what real DynamoDB rejects
(surfaced by the conformance suite). Dynoxide now turns away redundant
parentheses like &lt;code&gt;((a = :b))&lt;/code&gt; in condition, filter, and key-condition
expressions; &lt;code&gt;contains(x, x)&lt;/code&gt; with the same operand on both sides; and
&lt;code&gt;begins_with&lt;/code&gt; handed a number instead of a string or binary. These are
rejected up front, before any items are scanned
(&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/31&quot;&gt;#31&lt;/a&gt;).&lt;/li&gt;
&lt;li&gt;&lt;code&gt;size()&lt;/code&gt; now measures strings in UTF-16 code units rather than bytes, so
values with emoji or accented characters report the length DynamoDB returns.&lt;/li&gt;
&lt;li&gt;A negative &lt;code&gt;Segment&lt;/code&gt; on a parallel scan is now rejected rather than accepted.&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Notes&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Existing native code that names &lt;code&gt;Database&lt;/code&gt; keeps working unchanged: the new generic parameter defaults to the rusqlite backend and the synchronous method signatures are identical. The one deliberate behaviour change is index fan-out atomicity (see Fixed); it is more DynamoDB-correct, and the conformance suite still passes. Tests, conformance, and benchmarks pass against the same observable surface as before.&lt;/li&gt;
&lt;li&gt;Building dynoxide for &lt;code&gt;wasm32-unknown-unknown&lt;/code&gt; is now supported via the &lt;code&gt;wasm-sqlite&lt;/code&gt; feature (see Added). The wasm backend is a preview: it is not run against the conformance suite that covers the native build, so its correctness rests on its own CRUD/query/scan/GSI/LSI tests for now. The engine runs in a Web Worker (OPFS&#39;s synchronous file handles are Worker-only) and needs no cross-origin isolation, so it works on ordinary static hosting.&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.13</title>
    <link href="https://dynoxide.dev/changelog#0.9.13"/>
    <id>https://dynoxide.dev/changelog#0.9.13</id>
    <updated>2026-05-11T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Security&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;
&lt;p&gt;Close a DNS rebinding vulnerability in the MCP HTTP transport
(&lt;a href=&quot;https://github.com/modelcontextprotocol/rust-sdk/security/advisories/GHSA-89vp-x53w-74fx&quot;&gt;GHSA-89vp-x53w-74fx&lt;/a&gt; /
&lt;a href=&quot;https://www.cve.org/CVERecord?id=CVE-2026-42559&quot;&gt;CVE-2026-42559&lt;/a&gt;) by upgrading &lt;code&gt;rmcp&lt;/code&gt;
from 1.1.1 to 1.6.0 in both lockfiles. A malicious page could make the
user&#39;s browser send requests to a loopback MCP server with a non-loopback
&lt;code&gt;Host&lt;/code&gt; header, which the server would then process. Affects 0.9.3 to 0.9.12.
Users running &lt;code&gt;dynoxide mcp --http&lt;/code&gt; or &lt;code&gt;dynoxide serve --mcp&lt;/code&gt; should
upgrade; stdio transport is unaffected.&lt;/p&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;p&gt;Close a related cross-origin CSRF gap: a page could &lt;code&gt;fetch&lt;/code&gt; the loopback
endpoint with &lt;code&gt;mode: &#39;no-cors&#39;&lt;/code&gt;, and the Host check would pass while the
Origin header went unchecked. Affected write tools: &lt;code&gt;put_item&lt;/code&gt;,
&lt;code&gt;update_item&lt;/code&gt;, &lt;code&gt;delete_item&lt;/code&gt;, &lt;code&gt;create_table&lt;/code&gt;, and &lt;code&gt;batch_write_item&lt;/code&gt;.
Fixed by setting an explicit Host and Origin allowlist on
&lt;code&gt;StreamableHttpServerConfig&lt;/code&gt;. Native MCP clients (Claude Code, Cursor,
the dynoxide CLI) don&#39;t send an Origin header and are unaffected.&lt;/p&gt;
&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.12</title>
    <link href="https://dynoxide.dev/changelog#0.9.12"/>
    <id>https://dynoxide.dev/changelog#0.9.12</id>
    <updated>2026-05-04T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;
&lt;p&gt;Unix: port releases immediately after &lt;code&gt;dynoxide serve&lt;/code&gt; shuts down. The listener used to skip &lt;code&gt;SO_REUSEADDR&lt;/code&gt;, leaving leftover &lt;code&gt;TIME_WAIT&lt;/code&gt; sockets from connected clients to block restart for ~60s. Live-listener conflict detection is unaffected: &lt;code&gt;SO_REUSEADDR&lt;/code&gt; only bypasses &lt;code&gt;TIME_WAIT&lt;/code&gt;, not active sockets.&lt;/p&gt;
&lt;p&gt;Windows: unchanged. &lt;code&gt;SO_REUSEADDR&lt;/code&gt; lets another process hijack an active bind there, so we leave it off.&lt;/p&gt;
&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.11</title>
    <link href="https://dynoxide.dev/changelog#0.9.11"/>
    <id>https://dynoxide.dev/changelog#0.9.11</id>
    <updated>2026-05-04T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;dynoxide serve --mcp&lt;/code&gt; now exits cleanly on Ctrl+C when an MCP client (Claude Code, Cursor) is holding a connection open. The MCP server&#39;s graceful-shutdown drain used to wait for those connections forever, hanging the process until something SIGKILLed it (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/22&quot;&gt;#22&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Security&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Refresh &lt;code&gt;Cargo.lock&lt;/code&gt; for the dependabot patches reachable within MSRV: &lt;code&gt;aws-lc-sys&lt;/code&gt; 0.37.1 to 0.40.0 (5 high-severity AWS-LC issues), &lt;code&gt;openssl&lt;/code&gt; 0.10.75 to 0.10.79 (5 buffer-overflow advisories), &lt;code&gt;rand&lt;/code&gt; 0.8.5 to 0.8.6. Remaining &lt;code&gt;rustls-webpki&lt;/code&gt; / &lt;code&gt;time&lt;/code&gt; / &lt;code&gt;aws-sdk-dynamodb&lt;/code&gt; alerts are dev-dependency only (test-suite AWS SDK chain, not the production binary) and stay pinned by MSRV 1.85 until v0.10.0&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.10</title>
    <link href="https://dynoxide.dev/changelog#0.9.10"/>
    <id>https://dynoxide.dev/changelog#0.9.10</id>
    <updated>2026-04-27T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;16 places where dynoxide&#39;s error strings drifted from real AWS DynamoDB. Mostly small things you only notice when you assert the message: &lt;code&gt;tableName&lt;/code&gt; length validation is now per-operation (1 char on read/write, 3 stays on &lt;code&gt;CreateTable&lt;/code&gt;), &lt;code&gt;Select&lt;/code&gt; enum order matches AWS rather than alphabetical, &lt;code&gt;Query&lt;/code&gt; vs &lt;code&gt;Scan&lt;/code&gt; &lt;code&gt;Limit=0&lt;/code&gt; messages are different on purpose now, batch/transact empty and oversize requests use the standard validation envelope, and &lt;code&gt;UpdateExpression&lt;/code&gt;/&lt;code&gt;ProjectionExpression&lt;/code&gt; syntax errors include the AWS &lt;code&gt;near: &amp;quot;...&amp;quot;&lt;/code&gt; window (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/11&quot;&gt;#11&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/12&quot;&gt;#12&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/13&quot;&gt;#13&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/15&quot;&gt;#15&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/16&quot;&gt;#16&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/17&quot;&gt;#17&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/18&quot;&gt;#18&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;&lt;code&gt;TransactGetItems&lt;/code&gt; with a bad action key now comes back as a &lt;code&gt;TransactionCanceledException&lt;/code&gt; with &lt;code&gt;ValidationError&lt;/code&gt; rather than HTTP 500. The 500 was a real leak: the dedup loop called the server-fault helper before key validation (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/19&quot;&gt;#19&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.9</title>
    <link href="https://dynoxide.dev/changelog#0.9.9"/>
    <id>https://dynoxide.dev/changelog#0.9.9</id>
    <updated>2026-04-24T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;KeyConditionExpression&lt;/code&gt; now accepts parenthesised sub-expressions, matching DynamoDB. Forms like &lt;code&gt;(#pk = :pk) AND (#sk = :sk)&lt;/code&gt; previously returned &lt;code&gt;ValidationException: Expected attribute name, got (&lt;/code&gt;. Both outer-wrap and per-condition parens are now handled (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/4&quot;&gt;#4&lt;/a&gt;, &lt;a href=&quot;https://github.com/nubo-db/dynoxide/pull/7&quot;&gt;#7&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;&lt;code&gt;UpdateItem&lt;/code&gt; and &lt;code&gt;TransactWriteItems.Update&lt;/code&gt; now evaluate &lt;code&gt;ConditionExpression&lt;/code&gt; against the existing item before populating key attributes for upsert. Previously &lt;code&gt;attribute_exists(pk)&lt;/code&gt; on a non-existent key succeeded and created a ghost item (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/pull/5&quot;&gt;#5&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Paginated &lt;code&gt;Scan&lt;/code&gt; on a GSI now returns all items when multiple items share the same GSI partition key. Previously the second page returned 0 items because the pagination cursor used only &lt;code&gt;(gsi_pk, gsi_sk)&lt;/code&gt; instead of the full 4-tuple primary key (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/pull/6&quot;&gt;#6&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;&lt;code&gt;&amp;lt;&amp;gt;&lt;/code&gt; on missing attributes now returns true, matching DynamoDB. All other comparison operators continue to return false on missing operands. Previously &lt;code&gt;&amp;lt;&amp;gt;&lt;/code&gt; also returned false, breaking &lt;code&gt;PutItem&lt;/code&gt; conditional idioms like &lt;code&gt;status &amp;lt;&amp;gt; &amp;quot;working&amp;quot;&lt;/code&gt; against fresh keys (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/pull/8&quot;&gt;#8&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.8</title>
    <link href="https://dynoxide.dev/changelog#0.9.8"/>
    <id>https://dynoxide.dev/changelog#0.9.8</id>
    <updated>2026-04-06T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Dynoxide no longer orphans when backgrounded in npm scripts (&lt;code&gt;dynoxide &amp;amp; sleep 1 &amp;amp;&amp;amp; npm run seed &amp;amp;&amp;amp; react-router dev&lt;/code&gt;) -- the port is released when the parent process exits (&lt;a href=&quot;https://github.com/nubo-db/dynoxide/issues/2&quot;&gt;nubo-db/dynoxide#2&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;The Rust server now handles SIGTERM for graceful shutdown, not just SIGINT (Ctrl+C) -- &lt;code&gt;kill &amp;lt;pid&amp;gt;&lt;/code&gt; now works as expected&lt;/li&gt;
&lt;li&gt;The npm wrapper switches from &lt;code&gt;spawnSync&lt;/code&gt; to async &lt;code&gt;spawn&lt;/code&gt; with explicit signal forwarding (SIGINT, SIGTERM, SIGHUP) and double-signal SIGKILL escalation&lt;/li&gt;
&lt;li&gt;Parent-death detection via PPID polling catches the backgrounded case where no signal is delivered to the wrapper&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.7</title>
    <link href="https://dynoxide.dev/changelog#0.9.7"/>
    <id>https://dynoxide.dev/changelog#0.9.7</id>
    <updated>2026-04-02T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Benchmark sanity checks were blocking README updates during release - 10 stale values from the v0.9.6 pipeline now corrected&lt;/li&gt;
&lt;li&gt;Binary download size in README was wrong (~5 MB, actually ~3 MB compressed / ~6 MB on disk)&lt;/li&gt;
&lt;li&gt;Docker image sizes now show both download and on-disk measurements - the old &amp;quot;225 MB&amp;quot; was the compressed download, the actual on-disk size is 471 MB&lt;/li&gt;
&lt;li&gt;MCP tool count in README was 33, should be 34 - &lt;code&gt;execute_transaction_partiql&lt;/code&gt; was missing from the list&lt;/li&gt;
&lt;li&gt;npm README had incorrect &lt;code&gt;--input&lt;/code&gt; and &lt;code&gt;--db-path&lt;/code&gt; flags for the import command (should be &lt;code&gt;--source&lt;/code&gt;, &lt;code&gt;--schema&lt;/code&gt;, &lt;code&gt;--output&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;Dropped the &lt;code&gt;serve&lt;/code&gt; subcommand from npm examples (bare &lt;code&gt;dynoxide --port 8000&lt;/code&gt; is the preferred form)&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Restructured release pipeline for token efficiency and reliability - dispatch verification, idempotent crate/npm publishing, template-based Homebrew formula updates&lt;/li&gt;
&lt;li&gt;npm publishing uses OIDC provenance via a dedicated &lt;code&gt;npm.yml&lt;/code&gt; workflow&lt;/li&gt;
&lt;li&gt;Cross-compilation switched to cargo-zigbuild for aarch64-musl targets&lt;/li&gt;
&lt;li&gt;Commit Cargo.lock for reproducible CI builds (was previously gitignored)&lt;/li&gt;
&lt;li&gt;Updated npm package README to reflect current CLI usage and features&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Security&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Updated &lt;code&gt;aws-lc-sys&lt;/code&gt; 0.37.1 to 0.39.1 (10 high-severity advisories - PKCS7 verification bypass, timing side-channel in AES-CCM, CRL/name constraint issues)&lt;/li&gt;
&lt;li&gt;Updated &lt;code&gt;rustls-webpki&lt;/code&gt; 0.103.9 to 0.103.10 (2 medium-severity CRL Distribution Point matching issues)&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.6</title>
    <link href="https://dynoxide.dev/changelog#0.9.6"/>
    <id>https://dynoxide.dev/changelog#0.9.6</id>
    <updated>2026-03-27T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Statically link the MSVC C runtime on Windows so the release binary no longer requires VCRUNTIME140.dll&lt;/li&gt;
&lt;li&gt;Switch Linux aarch64 target to musl for fully static binaries (matching x86_64)&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Drop the separate x86_64-unknown-linux-gnu release target (the musl build is already fully portable)&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.5</title>
    <link href="https://dynoxide.dev/changelog#0.9.5"/>
    <id>https://dynoxide.dev/changelog#0.9.5</id>
    <updated>2026-03-24T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;DynamoDB conformance suite&lt;/strong&gt; - 526 independently written tests across 3 tiers, validated against real DynamoDB ground truth. Dynoxide: 100%. DynamoDB Local: 92%. See &lt;a href=&quot;https://github.com/paritysuite/dynamodb-conformance&quot;&gt;dynamodb-conformance&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Dynalite external conformance&lt;/strong&gt; - 817/1039 passing (87.1% DynamoDB parity) against Dynalite&#39;s test suite, where real DynamoDB itself only passes 51%&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;DynamoDB compatibility documentation&lt;/strong&gt; - a public compatibility summary covering operation, expression, index, and PartiQL support, with a DynamoDB Local comparison column&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Correctness fixes&lt;/strong&gt; - 41 issues resolved across core operations and PartiQL&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Reserved word validation&lt;/strong&gt; - 573 DynamoDB reserved keywords rejected in ConditionExpression, UpdateExpression, FilterExpression, and ProjectionExpression with correct error messages&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;README benchmark automation&lt;/strong&gt; - CI benchmark numbers auto-updated via template markers and Python script; PR-based review with sanity checking&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;IdempotentParameterMismatchException&lt;/strong&gt; - TransactWriteItems detects same token with different payload&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;AccessDeniedException&lt;/strong&gt; - returned for tag operations on non-existent ARNs (matches DynamoDB behaviour)&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;BigDecimal replaces f64&lt;/strong&gt; for all number comparisons and arithmetic - eliminates silent precision loss beyond 15 significant digits; f64 fast-path for ≤15 significant digits preserves performance&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL INSERT&lt;/strong&gt; now fails with &lt;code&gt;DuplicateItemException&lt;/code&gt; if item already exists (previously silently overwrote)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL tokeniser&lt;/strong&gt; - correct handling of negative numbers, escaped single quotes, unknown characters (error instead of silent skip)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Query/Scan COUNT&lt;/strong&gt; now returns filtered count, not scanned count, when &lt;code&gt;FilterExpression&lt;/code&gt; is present&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;begins_with sort key&lt;/strong&gt; - SQL LIKE wildcards (&lt;code&gt;%&lt;/code&gt;, &lt;code&gt;_&lt;/code&gt;) properly escaped&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Condition + write operations&lt;/strong&gt; wrapped in SQLite transactions to prevent TOCTOU races&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;1MB response limit&lt;/strong&gt; now counts all scanned items, not just filtered results&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;GSI query/scan LastEvaluatedKey&lt;/strong&gt; now includes base table key attributes&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;BatchWriteItem&lt;/strong&gt; rejects duplicate keys within the same request&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;TransactWriteItems&lt;/strong&gt; - 4MB size check uses accurate item size calculation; CancellationReasons returned as structured top-level JSON field; &lt;code&gt;ReturnValuesOnConditionCheckFailure&lt;/code&gt; returns ALL_OLD item on condition failure&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;UpdateItem&lt;/strong&gt; rejects empty update expressions; protects key attributes from REMOVE/ADD/DELETE&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;ReturnValues&lt;/strong&gt; validated against allowed values per operation&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;UnprocessedKeys&lt;/strong&gt; in BatchGetItem preserves per-table settings&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;SET on list index beyond bounds&lt;/strong&gt; extends the list with NULL padding (previously returned error)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;SET on empty list&lt;/strong&gt; at index 0 now succeeds&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Projection with list index&lt;/strong&gt; correctly reconstructs list structure (previously created Map where List was needed)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Select validation&lt;/strong&gt; - invalid Select values and SPECIFIC_ATTRIBUTES without ProjectionExpression rejected&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;ConsistentRead on GSI&lt;/strong&gt; rejected with correct error message&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Limit of 0&lt;/strong&gt; rejected with constraint error&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Query/Scan validation ordering&lt;/strong&gt; matches DynamoDB (input validation before table existence check)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Expression attribute usage&lt;/strong&gt; validated syntactically (at parse time) not semantically (at runtime) - fixes false positives with &lt;code&gt;if_not_exists&lt;/code&gt; short-circuiting&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;SerializationException&lt;/strong&gt; pre-checks for non-list field types with DynamoDB-compatible error format&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Error type prefix&lt;/strong&gt; - &lt;code&gt;ValidationException&lt;/code&gt; uses &lt;code&gt;com.amazon.coral.validate#&lt;/code&gt; prefix matching real DynamoDB&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;BatchExecuteStatement&lt;/strong&gt; uses short error codes (&lt;code&gt;ResourceNotFound&lt;/code&gt; not fully qualified type) and rejects empty Statements array&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;UpdateTable GSI delete&lt;/strong&gt; returns &lt;code&gt;ResourceNotFoundException&lt;/code&gt; for non-existent GSI (previously &lt;code&gt;ValidationException&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;StreamSpecification&lt;/strong&gt; included in DescribeTable response&lt;/li&gt;
&lt;li&gt;Stack overflow protection: 32-level nesting depth limit on item validation (matches DynamoDB)&lt;/li&gt;
&lt;li&gt;AND/OR short-circuit evaluation in condition expressions&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;size()&lt;/code&gt; function no longer evaluates on invalid attribute types&lt;/li&gt;
&lt;li&gt;Idempotency tokens correctly compared in TransactWriteItems&lt;/li&gt;
&lt;li&gt;PutItem no longer double-reads item for conditional checks&lt;/li&gt;
&lt;li&gt;GSI sort key replacement handles all edge cases&lt;/li&gt;
&lt;li&gt;Nested projection preserves document structure (no longer flattens)&lt;/li&gt;
&lt;li&gt;Double-quote identifier escaping in PartiQL&lt;/li&gt;
&lt;li&gt;PartiQL DELETE with missing sort key returns proper error&lt;/li&gt;
&lt;li&gt;PartiQL nested SET paths create correct nested structure (no longer creates literal dot-notation keys)&lt;/li&gt;
&lt;li&gt;PartiQL SELECT with nested map paths resolves correctly&lt;/li&gt;
&lt;li&gt;TTL expiry cleans up LSI entries (previously left orphans)&lt;/li&gt;
&lt;li&gt;GSI/LSI name collision detected and rejected at CreateTable time&lt;/li&gt;
&lt;li&gt;LSI pagination uses composite cursor to handle duplicate sort key values&lt;/li&gt;
&lt;li&gt;ExecuteTransaction breaks on first failure (previously continued executing then rolled back)&lt;/li&gt;
&lt;li&gt;Partition size calculation for ItemCollectionMetrics sums across base table and all LSI tables&lt;/li&gt;
&lt;li&gt;Error message fidelity improvements across empty string, deletion protection, scan segment, and query validation messages&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.4</title>
    <link href="https://dynoxide.dev/changelog#0.9.4"/>
    <id>https://dynoxide.dev/changelog#0.9.4</id>
    <updated>2026-03-16T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;Local Secondary Indexes (LSI)&lt;/strong&gt; - full lifecycle: creation, query/scan routing, projection types (ALL, KEYS_ONLY, INCLUDE), sparse index behaviour, write path maintenance across all operations including TTL expiry&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;ExecuteTransaction&lt;/strong&gt; - PartiQL transactional execution with all-or-nothing semantics, condition checks, per-statement cancellation reasons, ConsumedCapacity support&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Parallel Scan&lt;/strong&gt; - SQLite-level segment filtering via registered FNV-1a scalar function; validated segment/total parameters&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;CreateTable extensions&lt;/strong&gt; - &lt;code&gt;SSESpecification&lt;/code&gt;, &lt;code&gt;TableClass&lt;/code&gt; (validated), &lt;code&gt;Tags&lt;/code&gt; (inline), &lt;code&gt;DeletionProtectionEnabled&lt;/code&gt; with enforcement on DeleteTable and toggle via UpdateTable&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL WHERE clause extensions&lt;/strong&gt; - &lt;code&gt;BETWEEN&lt;/code&gt;, &lt;code&gt;IN&lt;/code&gt;, &lt;code&gt;CONTAINS&lt;/code&gt;, &lt;code&gt;IS MISSING&lt;/code&gt;, &lt;code&gt;IS NOT MISSING&lt;/code&gt;, &lt;code&gt;OR&lt;/code&gt;, &lt;code&gt;NOT&lt;/code&gt;, parenthesised grouping&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL nested path projections&lt;/strong&gt; - &lt;code&gt;SELECT address.city, tags[0] FROM ...&lt;/code&gt; with correct nested structure preservation&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL REMOVE clause&lt;/strong&gt; - &lt;code&gt;UPDATE ... REMOVE attribute&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL SET expressions&lt;/strong&gt; - arithmetic (&lt;code&gt;count + 1&lt;/code&gt;), &lt;code&gt;list_append&lt;/code&gt;, &lt;code&gt;if_not_exists&lt;/code&gt; in SET clauses&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL IF NOT EXISTS&lt;/strong&gt; - &lt;code&gt;INSERT ... VALUE {...} IF NOT EXISTS&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL set literals&lt;/strong&gt; - &lt;code&gt;&amp;lt;&amp;lt; &#39;a&#39;, &#39;b&#39;, &#39;c&#39; &amp;gt;&amp;gt;&lt;/code&gt; syntax for SS/NS/BS&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;PartiQL COUNT(*)&lt;/strong&gt; and &lt;strong&gt;LIMIT&lt;/strong&gt; support&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Item validation&lt;/strong&gt; - empty string/set rejection, number precision validation (38 significant digits, ±9.99E+125 range), set deduplication (NS by numeric equivalence)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Unused expression attribute rejection&lt;/strong&gt; - unreferenced &lt;code&gt;ExpressionAttributeNames&lt;/code&gt;/&lt;code&gt;ExpressionAttributeValues&lt;/code&gt; entries return &lt;code&gt;ValidationException&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;ReturnItemCollectionMetrics&lt;/strong&gt; - partition collection size across base table and all LSI tables&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Per-GSI ConsumedCapacity&lt;/strong&gt; - &lt;code&gt;INDEXES&lt;/code&gt; mode returns per-GSI breakdown in &lt;code&gt;GlobalSecondaryIndexes&lt;/code&gt; map&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;TrackedExpressionAttributes&lt;/strong&gt; - unified expression resolution with usage tracking; removed duplicate untracked code paths (~400 LOC reduction)&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;ScanParams / QueryParams structs&lt;/strong&gt; replace parameter sprawl in storage layer&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;CreateTableMetadata&lt;/strong&gt; consolidates previously triple-duplicated row mapping&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;GSI/LSI secondary indexes&lt;/strong&gt; on &lt;code&gt;(base_pk, base_sk)&lt;/code&gt; / &lt;code&gt;(table_pk, table_sk)&lt;/code&gt; columns - eliminates full table scans during index maintenance&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Schema v5 migration&lt;/strong&gt; with automatic secondary index creation on existing tables&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.3</title>
    <link href="https://dynoxide.dev/changelog#0.9.3"/>
    <id>https://dynoxide.dev/changelog#0.9.3</id>
    <updated>2026-03-12T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;MCP Server&lt;/strong&gt; - 33 tools exposing DynamoDB operations for coding agents (Claude Code, Cursor, etc.)
&lt;ul&gt;
&lt;li&gt;stdio and Streamable HTTP transports&lt;/li&gt;
&lt;li&gt;&lt;code&gt;--read-only&lt;/code&gt;, &lt;code&gt;--max-items&lt;/code&gt;, &lt;code&gt;--max-size-bytes&lt;/code&gt; safety flags&lt;/li&gt;
&lt;li&gt;&lt;code&gt;bulk_put_items&lt;/code&gt; tool for batch loading&lt;/li&gt;
&lt;li&gt;OneTable &lt;code&gt;--data-model&lt;/code&gt; integration with entity-aware agent context and &lt;code&gt;--data-model-summary-limit&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;code&gt;--mcp&lt;/code&gt; flag on &lt;code&gt;dynoxide serve&lt;/code&gt; to run MCP alongside HTTP server&lt;/li&gt;
&lt;li&gt;Snapshots: &lt;code&gt;create_snapshot&lt;/code&gt;, &lt;code&gt;restore_snapshot&lt;/code&gt;, &lt;code&gt;list_snapshots&lt;/code&gt;, &lt;code&gt;delete_snapshot&lt;/code&gt; with auto-snapshot before &lt;code&gt;delete_table&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;code&gt;get_database_info&lt;/code&gt; tool with data model context&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Import CLI&lt;/strong&gt; - &lt;code&gt;dynoxide import&lt;/code&gt; for DynamoDB Export data (JSON Lines format)
&lt;ul&gt;
&lt;li&gt;Anonymisation rules: fake, mask, hash, redact, null actions&lt;/li&gt;
&lt;li&gt;Cross-table consistency for specified fields&lt;/li&gt;
&lt;li&gt;zstd compression (&lt;code&gt;--compress&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;&lt;code&gt;--continue-on-error&lt;/code&gt;, &lt;code&gt;--tables&lt;/code&gt; filtering, atomic &lt;code&gt;--force&lt;/code&gt; overwrite&lt;/li&gt;
&lt;li&gt;Stream-aware import (reproduces source table&#39;s StreamSpecification)&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;CLI restructuring&lt;/strong&gt; - &lt;code&gt;dynoxide serve&lt;/code&gt;, &lt;code&gt;dynoxide mcp&lt;/code&gt;, &lt;code&gt;dynoxide import&lt;/code&gt; subcommands&lt;/li&gt;
&lt;li&gt;Database introspection and port conflict detection on startup&lt;/li&gt;
&lt;li&gt;&lt;code&gt;RUST_LOG&lt;/code&gt; debug tracing throughout HTTP and MCP servers&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.2</title>
    <link href="https://dynoxide.dev/changelog#0.9.2"/>
    <id>https://dynoxide.dev/changelog#0.9.2</id>
    <updated>2026-03-04T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;SQLCipher encryption&lt;/strong&gt; - &lt;code&gt;encryption&lt;/code&gt; feature (vendored OpenSSL via SQLCipher) and &lt;code&gt;encryption-cc&lt;/code&gt; feature (Apple CommonCrypto backend) for encryption at rest&lt;/li&gt;
&lt;li&gt;Secure key handling via &lt;code&gt;--encryption-key-file&lt;/code&gt; or &lt;code&gt;DYNOXIDE_ENCRYPTION_KEY&lt;/code&gt; environment variable&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;UpdateTable&lt;/strong&gt; - &lt;code&gt;StreamSpecification&lt;/code&gt; support, GSI create/delete with backfill&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Tag operations&lt;/strong&gt; - &lt;code&gt;TagResource&lt;/code&gt;, &lt;code&gt;UntagResource&lt;/code&gt;, &lt;code&gt;ListTagsOfResource&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;ReturnValuesOnConditionCheckFailure&lt;/strong&gt; for TransactWriteItems&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;GitHub Action&lt;/strong&gt; - &lt;code&gt;nubo-db/dynoxide@v1&lt;/code&gt; with optional &lt;code&gt;snapshot-url&lt;/code&gt; preloading&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Homebrew formula&lt;/strong&gt; - &lt;code&gt;brew install nubo-db/tap/dynoxide&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;Release CI workflow with cross-platform binary builds (Linux x86_64/aarch64/musl, macOS Intel/Apple Silicon, Windows)&lt;/li&gt;
&lt;li&gt;Private-to-public repo publishing pipeline&lt;/li&gt;
&lt;li&gt;DynamoDBStreams target prefix - server accepts &lt;code&gt;DynamoDB_20120810.ListStreams&lt;/code&gt; and Streams-prefixed actions&lt;/li&gt;
&lt;li&gt;&lt;code&gt;From&lt;/code&gt;/&lt;code&gt;TryFrom&lt;/code&gt; conversions for request/response types&lt;/li&gt;
&lt;li&gt;&lt;code&gt;item!&lt;/code&gt; macro for ergonomic item construction in tests&lt;/li&gt;
&lt;li&gt;Table metadata cache for reduced SQLite round-trips&lt;/li&gt;
&lt;li&gt;Stripped release binaries&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;nubo-app&lt;/code&gt; → &lt;code&gt;nubo-db&lt;/code&gt; GitHub organisation rename&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.1</title>
    <link href="https://dynoxide.dev/changelog#0.9.1"/>
    <id>https://dynoxide.dev/changelog#0.9.1</id>
    <updated>2026-02-16T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;Server&lt;/code&gt; and &lt;code&gt;X-Dynoxide-Version&lt;/code&gt; headers on all HTTP responses&lt;/li&gt;
&lt;li&gt;&lt;code&gt;TableArn&lt;/code&gt;, &lt;code&gt;LatestStreamArn&lt;/code&gt;, and related ARN fields in API responses&lt;/li&gt;
&lt;li&gt;Comprehensive benchmarking suite comparing Dynoxide against DynamoDB Local and LocalStack
&lt;ul&gt;
&lt;li&gt;Criterion, iai-callgrind, and custom benchmark binaries&lt;/li&gt;
&lt;li&gt;CI workflows for regression detection and historical tracking&lt;/li&gt;
&lt;li&gt;Standard 13-step workload with JVM warmup protocol&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Changed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;http-server&lt;/code&gt; feature is now enabled by default&lt;/li&gt;
&lt;li&gt;Package renamed to &lt;code&gt;dynoxide-rs&lt;/code&gt; for &lt;a href=&quot;http://crates.io&quot;&gt;crates.io&lt;/a&gt; publishing&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Fixed&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Rustdoc warnings&lt;/li&gt;
&lt;li&gt;README version reference&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
  <entry>
    <title>0.9.0</title>
    <link href="https://dynoxide.dev/changelog#0.9.0"/>
    <id>https://dynoxide.dev/changelog#0.9.0</id>
    <updated>2026-02-15T00:00:00Z</updated>
    <content type="html">&lt;h3&gt;Added&lt;/h3&gt;
&lt;ul&gt;
&lt;li&gt;Core DynamoDB emulator backed by SQLite via &lt;code&gt;rusqlite&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;In-memory and persistent database modes&lt;/li&gt;
&lt;li&gt;Table operations: CreateTable, DeleteTable, DescribeTable, ListTables&lt;/li&gt;
&lt;li&gt;Item operations: PutItem, GetItem, DeleteItem, UpdateItem&lt;/li&gt;
&lt;li&gt;Query and Scan with full expression support and pagination&lt;/li&gt;
&lt;li&gt;Batch operations: BatchGetItem, BatchWriteItem&lt;/li&gt;
&lt;li&gt;Transactions: TransactWriteItems, TransactGetItems&lt;/li&gt;
&lt;li&gt;Global Secondary Indexes (GSI)&lt;/li&gt;
&lt;li&gt;DynamoDB Streams (all four view types)&lt;/li&gt;
&lt;li&gt;TTL with background sweep&lt;/li&gt;
&lt;li&gt;Full expression language: KeyCondition, Filter, Condition, Projection, Update&lt;/li&gt;
&lt;li&gt;PartiQL: ExecuteStatement, BatchExecuteStatement&lt;/li&gt;
&lt;li&gt;ReturnConsumedCapacity (TOTAL and INDEXES modes)&lt;/li&gt;
&lt;li&gt;HTTP server (axum-based, DynamoDB JSON wire protocol)&lt;/li&gt;
&lt;li&gt;300+ tests&lt;/li&gt;
&lt;/ul&gt;
</content>
  </entry>
</feed>
