mirror of
https://github.com/loro-dev/loro.git
synced 2025-01-23 05:24:51 +00:00
fix: get deep value & throw mismatched context err
This commit is contained in:
parent
c461edd828
commit
88003bdffe
6 changed files with 159 additions and 49 deletions
|
@ -228,6 +228,7 @@ impl TextHandler {
|
|||
pos: start,
|
||||
}),
|
||||
Some(EventHint::Utf16 { pos, len: 0 }),
|
||||
&self.state,
|
||||
)
|
||||
}
|
||||
|
||||
|
@ -254,6 +255,7 @@ impl TextHandler {
|
|||
len: (end - start) as isize,
|
||||
})),
|
||||
Some(EventHint::Utf16 { pos, len: del }),
|
||||
&self.state,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
@ -280,6 +282,7 @@ impl ListHandler {
|
|||
pos,
|
||||
}),
|
||||
None,
|
||||
&self.state,
|
||||
)
|
||||
}
|
||||
|
||||
|
@ -301,6 +304,7 @@ impl ListHandler {
|
|||
pos,
|
||||
}),
|
||||
None,
|
||||
&self.state,
|
||||
)?;
|
||||
Ok(child_idx)
|
||||
}
|
||||
|
@ -317,6 +321,7 @@ impl ListHandler {
|
|||
len: len as isize,
|
||||
})),
|
||||
None,
|
||||
&self.state,
|
||||
)
|
||||
}
|
||||
|
||||
|
@ -344,6 +349,15 @@ impl ListHandler {
|
|||
.get_value_by_idx(self.container_idx)
|
||||
}
|
||||
|
||||
pub fn get_deep_value(&self) -> LoroValue {
|
||||
self.state
|
||||
.upgrade()
|
||||
.unwrap()
|
||||
.lock()
|
||||
.unwrap()
|
||||
.get_container_deep_value(self.container_idx)
|
||||
}
|
||||
|
||||
pub fn id(&self) -> ContainerID {
|
||||
self.state
|
||||
.upgrade()
|
||||
|
@ -390,6 +404,7 @@ impl MapHandler {
|
|||
value,
|
||||
}),
|
||||
None,
|
||||
&self.state,
|
||||
)
|
||||
}
|
||||
|
||||
|
@ -410,6 +425,7 @@ impl MapHandler {
|
|||
value: LoroValue::Container(container_id),
|
||||
}),
|
||||
None,
|
||||
&self.state,
|
||||
)?;
|
||||
Ok(child_idx)
|
||||
}
|
||||
|
@ -423,6 +439,7 @@ impl MapHandler {
|
|||
value: LoroValue::Null,
|
||||
}),
|
||||
None,
|
||||
&self.state,
|
||||
)
|
||||
}
|
||||
|
||||
|
@ -435,6 +452,15 @@ impl MapHandler {
|
|||
.get_value_by_idx(self.container_idx)
|
||||
}
|
||||
|
||||
pub fn get_deep_value(&self) -> LoroValue {
|
||||
self.state
|
||||
.upgrade()
|
||||
.unwrap()
|
||||
.lock()
|
||||
.unwrap()
|
||||
.get_container_deep_value(self.container_idx)
|
||||
}
|
||||
|
||||
pub fn get(&self, key: &str) -> Option<LoroValue> {
|
||||
self.state
|
||||
.upgrade()
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
use std::{
|
||||
borrow::Cow,
|
||||
mem::take,
|
||||
sync::{Arc, Mutex},
|
||||
sync::{Arc, Mutex, Weak},
|
||||
};
|
||||
|
||||
use debug_log::debug_dbg;
|
||||
|
@ -191,7 +191,16 @@ impl Transaction {
|
|||
content: RawOpContent,
|
||||
// we need extra hint to reduce calculation for utf16 text op
|
||||
hint: Option<EventHint>,
|
||||
// check whther context and txn are refering to the same state context
|
||||
state_ref: &Weak<Mutex<DocState>>,
|
||||
) -> LoroResult<()> {
|
||||
if Arc::as_ptr(&self.state) != Weak::as_ptr(state_ref) {
|
||||
return Err(LoroError::UnmatchedContext {
|
||||
expected: self.state.lock().unwrap().peer,
|
||||
found: state_ref.upgrade().unwrap().lock().unwrap().peer,
|
||||
});
|
||||
}
|
||||
|
||||
let len = content.content_len();
|
||||
let op = RawOp {
|
||||
id: ID {
|
||||
|
|
|
@ -291,7 +291,7 @@ fn call_after_micro_task(ob: observer::Observer, e: DiffEvent) {
|
|||
let ans = ob.call1(&event.into());
|
||||
drop(copy);
|
||||
if let Err(e) = ans {
|
||||
console_log!("Error when calling observer: {:#?}", e);
|
||||
console_error!("Error when calling observer: {:#?}", e);
|
||||
}
|
||||
});
|
||||
let _ = promise.then(&closure);
|
||||
|
@ -469,10 +469,9 @@ impl LoroMap {
|
|||
value.into()
|
||||
}
|
||||
|
||||
#[wasm_bindgen(js_name = "getValueDeep")]
|
||||
#[wasm_bindgen(js_name = "getDeepValue")]
|
||||
pub fn get_value_deep(&self) -> JsValue {
|
||||
todo!()
|
||||
// self.0.get_value_deep(ctx.deref()).into()
|
||||
self.0.get_deep_value().into()
|
||||
}
|
||||
|
||||
#[wasm_bindgen(js_name = "insertContainer")]
|
||||
|
@ -556,11 +555,10 @@ impl LoroList {
|
|||
self.0.get_value().into()
|
||||
}
|
||||
|
||||
#[wasm_bindgen(js_name = "getValueDeep")]
|
||||
pub fn get_value_deep(&self) -> JsValue {
|
||||
todo!()
|
||||
// let value = self.0.get_value_deep(ctx.deref());
|
||||
// value.into()
|
||||
#[wasm_bindgen(js_name = "getDeepValue")]
|
||||
pub fn get_deep_value(&self) -> JsValue {
|
||||
let value = self.0.get_deep_value();
|
||||
value.into()
|
||||
}
|
||||
|
||||
#[wasm_bindgen(js_name = "insertContainer")]
|
||||
|
|
|
@ -6,6 +6,8 @@ extern "C" {
|
|||
// `log(..)`
|
||||
#[wasm_bindgen(js_namespace = console)]
|
||||
pub fn log(s: &str);
|
||||
#[wasm_bindgen(js_namespace = console)]
|
||||
pub fn error(s: &str);
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
|
@ -14,3 +16,10 @@ macro_rules! console_log {
|
|||
// `bare_bones`
|
||||
($($t:tt)*) => ($crate::log::log(&format_args!($($t)*).to_string()))
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! console_error {
|
||||
// Note that this is using the `log` function imported above during
|
||||
// `bare_bones`
|
||||
($($t:tt)*) => ($crate::log::error(&format_args!($($t)*).to_string()))
|
||||
}
|
||||
|
|
|
@ -32,8 +32,8 @@ Loro.prototype.transact = function (cb, origin) {
|
|||
});
|
||||
};
|
||||
|
||||
Loro.prototype.getTypedMap = Loro.prototype.getMap;
|
||||
Loro.prototype.getTypedList = Loro.prototype.getList;
|
||||
Loro.prototype.getTypedMap = function (...args) { return this.getMap(...args) };
|
||||
Loro.prototype.getTypedList = function (...args) { return this.getList(...args) };
|
||||
LoroList.prototype.getTyped = function (loro, index) {
|
||||
const value = this.get(index);
|
||||
if (typeof value === "string" && isContainerId(value)) {
|
||||
|
@ -42,7 +42,9 @@ LoroList.prototype.getTyped = function (loro, index) {
|
|||
return value;
|
||||
}
|
||||
};
|
||||
LoroList.prototype.insertTyped = LoroList.prototype.insert;
|
||||
LoroList.prototype.insertTyped = function (...args) {
|
||||
return this.insert(...args)
|
||||
}
|
||||
LoroMap.prototype.getTyped = function (loro, key) {
|
||||
const value = this.get(key);
|
||||
if (typeof value === "string" && isContainerId(value)) {
|
||||
|
@ -51,7 +53,7 @@ LoroMap.prototype.getTyped = function (loro, key) {
|
|||
return value;
|
||||
}
|
||||
};
|
||||
LoroMap.prototype.setTyped = LoroMap.prototype.set;
|
||||
LoroMap.prototype.setTyped = function (...args) { return this.set(...args) };
|
||||
|
||||
LoroText.prototype.insert = function (txn, pos, text) {
|
||||
this.__txn_insert(txn, pos, text);
|
||||
|
|
|
@ -69,19 +69,29 @@ describe("subscribe", () => {
|
|||
let i = 1;
|
||||
const sub = loro.subscribe(() => {
|
||||
if (i > 0) {
|
||||
list.insert(loro, 0, i);
|
||||
loro.transact(txn => {
|
||||
list.insert(txn, 0, i);
|
||||
i--;
|
||||
})
|
||||
}
|
||||
|
||||
count += 1;
|
||||
});
|
||||
text.insert(loro, 0, "hello world");
|
||||
loro.transact((txn) => {
|
||||
text.insert(txn, 0, "hello world");
|
||||
})
|
||||
|
||||
await one_ms();
|
||||
assertEquals(count, 2);
|
||||
text.insert(loro, 0, "hello world");
|
||||
loro.transact((txn) => {
|
||||
text.insert(txn, 0, "hello world");
|
||||
});
|
||||
await one_ms();
|
||||
assertEquals(count, 3);
|
||||
loro.unsubscribe(sub);
|
||||
text.insert(loro, 0, "hello world");
|
||||
loro.transact(txn => {
|
||||
text.insert(txn, 0, "hello world");
|
||||
})
|
||||
await one_ms();
|
||||
assertEquals(count, 3);
|
||||
});
|
||||
|
@ -95,10 +105,16 @@ describe("subscribe", () => {
|
|||
loro.unsubscribe(sub);
|
||||
});
|
||||
assertEquals(count, 0);
|
||||
text.insert(loro, 0, "hello world");
|
||||
loro.transact(txn => {
|
||||
text.insert(txn, 0, "hello world");
|
||||
})
|
||||
|
||||
await one_ms();
|
||||
assertEquals(count, 1);
|
||||
text.insert(loro, 0, "hello world");
|
||||
loro.transact(txn => {
|
||||
text.insert(txn, 0, "hello world");
|
||||
})
|
||||
|
||||
assertEquals(count, 1);
|
||||
});
|
||||
|
||||
|
@ -109,14 +125,20 @@ describe("subscribe", () => {
|
|||
const sub = loro.subscribe(() => {
|
||||
count += 1;
|
||||
});
|
||||
loro.transact(loro => {
|
||||
text.insert(loro, 0, "hello world");
|
||||
})
|
||||
await one_ms();
|
||||
assertEquals(count, 1);
|
||||
loro.transact(loro => {
|
||||
text.insert(loro, 0, "hello world");
|
||||
})
|
||||
await one_ms();
|
||||
assertEquals(count, 2);
|
||||
loro.unsubscribe(sub);
|
||||
loro.transact(loro => {
|
||||
text.insert(loro, 0, "hello world");
|
||||
})
|
||||
await one_ms();
|
||||
assertEquals(count, 2);
|
||||
});
|
||||
|
@ -144,7 +166,10 @@ describe("sync", () => {
|
|||
});
|
||||
const aText = a.getText("text");
|
||||
const bText = b.getText("text");
|
||||
aText.insert(a, 0, "abc");
|
||||
a.transact(txn => {
|
||||
aText.insert(txn, 0, "abc");
|
||||
});
|
||||
|
||||
await one_ms();
|
||||
assertEquals(aText.toString(), bText.toString());
|
||||
});
|
||||
|
@ -152,17 +177,26 @@ describe("sync", () => {
|
|||
it("sync", () => {
|
||||
const loro = new Loro();
|
||||
const text = loro.getText("text");
|
||||
text.insert(loro, 0, "hello world");
|
||||
loro.transact(txn => {
|
||||
text.insert(txn, 0, "hello world");
|
||||
});
|
||||
|
||||
const loro_bk = new Loro();
|
||||
loro_bk.import(loro.exportFrom(undefined));
|
||||
assertEquals(loro_bk.toJson(), loro.toJson());
|
||||
const text_bk = loro_bk.getText("text");
|
||||
assertEquals(text_bk.toString(), "hello world");
|
||||
text_bk.insert(loro_bk, 0, "a ");
|
||||
loro_bk.transact(txn => {
|
||||
text_bk.insert(txn, 0, "a ");
|
||||
});
|
||||
|
||||
loro.import(loro_bk.exportFrom(undefined));
|
||||
assertEquals(text.toString(), "a hello world");
|
||||
const map = loro.getMap("map");
|
||||
map.set(loro, "key", "value");
|
||||
loro.transact(txn => {
|
||||
map.set(txn, "key", "value");
|
||||
});
|
||||
|
||||
});
|
||||
});
|
||||
|
||||
|
@ -198,10 +232,13 @@ describe("prelim", () => {
|
|||
});
|
||||
|
||||
it("prelim map integrate", () => {
|
||||
map.set(loro, "text", prelim_text);
|
||||
map.set(loro, "map", prelim_map);
|
||||
map.set(loro, "list", prelim_list);
|
||||
assertEquals(map.getValueDeep(loro), {
|
||||
loro.transact(txn => {
|
||||
map.set(txn, "text", prelim_text);
|
||||
map.set(txn, "map", prelim_map);
|
||||
map.set(txn, "list", prelim_list);
|
||||
});
|
||||
|
||||
assertEquals(map.getDeepValue(), {
|
||||
text: "hello everyone",
|
||||
map: { a: 1, ab: 123 },
|
||||
list: [0, { a: 4 }],
|
||||
|
@ -212,10 +249,13 @@ describe("prelim", () => {
|
|||
const prelim_text = new PrelimText("ttt");
|
||||
const prelim_map = new PrelimMap({ a: 1, b: 2 });
|
||||
const prelim_list = new PrelimList([1, "2", { a: 4 }]);
|
||||
list.insert(loro, 0, prelim_text);
|
||||
list.insert(loro, 1, prelim_map);
|
||||
list.insert(loro, 2, prelim_list);
|
||||
assertEquals(list.getValueDeep(loro), ["ttt", { a: 1, b: 2 }, [1, "2", {
|
||||
loro.transact(txn => {
|
||||
list.insert(txn, 0, prelim_text);
|
||||
list.insert(txn, 1, prelim_map);
|
||||
list.insert(txn, 2, prelim_list);
|
||||
});
|
||||
|
||||
assertEquals(list.getDeepValue(), ["ttt", { a: 1, b: 2 }, [1, "2", {
|
||||
a: 4,
|
||||
}]]);
|
||||
});
|
||||
|
@ -225,27 +265,40 @@ describe("prelim", () => {
|
|||
describe("wasm", () => {
|
||||
const loro = new Loro();
|
||||
const a = loro.getText("ha");
|
||||
a.insert(loro, 0, "hello world");
|
||||
a.delete(loro, 6, 5);
|
||||
a.insert(loro, 6, "everyone");
|
||||
const b = loro.getMap("ha");
|
||||
b.set(loro, "ab", 123);
|
||||
loro.transact(txn => {
|
||||
a.insert(txn, 0, "hello world");
|
||||
|
||||
const bText = b.insertContainer(loro, "hh", "Text");
|
||||
a.delete(txn, 6, 5);
|
||||
a.insert(txn, 6, "everyone");
|
||||
});
|
||||
const b = loro.getMap("ha");
|
||||
loro.transact(txn => {
|
||||
b.set(txn, "ab", 123);
|
||||
});
|
||||
|
||||
const bText = loro.transact(txn => {
|
||||
return b.insertContainer(txn, "hh", "Text")
|
||||
});
|
||||
|
||||
it("map get", () => {
|
||||
assertEquals(b.get("ab"), 123);
|
||||
});
|
||||
|
||||
it("getValueDeep", () => {
|
||||
bText.insert(loro, 0, "hello world Text");
|
||||
assertEquals(b.getValueDeep(loro), { ab: 123, hh: "hello world Text" });
|
||||
loro.transact(txn => {
|
||||
bText.insert(txn, 0, "hello world Text");
|
||||
});
|
||||
|
||||
assertEquals(b.getDeepValue(), { ab: 123, hh: "hello world Text" });
|
||||
});
|
||||
|
||||
it("should throw error when using the wrong context", () => {
|
||||
expect(() => {
|
||||
const loro2 = new Loro();
|
||||
bText.insert(loro2, 0, "hello world Text");
|
||||
loro2.transact(txn => {
|
||||
bText.insert(txn, 0, "hello world Text");
|
||||
});
|
||||
|
||||
}).toThrow();
|
||||
});
|
||||
|
||||
|
@ -254,7 +307,10 @@ describe("wasm", () => {
|
|||
const b2 = loro.getContainerById(id) as LoroMap;
|
||||
assertEquals(b2.value, b.value);
|
||||
assertEquals(b2.id, id);
|
||||
b2.set(loro, "0", 12);
|
||||
loro.transact(txn => {
|
||||
b2.set(txn, "0", 12);
|
||||
});
|
||||
|
||||
assertEquals(b2.value, b.value);
|
||||
});
|
||||
});
|
||||
|
@ -270,7 +326,10 @@ describe("type", () => {
|
|||
it("test recursive map type", () => {
|
||||
const loro = new Loro<{ map: LoroMap<{ map: LoroMap<{ name: "he" }> }> }>();
|
||||
const map = loro.getTypedMap("map");
|
||||
map.insertContainer(loro, "map", "Map");
|
||||
loro.transact(txn => {
|
||||
map.insertContainer(txn, "map", "Map");
|
||||
});
|
||||
|
||||
const subMap = map.getTyped(loro, "map");
|
||||
const name = subMap.getTyped(loro, "name");
|
||||
expectTypeOf(name).toEqualTypeOf<"he">();
|
||||
|
@ -279,8 +338,15 @@ describe("type", () => {
|
|||
it("works for list type", () => {
|
||||
const loro = new Loro<{ list: LoroList<[string, number]> }>();
|
||||
const list = loro.getTypedList("list");
|
||||
list.insertTyped(loro, 0, "123");
|
||||
list.insertTyped(loro, 1, 123);
|
||||
console.dir((list as any).__proto__);
|
||||
loro.transact(txn => {
|
||||
list.insertTyped(txn, 0, "123");
|
||||
});
|
||||
|
||||
loro.transact(txn => {
|
||||
list.insertTyped(txn, 1, 123);
|
||||
});
|
||||
|
||||
const v0 = list.getTyped(loro, 0);
|
||||
expectTypeOf(v0).toEqualTypeOf<string>();
|
||||
const v1 = list.getTyped(loro, 1);
|
||||
|
|
Loading…
Reference in a new issue