Start using the new sea-orm backed database
This commit is contained in:
parent
19d14737bf
commit
d2385bd6a0
21 changed files with 1131 additions and 4902 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -1159,7 +1159,6 @@ dependencies = [
|
||||||
"scrypt",
|
"scrypt",
|
||||||
"sea-orm",
|
"sea-orm",
|
||||||
"sea-query",
|
"sea-query",
|
||||||
"sea-query-binder",
|
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
"settings",
|
"settings",
|
||||||
|
|
|
@ -37,8 +37,7 @@ rand = "0.8"
|
||||||
reqwest = { version = "0.11", features = ["json"], optional = true }
|
reqwest = { version = "0.11", features = ["json"], optional = true }
|
||||||
scrypt = "0.7"
|
scrypt = "0.7"
|
||||||
sea-orm = { version = "0.10", features = ["sqlx-postgres", "postgres-array", "runtime-tokio-rustls"] }
|
sea-orm = { version = "0.10", features = ["sqlx-postgres", "postgres-array", "runtime-tokio-rustls"] }
|
||||||
sea-query = { version = "0.27", features = ["derive"] }
|
sea-query = "0.27"
|
||||||
sea-query-binder = { version = "0.2", features = ["sqlx-postgres"] }
|
|
||||||
serde = { version = "1.0", features = ["derive", "rc"] }
|
serde = { version = "1.0", features = ["derive", "rc"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
sha-1 = "0.9"
|
sha-1 = "0.9"
|
||||||
|
@ -76,7 +75,6 @@ log = { version = "0.4.16", features = ["kv_unstable_serde"] }
|
||||||
util = { path = "../util" }
|
util = { path = "../util" }
|
||||||
lazy_static = "1.4"
|
lazy_static = "1.4"
|
||||||
sea-orm = { version = "0.10", features = ["sqlx-sqlite"] }
|
sea-orm = { version = "0.10", features = ["sqlx-sqlite"] }
|
||||||
sea-query-binder = { version = "0.2", features = ["sqlx-sqlite"] }
|
|
||||||
serde_json = { version = "1.0", features = ["preserve_order"] }
|
serde_json = { version = "1.0", features = ["preserve_order"] }
|
||||||
sqlx = { version = "0.6", features = ["sqlite"] }
|
sqlx = { version = "0.6", features = ["sqlite"] }
|
||||||
unindent = "0.1"
|
unindent = "0.1"
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
auth,
|
auth,
|
||||||
db::{Invite, NewUserParams, Signup, User, UserId, WaitlistSummary},
|
db::{Invite, NewSignup, NewUserParams, User, UserId, WaitlistSummary},
|
||||||
rpc::{self, ResultExt},
|
rpc::{self, ResultExt},
|
||||||
AppState, Error, Result,
|
AppState, Error, Result,
|
||||||
};
|
};
|
||||||
|
@ -335,7 +335,7 @@ async fn get_user_for_invite_code(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn create_signup(
|
async fn create_signup(
|
||||||
Json(params): Json<Signup>,
|
Json(params): Json<NewSignup>,
|
||||||
Extension(app): Extension<Arc<AppState>>,
|
Extension(app): Extension<Arc<AppState>>,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
app.db.create_signup(params).await?;
|
app.db.create_signup(params).await?;
|
||||||
|
|
|
@ -75,7 +75,7 @@ pub async fn validate_header<B>(mut req: Request<B>, next: Next<B>) -> impl Into
|
||||||
|
|
||||||
const MAX_ACCESS_TOKENS_TO_STORE: usize = 8;
|
const MAX_ACCESS_TOKENS_TO_STORE: usize = 8;
|
||||||
|
|
||||||
pub async fn create_access_token(db: &db::DefaultDb, user_id: UserId) -> Result<String> {
|
pub async fn create_access_token(db: &db::Database, user_id: UserId) -> Result<String> {
|
||||||
let access_token = rpc::auth::random_token();
|
let access_token = rpc::auth::random_token();
|
||||||
let access_token_hash =
|
let access_token_hash =
|
||||||
hash_access_token(&access_token).context("failed to hash access token")?;
|
hash_access_token(&access_token).context("failed to hash access token")?;
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,43 +0,0 @@
|
||||||
pub mod project {
|
|
||||||
use sea_query::Iden;
|
|
||||||
|
|
||||||
#[derive(Iden)]
|
|
||||||
pub enum Definition {
|
|
||||||
#[iden = "projects"]
|
|
||||||
Table,
|
|
||||||
Id,
|
|
||||||
RoomId,
|
|
||||||
HostUserId,
|
|
||||||
HostConnectionId,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod worktree {
|
|
||||||
use sea_query::Iden;
|
|
||||||
|
|
||||||
#[derive(Iden)]
|
|
||||||
pub enum Definition {
|
|
||||||
#[iden = "worktrees"]
|
|
||||||
Table,
|
|
||||||
Id,
|
|
||||||
ProjectId,
|
|
||||||
AbsPath,
|
|
||||||
RootName,
|
|
||||||
Visible,
|
|
||||||
ScanId,
|
|
||||||
IsComplete,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod room_participant {
|
|
||||||
use sea_query::Iden;
|
|
||||||
|
|
||||||
#[derive(Iden)]
|
|
||||||
pub enum Definition {
|
|
||||||
#[iden = "room_participants"]
|
|
||||||
Table,
|
|
||||||
RoomId,
|
|
||||||
UserId,
|
|
||||||
AnsweringConnectionId,
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -27,7 +27,7 @@ pub enum Relation {}
|
||||||
|
|
||||||
impl ActiveModelBehavior for ActiveModel {}
|
impl ActiveModelBehavior for ActiveModel {}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq, FromQueryResult)]
|
#[derive(Debug, PartialEq, Eq, FromQueryResult, Serialize, Deserialize)]
|
||||||
pub struct Invite {
|
pub struct Invite {
|
||||||
pub email_address: String,
|
pub email_address: String,
|
||||||
pub email_confirmation_code: String,
|
pub email_confirmation_code: String,
|
|
@ -6,14 +6,14 @@ macro_rules! test_both_dbs {
|
||||||
($postgres_test_name:ident, $sqlite_test_name:ident, $db:ident, $body:block) => {
|
($postgres_test_name:ident, $sqlite_test_name:ident, $db:ident, $body:block) => {
|
||||||
#[gpui::test]
|
#[gpui::test]
|
||||||
async fn $postgres_test_name() {
|
async fn $postgres_test_name() {
|
||||||
let test_db = PostgresTestDb::new(Deterministic::new(0).build_background());
|
let test_db = TestDb::postgres(Deterministic::new(0).build_background());
|
||||||
let $db = test_db.db();
|
let $db = test_db.db();
|
||||||
$body
|
$body
|
||||||
}
|
}
|
||||||
|
|
||||||
#[gpui::test]
|
#[gpui::test]
|
||||||
async fn $sqlite_test_name() {
|
async fn $sqlite_test_name() {
|
||||||
let test_db = SqliteTestDb::new(Deterministic::new(0).build_background());
|
let test_db = TestDb::sqlite(Deterministic::new(0).build_background());
|
||||||
let $db = test_db.db();
|
let $db = test_db.db();
|
||||||
$body
|
$body
|
||||||
}
|
}
|
||||||
|
@ -26,9 +26,10 @@ test_both_dbs!(
|
||||||
db,
|
db,
|
||||||
{
|
{
|
||||||
let mut user_ids = Vec::new();
|
let mut user_ids = Vec::new();
|
||||||
|
let mut user_metric_ids = Vec::new();
|
||||||
for i in 1..=4 {
|
for i in 1..=4 {
|
||||||
user_ids.push(
|
let user = db
|
||||||
db.create_user(
|
.create_user(
|
||||||
&format!("user{i}@example.com"),
|
&format!("user{i}@example.com"),
|
||||||
false,
|
false,
|
||||||
NewUserParams {
|
NewUserParams {
|
||||||
|
@ -38,9 +39,9 @@ test_both_dbs!(
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap()
|
.unwrap();
|
||||||
.user_id,
|
user_ids.push(user.user_id);
|
||||||
);
|
user_metric_ids.push(user.metrics_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
|
@ -52,6 +53,7 @@ test_both_dbs!(
|
||||||
github_user_id: Some(1),
|
github_user_id: Some(1),
|
||||||
email_address: Some("user1@example.com".to_string()),
|
email_address: Some("user1@example.com".to_string()),
|
||||||
admin: false,
|
admin: false,
|
||||||
|
metrics_id: user_metric_ids[0].parse().unwrap(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
},
|
},
|
||||||
User {
|
User {
|
||||||
|
@ -60,6 +62,7 @@ test_both_dbs!(
|
||||||
github_user_id: Some(2),
|
github_user_id: Some(2),
|
||||||
email_address: Some("user2@example.com".to_string()),
|
email_address: Some("user2@example.com".to_string()),
|
||||||
admin: false,
|
admin: false,
|
||||||
|
metrics_id: user_metric_ids[1].parse().unwrap(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
},
|
},
|
||||||
User {
|
User {
|
||||||
|
@ -68,6 +71,7 @@ test_both_dbs!(
|
||||||
github_user_id: Some(3),
|
github_user_id: Some(3),
|
||||||
email_address: Some("user3@example.com".to_string()),
|
email_address: Some("user3@example.com".to_string()),
|
||||||
admin: false,
|
admin: false,
|
||||||
|
metrics_id: user_metric_ids[2].parse().unwrap(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
},
|
},
|
||||||
User {
|
User {
|
||||||
|
@ -76,6 +80,7 @@ test_both_dbs!(
|
||||||
github_user_id: Some(4),
|
github_user_id: Some(4),
|
||||||
email_address: Some("user4@example.com".to_string()),
|
email_address: Some("user4@example.com".to_string()),
|
||||||
admin: false,
|
admin: false,
|
||||||
|
metrics_id: user_metric_ids[3].parse().unwrap(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -399,14 +404,14 @@ test_both_dbs!(test_metrics_id_postgres, test_metrics_id_sqlite, db, {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_fuzzy_like_string() {
|
fn test_fuzzy_like_string() {
|
||||||
assert_eq!(DefaultDb::fuzzy_like_string("abcd"), "%a%b%c%d%");
|
assert_eq!(Database::fuzzy_like_string("abcd"), "%a%b%c%d%");
|
||||||
assert_eq!(DefaultDb::fuzzy_like_string("x y"), "%x%y%");
|
assert_eq!(Database::fuzzy_like_string("x y"), "%x%y%");
|
||||||
assert_eq!(DefaultDb::fuzzy_like_string(" z "), "%z%");
|
assert_eq!(Database::fuzzy_like_string(" z "), "%z%");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[gpui::test]
|
#[gpui::test]
|
||||||
async fn test_fuzzy_search_users() {
|
async fn test_fuzzy_search_users() {
|
||||||
let test_db = PostgresTestDb::new(build_background_executor());
|
let test_db = TestDb::postgres(build_background_executor());
|
||||||
let db = test_db.db();
|
let db = test_db.db();
|
||||||
for (i, github_login) in [
|
for (i, github_login) in [
|
||||||
"California",
|
"California",
|
||||||
|
@ -442,7 +447,7 @@ async fn test_fuzzy_search_users() {
|
||||||
&["rhode-island", "colorado", "oregon"],
|
&["rhode-island", "colorado", "oregon"],
|
||||||
);
|
);
|
||||||
|
|
||||||
async fn fuzzy_search_user_names(db: &Db<sqlx::Postgres>, query: &str) -> Vec<String> {
|
async fn fuzzy_search_user_names(db: &Database, query: &str) -> Vec<String> {
|
||||||
db.fuzzy_search_users(query, 10)
|
db.fuzzy_search_users(query, 10)
|
||||||
.await
|
.await
|
||||||
.unwrap()
|
.unwrap()
|
||||||
|
@ -454,7 +459,7 @@ async fn test_fuzzy_search_users() {
|
||||||
|
|
||||||
#[gpui::test]
|
#[gpui::test]
|
||||||
async fn test_invite_codes() {
|
async fn test_invite_codes() {
|
||||||
let test_db = PostgresTestDb::new(build_background_executor());
|
let test_db = TestDb::postgres(build_background_executor());
|
||||||
let db = test_db.db();
|
let db = test_db.db();
|
||||||
|
|
||||||
let NewUserResult { user_id: user1, .. } = db
|
let NewUserResult { user_id: user1, .. } = db
|
||||||
|
@ -659,12 +664,12 @@ async fn test_invite_codes() {
|
||||||
|
|
||||||
#[gpui::test]
|
#[gpui::test]
|
||||||
async fn test_signups() {
|
async fn test_signups() {
|
||||||
let test_db = PostgresTestDb::new(build_background_executor());
|
let test_db = TestDb::postgres(build_background_executor());
|
||||||
let db = test_db.db();
|
let db = test_db.db();
|
||||||
|
|
||||||
// people sign up on the waitlist
|
// people sign up on the waitlist
|
||||||
for i in 0..8 {
|
for i in 0..8 {
|
||||||
db.create_signup(Signup {
|
db.create_signup(NewSignup {
|
||||||
email_address: format!("person-{i}@example.com"),
|
email_address: format!("person-{i}@example.com"),
|
||||||
platform_mac: true,
|
platform_mac: true,
|
||||||
platform_linux: i % 2 == 0,
|
platform_linux: i % 2 == 0,
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
use super::UserId;
|
use super::UserId;
|
||||||
use sea_orm::entity::prelude::*;
|
use sea_orm::entity::prelude::*;
|
||||||
|
use serde::Serialize;
|
||||||
|
|
||||||
#[derive(Clone, Debug, Default, PartialEq, Eq, DeriveEntityModel)]
|
#[derive(Clone, Debug, Default, PartialEq, Eq, DeriveEntityModel, Serialize)]
|
||||||
#[sea_orm(table_name = "users")]
|
#[sea_orm(table_name = "users")]
|
||||||
pub struct Model {
|
pub struct Model {
|
||||||
#[sea_orm(primary_key)]
|
#[sea_orm(primary_key)]
|
||||||
|
@ -12,6 +13,7 @@ pub struct Model {
|
||||||
pub admin: bool,
|
pub admin: bool,
|
||||||
pub invite_code: Option<String>,
|
pub invite_code: Option<String>,
|
||||||
pub invite_count: i32,
|
pub invite_count: i32,
|
||||||
|
pub inviter_id: Option<UserId>,
|
||||||
pub connected_once: bool,
|
pub connected_once: bool,
|
||||||
pub metrics_id: Uuid,
|
pub metrics_id: Uuid,
|
||||||
}
|
}
|
File diff suppressed because it is too large
Load diff
|
@ -1,813 +0,0 @@
|
||||||
use super::*;
|
|
||||||
use gpui::executor::{Background, Deterministic};
|
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
macro_rules! test_both_dbs {
|
|
||||||
($postgres_test_name:ident, $sqlite_test_name:ident, $db:ident, $body:block) => {
|
|
||||||
#[gpui::test]
|
|
||||||
async fn $postgres_test_name() {
|
|
||||||
let test_db = TestDb::postgres(Deterministic::new(0).build_background());
|
|
||||||
let $db = test_db.db();
|
|
||||||
$body
|
|
||||||
}
|
|
||||||
|
|
||||||
#[gpui::test]
|
|
||||||
async fn $sqlite_test_name() {
|
|
||||||
let test_db = TestDb::sqlite(Deterministic::new(0).build_background());
|
|
||||||
let $db = test_db.db();
|
|
||||||
$body
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
test_both_dbs!(
|
|
||||||
test_get_users_by_ids_postgres,
|
|
||||||
test_get_users_by_ids_sqlite,
|
|
||||||
db,
|
|
||||||
{
|
|
||||||
let mut user_ids = Vec::new();
|
|
||||||
let mut user_metric_ids = Vec::new();
|
|
||||||
for i in 1..=4 {
|
|
||||||
let user = db
|
|
||||||
.create_user(
|
|
||||||
&format!("user{i}@example.com"),
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: format!("user{i}"),
|
|
||||||
github_user_id: i,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
user_ids.push(user.user_id);
|
|
||||||
user_metric_ids.push(user.metrics_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
db.get_users_by_ids(user_ids.clone()).await.unwrap(),
|
|
||||||
vec![
|
|
||||||
User {
|
|
||||||
id: user_ids[0],
|
|
||||||
github_login: "user1".to_string(),
|
|
||||||
github_user_id: Some(1),
|
|
||||||
email_address: Some("user1@example.com".to_string()),
|
|
||||||
admin: false,
|
|
||||||
metrics_id: user_metric_ids[0].parse().unwrap(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
User {
|
|
||||||
id: user_ids[1],
|
|
||||||
github_login: "user2".to_string(),
|
|
||||||
github_user_id: Some(2),
|
|
||||||
email_address: Some("user2@example.com".to_string()),
|
|
||||||
admin: false,
|
|
||||||
metrics_id: user_metric_ids[1].parse().unwrap(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
User {
|
|
||||||
id: user_ids[2],
|
|
||||||
github_login: "user3".to_string(),
|
|
||||||
github_user_id: Some(3),
|
|
||||||
email_address: Some("user3@example.com".to_string()),
|
|
||||||
admin: false,
|
|
||||||
metrics_id: user_metric_ids[2].parse().unwrap(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
User {
|
|
||||||
id: user_ids[3],
|
|
||||||
github_login: "user4".to_string(),
|
|
||||||
github_user_id: Some(4),
|
|
||||||
email_address: Some("user4@example.com".to_string()),
|
|
||||||
admin: false,
|
|
||||||
metrics_id: user_metric_ids[3].parse().unwrap(),
|
|
||||||
..Default::default()
|
|
||||||
}
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
test_both_dbs!(
|
|
||||||
test_get_user_by_github_account_postgres,
|
|
||||||
test_get_user_by_github_account_sqlite,
|
|
||||||
db,
|
|
||||||
{
|
|
||||||
let user_id1 = db
|
|
||||||
.create_user(
|
|
||||||
"user1@example.com",
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "login1".into(),
|
|
||||||
github_user_id: 101,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.user_id;
|
|
||||||
let user_id2 = db
|
|
||||||
.create_user(
|
|
||||||
"user2@example.com",
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "login2".into(),
|
|
||||||
github_user_id: 102,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.user_id;
|
|
||||||
|
|
||||||
let user = db
|
|
||||||
.get_user_by_github_account("login1", None)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.unwrap();
|
|
||||||
assert_eq!(user.id, user_id1);
|
|
||||||
assert_eq!(&user.github_login, "login1");
|
|
||||||
assert_eq!(user.github_user_id, Some(101));
|
|
||||||
|
|
||||||
assert!(db
|
|
||||||
.get_user_by_github_account("non-existent-login", None)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.is_none());
|
|
||||||
|
|
||||||
let user = db
|
|
||||||
.get_user_by_github_account("the-new-login2", Some(102))
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.unwrap();
|
|
||||||
assert_eq!(user.id, user_id2);
|
|
||||||
assert_eq!(&user.github_login, "the-new-login2");
|
|
||||||
assert_eq!(user.github_user_id, Some(102));
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
test_both_dbs!(
|
|
||||||
test_create_access_tokens_postgres,
|
|
||||||
test_create_access_tokens_sqlite,
|
|
||||||
db,
|
|
||||||
{
|
|
||||||
let user = db
|
|
||||||
.create_user(
|
|
||||||
"u1@example.com",
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "u1".into(),
|
|
||||||
github_user_id: 1,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.user_id;
|
|
||||||
|
|
||||||
db.create_access_token_hash(user, "h1", 3).await.unwrap();
|
|
||||||
db.create_access_token_hash(user, "h2", 3).await.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_access_token_hashes(user).await.unwrap(),
|
|
||||||
&["h2".to_string(), "h1".to_string()]
|
|
||||||
);
|
|
||||||
|
|
||||||
db.create_access_token_hash(user, "h3", 3).await.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_access_token_hashes(user).await.unwrap(),
|
|
||||||
&["h3".to_string(), "h2".to_string(), "h1".to_string(),]
|
|
||||||
);
|
|
||||||
|
|
||||||
db.create_access_token_hash(user, "h4", 3).await.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_access_token_hashes(user).await.unwrap(),
|
|
||||||
&["h4".to_string(), "h3".to_string(), "h2".to_string(),]
|
|
||||||
);
|
|
||||||
|
|
||||||
db.create_access_token_hash(user, "h5", 3).await.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_access_token_hashes(user).await.unwrap(),
|
|
||||||
&["h5".to_string(), "h4".to_string(), "h3".to_string()]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
test_both_dbs!(test_add_contacts_postgres, test_add_contacts_sqlite, db, {
|
|
||||||
let mut user_ids = Vec::new();
|
|
||||||
for i in 0..3 {
|
|
||||||
user_ids.push(
|
|
||||||
db.create_user(
|
|
||||||
&format!("user{i}@example.com"),
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: format!("user{i}"),
|
|
||||||
github_user_id: i,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.user_id,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
let user_1 = user_ids[0];
|
|
||||||
let user_2 = user_ids[1];
|
|
||||||
let user_3 = user_ids[2];
|
|
||||||
|
|
||||||
// User starts with no contacts
|
|
||||||
assert_eq!(db.get_contacts(user_1).await.unwrap(), &[]);
|
|
||||||
|
|
||||||
// User requests a contact. Both users see the pending request.
|
|
||||||
db.send_contact_request(user_1, user_2).await.unwrap();
|
|
||||||
assert!(!db.has_contact(user_1, user_2).await.unwrap());
|
|
||||||
assert!(!db.has_contact(user_2, user_1).await.unwrap());
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_1).await.unwrap(),
|
|
||||||
&[Contact::Outgoing { user_id: user_2 }],
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_2).await.unwrap(),
|
|
||||||
&[Contact::Incoming {
|
|
||||||
user_id: user_1,
|
|
||||||
should_notify: true
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
|
|
||||||
// User 2 dismisses the contact request notification without accepting or rejecting.
|
|
||||||
// We shouldn't notify them again.
|
|
||||||
db.dismiss_contact_notification(user_1, user_2)
|
|
||||||
.await
|
|
||||||
.unwrap_err();
|
|
||||||
db.dismiss_contact_notification(user_2, user_1)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_2).await.unwrap(),
|
|
||||||
&[Contact::Incoming {
|
|
||||||
user_id: user_1,
|
|
||||||
should_notify: false
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
|
|
||||||
// User can't accept their own contact request
|
|
||||||
db.respond_to_contact_request(user_1, user_2, true)
|
|
||||||
.await
|
|
||||||
.unwrap_err();
|
|
||||||
|
|
||||||
// User accepts a contact request. Both users see the contact.
|
|
||||||
db.respond_to_contact_request(user_2, user_1, true)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_1).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_2,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
}],
|
|
||||||
);
|
|
||||||
assert!(db.has_contact(user_1, user_2).await.unwrap());
|
|
||||||
assert!(db.has_contact(user_2, user_1).await.unwrap());
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_2).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Users cannot re-request existing contacts.
|
|
||||||
db.send_contact_request(user_1, user_2).await.unwrap_err();
|
|
||||||
db.send_contact_request(user_2, user_1).await.unwrap_err();
|
|
||||||
|
|
||||||
// Users can't dismiss notifications of them accepting other users' requests.
|
|
||||||
db.dismiss_contact_notification(user_2, user_1)
|
|
||||||
.await
|
|
||||||
.unwrap_err();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_1).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_2,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Users can dismiss notifications of other users accepting their requests.
|
|
||||||
db.dismiss_contact_notification(user_1, user_2)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_1).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_2,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Users send each other concurrent contact requests and
|
|
||||||
// see that they are immediately accepted.
|
|
||||||
db.send_contact_request(user_1, user_3).await.unwrap();
|
|
||||||
db.send_contact_request(user_3, user_1).await.unwrap();
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_1).await.unwrap(),
|
|
||||||
&[
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user_2,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
},
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user_3,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}
|
|
||||||
]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_3).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}],
|
|
||||||
);
|
|
||||||
|
|
||||||
// User declines a contact request. Both users see that it is gone.
|
|
||||||
db.send_contact_request(user_2, user_3).await.unwrap();
|
|
||||||
db.respond_to_contact_request(user_3, user_2, false)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
assert!(!db.has_contact(user_2, user_3).await.unwrap());
|
|
||||||
assert!(!db.has_contact(user_3, user_2).await.unwrap());
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_2).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user_3).await.unwrap(),
|
|
||||||
&[Contact::Accepted {
|
|
||||||
user_id: user_1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}],
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
test_both_dbs!(test_metrics_id_postgres, test_metrics_id_sqlite, db, {
|
|
||||||
let NewUserResult {
|
|
||||||
user_id: user1,
|
|
||||||
metrics_id: metrics_id1,
|
|
||||||
..
|
|
||||||
} = db
|
|
||||||
.create_user(
|
|
||||||
"person1@example.com",
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "person1".into(),
|
|
||||||
github_user_id: 101,
|
|
||||||
invite_count: 5,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
let NewUserResult {
|
|
||||||
user_id: user2,
|
|
||||||
metrics_id: metrics_id2,
|
|
||||||
..
|
|
||||||
} = db
|
|
||||||
.create_user(
|
|
||||||
"person2@example.com",
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "person2".into(),
|
|
||||||
github_user_id: 102,
|
|
||||||
invite_count: 5,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
assert_eq!(db.get_user_metrics_id(user1).await.unwrap(), metrics_id1);
|
|
||||||
assert_eq!(db.get_user_metrics_id(user2).await.unwrap(), metrics_id2);
|
|
||||||
assert_eq!(metrics_id1.len(), 36);
|
|
||||||
assert_eq!(metrics_id2.len(), 36);
|
|
||||||
assert_ne!(metrics_id1, metrics_id2);
|
|
||||||
});
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_fuzzy_like_string() {
|
|
||||||
assert_eq!(Database::fuzzy_like_string("abcd"), "%a%b%c%d%");
|
|
||||||
assert_eq!(Database::fuzzy_like_string("x y"), "%x%y%");
|
|
||||||
assert_eq!(Database::fuzzy_like_string(" z "), "%z%");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[gpui::test]
|
|
||||||
async fn test_fuzzy_search_users() {
|
|
||||||
let test_db = TestDb::postgres(build_background_executor());
|
|
||||||
let db = test_db.db();
|
|
||||||
for (i, github_login) in [
|
|
||||||
"California",
|
|
||||||
"colorado",
|
|
||||||
"oregon",
|
|
||||||
"washington",
|
|
||||||
"florida",
|
|
||||||
"delaware",
|
|
||||||
"rhode-island",
|
|
||||||
]
|
|
||||||
.into_iter()
|
|
||||||
.enumerate()
|
|
||||||
{
|
|
||||||
db.create_user(
|
|
||||||
&format!("{github_login}@example.com"),
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: github_login.into(),
|
|
||||||
github_user_id: i as i32,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
fuzzy_search_user_names(db, "clr").await,
|
|
||||||
&["colorado", "California"]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
fuzzy_search_user_names(db, "ro").await,
|
|
||||||
&["rhode-island", "colorado", "oregon"],
|
|
||||||
);
|
|
||||||
|
|
||||||
async fn fuzzy_search_user_names(db: &Database, query: &str) -> Vec<String> {
|
|
||||||
db.fuzzy_search_users(query, 10)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.into_iter()
|
|
||||||
.map(|user| user.github_login)
|
|
||||||
.collect::<Vec<_>>()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[gpui::test]
|
|
||||||
async fn test_invite_codes() {
|
|
||||||
let test_db = TestDb::postgres(build_background_executor());
|
|
||||||
let db = test_db.db();
|
|
||||||
|
|
||||||
let NewUserResult { user_id: user1, .. } = db
|
|
||||||
.create_user(
|
|
||||||
"user1@example.com",
|
|
||||||
false,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "user1".into(),
|
|
||||||
github_user_id: 0,
|
|
||||||
invite_count: 0,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
// Initially, user 1 has no invite code
|
|
||||||
assert_eq!(db.get_invite_code_for_user(user1).await.unwrap(), None);
|
|
||||||
|
|
||||||
// Setting invite count to 0 when no code is assigned does not assign a new code
|
|
||||||
db.set_invite_count_for_user(user1, 0).await.unwrap();
|
|
||||||
assert!(db.get_invite_code_for_user(user1).await.unwrap().is_none());
|
|
||||||
|
|
||||||
// User 1 creates an invite code that can be used twice.
|
|
||||||
db.set_invite_count_for_user(user1, 2).await.unwrap();
|
|
||||||
let (invite_code, invite_count) = db.get_invite_code_for_user(user1).await.unwrap().unwrap();
|
|
||||||
assert_eq!(invite_count, 2);
|
|
||||||
|
|
||||||
// User 2 redeems the invite code and becomes a contact of user 1.
|
|
||||||
let user2_invite = db
|
|
||||||
.create_invite_from_code(&invite_code, "user2@example.com", Some("user-2-device-id"))
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
let NewUserResult {
|
|
||||||
user_id: user2,
|
|
||||||
inviting_user_id,
|
|
||||||
signup_device_id,
|
|
||||||
metrics_id,
|
|
||||||
} = db
|
|
||||||
.create_user_from_invite(
|
|
||||||
&user2_invite,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "user2".into(),
|
|
||||||
github_user_id: 2,
|
|
||||||
invite_count: 7,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.unwrap();
|
|
||||||
let (_, invite_count) = db.get_invite_code_for_user(user1).await.unwrap().unwrap();
|
|
||||||
assert_eq!(invite_count, 1);
|
|
||||||
assert_eq!(inviting_user_id, Some(user1));
|
|
||||||
assert_eq!(signup_device_id.unwrap(), "user-2-device-id");
|
|
||||||
assert_eq!(db.get_user_metrics_id(user2).await.unwrap(), metrics_id);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user1).await.unwrap(),
|
|
||||||
[Contact::Accepted {
|
|
||||||
user_id: user2,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user2).await.unwrap(),
|
|
||||||
[Contact::Accepted {
|
|
||||||
user_id: user1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_invite_code_for_user(user2).await.unwrap().unwrap().1,
|
|
||||||
7
|
|
||||||
);
|
|
||||||
|
|
||||||
// User 3 redeems the invite code and becomes a contact of user 1.
|
|
||||||
let user3_invite = db
|
|
||||||
.create_invite_from_code(&invite_code, "user3@example.com", None)
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
let NewUserResult {
|
|
||||||
user_id: user3,
|
|
||||||
inviting_user_id,
|
|
||||||
signup_device_id,
|
|
||||||
..
|
|
||||||
} = db
|
|
||||||
.create_user_from_invite(
|
|
||||||
&user3_invite,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "user-3".into(),
|
|
||||||
github_user_id: 3,
|
|
||||||
invite_count: 3,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.unwrap();
|
|
||||||
let (_, invite_count) = db.get_invite_code_for_user(user1).await.unwrap().unwrap();
|
|
||||||
assert_eq!(invite_count, 0);
|
|
||||||
assert_eq!(inviting_user_id, Some(user1));
|
|
||||||
assert!(signup_device_id.is_none());
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user1).await.unwrap(),
|
|
||||||
[
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user2,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
},
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user3,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
}
|
|
||||||
]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user3).await.unwrap(),
|
|
||||||
[Contact::Accepted {
|
|
||||||
user_id: user1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_invite_code_for_user(user3).await.unwrap().unwrap().1,
|
|
||||||
3
|
|
||||||
);
|
|
||||||
|
|
||||||
// Trying to reedem the code for the third time results in an error.
|
|
||||||
db.create_invite_from_code(&invite_code, "user4@example.com", Some("user-4-device-id"))
|
|
||||||
.await
|
|
||||||
.unwrap_err();
|
|
||||||
|
|
||||||
// Invite count can be updated after the code has been created.
|
|
||||||
db.set_invite_count_for_user(user1, 2).await.unwrap();
|
|
||||||
let (latest_code, invite_count) = db.get_invite_code_for_user(user1).await.unwrap().unwrap();
|
|
||||||
assert_eq!(latest_code, invite_code); // Invite code doesn't change when we increment above 0
|
|
||||||
assert_eq!(invite_count, 2);
|
|
||||||
|
|
||||||
// User 4 can now redeem the invite code and becomes a contact of user 1.
|
|
||||||
let user4_invite = db
|
|
||||||
.create_invite_from_code(&invite_code, "user4@example.com", Some("user-4-device-id"))
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
let user4 = db
|
|
||||||
.create_user_from_invite(
|
|
||||||
&user4_invite,
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "user-4".into(),
|
|
||||||
github_user_id: 4,
|
|
||||||
invite_count: 5,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.unwrap()
|
|
||||||
.user_id;
|
|
||||||
|
|
||||||
let (_, invite_count) = db.get_invite_code_for_user(user1).await.unwrap().unwrap();
|
|
||||||
assert_eq!(invite_count, 1);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user1).await.unwrap(),
|
|
||||||
[
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user2,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
},
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user3,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
},
|
|
||||||
Contact::Accepted {
|
|
||||||
user_id: user4,
|
|
||||||
should_notify: true,
|
|
||||||
busy: false,
|
|
||||||
}
|
|
||||||
]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_contacts(user4).await.unwrap(),
|
|
||||||
[Contact::Accepted {
|
|
||||||
user_id: user1,
|
|
||||||
should_notify: false,
|
|
||||||
busy: false,
|
|
||||||
}]
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
db.get_invite_code_for_user(user4).await.unwrap().unwrap().1,
|
|
||||||
5
|
|
||||||
);
|
|
||||||
|
|
||||||
// An existing user cannot redeem invite codes.
|
|
||||||
db.create_invite_from_code(&invite_code, "user2@example.com", Some("user-2-device-id"))
|
|
||||||
.await
|
|
||||||
.unwrap_err();
|
|
||||||
let (_, invite_count) = db.get_invite_code_for_user(user1).await.unwrap().unwrap();
|
|
||||||
assert_eq!(invite_count, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[gpui::test]
|
|
||||||
async fn test_signups() {
|
|
||||||
let test_db = TestDb::postgres(build_background_executor());
|
|
||||||
let db = test_db.db();
|
|
||||||
|
|
||||||
// people sign up on the waitlist
|
|
||||||
for i in 0..8 {
|
|
||||||
db.create_signup(NewSignup {
|
|
||||||
email_address: format!("person-{i}@example.com"),
|
|
||||||
platform_mac: true,
|
|
||||||
platform_linux: i % 2 == 0,
|
|
||||||
platform_windows: i % 4 == 0,
|
|
||||||
editor_features: vec!["speed".into()],
|
|
||||||
programming_languages: vec!["rust".into(), "c".into()],
|
|
||||||
device_id: Some(format!("device_id_{i}")),
|
|
||||||
})
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
db.get_waitlist_summary().await.unwrap(),
|
|
||||||
WaitlistSummary {
|
|
||||||
count: 8,
|
|
||||||
mac_count: 8,
|
|
||||||
linux_count: 4,
|
|
||||||
windows_count: 2,
|
|
||||||
unknown_count: 0,
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
// retrieve the next batch of signup emails to send
|
|
||||||
let signups_batch1 = db.get_unsent_invites(3).await.unwrap();
|
|
||||||
let addresses = signups_batch1
|
|
||||||
.iter()
|
|
||||||
.map(|s| &s.email_address)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
assert_eq!(
|
|
||||||
addresses,
|
|
||||||
&[
|
|
||||||
"person-0@example.com",
|
|
||||||
"person-1@example.com",
|
|
||||||
"person-2@example.com"
|
|
||||||
]
|
|
||||||
);
|
|
||||||
assert_ne!(
|
|
||||||
signups_batch1[0].email_confirmation_code,
|
|
||||||
signups_batch1[1].email_confirmation_code
|
|
||||||
);
|
|
||||||
|
|
||||||
// the waitlist isn't updated until we record that the emails
|
|
||||||
// were successfully sent.
|
|
||||||
let signups_batch = db.get_unsent_invites(3).await.unwrap();
|
|
||||||
assert_eq!(signups_batch, signups_batch1);
|
|
||||||
|
|
||||||
// once the emails go out, we can retrieve the next batch
|
|
||||||
// of signups.
|
|
||||||
db.record_sent_invites(&signups_batch1).await.unwrap();
|
|
||||||
let signups_batch2 = db.get_unsent_invites(3).await.unwrap();
|
|
||||||
let addresses = signups_batch2
|
|
||||||
.iter()
|
|
||||||
.map(|s| &s.email_address)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
assert_eq!(
|
|
||||||
addresses,
|
|
||||||
&[
|
|
||||||
"person-3@example.com",
|
|
||||||
"person-4@example.com",
|
|
||||||
"person-5@example.com"
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
// the sent invites are excluded from the summary.
|
|
||||||
assert_eq!(
|
|
||||||
db.get_waitlist_summary().await.unwrap(),
|
|
||||||
WaitlistSummary {
|
|
||||||
count: 5,
|
|
||||||
mac_count: 5,
|
|
||||||
linux_count: 2,
|
|
||||||
windows_count: 1,
|
|
||||||
unknown_count: 0,
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
// user completes the signup process by providing their
|
|
||||||
// github account.
|
|
||||||
let NewUserResult {
|
|
||||||
user_id,
|
|
||||||
inviting_user_id,
|
|
||||||
signup_device_id,
|
|
||||||
..
|
|
||||||
} = db
|
|
||||||
.create_user_from_invite(
|
|
||||||
&Invite {
|
|
||||||
email_address: signups_batch1[0].email_address.clone(),
|
|
||||||
email_confirmation_code: signups_batch1[0].email_confirmation_code.clone(),
|
|
||||||
},
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "person-0".into(),
|
|
||||||
github_user_id: 0,
|
|
||||||
invite_count: 5,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.unwrap();
|
|
||||||
let user = db.get_user_by_id(user_id).await.unwrap().unwrap();
|
|
||||||
assert!(inviting_user_id.is_none());
|
|
||||||
assert_eq!(user.github_login, "person-0");
|
|
||||||
assert_eq!(user.email_address.as_deref(), Some("person-0@example.com"));
|
|
||||||
assert_eq!(user.invite_count, 5);
|
|
||||||
assert_eq!(signup_device_id.unwrap(), "device_id_0");
|
|
||||||
|
|
||||||
// cannot redeem the same signup again.
|
|
||||||
assert!(db
|
|
||||||
.create_user_from_invite(
|
|
||||||
&Invite {
|
|
||||||
email_address: signups_batch1[0].email_address.clone(),
|
|
||||||
email_confirmation_code: signups_batch1[0].email_confirmation_code.clone(),
|
|
||||||
},
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "some-other-github_account".into(),
|
|
||||||
github_user_id: 1,
|
|
||||||
invite_count: 5,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap()
|
|
||||||
.is_none());
|
|
||||||
|
|
||||||
// cannot redeem a signup with the wrong confirmation code.
|
|
||||||
db.create_user_from_invite(
|
|
||||||
&Invite {
|
|
||||||
email_address: signups_batch1[1].email_address.clone(),
|
|
||||||
email_confirmation_code: "the-wrong-code".to_string(),
|
|
||||||
},
|
|
||||||
NewUserParams {
|
|
||||||
github_login: "person-1".into(),
|
|
||||||
github_user_id: 2,
|
|
||||||
invite_count: 5,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.unwrap_err();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_background_executor() -> Arc<Background> {
|
|
||||||
Deterministic::new(0).build_background()
|
|
||||||
}
|
|
|
@ -1,5 +1,5 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
db::{self, NewUserParams, SqliteTestDb as TestDb, UserId},
|
db::{self, NewUserParams, TestDb, UserId},
|
||||||
rpc::{Executor, Server},
|
rpc::{Executor, Server},
|
||||||
AppState,
|
AppState,
|
||||||
};
|
};
|
||||||
|
@ -5665,7 +5665,7 @@ impl TestServer {
|
||||||
async fn start(background: Arc<executor::Background>) -> Self {
|
async fn start(background: Arc<executor::Background>) -> Self {
|
||||||
static NEXT_LIVE_KIT_SERVER_ID: AtomicUsize = AtomicUsize::new(0);
|
static NEXT_LIVE_KIT_SERVER_ID: AtomicUsize = AtomicUsize::new(0);
|
||||||
|
|
||||||
let test_db = TestDb::new(background.clone());
|
let test_db = TestDb::sqlite(background.clone());
|
||||||
let live_kit_server_id = NEXT_LIVE_KIT_SERVER_ID.fetch_add(1, SeqCst);
|
let live_kit_server_id = NEXT_LIVE_KIT_SERVER_ID.fetch_add(1, SeqCst);
|
||||||
let live_kit_server = live_kit_client::TestServer::create(
|
let live_kit_server = live_kit_client::TestServer::create(
|
||||||
format!("http://livekit.{}.test", live_kit_server_id),
|
format!("http://livekit.{}.test", live_kit_server_id),
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
mod api;
|
mod api;
|
||||||
mod auth;
|
mod auth;
|
||||||
mod db;
|
mod db;
|
||||||
mod db2;
|
|
||||||
mod env;
|
mod env;
|
||||||
mod rpc;
|
mod rpc;
|
||||||
|
|
||||||
|
@ -11,7 +10,7 @@ mod integration_tests;
|
||||||
use anyhow::anyhow;
|
use anyhow::anyhow;
|
||||||
use axum::{routing::get, Router};
|
use axum::{routing::get, Router};
|
||||||
use collab::{Error, Result};
|
use collab::{Error, Result};
|
||||||
use db::DefaultDb as Db;
|
use db::Database;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::{
|
use std::{
|
||||||
env::args,
|
env::args,
|
||||||
|
@ -45,14 +44,16 @@ pub struct MigrateConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct AppState {
|
pub struct AppState {
|
||||||
db: Arc<Db>,
|
db: Arc<Database>,
|
||||||
live_kit_client: Option<Arc<dyn live_kit_server::api::Client>>,
|
live_kit_client: Option<Arc<dyn live_kit_server::api::Client>>,
|
||||||
config: Config,
|
config: Config,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AppState {
|
impl AppState {
|
||||||
async fn new(config: Config) -> Result<Arc<Self>> {
|
async fn new(config: Config) -> Result<Arc<Self>> {
|
||||||
let db = Db::new(&config.database_url, 5).await?;
|
let mut db_options = db::ConnectOptions::new(config.database_url.clone());
|
||||||
|
db_options.max_connections(5);
|
||||||
|
let db = Database::new(db_options).await?;
|
||||||
let live_kit_client = if let Some(((server, key), secret)) = config
|
let live_kit_client = if let Some(((server, key), secret)) = config
|
||||||
.live_kit_server
|
.live_kit_server
|
||||||
.as_ref()
|
.as_ref()
|
||||||
|
@ -92,7 +93,9 @@ async fn main() -> Result<()> {
|
||||||
}
|
}
|
||||||
Some("migrate") => {
|
Some("migrate") => {
|
||||||
let config = envy::from_env::<MigrateConfig>().expect("error loading config");
|
let config = envy::from_env::<MigrateConfig>().expect("error loading config");
|
||||||
let db = Db::new(&config.database_url, 5).await?;
|
let mut db_options = db::ConnectOptions::new(config.database_url.clone());
|
||||||
|
db_options.max_connections(5);
|
||||||
|
let db = Database::new(db_options).await?;
|
||||||
|
|
||||||
let migrations_path = config
|
let migrations_path = config
|
||||||
.migrations_path
|
.migrations_path
|
||||||
|
|
|
@ -2,7 +2,7 @@ mod connection_pool;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
auth,
|
auth,
|
||||||
db::{self, DefaultDb, ProjectId, RoomId, User, UserId},
|
db::{self, Database, ProjectId, RoomId, User, UserId},
|
||||||
AppState, Result,
|
AppState, Result,
|
||||||
};
|
};
|
||||||
use anyhow::anyhow;
|
use anyhow::anyhow;
|
||||||
|
@ -128,10 +128,10 @@ impl fmt::Debug for Session {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct DbHandle(Arc<DefaultDb>);
|
struct DbHandle(Arc<Database>);
|
||||||
|
|
||||||
impl Deref for DbHandle {
|
impl Deref for DbHandle {
|
||||||
type Target = DefaultDb;
|
type Target = Database;
|
||||||
|
|
||||||
fn deref(&self) -> &Self::Target {
|
fn deref(&self) -> &Self::Target {
|
||||||
self.0.as_ref()
|
self.0.as_ref()
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue