mirror of
https://github.com/Phundrak/georm.git
synced 2025-08-30 22:25:35 +00:00
This commit abstracts the database operations to use the generic `sqlx::Executor` trait instead of a concrete `&sqlx::PgPool`. This change allows all generated methods (find, create, update, delete, and relationships) to be executed within a `sqlx::Transaction`, in addition to a connection pool. This is a crucial feature for ensuring atomic operations and data consistency. The public-facing traits `Georm` and `Defaultable` have been updated to require `sqlx::Executor`, and the documentation has been updated to reflect this new capability.
135 lines
3.2 KiB
Rust
135 lines
3.2 KiB
Rust
use georm::Georm;
|
|
use sqlx::{Postgres, types::BigDecimal};
|
|
|
|
#[derive(Debug, Georm, PartialEq, Eq, Default)]
|
|
#[georm(
|
|
table = "biographies",
|
|
one_to_one = [{
|
|
name = "author", remote_id = "biography_id", table = "authors", entity = Author
|
|
}]
|
|
)]
|
|
pub struct Biography {
|
|
#[georm(id)]
|
|
pub id: i32,
|
|
pub content: String,
|
|
}
|
|
|
|
#[derive(Debug, Georm, PartialEq, Eq, Default)]
|
|
#[georm(table = "authors")]
|
|
pub struct Author {
|
|
#[georm(id)]
|
|
pub id: i32,
|
|
pub name: String,
|
|
#[georm(relation = {entity = Biography, table = "biographies", name = "biography", nullable = true})]
|
|
pub biography_id: Option<i32>,
|
|
}
|
|
|
|
impl PartialOrd for Author {
|
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
Some(self.id.cmp(&other.id))
|
|
}
|
|
}
|
|
|
|
impl Ord for Author {
|
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
|
self.id.cmp(&other.id)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Georm, PartialEq, Eq, Default)]
|
|
#[georm(
|
|
table = "books",
|
|
one_to_many = [
|
|
{ name = "reviews", remote_id = "book_id", table = "reviews", entity = Review }
|
|
],
|
|
many_to_many = [{
|
|
name = "genres",
|
|
table = "genres",
|
|
entity = Genre,
|
|
link = { table = "book_genres", from = "book_id", to = "genre_id" }
|
|
}]
|
|
)]
|
|
pub struct Book {
|
|
#[georm(id)]
|
|
ident: i32,
|
|
title: String,
|
|
#[georm(relation = {entity = Author, table = "authors", name = "author"})]
|
|
author_id: i32,
|
|
}
|
|
|
|
impl PartialOrd for Book {
|
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
Some(self.ident.cmp(&other.ident))
|
|
}
|
|
}
|
|
|
|
impl Ord for Book {
|
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
|
self.ident.cmp(&other.ident)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Georm, PartialEq, Eq)]
|
|
#[georm(table = "reviews")]
|
|
pub struct Review {
|
|
#[georm(id)]
|
|
pub id: i32,
|
|
#[georm(relation = {entity = Book, table = "books", remote_id = "ident", name = "book"})]
|
|
pub book_id: i32,
|
|
pub review: String,
|
|
}
|
|
|
|
#[derive(Debug, Georm, PartialEq, Eq)]
|
|
#[georm(
|
|
table = "genres",
|
|
many_to_many = [{
|
|
name = "books",
|
|
table = "books",
|
|
entity = Book,
|
|
remote_id = "ident",
|
|
link = { table = "book_genres", from = "genre_id", to = "book_id" }
|
|
}]
|
|
)]
|
|
pub struct Genre {
|
|
#[georm(id)]
|
|
id: i32,
|
|
name: String,
|
|
}
|
|
|
|
#[derive(Debug, Georm, PartialEq, Eq, Default)]
|
|
#[georm(table = "UserRoles")]
|
|
pub struct UserRole {
|
|
#[georm(id)]
|
|
pub user_id: i32,
|
|
#[georm(id)]
|
|
pub role_id: i32,
|
|
#[georm(defaultable)]
|
|
pub assigned_at: chrono::DateTime<chrono::Utc>,
|
|
}
|
|
|
|
#[derive(Debug, Georm, PartialEq, Default, Clone)]
|
|
#[georm(table = "products")]
|
|
pub struct Product {
|
|
#[georm(id, generated_always)]
|
|
pub id: i32,
|
|
#[georm(generated)]
|
|
pub sku_number: i32,
|
|
pub name: String,
|
|
pub price: BigDecimal,
|
|
pub discount_percent: i32,
|
|
#[georm(generated_always)]
|
|
pub final_price: Option<BigDecimal>, // Apparently this can be null ?
|
|
}
|
|
|
|
impl Product {
|
|
#[allow(dead_code)]
|
|
pub async fn find_by_name<'e, E>(name: String, executor: E) -> ::sqlx::Result<Self>
|
|
where
|
|
E: sqlx::Executor<'e, Database = Postgres>,
|
|
{
|
|
::sqlx::query_as!(Self, "SELECT * FROM products WHERE name = $1", name)
|
|
.fetch_one(executor)
|
|
.await
|
|
}
|
|
}
|