y-rb 0.6.0-x86_64-linux-gnu
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/ext/yrb/Cargo.toml +21 -0
- data/ext/yrb/extconf.rb +6 -0
- data/ext/yrb/src/lib.rs +649 -0
- data/ext/yrb/src/utils.rs +64 -0
- data/ext/yrb/src/yany.rs +46 -0
- data/ext/yrb/src/yarray.rs +191 -0
- data/ext/yrb/src/yattrs.rs +39 -0
- data/ext/yrb/src/yawareness.rs +151 -0
- data/ext/yrb/src/ydiff.rs +19 -0
- data/ext/yrb/src/ydoc.rs +113 -0
- data/ext/yrb/src/ymap.rs +175 -0
- data/ext/yrb/src/ytext.rs +235 -0
- data/ext/yrb/src/ytransaction.rs +127 -0
- data/ext/yrb/src/yvalue.rs +256 -0
- data/ext/yrb/src/yxml_element.rs +280 -0
- data/ext/yrb/src/yxml_fragment.rs +129 -0
- data/ext/yrb/src/yxml_text.rs +177 -0
- data/lib/3.1/yrb.so +0 -0
- data/lib/3.2/yrb.so +0 -0
- data/lib/3.3/yrb.so +0 -0
- data/lib/3.4/yrb.so +0 -0
- data/lib/y/array.rb +371 -0
- data/lib/y/awareness.rb +290 -0
- data/lib/y/diff.rb +38 -0
- data/lib/y/doc.rb +313 -0
- data/lib/y/map.rb +199 -0
- data/lib/y/text.rb +383 -0
- data/lib/y/transaction.rb +189 -0
- data/lib/y/version.rb +5 -0
- data/lib/y/xml.rb +1141 -0
- data/lib/y-rb.rb +24 -0
- data/lib/y.rb +3 -0
- metadata +143 -0
@@ -0,0 +1,64 @@
|
|
1
|
+
use crate::yvalue::YValue;
|
2
|
+
use magnus::r_hash::ForEach::Continue;
|
3
|
+
use magnus::{exception, Error, RArray, RHash, RString, Symbol, Value};
|
4
|
+
use std::sync::Arc;
|
5
|
+
use yrs::types::{Attrs, Value as YrsValue};
|
6
|
+
use yrs::{Any, Array, Map, TransactionMut};
|
7
|
+
|
8
|
+
#[derive(Debug, Clone)]
|
9
|
+
pub(crate) struct TypeConversionError;
|
10
|
+
|
11
|
+
pub(crate) fn indifferent_hash_key(key: Value) -> Option<String> {
|
12
|
+
RString::from_value(key)
|
13
|
+
.map(|v| v.to_string().unwrap())
|
14
|
+
.or_else(|| Symbol::from_value(key).map(|v| v.name().unwrap().to_string()))
|
15
|
+
}
|
16
|
+
|
17
|
+
pub(crate) fn map_rhash_to_attrs(hash: RHash) -> Result<Attrs, Error> {
|
18
|
+
let mut a: Attrs = Default::default();
|
19
|
+
|
20
|
+
let result = hash.foreach(|key: Value, value: Value| {
|
21
|
+
let k = Arc::from(key.to_string());
|
22
|
+
let v = Any::from(YValue::from(value));
|
23
|
+
|
24
|
+
a.insert(k, v);
|
25
|
+
|
26
|
+
Ok(Continue)
|
27
|
+
});
|
28
|
+
|
29
|
+
if result.is_err() {
|
30
|
+
return Err(Error::new(
|
31
|
+
exception::runtime_error(),
|
32
|
+
"could not map hash to attrs",
|
33
|
+
));
|
34
|
+
}
|
35
|
+
|
36
|
+
Ok(a)
|
37
|
+
}
|
38
|
+
|
39
|
+
pub(crate) fn convert_yvalue_to_ruby_value(value: YrsValue, tx: &TransactionMut) -> YValue {
|
40
|
+
match value {
|
41
|
+
YrsValue::Any(val) => YValue::from(val),
|
42
|
+
YrsValue::YText(text) => YValue::from(text),
|
43
|
+
YrsValue::YXmlElement(el) => YValue::from(el),
|
44
|
+
YrsValue::YXmlText(text) => YValue::from(text),
|
45
|
+
YrsValue::YArray(val) => {
|
46
|
+
let arr = RArray::new();
|
47
|
+
for item in val.iter(tx) {
|
48
|
+
let val = convert_yvalue_to_ruby_value(item.clone(), tx);
|
49
|
+
let val = *val.0.borrow();
|
50
|
+
arr.push(val).expect("cannot push item event to array");
|
51
|
+
}
|
52
|
+
YValue::from(arr)
|
53
|
+
}
|
54
|
+
YrsValue::YMap(val) => {
|
55
|
+
let iter = val.iter(tx).map(|(key, val)| {
|
56
|
+
let val = convert_yvalue_to_ruby_value(val.clone(), tx);
|
57
|
+
let val = val.0.into_inner();
|
58
|
+
(key, val)
|
59
|
+
});
|
60
|
+
YValue::from(RHash::from_iter(iter))
|
61
|
+
}
|
62
|
+
v => panic!("cannot map given yrs values to yvalue: {:?}", v),
|
63
|
+
}
|
64
|
+
}
|
data/ext/yrb/src/yany.rs
ADDED
@@ -0,0 +1,46 @@
|
|
1
|
+
use magnus::r_string::IntoRString;
|
2
|
+
use magnus::value::ReprValue;
|
3
|
+
use magnus::{value, IntoValue, RArray, RHash, RString, Value};
|
4
|
+
use std::borrow::Borrow;
|
5
|
+
use std::ops::{Deref, DerefMut};
|
6
|
+
use yrs::Any;
|
7
|
+
|
8
|
+
pub(crate) struct YAny(pub(crate) Any);
|
9
|
+
|
10
|
+
impl Deref for YAny {
|
11
|
+
type Target = Any;
|
12
|
+
|
13
|
+
fn deref(&self) -> &Self::Target {
|
14
|
+
&self.0
|
15
|
+
}
|
16
|
+
}
|
17
|
+
|
18
|
+
impl DerefMut for YAny {
|
19
|
+
fn deref_mut(&mut self) -> &mut Self::Target {
|
20
|
+
&mut self.0
|
21
|
+
}
|
22
|
+
}
|
23
|
+
|
24
|
+
impl TryInto<Value> for YAny {
|
25
|
+
type Error = ();
|
26
|
+
|
27
|
+
fn try_into(self) -> Result<Value, Self::Error> {
|
28
|
+
return match self.0 {
|
29
|
+
Any::Array(_v) => {
|
30
|
+
let arr = RArray::new();
|
31
|
+
Ok(arr.as_value())
|
32
|
+
}
|
33
|
+
Any::Map(_v) => {
|
34
|
+
let hash = RHash::new();
|
35
|
+
Ok(hash.as_value())
|
36
|
+
}
|
37
|
+
Any::Null => Ok(value::qnil().as_value()),
|
38
|
+
Any::Undefined => Ok(value::qnil().as_value()),
|
39
|
+
Any::Bool(v) => Ok(v.into_value()),
|
40
|
+
Any::Number(v) => Ok(v.into_value()),
|
41
|
+
Any::BigInt(v) => Ok(v.into_value()),
|
42
|
+
Any::String(v) => Ok(v.into_r_string().as_value()),
|
43
|
+
Any::Buffer(v) => Ok(RString::from_slice(v.borrow()).as_value()),
|
44
|
+
};
|
45
|
+
}
|
46
|
+
}
|
@@ -0,0 +1,191 @@
|
|
1
|
+
use crate::utils::convert_yvalue_to_ruby_value;
|
2
|
+
use crate::ytransaction::YTransaction;
|
3
|
+
use crate::yvalue::YValue;
|
4
|
+
use magnus::block::Proc;
|
5
|
+
use magnus::value::Qnil;
|
6
|
+
use magnus::{Error, IntoValue, RArray, RHash, Symbol, Value};
|
7
|
+
use std::cell::RefCell;
|
8
|
+
use yrs::types::Change;
|
9
|
+
use yrs::{Any, Array, ArrayRef, Observable};
|
10
|
+
|
11
|
+
#[magnus::wrap(class = "Y::Array")]
|
12
|
+
pub(crate) struct YArray(pub(crate) RefCell<ArrayRef>);
|
13
|
+
|
14
|
+
/// SAFETY: This is safe because we only access this data when the GVL is held.
|
15
|
+
unsafe impl Send for YArray {}
|
16
|
+
|
17
|
+
impl YArray {
|
18
|
+
pub(crate) fn yarray_each(&self, transaction: &YTransaction, block: Proc) -> Result<(), Error> {
|
19
|
+
let tx = transaction.transaction();
|
20
|
+
let tx = tx.as_ref().unwrap();
|
21
|
+
|
22
|
+
let arr = self.0.borrow();
|
23
|
+
arr.iter(tx).for_each(|val| {
|
24
|
+
let yvalue = *convert_yvalue_to_ruby_value(val, tx).0.borrow();
|
25
|
+
let args = (yvalue,);
|
26
|
+
let _ = block.call::<(Value,), Qnil>(args);
|
27
|
+
});
|
28
|
+
|
29
|
+
Ok(())
|
30
|
+
}
|
31
|
+
|
32
|
+
pub(crate) fn yarray_get(&self, transaction: &YTransaction, index: u32) -> Value {
|
33
|
+
let tx = transaction.transaction();
|
34
|
+
let tx = tx.as_ref().unwrap();
|
35
|
+
|
36
|
+
let arr = self.0.borrow();
|
37
|
+
let v = arr.get(tx, index).unwrap();
|
38
|
+
*convert_yvalue_to_ruby_value(v, tx).0.borrow()
|
39
|
+
}
|
40
|
+
pub(crate) fn yarray_insert(&self, transaction: &YTransaction, index: u32, value: Value) {
|
41
|
+
let yvalue = YValue::from(value);
|
42
|
+
let avalue = Any::from(yvalue);
|
43
|
+
|
44
|
+
let mut tx = transaction.transaction();
|
45
|
+
let tx = tx.as_mut().unwrap();
|
46
|
+
|
47
|
+
let arr = self.0.borrow_mut();
|
48
|
+
arr.insert(tx, index, avalue);
|
49
|
+
}
|
50
|
+
pub(crate) fn yarray_insert_range(
|
51
|
+
&self,
|
52
|
+
transaction: &YTransaction,
|
53
|
+
index: u32,
|
54
|
+
values: RArray,
|
55
|
+
) {
|
56
|
+
let arr = self.0.borrow_mut();
|
57
|
+
let mut tx = transaction.transaction();
|
58
|
+
let tx = tx.as_mut().unwrap();
|
59
|
+
|
60
|
+
let add_values: Vec<Any> = values
|
61
|
+
.each()
|
62
|
+
.into_iter()
|
63
|
+
.map(|value| YValue::from(value.unwrap()).into())
|
64
|
+
.collect();
|
65
|
+
|
66
|
+
arr.insert_range(tx, index, add_values)
|
67
|
+
}
|
68
|
+
pub(crate) fn yarray_length(&self, transaction: &YTransaction) -> u32 {
|
69
|
+
let arr = self.0.borrow();
|
70
|
+
let tx = transaction.transaction();
|
71
|
+
let tx = tx.as_ref().unwrap();
|
72
|
+
|
73
|
+
arr.len(tx)
|
74
|
+
}
|
75
|
+
pub(crate) fn yarray_observe(&self, block: Proc) -> Result<u32, Error> {
|
76
|
+
let change_added = Symbol::new("added").to_static();
|
77
|
+
let change_retain = Symbol::new("retain").to_static();
|
78
|
+
let change_removed = Symbol::new("removed").to_static();
|
79
|
+
|
80
|
+
// let mut error: Option<Error> = None;
|
81
|
+
|
82
|
+
let subscription_id = self
|
83
|
+
.0
|
84
|
+
.borrow_mut()
|
85
|
+
.observe(move |transaction, array_event| {
|
86
|
+
let delta = array_event.delta(transaction);
|
87
|
+
// let mut changes = RArray::with_capacity(delta.len());
|
88
|
+
let (changes, errors): (Vec<_>, Vec<_>) = delta
|
89
|
+
.iter()
|
90
|
+
.map(|change| {
|
91
|
+
let payload = RHash::new();
|
92
|
+
let result = match change {
|
93
|
+
Change::Added(v) => {
|
94
|
+
let values = v
|
95
|
+
.iter()
|
96
|
+
.map(|v| <YValue as Into<Value>>::into(YValue::from(v.clone())))
|
97
|
+
.collect::<RArray>();
|
98
|
+
payload.aset(change_added, values)
|
99
|
+
}
|
100
|
+
Change::Retain(position) => {
|
101
|
+
payload.aset(change_retain, (*position).into_value())
|
102
|
+
}
|
103
|
+
Change::Removed(position) => {
|
104
|
+
payload.aset(change_removed, (*position).into_value())
|
105
|
+
}
|
106
|
+
};
|
107
|
+
|
108
|
+
match result {
|
109
|
+
Ok(()) => Ok(payload),
|
110
|
+
Err(e) => Err(e),
|
111
|
+
}
|
112
|
+
})
|
113
|
+
.partition(Result::is_ok);
|
114
|
+
|
115
|
+
if errors.is_empty() {
|
116
|
+
let args_changes = RArray::new();
|
117
|
+
for change in changes.iter() {
|
118
|
+
let c = *change.as_ref().unwrap();
|
119
|
+
args_changes
|
120
|
+
.push(c)
|
121
|
+
.expect("cannot push change event to args");
|
122
|
+
}
|
123
|
+
|
124
|
+
let args = (args_changes,);
|
125
|
+
let _ = block.call::<(RArray,), Qnil>(args);
|
126
|
+
// todo: make sure we respect the result and bubble up the
|
127
|
+
// error so that we can return as part of the Result
|
128
|
+
}
|
129
|
+
|
130
|
+
// todo: make sure we respect errors and let the method fail by
|
131
|
+
// by returning a Result containing an Error
|
132
|
+
})
|
133
|
+
.into();
|
134
|
+
|
135
|
+
Ok(subscription_id)
|
136
|
+
}
|
137
|
+
pub(crate) fn yarray_push_back(&self, transaction: &YTransaction, value: Value) {
|
138
|
+
let yvalue = YValue::from(value);
|
139
|
+
let avalue = Any::from(yvalue);
|
140
|
+
let mut tx = transaction.transaction();
|
141
|
+
let tx = tx.as_mut().unwrap();
|
142
|
+
|
143
|
+
self.0.borrow_mut().push_back(tx, avalue);
|
144
|
+
}
|
145
|
+
pub(crate) fn yarray_push_front(&self, transaction: &YTransaction, value: Value) {
|
146
|
+
let yvalue = YValue::from(value);
|
147
|
+
let avalue = Any::from(yvalue);
|
148
|
+
|
149
|
+
let mut tx = transaction.transaction();
|
150
|
+
let tx = tx.as_mut().unwrap();
|
151
|
+
|
152
|
+
let arr = self.0.borrow_mut();
|
153
|
+
arr.push_front(tx, avalue);
|
154
|
+
}
|
155
|
+
pub(crate) fn yarray_remove(&self, transaction: &YTransaction, index: u32) {
|
156
|
+
let mut tx = transaction.transaction();
|
157
|
+
let tx = tx.as_mut().unwrap();
|
158
|
+
|
159
|
+
let arr = self.0.borrow_mut();
|
160
|
+
arr.remove(tx, index)
|
161
|
+
}
|
162
|
+
pub(crate) fn yarray_remove_range(&self, transaction: &YTransaction, index: u32, len: u32) {
|
163
|
+
let mut tx = transaction.transaction();
|
164
|
+
let tx = tx.as_mut().unwrap();
|
165
|
+
|
166
|
+
let arr = self.0.borrow_mut();
|
167
|
+
arr.remove_range(tx, index, len)
|
168
|
+
}
|
169
|
+
pub(crate) fn yarray_to_a(&self, transaction: &YTransaction) -> RArray {
|
170
|
+
let arr = self.0.borrow();
|
171
|
+
let tx = transaction.transaction();
|
172
|
+
let tx = tx.as_ref().unwrap();
|
173
|
+
|
174
|
+
let r_arr = RArray::new();
|
175
|
+
for item in arr.iter(tx) {
|
176
|
+
let r_val = YValue::from(item);
|
177
|
+
let r_val = *r_val.0.borrow();
|
178
|
+
r_arr.push(r_val).expect("cannot push item event to array");
|
179
|
+
}
|
180
|
+
r_arr
|
181
|
+
}
|
182
|
+
pub(crate) fn yarray_unobserve(&self, subscription_id: u32) {
|
183
|
+
self.0.borrow_mut().unobserve(subscription_id);
|
184
|
+
}
|
185
|
+
}
|
186
|
+
|
187
|
+
impl From<ArrayRef> for YArray {
|
188
|
+
fn from(v: ArrayRef) -> Self {
|
189
|
+
YArray(RefCell::from(v))
|
190
|
+
}
|
191
|
+
}
|
@@ -0,0 +1,39 @@
|
|
1
|
+
use crate::yvalue::YValue;
|
2
|
+
use magnus::r_hash::ForEach::Continue;
|
3
|
+
use magnus::{RHash, Value};
|
4
|
+
use std::cell::RefCell;
|
5
|
+
use std::sync::Arc;
|
6
|
+
use yrs::types::Attrs;
|
7
|
+
use yrs::Any;
|
8
|
+
|
9
|
+
#[magnus::wrap(class = "Y::Attrs")]
|
10
|
+
#[derive(Clone)]
|
11
|
+
pub(crate) struct YAttrs(pub(crate) RefCell<Attrs>);
|
12
|
+
|
13
|
+
/// SAFETY: This is safe because we only access this data when the GVL is held.
|
14
|
+
unsafe impl Send for YAttrs {}
|
15
|
+
|
16
|
+
impl From<Attrs> for YAttrs {
|
17
|
+
fn from(value: Attrs) -> Self {
|
18
|
+
YAttrs(RefCell::from(value))
|
19
|
+
}
|
20
|
+
}
|
21
|
+
|
22
|
+
impl From<RHash> for YAttrs {
|
23
|
+
fn from(value: RHash) -> Self {
|
24
|
+
let mut attrs = Attrs::new();
|
25
|
+
|
26
|
+
value
|
27
|
+
.foreach(|key: Value, value: Value| {
|
28
|
+
let k = key.to_string();
|
29
|
+
let yvalue = YValue::from(value);
|
30
|
+
let avalue = Any::from(yvalue);
|
31
|
+
attrs.insert(Arc::from(k), avalue);
|
32
|
+
|
33
|
+
Ok(Continue)
|
34
|
+
})
|
35
|
+
.expect("cannot iterate attributes hash");
|
36
|
+
|
37
|
+
YAttrs(RefCell::from(attrs))
|
38
|
+
}
|
39
|
+
}
|
@@ -0,0 +1,151 @@
|
|
1
|
+
use magnus::{block::Proc, exception, Error, Value};
|
2
|
+
use std::borrow::Borrow;
|
3
|
+
use std::cell::RefCell;
|
4
|
+
use std::collections::HashMap;
|
5
|
+
use y_sync::awareness::{Awareness, AwarenessUpdate, Event, UpdateSubscription};
|
6
|
+
use yrs::block::ClientID;
|
7
|
+
use yrs::updates::decoder::Decode;
|
8
|
+
use yrs::updates::encoder::Encode;
|
9
|
+
use yrs::{Doc, OffsetKind, Options};
|
10
|
+
|
11
|
+
#[magnus::wrap(class = "Y::Awareness")]
|
12
|
+
pub(crate) struct YAwareness(pub(crate) RefCell<Awareness>);
|
13
|
+
|
14
|
+
/// SAFETY: This is safe because we only access this data when the GVL is held.
|
15
|
+
unsafe impl Send for YAwareness {}
|
16
|
+
|
17
|
+
impl YAwareness {
|
18
|
+
pub(crate) fn yawareness_new() -> Self {
|
19
|
+
let mut options = Options {
|
20
|
+
offset_kind: OffsetKind::Utf16,
|
21
|
+
..Default::default()
|
22
|
+
};
|
23
|
+
options.offset_kind = OffsetKind::Utf16;
|
24
|
+
|
25
|
+
let doc = Doc::with_options(options);
|
26
|
+
|
27
|
+
let awareness = Awareness::new(doc);
|
28
|
+
|
29
|
+
Self(RefCell::new(awareness))
|
30
|
+
}
|
31
|
+
|
32
|
+
pub(crate) fn yawareness_apply_update(&self, update: Vec<u8>) -> Result<(), Error> {
|
33
|
+
AwarenessUpdate::decode_v1(update.as_slice())
|
34
|
+
.map_err(|_error| Error::new(exception::runtime_error(), "cannot decode update"))
|
35
|
+
.and_then(|value| {
|
36
|
+
self.0.borrow_mut().apply_update(value).map_err(|_error| {
|
37
|
+
Error::new(exception::runtime_error(), "cannot apply awareness update")
|
38
|
+
})
|
39
|
+
})
|
40
|
+
}
|
41
|
+
|
42
|
+
pub(crate) fn yawareness_clean_local_state(&self) {
|
43
|
+
self.0.borrow_mut().clean_local_state();
|
44
|
+
}
|
45
|
+
|
46
|
+
pub(crate) fn yawareness_client_id(&self) -> ClientID {
|
47
|
+
self.0.borrow().client_id()
|
48
|
+
}
|
49
|
+
|
50
|
+
pub(crate) fn yawareness_clients(&self) -> HashMap<ClientID, String> {
|
51
|
+
self.0.borrow().clients().to_owned()
|
52
|
+
}
|
53
|
+
|
54
|
+
pub(crate) fn yawareness_local_state(&self) -> Option<String> {
|
55
|
+
self.0.borrow().local_state().map(|value| value.to_string())
|
56
|
+
}
|
57
|
+
|
58
|
+
pub(crate) fn yawareness_on_update(&self, block: Proc) -> YAwarenessSubscription {
|
59
|
+
let subscription = self.0.borrow_mut().on_update(move |_awareness, event| {
|
60
|
+
let awareness_event = YAwarenessEvent::from(event);
|
61
|
+
let args = (awareness_event,);
|
62
|
+
block
|
63
|
+
.call::<(YAwarenessEvent,), Value>(args)
|
64
|
+
.expect("cannot call block: on_update");
|
65
|
+
});
|
66
|
+
|
67
|
+
// we need to make sure the event handler "survives" and is not being
|
68
|
+
// dropped after leaving this scope, so we pass it back to Ruby.
|
69
|
+
YAwarenessSubscription::from(subscription)
|
70
|
+
}
|
71
|
+
|
72
|
+
pub(crate) fn yawareness_remove_state(&self, client_id: ClientID) {
|
73
|
+
self.0.borrow_mut().remove_state(client_id)
|
74
|
+
}
|
75
|
+
|
76
|
+
pub(crate) fn yawareness_set_local_state(&self, json: String) {
|
77
|
+
self.0.borrow_mut().set_local_state(json)
|
78
|
+
}
|
79
|
+
|
80
|
+
pub(crate) fn yawareness_update(&self) -> Result<Vec<u8>, Error> {
|
81
|
+
self.0
|
82
|
+
.borrow_mut()
|
83
|
+
.update()
|
84
|
+
.map(|update| update.encode_v1())
|
85
|
+
.map_err(|_error| {
|
86
|
+
Error::new(
|
87
|
+
exception::runtime_error(),
|
88
|
+
"cannot create update for current state",
|
89
|
+
)
|
90
|
+
})
|
91
|
+
}
|
92
|
+
|
93
|
+
pub(crate) fn yawareness_update_with_clients(
|
94
|
+
&self,
|
95
|
+
clients: Vec<ClientID>,
|
96
|
+
) -> Result<Vec<u8>, Error> {
|
97
|
+
self.0
|
98
|
+
.borrow_mut()
|
99
|
+
.update_with_clients(clients)
|
100
|
+
.map(|update| update.encode_v1())
|
101
|
+
.map_err(|_error| {
|
102
|
+
Error::new(
|
103
|
+
exception::runtime_error(),
|
104
|
+
"cannot create update for current state and given clients",
|
105
|
+
)
|
106
|
+
})
|
107
|
+
}
|
108
|
+
}
|
109
|
+
|
110
|
+
impl From<Awareness> for YAwareness {
|
111
|
+
fn from(value: Awareness) -> Self {
|
112
|
+
Self(RefCell::from(value))
|
113
|
+
}
|
114
|
+
}
|
115
|
+
|
116
|
+
#[magnus::wrap(class = "Y::AwarenessEvent")]
|
117
|
+
pub(crate) struct YAwarenessEvent(Event);
|
118
|
+
|
119
|
+
/// SAFETY: This is safe because we only access this data when the GVL is held.
|
120
|
+
unsafe impl Send for YAwarenessEvent {}
|
121
|
+
|
122
|
+
impl YAwarenessEvent {
|
123
|
+
pub(crate) fn added(&self) -> Vec<ClientID> {
|
124
|
+
self.0.borrow().added().to_vec()
|
125
|
+
}
|
126
|
+
pub(crate) fn updated(&self) -> Vec<ClientID> {
|
127
|
+
self.0.borrow().updated().to_vec()
|
128
|
+
}
|
129
|
+
pub(crate) fn removed(&self) -> Vec<ClientID> {
|
130
|
+
self.0.borrow().removed().to_vec()
|
131
|
+
}
|
132
|
+
}
|
133
|
+
|
134
|
+
impl From<&Event> for YAwarenessEvent {
|
135
|
+
fn from(value: &Event) -> Self {
|
136
|
+
Self(value.clone())
|
137
|
+
}
|
138
|
+
}
|
139
|
+
|
140
|
+
#[magnus::wrap(class = "Y::AwarenessEvent")]
|
141
|
+
pub(crate) struct YAwarenessSubscription(UpdateSubscription);
|
142
|
+
|
143
|
+
unsafe impl Send for YAwarenessSubscription {}
|
144
|
+
|
145
|
+
impl YAwarenessSubscription {}
|
146
|
+
|
147
|
+
impl From<UpdateSubscription> for YAwarenessSubscription {
|
148
|
+
fn from(v: UpdateSubscription) -> Self {
|
149
|
+
YAwarenessSubscription(v)
|
150
|
+
}
|
151
|
+
}
|
@@ -0,0 +1,19 @@
|
|
1
|
+
use magnus::{IntoValue, RHash, Value};
|
2
|
+
|
3
|
+
unsafe impl Send for YDiff {}
|
4
|
+
|
5
|
+
#[magnus::wrap(class = "Y::Diff")]
|
6
|
+
pub(crate) struct YDiff {
|
7
|
+
pub(crate) ydiff_insert: Value,
|
8
|
+
pub(crate) ydiff_attrs: Option<RHash>,
|
9
|
+
}
|
10
|
+
|
11
|
+
impl YDiff {
|
12
|
+
pub(crate) fn ydiff_insert(&self) -> Value {
|
13
|
+
self.ydiff_insert
|
14
|
+
}
|
15
|
+
|
16
|
+
pub(crate) fn ydiff_attrs(&self) -> Option<Value> {
|
17
|
+
self.ydiff_attrs.as_ref().map(|value| value.into_value())
|
18
|
+
}
|
19
|
+
}
|
data/ext/yrb/src/ydoc.rs
ADDED
@@ -0,0 +1,113 @@
|
|
1
|
+
use crate::yarray::YArray;
|
2
|
+
use crate::ymap::YMap;
|
3
|
+
use crate::ytext::YText;
|
4
|
+
use crate::yxml_element::YXmlElement;
|
5
|
+
use crate::yxml_fragment::YXmlFragment;
|
6
|
+
use crate::yxml_text::YXmlText;
|
7
|
+
use crate::YTransaction;
|
8
|
+
use magnus::block::Proc;
|
9
|
+
use magnus::{exception::runtime_error, Error, Integer, RArray, Value};
|
10
|
+
use std::borrow::Borrow;
|
11
|
+
use std::cell::RefCell;
|
12
|
+
use yrs::updates::decoder::Decode;
|
13
|
+
use yrs::updates::encoder::{Encoder, EncoderV2};
|
14
|
+
use yrs::{Doc, OffsetKind, Options, ReadTxn, StateVector, SubscriptionId, Transact};
|
15
|
+
|
16
|
+
#[magnus::wrap(class = "Y::Doc")]
|
17
|
+
pub(crate) struct YDoc(pub(crate) RefCell<Doc>);
|
18
|
+
|
19
|
+
unsafe impl Send for YDoc {}
|
20
|
+
|
21
|
+
impl YDoc {
|
22
|
+
pub(crate) fn ydoc_new(client_id: &[Value]) -> Self {
|
23
|
+
let mut options = Options::default();
|
24
|
+
if client_id.len() == 1 {
|
25
|
+
let value = client_id.first().unwrap();
|
26
|
+
options.client_id = Integer::from_value(*value).unwrap().to_u64().unwrap();
|
27
|
+
}
|
28
|
+
options.offset_kind = OffsetKind::Utf16;
|
29
|
+
|
30
|
+
let doc = Doc::with_options(options);
|
31
|
+
Self(RefCell::new(doc))
|
32
|
+
}
|
33
|
+
|
34
|
+
pub(crate) fn ydoc_encode_diff_v1(
|
35
|
+
&self,
|
36
|
+
transaction: &YTransaction,
|
37
|
+
state_vector: Vec<u8>,
|
38
|
+
) -> Result<Vec<u8>, Error> {
|
39
|
+
let mut tx = transaction.transaction();
|
40
|
+
let tx = tx.as_mut().unwrap();
|
41
|
+
|
42
|
+
StateVector::decode_v1(state_vector.borrow())
|
43
|
+
.map(|sv| tx.encode_diff_v1(&sv))
|
44
|
+
.map_err(|_e| Error::new(runtime_error(), "cannot encode diff"))
|
45
|
+
}
|
46
|
+
|
47
|
+
pub(crate) fn ydoc_encode_diff_v2(
|
48
|
+
&self,
|
49
|
+
transaction: &YTransaction,
|
50
|
+
state_vector: Vec<u8>,
|
51
|
+
) -> Result<Vec<u8>, Error> {
|
52
|
+
let mut tx = transaction.transaction();
|
53
|
+
let tx = tx.as_mut().unwrap();
|
54
|
+
let mut encoder = EncoderV2::new();
|
55
|
+
|
56
|
+
StateVector::decode_v2(state_vector.borrow())
|
57
|
+
.map(|sv| tx.encode_diff(&sv, &mut encoder))
|
58
|
+
.map(|_| encoder.to_vec())
|
59
|
+
.map_err(|_e| Error::new(runtime_error(), "cannot encode diff"))
|
60
|
+
}
|
61
|
+
|
62
|
+
pub(crate) fn ydoc_get_or_insert_array(&self, name: String) -> YArray {
|
63
|
+
let array_ref = self.0.borrow().get_or_insert_array(name.as_str());
|
64
|
+
YArray::from(array_ref)
|
65
|
+
}
|
66
|
+
|
67
|
+
pub(crate) fn ydoc_get_or_insert_map(&self, name: String) -> YMap {
|
68
|
+
let map_ref = self.0.borrow().get_or_insert_map(name.as_str());
|
69
|
+
YMap::from(map_ref)
|
70
|
+
}
|
71
|
+
|
72
|
+
pub(crate) fn ydoc_get_or_insert_text(&self, name: String) -> YText {
|
73
|
+
let text_ref = self.0.borrow().get_or_insert_text(name.as_str());
|
74
|
+
YText::from(text_ref)
|
75
|
+
}
|
76
|
+
|
77
|
+
pub(crate) fn ydoc_get_or_insert_xml_element(&self, name: String) -> YXmlElement {
|
78
|
+
let xml_element_ref = self.0.borrow_mut().get_or_insert_xml_element(name.as_str());
|
79
|
+
YXmlElement::from(xml_element_ref) // ::into() maps to YXmlFragment instead of YXmlElement :-(
|
80
|
+
}
|
81
|
+
|
82
|
+
pub(crate) fn ydoc_get_or_insert_xml_fragment(&self, name: String) -> YXmlFragment {
|
83
|
+
let xml_fragment_ref = self.0.borrow().get_or_insert_xml_fragment(name.as_str());
|
84
|
+
YXmlFragment::from(xml_fragment_ref)
|
85
|
+
}
|
86
|
+
|
87
|
+
pub(crate) fn ydoc_get_or_insert_xml_text(&self, name: String) -> YXmlText {
|
88
|
+
let xml_text_ref = self.0.borrow().get_or_insert_xml_text(name.as_str());
|
89
|
+
YXmlText::from(xml_text_ref)
|
90
|
+
}
|
91
|
+
|
92
|
+
pub(crate) fn ydoc_transact(&self) -> YTransaction {
|
93
|
+
let doc = self.0.borrow();
|
94
|
+
let transaction = doc.transact_mut();
|
95
|
+
YTransaction::from(transaction)
|
96
|
+
}
|
97
|
+
|
98
|
+
pub(crate) fn ydoc_observe_update(&self, block: Proc) -> Result<SubscriptionId, Error> {
|
99
|
+
self.0
|
100
|
+
.borrow()
|
101
|
+
.observe_update_v1(move |_tx, update_event| {
|
102
|
+
let update = update_event.update.to_vec();
|
103
|
+
let update = RArray::from_vec(update);
|
104
|
+
|
105
|
+
let args: (RArray,) = (update,);
|
106
|
+
block
|
107
|
+
.call::<(RArray,), Value>(args)
|
108
|
+
.expect("cannot call update block");
|
109
|
+
})
|
110
|
+
.map(|v| v.into())
|
111
|
+
.map_err(|err| Error::new(runtime_error(), err.to_string()))
|
112
|
+
}
|
113
|
+
}
|