1
0
Fork 0
mirror of https://github.com/Nutomic/ibis.git synced 2024-12-25 13:31:22 +00:00
ibis/tests/common.rs

179 lines
5.2 KiB
Rust
Raw Normal View History

2023-11-24 14:48:43 +00:00
use fediwiki::api::{
2023-11-28 12:04:33 +00:00
ApiConflict, CreateArticleData, EditArticleData, FollowInstance, GetArticleData, ResolveObject,
2023-11-24 14:48:43 +00:00
};
2023-11-16 14:27:35 +00:00
use fediwiki::error::MyResult;
2023-11-24 14:31:31 +00:00
use fediwiki::federation::objects::article::DbArticle;
2023-11-17 13:22:31 +00:00
use fediwiki::federation::objects::instance::DbInstance;
2023-11-20 15:48:29 +00:00
use fediwiki::start;
2023-11-16 14:27:35 +00:00
use once_cell::sync::Lazy;
use reqwest::Client;
use serde::de::Deserialize;
use serde::ser::Serialize;
use std::sync::Once;
2023-11-17 13:36:56 +00:00
use tokio::task::JoinHandle;
2023-11-16 14:27:35 +00:00
use tracing::log::LevelFilter;
2023-11-17 13:22:31 +00:00
use url::Url;
2023-11-16 14:27:35 +00:00
pub static CLIENT: Lazy<Client> = Lazy::new(Client::new);
2023-11-16 14:27:35 +00:00
2023-11-17 13:36:56 +00:00
pub struct TestData {
pub hostname_alpha: &'static str,
2023-11-20 15:48:29 +00:00
pub hostname_beta: &'static str,
pub hostname_gamma: &'static str,
2023-11-17 13:36:56 +00:00
handle_alpha: JoinHandle<()>,
handle_beta: JoinHandle<()>,
handle_gamma: JoinHandle<()>,
2023-11-17 13:36:56 +00:00
}
impl TestData {
pub fn start() -> Self {
static INIT: Once = Once::new();
INIT.call_once(|| {
env_logger::builder()
.filter_level(LevelFilter::Warn)
.filter_module("activitypub_federation", LevelFilter::Info)
.filter_module("fediwiki", LevelFilter::Info)
.init();
});
let hostname_alpha = "localhost:8131";
let hostname_beta = "localhost:8132";
let hostname_gamma = "localhost:8133";
2023-11-17 13:36:56 +00:00
let handle_alpha = tokio::task::spawn(async {
start(hostname_alpha).await.unwrap();
});
let handle_beta = tokio::task::spawn(async {
start(hostname_beta).await.unwrap();
});
let handle_gamma = tokio::task::spawn(async {
start(hostname_gamma).await.unwrap();
});
2023-11-17 13:36:56 +00:00
Self {
hostname_alpha,
hostname_beta,
hostname_gamma,
2023-11-17 13:36:56 +00:00
handle_alpha,
handle_beta,
handle_gamma,
2023-11-17 13:36:56 +00:00
}
}
2023-11-20 15:48:29 +00:00
pub fn stop(self) -> MyResult<()> {
2023-11-17 13:36:56 +00:00
self.handle_alpha.abort();
self.handle_beta.abort();
self.handle_gamma.abort();
2023-11-17 13:36:56 +00:00
Ok(())
}
2023-11-16 14:27:35 +00:00
}
pub const TEST_ARTICLE_DEFAULT_TEXT: &str = "some\nexample\ntext\n";
2023-11-24 14:48:43 +00:00
pub async fn create_article(hostname: &str, title: String) -> MyResult<DbArticle> {
let create_form = CreateArticleData {
title: title.clone(),
};
let article: DbArticle = post(hostname, "article", &create_form).await?;
// create initial edit to ensure that conflicts are generated (there are no conflicts on empty file)
let edit_form = EditArticleData {
ap_id: article.ap_id,
new_text: TEST_ARTICLE_DEFAULT_TEXT.to_string(),
2023-11-27 15:34:45 +00:00
previous_version: article.latest_version,
resolve_conflict_id: None,
};
edit_article(hostname, &title, &edit_form).await
2023-11-24 14:48:43 +00:00
}
pub async fn get_article(hostname: &str, title: &str) -> MyResult<DbArticle> {
let get_article = GetArticleData {
title: title.to_string(),
};
get_query::<DbArticle, _>(hostname, "article", Some(get_article.clone())).await
}
2023-11-27 15:34:45 +00:00
pub async fn edit_article_with_conflict(
hostname: &str,
edit_form: &EditArticleData,
2023-11-28 12:04:33 +00:00
) -> MyResult<Option<ApiConflict>> {
2023-11-27 15:34:45 +00:00
Ok(CLIENT
.patch(format!("http://{}/api/v1/article", hostname))
.form(edit_form)
.send()
.await?
.json()
.await?)
}
2023-11-24 14:31:31 +00:00
pub async fn edit_article(
hostname: &str,
title: &str,
edit_form: &EditArticleData,
) -> MyResult<DbArticle> {
2023-11-28 12:04:33 +00:00
let edit_res: Option<ApiConflict> = CLIENT
2023-11-24 14:31:31 +00:00
.patch(format!("http://{}/api/v1/article", hostname))
.form(edit_form)
.send()
2023-11-27 15:34:45 +00:00
.await?
.json()
2023-11-24 14:31:31 +00:00
.await?;
2023-11-27 15:34:45 +00:00
assert!(edit_res.is_none());
2023-11-24 14:31:31 +00:00
let get_article = GetArticleData {
title: title.to_string(),
};
let updated_article: DbArticle = get_query(hostname, "article", Some(get_article)).await?;
2023-11-24 14:31:31 +00:00
Ok(updated_article)
}
2023-11-16 14:27:35 +00:00
pub async fn get<T>(hostname: &str, endpoint: &str) -> MyResult<T>
where
T: for<'de> Deserialize<'de>,
{
get_query(hostname, endpoint, None::<i32>).await
}
pub async fn get_query<T, R>(hostname: &str, endpoint: &str, query: Option<R>) -> MyResult<T>
where
T: for<'de> Deserialize<'de>,
R: Serialize,
{
let mut res = CLIENT.get(format!("http://{}/api/v1/{}", hostname, endpoint));
if let Some(query) = query {
res = res.query(&query);
}
let alpha_instance: T = res.send().await?.json().await?;
Ok(alpha_instance)
}
2023-11-24 14:48:43 +00:00
async fn post<T: Serialize, R>(hostname: &str, endpoint: &str, form: &T) -> MyResult<R>
2023-11-16 14:27:35 +00:00
where
R: for<'de> Deserialize<'de>,
{
Ok(CLIENT
.post(format!("http://{}/api/v1/{}", hostname, endpoint))
.form(form)
.send()
.await?
.json()
.await?)
}
2023-11-17 13:22:31 +00:00
pub async fn follow_instance(follow_instance: &str, followed_instance: &str) -> MyResult<()> {
// fetch beta instance on alpha
let resolve_form = ResolveObject {
id: Url::parse(&format!("http://{}", followed_instance))?,
};
2023-11-22 15:41:34 +00:00
let instance_resolved: DbInstance =
get_query(followed_instance, "resolve_instance", Some(resolve_form)).await?;
2023-11-17 13:22:31 +00:00
// send follow
let follow_form = FollowInstance {
2023-11-22 15:41:34 +00:00
instance_id: instance_resolved.ap_id,
2023-11-17 13:22:31 +00:00
};
// cant use post helper because follow doesnt return json
CLIENT
.post(format!("http://{}/api/v1/instance/follow", follow_instance))
.form(&follow_form)
.send()
.await?;
Ok(())
}