diff --git a/docs/content/concepts/app-model.md b/docs/content/concepts/app-model.md
new file mode 100644
index 0000000000000..51c88c071dbe7
--- /dev/null
+++ b/docs/content/concepts/app-model.md
@@ -0,0 +1,189 @@
+---
+title: Application model
+order: 0
+---
+
+The Rerun distribution comes with numerous moving pieces:
+* The **SDKs** (Python, Rust & C++), for logging data and querying it back. These are libraries running directly in the end user's process.
+* The **Native Viewer**: the Rerun GUI application for native platforms (Linux, macOS, Windows).
+* The **TCP server**, which receives data from the **SDKs** and forwards it to the **Native Viewer** and/or **WebSocket Server**. The communication is unidirectional: clients push data into the TCP connection, never the other way around.
+* The **Web Viewer**, which packs the **Native Viewer** into a WASM application that can run on the Web and its derivatives (notebooks, etc).
+* The **Web/HTTP Server**, for serving the web page that hosts the **Web Viewer**.
+* The **WebSocket server**, for serving data to the **Web Viewer**. The communication is unidirectional: the server pushes data to the **Web Viewer**, never the other way around.
+* The **CLI**, which allows you to control all the pieces above as well as manipulate RRD files.
+
+The **Native Viewer** always includes:
+ * A **Chunk Store**: an in-memory database that stores the logged data.
+ * A **Renderer**: a 3D engine that renders the contents of the **Chunk Store**.
+
+
+## What runs where?
+
+This is a lot to take in at first, but as we'll see these different pieces are generally deployed in just a few unique configurations for most common cases.
+
+The first thing to understand is what process do each of these things run in.
+
+The **CLI**, **Native Viewer**, **TCP server**, **Web/HTTP Server** and **WebSocket Server** are all part of the same binary: `rerun`.
+Some of them can be enabled or disabled on demand using the appropriate flags but, no matter what, all these pieces are part of the same binary and execute in the same process.
+Keep in mind that even the **Native Viewer** can be disabled (headless mode).
+
+The **SDKs** are vanilla software libraries and therefore always executes in the same context as the end-user's code.
+
+Finally, the **Web Viewer** is a WASM application and therefore has its own dedicated `.wasm` artifact, and always runs in isolation in the end-user's web browser.
+
+The best way to make sense of it all it to look at some of the most common scenarios when:
+* Logging and visualizing data on native.
+* Logging data on native and visualizing it on the web.
+
+
+## Logging and visualizing data on native
+
+There are two common sub-scenarios when working natively:
+* Data is being logged and visualized at the same time (synchronous workflow).
+* Data is being logged first to some persistent storage, and visualized at a later time (asynchronous workflow).
+
+
+### Synchronous workflow
+
+This is the most common kind of Rerun deployment, and also the simplest: one or more **SDKs**, embedded into the user's process, are logging data directly to a **TCP Server**, which in turns feeds the **Native Viewer**.
+Both the **Native Viewer** and the **TCP Server** are running in the same `rerun` process.
+
+Logging script:
+
+snippet: concepts/app-model/native-sync
+
+Deployment:
+
+```sh
+# Start the Rerun Native Viewer in the background.
+#
+# This will also start the TCP server on its default port (9876, use `--port`
+# to pick another one).
+#
+# We could also have just used `spawn()` instead of `connect()` in the logging
+# script above, and # we wouldn't have had to start the Native Viewer manually.
+# `spawn()` does exactly this: it fork-execs a Native Viewer in the background
+# using the first `rerun` # binary available # on your $PATH.
+$ rerun &
+
+# Start logging data. It will be pushed to the Native Viewer through the TCP link.
+$ ./logging_script
+```
+
+
+Dataflow:
+
+
+
+
+Reference:
+* [SDK operating modes: `connect`](../reference/sdk/operating-modes.md#connect)
+* [🐍 Python `connect`](https://ref.rerun.io/docs/python/0.19.0/common/initialization_functions/#rerun.connect)
+* [🦀 Rust `connect`](https://docs.rs/rerun/latest/rerun/struct.RecordingStreamBuilder.html#method.connect)
+* [🌊 C++ `connect`](https://ref.rerun.io/docs/cpp/stable/classrerun_1_1RecordingStream.html#aef3377ffaa2441b906d2bac94dd8fc64)
+
+### Asynchronous workflow
+
+The asynchronous native workflow is similarly simple: one or more **SDKs**, embedded into the user's process, are logging data directly to one or more files.
+The user will then manually start the **Native Viewer** at some later point, in order to visualize these files.
+
+Note: the `rerun` process still embeds both a **Native Viewer** and a **TCP Server**. For each **Native Viewer**, there is **always** an accompanying **TCP Server**, no exception.
+
+Logging script:
+
+snippet: concepts/app-model/native-async
+
+Deployment:
+```sh
+# Log the data into one or more files.
+$ ./logging_script
+
+# Start the Rerun Native Viewer and feed it the RRD file directly.
+#
+# This will also start the TCP server on its default port (9876, use `--port`
+# to pick another one). Although it is not used yet, some client might want
+# to connect in the future.
+$ rerun /tmp/my_recording.rrd
+```
+
+Dataflow:
+
+
+
+
+Reference:
+* [SDK operating modes: `save`](../reference/sdk/operating-modes.md#save)
+* [🐍 Python `save`](https://ref.rerun.io/docs/python/0.19.0/common/initialization_functions/#rerun.save)
+* [🦀 Rust `save`](https://docs.rs/rerun/latest/rerun/struct.RecordingStreamBuilder.html#method.save)
+* [🌊 C++ `save`](https://ref.rerun.io/docs/cpp/stable/classrerun_1_1RecordingStream.html#a555a7940a076c93d951de5b139d14918)
+
+## Logging data on native and visualizing it on the web.
+
+TODO(cmc): incoming.
+
+
+## FAQ
+
+### How can I use multiple **Native Viewers** at the same (i.e. multiple windows)?
+
+Every **Native Viewer** comes with a corresponding **TCP Server** -- always. You cannot start a **Native Viewer** without starting a **TCP server**.
+
+The only way to have more than one Rerun window is to have more than one **TCP server**, by means of the `--port` flag.
+
+E.g.:
+```sh
+# starts a new viewer, listening for TCP connections on :9876
+rerun &
+
+# does nothing, there's already a viewer session running at that address
+rerun &
+
+# does nothing, there's already a viewer session running at that address
+rerun --port 9876 &
+
+# logs the image file to the existing viewer running on :9876
+rerun image.jpg
+
+# logs the image file to the existing viewer running on :9876
+rerun --port 9876 image.jpg
+
+# starts a new viewer, listening for TCP connections on :6789, and logs the image data to it
+rerun --port 6789 image.jpg
+
+# does nothing, there's already a viewer session running at that address
+rerun --port 6789 &
+
+# logs the image file to the existing viewer running on :6789
+rerun --port 6789 image.jpg &
+```
+
+
+### What happens when I use `rr.spawn()` from my SDK of choice?
+
+TODO(cmc): incoming.
+
+
+### What happens when I use `rr.serve()` from my SDK of choice?
+
+TODO(cmc): incoming.
+
+
+### What happens when I use `rerun --serve`?
+
+TODO(cmc): incoming.
+
+
+### Can the **Native Viewer** pull data from a **WebSocket Server**, like the **Web Viewer** does?
+
+TODO(cmc): incoming.
diff --git a/docs/content/reference/sdk/operating-modes.md b/docs/content/reference/sdk/operating-modes.md
index 2606a9f130271..cf6c729067ee5 100644
--- a/docs/content/reference/sdk/operating-modes.md
+++ b/docs/content/reference/sdk/operating-modes.md
@@ -8,6 +8,8 @@ There are many different ways of sending data to the Rerun Viewer depending on w
In the [official examples](/examples), these different modes of operation are exposed via a standardized set of flags that we'll cover below.
We will also demonstrate how you can achieve the same behavior in your own code.
+Before reading this document, you might want to familiarize yourself with the [Rerun application model](../../concepts/app-model.md).
+
## Operating modes
The Rerun SDK provides 4 modes of operation: `spawn`, `connect`, `serve` & `save`.
diff --git a/docs/snippets/all/concepts/app-model/native-async.cpp b/docs/snippets/all/concepts/app-model/native-async.cpp
new file mode 100644
index 0000000000000..77fc4f07908f8
--- /dev/null
+++ b/docs/snippets/all/concepts/app-model/native-async.cpp
@@ -0,0 +1,12 @@
+#include
+
+int main() {
+ // Open a local file handle to stream the data into.
+ const auto rec = rerun::RecordingStream("rerun_example_native_sync");
+ rec.save("/tmp/my_recording.rrd").exit_on_failure();
+
+ // Log data as usual, thereby writing it into the file.
+ while true {
+ rec.log("log", rerun::TextLog("Logging things..."));
+ }
+}
diff --git a/docs/snippets/all/concepts/app-model/native-async.py b/docs/snippets/all/concepts/app-model/native-async.py
new file mode 100755
index 0000000000000..ed7cc903682bb
--- /dev/null
+++ b/docs/snippets/all/concepts/app-model/native-async.py
@@ -0,0 +1,12 @@
+#!/usr/bin/env python3
+
+import rerun as rr
+
+rr.init("rerun_example_native_sync")
+
+# Open a local file handle to stream the data into.
+rr.save("/tmp/my_recording.rrd")
+
+# Log data as usual, thereby writing it into the file.
+while True:
+ rr.log("/", rr.TextLog("Logging things..."))
diff --git a/docs/snippets/all/concepts/app-model/native-async.rs b/docs/snippets/all/concepts/app-model/native-async.rs
new file mode 100644
index 0000000000000..e58ca5e8a7d8e
--- /dev/null
+++ b/docs/snippets/all/concepts/app-model/native-async.rs
@@ -0,0 +1,10 @@
+fn main() -> Result<(), Box> {
+ // Open a local file handle to stream the data into.
+ let rec = rerun::RecordingStreamBuilder::new("rerun_example_native_sync")
+ .save("/tmp/my_recording.rrd")?;
+
+ // Log data as usual, thereby writing it into the file.
+ loop {
+ rec.log("/", &rerun::TextLog::new("Logging things..."))?;
+ }
+}
diff --git a/docs/snippets/all/concepts/app-model/native-sync.cpp b/docs/snippets/all/concepts/app-model/native-sync.cpp
new file mode 100644
index 0000000000000..29f93caaeb1b3
--- /dev/null
+++ b/docs/snippets/all/concepts/app-model/native-sync.cpp
@@ -0,0 +1,13 @@
+#include
+
+int main() {
+ // Connect to the Rerun TCP server using the default address and
+ // port: localhost:9876
+ const auto rec = rerun::RecordingStream("rerun_example_native_sync");
+ rec.connect().exit_on_failure();
+
+ // Log data as usual, thereby pushing it into the TCP socket.
+ while true {
+ rec.log("log", rerun::TextLog("Logging things..."));
+ }
+}
diff --git a/docs/snippets/all/concepts/app-model/native-sync.py b/docs/snippets/all/concepts/app-model/native-sync.py
new file mode 100755
index 0000000000000..6847e4e3fb71a
--- /dev/null
+++ b/docs/snippets/all/concepts/app-model/native-sync.py
@@ -0,0 +1,13 @@
+#!/usr/bin/env python3
+
+import rerun as rr
+
+rr.init("rerun_example_native_sync")
+
+# Connect to the Rerun TCP server using the default address and
+# port: localhost:9876
+rr.connect()
+
+# Log data as usual, thereby pushing it into the TCP socket.
+while True:
+ rr.log("/", rr.TextLog("Logging things..."))
diff --git a/docs/snippets/all/concepts/app-model/native-sync.rs b/docs/snippets/all/concepts/app-model/native-sync.rs
new file mode 100644
index 0000000000000..2db8313dd9dfc
--- /dev/null
+++ b/docs/snippets/all/concepts/app-model/native-sync.rs
@@ -0,0 +1,10 @@
+fn main() -> Result<(), Box> {
+ // Connect to the Rerun TCP server using the default address and
+ // port: localhost:9876
+ let rec = rerun::RecordingStreamBuilder::new("rerun_example_native_sync").connect()?;
+
+ // Log data as usual, thereby pushing it into the TCP socket.
+ loop {
+ rec.log("/", &rerun::TextLog::new("Logging things..."))?;
+ }
+}
diff --git a/scripts/lint.py b/scripts/lint.py
index 678f34d8ff0ee..7a7910d6c4033 100755
--- a/scripts/lint.py
+++ b/scripts/lint.py
@@ -1204,6 +1204,7 @@ def main() -> None:
"./CODE_STYLE.md",
"./crates/build/re_types_builder/src/reflection.rs", # auto-generated
"./crates/store/re_remote_store_types/src/v0/rerun.remote_store.v0.rs", # auto-generated
+ "./docs/content/concepts/app-model.md", # this really needs custom letter casing
"./docs/content/reference/cli.md", # auto-generated
"./examples/assets",
"./examples/python/detect_and_track_objects/cache/version.txt",