2022-12-31 00:59:14 +11:00
|
|
|
use crate::message_handler::user_commands::parsing::parse_offset;
|
2023-06-07 22:38:46 +10:00
|
|
|
use crate::message_handler::ListenerSession;
|
2023-01-02 13:25:05 +11:00
|
|
|
use crate::models::{
|
2023-06-07 22:38:46 +10:00
|
|
|
consent::{Consent, ConsentType},
|
|
|
|
corp::{Corp, CorpCommType, CorpId, CorpMembership},
|
|
|
|
item::{Item, LocationActionType},
|
2023-01-02 13:25:05 +11:00
|
|
|
session::Session,
|
2023-03-13 15:23:07 +11:00
|
|
|
task::{Task, TaskParse},
|
2023-06-07 22:38:46 +10:00
|
|
|
user::User,
|
2023-01-02 13:25:05 +11:00
|
|
|
};
|
2023-06-07 22:38:46 +10:00
|
|
|
use crate::static_content::{possession_type::PossessionType, room::Direction};
|
|
|
|
use crate::DResult;
|
2023-01-28 23:00:53 +11:00
|
|
|
use chrono::{DateTime, Utc};
|
2023-06-07 22:38:46 +10:00
|
|
|
use deadpool_postgres::{Manager, ManagerConfig, Object, Pool, RecyclingMethod, Transaction};
|
|
|
|
use futures::FutureExt;
|
2023-02-19 14:03:15 +11:00
|
|
|
#[cfg(test)]
|
|
|
|
use mockall::automock;
|
2023-06-07 22:38:46 +10:00
|
|
|
use ouroboros::self_referencing;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use serde_json::{self, Value};
|
|
|
|
use std::collections::BTreeSet;
|
|
|
|
use std::error::Error;
|
|
|
|
use std::str::FromStr;
|
|
|
|
use std::sync::Arc;
|
|
|
|
use tokio_postgres::types::ToSql;
|
|
|
|
use tokio_postgres::NoTls;
|
|
|
|
use tokio_postgres::{config::Config as PgConfig, row::Row};
|
|
|
|
use uuid::Uuid;
|
2022-12-14 23:48:00 +11:00
|
|
|
|
2022-12-18 23:44:04 +11:00
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct DBPool {
|
2023-06-07 22:38:46 +10:00
|
|
|
pool: Pool,
|
2022-12-18 02:30:30 +11:00
|
|
|
}
|
|
|
|
|
2022-12-25 00:25:52 +11:00
|
|
|
#[self_referencing]
|
|
|
|
pub struct DBTrans {
|
|
|
|
conn: Object,
|
|
|
|
#[borrows(mut conn)]
|
|
|
|
#[covariant]
|
2023-06-07 22:38:46 +10:00
|
|
|
pub trans: Option<Transaction<'this>>,
|
2022-12-25 00:25:52 +11:00
|
|
|
}
|
|
|
|
|
2022-12-23 23:31:49 +11:00
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct SendqueueItem {
|
|
|
|
pub item: i64,
|
|
|
|
pub session: ListenerSession,
|
2023-06-07 22:38:46 +10:00
|
|
|
pub message: Option<String>,
|
2022-12-23 23:31:49 +11:00
|
|
|
}
|
|
|
|
impl From<Row> for SendqueueItem {
|
|
|
|
fn from(row: Row) -> Self {
|
|
|
|
SendqueueItem {
|
|
|
|
item: row.get("item"),
|
|
|
|
session: ListenerSession {
|
|
|
|
session: row.get("session"),
|
2023-06-07 22:38:46 +10:00
|
|
|
listener: row.get("listener"),
|
2022-12-23 23:31:49 +11:00
|
|
|
},
|
2023-06-07 22:38:46 +10:00
|
|
|
message: row.get("message"),
|
2022-12-23 23:31:49 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-28 23:00:53 +11:00
|
|
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
|
|
|
pub struct OnlineInfo {
|
|
|
|
pub username: String,
|
2023-03-19 15:41:48 +11:00
|
|
|
pub corp: Option<String>,
|
2023-04-01 23:31:42 +11:00
|
|
|
pub time: Option<DateTime<Utc>>,
|
2023-01-28 23:00:53 +11:00
|
|
|
}
|
|
|
|
|
2023-02-03 23:26:24 +11:00
|
|
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
|
|
|
pub struct LocationStats {
|
|
|
|
pub total_count: u64,
|
|
|
|
pub total_weight: u64,
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
#[cfg_attr(test, allow(dead_code))]
|
2022-12-23 21:37:28 +11:00
|
|
|
impl DBPool {
|
2022-12-24 21:16:23 +11:00
|
|
|
pub async fn record_listener_ping(self: &DBPool, listener: Uuid) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.get_conn()
|
|
|
|
.await?
|
|
|
|
.execute(
|
|
|
|
"INSERT INTO listeners (listener, last_seen) \
|
2022-12-23 21:37:28 +11:00
|
|
|
VALUES ($1, NOW()) \
|
|
|
|
ON CONFLICT (listener) \
|
2023-06-07 22:38:46 +10:00
|
|
|
DO UPDATE SET last_seen = EXCLUDED.last_seen",
|
|
|
|
&[&listener],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-23 21:37:28 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2022-12-18 23:44:04 +11:00
|
|
|
|
2022-12-24 21:16:23 +11:00
|
|
|
pub async fn get_dead_listeners(self: &Self) -> DResult<Vec<Uuid>> {
|
2023-06-07 22:38:46 +10:00
|
|
|
Ok(self
|
|
|
|
.get_conn()
|
|
|
|
.await?
|
|
|
|
.query(
|
|
|
|
"SELECT listener FROM listeners WHERE last_seen < NOW() - \
|
|
|
|
INTERVAL '2 minutes'",
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.map(|r| r.get(0))
|
|
|
|
.collect())
|
2022-12-23 21:37:28 +11:00
|
|
|
}
|
2022-12-18 23:44:04 +11:00
|
|
|
|
2022-12-24 21:16:23 +11:00
|
|
|
pub async fn cleanup_listener(self: &Self, listener: Uuid) -> DResult<()> {
|
2022-12-23 21:37:28 +11:00
|
|
|
let mut conn = self.get_conn().await?;
|
|
|
|
let tx = conn.transaction().await?;
|
2023-06-07 22:38:46 +10:00
|
|
|
tx.execute(
|
|
|
|
"UPDATE users SET current_session = NULL, \
|
2022-12-23 21:37:28 +11:00
|
|
|
current_listener = NULL WHERE current_listener = $1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&listener],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
tx.execute("DELETE FROM sendqueue WHERE listener = $1", &[&listener])
|
|
|
|
.await?;
|
|
|
|
tx.execute("DELETE FROM sessions WHERE listener = $1", &[&listener])
|
|
|
|
.await?;
|
|
|
|
tx.execute("DELETE FROM listeners WHERE listener = $1", &[&listener])
|
|
|
|
.await?;
|
2022-12-23 21:37:28 +11:00
|
|
|
tx.commit().await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2022-12-14 23:48:00 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn start_session(
|
|
|
|
self: &Self,
|
|
|
|
session: &ListenerSession,
|
|
|
|
details: &Session,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.get_conn()
|
|
|
|
.await?
|
|
|
|
.execute(
|
|
|
|
"INSERT INTO sessions (session, listener, details) \
|
2022-12-24 21:16:23 +11:00
|
|
|
VALUES ($1, $2, $3) ON CONFLICT (session) DO NOTHING",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&session.session,
|
|
|
|
&session.listener,
|
|
|
|
&serde_json::to_value(details)?,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-23 21:37:28 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2022-12-23 23:31:49 +11:00
|
|
|
|
2022-12-24 21:16:23 +11:00
|
|
|
pub async fn end_session(self: &Self, session: ListenerSession) -> DResult<()> {
|
2022-12-23 23:31:49 +11:00
|
|
|
let mut conn = self.get_conn().await?;
|
|
|
|
let tx = conn.transaction().await?;
|
2023-06-07 22:38:46 +10:00
|
|
|
tx.execute(
|
|
|
|
"UPDATE users SET current_session = NULL, \
|
2022-12-23 23:31:49 +11:00
|
|
|
current_listener = NULL WHERE current_session = $1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&session.session],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
tx.execute(
|
|
|
|
"DELETE FROM sendqueue WHERE session = $1",
|
|
|
|
&[&session.session],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
tx.execute(
|
|
|
|
"DELETE FROM sessions WHERE session = $1",
|
|
|
|
&[&session.session],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-23 23:31:49 +11:00
|
|
|
tx.commit().await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-12-25 00:25:52 +11:00
|
|
|
pub async fn start_transaction(self: &Self) -> DResult<DBTrans> {
|
|
|
|
let conn = self.get_conn().await?;
|
|
|
|
Ok(DBTransAsyncSendTryBuilder {
|
|
|
|
conn,
|
2023-06-07 22:38:46 +10:00
|
|
|
trans_builder: |conn| Box::pin(conn.transaction().map(|r| r.map(Some))),
|
|
|
|
}
|
|
|
|
.try_build()
|
|
|
|
.await?)
|
2022-12-25 00:25:52 +11:00
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn queue_for_session(
|
|
|
|
self: &Self,
|
|
|
|
session: &ListenerSession,
|
|
|
|
message: Option<&str>,
|
|
|
|
) -> DResult<()> {
|
2022-12-23 23:31:49 +11:00
|
|
|
let conn = self.get_conn().await?;
|
2023-06-07 22:38:46 +10:00
|
|
|
conn.execute(
|
|
|
|
"INSERT INTO sendqueue (session, listener, message) VALUES ($1, $2, $3)",
|
|
|
|
&[&session.session, &session.listener, &message],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-23 23:31:49 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-12-24 21:16:23 +11:00
|
|
|
pub async fn get_from_sendqueue(self: &Self) -> DResult<Vec<SendqueueItem>> {
|
2022-12-23 23:31:49 +11:00
|
|
|
let conn = self.get_conn().await?;
|
2023-06-07 22:38:46 +10:00
|
|
|
Ok(conn
|
|
|
|
.query(
|
|
|
|
"SELECT item, session, listener, message FROM sendqueue ORDER BY item ASC LIMIT 10",
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.map(SendqueueItem::from)
|
|
|
|
.collect())
|
2022-12-23 23:31:49 +11:00
|
|
|
}
|
|
|
|
|
2022-12-24 21:16:23 +11:00
|
|
|
pub async fn delete_from_sendqueue(self: &DBPool, item: &SendqueueItem) -> DResult<()> {
|
2022-12-23 23:31:49 +11:00
|
|
|
let conn = self.get_conn().await?;
|
2023-06-07 22:38:46 +10:00
|
|
|
conn.execute("DELETE FROM sendqueue WHERE item=$1", &[&item.item])
|
|
|
|
.await?;
|
2022-12-23 23:31:49 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2022-12-27 20:16:35 +11:00
|
|
|
|
|
|
|
pub async fn find_static_item_types(self: &Self) -> DResult<Box<BTreeSet<String>>> {
|
|
|
|
Ok(Box::new(
|
2023-06-07 22:38:46 +10:00
|
|
|
self.get_conn()
|
|
|
|
.await?
|
|
|
|
.query(
|
|
|
|
"SELECT DISTINCT details->>'item_type' AS item_type \
|
|
|
|
FROM items WHERE details->>'is_static' = 'true'",
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.iter()
|
2022-12-27 20:16:35 +11:00
|
|
|
.map(|r| r.get("item_type"))
|
2023-06-07 22:38:46 +10:00
|
|
|
.collect(),
|
|
|
|
))
|
2022-12-27 20:16:35 +11:00
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-01-07 23:06:02 +11:00
|
|
|
pub async fn find_static_task_types(self: &Self) -> DResult<Box<BTreeSet<String>>> {
|
|
|
|
Ok(Box::new(
|
2023-06-07 22:38:46 +10:00
|
|
|
self.get_conn()
|
|
|
|
.await?
|
|
|
|
.query(
|
|
|
|
"SELECT DISTINCT details->>'task_type' AS task_type \
|
|
|
|
FROM tasks WHERE details->>'is_static' = 'true'",
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.iter()
|
2023-01-07 23:06:02 +11:00
|
|
|
.map(|r| r.get("task_type"))
|
2023-06-07 22:38:46 +10:00
|
|
|
.collect(),
|
|
|
|
))
|
2023-01-07 23:06:02 +11:00
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2022-12-27 20:16:35 +11:00
|
|
|
pub async fn delete_static_items_by_type(self: &Self, item_type: &str) -> DResult<()> {
|
|
|
|
self.get_conn().await?.query(
|
2023-01-25 23:59:19 +11:00
|
|
|
"DELETE FROM items WHERE details->>'is_static' = 'true' AND details->>'item_type' = $1",
|
2022-12-27 20:16:35 +11:00
|
|
|
&[&item_type]).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-01-07 23:06:02 +11:00
|
|
|
pub async fn delete_static_tasks_by_type(self: &Self, task_type: &str) -> DResult<()> {
|
|
|
|
self.get_conn().await?.query(
|
2023-01-25 23:59:19 +11:00
|
|
|
"DELETE FROM tasks WHERE details->>'is_static' = 'true' AND details->>'task_type' = $1",
|
2023-01-07 23:06:02 +11:00
|
|
|
&[&task_type]).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-01-28 23:00:53 +11:00
|
|
|
pub async fn bump_session_time(&self, session: &ListenerSession) -> DResult<()> {
|
|
|
|
self.get_conn().await?.query(
|
|
|
|
"UPDATE sessions SET details=JSONB_SET(details, '{last_active}', to_json(NOW())::jsonb) \
|
|
|
|
WHERE session = $1", &[&session.session]
|
|
|
|
).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn get_conn(self: &DBPool) -> DResult<Object> {
|
|
|
|
let conn = self.pool.get().await?;
|
|
|
|
conn.execute("SET synchronous_commit=off", &[]).await?;
|
|
|
|
Ok(conn)
|
|
|
|
}
|
2022-12-23 23:31:49 +11:00
|
|
|
|
2022-12-23 21:37:28 +11:00
|
|
|
pub fn start(connstr: &str) -> DResult<DBPool> {
|
|
|
|
let mgr_config = ManagerConfig {
|
2023-06-07 22:38:46 +10:00
|
|
|
recycling_method: RecyclingMethod::Fast,
|
2022-12-23 21:37:28 +11:00
|
|
|
};
|
|
|
|
let mgr = Manager::from_config(
|
2023-06-07 22:38:46 +10:00
|
|
|
PgConfig::from_str(connstr).map_err(|e| Box::new(e) as Box<dyn Error + Send + Sync>)?,
|
|
|
|
NoTls,
|
|
|
|
mgr_config,
|
2022-12-23 21:37:28 +11:00
|
|
|
);
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
Pool::builder(mgr)
|
|
|
|
.max_size(4)
|
|
|
|
.build()
|
2022-12-23 21:37:28 +11:00
|
|
|
.map_err(|e| Box::new(e) as Box<dyn Error + Send + Sync>)
|
2022-12-23 23:31:49 +11:00
|
|
|
.map(|pool| Self { pool })
|
2022-12-23 21:37:28 +11:00
|
|
|
}
|
2022-12-14 23:48:00 +11:00
|
|
|
}
|
2022-12-25 00:25:52 +11:00
|
|
|
|
2022-12-31 00:59:14 +11:00
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct ItemSearchParams<'l> {
|
|
|
|
pub from_item: &'l Item,
|
|
|
|
pub query: &'l str,
|
|
|
|
pub include_contents: bool,
|
|
|
|
pub include_loc_contents: bool,
|
|
|
|
pub include_active_players: bool,
|
2023-02-20 00:02:29 +11:00
|
|
|
pub include_all_players: bool,
|
2023-02-20 22:27:43 +11:00
|
|
|
pub item_type_only: Option<&'l str>,
|
2023-05-23 20:37:27 +10:00
|
|
|
pub item_action_type_only: Option<&'l LocationActionType>,
|
2023-02-20 22:27:43 +11:00
|
|
|
pub limit: u8,
|
2023-02-20 00:02:29 +11:00
|
|
|
pub dead_first: bool,
|
2022-12-31 00:59:14 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl ItemSearchParams<'_> {
|
|
|
|
pub fn base<'l>(from_item: &'l Item, query: &'l str) -> ItemSearchParams<'l> {
|
|
|
|
ItemSearchParams {
|
2023-06-07 22:38:46 +10:00
|
|
|
from_item,
|
|
|
|
query,
|
2022-12-31 00:59:14 +11:00
|
|
|
include_contents: false,
|
|
|
|
include_loc_contents: false,
|
|
|
|
include_active_players: false,
|
2023-02-20 00:02:29 +11:00
|
|
|
include_all_players: false,
|
|
|
|
dead_first: false,
|
2023-02-20 22:27:43 +11:00
|
|
|
limit: 100,
|
|
|
|
item_type_only: None,
|
2023-05-23 20:37:27 +10:00
|
|
|
item_action_type_only: None,
|
2022-12-31 00:59:14 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
#[cfg_attr(test, automock)]
|
|
|
|
#[cfg_attr(test, allow(dead_code))]
|
2022-12-25 00:25:52 +11:00
|
|
|
impl DBTrans {
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn queue_for_session<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
session: &'a ListenerSession,
|
|
|
|
message: Option<&'a str>,
|
|
|
|
) -> DResult<()> {
|
2022-12-25 00:25:52 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"INSERT INTO sendqueue (session, listener, message) VALUES ($1, $2, $3)",
|
|
|
|
&[&session.session, &session.listener, &message],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-25 00:25:52 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn get_session_user_model<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
session: &'a ListenerSession,
|
|
|
|
) -> DResult<Option<(Session, Option<User>)>> {
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT s.details AS sess_details, \
|
2022-12-26 01:30:59 +11:00
|
|
|
u.details AS user_details FROM sessions s \
|
|
|
|
LEFT JOIN users u ON u.current_session = s.session \
|
2023-06-07 22:38:46 +10:00
|
|
|
WHERE s.session = $1",
|
|
|
|
&[&session.session],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
2022-12-25 01:42:51 +11:00
|
|
|
None => Ok(None),
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(row) => Ok(Some((
|
|
|
|
serde_json::from_value(row.get("sess_details"))?,
|
|
|
|
match row.get::<&str, Option<serde_json::Value>>("user_details") {
|
|
|
|
None => None,
|
|
|
|
Some(v) => serde_json::from_value(v)?,
|
|
|
|
},
|
|
|
|
))),
|
2022-12-25 01:42:51 +11:00
|
|
|
}
|
|
|
|
}
|
2022-12-26 01:30:59 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn save_session_model<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
session: &'a ListenerSession,
|
|
|
|
details: &Session,
|
|
|
|
) -> DResult<()> {
|
2022-12-26 01:30:59 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"UPDATE sessions SET details = $1 WHERE session = $2",
|
|
|
|
&[&serde_json::to_value(details)?, &session.session],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-26 01:30:59 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub async fn find_by_username<'a>(self: &'a Self, username: &'a str) -> DResult<Option<User>> {
|
2023-06-07 22:38:46 +10:00
|
|
|
if let Some(details_json) = self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM users WHERE username=$1",
|
|
|
|
&[&username.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
return Ok(Some(serde_json::from_value(details_json.get("details"))?));
|
|
|
|
}
|
2022-12-27 00:20:09 +11:00
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub async fn create_item<'a>(self: &'a Self, item: &'a Item) -> DResult<i64> {
|
2023-06-07 22:38:46 +10:00
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_one(
|
|
|
|
"INSERT INTO items (details) VALUES ($1) RETURNING item_id",
|
|
|
|
&[&serde_json::to_value(item)?],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.get("item_id"))
|
2022-12-27 00:20:09 +11:00
|
|
|
}
|
2022-12-29 23:16:52 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn find_exact_dyn_exit<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
source: &'a str,
|
|
|
|
direction: &'a Direction,
|
|
|
|
) -> DResult<Option<Item>> {
|
|
|
|
if let Some(details_json) = self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM items WHERE \
|
2023-04-08 23:51:18 +10:00
|
|
|
details->'dynamic_entrance'->>'source_item' = $1 AND \
|
|
|
|
LOWER(details->'dynamic_entrance'->>'direction') = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&source, &direction.describe().to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
return Ok(Some(serde_json::from_value(details_json.get("details"))?));
|
|
|
|
}
|
2023-04-08 23:51:18 +10:00
|
|
|
Ok(None)
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub async fn limited_update_static_item<'a>(self: &'a Self, item: &'a Item) -> DResult<()> {
|
2022-12-29 23:16:52 +11:00
|
|
|
let value = serde_json::to_value(item)?;
|
2023-06-07 22:38:46 +10:00
|
|
|
let obj_map = value.as_object().expect("Static item to be object in JSON");
|
|
|
|
let mut params: Vec<&(dyn ToSql + Sync)> = vec![&item.item_type, &item.item_code];
|
2022-12-29 23:16:52 +11:00
|
|
|
let mut det_ex: String = "details".to_owned();
|
|
|
|
let mut var_id = 3;
|
|
|
|
// Only copy more permanent fields, others are supposed to change over time and shouldn't
|
|
|
|
// be reset on restart.
|
2023-06-07 22:38:46 +10:00
|
|
|
for to_copy in [
|
|
|
|
"display",
|
|
|
|
"display_less_explicit",
|
|
|
|
"details",
|
|
|
|
"details_less_explicit",
|
|
|
|
"total_xp",
|
|
|
|
"total_stats",
|
|
|
|
"total_skills",
|
|
|
|
"pronouns",
|
|
|
|
"flags",
|
|
|
|
"sex",
|
|
|
|
"is_challenge_attack_only",
|
|
|
|
"aliases",
|
|
|
|
"species",
|
|
|
|
] {
|
2022-12-29 23:16:52 +11:00
|
|
|
det_ex = format!("jsonb_set({}, '{{{}}}', ${})", det_ex, to_copy, var_id);
|
|
|
|
params.push(obj_map.get(to_copy).unwrap_or(&Value::Null));
|
|
|
|
var_id += 1;
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
&("UPDATE items SET details = ".to_owned()
|
|
|
|
+ &det_ex
|
|
|
|
+ " WHERE details->>'item_type' = $1 AND details->>'item_code' = $2"),
|
|
|
|
¶ms,
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-29 23:16:52 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub async fn limited_update_static_task<'a>(self: &'a Self, task: &'a Task) -> DResult<()> {
|
2023-01-07 23:06:02 +11:00
|
|
|
let value = serde_json::to_value(task)?;
|
2023-06-07 22:38:46 +10:00
|
|
|
let obj_map = value.as_object().expect("Static task to be object in JSON");
|
2023-01-07 23:06:02 +11:00
|
|
|
let task_name: &(dyn ToSql + Sync) = &task.details.name();
|
2023-06-07 22:38:46 +10:00
|
|
|
let mut params: Vec<&(dyn ToSql + Sync)> = vec![task_name, &task.meta.task_code];
|
2023-01-07 23:06:02 +11:00
|
|
|
let mut det_ex: String = "details".to_owned();
|
|
|
|
let mut var_id = 3;
|
|
|
|
// Only copy more permanent fields, others are supposed to change over time and shouldn't
|
|
|
|
// be reset on restart. We do reset failure count since the problem may be fixed.
|
|
|
|
for to_copy in ["recurrence", "consecutive_failure_count", "task_details"] {
|
|
|
|
det_ex = format!("jsonb_set({}, '{{{}}}', ${})", det_ex, to_copy, var_id);
|
|
|
|
params.push(obj_map.get(to_copy).unwrap_or(&Value::Null));
|
|
|
|
var_id += 1;
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
&("UPDATE tasks SET details = ".to_owned()
|
|
|
|
+ &det_ex
|
|
|
|
+ " WHERE details->>'task_type' = $1 AND details->>'task_code' = $2"),
|
|
|
|
¶ms,
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-07 23:06:02 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn create_user<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
session: &'a ListenerSession,
|
|
|
|
user_dat: &'a User,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"INSERT INTO users (\
|
2022-12-27 00:20:09 +11:00
|
|
|
username, current_session, current_listener, details\
|
2023-06-07 22:38:46 +10:00
|
|
|
) VALUES ($1, $2, $3, $4)",
|
|
|
|
&[
|
|
|
|
&user_dat.username.to_lowercase(),
|
|
|
|
&session.session,
|
|
|
|
&session.listener,
|
|
|
|
&serde_json::to_value(user_dat)?,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn find_dynzone_for_user(&self, username: &str) -> DResult<Vec<Arc<Item>>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT details FROM items WHERE details->>'owner' = $1 \
|
|
|
|
AND details->>'item_type' = 'dynzone'",
|
|
|
|
&[&format!("player/{}", username.to_lowercase())],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|i| serde_json::from_value(i.get("details")).ok())
|
|
|
|
.map(Arc::new)
|
|
|
|
.collect())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn delete_user(&self, username: &str) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM users WHERE username = $1",
|
|
|
|
&[&username.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-27 00:20:09 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn save_user_model<'a>(self: &'a Self, details: &'a User) -> DResult<()> {
|
2022-12-27 00:20:09 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"UPDATE users SET details = $1 WHERE username = $2",
|
|
|
|
&[
|
|
|
|
&serde_json::to_value(details)?,
|
|
|
|
&details.username.to_lowercase(),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-27 00:20:09 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2022-12-27 16:08:27 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn attach_user_to_session<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
username: &'a str,
|
|
|
|
session: &'a ListenerSession,
|
|
|
|
) -> DResult<()> {
|
2022-12-27 16:08:27 +11:00
|
|
|
let username_l = username.to_lowercase();
|
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"INSERT INTO sendqueue (session, listener, message) \
|
2022-12-27 16:08:27 +11:00
|
|
|
SELECT current_session, current_listener, $1 FROM users \
|
|
|
|
WHERE username = $2 AND current_session IS NOT NULL \
|
|
|
|
AND current_listener IS NOT NULL",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&"Logged in from another session\r\n", &username_l],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-27 16:08:27 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"INSERT INTO sendqueue (session, listener, message) \
|
2022-12-29 00:43:23 +11:00
|
|
|
SELECT current_session, current_listener, null FROM users \
|
|
|
|
WHERE username = $1 AND current_session IS NOT NULL \
|
|
|
|
AND current_listener IS NOT NULL",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&username_l],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-29 00:43:23 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"UPDATE users SET current_session = $1, current_listener = $2 WHERE username = $3",
|
|
|
|
&[
|
|
|
|
&session.session as &(dyn ToSql + Sync),
|
|
|
|
&session.listener,
|
|
|
|
&username_l,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-27 16:08:27 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn find_static_items_by_type<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
item_type: &'a str,
|
|
|
|
) -> DResult<Box<BTreeSet<String>>> {
|
2022-12-27 20:16:35 +11:00
|
|
|
Ok(Box::new(
|
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.query(
|
|
|
|
"SELECT DISTINCT details->>'item_code' AS item_code FROM items WHERE \
|
2022-12-27 20:16:35 +11:00
|
|
|
details->>'is_static' = 'true' AND \
|
2023-06-07 22:38:46 +10:00
|
|
|
details->>'item_type' = $1",
|
|
|
|
&[&item_type],
|
|
|
|
)
|
2022-12-27 20:16:35 +11:00
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.map(|v| v.get("item_code"))
|
2023-06-07 22:38:46 +10:00
|
|
|
.collect(),
|
|
|
|
))
|
2022-12-27 20:16:35 +11:00
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn find_static_tasks_by_type<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
task_type: &'a str,
|
|
|
|
) -> DResult<Box<BTreeSet<String>>> {
|
2023-01-07 23:06:02 +11:00
|
|
|
Ok(Box::new(
|
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.query(
|
|
|
|
"SELECT DISTINCT details->>'task_code' AS task_code FROM tasks WHERE \
|
2023-01-07 23:06:02 +11:00
|
|
|
details->>'is_static' = 'true' AND \
|
2023-06-07 22:38:46 +10:00
|
|
|
details->>'task_type' = $1",
|
|
|
|
&[&task_type],
|
|
|
|
)
|
2023-01-07 23:06:02 +11:00
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.map(|v| v.get("task_code"))
|
2023-06-07 22:38:46 +10:00
|
|
|
.collect(),
|
|
|
|
))
|
2023-01-07 23:06:02 +11:00
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn delete_static_items_by_code<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
item_type: &'a str,
|
|
|
|
item_code: &str,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"DELETE FROM items WHERE details->>'is_static' = 'true' AND \
|
2023-01-02 15:39:55 +11:00
|
|
|
details->>'item_type' = $1 AND \
|
|
|
|
details->>'item_code' = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&item_type, &item_code],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-27 20:16:35 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2022-12-29 00:37:14 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn delete_static_tasks_by_code<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
task_type: &'a str,
|
|
|
|
task_code: &'a str,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"DELETE FROM task WHERE details->>'is_static' = 'true' AND \
|
2023-01-07 23:06:02 +11:00
|
|
|
details->>'task_type' = $1 AND \
|
|
|
|
details->>'task_code' = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&task_type, &task_code],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-07 23:06:02 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn find_item_by_type_code<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
item_type: &'a str,
|
|
|
|
item_code: &'a str,
|
|
|
|
) -> DResult<Option<Arc<Item>>> {
|
|
|
|
if let Some(item) = self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM items WHERE \
|
2022-12-29 00:37:14 +11:00
|
|
|
details->>'item_type' = $1 AND \
|
2023-06-07 22:38:46 +10:00
|
|
|
details->>'item_code' = $2",
|
|
|
|
&[&item_type, &item_code],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
return Ok(Some(Arc::new(serde_json::from_value::<Item>(
|
|
|
|
item.get("details"),
|
|
|
|
)?)));
|
2022-12-29 00:37:14 +11:00
|
|
|
}
|
|
|
|
Ok(None)
|
|
|
|
}
|
2022-12-29 22:17:55 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn transfer_all_possessions_code<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
src_loc: &'a str,
|
|
|
|
dst_loc: &'a str,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"UPDATE items SET details=JSONB_SET(details, '{location}', $1) \
|
2023-01-25 23:59:19 +11:00
|
|
|
WHERE details->>'location' = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&serde_json::to_value(dst_loc)?, &src_loc],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-25 23:59:19 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn transfer_all_possessions<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
source: &'a Item,
|
|
|
|
dest: &'a Item,
|
|
|
|
) -> DResult<()> {
|
2023-01-25 23:59:19 +11:00
|
|
|
let src_loc = format!("{}/{}", &source.item_type, &source.item_code);
|
|
|
|
let dst_loc = format!("{}/{}", &dest.item_type, &dest.item_code);
|
2023-06-07 22:38:46 +10:00
|
|
|
self.transfer_all_possessions_code(&src_loc, &dst_loc)
|
|
|
|
.await?;
|
2023-01-25 23:59:19 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn find_items_by_location<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
location: &'a str,
|
|
|
|
) -> DResult<Vec<Arc<Item>>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT details FROM items WHERE details->>'location' = $1 \
|
2022-12-31 00:59:14 +11:00
|
|
|
ORDER BY details->>'display'
|
2023-06-07 22:38:46 +10:00
|
|
|
LIMIT 100",
|
|
|
|
&[&location],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|i| serde_json::from_value(i.get("details")).ok())
|
|
|
|
.map(Arc::new)
|
|
|
|
.collect())
|
2022-12-29 22:17:55 +11:00
|
|
|
}
|
2022-12-30 00:41:46 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn find_item_by_location_dynroom_code<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
location: &'a str,
|
|
|
|
dynroom_code: &'a str,
|
|
|
|
) -> DResult<Option<Item>> {
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM items WHERE details->>'location' = $1 \
|
2023-04-08 23:51:18 +10:00
|
|
|
AND details->'special_data'->'DynroomData'->>'dynroom_code' = $2 LIMIT 1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&location, &dynroom_code],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
2023-04-08 23:51:18 +10:00
|
|
|
None => Ok(None),
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(v) => Ok(Some(serde_json::from_value(v.get("details"))?)),
|
2023-04-08 23:51:18 +10:00
|
|
|
}
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn save_item_model(self: &Self, details: &Item) -> DResult<()> {
|
2022-12-31 18:42:13 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"INSERT INTO items (details) VALUES ($1) \
|
2023-01-25 23:59:19 +11:00
|
|
|
ON CONFLICT ((details->>'item_type'), \
|
|
|
|
(details->>'item_code')) DO UPDATE SET \
|
|
|
|
details = EXCLUDED.details",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&serde_json::to_value(details)?],
|
|
|
|
)
|
|
|
|
.await?;
|
2022-12-31 18:42:13 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn delete_item<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
item_type: &'a str,
|
|
|
|
item_code: &'a str,
|
|
|
|
) -> DResult<()> {
|
2023-01-23 22:52:01 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"DELETE FROM items WHERE \
|
2023-01-25 23:59:19 +11:00
|
|
|
details->>'item_type' = $1 AND \
|
|
|
|
details->>'item_code' = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&item_type, &item_code],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-23 22:52:01 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn find_session_for_player<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
item_code: &'a str,
|
|
|
|
) -> DResult<Option<(ListenerSession, Session)>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT u.current_listener, u.current_session, s.details \
|
2022-12-31 19:33:31 +11:00
|
|
|
FROM users u JOIN sessions s ON s.session = u.current_session \
|
2023-06-07 22:38:46 +10:00
|
|
|
WHERE u.username=$1",
|
|
|
|
&[&item_code],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.and_then(|r| {
|
|
|
|
match (
|
|
|
|
r.get("current_listener"),
|
|
|
|
r.get("current_session"),
|
|
|
|
r.get("details"),
|
|
|
|
) {
|
|
|
|
(Some(listener), Some(session), details) => Some((
|
|
|
|
ListenerSession { listener, session },
|
|
|
|
serde_json::from_value(details).ok()?,
|
|
|
|
)),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2022-12-31 00:59:14 +11:00
|
|
|
pub async fn resolve_items_by_display_name_for_player<'l>(
|
2023-02-19 14:03:15 +11:00
|
|
|
self: &'l Self,
|
2023-06-07 22:38:46 +10:00
|
|
|
search: &'l ItemSearchParams<'l>,
|
2022-12-31 00:59:14 +11:00
|
|
|
) -> DResult<Arc<Vec<Arc<Item>>>> {
|
2022-12-30 00:41:46 +11:00
|
|
|
let mut ctes: Vec<String> = Vec::new();
|
|
|
|
let mut include_tables: Vec<&'static str> = Vec::new();
|
|
|
|
|
2022-12-31 00:59:14 +11:00
|
|
|
let player_loc = &search.from_item.location;
|
2023-06-07 22:38:46 +10:00
|
|
|
let player_desig = format!(
|
|
|
|
"{}/{}",
|
|
|
|
search.from_item.item_type, search.from_item.item_code
|
|
|
|
);
|
|
|
|
|
2023-02-20 22:27:43 +11:00
|
|
|
let (offset, mut query) = parse_offset(search.query);
|
2023-02-23 21:33:26 +11:00
|
|
|
let mut param_no: usize = 5;
|
2023-02-20 22:27:43 +11:00
|
|
|
let mut extra_order: String = String::new();
|
|
|
|
let mut extra_where: String = String::new();
|
|
|
|
|
|
|
|
let mut dead_only = false;
|
|
|
|
if query.starts_with("dead") {
|
|
|
|
query = query[4..].trim();
|
|
|
|
dead_only = true;
|
|
|
|
} else if query.starts_with("corpse of") {
|
|
|
|
query = query[9..].trim();
|
|
|
|
dead_only = true;
|
|
|
|
} else if query.starts_with("corpse") {
|
|
|
|
query = query[6..].trim();
|
|
|
|
dead_only = true;
|
|
|
|
}
|
|
|
|
if dead_only {
|
2023-04-24 00:56:42 +10:00
|
|
|
extra_where.push_str(" AND COALESCE(details->>'death_data' IS NOT NULL, false) = true");
|
2023-02-20 22:27:43 +11:00
|
|
|
} else if search.dead_first {
|
2023-04-24 00:56:42 +10:00
|
|
|
extra_order.push_str(" COALESCE(details->>'death_data' IS NOT NULL, false) DESC,");
|
2023-02-20 22:27:43 +11:00
|
|
|
} else {
|
2023-04-24 00:56:42 +10:00
|
|
|
extra_order.push_str(" COALESCE(details->>'death_data' IS NOT NULL, false) ASC,");
|
2023-02-20 22:27:43 +11:00
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
let query_wildcard = query
|
|
|
|
.replace("\\", "\\\\")
|
|
|
|
.replace("_", "\\_")
|
|
|
|
.replace("%", "")
|
|
|
|
.to_lowercase()
|
|
|
|
+ "%";
|
|
|
|
let offset_sql = offset
|
|
|
|
.map(|x| (if x >= 1 { x - 1 } else { x }) as i64)
|
|
|
|
.unwrap_or(0);
|
2022-12-31 19:33:31 +11:00
|
|
|
let query_len = query.len() as i32;
|
2023-02-20 22:27:43 +11:00
|
|
|
let limit = search.limit as i64;
|
2023-06-07 22:38:46 +10:00
|
|
|
let mut params: Vec<&(dyn ToSql + Sync)> =
|
|
|
|
vec![&query_wildcard, &offset_sql, &query_len, &limit];
|
2023-02-20 22:27:43 +11:00
|
|
|
|
|
|
|
match search.item_type_only {
|
|
|
|
None => {}
|
|
|
|
Some(ref item_type) => {
|
|
|
|
extra_where.push_str(&format!(" AND details->>'item_type' = ${}", param_no));
|
|
|
|
param_no += 1;
|
|
|
|
params.push(item_type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-23 20:37:27 +10:00
|
|
|
let item_action_type_value: Option<serde_json::Value> = match search.item_action_type_only {
|
|
|
|
None => None,
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(v) => Some(serde_json::to_value(v)?),
|
2023-05-23 20:37:27 +10:00
|
|
|
};
|
|
|
|
match item_action_type_value {
|
|
|
|
None => {}
|
|
|
|
Some(ref item_action_type) => {
|
2023-06-07 22:38:46 +10:00
|
|
|
extra_where.push_str(&format!(
|
|
|
|
" AND (details->'action_type')::TEXT = ${}::JSONB::TEXT",
|
|
|
|
param_no
|
|
|
|
));
|
2023-05-23 20:37:27 +10:00
|
|
|
param_no += 1;
|
|
|
|
params.push(item_action_type);
|
|
|
|
}
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2022-12-31 00:59:14 +11:00
|
|
|
if search.include_contents {
|
|
|
|
ctes.push(format!("contents AS (\
|
2023-01-20 23:38:57 +11:00
|
|
|
SELECT details, details->'aliases' AS aliases FROM items WHERE details->>'location' = ${}\
|
2022-12-31 00:59:14 +11:00
|
|
|
)", param_no));
|
|
|
|
param_no += 1;
|
|
|
|
params.push(&player_desig);
|
2023-01-20 23:38:57 +11:00
|
|
|
include_tables.push("SELECT details, aliases FROM contents");
|
2022-12-30 00:41:46 +11:00
|
|
|
}
|
2022-12-31 00:59:14 +11:00
|
|
|
if search.include_loc_contents {
|
|
|
|
ctes.push(format!("loc_contents AS (\
|
2023-01-20 23:38:57 +11:00
|
|
|
SELECT details, details->'aliases' AS aliases FROM items WHERE details->>'location' = ${}\
|
2022-12-31 00:59:14 +11:00
|
|
|
)", param_no));
|
|
|
|
drop(param_no); // or increment if this is a problem.
|
|
|
|
params.push(&player_loc);
|
2023-01-20 23:38:57 +11:00
|
|
|
include_tables.push("SELECT details, aliases FROM loc_contents");
|
2022-12-30 00:41:46 +11:00
|
|
|
}
|
2022-12-31 00:59:14 +11:00
|
|
|
if search.include_active_players {
|
2023-06-07 22:38:46 +10:00
|
|
|
ctes.push(
|
|
|
|
"active_players AS (\
|
2023-02-26 17:01:05 +11:00
|
|
|
SELECT i.details, ('[]'::JSONB) AS aliases FROM items i \
|
|
|
|
JOIN users u ON u.username = i.details->>'item_code' \
|
|
|
|
WHERE i.details->>'item_type' = 'player' \
|
|
|
|
AND u.current_session IS NOT NULL \
|
2023-06-07 22:38:46 +10:00
|
|
|
)"
|
|
|
|
.to_owned(),
|
|
|
|
);
|
2023-01-20 23:38:57 +11:00
|
|
|
include_tables.push("SELECT details, aliases FROM active_players");
|
2022-12-30 00:41:46 +11:00
|
|
|
}
|
2022-12-31 00:59:14 +11:00
|
|
|
if search.include_all_players {
|
2022-12-30 00:41:46 +11:00
|
|
|
ctes.push("all_players AS (\
|
2023-01-20 23:38:57 +11:00
|
|
|
SELECT details, ('[]'::JSONB) AS aliases FROM items WHERE details->>'item_type' = 'player'
|
2022-12-30 00:41:46 +11:00
|
|
|
)".to_owned());
|
2023-01-20 23:38:57 +11:00
|
|
|
include_tables.push("SELECT details, aliases FROM all_players");
|
2022-12-30 00:41:46 +11:00
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
ctes.push(format!(
|
|
|
|
"relevant_items AS ({})",
|
|
|
|
include_tables.join(" UNION ")
|
|
|
|
));
|
2022-12-30 00:41:46 +11:00
|
|
|
|
|
|
|
let cte_str: String = ctes.join(", ");
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
Ok(Arc::new(
|
|
|
|
self.pg_trans()?
|
|
|
|
.query(
|
|
|
|
&format!(
|
|
|
|
"WITH {} SELECT details FROM relevant_items \
|
2023-02-23 21:33:26 +11:00
|
|
|
WHERE \
|
2023-02-20 22:27:43 +11:00
|
|
|
((lower(details->>'display') LIKE $1) \
|
|
|
|
OR (lower(details ->>'display_less_explicit') LIKE $1) \
|
2023-02-23 21:33:26 +11:00
|
|
|
OR EXISTS (SELECT 1 FROM jsonb_array_elements(aliases) AS al(alias) WHERE \
|
|
|
|
LOWER(alias#>>'{{}}') LIKE $1)) {} \
|
2023-02-20 22:27:43 +11:00
|
|
|
ORDER BY {} ABS(length(details->>'display')-$3) ASC \
|
2023-06-07 22:38:46 +10:00
|
|
|
LIMIT $4 OFFSET $2",
|
|
|
|
&cte_str, &extra_where, &extra_order
|
|
|
|
),
|
|
|
|
¶ms,
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|i| serde_json::from_value(i.get("details")).ok())
|
|
|
|
.map(Arc::new)
|
|
|
|
.collect(),
|
|
|
|
))
|
2022-12-30 00:41:46 +11:00
|
|
|
}
|
2023-01-01 19:24:13 +11:00
|
|
|
|
|
|
|
pub async fn get_next_scheduled_task(&self) -> DResult<Option<TaskParse>> {
|
2023-06-07 22:38:46 +10:00
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM tasks WHERE \
|
2023-01-01 19:24:13 +11:00
|
|
|
CAST(details->>'next_scheduled' AS TIMESTAMPTZ) <= now() \
|
2023-06-07 22:38:46 +10:00
|
|
|
ORDER BY details->>'next_scheduled' ASC LIMIT 1",
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
2023-01-01 19:24:13 +11:00
|
|
|
None => Ok(None),
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(row) => Ok(serde_json::from_value(row.get("details"))?),
|
2023-01-01 19:24:13 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub async fn delete_task<'a>(&'a self, task_type: &'a str, task_code: &'a str) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM tasks WHERE details->>'task_type' = $1 AND \
|
|
|
|
details->>'task_code' = $2",
|
|
|
|
&[&task_type, &task_code],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-01 19:24:13 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub async fn upsert_task<'a>(&'a self, task: &'a Task) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"INSERT INTO tasks (details) \
|
2023-01-02 13:25:05 +11:00
|
|
|
VALUES ($1) \
|
|
|
|
ON CONFLICT ((details->>'task_code'), (details->>'task_type')) \
|
2023-06-07 22:38:46 +10:00
|
|
|
DO UPDATE SET details = $1",
|
|
|
|
&[&serde_json::to_value(task)?],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-02 13:25:05 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn update_task<'a>(
|
|
|
|
&'a self,
|
|
|
|
task_type: &'a str,
|
|
|
|
task_code: &'a str,
|
|
|
|
task: &'a TaskParse,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"UPDATE tasks SET details = $3 WHERE details->>'task_type' = $1 AND \
|
2023-01-01 19:24:13 +11:00
|
|
|
details->>'task_code' = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&task_type, &task_code, &serde_json::to_value(task)?],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-01-01 19:24:13 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-01-23 22:52:01 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn check_task_by_type_code<'a>(
|
|
|
|
&'a self,
|
|
|
|
task_type: &'a str,
|
|
|
|
task_code: &'a str,
|
|
|
|
) -> DResult<bool> {
|
|
|
|
let n: i64 = self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_one(
|
|
|
|
"SELECT COUNT(*) FROM tasks WHERE \
|
2023-02-26 00:56:22 +11:00
|
|
|
details->>'task_type' = $1 AND \
|
2023-06-07 22:38:46 +10:00
|
|
|
details->>'task_code' = $2",
|
|
|
|
&[&task_type, &task_code],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.get(0);
|
2023-02-26 00:56:22 +11:00
|
|
|
Ok(n > 0)
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-01-23 22:52:01 +11:00
|
|
|
pub async fn alloc_item_code(&self) -> DResult<i64> {
|
2023-06-07 22:38:46 +10:00
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_one("SELECT NEXTVAL('item_seq')", &[])
|
|
|
|
.await?
|
|
|
|
.get(0))
|
2023-01-23 22:52:01 +11:00
|
|
|
}
|
2023-01-28 23:00:53 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn get_online_info(&self) -> DResult<Vec<OnlineInfo>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"WITH show_corps AS (\
|
2023-03-19 15:41:48 +11:00
|
|
|
SELECT DISTINCT ON (member_username) \
|
|
|
|
m.member_username AS username, \
|
|
|
|
c.details ->> 'name' AS corpname \
|
|
|
|
FROM corps c JOIN corp_membership m ON m.corp_id = c.corp_id \
|
|
|
|
ORDER BY m.member_username, (m.details->>'priority')::int DESC NULLS LAST, \
|
|
|
|
(m.details->>'joined_at') :: TIMESTAMPTZ ASC \
|
|
|
|
)\
|
|
|
|
SELECT jsonb_build_object(\
|
2023-01-28 23:00:53 +11:00
|
|
|
'username', u.details->>'username',\
|
2023-03-19 15:41:48 +11:00
|
|
|
'corp', c.corpname, \
|
2023-01-28 23:00:53 +11:00
|
|
|
'time', s.details->>'last_active'\
|
|
|
|
) FROM sessions s \
|
|
|
|
JOIN users u ON u.current_session = s.session \
|
2023-03-19 15:41:48 +11:00
|
|
|
LEFT JOIN show_corps c ON c.username = LOWER(u.username) \
|
2023-06-07 22:38:46 +10:00
|
|
|
ORDER BY s.details->>'last_active' DESC",
|
|
|
|
&[],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|row| serde_json::from_value(row.get(0)).ok())
|
|
|
|
.collect())
|
2023-01-28 23:00:53 +11:00
|
|
|
}
|
2023-02-03 23:26:24 +11:00
|
|
|
|
|
|
|
pub async fn get_location_stats(&self, location: &str) -> DResult<LocationStats> {
|
|
|
|
Ok(serde_json::from_value(self.pg_trans()?.query_one(
|
2023-02-19 01:18:08 +11:00
|
|
|
"SELECT JSON_BUILD_OBJECT('total_count', COUNT(*), 'total_weight', COALESCE(SUM(CAST(details->>'weight' AS NUMERIC)), 0)) \
|
|
|
|
FROM items WHERE details->>'location' = $1", &[&location]
|
2023-02-03 23:26:24 +11:00
|
|
|
).await?.get(0))?)
|
|
|
|
}
|
2023-02-19 01:18:08 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn set_exclusive_action_type_to(
|
|
|
|
&self,
|
|
|
|
item: &Item,
|
|
|
|
new_action_type: &LocationActionType,
|
|
|
|
other_item_action_type: &LocationActionType,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"UPDATE items SET details=\
|
2023-02-19 01:18:08 +11:00
|
|
|
JSONB_SET(details, '{action_type}', $1) \
|
|
|
|
WHERE details->>'location' = $2 AND \
|
2023-04-21 23:33:23 +10:00
|
|
|
(details->'action_type')::TEXT = $3::JSONB::TEXT",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&serde_json::to_value(other_item_action_type)?,
|
|
|
|
&item.location,
|
|
|
|
&serde_json::to_value(new_action_type)?,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"UPDATE items SET details=\
|
2023-02-19 01:18:08 +11:00
|
|
|
JSONB_SET(details, '{action_type}', $1) \
|
|
|
|
WHERE details->>'item_type' = $2 AND \
|
|
|
|
details->>'item_code' = $3",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&serde_json::to_value(new_action_type)?,
|
|
|
|
&item.item_type,
|
|
|
|
&item.item_code,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-02-19 01:18:08 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn find_by_action_and_location(
|
|
|
|
&self,
|
|
|
|
location: &str,
|
|
|
|
action_type: &LocationActionType,
|
|
|
|
) -> DResult<Vec<Arc<Item>>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT details FROM items WHERE \
|
2023-02-19 01:18:08 +11:00
|
|
|
details->>'location' = $1 AND \
|
2023-05-23 20:37:27 +10:00
|
|
|
((details->'action_type')::TEXT = $2::JSONB::TEXT) LIMIT 100",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&location, &serde_json::to_value(action_type)?],
|
|
|
|
)
|
|
|
|
.await?
|
2023-05-23 20:37:27 +10:00
|
|
|
.into_iter()
|
|
|
|
.filter_map(
|
|
|
|
|row| match serde_json::from_value::<Item>(row.get("details")) {
|
|
|
|
Err(_) => None,
|
2023-06-07 22:38:46 +10:00
|
|
|
Ok(item) => Some(Arc::new(item)),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.collect())
|
2023-02-19 01:18:08 +11:00
|
|
|
}
|
2023-03-13 15:23:07 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn list_consents(
|
|
|
|
&self,
|
|
|
|
consenting: &str,
|
|
|
|
) -> DResult<Vec<(String, ConsentType, Consent)>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT consented_user, consent_type, details \
|
2023-04-02 12:30:50 +10:00
|
|
|
FROM user_consent \
|
|
|
|
WHERE consenting_user = $1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&consenting.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|row| match serde_json::from_value(row.get(2)) {
|
|
|
|
Err(_) => None,
|
|
|
|
Ok(consent) => ConsentType::from_str(row.get(1))
|
|
|
|
.map(|consent_type| (row.get(0), consent_type, consent)),
|
|
|
|
})
|
2023-04-02 12:30:50 +10:00
|
|
|
.collect())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn find_user_consent_by_parties_type(
|
|
|
|
&self,
|
|
|
|
consenting: &str,
|
|
|
|
consented: &str,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
) -> DResult<Option<Consent>> {
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM user_consent WHERE consenting_user = $1 AND \
|
2023-03-13 15:23:07 +11:00
|
|
|
consented_user = $2 AND consent_type = $3",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&consenting, &consented, &ConsentType::to_str(consent_type)],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
2023-03-13 15:23:07 +11:00
|
|
|
None => Ok(None),
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(row) => Ok(Some(serde_json::from_value(row.get(0))?)),
|
2023-03-13 15:23:07 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn find_corp_consent_by_parties_type(
|
|
|
|
&self,
|
|
|
|
consenting: &CorpId,
|
|
|
|
consented: &CorpId,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
) -> DResult<Option<Consent>> {
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT details FROM corp_consent WHERE consenting_corp = $1 AND \
|
2023-03-27 22:41:39 +11:00
|
|
|
consented_corp = $2 AND consent_type = $3",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&consenting.0,
|
|
|
|
&consented.0,
|
|
|
|
&ConsentType::to_str(consent_type),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
2023-03-28 22:31:59 +11:00
|
|
|
None => Ok(None),
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(row) => Ok(Some(serde_json::from_value(row.get(0))?)),
|
2023-03-28 22:31:59 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn find_corp_consent_by_user_parties_type(
|
|
|
|
&self,
|
|
|
|
usr_consenting: &str,
|
|
|
|
usr_consented: &str,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
) -> DResult<Option<Consent>> {
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT cc.details FROM corp_consent cc \
|
2023-03-28 22:31:59 +11:00
|
|
|
JOIN corp_membership cm_consenting ON cc.consenting_corp = cm_consenting.corp_id \
|
|
|
|
JOIN corp_membership cm_consented ON cc.consented_corp = cm_consented.corp_id \
|
|
|
|
WHERE cm_consenting.member_username = $1 AND \
|
|
|
|
cm_consented.member_username = $2 AND cc.consent_type = $3",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&usr_consenting,
|
|
|
|
&usr_consented,
|
|
|
|
&ConsentType::to_str(consent_type),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
2023-03-27 22:41:39 +11:00
|
|
|
None => Ok(None),
|
2023-06-07 22:38:46 +10:00
|
|
|
Some(row) => Ok(Some(serde_json::from_value(row.get(0))?)),
|
2023-03-27 22:41:39 +11:00
|
|
|
}
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-03-13 22:38:54 +11:00
|
|
|
pub async fn revoke_until_death_consent(&self, party: &str) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM user_consent WHERE (consenting_user = $1 OR \
|
2023-03-13 22:38:54 +11:00
|
|
|
(consented_user = $1 AND consent_type = 'fight')) AND \
|
|
|
|
details->>'until_death'='true'",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&party],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM corp_consent cc USING
|
2023-03-27 22:41:39 +11:00
|
|
|
corp_membership cm
|
|
|
|
WHERE (cc.consenting_corp = cm.corp_id OR cc.consented_corp = cm.corp_id) AND \
|
|
|
|
cm.member_username = $1 AND \
|
|
|
|
cm.details->>'joined_at' IS NOT NULL AND \
|
|
|
|
cc.consent_type = 'fight' AND \
|
|
|
|
cc.details->>'until_death'='true'",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&party],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-13 22:38:54 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-03-13 15:23:07 +11:00
|
|
|
pub async fn delete_expired_user_consent(&self) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM user_consent WHERE details->>'expires' < $1",
|
|
|
|
&[&Utc::now().to_rfc3339_opts(chrono::SecondsFormat::Nanos, true)],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-13 15:23:07 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-03-27 22:41:39 +11:00
|
|
|
pub async fn delete_expired_corp_consent(&self) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM corp_consent WHERE details->>'expires' < $1",
|
|
|
|
&[&Utc::now().to_rfc3339_opts(chrono::SecondsFormat::Nanos, true)],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-27 22:41:39 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn delete_user_consent(
|
|
|
|
&self,
|
|
|
|
consenting: &str,
|
|
|
|
consented: &str,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM user_consent WHERE consenting_user = $1 AND \
|
2023-03-13 15:23:07 +11:00
|
|
|
consented_user = $2 AND consent_type = $3",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&consenting, &consented, &ConsentType::to_str(consent_type)],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-13 15:23:07 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn delete_corp_consent(
|
|
|
|
&self,
|
|
|
|
consenting: &CorpId,
|
|
|
|
consented: &CorpId,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM corp_consent WHERE consenting_corp = $1 AND \
|
2023-03-27 22:41:39 +11:00
|
|
|
consented_corp = $2 AND consent_type = $3",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&consenting.0,
|
|
|
|
&consented.0,
|
|
|
|
&ConsentType::to_str(consent_type),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-27 22:41:39 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn upsert_user_consent(
|
|
|
|
&self,
|
|
|
|
consenting: &str,
|
|
|
|
consented: &str,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
details: &Consent,
|
2023-03-13 15:23:07 +11:00
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute("INSERT INTO user_consent (consenting_user, consented_user, consent_type, details) VALUES ($1, $2, $3, $4) \
|
|
|
|
ON CONFLICT (consenting_user, consented_user, consent_type) DO UPDATE SET \
|
|
|
|
details = EXCLUDED.details",
|
|
|
|
&[&consenting, &consented, &ConsentType::to_str(consent_type),
|
|
|
|
&serde_json::to_value(details)?]).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn upsert_corp_consent(
|
|
|
|
&self,
|
|
|
|
consenting: &CorpId,
|
|
|
|
consented: &CorpId,
|
|
|
|
consent_type: &ConsentType,
|
|
|
|
details: &Consent,
|
2023-03-27 22:41:39 +11:00
|
|
|
) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute("INSERT INTO corp_consent (consenting_corp, consented_corp, consent_type, details) VALUES ($1, $2, $3, $4) \
|
|
|
|
ON CONFLICT (consenting_corp, consented_corp, consent_type) DO UPDATE SET \
|
|
|
|
details = EXCLUDED.details",
|
|
|
|
&[&consenting.0, &consented.0, &ConsentType::to_str(consent_type),
|
|
|
|
&serde_json::to_value(details)?]).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-03-19 00:04:59 +11:00
|
|
|
pub async fn find_corp_by_name(&self, name: &str) -> DResult<Option<(CorpId, Corp)>> {
|
2023-06-07 22:38:46 +10:00
|
|
|
Ok(
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT corp_id, details FROM corps WHERE LOWER(details->>'name') = $1",
|
|
|
|
&[&name.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
None => None,
|
|
|
|
Some(row) => Some((
|
|
|
|
CorpId(row.get("corp_id")),
|
|
|
|
serde_json::from_value(row.get("details"))?,
|
|
|
|
)),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn match_user_corp_by_name(
|
|
|
|
&self,
|
|
|
|
corpname: &str,
|
|
|
|
username: &str,
|
|
|
|
) -> DResult<Option<(CorpId, Corp, CorpMembership)>> {
|
|
|
|
Ok(
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT c.corp_id, c.details AS cdetails, m.details AS mdetails FROM corps c \
|
2023-03-19 22:59:35 +11:00
|
|
|
JOIN corp_membership m ON m.corp_id = c.corp_id \
|
|
|
|
WHERE LOWER(c.details->>'name') LIKE $1 AND \
|
|
|
|
m.member_username = $2 \
|
|
|
|
ORDER BY \
|
|
|
|
ABS(length(c.details->>'name')-length($1)) DESC \
|
|
|
|
LIMIT 1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&(corpname
|
|
|
|
.replace("\\", "\\\\")
|
|
|
|
.replace("_", "\\_")
|
|
|
|
.replace("%", "")
|
|
|
|
.to_lowercase()
|
|
|
|
+ "%"),
|
|
|
|
&(username.to_lowercase()),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
None => None,
|
|
|
|
Some(row) => Some((
|
|
|
|
CorpId(row.get("corp_id")),
|
|
|
|
serde_json::from_value(row.get("cdetails"))?,
|
|
|
|
serde_json::from_value(row.get("mdetails"))?,
|
|
|
|
)),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2023-03-19 00:04:59 +11:00
|
|
|
pub async fn create_corp(&self, details: &Corp) -> DResult<CorpId> {
|
2023-06-07 22:38:46 +10:00
|
|
|
let id = self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_one(
|
|
|
|
"INSERT INTO corps (details) VALUES ($1) RETURNING corp_id",
|
|
|
|
&[&serde_json::to_value(details)?],
|
|
|
|
)
|
|
|
|
.await?
|
2023-03-19 00:04:59 +11:00
|
|
|
.get("corp_id");
|
|
|
|
Ok(CorpId(id))
|
|
|
|
}
|
|
|
|
|
2023-03-26 16:51:10 +11:00
|
|
|
pub async fn update_corp_details(&self, corp: &CorpId, details: &Corp) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"UPDATE corps SET details=$1 WHERE corp_id = $2",
|
|
|
|
&[&serde_json::to_value(details)?, &corp.0],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-26 16:51:10 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-03-19 22:59:35 +11:00
|
|
|
pub async fn expire_old_invites(&self) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM corp_membership WHERE details->>'invited_at' <= $1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&(Utc::now() - chrono::Duration::hours(4))
|
|
|
|
.to_rfc3339_opts(chrono::SecondsFormat::Nanos, true)],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-19 22:59:35 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
|
|
|
pub async fn upsert_corp_membership(
|
|
|
|
&self,
|
|
|
|
corp: &CorpId,
|
|
|
|
username: &str,
|
|
|
|
details: &CorpMembership,
|
|
|
|
) -> DResult<()> {
|
2023-03-19 00:04:59 +11:00
|
|
|
self.pg_trans()?
|
2023-06-07 22:38:46 +10:00
|
|
|
.execute(
|
|
|
|
"INSERT INTO corp_membership (corp_id, member_username, details) \
|
2023-03-19 00:04:59 +11:00
|
|
|
VALUES ($1, $2, $3) \
|
|
|
|
ON CONFLICT (corp_id, member_username) DO UPDATE SET \
|
|
|
|
details = excluded.details",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[
|
|
|
|
&corp.0,
|
|
|
|
&username.to_lowercase(),
|
|
|
|
&serde_json::to_value(details)?,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
.await?;
|
2023-03-19 00:04:59 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn get_corp_memberships_for_user(
|
|
|
|
&self,
|
|
|
|
username: &str,
|
|
|
|
) -> DResult<Vec<(CorpId, String, CorpMembership)>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT m.corp_id, c.details->>'name', m.details FROM corp_membership m \
|
2023-03-19 00:04:59 +11:00
|
|
|
JOIN corps c ON c.corp_id = m.corp_id WHERE m.member_username = $1 \
|
2023-03-19 22:59:35 +11:00
|
|
|
AND m.details->>'joined_at' IS NOT NULL \
|
|
|
|
ORDER BY (m.details->>'priority')::int DESC NULLS LAST, \
|
|
|
|
(m.details->>'joined_at') :: TIMESTAMPTZ ASC",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&username.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|row| match serde_json::from_value(row.get(2)) {
|
|
|
|
Err(_) => None,
|
|
|
|
Ok(j) => Some((CorpId(row.get(0)), row.get(1), j)),
|
|
|
|
})
|
2023-03-19 00:04:59 +11:00
|
|
|
.collect())
|
|
|
|
}
|
2023-03-26 22:29:39 +11:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn get_default_corp_for_user(
|
|
|
|
&self,
|
|
|
|
username: &str,
|
|
|
|
) -> DResult<Option<(CorpId, Corp)>> {
|
|
|
|
Ok(
|
|
|
|
match self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_opt(
|
|
|
|
"SELECT m.corp_id, c.details FROM corp_membership m \
|
2023-03-26 22:29:39 +11:00
|
|
|
JOIN corps c ON c.corp_id = m.corp_id WHERE m.member_username = $1 \
|
|
|
|
AND m.details->>'joined_at' IS NOT NULL \
|
|
|
|
ORDER BY (m.details->>'priority')::int ASC NULLS LAST, \
|
|
|
|
(m.details->>'joined_at') :: TIMESTAMPTZ ASC LIMIT 1",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&username.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
None => None,
|
|
|
|
Some(row) => match serde_json::from_value(row.get(1)) {
|
|
|
|
Err(_) => None,
|
|
|
|
Ok(j) => Some((CorpId(row.get(0)), j)),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn broadcast_to_corp<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
corp_id: &'a CorpId,
|
|
|
|
comm_type: &'a CorpCommType,
|
|
|
|
except_user: Option<&'a str>,
|
|
|
|
message: &'a str,
|
|
|
|
) -> DResult<()> {
|
|
|
|
let comm_type_s = serde_json::to_value(comm_type)?
|
|
|
|
.as_str()
|
2023-03-25 00:58:19 +11:00
|
|
|
.ok_or("comm type doesn't serialise to JSON string")?
|
|
|
|
.to_owned();
|
2023-06-07 22:38:46 +10:00
|
|
|
let mut params: Vec<&(dyn ToSql + Sync)> = vec![&message, &corp_id.0, &comm_type_s];
|
2023-03-25 00:58:19 +11:00
|
|
|
let mut query = "INSERT INTO sendqueue (session, listener, message) \
|
|
|
|
SELECT s.session, s.listener, $1 FROM \
|
|
|
|
sessions s \
|
|
|
|
JOIN users u ON u.current_session = s.session \
|
|
|
|
JOIN corp_membership m ON m.member_username = u.username \
|
|
|
|
WHERE m.corp_id = $2 AND \
|
|
|
|
(m.details->'comms_on') ? $3 AND \
|
2023-06-07 22:38:46 +10:00
|
|
|
m.details->>'joined_at' IS NOT NULL"
|
|
|
|
.to_owned();
|
2023-03-25 00:58:19 +11:00
|
|
|
match except_user.as_ref() {
|
2023-06-07 22:38:46 +10:00
|
|
|
None => {}
|
2023-03-25 00:58:19 +11:00
|
|
|
Some(u) => {
|
|
|
|
query.push_str(" AND u.username <> $4");
|
|
|
|
params.push(u);
|
|
|
|
}
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?.execute(&query, ¶ms).await?;
|
2023-03-25 00:58:19 +11:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn list_corp_members<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
corp_id: &'a CorpId,
|
|
|
|
) -> DResult<Vec<(String, CorpMembership)>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT member_username, details \
|
2023-03-25 00:58:19 +11:00
|
|
|
FROM corp_membership WHERE \
|
2023-06-07 22:38:46 +10:00
|
|
|
corp_id = $1",
|
|
|
|
&[&corp_id.0],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.iter()
|
|
|
|
.filter_map(|i| {
|
|
|
|
Some((
|
|
|
|
i.get("member_username"),
|
|
|
|
serde_json::from_value(i.get("details")).ok()?,
|
|
|
|
))
|
|
|
|
})
|
|
|
|
.collect())
|
2023-03-25 00:58:19 +11:00
|
|
|
}
|
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn delete_corp<'a>(self: &'a Self, corp_id: &'a CorpId) -> DResult<()> {
|
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM corps WHERE \
|
|
|
|
corp_id = $1",
|
|
|
|
&[&corp_id.0],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
2023-03-25 00:58:19 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn delete_corp_membership<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
corp_id: &'a CorpId,
|
2023-06-07 22:38:46 +10:00
|
|
|
username: &'a str,
|
2023-03-25 00:58:19 +11:00
|
|
|
) -> DResult<()> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.pg_trans()?
|
|
|
|
.execute(
|
|
|
|
"DELETE FROM corp_membership WHERE \
|
2023-03-25 00:58:19 +11:00
|
|
|
corp_id = $1 AND member_username = $2",
|
2023-06-07 22:38:46 +10:00
|
|
|
&[&corp_id.0, &username.to_lowercase()],
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
2023-03-25 00:58:19 +11:00
|
|
|
}
|
2023-04-24 16:47:08 +10:00
|
|
|
|
2023-06-07 22:38:46 +10:00
|
|
|
pub async fn count_matching_possessions<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
location: &str,
|
|
|
|
allowed_types: &'a [PossessionType],
|
|
|
|
) -> DResult<i64> {
|
2023-04-24 16:47:08 +10:00
|
|
|
Ok(self.pg_trans()?
|
|
|
|
.query_one(
|
|
|
|
"SELECT COUNT(*) FROM items WHERE details->>'location' = $1 AND $2 @> (details->'possession_type')",
|
|
|
|
&[&location, &serde_json::to_value(allowed_types)?]
|
|
|
|
).await?.get(0))
|
|
|
|
}
|
2023-06-07 22:38:46 +10:00
|
|
|
|
2023-06-30 23:46:38 +10:00
|
|
|
pub async fn count_followers_by_leader<'a>(self: &'a Self, leader: &str) -> DResult<i64> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query_one(
|
|
|
|
"SELECT COUNT(*) FROM items WHERE details->'following'->>'follow_whom' = $1",
|
|
|
|
&[&leader],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.get(0))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn find_followers_by_leader<'a>(
|
|
|
|
self: &'a Self,
|
|
|
|
leader: &str,
|
|
|
|
) -> DResult<Vec<Arc<Item>>> {
|
|
|
|
Ok(self
|
|
|
|
.pg_trans()?
|
|
|
|
.query(
|
|
|
|
"SELECT details FROM items WHERE details->'following'->>'follow_whom' = $1",
|
|
|
|
&[&leader],
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|i| serde_json::from_value(i.get("details")).ok())
|
|
|
|
.map(Arc::new)
|
|
|
|
.collect())
|
|
|
|
}
|
|
|
|
|
2022-12-25 00:25:52 +11:00
|
|
|
pub async fn commit(mut self: Self) -> DResult<()> {
|
|
|
|
let trans_opt = self.with_trans_mut(|t| std::mem::replace(t, None));
|
2022-12-27 00:20:09 +11:00
|
|
|
if let Some(trans) = trans_opt {
|
2022-12-25 00:25:52 +11:00
|
|
|
trans.commit().await?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-02-19 14:03:15 +11:00
|
|
|
pub fn pg_trans<'a>(self: &'a Self) -> DResult<&'a Transaction<'a>> {
|
2023-06-07 22:38:46 +10:00
|
|
|
self.borrow_trans()
|
|
|
|
.as_ref()
|
|
|
|
.ok_or("Transaction already closed".into())
|
2022-12-25 00:25:52 +11:00
|
|
|
}
|
|
|
|
}
|