diff --git a/Cargo.toml b/Cargo.toml
index a8191ab86d5567b9322b857b29ddf5181653e3f6..e65b2c51648e0d225f9c160dff33d34015b3aadc 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -19,6 +19,7 @@ serde_derive = "1.0.104"
 serde-wasm-bindgen = "0.1.3"
 vec_map = { path = "vec-map", features = ["serde"] }
 bincode = "1.2.1"
+fixedbitset = "0.2.0"
 # web-sys = { version = "0.3.33", features = ["console"] }
 
 # The `console_error_panic_hook` crate provides better debugging of panics by
diff --git a/src/crdt.rs b/src/crdt.rs
index b69838e00389ac68150fffbe0ff1fe3b0a44b0f0..63248628884c4a3df26bcc23d9f119263bf7233c 100644
--- a/src/crdt.rs
+++ b/src/crdt.rs
@@ -1,7 +1,10 @@
+use fixedbitset::FixedBitSet;
 use serde::de::{self, Deserialize, Deserializer, SeqAccess, Visitor};
 use serde::ser::{Serialize, SerializeSeq, SerializeStruct, SerializeTuple, Serializer};
 use std::cmp;
+use std::collections::hash_map::Entry::{Occupied, Vacant};
 use std::collections::HashMap;
+use std::convert::TryInto;
 use std::fmt;
 use std::ops::{Deref, DerefMut};
 use uuid::Uuid;
@@ -627,6 +630,7 @@ pub trait EventListener {
 
 pub struct CRDT {
     user: Option<u128>,
+    active_strokes: (FixedBitSet, HashMap<StrokeID, u8>),
 
     crdt: Dirty<HashMap<u128, Dirty<User>>>,
     deltas: HashMap<StrokeID, StrokeDelta>,
@@ -638,6 +642,14 @@ impl CRDT {
     pub fn new(event_listener: Box<dyn EventListener>) -> CRDT {
         CRDT {
             user: None,
+            active_strokes: (
+                {
+                    let mut set = FixedBitSet::with_capacity(std::u8::MAX as usize);
+                    set.insert_range(..);
+                    set
+                },
+                HashMap::with_capacity(std::u8::MAX as usize),
+            ),
 
             crdt: Dirty::new(HashMap::new(), false),
             deltas: HashMap::new(),
@@ -669,12 +681,48 @@ impl CRDT {
         self.user.map(CRDT::uuid_to_string)
     }
 
+    pub fn canonicalise_user(uuid: &str) -> Option<String> {
+        match Uuid::parse_str(uuid) {
+            Ok(uuid) => Some(CRDT::uuid_to_string(
+                (uuid.as_u128() & 0xFFFFFFFF_FFFF_0FFF_0FFF_FFFFFFFFFFFFu128)
+                    | 0x00000000_0000_C000_3000_000000000000u128,
+            )),
+            Err(_) => None,
+        }
+    }
+
+    pub fn canonicalise_stroke_author(stroke_id: &str) -> Option<String> {
+        let (author, stroke) = match stroke_id.rfind('-') {
+            Some(split) => stroke_id.split_at(split),
+            None => return None,
+        };
+
+        match stroke[1..].parse::<usize>() {
+            Ok(_) => CRDT::canonicalise_user(author),
+            Err(_) => None,
+        }
+    }
+
     pub fn add_stroke(&mut self, x: i32, y: i32, weight: f32, colour: &str) -> Option<String> {
         let user = match self.user {
             Some(user) => user,
             None => return None,
         };
 
+        let active_id: u8 = match self
+            .active_strokes
+            .0
+            .ones()
+            .next()
+            .and_then(|active_id| active_id.try_into().ok())
+        {
+            Some(active_id) => active_id,
+            None => return None,
+        };
+
+        let user = user
+            ^ ((((active_id & 0xF0u8) as u128) << 72) | (((active_id & 0x0Fu8) as u128) << 60));
+
         let point = match Point::try_new(x, y, weight, colour) {
             Some(point) => point,
             None => return None,
@@ -690,6 +738,11 @@ impl CRDT {
             None => 0,
         };
 
+        self.active_strokes.0.set(active_id as usize, false);
+        self.active_strokes
+            .1
+            .insert(StrokeID::new(user, stroke_idx), active_id);
+
         let mut stroke = Stroke::new(stroke_idx);
         stroke.points.push(point.clone());
 
@@ -708,7 +761,10 @@ impl CRDT {
         Some(CRDT::uuid_to_string(user) + "-" + &stroke_idx.to_string())
     }
 
-    fn split_stroke_id(&self, stroke_id: &str) -> Option<(&Dirty<User>, u128, usize)> {
+    fn split_stroke_id<'crdt>(
+        crdt: &'crdt Dirty<HashMap<u128, Dirty<User>>>,
+        stroke_id: &str,
+    ) -> Option<(&'crdt Dirty<User>, u128, usize)> {
         let (author, stroke) = match stroke_id.rfind('-') {
             Some(split) => stroke_id.split_at(split),
             None => return None,
@@ -719,7 +775,7 @@ impl CRDT {
             None => return None,
         };
 
-        let entry = match self.crdt.get(&author) {
+        let entry = match crdt.get(&author) {
             Some(entry) => entry,
             None => return None,
         };
@@ -730,7 +786,10 @@ impl CRDT {
         }
     }
 
-    fn split_stroke_id_mut(&mut self, stroke_id: &str) -> Option<(&mut Dirty<User>, u128, usize)> {
+    fn split_stroke_id_mut<'crdt>(
+        crdt: &'crdt mut Dirty<HashMap<u128, Dirty<User>>>,
+        stroke_id: &str,
+    ) -> Option<(&'crdt mut Dirty<User>, u128, usize)> {
         let (author, stroke) = match stroke_id.rfind('-') {
             Some(split) => stroke_id.split_at(split),
             None => return None,
@@ -741,7 +800,7 @@ impl CRDT {
             None => return None,
         };
 
-        let entry = match self.crdt.get_mut(&author) {
+        let entry = match crdt.get_mut(&author) {
             Some(entry) => entry,
             None => return None,
         };
@@ -760,16 +819,26 @@ impl CRDT {
         weight: f32,
         colour: &str,
     ) -> bool {
-        let (entry, user, stroke_idx) = match self.split_stroke_id_mut(stroke_id) {
+        let (entry, user, stroke_idx) = match Self::split_stroke_id_mut(&mut self.crdt, stroke_id) {
             Some(stroke) => stroke,
             None => return false,
         };
 
+        if !self
+            .active_strokes
+            .1
+            .contains_key(&StrokeID::new(user, stroke_idx))
+        {
+            return false;
+        };
+
         let idx = match entry.get_stroke_idx(stroke_idx) {
-            Some(idx) if idx == (entry.strokes.len() - 1) => idx,
+            Some(idx) => idx,
             _ => return false,
         };
 
+        assert!(idx == (entry.strokes.len() - 1));
+
         let stroke = match entry.strokes.get_mut(idx) {
             Some(stroke) => stroke,
             _ => return false,
@@ -796,12 +865,28 @@ impl CRDT {
         true
     }
 
+    pub fn end_stroke(&mut self, stroke_id: &str) -> bool {
+        let (_entry, user, stroke_idx) = match Self::split_stroke_id(&self.crdt, stroke_id) {
+            Some(stroke) => stroke,
+            None => return false,
+        };
+
+        match self.active_strokes.1.entry(StrokeID::new(user, stroke_idx)) {
+            Occupied(entry) => {
+                let (_stroke_id, active_id) = entry.remove_entry();
+                self.active_strokes.0.insert(active_id as usize);
+                true
+            }
+            Vacant(_) => false,
+        }
+    }
+
     pub fn erase_stroke(&mut self, stroke_id: &str, from: f32, to: f32) -> bool {
         if from < 0.0 || to < from {
             return false;
         };
 
-        let (entry, user, stroke_idx) = match self.split_stroke_id_mut(stroke_id) {
+        let (entry, user, stroke_idx) = match Self::split_stroke_id_mut(&mut self.crdt, stroke_id) {
             Some(stroke) => stroke,
             None => return false,
         };
@@ -834,7 +919,7 @@ impl CRDT {
     }
 
     pub fn get_stroke_points(&self, stroke_id: &str) -> Option<&VecMap<Point>> {
-        let (entry, _user, stroke_idx) = match self.split_stroke_id(stroke_id) {
+        let (entry, _user, stroke_idx) = match Self::split_stroke_id(&self.crdt, stroke_id) {
             Some(stroke) => stroke,
             None => return None,
         };
@@ -848,7 +933,7 @@ impl CRDT {
     }
 
     pub fn get_stroke_intervals(&self, stroke_id: &str) -> Option<&IntervalUnion<f32>> {
-        let (entry, _user, stroke_idx) = match self.split_stroke_id(stroke_id) {
+        let (entry, _user, stroke_idx) = match Self::split_stroke_id(&self.crdt, stroke_id) {
             Some(stroke) => stroke,
             None => return None,
         };
@@ -986,26 +1071,18 @@ impl CRDT {
     pub fn fetch_deltas_from_state_vector(&self, user: String, remote_state: &StateVec) {
         let mut deltas = HashMap::new();
 
-        //web_sys::console::log_1(&format!("remote state: {:?}", remote_state).into());
-        //web_sys::console::log_1(&format!("local state: {:?}", self.get_state_vector()).into());
-
         for (user, strokes_state) in &self.get_state_vector().strokes {
             let strokes = &self.crdt.get(user).unwrap().strokes;
 
             if let Some(remote_strokes_state) = remote_state.strokes.get(user) {
-                //web_sys::console::log_1(&format!("remote strokes state: {:?}", remote_strokes_state).into());
-                //web_sys::console::log_1(&format!("local strokes state: {:?}", strokes_state).into());
-
                 let mut difference = (*strokes_state).clone();
                 difference.difference(remote_strokes_state);
 
-                //web_sys::console::log_1(&format!("strokes state difference: {:?}", difference).into());
-
                 if difference.is_empty() {
                     continue;
                 };
 
-                for interval in difference {
+                for mut interval in difference {
                     loop {
                         let stroke_idx = match strokes
                             .binary_search_by_key(&interval.from, |stroke| stroke.index)
@@ -1017,17 +1094,11 @@ impl CRDT {
                         let stroke = strokes.get(stroke_idx).unwrap();
                         let points_len = stroke.points.vec_len();
 
-                        //web_sys::console::log_1(&format!("stroke_idx {:?}", stroke_idx).into());
-                        //web_sys::console::log_1(&format!("stroke {:?}", stroke).into());
-                        //web_sys::console::log_1(&format!("points_len {:?}", points_len).into());
-
                         let mut stroke_delta = StrokeDelta::new();
                         stroke_delta
                             .points
                             .reserve(cmp::min(points_len, interval.to - interval.from + 1));
 
-                        //web_sys::console::log_1(&format!("point_idx range {:?}", (interval.from - stroke.index)..=cmp::min(interval.to - stroke.index, points_len - 1)).into());
-
                         for point_idx in (interval.from - stroke.index)
                             ..=cmp::min(interval.to - stroke.index, points_len - 1)
                         {
@@ -1040,6 +1111,8 @@ impl CRDT {
 
                         if interval.to < (stroke.index + points_len) {
                             break;
+                        } else {
+                            interval.from = stroke.index + points_len
                         };
                     }
                 }
@@ -1063,14 +1136,8 @@ impl CRDT {
             }
         }
 
-        //web_sys::console::log_1(&format!("deltas {:?}", deltas).into());
-
         self.event_listener.on_deltas_from_state(user, deltas)
     }
 
-    // TODO: add endStroke() method that finishes the named stroke
-    // TODO: update addStroke() to use a new username (XOR variant + version bits in UUID) if last stroke still unfinished
-    // TODO: keep track of users with unfinished strokes to use "fake" usernames optimally
-
     // TODO: intervals in StateVec
 }
diff --git a/src/lib.rs b/src/lib.rs
index 44daf96cc79de47ea3efd0ea499965537c6b3e3e..e9ad5d2dd97d5ab64e92617f672e4be6f49699bd 100644
--- a/src/lib.rs
+++ b/src/lib.rs
@@ -78,16 +78,35 @@ impl WasmCRDT {
         self.0.get_user()
     }
 
+    pub fn canonicalise_user(user: &str) -> Option<String> {
+        CRDT::canonicalise_user(user)
+    }
+
+    pub fn canonicalise_stroke_author(stroke_id: &str) -> Option<String> {
+        CRDT::canonicalise_stroke_author(stroke_id)
+    }
+
     pub fn add_stroke(&mut self, x: i32, y: i32, weight: f32, colour: &str) -> Option<String> {
         self.0.add_stroke(x, y, weight, colour)
     }
 
-    pub fn add_point(&mut self, stroke: &str, x: i32, y: i32, weight: f32, colour: &str) -> bool {
-        self.0.add_point(stroke, x, y, weight, colour)
+    pub fn add_point(
+        &mut self,
+        stroke_id: &str,
+        x: i32,
+        y: i32,
+        weight: f32,
+        colour: &str,
+    ) -> bool {
+        self.0.add_point(stroke_id, x, y, weight, colour)
+    }
+
+    pub fn end_stroke(&mut self, stroke_id: &str) -> bool {
+        self.0.end_stroke(stroke_id)
     }
 
-    pub fn erase_stroke(&mut self, stroke: &str, from: f32, to: f32) -> bool {
-        self.0.erase_stroke(stroke, from, to)
+    pub fn erase_stroke(&mut self, stroke_id: &str, from: f32, to: f32) -> bool {
+        self.0.erase_stroke(stroke_id, from, to)
     }
 
     pub fn get_stroke_points(&self, stroke_id: &str) -> Result<JsValue, JsValue> {
diff --git a/www/index.js b/www/index.js
index 3d286bdff6cd1f6485b42b620a2507c8dd323fba..d18131d8dc2cfd6544ff1da9fc0a5ec3e01b967d 100644
--- a/www/index.js
+++ b/www/index.js
@@ -1,8 +1,8 @@
-import * as wasm from "drawing-crdt"
+import { WasmCRDT } from "drawing-crdt"
 
 const broadcasts = []
 
-const crdt = new wasm.WasmCRDT({
+const crdt = new WasmCRDT({
   on_stroke: (stroke, points) => {
     console.log("on_stroke:", stroke, points)
   },
@@ -21,6 +21,22 @@ const crdt = new wasm.WasmCRDT({
 console.log("CRDT original StateVec", crdt.get_state_vector())
 
 crdt.set_user("36577c51-a80b-47d6-b3c3-cfb11f705b87")
+console.log("original user:", crdt.get_user())
+console.log("canonical user:", WasmCRDT.canonicalise_user(crdt.get_user()))
+
+console.log("canonical stroke author 1:", WasmCRDT.canonicalise_stroke_author(crdt.add_stroke(4, 2, 3.14, "ffff00")))
+crdt.add_stroke(4, 2, 3.14, "ffff00")
+crdt.add_stroke(4, 2, 3.14, "ffff00")
+
+let end_id = crdt.add_stroke(4, 2, 3.14, "ffff00")
+console.log("canonical stroke author 4:", WasmCRDT.canonicalise_stroke_author(end_id))
+
+crdt.add_stroke(4, 2, 3.14, "ffff00")
+crdt.add_stroke(4, 2, 3.14, "ffff00")
+crdt.add_stroke(4, 2, 3.14, "ffff00")
+crdt.add_stroke(4, 2, 3.14, "ffff00")
+
+crdt.end_stroke(end_id)
 
 let stroke_id = crdt.add_stroke(4, 2, 3.14, "ffff00")
 crdt.add_point(stroke_id, 2, 4, 4.13, "0000ff")
@@ -46,7 +62,7 @@ console.log("pre fetch deltas")
 crdt.fetch_deltas();
 console.log("post fetch deltas")
 
-const crdt2 = new wasm.WasmCRDT({
+const crdt2 = new WasmCRDT({
   on_stroke: (stroke, points) => {
     console.log("on_stroke2:", stroke, points)
   },