|
@@ -1,247 +1,24 @@
|
|
|
-use actix_web::{
|
|
|
- error::InternalError, get, middleware::Logger, post, web, App, HttpResponse, HttpServer,
|
|
|
- Responder,
|
|
|
-};
|
|
|
-use serde::{Deserialize, Serialize};
|
|
|
-use serde_json::json;
|
|
|
+use actix_web::{error::InternalError, middleware::Logger, web, App, HttpResponse, HttpServer};
|
|
|
+use serde::Serialize;
|
|
|
use std::sync::Arc;
|
|
|
-use subscribe::subscribe_by_tag;
|
|
|
-use verax::{AccountId, AnyAmount, AnyId, Asset, Filter, RevId, Status, Tag, Type};
|
|
|
|
|
|
-mod subscribe;
|
|
|
-
|
|
|
-#[derive(Deserialize)]
|
|
|
-pub struct Movement {
|
|
|
- pub account: AccountId,
|
|
|
- #[serde(flatten)]
|
|
|
- pub amount: AnyAmount,
|
|
|
-}
|
|
|
-
|
|
|
-#[derive(Deserialize)]
|
|
|
-pub struct Deposit {
|
|
|
- pub account: AccountId,
|
|
|
- #[serde(flatten)]
|
|
|
- pub amount: AnyAmount,
|
|
|
- pub memo: String,
|
|
|
- pub tags: Vec<Tag>,
|
|
|
- pub status: Status,
|
|
|
-}
|
|
|
-
|
|
|
-impl Deposit {
|
|
|
- pub async fn to_ledger_transaction(
|
|
|
- self,
|
|
|
- ledger: &Ledger,
|
|
|
- ) -> Result<verax::Transaction, verax::Error> {
|
|
|
- let zdeposit = ledger
|
|
|
- ._inner
|
|
|
- .deposit(
|
|
|
- &self.account,
|
|
|
- self.amount.try_into()?,
|
|
|
- self.status,
|
|
|
- vec![],
|
|
|
- self.memo,
|
|
|
- )
|
|
|
- .await?;
|
|
|
-
|
|
|
- Ok(if !self.tags.is_empty() {
|
|
|
- ledger
|
|
|
- ._inner
|
|
|
- .set_tags(zdeposit.revision_id, self.tags, "Update tags".to_owned())
|
|
|
- .await?
|
|
|
- } else {
|
|
|
- zdeposit
|
|
|
- })
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-#[derive(Deserialize)]
|
|
|
-pub struct Transaction {
|
|
|
- pub debit: Vec<Movement>,
|
|
|
- pub credit: Vec<Movement>,
|
|
|
- pub memo: String,
|
|
|
- pub status: Status,
|
|
|
-}
|
|
|
-
|
|
|
-#[derive(Deserialize)]
|
|
|
-pub struct UpdateTransaction {
|
|
|
- pub status: Status,
|
|
|
- pub memo: String,
|
|
|
-}
|
|
|
-
|
|
|
-impl UpdateTransaction {
|
|
|
- pub async fn to_ledger_transaction(
|
|
|
- self,
|
|
|
- id: RevId,
|
|
|
- ledger: &Ledger,
|
|
|
- ) -> Result<verax::Transaction, verax::Error> {
|
|
|
- ledger
|
|
|
- ._inner
|
|
|
- .change_status(id, self.status, self.memo)
|
|
|
- .await
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-impl Transaction {
|
|
|
- pub async fn to_ledger_transaction(
|
|
|
- self,
|
|
|
- ledger: &Ledger,
|
|
|
- ) -> Result<verax::Transaction, verax::Error> {
|
|
|
- let from = self
|
|
|
- .debit
|
|
|
- .into_iter()
|
|
|
- .map(|x| x.amount.try_into().map(|amount| (x.account, amount)))
|
|
|
- .collect::<Result<Vec<_>, _>>()?;
|
|
|
-
|
|
|
- let to = self
|
|
|
- .credit
|
|
|
- .into_iter()
|
|
|
- .map(|x| x.amount.try_into().map(|amount| (x.account, amount)))
|
|
|
- .collect::<Result<Vec<_>, _>>()?;
|
|
|
-
|
|
|
- ledger
|
|
|
- ._inner
|
|
|
- .new_transaction(self.memo, self.status, from, to)
|
|
|
- .await
|
|
|
- }
|
|
|
-}
|
|
|
+#[async_trait::async_trait]
|
|
|
+pub trait Handler {
|
|
|
+ type Ok: Serialize;
|
|
|
+ type Err: Serialize;
|
|
|
|
|
|
-#[derive(Serialize)]
|
|
|
-struct AccountResponse {
|
|
|
- amount: String,
|
|
|
- cents: String,
|
|
|
- asset: Asset,
|
|
|
+ async fn handle(self, ctx: &Context) -> Result<Self::Ok, Self::Err>;
|
|
|
}
|
|
|
|
|
|
-#[get("/balance/{id}")]
|
|
|
-async fn get_balance(info: web::Path<AccountId>, ctx: web::Data<Ledger>) -> impl Responder {
|
|
|
- match ctx._inner.get_balance(&info.0).await {
|
|
|
- Ok(balances) => HttpResponse::Ok().json(
|
|
|
- balances
|
|
|
- .into_iter()
|
|
|
- .map(|amount| AccountResponse {
|
|
|
- amount: amount.to_string(),
|
|
|
- cents: amount.cents().to_string(),
|
|
|
- asset: amount.asset().clone(),
|
|
|
- })
|
|
|
- .collect::<Vec<_>>(),
|
|
|
- ),
|
|
|
- Err(err) => HttpResponse::BadRequest().json(json!({ "text": err.to_string(), "err": err})),
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-#[get("/{id}")]
|
|
|
-async fn get_info(info: web::Path<AnyId>, ctx: web::Data<Ledger>) -> impl Responder {
|
|
|
- let (cache_for_ever, filter) = match info.0 {
|
|
|
- AnyId::Account(account_id) => (
|
|
|
- false,
|
|
|
- Filter {
|
|
|
- accounts: vec![account_id],
|
|
|
- typ: vec![Type::Deposit, Type::Withdrawal, Type::Transaction],
|
|
|
- ..Default::default()
|
|
|
- },
|
|
|
- ),
|
|
|
- AnyId::Revision(rev_id) => (
|
|
|
- true,
|
|
|
- Filter {
|
|
|
- revisions: vec![rev_id],
|
|
|
- limit: 1,
|
|
|
- ..Default::default()
|
|
|
- },
|
|
|
- ),
|
|
|
-
|
|
|
- AnyId::Transaction(transaction_id) => (
|
|
|
- false,
|
|
|
- Filter {
|
|
|
- ids: vec![transaction_id],
|
|
|
- limit: 1,
|
|
|
- ..Default::default()
|
|
|
- },
|
|
|
- ),
|
|
|
-
|
|
|
- AnyId::Payment(payment_id) => {
|
|
|
- let _ = ctx
|
|
|
- ._inner
|
|
|
- .get_payment_info(&payment_id)
|
|
|
- .await
|
|
|
- .map(|tx| HttpResponse::Ok().json(tx));
|
|
|
-
|
|
|
- todo!()
|
|
|
- }
|
|
|
- };
|
|
|
-
|
|
|
- let limit = filter.limit;
|
|
|
-
|
|
|
- ctx._inner
|
|
|
- .get_transactions(filter)
|
|
|
- .await
|
|
|
- .map(|results| {
|
|
|
- let json_response = if limit == 1 {
|
|
|
- serde_json::to_value(&results[0])
|
|
|
- } else {
|
|
|
- serde_json::to_value(&results)
|
|
|
- }
|
|
|
- .unwrap();
|
|
|
-
|
|
|
- if cache_for_ever {
|
|
|
- HttpResponse::Ok()
|
|
|
- .header(
|
|
|
- "Cache-Control",
|
|
|
- "public, max-age=31536000, s-maxage=31536000, immutable",
|
|
|
- )
|
|
|
- .header("Vary", "Accept-Encoding")
|
|
|
- .json(json_response)
|
|
|
- } else {
|
|
|
- HttpResponse::Ok().json(json_response)
|
|
|
- }
|
|
|
- })
|
|
|
- .map_err(|err| {
|
|
|
- HttpResponse::InternalServerError().json(json!({ "text": err.to_string(), "err": err}))
|
|
|
- })
|
|
|
-}
|
|
|
-
|
|
|
-#[post("/deposit")]
|
|
|
-async fn deposit(item: web::Json<Deposit>, ledger: web::Data<Ledger>) -> impl Responder {
|
|
|
- // Insert the item into a database or another data source.
|
|
|
- // For this example, we'll just echo the received item.
|
|
|
- match item.into_inner().to_ledger_transaction(&ledger).await {
|
|
|
- Ok(tx) => {
|
|
|
- // Insert the item into a database or another data source.
|
|
|
- // For this example, we'll just echo the received item.
|
|
|
- HttpResponse::Created().json(tx)
|
|
|
- }
|
|
|
- Err(err) => HttpResponse::BadRequest().json(json!({ "text": err.to_string(), "err": err})),
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-#[post("/tx")]
|
|
|
-async fn create_transaction(
|
|
|
- item: web::Json<Transaction>,
|
|
|
- ledger: web::Data<Ledger>,
|
|
|
-) -> impl Responder {
|
|
|
- match item.into_inner().to_ledger_transaction(&ledger).await {
|
|
|
- Ok(tx) => HttpResponse::Accepted().json(tx),
|
|
|
- Err(err) => {
|
|
|
- HttpResponse::InternalServerError().json(json!({ "text": err.to_string(), "err": err}))
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-#[post("/{id}")]
|
|
|
-async fn update_status(
|
|
|
- info: web::Path<RevId>,
|
|
|
- item: web::Json<UpdateTransaction>,
|
|
|
- ctx: web::Data<Ledger>,
|
|
|
-) -> impl Responder {
|
|
|
- match item.into_inner().to_ledger_transaction(info.0, &ctx).await {
|
|
|
- Ok(tx) => HttpResponse::Accepted().json(tx),
|
|
|
- Err(err) => {
|
|
|
- HttpResponse::InternalServerError().json(json!({ "text": err.to_string(), "err": err}))
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
+mod balance;
|
|
|
+mod deposit;
|
|
|
+mod get;
|
|
|
+mod subscribe;
|
|
|
+mod tx;
|
|
|
+mod update;
|
|
|
|
|
|
-pub struct Ledger {
|
|
|
- _inner: Arc<verax::Ledger<verax::storage::Cache<verax::storage::SQLite>>>,
|
|
|
+pub struct Context {
|
|
|
+ ledger: Arc<verax::Ledger<verax::storage::Cache<verax::storage::SQLite>>>,
|
|
|
}
|
|
|
|
|
|
#[actix_web::main]
|
|
@@ -274,7 +51,7 @@ async fn main() -> std::io::Result<()> {
|
|
|
|
|
|
App::new()
|
|
|
.wrap(Logger::default())
|
|
|
- .app_data(web::Data::new(Ledger { _inner: ledger }))
|
|
|
+ .app_data(web::Data::new(Context { ledger: ledger }))
|
|
|
.app_data(web::JsonConfig::default().error_handler(|err, _req| {
|
|
|
InternalError::from_response(
|
|
|
"",
|
|
@@ -284,12 +61,12 @@ async fn main() -> std::io::Result<()> {
|
|
|
)
|
|
|
.into()
|
|
|
}))
|
|
|
- .service(subscribe_by_tag)
|
|
|
- .service(get_balance)
|
|
|
- .service(get_info)
|
|
|
- .service(deposit)
|
|
|
- .service(create_transaction)
|
|
|
- .service(update_status)
|
|
|
+ .service(subscribe::handler)
|
|
|
+ .service(deposit::handler)
|
|
|
+ .service(balance::handler)
|
|
|
+ .service(tx::handler)
|
|
|
+ .service(update::handler)
|
|
|
+ .service(get::handler)
|
|
|
})
|
|
|
.bind("127.0.0.1:8080")?
|
|
|
.run()
|