mirror of
https://github.com/alvierahman90/rabbit.git
synced 2025-01-26 02:23:24 +00:00
Compare commits
2 Commits
53208b73f9
...
088d3bc0a0
Author | SHA1 | Date | |
---|---|---|---|
088d3bc0a0 | |||
c3ae3e91e4 |
@ -1,11 +0,0 @@
|
|||||||
install:
|
|
||||||
- appveyor-retry appveyor DownloadFile https://win.rustup.rs/ -FileName rustup-init.exe
|
|
||||||
- if not defined RUSTFLAGS rustup-init.exe -y --default-host x86_64-pc-windows-msvc --default-toolchain nightly
|
|
||||||
- set PATH=%PATH%;C:\Users\appveyor\.cargo\bin
|
|
||||||
- rustc -V
|
|
||||||
- cargo -V
|
|
||||||
|
|
||||||
build: false
|
|
||||||
|
|
||||||
test_script:
|
|
||||||
- cargo test --locked
|
|
8
.github/dependabot.yml
vendored
8
.github/dependabot.yml
vendored
@ -1,8 +0,0 @@
|
|||||||
version: 2
|
|
||||||
updates:
|
|
||||||
- package-ecosystem: cargo
|
|
||||||
directory: "/"
|
|
||||||
schedule:
|
|
||||||
interval: daily
|
|
||||||
time: "08:00"
|
|
||||||
open-pull-requests-limit: 10
|
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -4,3 +4,4 @@ Cargo.lock
|
|||||||
bin/
|
bin/
|
||||||
pkg/
|
pkg/
|
||||||
wasm-pack.log
|
wasm-pack.log
|
||||||
|
.env
|
||||||
|
69
.travis.yml
69
.travis.yml
@ -1,69 +0,0 @@
|
|||||||
language: rust
|
|
||||||
sudo: false
|
|
||||||
|
|
||||||
cache: cargo
|
|
||||||
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
|
|
||||||
# Builds with wasm-pack.
|
|
||||||
- rust: beta
|
|
||||||
env: RUST_BACKTRACE=1
|
|
||||||
addons:
|
|
||||||
firefox: latest
|
|
||||||
chrome: stable
|
|
||||||
before_script:
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-install-update || cargo install cargo-update)
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-generate || cargo install --vers "^0.2" cargo-generate)
|
|
||||||
- cargo install-update -a
|
|
||||||
- curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh -s -- -f
|
|
||||||
script:
|
|
||||||
- cargo generate --git . --name testing
|
|
||||||
# Having a broken Cargo.toml (in that it has curlies in fields) anywhere
|
|
||||||
# in any of our parent dirs is problematic.
|
|
||||||
- mv Cargo.toml Cargo.toml.tmpl
|
|
||||||
- cd testing
|
|
||||||
- wasm-pack build
|
|
||||||
- wasm-pack test --chrome --firefox --headless
|
|
||||||
|
|
||||||
# Builds on nightly.
|
|
||||||
- rust: nightly
|
|
||||||
env: RUST_BACKTRACE=1
|
|
||||||
before_script:
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-install-update || cargo install cargo-update)
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-generate || cargo install --vers "^0.2" cargo-generate)
|
|
||||||
- cargo install-update -a
|
|
||||||
- rustup target add wasm32-unknown-unknown
|
|
||||||
script:
|
|
||||||
- cargo generate --git . --name testing
|
|
||||||
- mv Cargo.toml Cargo.toml.tmpl
|
|
||||||
- cd testing
|
|
||||||
- cargo check
|
|
||||||
- cargo check --target wasm32-unknown-unknown
|
|
||||||
- cargo check --no-default-features
|
|
||||||
- cargo check --target wasm32-unknown-unknown --no-default-features
|
|
||||||
- cargo check --no-default-features --features console_error_panic_hook
|
|
||||||
- cargo check --target wasm32-unknown-unknown --no-default-features --features console_error_panic_hook
|
|
||||||
- cargo check --no-default-features --features "console_error_panic_hook wee_alloc"
|
|
||||||
- cargo check --target wasm32-unknown-unknown --no-default-features --features "console_error_panic_hook wee_alloc"
|
|
||||||
|
|
||||||
# Builds on beta.
|
|
||||||
- rust: beta
|
|
||||||
env: RUST_BACKTRACE=1
|
|
||||||
before_script:
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-install-update || cargo install cargo-update)
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-generate || cargo install --vers "^0.2" cargo-generate)
|
|
||||||
- cargo install-update -a
|
|
||||||
- rustup target add wasm32-unknown-unknown
|
|
||||||
script:
|
|
||||||
- cargo generate --git . --name testing
|
|
||||||
- mv Cargo.toml Cargo.toml.tmpl
|
|
||||||
- cd testing
|
|
||||||
- cargo check
|
|
||||||
- cargo check --target wasm32-unknown-unknown
|
|
||||||
- cargo check --no-default-features
|
|
||||||
- cargo check --target wasm32-unknown-unknown --no-default-features
|
|
||||||
- cargo check --no-default-features --features console_error_panic_hook
|
|
||||||
- cargo check --target wasm32-unknown-unknown --no-default-features --features console_error_panic_hook
|
|
||||||
# Note: no enabling the `wee_alloc` feature here because it requires
|
|
||||||
# nightly for now.
|
|
22
Cargo.toml
22
Cargo.toml
@ -4,28 +4,12 @@ version = "0.1.0"
|
|||||||
authors = ["Akbar Rahman <hi@alv.cx>"]
|
authors = ["Akbar Rahman <hi@alv.cx>"]
|
||||||
edition = "2018"
|
edition = "2018"
|
||||||
|
|
||||||
[lib]
|
|
||||||
crate-type = ["cdylib", "rlib"]
|
|
||||||
|
|
||||||
[features]
|
|
||||||
default = ["console_error_panic_hook"]
|
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
wasm-bindgen = "0.2.84"
|
|
||||||
chrono = { version = "0.4.31", features = [ "clock", "serde" ] }
|
chrono = { version = "0.4.31", features = [ "clock", "serde" ] }
|
||||||
|
diesel = { version = "2.1.4", features = ["postgres", "chrono" ] }
|
||||||
|
dotenvy = "0.15"
|
||||||
serde = { version = "1.0", features = [ "derive" ] }
|
serde = { version = "1.0", features = [ "derive" ] }
|
||||||
serde_json = { version = "1.0", features = [ "std" ] }
|
serde_json = { version = "1.0", features = [ "std" ] }
|
||||||
serde_with = { version = "3.4.0", features = [ "std", "chrono_0_4", "json" ] }
|
serde_with = { version = "3.4.0", features = [ "std", "chrono_0_4", "json" ] }
|
||||||
|
|
||||||
# The `console_error_panic_hook` crate provides better debugging of panics by
|
rocket = { version = "0.5", features = [ "json" ] }
|
||||||
# logging them with `console.error`. This is great for development, but requires
|
|
||||||
# all the `std::fmt` and `std::panicking` infrastructure, so isn't great for
|
|
||||||
# code size when deploying.
|
|
||||||
console_error_panic_hook = { version = "0.1.7", optional = true }
|
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
wasm-bindgen-test = "0.3.34"
|
|
||||||
|
|
||||||
[profile.release]
|
|
||||||
# Tell `rustc` to optimize for small code size.
|
|
||||||
opt-level = "s"
|
|
||||||
|
27
compose.yml
Normal file
27
compose.yml
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
services:
|
||||||
|
db:
|
||||||
|
image: postgres
|
||||||
|
#volumes:
|
||||||
|
#- ./dbdata:/var/lib/postgresql/data
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
environment:
|
||||||
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
||||||
|
POSTGRES_USER: ${POSTGRES_USERNAME}
|
||||||
|
POSTGRES_DB: ${POSTGRES_DB}
|
||||||
|
healthcheck:
|
||||||
|
test: [ "CMD", "pg_isready" ]
|
||||||
|
interval: 10s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 5
|
||||||
|
|
||||||
|
pgweb:
|
||||||
|
image: sosedoff/pgweb
|
||||||
|
ports:
|
||||||
|
- 8081:8081
|
||||||
|
environment:
|
||||||
|
PGWEB_DATABASE_URL: postgres://${POSTGRES_USERNAME}:${POSTGRES_PASSWORD}@db/${POSTGRES_DB}?sslmode=disable
|
||||||
|
depends_on:
|
||||||
|
db:
|
||||||
|
condition: service_healthy
|
||||||
|
restart: true
|
9
diesel.toml
Normal file
9
diesel.toml
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
# For documentation on how to configure this file,
|
||||||
|
# see https://diesel.rs/guides/configuring-diesel-cli
|
||||||
|
|
||||||
|
[print_schema]
|
||||||
|
file = "src/schema.rs"
|
||||||
|
custom_type_derives = ["diesel::query_builder::QueryId"]
|
||||||
|
|
||||||
|
[migrations_directory]
|
||||||
|
dir = "migrations"
|
0
migrations/.keep
Normal file
0
migrations/.keep
Normal file
6
migrations/00000000000000_diesel_initial_setup/down.sql
Normal file
6
migrations/00000000000000_diesel_initial_setup/down.sql
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
-- This file was automatically created by Diesel to setup helper functions
|
||||||
|
-- and other internal bookkeeping. This file is safe to edit, any future
|
||||||
|
-- changes will be added to existing projects as new migrations.
|
||||||
|
|
||||||
|
DROP FUNCTION IF EXISTS diesel_manage_updated_at(_tbl regclass);
|
||||||
|
DROP FUNCTION IF EXISTS diesel_set_updated_at();
|
36
migrations/00000000000000_diesel_initial_setup/up.sql
Normal file
36
migrations/00000000000000_diesel_initial_setup/up.sql
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
-- This file was automatically created by Diesel to setup helper functions
|
||||||
|
-- and other internal bookkeeping. This file is safe to edit, any future
|
||||||
|
-- changes will be added to existing projects as new migrations.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- Sets up a trigger for the given table to automatically set a column called
|
||||||
|
-- `updated_at` whenever the row is modified (unless `updated_at` was included
|
||||||
|
-- in the modified columns)
|
||||||
|
--
|
||||||
|
-- # Example
|
||||||
|
--
|
||||||
|
-- ```sql
|
||||||
|
-- CREATE TABLE users (id SERIAL PRIMARY KEY, updated_at TIMESTAMP NOT NULL DEFAULT NOW());
|
||||||
|
--
|
||||||
|
-- SELECT diesel_manage_updated_at('users');
|
||||||
|
-- ```
|
||||||
|
CREATE OR REPLACE FUNCTION diesel_manage_updated_at(_tbl regclass) RETURNS VOID AS $$
|
||||||
|
BEGIN
|
||||||
|
EXECUTE format('CREATE TRIGGER set_updated_at BEFORE UPDATE ON %s
|
||||||
|
FOR EACH ROW EXECUTE PROCEDURE diesel_set_updated_at()', _tbl);
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION diesel_set_updated_at() RETURNS trigger AS $$
|
||||||
|
BEGIN
|
||||||
|
IF (
|
||||||
|
NEW IS DISTINCT FROM OLD AND
|
||||||
|
NEW.updated_at IS NOT DISTINCT FROM OLD.updated_at
|
||||||
|
) THEN
|
||||||
|
NEW.updated_at := current_timestamp;
|
||||||
|
END IF;
|
||||||
|
RETURN NEW;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
1
migrations/2023-12-22-135046_create_users/down.sql
Normal file
1
migrations/2023-12-22-135046_create_users/down.sql
Normal file
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE users;
|
5
migrations/2023-12-22-135046_create_users/up.sql
Normal file
5
migrations/2023-12-22-135046_create_users/up.sql
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
CREATE TABLE users (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR NOT NULL,
|
||||||
|
email VARCHAR NOT NULL
|
||||||
|
);
|
1
migrations/2023-12-22-135650_create_categories/down.sql
Normal file
1
migrations/2023-12-22-135650_create_categories/down.sql
Normal file
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE categories;
|
5
migrations/2023-12-22-135650_create_categories/up.sql
Normal file
5
migrations/2023-12-22-135650_create_categories/up.sql
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
CREATE TABLE categories (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR NOT NULL,
|
||||||
|
user_id SERIAL REFERENCES users(id)
|
||||||
|
);
|
1
migrations/2023-12-22-135844_create_series/down.sql
Normal file
1
migrations/2023-12-22-135844_create_series/down.sql
Normal file
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE series;
|
7
migrations/2023-12-22-135844_create_series/up.sql
Normal file
7
migrations/2023-12-22-135844_create_series/up.sql
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
CREATE TABLE series (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR NOT NULL,
|
||||||
|
repeat INTEGER NOT NULL,
|
||||||
|
good BOOLEAN NOT NULL,
|
||||||
|
category_id SERIAL REFERENCES categories(id)
|
||||||
|
);
|
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE series_points;
|
6
migrations/2023-12-22-141631_create_series_points/up.sql
Normal file
6
migrations/2023-12-22-141631_create_series_points/up.sql
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
CREATE TABLE series_points (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
timestamp TIMESTAMP WITHOUT TIME ZONE NOT NULL,
|
||||||
|
value INTEGER NOT NULL,
|
||||||
|
series_id SERIAL REFERENCES series(id)
|
||||||
|
);
|
@ -4,10 +4,8 @@
|
|||||||
|
|
||||||
## Why
|
## Why
|
||||||
|
|
||||||
- Cross platform (just a web app)
|
- Simple
|
||||||
- Flexible (tracks whatever)
|
- Flexible (tracks whatever)
|
||||||
- No sign up required
|
|
||||||
- Can sign up if you want
|
|
||||||
- Self-hostable
|
- Self-hostable
|
||||||
- CalDAV reminders and calendars to remind you to build habits
|
- CalDAV reminders and calendars to remind you to build habits
|
||||||
- Rust :crab: :muscle:
|
- Rust :crab: :muscle:
|
||||||
|
30
src/db.rs
30
src/db.rs
@ -1,4 +1,5 @@
|
|||||||
pub mod json;
|
//pub mod json;
|
||||||
|
pub mod pg;
|
||||||
|
|
||||||
use crate::models::*;
|
use crate::models::*;
|
||||||
use serde_json;
|
use serde_json;
|
||||||
@ -9,6 +10,8 @@ pub enum Error {
|
|||||||
Generic(String),
|
Generic(String),
|
||||||
FsIo(String),
|
FsIo(String),
|
||||||
Json(String),
|
Json(String),
|
||||||
|
PgDb(String),
|
||||||
|
Parsing(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<io::Error> for Error {
|
impl From<io::Error> for Error {
|
||||||
@ -23,19 +26,22 @@ impl From<serde_json::Error> for Error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<diesel::result::Error> for Error {
|
||||||
|
fn from(e: diesel::result::Error) -> Self {
|
||||||
|
Self::PgDb(e.to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub trait Storage {
|
pub trait Storage {
|
||||||
|
fn new() -> Self;
|
||||||
fn add_category(&mut self, category: NewCategory) -> Result<i32, Error>;
|
fn add_category(&mut self, category: NewCategory) -> Result<i32, Error>;
|
||||||
fn add_series(&mut self, category_id: i32, series: NewSeries) -> Result<i32, Error>;
|
fn add_series(&mut self, series: NewSeries) -> Result<i32, Error>;
|
||||||
fn add_series_point(
|
fn add_series_point(&mut self, series_point: NewSeriesPoint) -> Result<i32, Error>;
|
||||||
&mut self,
|
fn add_user(&mut self, user: NewUser) -> Result<i32, Error>;
|
||||||
category_id: i32,
|
fn get_categories(&mut self, filter: CategoryFilter) -> Result<Vec<Category>, Error>;
|
||||||
series_id: i32,
|
fn get_series(&mut self, filter: SeriesFilter) -> Result<Vec<Series>, Error>;
|
||||||
series_point: NewSeriesPoint,
|
fn get_series_points(&mut self, filter: SeriesPointFilter) -> Result<Vec<SeriesPoint>, Error>;
|
||||||
) -> Result<i32, Error>;
|
fn get_users(&mut self, filter: UserFilter) -> Result<Vec<User>, Error>;
|
||||||
//fn add_user(&mut self, id: i32, user: NewUser) -> Result<User, Error>;
|
|
||||||
fn get_category(&self, id: i32) -> Option<Category>;
|
|
||||||
fn get_series(&self, category_id: i32, series_id: i32) -> Option<Series>;
|
|
||||||
//fn get_user(&self, id: i32) -> Option<User>;
|
|
||||||
//fn update_category(&mut self, id: i32, changeset: CategoryChangeset) -> Result<(), Error>;
|
//fn update_category(&mut self, id: i32, changeset: CategoryChangeset) -> Result<(), Error>;
|
||||||
//fn update_series(&mut self, id: i32, changeset: SeriesChangeset) -> Result<(), Error>;
|
//fn update_series(&mut self, id: i32, changeset: SeriesChangeset) -> Result<(), Error>;
|
||||||
//fn update_user(&mut self, id: i32, changeset: UserChangeset) -> Result<(), Error>;
|
//fn update_user(&mut self, id: i32, changeset: UserChangeset) -> Result<(), Error>;
|
||||||
|
@ -6,15 +6,6 @@ pub struct JsonDb {
|
|||||||
|
|
||||||
/// JsonDb is single user.
|
/// JsonDb is single user.
|
||||||
impl JsonDb {
|
impl JsonDb {
|
||||||
pub fn new(value: Option<String>) -> Self {
|
|
||||||
match value {
|
|
||||||
Some(value) => JsonDb { value },
|
|
||||||
None => JsonDb {
|
|
||||||
value: String::from("[]"),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn load(&self) -> Result<Vec<Category>, Error> {
|
fn load(&self) -> Result<Vec<Category>, Error> {
|
||||||
match serde_json::from_str::<Vec<Category>>(&self.value) {
|
match serde_json::from_str::<Vec<Category>>(&self.value) {
|
||||||
Ok(d) => Ok(d),
|
Ok(d) => Ok(d),
|
||||||
@ -35,6 +26,12 @@ impl JsonDb {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Storage for JsonDb {
|
impl Storage for JsonDb {
|
||||||
|
fn new() -> Self {
|
||||||
|
JsonDb {
|
||||||
|
value: String::from("[]"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn add_category(&mut self, category: NewCategory) -> Result<i32, Error> {
|
fn add_category(&mut self, category: NewCategory) -> Result<i32, Error> {
|
||||||
let mut max_id = 1;
|
let mut max_id = 1;
|
||||||
let mut data = self.load()?;
|
let mut data = self.load()?;
|
||||||
|
181
src/db/pg.rs
Normal file
181
src/db/pg.rs
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
use super::{Error, Storage};
|
||||||
|
use crate::{models::*, schema};
|
||||||
|
use diesel::pg::PgConnection;
|
||||||
|
use diesel::prelude::*;
|
||||||
|
use dotenvy::dotenv;
|
||||||
|
use std::env;
|
||||||
|
|
||||||
|
pub struct PgDb {
|
||||||
|
conn: PgConnection,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Storage for PgDb {
|
||||||
|
fn new() -> Self {
|
||||||
|
dotenv().ok();
|
||||||
|
|
||||||
|
let database_url = env::var("DATABASE_URL").expect("DATABASE_URL must be set");
|
||||||
|
Self {
|
||||||
|
conn: PgConnection::establish(&database_url)
|
||||||
|
.unwrap_or_else(|_| panic!("Error connecting to {}", database_url)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_user(&mut self, user: NewUser) -> Result<i32, Error> {
|
||||||
|
use schema::users;
|
||||||
|
let ret = diesel::insert_into(users::table)
|
||||||
|
.values(&user)
|
||||||
|
.returning(users::dsl::id)
|
||||||
|
.get_result(&mut self.conn);
|
||||||
|
|
||||||
|
match ret {
|
||||||
|
Ok(val) => Ok(val),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_category(&mut self, category: NewCategory) -> Result<i32, Error> {
|
||||||
|
use schema::categories;
|
||||||
|
let ret = diesel::insert_into(categories::table)
|
||||||
|
.values(&category)
|
||||||
|
.returning(categories::dsl::id)
|
||||||
|
.get_result(&mut self.conn);
|
||||||
|
|
||||||
|
match ret {
|
||||||
|
Ok(val) => Ok(val),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_series(&mut self, series: NewSeries) -> Result<i32, Error> {
|
||||||
|
use schema::series;
|
||||||
|
let ret = diesel::insert_into(series::table)
|
||||||
|
.values(&series)
|
||||||
|
.returning(series::dsl::id)
|
||||||
|
.get_result(&mut self.conn);
|
||||||
|
|
||||||
|
match ret {
|
||||||
|
Ok(val) => Ok(val),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_series_point(&mut self, series_point: NewSeriesPoint) -> Result<i32, Error> {
|
||||||
|
use schema::series_points;
|
||||||
|
let ret = diesel::insert_into(series_points::table)
|
||||||
|
.values(&series_point)
|
||||||
|
.returning(series_points::dsl::id)
|
||||||
|
.get_result(&mut self.conn);
|
||||||
|
|
||||||
|
match ret {
|
||||||
|
Ok(val) => Ok(val),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_categories(&mut self, filter: CategoryFilter) -> Result<Vec<Category>, Error> {
|
||||||
|
use schema::categories;
|
||||||
|
|
||||||
|
let mut query = categories::table.into_boxed();
|
||||||
|
|
||||||
|
if let Some(val) = filter.id {
|
||||||
|
query = query.filter(categories::id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.name {
|
||||||
|
query = query.filter(categories::name.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.user_id {
|
||||||
|
query = query.filter(categories::user_id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
match query.select(Category::as_select()).load(&mut self.conn) {
|
||||||
|
Ok(q) => Ok(q),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_series(&mut self, filter: SeriesFilter) -> Result<Vec<Series>, Error> {
|
||||||
|
use schema::series;
|
||||||
|
|
||||||
|
let mut query = series::table.into_boxed();
|
||||||
|
|
||||||
|
if let Some(val) = filter.id {
|
||||||
|
query = query.filter(series::id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.name {
|
||||||
|
query = query.filter(series::name.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.repeat {
|
||||||
|
query = query.filter(series::repeat.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.good {
|
||||||
|
query = query.filter(series::good.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.category_id {
|
||||||
|
query = query.filter(series::category_id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
match query.select(Series::as_select()).load(&mut self.conn) {
|
||||||
|
Ok(q) => Ok(q),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_series_points(&mut self, filter: SeriesPointFilter) -> Result<Vec<SeriesPoint>, Error> {
|
||||||
|
use schema::series_points;
|
||||||
|
|
||||||
|
let mut query = series_points::table.into_boxed();
|
||||||
|
|
||||||
|
if let Some(val) = filter.id {
|
||||||
|
query = query.filter(series_points::id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.timestamp_millis {
|
||||||
|
match chrono::NaiveDateTime::from_timestamp_millis(val) {
|
||||||
|
Some(val) => query = query.filter(series_points::timestamp.eq(val)),
|
||||||
|
_ => return Err(Error::Parsing("Failed to parse timestamp".to_owned())),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.value {
|
||||||
|
query = query.filter(series_points::value.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.series_id {
|
||||||
|
query = query.filter(series_points::series_id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
match query.select(SeriesPoint::as_select()).load(&mut self.conn) {
|
||||||
|
Ok(q) => Ok(q),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_users(&mut self, filter: UserFilter) -> Result<Vec<User>, Error> {
|
||||||
|
use schema::users;
|
||||||
|
|
||||||
|
let mut query = users::table.into_boxed();
|
||||||
|
|
||||||
|
if let Some(val) = filter.id {
|
||||||
|
query = query.filter(users::id.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.name {
|
||||||
|
query = query.filter(users::name.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = filter.email {
|
||||||
|
query = query.filter(users::email.eq(val));
|
||||||
|
}
|
||||||
|
|
||||||
|
match query.select(User::as_select()).load(&mut self.conn) {
|
||||||
|
Ok(q) => Ok(q),
|
||||||
|
Err(e) => Err(Error::from(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,3 @@
|
|||||||
pub mod db;
|
pub mod db;
|
||||||
pub mod models;
|
pub mod models;
|
||||||
mod utils;
|
pub mod schema;
|
||||||
|
|
||||||
use wasm_bindgen::prelude::*;
|
|
||||||
|
@ -1,11 +1,9 @@
|
|||||||
mod category;
|
mod category;
|
||||||
mod series;
|
mod series;
|
||||||
mod series_point;
|
mod series_point;
|
||||||
mod series_type;
|
|
||||||
mod user;
|
mod user;
|
||||||
|
|
||||||
pub use category::*;
|
pub use category::*;
|
||||||
pub use series::*;
|
pub use series::*;
|
||||||
pub use series_point::*;
|
pub use series_point::*;
|
||||||
pub use series_type::*;
|
|
||||||
pub use user::*;
|
pub use user::*;
|
||||||
|
@ -1,33 +1,34 @@
|
|||||||
use crate::models::Series;
|
use crate::schema::categories;
|
||||||
|
use diesel;
|
||||||
|
use diesel::prelude::*;
|
||||||
|
use rocket::form::FromForm;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize, Queryable, Selectable, Identifiable, PartialEq)]
|
||||||
|
#[diesel(table_name = categories)]
|
||||||
|
#[diesel(check_for_backend(diesel::pg::Pg))]
|
||||||
pub struct Category {
|
pub struct Category {
|
||||||
pub id: i32,
|
pub id: i32,
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub series: Vec<Series>,
|
pub user_id: i32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Category {
|
impl Category {}
|
||||||
pub fn new(id: i32, new: NewCategory) -> Category {
|
|
||||||
Category {
|
|
||||||
id,
|
|
||||||
name: new.name,
|
|
||||||
series: new.series,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn add_series(&mut self, series: Series) {
|
#[derive(FromForm)]
|
||||||
self.series.push(series);
|
pub struct CategoryFilter {
|
||||||
}
|
pub id: Option<i32>,
|
||||||
|
pub name: Option<String>,
|
||||||
|
pub user_id: Option<i32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct CategoryChangeset {
|
pub struct CategoryChangeset {
|
||||||
pub name: Option<String>,
|
pub name: Option<String>,
|
||||||
pub series: Option<Vec<Series>>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable, Deserialize, FromForm)]
|
||||||
|
#[diesel(table_name = categories)]
|
||||||
pub struct NewCategory {
|
pub struct NewCategory {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub series: Vec<Series>,
|
pub user_id: i32,
|
||||||
}
|
}
|
||||||
|
@ -1,45 +1,45 @@
|
|||||||
use super::series_point::SeriesPoint;
|
use crate::schema::series;
|
||||||
use chrono;
|
use diesel;
|
||||||
|
use diesel::prelude::*;
|
||||||
|
use rocket::form::FromForm;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use serde_with;
|
use serde_with;
|
||||||
|
|
||||||
#[serde_with::serde_as]
|
#[serde_with::serde_as]
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize, Queryable, Selectable, Identifiable, PartialEq)]
|
||||||
|
#[diesel(table_name = series)]
|
||||||
|
#[diesel(check_for_backend(diesel::pg::Pg))]
|
||||||
pub struct Series {
|
pub struct Series {
|
||||||
pub id: i32,
|
pub id: i32,
|
||||||
pub name: String,
|
pub name: String,
|
||||||
#[serde_as(as = "serde_with::DurationSeconds<i64>")]
|
pub repeat: i32,
|
||||||
pub repeat: chrono::Duration,
|
|
||||||
pub good: bool,
|
pub good: bool,
|
||||||
pub points: Vec<SeriesPoint>,
|
pub category_id: i32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Series {
|
#[derive(FromForm)]
|
||||||
pub fn new(id: i32, series: NewSeries) -> Series {
|
pub struct SeriesFilter {
|
||||||
Series {
|
pub id: Option<i32>,
|
||||||
id,
|
pub name: Option<String>,
|
||||||
name: series.name,
|
pub repeat: Option<i32>,
|
||||||
repeat: series.repeat,
|
pub good: Option<bool>,
|
||||||
good: series.good,
|
pub category_id: Option<i32>,
|
||||||
points: series.points,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn add_point(&mut self, point: SeriesPoint) {
|
|
||||||
self.points.push(point);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(AsChangeset)]
|
||||||
|
#[diesel(table_name = series)]
|
||||||
pub struct SeriesChangeset {
|
pub struct SeriesChangeset {
|
||||||
pub name: Option<String>,
|
pub name: Option<String>,
|
||||||
pub repeat: Option<chrono::Duration>,
|
pub repeat: Option<i32>,
|
||||||
pub good: Option<bool>,
|
pub good: Option<bool>,
|
||||||
pub points: Option<Vec<SeriesPoint>>,
|
pub category_id: Option<i32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable, Deserialize)]
|
||||||
|
#[diesel(table_name = series)]
|
||||||
pub struct NewSeries {
|
pub struct NewSeries {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub repeat: chrono::Duration,
|
pub repeat: i32,
|
||||||
pub good: bool,
|
pub good: bool,
|
||||||
pub points: Vec<SeriesPoint>,
|
pub category_id: i32,
|
||||||
}
|
}
|
||||||
|
@ -1,25 +1,32 @@
|
|||||||
use super::series_type::SeriesType;
|
use crate::schema::series_points;
|
||||||
use chrono;
|
use chrono;
|
||||||
|
use diesel;
|
||||||
|
use diesel::prelude::*;
|
||||||
|
use rocket::form::FromForm;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize, Queryable, Selectable, Identifiable, PartialEq)]
|
||||||
|
#[diesel(table_name = series_points)]
|
||||||
|
#[diesel(check_for_backend(diesel::pg::Pg))]
|
||||||
pub struct SeriesPoint {
|
pub struct SeriesPoint {
|
||||||
pub id: i32,
|
pub id: i32,
|
||||||
pub timestamp: chrono::NaiveDateTime,
|
pub timestamp: chrono::NaiveDateTime,
|
||||||
pub value: SeriesType,
|
pub value: i32,
|
||||||
|
pub series_id: i32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SeriesPoint {
|
#[derive(FromForm)]
|
||||||
pub fn new(id: i32, new: NewSeriesPoint) -> SeriesPoint {
|
pub struct SeriesPointFilter {
|
||||||
SeriesPoint {
|
pub id: Option<i32>,
|
||||||
id,
|
pub timestamp_millis: Option<i64>,
|
||||||
timestamp: new.timestamp,
|
pub value: Option<i32>,
|
||||||
value: new.value,
|
pub series_id: Option<i32>,
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable, Deserialize)]
|
||||||
|
#[diesel(table_name = series_points)]
|
||||||
pub struct NewSeriesPoint {
|
pub struct NewSeriesPoint {
|
||||||
pub timestamp: chrono::NaiveDateTime,
|
pub timestamp: chrono::NaiveDateTime,
|
||||||
pub value: SeriesType,
|
pub value: i32,
|
||||||
|
pub series_id: i32,
|
||||||
}
|
}
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
|
||||||
pub enum SeriesType {
|
|
||||||
Bool(bool),
|
|
||||||
Count(u32),
|
|
||||||
Signed(i32),
|
|
||||||
Float(f32),
|
|
||||||
}
|
|
@ -1,18 +1,35 @@
|
|||||||
|
use crate::schema::users;
|
||||||
|
use diesel;
|
||||||
|
use diesel::prelude::*;
|
||||||
|
use rocket::form::FromForm;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize, Queryable, Selectable, Identifiable, PartialEq)]
|
||||||
|
#[diesel(table_name = users)]
|
||||||
|
#[diesel(check_for_backend(diesel::pg::Pg))]
|
||||||
pub struct User {
|
pub struct User {
|
||||||
id: i32,
|
pub id: i32,
|
||||||
name: String,
|
pub name: String,
|
||||||
email: String,
|
pub email: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(FromForm)]
|
||||||
|
pub struct UserFilter {
|
||||||
|
pub id: Option<i32>,
|
||||||
|
pub name: Option<String>,
|
||||||
|
pub email: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(AsChangeset)]
|
||||||
|
#[diesel(table_name = users)]
|
||||||
pub struct UserChangeset {
|
pub struct UserChangeset {
|
||||||
name: Option<String>,
|
pub name: Option<String>,
|
||||||
email: Option<String>,
|
pub email: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Insertable, Deserialize)]
|
||||||
|
#[diesel(table_name = users)]
|
||||||
pub struct NewUser {
|
pub struct NewUser {
|
||||||
name: String,
|
pub name: String,
|
||||||
email: String,
|
pub email: String,
|
||||||
}
|
}
|
||||||
|
47
src/schema.rs
Normal file
47
src/schema.rs
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
// @generated automatically by Diesel CLI.
|
||||||
|
|
||||||
|
diesel::table! {
|
||||||
|
categories (id) {
|
||||||
|
id -> Int4,
|
||||||
|
name -> Varchar,
|
||||||
|
user_id -> Int4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
diesel::table! {
|
||||||
|
series (id) {
|
||||||
|
id -> Int4,
|
||||||
|
name -> Varchar,
|
||||||
|
repeat -> Int4,
|
||||||
|
good -> Bool,
|
||||||
|
category_id -> Int4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
diesel::table! {
|
||||||
|
series_points (id) {
|
||||||
|
id -> Int4,
|
||||||
|
timestamp -> Timestamp,
|
||||||
|
value -> Int4,
|
||||||
|
series_id -> Int4,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
diesel::table! {
|
||||||
|
users (id) {
|
||||||
|
id -> Int4,
|
||||||
|
name -> Varchar,
|
||||||
|
email -> Varchar,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
diesel::joinable!(categories -> users (user_id));
|
||||||
|
diesel::joinable!(series -> categories (category_id));
|
||||||
|
diesel::joinable!(series_points -> series (series_id));
|
||||||
|
|
||||||
|
diesel::allow_tables_to_appear_in_same_query!(
|
||||||
|
categories,
|
||||||
|
series,
|
||||||
|
series_points,
|
||||||
|
users,
|
||||||
|
);
|
10
src/utils.rs
10
src/utils.rs
@ -1,10 +0,0 @@
|
|||||||
pub fn set_panic_hook() {
|
|
||||||
// When the `console_error_panic_hook` feature is enabled, we can call the
|
|
||||||
// `set_panic_hook` function at least once during initialization, and then
|
|
||||||
// we will get better error messages if our code ever panics.
|
|
||||||
//
|
|
||||||
// For more details see
|
|
||||||
// https://github.com/rustwasm/console_error_panic_hook#readme
|
|
||||||
#[cfg(feature = "console_error_panic_hook")]
|
|
||||||
console_error_panic_hook::set_once();
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user