Please contact the owner of the site that linked you to the original URL and let them know their link is broken.
diff --git a/assets/js/c08e7d3f.4e19ecef.js b/assets/js/c08e7d3f.4e19ecef.js
new file mode 100644
index 00000000..02cc3ed5
--- /dev/null
+++ b/assets/js/c08e7d3f.4e19ecef.js
@@ -0,0 +1 @@
+"use strict";(self.webpackChunkdocs=self.webpackChunkdocs||[]).push([[8121],{4372:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>c,contentTitle:()=>r,default:()=>l,frontMatter:()=>a,metadata:()=>s,toc:()=>d});var o=n(4848),i=n(8453);const a={sidebar_position:1},r="Modeling Data",s={id:"cookbook/modeling-data",title:"Modeling Data",description:"All data in Automerge must be stored in a document. A document can be modeled in a variety of ways, and there are many design patterns that can be used. An application could have many documents, typically identified by a UUID.",source:"@site/docs/cookbook/modeling-data.md",sourceDirName:"cookbook",slug:"/cookbook/modeling-data",permalink:"/docs/cookbook/modeling-data",draft:!1,unlisted:!1,editUrl:"https://github.com/automerge/automerge.github.io/edit/main/docs/cookbook/modeling-data.md",tags:[],version:"current",sidebarPosition:1,frontMatter:{sidebar_position:1},sidebar:"tutorialSidebar",previous:{title:"Ephemeral Data",permalink:"/docs/repositories/ephemeral"},next:{title:"Prosemirror + React + Automerge",permalink:"/docs/cookbook/rich-text-prosemirror-react"}},c={},d=[{value:"How many documents?",id:"how-many-documents",level:2},{value:"Setting up an initial document structure",id:"setting-up-an-initial-document-structure",level:2},{value:"Versioning",id:"versioning",level:2},{value:"Performance",id:"performance",level:2}];function h(e){const t={a:"a",code:"code",em:"em",h1:"h1",h2:"h2",p:"p",pre:"pre",strong:"strong",...(0,i.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(t.h1,{id:"modeling-data",children:"Modeling Data"}),"\n",(0,o.jsx)(t.p,{children:"All data in Automerge must be stored in a document. A document can be modeled in a variety of ways, and there are many design patterns that can be used. An application could have many documents, typically identified by a UUID."}),"\n",(0,o.jsx)(t.p,{children:"In this section, we will discuss how to model data within a particular document, including how to version and manage data with Automerge in production scenarios."}),"\n",(0,o.jsx)(t.h2,{id:"how-many-documents",children:"How many documents?"}),"\n",(0,o.jsxs)(t.p,{children:["You can decide which things to group together as one Automerge document (more fine grained or more coarse grained) based on what makes sense in your app. Having hundreds of docs should be fine \u2014 we've built prototypes of that scale. One major automerge project, ",(0,o.jsx)(t.a,{href:"https://github.com/automerge/pushpin",children:"PushPin"}),", was built around very granular documents. This had a lot of benefits, but the overhead of syncing many thousands of documents was high."]}),"\n",(0,o.jsx)(t.p,{children:"We believe on the whole there's an art to the granularity of data that is universal. When should you have two JSON documents or two SQLite databases or two rows? We suspect that an Automerge document is best suited to being a unit of collaboration between two people or a small group."}),"\n",(0,o.jsx)(t.h2,{id:"setting-up-an-initial-document-structure",children:"Setting up an initial document structure"}),"\n",(0,o.jsxs)(t.p,{children:["When you create a document using ",(0,o.jsx)(t.code,{children:"Automerge.init()"}),", it's just an empty JSON document with no properties. As the first change, most applications will need to initialize some empty collection objects that are expected to be present within the document."]}),"\n",(0,o.jsxs)(t.p,{children:["The easiest way of doing this is with a call to ",(0,o.jsx)(t.code,{children:"Automerge.change()"})," that sets up the document schema in the form that you need it, like in the code sample above. You can then sync this initial change to all of your devices; once everybody has the schema, you can have different users updating the document on different devices, and the updates should merge nicely. For example:"]}),"\n",(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-js",children:'// Set up the `cards` array in doc1\nlet doc1 = Automerge.change(Automerge.init(), (doc) => {\n doc.cards = [];\n});\n\n// In doc2, don\'t create `cards` again! Instead, merge\n// the schema initialization from doc1\nlet doc2 = Automerge.merge(Automerge.init(), doc1);\n\n// Now we can update both documents\ndoc1 = Automerge.change(doc1, (doc) => {\n doc.cards.push({ title: "card1" });\n});\n\ndoc2 = Automerge.change(doc2, (doc) => {\n doc.cards.push({ title: "card2" });\n});\n\n// The merged document will contain both cards\ndoc1 = Automerge.merge(doc1, doc2);\ndoc2 = Automerge.merge(doc2, doc1);\n'})}),"\n",(0,o.jsxs)(t.p,{children:["However, sometimes it's inconvenient to have to sync the initial change to a device before you can modify the document on that device. If you want two devices to be able to independently set up their own document schema, but still to be able to merge those documents, you have to be careful. Simply doing ",(0,o.jsx)(t.code,{children:"Automerge.change()"})," on each device to initialize the schema ",(0,o.jsx)(t.strong,{children:"will not work"}),", because you now have two different documents with no shared ancestry (even if the initial change performs the same operations, each device has a different actorId and so the changes will be different)."]}),"\n",(0,o.jsxs)(t.p,{children:["If you really must initialize each device's copy of a document independently, one option is to do the initial ",(0,o.jsx)(t.code,{children:"Automerge.change()"})," once to set up your schema, then call ",(0,o.jsx)(t.code,{children:"Automerge.save()"})," on the document (which returns a byte array), and ",(0,o.jsx)(t.em,{children:"hard-code that byte array into your application"}),". Now, on each device that needs to initialize a document, you do this:"]}),"\n",(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-js",children:"// hard-code the initial change here\nconst initChange = new Uint8Array([133, 111, 74, 131, ...])\nlet [doc] = Automerge.load(initChange)\n"})}),"\n",(0,o.jsx)(t.p,{children:"This will set you up with a document whose initial change is the one you hard-coded. Any documents you set up with the same initial change will be able to merge."}),"\n",(0,o.jsx)(t.h2,{id:"versioning",children:"Versioning"}),"\n",(0,o.jsx)(t.p,{children:"Often, there comes a time in the production lifecycle where you will need to change the schema of a document. Because Automerge uses a JSON document model, it's similar to a NoSQL database, where properties can be arbitrarily removed and added at will."}),"\n",(0,o.jsx)(t.p,{children:"You can implement your own versioning scheme, for example by embedding a schema version number into the document, and writing a function that can upgrade a document from one schema version to the next. However, doing this in a CRDT like Automerge is more difficult than migrations in a centralized relational database, because it could happen that two users independently perform the same migration. In this case, you need to ensure that the two migrations don't clash with each other, which is difficult."}),"\n",(0,o.jsx)(t.p,{children:"One way of making migrations safe is by using the tricks from the previous section: in addition to hard-coding the initial change that sets up the document, you can also hard-code migrations that upgrade from one schema version to the next, using the same technique (either hard-coding the change as a byte array, or making a change on the fly with hard-coded actorId and timestamp). Do not modify the initial change; instead, every migration should be a separate hard-coded change that depends only on the preceding change. This way, you can have multiple devices independently applying the same migration, and they will all be compatible because the migration is performed identically on every device."}),"\n",(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-js",children:"type DocV1 = {\n version: 1,\n cards: Card[]\n}\n\ntype DocV2 = {\n version: 2,\n title: Automerge.Text,\n cards: Card[]\n}\n\n// This change creates the `title` property required in V2,\n// and updates the `version` property from 1 to 2\nconst migrateV1toV2 = new Uint8Array([133, 111, 74, 131, ...])\n\nlet doc = getDocumentFromNetwork()\nif (doc.version === 1) {\n [doc] = Automerge.applyChange(doc, [migrateV1toV2])\n}\n"})}),"\n",(0,o.jsx)(t.p,{children:"Also keep in mind that in your app there might be some users using an old version of the app while other users are using a newer version; you will need to take care with migrations to ensure that they do not break compatibility with older app versions, or force all users to update to the latest version."}),"\n",(0,o.jsxs)(t.p,{children:["Some further ideas on safe schema migrations in CRDT apps are discussed in the ",(0,o.jsx)(t.a,{href:"https://www.inkandswitch.com/cambria",children:"Cambria"})," paper, but these are not yet implemented in Automerge. If you want to work on improving schema migrations in Automerge, please get in touch \u2014\xa0contributions are welcome!"]}),"\n",(0,o.jsx)(t.h2,{id:"performance",children:"Performance"}),"\n",(0,o.jsx)(t.p,{children:"Automerge documents hold their entire change histories. It is fairly performant, and can handle a significant amount of data in a single document's history. Performance depends very much on your workload, so we strongly suggest you do your own measurements with the type and quantity of data that you will have in your app."}),"\n",(0,o.jsxs)(t.p,{children:["Some developers have proposed \u201cgarbage collecting\u201d large documents. If a document gets to a certain size, a central authority could emit a message to each peer that it would like to reduce it in size and only save the history from a specific change (hash). Martin Kleppmann did some experiments with a benchmark document to see how much space would be saved by discarding history, with and without preserving tombstones. See ",(0,o.jsx)(t.a,{href:"https://youtu.be/x7drE24geUw?t=3289",children:"this video at 55 minutes in"}),". The savings are not all that great, which is why we haven't prioritised history truncation so far."]}),"\n",(0,o.jsx)(t.p,{children:"Typically, performance improvements can come at the networking level. You can set up a single connection (between peers or client-server) and sync many docs over a single connection. The basic idea is to tag each message with the ID of the document it belongs to. There are possible ways of optimising this if necessary. In general, having fewer documents that a client must load over the network or into memory at any given time will reduce the synchronization and startup time for your application."})]})}function l(e={}){const{wrapper:t}={...(0,i.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(h,{...e})}):h(e)}},8453:(e,t,n)=>{n.d(t,{R:()=>r,x:()=>s});var o=n(6540);const i={},a=o.createContext(i);function r(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function s(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:r(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]);
\ No newline at end of file
diff --git a/assets/js/c08e7d3f.b92390d9.js b/assets/js/c08e7d3f.b92390d9.js
deleted file mode 100644
index 55d43bf4..00000000
--- a/assets/js/c08e7d3f.b92390d9.js
+++ /dev/null
@@ -1 +0,0 @@
-"use strict";(self.webpackChunkdocs=self.webpackChunkdocs||[]).push([[8121],{4372:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>c,contentTitle:()=>r,default:()=>l,frontMatter:()=>a,metadata:()=>s,toc:()=>d});var o=n(4848),i=n(8453);const a={sidebar_position:1},r="Modeling Data",s={id:"cookbook/modeling-data",title:"Modeling Data",description:"All data in Automerge must be stored in a document. A document can be modeled in a variety of ways, and there are many design patterns that can be used. An application could have many documents, typically identified by a UUID.",source:"@site/docs/cookbook/modeling-data.md",sourceDirName:"cookbook",slug:"/cookbook/modeling-data",permalink:"/docs/cookbook/modeling-data",draft:!1,unlisted:!1,editUrl:"https://github.com/automerge/automerge.github.io/edit/main/docs/cookbook/modeling-data.md",tags:[],version:"current",sidebarPosition:1,frontMatter:{sidebar_position:1},sidebar:"tutorialSidebar",previous:{title:"Ephemeral Data",permalink:"/docs/repositories/ephemeral"},next:{title:"Prosemirror + React + Automerge",permalink:"/docs/cookbook/rich-text-prosemirror-react"}},c={},d=[{value:"How many documents?",id:"how-many-documents",level:2},{value:"Setting up an initial document structure",id:"setting-up-an-initial-document-structure",level:2},{value:"Versioning",id:"versioning",level:2},{value:"Performance",id:"performance",level:2}];function h(e){const t={a:"a",code:"code",em:"em",h1:"h1",h2:"h2",p:"p",pre:"pre",strong:"strong",...(0,i.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(t.h1,{id:"modeling-data",children:"Modeling Data"}),"\n",(0,o.jsx)(t.p,{children:"All data in Automerge must be stored in a document. A document can be modeled in a variety of ways, and there are many design patterns that can be used. An application could have many documents, typically identified by a UUID."}),"\n",(0,o.jsx)(t.p,{children:"In this section, we will discuss how to model data within a particular document, including how to version and manage data with Automerge in production scenarios."}),"\n",(0,o.jsx)(t.h2,{id:"how-many-documents",children:"How many documents?"}),"\n",(0,o.jsxs)(t.p,{children:["You can decide which things to group together as one Automerge document (more fine grained or more coarse grained) based on what makes sense in your app. Having hundreds of docs should be fine \u2014 we've built prototypes of that scale. One major automerge project, ",(0,o.jsx)(t.a,{href:"https://github.com/automerge/pushpin",children:"PushPin"}),", was built around very granular documents. This had a lot of benefits, but the overhead of syncing many thousands of documents was high."]}),"\n",(0,o.jsx)(t.p,{children:"We believe on the whole there's an art to the granularity of data that is universal. When should you have two JSON documents or two SQLite databases or two rows? We suspect that an Automerge document is best suited to being a unit of collaboration between two people or a small group."}),"\n",(0,o.jsx)(t.h2,{id:"setting-up-an-initial-document-structure",children:"Setting up an initial document structure"}),"\n",(0,o.jsxs)(t.p,{children:["When you create a document using ",(0,o.jsx)(t.code,{children:"Automerge.init()"}),", it's just an empty JSON document with no properties. As the first change, most applications will need to initialize some empty collection objects that are expected to be present within the document."]}),"\n",(0,o.jsxs)(t.p,{children:["The easiest way of doing this is with a call to ",(0,o.jsx)(t.code,{children:"Automerge.change()"})," that sets up the document schema in the form that you need it, like in the code sample above. You can then ",(0,o.jsx)(t.a,{href:"/docs/cookbook/real-time/",children:"sync"})," this initial change to all of your devices; once everybody has the schema, you can have different users updating the document on different devices, and the updates should merge nicely. For example:"]}),"\n",(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-js",children:'// Set up the `cards` array in doc1\nlet doc1 = Automerge.change(Automerge.init(), (doc) => {\n doc.cards = [];\n});\n\n// In doc2, don\'t create `cards` again! Instead, merge\n// the schema initialization from doc1\nlet doc2 = Automerge.merge(Automerge.init(), doc1);\n\n// Now we can update both documents\ndoc1 = Automerge.change(doc1, (doc) => {\n doc.cards.push({ title: "card1" });\n});\n\ndoc2 = Automerge.change(doc2, (doc) => {\n doc.cards.push({ title: "card2" });\n});\n\n// The merged document will contain both cards\ndoc1 = Automerge.merge(doc1, doc2);\ndoc2 = Automerge.merge(doc2, doc1);\n'})}),"\n",(0,o.jsxs)(t.p,{children:["However, sometimes it's inconvenient to have to sync the initial change to a device before you can modify the document on that device. If you want two devices to be able to independently set up their own document schema, but still to be able to merge those documents, you have to be careful. Simply doing ",(0,o.jsx)(t.code,{children:"Automerge.change()"})," on each device to initialize the schema ",(0,o.jsx)(t.strong,{children:"will not work"}),", because you now have two different documents with no shared ancestry (even if the initial change performs the same operations, each device has a different actorId and so the changes will be different)."]}),"\n",(0,o.jsxs)(t.p,{children:["If you really must initialize each device's copy of a document independently, one option is to do the initial ",(0,o.jsx)(t.code,{children:"Automerge.change()"})," once to set up your schema, then call ",(0,o.jsx)(t.code,{children:"Automerge.save()"})," on the document (which returns a byte array), and ",(0,o.jsx)(t.em,{children:"hard-code that byte array into your application"}),". Now, on each device that needs to initialize a document, you do this:"]}),"\n",(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-js",children:"// hard-code the initial change here\nconst initChange = new Uint8Array([133, 111, 74, 131, ...])\nlet [doc] = Automerge.load(initChange)\n"})}),"\n",(0,o.jsx)(t.p,{children:"This will set you up with a document whose initial change is the one you hard-coded. Any documents you set up with the same initial change will be able to merge."}),"\n",(0,o.jsx)(t.h2,{id:"versioning",children:"Versioning"}),"\n",(0,o.jsx)(t.p,{children:"Often, there comes a time in the production lifecycle where you will need to change the schema of a document. Because Automerge uses a JSON document model, it's similar to a NoSQL database, where properties can be arbitrarily removed and added at will."}),"\n",(0,o.jsx)(t.p,{children:"You can implement your own versioning scheme, for example by embedding a schema version number into the document, and writing a function that can upgrade a document from one schema version to the next. However, doing this in a CRDT like Automerge is more difficult than migrations in a centralized relational database, because it could happen that two users independently perform the same migration. In this case, you need to ensure that the two migrations don't clash with each other, which is difficult."}),"\n",(0,o.jsx)(t.p,{children:"One way of making migrations safe is by using the tricks from the previous section: in addition to hard-coding the initial change that sets up the document, you can also hard-code migrations that upgrade from one schema version to the next, using the same technique (either hard-coding the change as a byte array, or making a change on the fly with hard-coded actorId and timestamp). Do not modify the initial change; instead, every migration should be a separate hard-coded change that depends only on the preceding change. This way, you can have multiple devices independently applying the same migration, and they will all be compatible because the migration is performed identically on every device."}),"\n",(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-js",children:"type DocV1 = {\n version: 1,\n cards: Card[]\n}\n\ntype DocV2 = {\n version: 2,\n title: Automerge.Text,\n cards: Card[]\n}\n\n// This change creates the `title` property required in V2,\n// and updates the `version` property from 1 to 2\nconst migrateV1toV2 = new Uint8Array([133, 111, 74, 131, ...])\n\nlet doc = getDocumentFromNetwork()\nif (doc.version === 1) {\n [doc] = Automerge.applyChange(doc, [migrateV1toV2])\n}\n"})}),"\n",(0,o.jsx)(t.p,{children:"Also keep in mind that in your app there might be some users using an old version of the app while other users are using a newer version; you will need to take care with migrations to ensure that they do not break compatibility with older app versions, or force all users to update to the latest version."}),"\n",(0,o.jsxs)(t.p,{children:["Some further ideas on safe schema migrations in CRDT apps are discussed in the ",(0,o.jsx)(t.a,{href:"https://www.inkandswitch.com/cambria",children:"Cambria"})," paper, but these are not yet implemented in Automerge. If you want to work on improving schema migrations in Automerge, please get in touch \u2014\xa0contributions are welcome!"]}),"\n",(0,o.jsx)(t.h2,{id:"performance",children:"Performance"}),"\n",(0,o.jsx)(t.p,{children:"Automerge documents hold their entire change histories. It is fairly performant, and can handle a significant amount of data in a single document's history. Performance depends very much on your workload, so we strongly suggest you do your own measurements with the type and quantity of data that you will have in your app."}),"\n",(0,o.jsxs)(t.p,{children:["Some developers have proposed \u201cgarbage collecting\u201d large documents. If a document gets to a certain size, a central authority could emit a message to each peer that it would like to reduce it in size and only save the history from a specific change (hash). Martin Kleppmann did some experiments with a benchmark document to see how much space would be saved by discarding history, with and without preserving tombstones. See ",(0,o.jsx)(t.a,{href:"https://youtu.be/x7drE24geUw?t=3289",children:"this video at 55 minutes in"}),". The savings are not all that great, which is why we haven't prioritised history truncation so far."]}),"\n",(0,o.jsx)(t.p,{children:"Typically, performance improvements can come at the networking level. You can set up a single connection (between peers or client-server) and sync many docs over a single connection. The basic idea is to tag each message with the ID of the document it belongs to. There are possible ways of optimising this if necessary. In general, having fewer documents that a client must load over the network or into memory at any given time will reduce the synchronization and startup time for your application."})]})}function l(e={}){const{wrapper:t}={...(0,i.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(h,{...e})}):h(e)}},8453:(e,t,n)=>{n.d(t,{R:()=>r,x:()=>s});var o=n(6540);const i={},a=o.createContext(i);function r(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function s(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:r(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]);
\ No newline at end of file
diff --git a/assets/js/main.ecfd1664.js b/assets/js/main.b04d9612.js
similarity index 99%
rename from assets/js/main.ecfd1664.js
rename to assets/js/main.b04d9612.js
index 2f83b475..37801a9f 100644
--- a/assets/js/main.ecfd1664.js
+++ b/assets/js/main.b04d9612.js
@@ -1,2 +1,2 @@
-/*! For license information please see main.ecfd1664.js.LICENSE.txt */
-(self.webpackChunkdocs=self.webpackChunkdocs||[]).push([[8792],{8328:(e,t,n)=>{"use strict";n.d(t,{A:()=>p});n(6540);var r=n(3259),o=n.n(r),a=n(4054);const i={"01a85c17":[()=>Promise.all([n.e(1869),n.e(8209)]).then(n.bind(n,9158)),"@theme/BlogTagsListPage",9158],"0ab6c356":[()=>n.e(8409).then(n.bind(n,5204)),"@site/docs/documents/text.md",5204],"164f3716":[()=>n.e(5975).then(n.bind(n,7973)),"@site/docs/repositories/networking.md",7973],"171c4778":[()=>n.e(453).then(n.bind(n,9686)),"@site/blog/2023-01-17-automerge-2/index.md?truncated=true",9686],17896441:[()=>Promise.all([n.e(1869),n.e(1544),n.e(8401)]).then(n.bind(n,2447)),"@theme/DocItem",2447],"1f391b9e":[()=>Promise.all([n.e(1869),n.e(1544),n.e(6061)]).then(n.bind(n,354)),"@theme/MDXPage",354],"1f43bfb7":[()=>n.e(3788).then(n.bind(n,3435)),"@site/docs/documents/rich_text.md",3435],"1fd5fe2b":[()=>n.e(6644).then(n.bind(n,1408)),"@site/blog/2023-11-06-automerge-repo.md?truncated=true",1408],"2c78790a":[()=>n.e(1622).then(n.bind(n,1882)),"@site/docs/under-the-hood/merge_rules.md",1882],"30a24c52":[()=>n.e(3669).then(n.t.bind(n,6833,19)),"~blog/default/blog-tags-hello-039.json",6833],"393be207":[()=>n.e(4134).then(n.bind(n,6602)),"@site/src/pages/markdown-page.md",6602],"3faabd13":[()=>n.e(4596).then(n.bind(n,9232)),"@site/docs/repositories/ephemeral.md",9232],"4115a37b":[()=>n.e(5023).then(n.bind(n,4940)),"@site/blog/2024-08-23-wasm-packaging.md",4940],"45a5cd1f":[()=>n.e(4e3).then(n.bind(n,3357)),"@site/docs/concepts.md",3357],"4b1ab083":[()=>n.e(5446).then(n.t.bind(n,1966,19)),"/home/runner/work/automerge.github.io/automerge.github.io/.docusaurus/docusaurus-plugin-content-docs/default/plugin-route-context-module-100.json",1966],"4c11766a":[()=>n.e(3850).then(n.bind(n,2199)),"@site/docs/cookbook/rich-text-prosemirror-vanilla.md",2199],"5e95c892":[()=>n.e(9647).then(n.bind(n,7121)),"@theme/DocsRoot",7121],"5e9f5e1a":[()=>Promise.resolve().then(n.bind(n,4784)),"@generated/docusaurus.config",4784],"5fbc5cf1":[()=>n.e(7624).then(n.bind(n,8946)),"@site/docs/api.md",8946],"6152c90d":[()=>n.e(6905).then(n.bind(n,8548)),"@site/docs/documents/conflicts.md",8548],66406991:[()=>n.e(4374).then(n.t.bind(n,576,19)),"~blog/default/blog-tags-hello-039-list.json",576],"6875c492":[()=>Promise.all([n.e(1869),n.e(1544),n.e(3242),n.e(4813)]).then(n.bind(n,3069)),"@theme/BlogTagsPostsPage",3069],"69a12fef":[()=>n.e(9599).then(n.bind(n,9723)),"@site/docs/documents/index.md",9723],74876495:[()=>Promise.all([n.e(1869),n.e(7336),n.e(6505)]).then(n.bind(n,2622)),"@site/docs/quickstart.mdx",2622],"77c3add4":[()=>n.e(9172).then(n.bind(n,7850)),"@site/docs/repositories/storage.md",7850],"7bded21b":[()=>n.e(2930).then(n.bind(n,8312)),"@site/docs/library_initialization.md",8312],"7cafe14d":[()=>n.e(8434).then(n.bind(n,9400)),"@site/blog/2024-08-23-wasm-packaging.md?truncated=true",9400],"7cdda22f":[()=>n.e(2392).then(n.bind(n,5871)),"@site/docs/under-the-hood/storage.md",5871],"814f3328":[()=>n.e(7472).then(n.t.bind(n,5513,19)),"~blog/default/blog-post-list-prop-default.json",5513],"89d3c875":[()=>n.e(5216).then(n.bind(n,6062)),"@site/docs/under-the-hood/rich_text_schema.md",6062],"8e10ab80":[()=>n.e(1424).then(n.bind(n,973)),"@site/blog/2022-01-05-welcome.md/index.md?truncated=true",973],"935f2afb":[()=>n.e(8581).then(n.t.bind(n,5610,19)),"~docs/default/version-current-metadata-prop-751.json",5610],"9dfd250b":[()=>n.e(2887).then(n.bind(n,5310)),"@site/docs/hello.md",5310],"9e4087bc":[()=>n.e(2711).then(n.bind(n,9331)),"@theme/BlogArchivePage",9331],a2d2eeed:[()=>n.e(757).then(n.bind(n,7499)),"@site/blog/2024-04-06-richtext/index.mdx?truncated=true",7499],a6aa9e1f:[()=>Promise.all([n.e(1869),n.e(1544),n.e(3242),n.e(7643)]).then(n.bind(n,5124)),"@theme/BlogListPage",5124],a7023ddc:[()=>n.e(9267).then(n.t.bind(n,8289,19)),"~blog/default/blog-tags-tags-4c2.json",8289],a7bd4aaa:[()=>n.e(7098).then(n.bind(n,4532)),"@theme/DocVersionRoot",4532],a94703ab:[()=>Promise.all([n.e(1869),n.e(9048)]).then(n.bind(n,2559)),"@theme/DocRoot",2559],a96de142:[()=>n.e(8167).then(n.bind(n,3650)),"@site/blog/2023-12-21-merry-commitmas.md",3650],ab57a952:[()=>n.e(4049).then(n.bind(n,5743)),"@site/docs/the_js_packages.md",5743],acecf23e:[()=>n.e(1903).then(n.t.bind(n,1912,19)),"~blog/default/blogMetadata-default.json",1912],b2b675dd:[()=>n.e(1991).then(n.t.bind(n,9775,19)),"~blog/default/blog-c06.json",9775],b2f554cd:[()=>n.e(5894).then(n.t.bind(n,6042,19)),"~blog/default/blog-archive-80c.json",6042],b77ff49f:[()=>n.e(3402).then(n.bind(n,2847)),"@site/docs/repositories/index.md",2847],b826a138:[()=>n.e(3170).then(n.bind(n,2482)),"@site/blog/2023-01-17-automerge-2/index.md",2482],bb19a7e4:[()=>n.e(610).then(n.t.bind(n,2945,19)),"/home/runner/work/automerge.github.io/automerge.github.io/.docusaurus/docusaurus-plugin-content-blog/default/plugin-route-context-module-100.json",2945],c08e7d3f:[()=>n.e(8121).then(n.bind(n,4372)),"@site/docs/cookbook/modeling-data.md",4372],c4f5d8e4:[()=>Promise.all([n.e(1869),n.e(2634)]).then(n.bind(n,8655)),"@site/src/pages/index.js",8655],ccc49370:[()=>Promise.all([n.e(1869),n.e(1544),n.e(3242),n.e(3249)]).then(n.bind(n,1477)),"@theme/BlogPostPage",1477],d184be29:[()=>n.e(6).then(n.bind(n,659)),"@site/docs/cookbook/rich-text-prosemirror-react.md",659],d62774ea:[()=>n.e(7931).then(n.bind(n,2743)),"@site/blog/2023-11-06-automerge-repo.md",2743],db84ab83:[()=>n.e(1323).then(n.bind(n,974)),"@site/docs/documents/lists.md",974],db8a9483:[()=>n.e(1548).then(n.bind(n,8741)),"@site/docs/documents/counters.md",8741],e0030a30:[()=>n.e(3471).then(n.bind(n,8166)),"@site/docs/repositories/dochandles.md",8166],e5e39ef1:[()=>n.e(1296).then(n.bind(n,4603)),"@site/blog/2022-01-05-welcome.md/index.md",4603],e747ec83:[()=>n.e(7051).then(n.bind(n,3340)),"@site/docs/glossary.md",3340],edeae7d6:[()=>n.e(5412).then(n.t.bind(n,4061,19)),"/home/runner/work/automerge.github.io/automerge.github.io/.docusaurus/docusaurus-plugin-content-pages/default/plugin-route-context-module-100.json",4061],ee8e1f9e:[()=>n.e(1631).then(n.bind(n,9974)),"@site/blog/2023-12-21-merry-commitmas.md?truncated=true",9974],f51dd59e:[()=>n.e(752).then(n.bind(n,8665)),"@site/blog/2024-04-06-richtext/index.mdx",8665],f80f5a42:[()=>n.e(5463).then(n.bind(n,2649)),"@site/docs/documents/values.md",2649]};var l=n(4848);function s(e){let{error:t,retry:n,pastDelay:r}=e;return t?(0,l.jsxs)("div",{style:{textAlign:"center",color:"#fff",backgroundColor:"#fa383e",borderColor:"#fa383e",borderStyle:"solid",borderRadius:"0.25rem",borderWidth:"1px",boxSizing:"border-box",display:"block",padding:"1rem",flex:"0 0 50%",marginLeft:"25%",marginRight:"25%",marginTop:"5rem",maxWidth:"50%",width:"100%"},children:[(0,l.jsx)("p",{children:String(t)}),(0,l.jsx)("div",{children:(0,l.jsx)("button",{type:"button",onClick:n,children:"Retry"})})]}):r?(0,l.jsx)("div",{style:{display:"flex",justifyContent:"center",alignItems:"center",height:"100vh"},children:(0,l.jsx)("svg",{id:"loader",style:{width:128,height:110,position:"absolute",top:"calc(100vh - 64%)"},viewBox:"0 0 45 45",xmlns:"http://www.w3.org/2000/svg",stroke:"#61dafb",children:(0,l.jsxs)("g",{fill:"none",fillRule:"evenodd",transform:"translate(1 1)",strokeWidth:"2",children:[(0,l.jsxs)("circle",{cx:"22",cy:"22",r:"6",strokeOpacity:"0",children:[(0,l.jsx)("animate",{attributeName:"r",begin:"1.5s",dur:"3s",values:"6;22",calcMode:"linear",repeatCount:"indefinite"}),(0,l.jsx)("animate",{attributeName:"stroke-opacity",begin:"1.5s",dur:"3s",values:"1;0",calcMode:"linear",repeatCount:"indefinite"}),(0,l.jsx)("animate",{attributeName:"stroke-width",begin:"1.5s",dur:"3s",values:"2;0",calcMode:"linear",repeatCount:"indefinite"})]}),(0,l.jsxs)("circle",{cx:"22",cy:"22",r:"6",strokeOpacity:"0",children:[(0,l.jsx)("animate",{attributeName:"r",begin:"3s",dur:"3s",values:"6;22",calcMode:"linear",repeatCount:"indefinite"}),(0,l.jsx)("animate",{attributeName:"stroke-opacity",begin:"3s",dur:"3s",values:"1;0",calcMode:"linear",repeatCount:"indefinite"}),(0,l.jsx)("animate",{attributeName:"stroke-width",begin:"3s",dur:"3s",values:"2;0",calcMode:"linear",repeatCount:"indefinite"})]}),(0,l.jsx)("circle",{cx:"22",cy:"22",r:"8",children:(0,l.jsx)("animate",{attributeName:"r",begin:"0s",dur:"1.5s",values:"6;1;2;3;4;5;6",calcMode:"linear",repeatCount:"indefinite"})})]})})}):null}var u=n(6921),c=n(3102);function d(e,t){if("*"===e)return o()({loading:s,loader:()=>n.e(2237).then(n.bind(n,2237)),modules:["@theme/NotFound"],webpack:()=>[2237],render(e,t){const n=e.default;return(0,l.jsx)(c.W,{value:{plugin:{name:"native",id:"default"}},children:(0,l.jsx)(n,{...t})})}});const r=a[`${e}-${t}`],d={},p=[],f=[],m=(0,u.A)(r);return Object.entries(m).forEach((e=>{let[t,n]=e;const r=i[n];r&&(d[t]=r[0],p.push(r[1]),f.push(r[2]))})),o().Map({loading:s,loader:d,modules:p,webpack:()=>f,render(t,n){const o=JSON.parse(JSON.stringify(r));Object.entries(t).forEach((t=>{let[n,r]=t;const a=r.default;if(!a)throw new Error(`The page component at ${e} doesn't have a default export. This makes it impossible to render anything. Consider default-exporting a React component.`);"object"!=typeof a&&"function"!=typeof a||Object.keys(r).filter((e=>"default"!==e)).forEach((e=>{a[e]=r[e]}));let i=o;const l=n.split(".");l.slice(0,-1).forEach((e=>{i=i[e]})),i[l[l.length-1]]=a}));const a=o.__comp;delete o.__comp;const i=o.__context;return delete o.__context,(0,l.jsx)(c.W,{value:i,children:(0,l.jsx)(a,{...o,...n})})}})}const p=[{path:"/blog/",component:d("/blog/","26b"),exact:!0},{path:"/blog/2023/11/06/automerge-repo/",component:d("/blog/2023/11/06/automerge-repo/","098"),exact:!0},{path:"/blog/2023/12/21/merry-commitmas/",component:d("/blog/2023/12/21/merry-commitmas/","e9e"),exact:!0},{path:"/blog/2024/04/06/richtext/",component:d("/blog/2024/04/06/richtext/","651"),exact:!0},{path:"/blog/2024/08/23/wasm-packaging/",component:d("/blog/2024/08/23/wasm-packaging/","501"),exact:!0},{path:"/blog/archive/",component:d("/blog/archive/","c57"),exact:!0},{path:"/blog/automerge-2/",component:d("/blog/automerge-2/","ec7"),exact:!0},{path:"/blog/tags/",component:d("/blog/tags/","16b"),exact:!0},{path:"/blog/tags/hello/",component:d("/blog/tags/hello/","33e"),exact:!0},{path:"/blog/welcome/",component:d("/blog/welcome/","713"),exact:!0},{path:"/markdown-page/",component:d("/markdown-page/","54c"),exact:!0},{path:"/docs/",component:d("/docs/","c05"),routes:[{path:"/docs/",component:d("/docs/","e5f"),routes:[{path:"/docs/",component:d("/docs/","92c"),routes:[{path:"/docs/api/",component:d("/docs/api/","fd4"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/concepts/",component:d("/docs/concepts/","5a9"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/cookbook/modeling-data/",component:d("/docs/cookbook/modeling-data/","fa5"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/cookbook/rich-text-prosemirror-react/",component:d("/docs/cookbook/rich-text-prosemirror-react/","316"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/cookbook/rich-text-prosemirror-vanilla/",component:d("/docs/cookbook/rich-text-prosemirror-vanilla/","48e"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/",component:d("/docs/documents/","d9b"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/conflicts/",component:d("/docs/documents/conflicts/","0b6"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/counters/",component:d("/docs/documents/counters/","6e9"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/lists/",component:d("/docs/documents/lists/","db6"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/rich_text/",component:d("/docs/documents/rich_text/","7f3"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/text/",component:d("/docs/documents/text/","5a0"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/documents/values/",component:d("/docs/documents/values/","d4d"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/glossary/",component:d("/docs/glossary/","49b"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/hello/",component:d("/docs/hello/","2d3"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/library_initialization/",component:d("/docs/library_initialization/","bcc"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/quickstart/",component:d("/docs/quickstart/","6cf"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/repositories/",component:d("/docs/repositories/","4d3"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/repositories/dochandles/",component:d("/docs/repositories/dochandles/","fac"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/repositories/ephemeral/",component:d("/docs/repositories/ephemeral/","36b"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/repositories/networking/",component:d("/docs/repositories/networking/","4f1"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/repositories/storage/",component:d("/docs/repositories/storage/","794"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/the_js_packages/",component:d("/docs/the_js_packages/","b04"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/under-the-hood/merge_rules/",component:d("/docs/under-the-hood/merge_rules/","48d"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/under-the-hood/rich_text_schema/",component:d("/docs/under-the-hood/rich_text_schema/","245"),exact:!0,sidebar:"tutorialSidebar"},{path:"/docs/under-the-hood/storage/",component:d("/docs/under-the-hood/storage/","bc7"),exact:!0,sidebar:"tutorialSidebar"}]}]}]},{path:"/",component:d("/","03a"),exact:!0},{path:"*",component:d("*")}]},6125:(e,t,n)=>{"use strict";n.d(t,{o:()=>a,x:()=>i});var r=n(6540),o=n(4848);const a=r.createContext(!1);function i(e){let{children:t}=e;const[n,i]=(0,r.useState)(!1);return(0,r.useEffect)((()=>{i(!0)}),[]),(0,o.jsx)(a.Provider,{value:n,children:t})}},8536:(e,t,n)=>{"use strict";var r=n(6540),o=n(5338),a=n(4625),i=n(545),l=n(8193);const s=[n(119),n(6134),n(6294),n(1043)];var u=n(8328),c=n(6347),d=n(2831),p=n(4848);function f(e){let{children:t}=e;return(0,p.jsx)(p.Fragment,{children:t})}var m=n(5260),g=n(4586),h=n(6025),b=n(6342),y=n(1213),v=n(2131),w=n(4090),k=n(2967),x=n(440),S=n(1463);function E(){const{i18n:{currentLocale:e,defaultLocale:t,localeConfigs:n}}=(0,g.A)(),r=(0,v.o)(),o=n[e].htmlLang,a=e=>e.replace("-","_");return(0,p.jsxs)(m.A,{children:[Object.entries(n).map((e=>{let[t,{htmlLang:n}]=e;return(0,p.jsx)("link",{rel:"alternate",href:r.createUrl({locale:t,fullyQualified:!0}),hrefLang:n},t)})),(0,p.jsx)("link",{rel:"alternate",href:r.createUrl({locale:t,fullyQualified:!0}),hrefLang:"x-default"}),(0,p.jsx)("meta",{property:"og:locale",content:a(o)}),Object.values(n).filter((e=>o!==e.htmlLang)).map((e=>(0,p.jsx)("meta",{property:"og:locale:alternate",content:a(e.htmlLang)},`meta-og-${e.htmlLang}`)))]})}function _(e){let{permalink:t}=e;const{siteConfig:{url:n}}=(0,g.A)(),r=function(){const{siteConfig:{url:e,baseUrl:t,trailingSlash:n}}=(0,g.A)(),{pathname:r}=(0,c.zy)();return e+(0,x.applyTrailingSlash)((0,h.A)(r),{trailingSlash:n,baseUrl:t})}(),o=t?`${n}${t}`:r;return(0,p.jsxs)(m.A,{children:[(0,p.jsx)("meta",{property:"og:url",content:o}),(0,p.jsx)("link",{rel:"canonical",href:o})]})}function C(){const{i18n:{currentLocale:e}}=(0,g.A)(),{metadata:t,image:n}=(0,b.p)();return(0,p.jsxs)(p.Fragment,{children:[(0,p.jsxs)(m.A,{children:[(0,p.jsx)("meta",{name:"twitter:card",content:"summary_large_image"}),(0,p.jsx)("body",{className:w.w})]}),n&&(0,p.jsx)(y.be,{image:n}),(0,p.jsx)(_,{}),(0,p.jsx)(E,{}),(0,p.jsx)(S.A,{tag:k.Cy,locale:e}),(0,p.jsx)(m.A,{children:t.map(((e,t)=>(0,p.jsx)("meta",{...e},t)))})]})}const A=new Map;function T(e){if(A.has(e.pathname))return{...e,pathname:A.get(e.pathname)};if((0,d.u)(u.A,e.pathname).some((e=>{let{route:t}=e;return!0===t.exact})))return A.set(e.pathname,e.pathname),e;const t=e.pathname.trim().replace(/(?:\/index)?\.html$/,"")||"/";return A.set(e.pathname,t),{...e,pathname:t}}var j=n(6125),N=n(6988),L=n(205);function R(e){for(var t=arguments.length,n=new Array(t>1?t-1:0),r=1;r{const r=t.default?.[e]??t[e];return r?.(...n)}));return()=>o.forEach((e=>e?.()))}const P=function(e){let{children:t,location:n,previousLocation:r}=e;return(0,L.A)((()=>{r!==n&&(!function(e){let{location:t,previousLocation:n}=e;if(!n)return;const r=t.pathname===n.pathname,o=t.hash===n.hash,a=t.search===n.search;if(r&&o&&!a)return;const{hash:i}=t;if(i){const e=decodeURIComponent(i.substring(1)),t=document.getElementById(e);t?.scrollIntoView()}else window.scrollTo(0,0)}({location:n,previousLocation:r}),R("onRouteDidUpdate",{previousLocation:r,location:n}))}),[r,n]),t};function O(e){const t=Array.from(new Set([e,decodeURI(e)])).map((e=>(0,d.u)(u.A,e))).flat();return Promise.all(t.map((e=>e.route.component.preload?.())))}class D extends r.Component{previousLocation;routeUpdateCleanupCb;constructor(e){super(e),this.previousLocation=null,this.routeUpdateCleanupCb=l.A.canUseDOM?R("onRouteUpdate",{previousLocation:null,location:this.props.location}):()=>{},this.state={nextRouteHasLoaded:!0}}shouldComponentUpdate(e,t){if(e.location===this.props.location)return t.nextRouteHasLoaded;const n=e.location;return this.previousLocation=this.props.location,this.setState({nextRouteHasLoaded:!1}),this.routeUpdateCleanupCb=R("onRouteUpdate",{previousLocation:this.previousLocation,location:n}),O(n.pathname).then((()=>{this.routeUpdateCleanupCb(),this.setState({nextRouteHasLoaded:!0})})).catch((e=>{console.warn(e),window.location.reload()})),!1}render(){const{children:e,location:t}=this.props;return(0,p.jsx)(P,{previousLocation:this.previousLocation,location:t,children:(0,p.jsx)(c.qh,{location:t,render:()=>e})})}}const I=D,M="__docusaurus-base-url-issue-banner-container",F="__docusaurus-base-url-issue-banner",z="__docusaurus-base-url-issue-banner-suggestion-container";function B(e){return`\ndocument.addEventListener('DOMContentLoaded', function maybeInsertBanner() {\n var shouldInsert = typeof window['docusaurus'] === 'undefined';\n shouldInsert && insertBanner();\n});\n\nfunction insertBanner() {\n var bannerContainer = document.createElement('div');\n bannerContainer.id = '${M}';\n var bannerHtml = ${JSON.stringify(function(e){return`\n
Automerge-Repo: A "batteries-included" toolkit for building local-first applications
· 10 min read
Today we are announcing our new library, automerge-repo, which makes it vastly easier to build local-first applications with Automerge. Take a look at our quickstart guide or read on for some background and examples.