mirror of
https://github.com/lennart-k/rustical.git
synced 2026-01-30 10:38:16 +00:00
Compare commits
6 Commits
c763a682ed
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
008e40e17f | ||
|
|
0703b7b470 | ||
|
|
233cf2ea37 | ||
|
|
494f31f992 | ||
|
|
c1758e2cba | ||
|
|
af60a446ad |
4
Cargo.lock
generated
4
Cargo.lock
generated
@@ -3319,6 +3319,7 @@ dependencies = [
|
||||
"caldata",
|
||||
"clap",
|
||||
"figment",
|
||||
"futures-util",
|
||||
"headers",
|
||||
"http",
|
||||
"insta",
|
||||
@@ -3343,7 +3344,9 @@ dependencies = [
|
||||
"serde",
|
||||
"similar-asserts",
|
||||
"sqlx",
|
||||
"tempfile",
|
||||
"tokio",
|
||||
"tokio-util",
|
||||
"toml 0.9.11+spec-1.1.0",
|
||||
"tower",
|
||||
"tower-http",
|
||||
@@ -4471,6 +4474,7 @@ dependencies = [
|
||||
"bytes",
|
||||
"futures-core",
|
||||
"futures-sink",
|
||||
"futures-util",
|
||||
"pin-project-lite",
|
||||
"tokio",
|
||||
]
|
||||
|
||||
@@ -32,9 +32,6 @@ opentelemetry = [
|
||||
"dep:tracing-opentelemetry",
|
||||
]
|
||||
|
||||
[profile.dev]
|
||||
debug = 0
|
||||
|
||||
[lib]
|
||||
doc = true
|
||||
name = "rustical"
|
||||
@@ -76,6 +73,7 @@ tokio = { version = "1.48", features = [
|
||||
"rt-multi-thread",
|
||||
"full",
|
||||
] }
|
||||
tokio-util = { version = "0.7", features = ["rt"] }
|
||||
url = "2.5"
|
||||
base64 = "0.22"
|
||||
thiserror = "2.0"
|
||||
@@ -158,6 +156,7 @@ rstest.workspace = true
|
||||
rustical_store_sqlite = { workspace = true, features = ["test"] }
|
||||
insta.workspace = true
|
||||
similar-asserts.workspace = true
|
||||
tempfile = "3.24"
|
||||
|
||||
[dependencies]
|
||||
rustical_store.workspace = true
|
||||
@@ -169,6 +168,7 @@ caldata.workspace = true
|
||||
toml.workspace = true
|
||||
serde.workspace = true
|
||||
tokio.workspace = true
|
||||
tokio-util.workspace = true
|
||||
tracing.workspace = true
|
||||
anyhow.workspace = true
|
||||
clap.workspace = true
|
||||
@@ -207,3 +207,4 @@ tower-http.workspace = true
|
||||
axum-extra.workspace = true
|
||||
headers.workspace = true
|
||||
http.workspace = true
|
||||
futures-util.workspace = true
|
||||
|
||||
@@ -37,11 +37,11 @@ impl SqliteStore {
|
||||
}
|
||||
|
||||
pub async fn create_db_pool(db_url: &str, migrate: bool) -> Result<Pool<Sqlite>, sqlx::Error> {
|
||||
let options: SqliteConnectOptions = db_url.parse()?;
|
||||
|
||||
let db = SqlitePool::connect_with(
|
||||
SqliteConnectOptions::new()
|
||||
options
|
||||
.journal_mode(sqlx::sqlite::SqliteJournalMode::Wal)
|
||||
.synchronous(sqlx::sqlite::SqliteSynchronous::Normal)
|
||||
.filename(db_url)
|
||||
.create_if_missing(true),
|
||||
)
|
||||
.await?;
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
use crate::{
|
||||
SqliteStore, addressbook_store::SqliteAddressbookStore, calendar_store::SqliteCalendarStore,
|
||||
principal_store::SqlitePrincipalStore,
|
||||
create_db_pool, principal_store::SqlitePrincipalStore,
|
||||
};
|
||||
use rstest::fixture;
|
||||
use rustical_store::auth::{AuthenticationProvider, Principal, PrincipalType};
|
||||
@@ -9,12 +9,23 @@ use sqlx::SqlitePool;
|
||||
mod addressbook_store;
|
||||
mod calendar_store;
|
||||
|
||||
async fn get_test_db() -> SqlitePool {
|
||||
let db = SqlitePool::connect("sqlite::memory:").await.unwrap();
|
||||
sqlx::migrate!("./migrations").run(&db).await.unwrap();
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct TestStoreContext {
|
||||
pub db: SqlitePool,
|
||||
pub addr_store: SqliteAddressbookStore,
|
||||
pub cal_store: SqliteCalendarStore,
|
||||
pub principal_store: SqlitePrincipalStore,
|
||||
pub sub_store: SqliteStore,
|
||||
}
|
||||
|
||||
#[fixture]
|
||||
pub async fn test_store_context() -> TestStoreContext {
|
||||
let (send_addr, _recv) = tokio::sync::mpsc::channel(1);
|
||||
let (send_cal, _recv) = tokio::sync::mpsc::channel(1);
|
||||
let db = create_db_pool(":memory:", true).await.unwrap();
|
||||
|
||||
// Populate with test data
|
||||
let principal_store = SqlitePrincipalStore::new(db.clone());
|
||||
// Populate with test data
|
||||
principal_store
|
||||
.insert_principal(
|
||||
Principal {
|
||||
@@ -33,28 +44,11 @@ async fn get_test_db() -> SqlitePool {
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
db
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct TestStoreContext {
|
||||
pub db: SqlitePool,
|
||||
pub addr_store: SqliteAddressbookStore,
|
||||
pub cal_store: SqliteCalendarStore,
|
||||
pub principal_store: SqlitePrincipalStore,
|
||||
pub sub_store: SqliteStore,
|
||||
}
|
||||
|
||||
#[fixture]
|
||||
pub async fn test_store_context() -> TestStoreContext {
|
||||
let (send_addr, _recv) = tokio::sync::mpsc::channel(1);
|
||||
let (send_cal, _recv) = tokio::sync::mpsc::channel(1);
|
||||
let db = get_test_db().await;
|
||||
TestStoreContext {
|
||||
db: db.clone(),
|
||||
addr_store: SqliteAddressbookStore::new(db.clone(), send_addr, false),
|
||||
cal_store: SqliteCalendarStore::new(db.clone(), send_cal, false),
|
||||
principal_store: SqlitePrincipalStore::new(db.clone()),
|
||||
principal_store,
|
||||
sub_store: SqliteStore::new(db),
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2,7 +2,7 @@ use crate::config::HttpConfig;
|
||||
use clap::Parser;
|
||||
use http::Method;
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
#[derive(Parser, Debug, Default)]
|
||||
pub struct HealthArgs {}
|
||||
|
||||
/// Healthcheck for running rustical instance
|
||||
|
||||
@@ -8,7 +8,7 @@ use rustical_frontend::FrontendConfig;
|
||||
|
||||
mod health;
|
||||
pub mod membership;
|
||||
mod principals;
|
||||
pub mod principals;
|
||||
|
||||
pub use health::{HealthArgs, cmd_health};
|
||||
pub use principals::{PrincipalsArgs, cmd_principals};
|
||||
|
||||
@@ -1,56 +1,49 @@
|
||||
use super::membership::MembershipArgs;
|
||||
use crate::{config::Config, get_data_stores, membership::cmd_membership};
|
||||
use clap::{Parser, Subcommand};
|
||||
use figment::{
|
||||
Figment,
|
||||
providers::{Env, Format, Toml},
|
||||
};
|
||||
use password_hash::{PasswordHasher, SaltString, rand_core::OsRng};
|
||||
use rustical_store::auth::{AuthenticationProvider, Principal, PrincipalType};
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
pub struct PrincipalsArgs {
|
||||
#[arg(short, long, env, default_value = "/etc/rustical/config.toml")]
|
||||
config_file: String,
|
||||
|
||||
#[command(subcommand)]
|
||||
command: Command,
|
||||
pub command: PrincipalsCommand,
|
||||
}
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
struct CreateArgs {
|
||||
id: String,
|
||||
pub struct CreateArgs {
|
||||
pub id: String,
|
||||
#[arg(value_enum, short, long)]
|
||||
principal_type: Option<PrincipalType>,
|
||||
pub principal_type: Option<PrincipalType>,
|
||||
#[arg(short, long)]
|
||||
name: Option<String>,
|
||||
pub name: Option<String>,
|
||||
#[arg(long, help = "Ask for password input")]
|
||||
password: bool,
|
||||
pub password: bool,
|
||||
}
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
struct RemoveArgs {
|
||||
id: String,
|
||||
pub struct RemoveArgs {
|
||||
pub id: String,
|
||||
}
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
struct EditArgs {
|
||||
id: String,
|
||||
pub struct EditArgs {
|
||||
pub id: String,
|
||||
#[arg(long, help = "Ask for password input")]
|
||||
password: bool,
|
||||
pub password: bool,
|
||||
#[arg(
|
||||
long,
|
||||
help = "Remove password (If you only want to use OIDC for example)"
|
||||
)]
|
||||
remove_password: bool,
|
||||
pub remove_password: bool,
|
||||
#[arg(short, long, help = "Change principal displayname")]
|
||||
name: Option<String>,
|
||||
pub name: Option<String>,
|
||||
#[arg(value_enum, short, long, help = "Change the principal type")]
|
||||
principal_type: Option<PrincipalType>,
|
||||
pub principal_type: Option<PrincipalType>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Subcommand)]
|
||||
enum Command {
|
||||
pub enum PrincipalsCommand {
|
||||
List,
|
||||
Create(CreateArgs),
|
||||
Remove(RemoveArgs),
|
||||
@@ -59,16 +52,11 @@ enum Command {
|
||||
}
|
||||
|
||||
#[allow(clippy::missing_errors_doc, clippy::missing_panics_doc)]
|
||||
pub async fn cmd_principals(args: PrincipalsArgs) -> anyhow::Result<()> {
|
||||
let config: Config = Figment::new()
|
||||
.merge(Toml::file(&args.config_file))
|
||||
.merge(Env::prefixed("RUSTICAL_").split("__"))
|
||||
.extract()?;
|
||||
|
||||
pub async fn cmd_principals(args: PrincipalsArgs, config: Config) -> anyhow::Result<()> {
|
||||
let (_, _, _, principal_store, _) = get_data_stores(true, &config.data_store).await?;
|
||||
|
||||
match args.command {
|
||||
Command::List => {
|
||||
PrincipalsCommand::List => {
|
||||
for principal in principal_store.get_principals().await? {
|
||||
println!(
|
||||
"{} (displayname={}) [{}]",
|
||||
@@ -78,7 +66,7 @@ pub async fn cmd_principals(args: PrincipalsArgs) -> anyhow::Result<()> {
|
||||
);
|
||||
}
|
||||
}
|
||||
Command::Create(CreateArgs {
|
||||
PrincipalsCommand::Create(CreateArgs {
|
||||
id,
|
||||
principal_type,
|
||||
name,
|
||||
@@ -112,11 +100,11 @@ pub async fn cmd_principals(args: PrincipalsArgs) -> anyhow::Result<()> {
|
||||
.await?;
|
||||
println!("Principal created");
|
||||
}
|
||||
Command::Remove(RemoveArgs { id }) => {
|
||||
PrincipalsCommand::Remove(RemoveArgs { id }) => {
|
||||
principal_store.remove_principal(&id).await?;
|
||||
println!("Principal {id} removed");
|
||||
}
|
||||
Command::Edit(EditArgs {
|
||||
PrincipalsCommand::Edit(EditArgs {
|
||||
id,
|
||||
remove_password,
|
||||
password,
|
||||
@@ -152,7 +140,7 @@ pub async fn cmd_principals(args: PrincipalsArgs) -> anyhow::Result<()> {
|
||||
principal_store.insert_principal(principal, true).await?;
|
||||
println!("Principal {id} updated");
|
||||
}
|
||||
Command::Membership(args) => {
|
||||
PrincipalsCommand::Membership(args) => {
|
||||
cmd_membership(principal_store.as_ref(), args).await?;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,107 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCALENDAR
|
||||
VERSION:2.0
|
||||
CALSCALE:GREGORIAN
|
||||
PRODID:RustiCal
|
||||
BEGIN:VTIMEZONE
|
||||
LAST-MODIFIED:20040110T032845Z
|
||||
TZID:US/Eastern
|
||||
BEGIN:DAYLIGHT
|
||||
DTSTART:20000404T020000
|
||||
RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
|
||||
TZNAME:EDT
|
||||
TZOFFSETFROM:-0500
|
||||
TZOFFSETTO:-0400
|
||||
END:DAYLIGHT
|
||||
BEGIN:STANDARD
|
||||
DTSTART:20001026T020000
|
||||
RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
|
||||
TZNAME:EST
|
||||
TZOFFSETFROM:-0400
|
||||
TZOFFSETTO:-0500
|
||||
END:STANDARD
|
||||
END:VTIMEZONE
|
||||
BEGIN:VEVENT
|
||||
DTSTAMP:20060206T001121Z
|
||||
DTSTART;TZID=US/Eastern:20060104T140000
|
||||
DURATION:PT1H
|
||||
RECURRENCE-ID;TZID=US/Eastern:20060104T120000
|
||||
SUMMARY:Event #2 bis
|
||||
UID:[UID]
|
||||
END:VEVENT
|
||||
BEGIN:VEVENT
|
||||
DTSTAMP:20060206T001102Z
|
||||
DTSTART;TZID=US/Eastern:20060102T100000
|
||||
DURATION:PT1H
|
||||
SUMMARY:Event #1
|
||||
Description:Go Steelers!
|
||||
UID:[UID]
|
||||
END:VEVENT
|
||||
BEGIN:VEVENT
|
||||
DTSTAMP:20060206T001121Z
|
||||
DTSTART;TZID=US/Eastern:20060102T120000
|
||||
DURATION:PT1H
|
||||
RRULE:FREQ=DAILY;COUNT=5
|
||||
SUMMARY:Event #2
|
||||
UID:[UID]
|
||||
END:VEVENT
|
||||
BEGIN:VEVENT
|
||||
ATTENDEE;PARTSTAT=ACCEPTED;ROLE=CHAIR:mailto:cyrus@example.com
|
||||
ATTENDEE;PARTSTAT=NEEDS-ACTION:mailto:lisa@example.com
|
||||
DTSTAMP:20060206T001220Z
|
||||
DTSTART;TZID=US/Eastern:20060104T100000
|
||||
DURATION:PT1H
|
||||
LAST-MODIFIED:20060206T001330Z
|
||||
ORGANIZER:mailto:cyrus@example.com
|
||||
SEQUENCE:1
|
||||
STATUS:TENTATIVE
|
||||
SUMMARY:Event #3
|
||||
UID:[UID]
|
||||
END:VEVENT
|
||||
BEGIN:VTODO
|
||||
DTSTAMP:20060205T235335Z
|
||||
DUE;VALUE=DATE:20060104
|
||||
STATUS:NEEDS-ACTION
|
||||
SUMMARY:Task #1
|
||||
UID:[UID]
|
||||
BEGIN:VALARM
|
||||
ACTION:AUDIO
|
||||
TRIGGER;RELATED=START:-PT10M
|
||||
END:VALARM
|
||||
END:VTODO
|
||||
BEGIN:VTODO
|
||||
DTSTAMP:20060205T235300Z
|
||||
DUE;VALUE=DATE:20060106
|
||||
LAST-MODIFIED:20060205T235308Z
|
||||
SEQUENCE:1
|
||||
STATUS:NEEDS-ACTION
|
||||
SUMMARY:Task #2
|
||||
UID:[UID]
|
||||
BEGIN:VALARM
|
||||
ACTION:AUDIO
|
||||
TRIGGER;RELATED=START:-PT10M
|
||||
END:VALARM
|
||||
END:VTODO
|
||||
BEGIN:VTODO
|
||||
COMPLETED:20051223T122322Z
|
||||
DTSTAMP:20060205T235400Z
|
||||
DUE;VALUE=DATE:20051225
|
||||
LAST-MODIFIED:20060205T235308Z
|
||||
SEQUENCE:1
|
||||
STATUS:COMPLETED
|
||||
SUMMARY:Task #3
|
||||
UID:[UID]
|
||||
END:VTODO
|
||||
BEGIN:VTODO
|
||||
DTSTAMP:20060205T235600Z
|
||||
DUE;VALUE=DATE:20060101
|
||||
LAST-MODIFIED:20060205T235308Z
|
||||
SEQUENCE:1
|
||||
STATUS:CANCELLED
|
||||
SUMMARY:Task #4
|
||||
UID:[UID]
|
||||
END:VTODO
|
||||
END:VCALENDAR
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,13 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook_import.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
FN:Simon Perreault
|
||||
N:Perreault;Simon;;;ing. jr,M.Sc.
|
||||
BDAY:--0203
|
||||
GENDER:M
|
||||
EMAIL;TYPE=work:simon.perreault@viagenie.ca
|
||||
UID:[UID]
|
||||
END:VCARD
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
16
src/lib.rs
16
src/lib.rs
@@ -17,6 +17,7 @@ use rustical_store_sqlite::principal_store::SqlitePrincipalStore;
|
||||
use rustical_store_sqlite::{SqliteStore, create_db_pool};
|
||||
use setup_tracing::setup_tracing;
|
||||
use std::sync::Arc;
|
||||
use tokio::sync::Notify;
|
||||
use tokio::sync::mpsc::Receiver;
|
||||
use tower::Layer;
|
||||
use tower_http::normalize_path::NormalizePathLayer;
|
||||
@@ -26,8 +27,6 @@ pub mod app;
|
||||
mod commands;
|
||||
pub use commands::*;
|
||||
pub mod config;
|
||||
#[cfg(test)]
|
||||
pub mod integration_tests;
|
||||
mod setup_tracing;
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
@@ -70,6 +69,7 @@ pub async fn get_data_stores(
|
||||
skip_broken,
|
||||
}) => {
|
||||
let db = create_db_pool(db_url, migrate).await?;
|
||||
|
||||
// Channel to watch for changes (for DAV Push)
|
||||
let (send, recv) = tokio::sync::mpsc::channel(1000);
|
||||
|
||||
@@ -106,8 +106,15 @@ pub async fn get_data_stores(
|
||||
}
|
||||
|
||||
#[allow(clippy::missing_errors_doc, clippy::missing_panics_doc)]
|
||||
pub async fn cmd_default(args: Args, config: Config) -> Result<()> {
|
||||
pub async fn cmd_default(
|
||||
args: Args,
|
||||
config: Config,
|
||||
start_notifier: Option<Arc<Notify>>,
|
||||
tracing: bool,
|
||||
) -> Result<()> {
|
||||
if tracing {
|
||||
setup_tracing(&config.tracing);
|
||||
}
|
||||
|
||||
let (addr_store, cal_store, subscription_store, principal_store, update_recv) =
|
||||
get_data_stores(!args.no_migrations, &config.data_store).await?;
|
||||
@@ -145,6 +152,9 @@ pub async fn cmd_default(args: Args, config: Config) -> Result<()> {
|
||||
let listener = tokio::net::TcpListener::bind(&address).await?;
|
||||
tasks.push(tokio::spawn(async move {
|
||||
info!("RustiCal serving on http://{address}");
|
||||
if let Some(start_notifier) = start_notifier {
|
||||
start_notifier.notify_waiters();
|
||||
}
|
||||
axum::serve(listener, app).await.unwrap();
|
||||
}));
|
||||
|
||||
|
||||
@@ -21,14 +21,16 @@ async fn main() -> Result<()> {
|
||||
|
||||
match args.command {
|
||||
Some(Command::GenConfig(gen_config_args)) => cmd_gen_config(gen_config_args),
|
||||
Some(Command::Principals(principals_args)) => cmd_principals(principals_args).await,
|
||||
Some(Command::Principals(principals_args)) => {
|
||||
cmd_principals(principals_args, parse_config()?).await
|
||||
}
|
||||
Some(Command::Health(health_args)) => {
|
||||
let config: Config = parse_config()?;
|
||||
cmd_health(config.http, health_args).await
|
||||
}
|
||||
None => {
|
||||
let config: Config = parse_config()?;
|
||||
cmd_default(args, config).await
|
||||
cmd_default(args, config, None, true).await
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
84
tests/common/mod.rs
Normal file
84
tests/common/mod.rs
Normal file
@@ -0,0 +1,84 @@
|
||||
use rustical::{
|
||||
Args, cmd_default,
|
||||
config::{Config, DataStoreConfig, HttpConfig, SqliteDataStoreConfig},
|
||||
};
|
||||
use std::{
|
||||
collections::HashSet,
|
||||
net::{Ipv4Addr, SocketAddrV4, TcpListener},
|
||||
sync::{Arc, Mutex, OnceLock},
|
||||
thread::{self, JoinHandle},
|
||||
};
|
||||
use tokio::sync::Notify;
|
||||
use tokio_util::sync::CancellationToken;
|
||||
|
||||
// When running multiple integration tests we need to make sure that they don't get the same port
|
||||
static BOUND_PORTS: OnceLock<Mutex<HashSet<u16>>> = OnceLock::new();
|
||||
|
||||
pub fn find_free_port() -> Option<u16> {
|
||||
let bound_ports = BOUND_PORTS.get_or_init(Mutex::default);
|
||||
let mut bound_ports_write = bound_ports.lock().unwrap();
|
||||
let mut port = 15000;
|
||||
// Frees the socket on drop such that this function returns a free port
|
||||
while TcpListener::bind(SocketAddrV4::new(Ipv4Addr::UNSPECIFIED, port)).is_err()
|
||||
|| bound_ports_write.contains(&port)
|
||||
{
|
||||
port += 1;
|
||||
|
||||
if port >= 16000 {
|
||||
return None;
|
||||
}
|
||||
}
|
||||
bound_ports_write.insert(port);
|
||||
Some(port)
|
||||
}
|
||||
|
||||
pub fn rustical_process(
|
||||
db_url: Option<String>,
|
||||
) -> (CancellationToken, u16, JoinHandle<()>, Arc<Notify>) {
|
||||
let port = find_free_port().unwrap();
|
||||
let token = CancellationToken::new();
|
||||
let cloned_token = token.clone();
|
||||
let start_notify = Arc::new(Notify::new());
|
||||
let cloned_start_notify = start_notify.clone();
|
||||
|
||||
let main_process = thread::spawn(move || {
|
||||
let rt = tokio::runtime::Runtime::new().unwrap();
|
||||
let fut = async {
|
||||
cmd_default(
|
||||
Args {
|
||||
config_file: "asldajldakjsdkj".to_owned(),
|
||||
no_migrations: false,
|
||||
command: None,
|
||||
},
|
||||
Config {
|
||||
data_store: DataStoreConfig::Sqlite(SqliteDataStoreConfig {
|
||||
db_url: db_url.unwrap_or(":memory:".to_owned()),
|
||||
run_repairs: true,
|
||||
skip_broken: false,
|
||||
}),
|
||||
http: HttpConfig {
|
||||
host: "127.0.0.1".to_owned(),
|
||||
port,
|
||||
..Default::default()
|
||||
},
|
||||
frontend: Default::default(),
|
||||
oidc: None,
|
||||
tracing: Default::default(),
|
||||
dav_push: Default::default(),
|
||||
nextcloud_login: Default::default(),
|
||||
caldav: Default::default(),
|
||||
},
|
||||
Some(cloned_start_notify),
|
||||
false,
|
||||
)
|
||||
.await
|
||||
};
|
||||
rt.block_on(async {
|
||||
tokio::select! {
|
||||
_ = cloned_token.cancelled() => {},
|
||||
_ = fut => {}
|
||||
}
|
||||
});
|
||||
});
|
||||
(token, port, main_process, start_notify)
|
||||
}
|
||||
137
tests/http_integration.rs
Normal file
137
tests/http_integration.rs
Normal file
@@ -0,0 +1,137 @@
|
||||
// This integration test checks whether the HTTP server works by actually running rustical in a new
|
||||
// thread.
|
||||
use common::rustical_process;
|
||||
use http::{Method, StatusCode};
|
||||
use rustical::{
|
||||
PrincipalsArgs, cmd_health, cmd_principals,
|
||||
config::{Config, DataStoreConfig, HttpConfig, SqliteDataStoreConfig},
|
||||
principals::{CreateArgs, PrincipalsCommand},
|
||||
};
|
||||
use rustical_store::auth::{AuthenticationProvider, PrincipalType};
|
||||
use rustical_store_sqlite::{create_db_pool, principal_store::SqlitePrincipalStore};
|
||||
use std::time::Duration;
|
||||
|
||||
mod common;
|
||||
|
||||
pub async fn test_runner<O, F>(db_path: Option<String>, inner: F)
|
||||
where
|
||||
O: IntoFuture<Output = ()>,
|
||||
// <O as IntoFuture>::IntoFuture: UnwindSafe,
|
||||
F: FnOnce(u16) -> O,
|
||||
{
|
||||
// Start RustiCal process
|
||||
let (token, port, main_process, start_notify) = rustical_process(db_path);
|
||||
|
||||
// Wait for RustiCal server to listen
|
||||
tokio::time::timeout(Duration::new(2, 0), start_notify.notified())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
// We use catch_unwind to make sure we'll always correctly stop RustiCal
|
||||
// Otherwise, our process would just run indefinitely
|
||||
inner(port).into_future().await;
|
||||
|
||||
// Signal RustiCal to stop
|
||||
token.cancel();
|
||||
main_process.join().unwrap();
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn test_ping() {
|
||||
test_runner(None, async |port| {
|
||||
let origin = format!("http://localhost:{port}");
|
||||
let resp = reqwest::get(origin.clone() + "/ping").await.unwrap();
|
||||
assert_eq!(resp.status(), StatusCode::OK);
|
||||
|
||||
// Ensure that path normalisation works as intended
|
||||
let resp = reqwest::get(origin + "/ping/").await.unwrap();
|
||||
assert_eq!(resp.status(), StatusCode::OK);
|
||||
|
||||
cmd_health(
|
||||
HttpConfig {
|
||||
host: "localhost".to_owned(),
|
||||
port,
|
||||
..Default::default()
|
||||
},
|
||||
Default::default(),
|
||||
)
|
||||
.await
|
||||
.unwrap();
|
||||
})
|
||||
.await
|
||||
}
|
||||
|
||||
// When setting a use password from the CLI we effectively have two processes accessing the same
|
||||
// database: The server and the CLI.
|
||||
// This test ensures that the server correctly picks up the changes made by the CLI.
|
||||
#[tokio::test]
|
||||
async fn test_initial_setup() {
|
||||
let db_tempfile = tempfile::NamedTempFile::with_suffix(".rustical-test.sqlite3").unwrap();
|
||||
let db_path = db_tempfile.path().to_string_lossy().into_owned();
|
||||
|
||||
test_runner(Some(db_path.clone()), async |port| {
|
||||
let origin = format!("http://localhost:{port}");
|
||||
// Create principal
|
||||
cmd_principals(
|
||||
PrincipalsArgs {
|
||||
command: PrincipalsCommand::Create(CreateArgs {
|
||||
id: "user".to_owned(),
|
||||
name: Some("Test User".to_owned()),
|
||||
password: false,
|
||||
principal_type: Some(PrincipalType::Individual),
|
||||
}),
|
||||
},
|
||||
Config {
|
||||
data_store: DataStoreConfig::Sqlite(SqliteDataStoreConfig {
|
||||
db_url: db_path.clone(),
|
||||
run_repairs: true,
|
||||
skip_broken: false,
|
||||
}),
|
||||
http: Default::default(),
|
||||
frontend: Default::default(),
|
||||
oidc: None,
|
||||
tracing: Default::default(),
|
||||
dav_push: Default::default(),
|
||||
nextcloud_login: Default::default(),
|
||||
caldav: Default::default(),
|
||||
},
|
||||
)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
// Bodge to set password without using command (since that reads stdin)
|
||||
let db = create_db_pool(&db_path, false).await.unwrap();
|
||||
let principal_store = SqlitePrincipalStore::new(db);
|
||||
let app_token = "token";
|
||||
principal_store
|
||||
.add_app_token("user", "Test Token".to_owned(), app_token.to_owned())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let url = origin.clone() + "/caldav/principal/user";
|
||||
let resp = reqwest::Client::new()
|
||||
.request(Method::from_bytes(b"PROPFIND").unwrap(), &url)
|
||||
.send()
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(resp.status(), StatusCode::UNAUTHORIZED);
|
||||
|
||||
let resp = reqwest::Client::new()
|
||||
.request(Method::from_bytes(b"PROPFIND").unwrap(), &url)
|
||||
.basic_auth("user", Some("token"))
|
||||
.send()
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(resp.status(), StatusCode::MULTI_STATUS);
|
||||
|
||||
principal_store.remove_principal("user").await.unwrap();
|
||||
|
||||
let resp = reqwest::Client::new()
|
||||
.request(Method::from_bytes(b"PROPFIND").unwrap(), &url)
|
||||
.send()
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(resp.status(), StatusCode::UNAUTHORIZED);
|
||||
})
|
||||
.await;
|
||||
}
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -1,3 +1,4 @@
|
||||
use super::{ResponseExtractString, calendar::mkcalendar_template, get_app};
|
||||
use axum::body::Body;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
use http::{Request, StatusCode};
|
||||
@@ -6,10 +7,6 @@ use rustical_store::CalendarMetadata;
|
||||
use rustical_store_sqlite::tests::{TestStoreContext, test_store_context};
|
||||
use tower::ServiceExt;
|
||||
|
||||
use crate::integration_tests::{
|
||||
ResponseExtractString, caldav::calendar::mkcalendar_template, get_app,
|
||||
};
|
||||
|
||||
#[rstest]
|
||||
#[tokio::test]
|
||||
async fn test_put_invalid(
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar.rs
|
||||
source: tests/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCALENDAR
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar.rs
|
||||
source: tests/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar.rs
|
||||
source: tests/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar.rs
|
||||
source: tests/integration_tests/caldav/calendar.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_import.rs
|
||||
source: tests/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCALENDAR
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_import.rs
|
||||
source: tests/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCALENDAR
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/caldav/calendar_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_report.rs
|
||||
source: tests/integration_tests/caldav/calendar_report.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_report.rs
|
||||
source: tests/integration_tests/caldav/calendar_report.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/calendar_report.rs
|
||||
source: tests/integration_tests/caldav/calendar_report.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/mod.rs
|
||||
source: tests/integration_tests/caldav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/mod.rs
|
||||
source: tests/integration_tests/caldav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/caldav/mod.rs
|
||||
source: tests/integration_tests/caldav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -27,11 +27,10 @@ async fn test_import(
|
||||
.body(Body::from(
|
||||
r"BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
FN:Simon Perreault
|
||||
N:Perreault;Simon;;;ing. jr,M.Sc.
|
||||
FN:John Doe
|
||||
N:Doe;John;;;,
|
||||
BDAY:--0203
|
||||
GENDER:M
|
||||
EMAIL;TYPE=work:simon.perreault@viagenie.ca
|
||||
END:VCARD",
|
||||
))
|
||||
.unwrap()
|
||||
@@ -1,4 +1,4 @@
|
||||
use crate::integration_tests::{ResponseExtractString, get_app};
|
||||
use super::{ResponseExtractString, get_app};
|
||||
use axum::body::Body;
|
||||
use axum::extract::Request;
|
||||
use headers::{Authorization, HeaderMapExt};
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook.rs
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook.rs
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/addressbook.rs
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -0,0 +1,12 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook_import.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
FN:John Doe
|
||||
N:Doe;John;;;,
|
||||
BDAY:--0203
|
||||
GENDER:M
|
||||
UID:[UID]
|
||||
END:VCARD
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
---
|
||||
source: src/integration_tests/carddav/mod.rs
|
||||
source: tests/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,35 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<multistatus xmlns="DAV:" xmlns:CAL="urn:ietf:params:xml:ns:caldav" xmlns:CARD="urn:ietf:params:xml:ns:carddav" xmlns:CS="http://calendarserver.org/ns/" xmlns:PUSH="https://bitfire.at/webdav-push">
|
||||
<response>
|
||||
<href>/carddav/principal/user/contacts/newcard.vcf</href>
|
||||
<propstat>
|
||||
<prop>
|
||||
<getetag>"ea0bf4a2ce7ef84606a4cf9235776dbc11b3e7ce351ddf35f27cbc0088acca7e"</getetag>
|
||||
<CARD:address-data>BEGIN:VCARD
|
||||
VERSION:3.0
|
||||
FN:Cyrus Daboo
|
||||
N:Daboo;Cyrus
|
||||
ADR;TYPE=POSTAL:;2822 Email HQ;Suite 2821;RFCVille;PA;15213;USA
|
||||
EMAIL;TYPE=INTERNET,PREF:cyrus@example.com
|
||||
NICKNAME:me
|
||||
NOTE:Example VCard.
|
||||
ORG:Self Employed
|
||||
TEL;TYPE=WORK,VOICE:412 605 0499
|
||||
TEL;TYPE=FAX:412 605 0705
|
||||
URL:http://www.example.com
|
||||
UID:1234-5678-9000-1
|
||||
END:VCARD
|
||||
</CARD:address-data>
|
||||
</prop>
|
||||
<status>HTTP/1.1 200 OK</status>
|
||||
</propstat>
|
||||
</response>
|
||||
<response>
|
||||
<href>/home/bernard/addressbook/vcf1.vcf</href>
|
||||
<status>HTTP/1.1 404 Not Found</status>
|
||||
</response>
|
||||
</multistatus>
|
||||
@@ -0,0 +1,68 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<multistatus xmlns="DAV:" xmlns:CAL="urn:ietf:params:xml:ns:caldav" xmlns:CARD="urn:ietf:params:xml:ns:carddav" xmlns:CS="http://calendarserver.org/ns/" xmlns:PUSH="https://bitfire.at/webdav-push">
|
||||
<response>
|
||||
<href>/carddav/principal/user/contacts/</href>
|
||||
<propstat>
|
||||
<prop>
|
||||
<CARD:addressbook-description>Amazing contacts!</CARD:addressbook-description>
|
||||
<CARD:supported-address-data>
|
||||
<CARD:address-data-type content-type="text/vcard" version="3.0"/>
|
||||
<CARD:address-data-type content-type="text/vcard" version="4.0"/>
|
||||
</CARD:supported-address-data>
|
||||
<CARD:supported-collation-set>
|
||||
<CARD:supported-collation>i;ascii-casemap</CARD:supported-collation>
|
||||
<CARD:supported-collation>i;unicode-casemap</CARD:supported-collation>
|
||||
<CARD:supported-collation>i;octet</CARD:supported-collation>
|
||||
</CARD:supported-collation-set>
|
||||
<supported-report-set>
|
||||
<supported-report>
|
||||
<report>
|
||||
<CARD:addressbook-multiget/>
|
||||
</report>
|
||||
</supported-report>
|
||||
<supported-report>
|
||||
<report>
|
||||
<sync-collection/>
|
||||
</report>
|
||||
</supported-report>
|
||||
</supported-report-set>
|
||||
<CARD:max-resource-size>10000000</CARD:max-resource-size>
|
||||
<sync-token>github.com/lennart-k/rustical/ns/0</sync-token>
|
||||
<CS:getctag>github.com/lennart-k/rustical/ns/0</CS:getctag>
|
||||
<PUSH:transports>
|
||||
<PUSH:web-push/>
|
||||
</PUSH:transports>
|
||||
<PUSH:topic>[PUSH_TOPIC]</PUSH:topic>
|
||||
<PUSH:supported-triggers>
|
||||
<PUSH:content-update>
|
||||
<depth>1</depth>
|
||||
</PUSH:content-update>
|
||||
<PUSH:property-update>
|
||||
<depth>1</depth>
|
||||
</PUSH:property-update>
|
||||
</PUSH:supported-triggers>
|
||||
<resourcetype>
|
||||
<collection/>
|
||||
<CARD:addressbook/>
|
||||
</resourcetype>
|
||||
<displayname>Contacts</displayname>
|
||||
<current-user-principal>
|
||||
<href>/carddav/principal/user/</href>
|
||||
</current-user-principal>
|
||||
<current-user-privilege-set>
|
||||
<privilege>
|
||||
<all/>
|
||||
</privilege>
|
||||
</current-user-privilege-set>
|
||||
<owner>
|
||||
<href>/carddav/principal/user/</href>
|
||||
</owner>
|
||||
</prop>
|
||||
<status>HTTP/1.1 200 OK</status>
|
||||
</propstat>
|
||||
</response>
|
||||
</multistatus>
|
||||
@@ -0,0 +1,28 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<multistatus xmlns="DAV:" xmlns:CAL="urn:ietf:params:xml:ns:caldav" xmlns:CARD="urn:ietf:params:xml:ns:carddav" xmlns:CS="http://calendarserver.org/ns/" xmlns:PUSH="https://bitfire.at/webdav-push">
|
||||
<response>
|
||||
<href>/carddav/principal/user/contacts</href>
|
||||
<propstat>
|
||||
<prop>
|
||||
<displayname xmlns="DAV:"/>
|
||||
<addressbook-description xmlns="urn:ietf:params:xml:ns:carddav"/>
|
||||
<addressbook-description/>
|
||||
</prop>
|
||||
<status>HTTP/1.1 200 OK</status>
|
||||
</propstat>
|
||||
<propstat>
|
||||
<prop>
|
||||
</prop>
|
||||
<status>HTTP/1.1 404 Not Found</status>
|
||||
</propstat>
|
||||
<propstat>
|
||||
<prop>
|
||||
</prop>
|
||||
<status>HTTP/1.1 409 Conflict</status>
|
||||
</propstat>
|
||||
</response>
|
||||
</multistatus>
|
||||
@@ -0,0 +1,12 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook_import.rs
|
||||
expression: body
|
||||
---
|
||||
BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
FN:John Doe
|
||||
N:Doe;John;;;,
|
||||
BDAY:--0203
|
||||
GENDER:M
|
||||
UID:[UID]
|
||||
END:VCARD
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/addressbook_import.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -0,0 +1,27 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<multistatus xmlns="DAV:" xmlns:CAL="urn:ietf:params:xml:ns:caldav" xmlns:CARD="urn:ietf:params:xml:ns:carddav" xmlns:CS="http://calendarserver.org/ns/" xmlns:PUSH="https://bitfire.at/webdav-push">
|
||||
<response>
|
||||
<href>/carddav/</href>
|
||||
<propstat>
|
||||
<prop>
|
||||
<resourcetype>
|
||||
<collection/>
|
||||
</resourcetype>
|
||||
<displayname>RustiCal DAV root</displayname>
|
||||
<current-user-principal>
|
||||
<href>/carddav/principal/user/</href>
|
||||
</current-user-principal>
|
||||
<current-user-privilege-set>
|
||||
<privilege>
|
||||
<all/>
|
||||
</privilege>
|
||||
</current-user-privilege-set>
|
||||
</prop>
|
||||
<status>HTTP/1.1 200 OK</status>
|
||||
</propstat>
|
||||
</response>
|
||||
</multistatus>
|
||||
@@ -0,0 +1,5 @@
|
||||
---
|
||||
source: tests/integration_tests/carddav/mod.rs
|
||||
expression: body
|
||||
---
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
use crate::{app::make_app, config::NextcloudLoginConfig};
|
||||
use axum::extract::Request;
|
||||
use axum::{body::Body, response::Response};
|
||||
use rstest::rstest;
|
||||
use rustical::{app::make_app, config::NextcloudLoginConfig};
|
||||
use rustical_caldav::CalDavConfig;
|
||||
use rustical_frontend::FrontendConfig;
|
||||
use rustical_store_sqlite::tests::{TestStoreContext, test_store_context};
|
||||
3
tests/run_integration_tests.rs
Normal file
3
tests/run_integration_tests.rs
Normal file
@@ -0,0 +1,3 @@
|
||||
// This file is just an entrypoint for integration_tests
|
||||
// since the test runner only looks for files to run.
|
||||
mod integration_tests;
|
||||
Reference in New Issue
Block a user