mirror of
https://github.com/hexedtech/codemp.git
synced 2024-11-22 07:14:50 +01:00
feat: send granular changes, added deadlocked error
This commit is contained in:
parent
39f2bd6ac2
commit
9df4594408
2 changed files with 27 additions and 32 deletions
|
@ -9,7 +9,7 @@ use tokio::sync::{watch, mpsc, RwLock};
|
|||
use tonic::async_trait;
|
||||
|
||||
use crate::errors::IgnorableError;
|
||||
use crate::{api::Controller, Error};
|
||||
use crate::api::Controller;
|
||||
|
||||
use crate::api::TextChange;
|
||||
|
||||
|
@ -61,7 +61,7 @@ impl Drop for StopOnDrop {
|
|||
impl Controller<TextChange> for BufferController {
|
||||
type Input = TextChange;
|
||||
|
||||
async fn poll(&self) -> Result<(), Error> {
|
||||
async fn poll(&self) -> crate::Result<()> {
|
||||
let mut poller = self.content.clone();
|
||||
loop {
|
||||
poller.changed().await?;
|
||||
|
@ -73,45 +73,37 @@ impl Controller<TextChange> for BufferController {
|
|||
Ok(())
|
||||
}
|
||||
|
||||
fn try_recv(&self) -> Result<Option<TextChange>, Error> {
|
||||
let cur = match self.seen.try_read() {
|
||||
Err(e) => {
|
||||
tracing::error!("try_recv invoked while being mutated: {}", e);
|
||||
return Ok(None);
|
||||
},
|
||||
Ok(x) => x.clone(),
|
||||
};
|
||||
if *self.content.borrow() != cur {
|
||||
match self.seen.try_write() {
|
||||
Err(e) => {
|
||||
tracing::error!("try_recv mutating while being mutated: {}", e);
|
||||
return Ok(None);
|
||||
},
|
||||
Ok(mut w) => {
|
||||
*w = self.content.borrow().clone();
|
||||
// TODO it's not the whole buffer that changed
|
||||
return Ok(Some(TextChange {
|
||||
span: 0..cur.len(),
|
||||
content: self.content.borrow().clone(),
|
||||
after: "".to_string(),
|
||||
}));
|
||||
fn try_recv(&self) -> crate::Result<Option<TextChange>> {
|
||||
match self.seen.try_read() {
|
||||
Err(_) => Err(crate::Error::Deadlocked),
|
||||
Ok(x) => {
|
||||
if *self.content.borrow() != *x {
|
||||
match self.seen.try_write() {
|
||||
Err(_) => Err(crate::Error::Deadlocked),
|
||||
Ok(mut w) => {
|
||||
let change = TextChange::from_diff(&w, &self.content.borrow());
|
||||
*w = self.content.borrow().clone();
|
||||
Ok(Some(change))
|
||||
}
|
||||
}
|
||||
} else {
|
||||
Ok(None)
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
return Ok(None);
|
||||
}
|
||||
|
||||
async fn recv(&self) -> Result<TextChange, Error> {
|
||||
async fn recv(&self) -> crate::Result<TextChange> {
|
||||
self.poll().await?;
|
||||
match self.try_recv()? {
|
||||
Some(x) => Ok(x),
|
||||
None => Err(crate::Error::Filler { message: "wtfff".into() }),
|
||||
}
|
||||
let cur = self.seen.read().await.clone();
|
||||
let change = TextChange::from_diff(&cur, &self.content.borrow());
|
||||
let mut seen = self.seen.write().await;
|
||||
*seen = self.content.borrow().clone();
|
||||
Ok(change)
|
||||
}
|
||||
|
||||
/// enqueue an opseq for processing
|
||||
fn send(&self, op: TextChange) -> Result<(), Error> {
|
||||
fn send(&self, op: TextChange) -> crate::Result<()> {
|
||||
Ok(self.operations.send(op)?)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -61,6 +61,9 @@ pub enum Error {
|
|||
msg: String,
|
||||
},
|
||||
|
||||
/// errors caused by wrong interlocking, safe to retry
|
||||
Deadlocked,
|
||||
|
||||
/// if you see these errors someone is being lazy (:
|
||||
Filler { // TODO filler error, remove later
|
||||
message: String,
|
||||
|
|
Loading…
Reference in a new issue