You've already forked AstralRinth
* Prorations * Fix pyro integration * set server uuid on creation * fix comp * Fix new charge date, pyro suspend reason * Update server creation endpoint
131 lines
4.0 KiB
Rust
131 lines
4.0 KiB
Rust
use crate::database::models::{DatabaseError, ProductPriceId, UserId, UserSubscriptionId};
|
|
use crate::models::billing::{PriceDuration, SubscriptionMetadata, SubscriptionStatus};
|
|
use chrono::{DateTime, Utc};
|
|
use itertools::Itertools;
|
|
use std::convert::{TryFrom, TryInto};
|
|
|
|
pub struct UserSubscriptionItem {
|
|
pub id: UserSubscriptionId,
|
|
pub user_id: UserId,
|
|
pub price_id: ProductPriceId,
|
|
pub interval: PriceDuration,
|
|
pub created: DateTime<Utc>,
|
|
pub status: SubscriptionStatus,
|
|
pub metadata: Option<SubscriptionMetadata>,
|
|
}
|
|
|
|
struct UserSubscriptionResult {
|
|
id: i64,
|
|
user_id: i64,
|
|
price_id: i64,
|
|
interval: String,
|
|
pub created: DateTime<Utc>,
|
|
pub status: String,
|
|
pub metadata: serde_json::Value,
|
|
}
|
|
|
|
macro_rules! select_user_subscriptions_with_predicate {
|
|
($predicate:tt, $param:ident) => {
|
|
sqlx::query_as!(
|
|
UserSubscriptionResult,
|
|
r#"
|
|
SELECT
|
|
us.id, us.user_id, us.price_id, us.interval, us.created, us.status, us.metadata
|
|
FROM users_subscriptions us
|
|
"#
|
|
+ $predicate,
|
|
$param
|
|
)
|
|
};
|
|
}
|
|
|
|
impl TryFrom<UserSubscriptionResult> for UserSubscriptionItem {
|
|
type Error = serde_json::Error;
|
|
|
|
fn try_from(r: UserSubscriptionResult) -> Result<Self, Self::Error> {
|
|
Ok(UserSubscriptionItem {
|
|
id: UserSubscriptionId(r.id),
|
|
user_id: UserId(r.user_id),
|
|
price_id: ProductPriceId(r.price_id),
|
|
interval: PriceDuration::from_string(&r.interval),
|
|
created: r.created,
|
|
status: SubscriptionStatus::from_string(&r.status),
|
|
metadata: serde_json::from_value(r.metadata)?,
|
|
})
|
|
}
|
|
}
|
|
|
|
impl UserSubscriptionItem {
|
|
pub async fn get(
|
|
id: UserSubscriptionId,
|
|
exec: impl sqlx::Executor<'_, Database = sqlx::Postgres>,
|
|
) -> Result<Option<UserSubscriptionItem>, DatabaseError> {
|
|
Ok(Self::get_many(&[id], exec).await?.into_iter().next())
|
|
}
|
|
|
|
pub async fn get_many(
|
|
ids: &[UserSubscriptionId],
|
|
exec: impl sqlx::Executor<'_, Database = sqlx::Postgres>,
|
|
) -> Result<Vec<UserSubscriptionItem>, DatabaseError> {
|
|
let ids = ids.iter().map(|id| id.0).collect_vec();
|
|
let ids_ref: &[i64] = &ids;
|
|
let results =
|
|
select_user_subscriptions_with_predicate!("WHERE us.id = ANY($1::bigint[])", ids_ref)
|
|
.fetch_all(exec)
|
|
.await?;
|
|
|
|
Ok(results
|
|
.into_iter()
|
|
.map(|r| r.try_into())
|
|
.collect::<Result<Vec<_>, serde_json::Error>>()?)
|
|
}
|
|
|
|
pub async fn get_all_user(
|
|
user_id: UserId,
|
|
exec: impl sqlx::Executor<'_, Database = sqlx::Postgres>,
|
|
) -> Result<Vec<UserSubscriptionItem>, DatabaseError> {
|
|
let user_id = user_id.0;
|
|
let results = select_user_subscriptions_with_predicate!("WHERE us.user_id = $1", user_id)
|
|
.fetch_all(exec)
|
|
.await?;
|
|
|
|
Ok(results
|
|
.into_iter()
|
|
.map(|r| r.try_into())
|
|
.collect::<Result<Vec<_>, serde_json::Error>>()?)
|
|
}
|
|
|
|
pub async fn upsert(
|
|
&self,
|
|
transaction: &mut sqlx::Transaction<'_, sqlx::Postgres>,
|
|
) -> Result<(), DatabaseError> {
|
|
sqlx::query!(
|
|
"
|
|
INSERT INTO users_subscriptions (
|
|
id, user_id, price_id, interval, created, status, metadata
|
|
)
|
|
VALUES (
|
|
$1, $2, $3, $4, $5, $6, $7
|
|
)
|
|
ON CONFLICT (id)
|
|
DO UPDATE
|
|
SET interval = EXCLUDED.interval,
|
|
status = EXCLUDED.status,
|
|
price_id = EXCLUDED.price_id,
|
|
metadata = EXCLUDED.metadata
|
|
",
|
|
self.id.0,
|
|
self.user_id.0,
|
|
self.price_id.0,
|
|
self.interval.as_str(),
|
|
self.created,
|
|
self.status.as_str(),
|
|
serde_json::to_value(&self.metadata)?,
|
|
)
|
|
.execute(&mut **transaction)
|
|
.await?;
|
|
|
|
Ok(())
|
|
}
|
|
}
|