Rebase title edits, #23
This commit is contained in:
parent
b862ad8c73
commit
bfca5d6e78
2 changed files with 76 additions and 21 deletions
|
@ -43,6 +43,30 @@ pub fn merge_lines<'a>(a: &'a str, o: &'a str, b: &'a str) -> MergeResult<&'a st
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn merge_chars<'a>(a: &'a str, o: &'a str, b: &'a str) -> MergeResult<char> {
|
||||||
|
let oa = diff::chars(o, a);
|
||||||
|
let ob = diff::chars(o, b);
|
||||||
|
|
||||||
|
let chunks = ChunkIterator::new(&oa, &ob);
|
||||||
|
let hunks: Vec<_> = chunks.map(resolve).collect();
|
||||||
|
|
||||||
|
let clean = hunks.iter().all(|x| match x { &Resolved(..) => true, _ => false });
|
||||||
|
|
||||||
|
if clean {
|
||||||
|
MergeResult::Clean(
|
||||||
|
hunks
|
||||||
|
.into_iter()
|
||||||
|
.flat_map(|x| match x {
|
||||||
|
Resolved(y) => y.into_iter(),
|
||||||
|
_ => unreachable!()
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
MergeResult::Conflicted(hunks)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use diff;
|
use diff;
|
||||||
|
@ -51,16 +75,16 @@ mod test {
|
||||||
use super::output::*;
|
use super::output::*;
|
||||||
use super::output::Output::*;
|
use super::output::Output::*;
|
||||||
|
|
||||||
fn merge_chars(a: &str, o: &str, b: &str) -> Vec<Output<char>> {
|
|
||||||
let oa = diff::chars(o, a);
|
|
||||||
let ob = diff::chars(o, b);
|
|
||||||
|
|
||||||
let chunks = super::chunk_iterator::ChunkIterator::new(&oa, &ob);
|
|
||||||
chunks.map(resolve).collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn simple_case() {
|
fn simple_case() {
|
||||||
|
fn merge_chars(a: &str, o: &str, b: &str) -> Vec<Output<char>> {
|
||||||
|
let oa = diff::chars(o, a);
|
||||||
|
let ob = diff::chars(o, b);
|
||||||
|
|
||||||
|
let chunks = super::chunk_iterator::ChunkIterator::new(&oa, &ob);
|
||||||
|
chunks.map(resolve).collect()
|
||||||
|
}
|
||||||
|
|
||||||
assert_eq!(vec![
|
assert_eq!(vec![
|
||||||
Resolved("aaa".chars().collect()),
|
Resolved("aaa".chars().collect()),
|
||||||
Resolved("xxx".chars().collect()),
|
Resolved("xxx".chars().collect()),
|
||||||
|
@ -89,6 +113,15 @@ mod test {
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn clean_case_chars() {
|
||||||
|
assert_eq!(MergeResult::Clean("Title".into()), merge_chars(
|
||||||
|
"Titlle",
|
||||||
|
"titlle",
|
||||||
|
"title",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn false_conflict() {
|
fn false_conflict() {
|
||||||
assert_eq!(MergeResult::Clean("\
|
assert_eq!(MergeResult::Clean("\
|
||||||
|
|
48
src/state.rs
48
src/state.rs
|
@ -174,12 +174,11 @@ impl<'a> SyncState<'a> {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn rebase_update(&self, article_id: i32, target_base_revision: i32, existing_base_revision: i32, _title: &str, body: String)
|
fn rebase_update(&self, article_id: i32, target_base_revision: i32, existing_base_revision: i32, title: String, body: String)
|
||||||
-> Result<String, Error>
|
-> Result<(String, String), Error>
|
||||||
{
|
{
|
||||||
// TODO Also rebase title
|
let mut title_a = title;
|
||||||
|
let mut body_a = body;
|
||||||
let mut a = body;
|
|
||||||
|
|
||||||
for revision in existing_base_revision..target_base_revision {
|
for revision in existing_base_revision..target_base_revision {
|
||||||
let mut stored = article_revisions::table
|
let mut stored = article_revisions::table
|
||||||
|
@ -187,19 +186,27 @@ impl<'a> SyncState<'a> {
|
||||||
.filter(article_revisions::revision.ge(revision))
|
.filter(article_revisions::revision.ge(revision))
|
||||||
.filter(article_revisions::revision.le(revision+1))
|
.filter(article_revisions::revision.le(revision+1))
|
||||||
.order(article_revisions::revision.asc())
|
.order(article_revisions::revision.asc())
|
||||||
.select((article_revisions::body))
|
.select((
|
||||||
.load::<(String)>(self.db_connection)?;
|
article_revisions::title,
|
||||||
|
article_revisions::body,
|
||||||
|
))
|
||||||
|
.load::<(String, String)>(self.db_connection)?;
|
||||||
|
|
||||||
let b = stored.pop().expect("Application layer guarantee");
|
let (title_b, body_b) = stored.pop().expect("Application layer guarantee");
|
||||||
let o = stored.pop().expect("Application layer guarantee");
|
let (title_o, body_o) = stored.pop().expect("Application layer guarantee");
|
||||||
|
|
||||||
a = match merge::merge_lines(&a, &o, &b) {
|
title_a = match merge::merge_chars(&title_a, &title_o, &title_b) {
|
||||||
merge::MergeResult::Clean(merged) => merged,
|
merge::MergeResult::Clean(merged) => merged,
|
||||||
_ => unimplemented!("Missing handling of merge conflicts"),
|
_ => unimplemented!("Missing handling of merge conflicts"),
|
||||||
}
|
};
|
||||||
|
|
||||||
|
body_a = match merge::merge_lines(&body_a, &body_o, &body_b) {
|
||||||
|
merge::MergeResult::Clean(merged) => merged,
|
||||||
|
_ => unimplemented!("Missing handling of merge conflicts"),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(a)
|
Ok((title_a, body_a))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn update_article(&self, article_id: i32, base_revision: i32, title: String, body: String, author: Option<String>)
|
pub fn update_article(&self, article_id: i32, base_revision: i32, title: String, body: String, author: Option<String>)
|
||||||
|
@ -229,7 +236,7 @@ impl<'a> SyncState<'a> {
|
||||||
Err("This edit is based on a future version of the article")?;
|
Err("This edit is based on a future version of the article")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
let body = self.rebase_update(article_id, latest_revision, base_revision, &title, body)?;
|
let (title, body) = self.rebase_update(article_id, latest_revision, base_revision, title, body)?;
|
||||||
|
|
||||||
let new_revision = latest_revision + 1;
|
let new_revision = latest_revision + 1;
|
||||||
|
|
||||||
|
@ -517,4 +524,19 @@ mod test {
|
||||||
|
|
||||||
assert_eq!("a\nx1\nx2\nx3\nb\ny\nc\n", rebase_edit.body);
|
assert_eq!("a\nx1\nx2\nx3\nb\ny\nc\n", rebase_edit.body);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn update_article_when_title_edit_conflict_then_merge_title() {
|
||||||
|
init!(state);
|
||||||
|
|
||||||
|
let article = state.create_article(None, "titlle".into(), "".into(), None).unwrap();
|
||||||
|
|
||||||
|
let first_edit = state.update_article(article.article_id, article.revision, "Titlle".into(), article.body.clone(), None).unwrap();
|
||||||
|
let second_edit = state.update_article(article.article_id, article.revision, "title".into(), article.body.clone(), None).unwrap();
|
||||||
|
|
||||||
|
assert!(article.revision < first_edit.revision);
|
||||||
|
assert!(first_edit.revision < second_edit.revision);
|
||||||
|
|
||||||
|
assert_eq!("Title", second_edit.title);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue