summaryrefslogtreecommitdiff
path: root/src/channel
diff options
context:
space:
mode:
Diffstat (limited to 'src/channel')
-rw-r--r--src/channel/mod.rs4
-rw-r--r--src/channel/repo.rs108
-rw-r--r--src/channel/routes.rs32
3 files changed, 144 insertions, 0 deletions
diff --git a/src/channel/mod.rs b/src/channel/mod.rs
new file mode 100644
index 0000000..238e116
--- /dev/null
+++ b/src/channel/mod.rs
@@ -0,0 +1,4 @@
+pub mod repo;
+mod routes;
+
+pub use self::routes::router;
diff --git a/src/channel/repo.rs b/src/channel/repo.rs
new file mode 100644
index 0000000..e6a5e5c
--- /dev/null
+++ b/src/channel/repo.rs
@@ -0,0 +1,108 @@
+use std::fmt;
+
+use sqlx::{sqlite::Sqlite, SqliteConnection, Transaction};
+
+use crate::id::Id as BaseId;
+use crate::{error::BoxedError, login::repo::logins::Id as LoginId};
+
+pub trait Provider {
+ fn channels(&mut self) -> Channels;
+}
+
+impl<'c> Provider for Transaction<'c, Sqlite> {
+ fn channels(&mut self) -> Channels {
+ Channels(self)
+ }
+}
+
+pub struct Channels<'t>(&'t mut SqliteConnection);
+
+#[derive(Debug)]
+pub struct Channel {
+ pub id: Id,
+ pub name: String,
+}
+
+impl<'c> Channels<'c> {
+ /// Create a new channel.
+ pub async fn create(&mut self, name: &str) -> Result<Channel, BoxedError> {
+ let id = Id::generate();
+
+ let channel = sqlx::query_as!(
+ Channel,
+ r#"
+ insert
+ into channel (id, name)
+ values ($1, $2)
+ returning id as "id: Id", name
+ "#,
+ id,
+ name,
+ )
+ .fetch_one(&mut *self.0)
+ .await?;
+
+ Ok(channel)
+ }
+
+ /// Enrol a login in a channel.
+ pub async fn join(&mut self, channel: &Id, login: &LoginId) -> Result<(), BoxedError> {
+ sqlx::query!(
+ r#"
+ insert
+ into channel_member (channel, login)
+ values ($1, $2)
+ "#,
+ channel,
+ login,
+ )
+ .execute(&mut *self.0)
+ .await?;
+
+ Ok(())
+ }
+
+ pub async fn for_login(&mut self, login: &LoginId) -> Result<Vec<Channel>, BoxedError> {
+ let channels = sqlx::query_as!(
+ Channel,
+ r#"
+ select
+ channel.id as "id: Id",
+ channel.name
+ from channel
+ join channel_member
+ on (channel.id = channel_member.channel)
+ where channel_member.login = $1
+ order by channel.name
+ "#,
+ login,
+ )
+ .fetch_all(&mut *self.0)
+ .await?;
+
+ Ok(channels)
+ }
+}
+
+/// Stable identifier for a [Channel]. Prefixed with `C`.
+#[derive(Debug, sqlx::Type)]
+#[sqlx(transparent)]
+pub struct Id(BaseId);
+
+impl From<BaseId> for Id {
+ fn from(id: BaseId) -> Self {
+ Self(id)
+ }
+}
+
+impl Id {
+ pub fn generate() -> Self {
+ BaseId::generate("C")
+ }
+}
+
+impl fmt::Display for Id {
+ fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
+ self.0.fmt(f)
+ }
+}
diff --git a/src/channel/routes.rs b/src/channel/routes.rs
new file mode 100644
index 0000000..c8d6c3f
--- /dev/null
+++ b/src/channel/routes.rs
@@ -0,0 +1,32 @@
+use axum::{
+ extract::{Form, State},
+ response::{IntoResponse, Redirect},
+ routing::post,
+ Router,
+};
+use sqlx::sqlite::SqlitePool;
+
+use super::repo::Provider as _;
+use crate::{error::InternalError, login::repo::logins::Login};
+
+pub fn router() -> Router<SqlitePool> {
+ Router::new().route("/create", post(on_create))
+}
+
+#[derive(serde::Deserialize)]
+struct CreateRequest {
+ name: String,
+}
+
+async fn on_create(
+ State(db): State<SqlitePool>,
+ login: Login,
+ Form(form): Form<CreateRequest>,
+) -> Result<impl IntoResponse, InternalError> {
+ let mut tx = db.begin().await?;
+ let channel = tx.channels().create(&form.name).await?;
+ tx.channels().join(&channel.id, &login.id).await?;
+ tx.commit().await?;
+
+ Ok(Redirect::to("/"))
+}