use crate::api;
use crate::api::client;
use crate::error::OxenError;
use crate::model::RemoteRepository;
use crate::view::merge::{MergeResult, MergeSuccessResponse, Mergeable, MergeableResponse};
pub async fn mergeable(
remote_repo: &RemoteRepository,
base: &str,
head: &str,
) -> Result<Mergeable, OxenError> {
let uri = format!("/merge/{base}..{head}");
let url = api::endpoint::url_from_repo(remote_repo, &uri)?;
log::debug!("api::client::merger::mergeability url: {url}");
let client = client::new_for_url(&url)?;
let res = client.get(&url).send().await?;
let body = client::parse_json_body(&url, res).await?;
let response: MergeableResponse = serde_json::from_str(&body)?;
Ok(response.mergeable)
}
pub async fn merge(
remote_repo: &RemoteRepository,
base: &str,
head: &str,
) -> Result<MergeResult, OxenError> {
let uri = format!("/merge/{base}..{head}");
let url = api::endpoint::url_from_repo(remote_repo, &uri)?;
log::debug!("api::client::merger::merge url: {url}");
let client = client::new_for_url(&url)?;
let res = client.post(&url).send().await?;
let body = client::parse_json_body(&url, res).await?;
let response: MergeSuccessResponse = serde_json::from_str(&body)?;
Ok(response.commits)
}
#[cfg(test)]
mod tests {
use crate::api;
use crate::constants::DEFAULT_REMOTE_NAME;
use crate::error::OxenError;
use crate::opts::FetchOpts;
use crate::opts::PushOpts;
use crate::repositories;
use crate::test;
#[tokio::test]
async fn test_remote_merger_no_commits() -> Result<(), OxenError> {
test::run_training_data_fully_sync_remote(|local_repo, remote_repo| async move {
let base = "main";
let head = "add-data";
repositories::branches::create_checkout(&local_repo, head)?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let mergeability = api::client::merger::mergeable(&remote_repo, base, head).await?;
assert!(mergeability.is_mergeable);
assert_eq!(mergeability.commits.len(), 1);
Ok(remote_repo)
})
.await
}
#[tokio::test]
async fn test_remote_merger_base_is_ahead() -> Result<(), OxenError> {
test::run_training_data_fully_sync_remote(|local_repo, remote_repo| async move {
let base = "main";
let head = "add-data";
repositories::branches::create_checkout(&local_repo, head)?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
repositories::checkout(&local_repo, base).await?;
let path = local_repo.path.join("file_1.txt");
test::write_txt_file_to_path(&path, "hello")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "adding file 1")?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: base.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let mergeability = api::client::merger::mergeable(&remote_repo, base, head).await?;
assert!(mergeability.is_mergeable);
assert_eq!(mergeability.commits.len(), 1);
Ok(remote_repo)
})
.await
}
#[tokio::test]
async fn test_remote_merger_mergeable_multiple_commits() -> Result<(), OxenError> {
test::run_training_data_fully_sync_remote(|local_repo, remote_repo| async move {
let base = "main";
let head = "add-data";
repositories::branches::create_checkout(&local_repo, head)?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let path = local_repo.path.join("README.md");
test::write_txt_file_to_path(&path, "I am the README now")?;
repositories::add(&local_repo, &path).await?;
let path = local_repo.path.join("file_1.txt");
test::write_txt_file_to_path(&path, "hello")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "adding file 1")?;
let path = local_repo.path.join("file_2.txt");
test::write_txt_file_to_path(&path, "world")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "adding file 2")?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let mergeability = api::client::merger::mergeable(&remote_repo, base, head).await?;
println!("Got {} commits", mergeability.commits.len());
for commit in &mergeability.commits {
println!("mergeability commit: {commit:?}");
}
assert!(mergeability.is_mergeable);
assert_eq!(mergeability.commits.len(), 3);
assert_eq!(mergeability.conflicts.len(), 0);
Ok(remote_repo)
})
.await
}
#[tokio::test]
async fn test_remote_merger_multiple_commits_conflict_head_is_ahead() -> Result<(), OxenError> {
test::run_training_data_fully_sync_remote(|local_repo, remote_repo| async move {
let base = "main";
let head = "add-data";
repositories::branches::create_checkout(&local_repo, head)?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let path = local_repo.path.join("README.md");
test::write_txt_file_to_path(&path, "I am the README now")?;
repositories::add(&local_repo, &path).await?;
let path = local_repo.path.join("file_1.txt");
test::write_txt_file_to_path(&path, "hello")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "adding file 1")?;
let path = local_repo.path.join("file_2.txt");
test::write_txt_file_to_path(&path, "world")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "adding file 2")?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
repositories::checkout(&local_repo, base).await?;
let path = local_repo.path.join("README.md");
test::write_txt_file_to_path(&path, "I am on main conflicting the README")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "modifying readme on main")?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: base.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let mergeability = api::client::merger::mergeable(&remote_repo, base, head).await?;
assert!(!mergeability.is_mergeable);
assert_eq!(mergeability.commits.len(), 3);
assert_eq!(mergeability.conflicts.len(), 1);
Ok(remote_repo)
})
.await
}
#[tokio::test]
async fn test_remote_merger_merge_unique() -> Result<(), OxenError> {
test::run_training_data_fully_sync_remote(|local_repo, remote_repo| async move {
let base = "main";
let head = "add-data";
repositories::branches::create_checkout(&local_repo, head)?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let new_file_name = "merge_file.txt";
let path = local_repo.path.join(new_file_name);
test::write_txt_file_to_path(&path, "hello")?;
repositories::add(&local_repo, &path).await?;
repositories::commit(&local_repo, "adding file")?;
let opts = PushOpts {
remote: DEFAULT_REMOTE_NAME.to_string(),
branch: head.to_string(),
..Default::default()
};
repositories::push::push_remote_branch(&local_repo, &opts).await?;
let merge_result = api::client::merger::merge(&remote_repo, base, head).await?;
repositories::checkout::checkout(&local_repo, base).await?;
let commits_before = repositories::commits::list(&local_repo)?;
let fetch_opts = FetchOpts::new();
repositories::pull::pull_remote_branch(&local_repo, &fetch_opts).await?;
let commits_after = repositories::commits::list(&local_repo)?;
assert!(commits_after.len() > commits_before.len());
let path = local_repo.path.join(new_file_name);
assert!(path.exists());
let new_head = repositories::branches::current_branch(&local_repo)?.unwrap();
assert_eq!(new_head.commit_id, merge_result.merge.id);
Ok(remote_repo)
})
.await
}
}