Procházet zdrojové kódy

Move handlers around

Cesar Rodas před 10 měsíci
rodič
revize
2e68b44fa5
9 změnil soubory, kde provedl 364 přidání a 278 odebrání
  1. 21 20
      Cargo.lock
  2. 1 0
      Cargo.toml
  3. 49 0
      src/balance.rs
  4. 47 0
      src/deposit.rs
  5. 74 0
      src/get.rs
  6. 22 245
      src/main.rs
  7. 19 13
      src/subscribe.rs
  8. 55 0
      src/tx.rs
  9. 76 0
      src/update.rs

+ 21 - 20
Cargo.lock

@@ -438,13 +438,13 @@ checksum = "ecc7ab41815b3c653ccd2978ec3255c81349336702dfdf62ee6f7069b12a3aae"
 
 [[package]]
 name = "async-trait"
-version = "0.1.73"
+version = "0.1.80"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "bc00ceb34980c03614e35a3a4e218276a0a824e911d07651cd0d858a51e8c0f0"
+checksum = "c6fa2087f2753a7da8cc1c0dbfcf89579dd57458e36769de5ac750b4671737ca"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -607,7 +607,7 @@ dependencies = [
  "proc-macro-crate",
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
  "syn_derive",
 ]
 
@@ -1143,7 +1143,7 @@ checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -1547,6 +1547,7 @@ name = "ledger"
 version = "0.1.0"
 dependencies = [
  "actix-web",
+ "async-trait",
  "env_logger",
  "futures-util",
  "serde",
@@ -1867,7 +1868,7 @@ checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -2000,7 +2001,7 @@ checksum = "4359fd9c9171ec6e8c62926d6faaf553a8dc3f64e1507e76da7911b4f6a04405"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -2121,9 +2122,9 @@ checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068"
 
 [[package]]
 name = "proc-macro2"
-version = "1.0.67"
+version = "1.0.83"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3d433d9f1a3e8c1263d9456598b16fec66f4acc9a74dacffd35c7bb09b3a1328"
+checksum = "0b33eb56c327dec362a9e55b3ad14f9d2f0904fb5a5b03b513ab5465399e9f43"
 dependencies = [
  "unicode-ident",
 ]
@@ -2136,9 +2137,9 @@ checksum = "a1d01941d82fa2ab50be1e79e6714289dd7cde78eba4c074bc5a4374f650dfe0"
 
 [[package]]
 name = "quote"
-version = "1.0.33"
+version = "1.0.36"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "5267fca4496028628a95160fc423a33e8b2e6af8a5302579e322e4b520293cae"
+checksum = "0fa76aaf39101c457836aec0ce2316dbdc3ab723cdda1c6bd4e6ad4208acaca7"
 dependencies = [
  "proc-macro2",
 ]
@@ -2426,7 +2427,7 @@ checksum = "4eca7ac642d82aa35b60049a6eccb4be6be75e599bd2e9adb5f875a737654af2"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -2894,9 +2895,9 @@ dependencies = [
 
 [[package]]
 name = "syn"
-version = "2.0.37"
+version = "2.0.65"
 source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "7303ef2c05cd654186cb250d29049a24840ca25d2747c25c0381c8d9e2f582e8"
+checksum = "d2863d96a84c6439701d7a38f9de935ec562c8832cc55d1dde0f513b52fad106"
 dependencies = [
  "proc-macro2",
  "quote",
@@ -2912,7 +2913,7 @@ dependencies = [
  "proc-macro-error",
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -2954,7 +2955,7 @@ checksum = "10712f02019e9288794769fba95cd6847df9874d49d871d062172f9dd41bc4cc"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -3066,7 +3067,7 @@ checksum = "630bdcf245f78637c13ec01ffae6187cca34625e8c63150d424b59e55af2675e"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -3132,7 +3133,7 @@ checksum = "5f4f31f56159e98206da9efd823404b79b6ef3143b4a7ab76e67b1751b25a4ab"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
 ]
 
 [[package]]
@@ -3318,7 +3319,7 @@ dependencies = [
  "once_cell",
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
  "wasm-bindgen-shared",
 ]
 
@@ -3352,7 +3353,7 @@ checksum = "54681b18a46765f095758388f2d0cf16eb8d4169b639ab575a8f5693af210c7b"
 dependencies = [
  "proc-macro2",
  "quote",
- "syn 2.0.37",
+ "syn 2.0.65",
  "wasm-bindgen-backend",
  "wasm-bindgen-shared",
 ]

+ 1 - 0
Cargo.toml

@@ -15,3 +15,4 @@ serde_json = "1"
 tokio = { version = "1.32.0", features = ["full"] }
 env_logger = "0.10.0"
 futures-util = "0.3.30"
+async-trait = "0.1.80"

+ 49 - 0
src/balance.rs

@@ -0,0 +1,49 @@
+use crate::{Context, Handler};
+use actix_web::{get, web, HttpResponse, Responder};
+use serde::{Deserialize, Serialize};
+use serde_json::json;
+use verax::Asset;
+
+#[derive(Deserialize)]
+pub struct AccountId(verax::AccountId);
+
+#[derive(Serialize)]
+pub struct Balance {
+    amount: String,
+    cents: String,
+    asset: Asset,
+}
+
+impl From<verax::Amount> for Balance {
+    fn from(amount: verax::Amount) -> Self {
+        Balance {
+            amount: amount.to_string(),
+            cents: amount.cents().to_string(),
+            asset: amount.asset().clone(),
+        }
+    }
+}
+
+#[async_trait::async_trait]
+impl Handler for AccountId {
+    type Ok = Vec<Balance>;
+    type Err = verax::Error;
+
+    async fn handle(self, ctx: &Context) -> Result<Self::Ok, Self::Err> {
+        Ok(ctx
+            .ledger
+            .get_balance(&self.0)
+            .await?
+            .into_iter()
+            .map(|x| x.into())
+            .collect())
+    }
+}
+
+#[get("/balance/{id}")]
+pub async fn handler(info: web::Path<AccountId>, ctx: web::Data<Context>) -> impl Responder {
+    match info.0.handle(&ctx).await {
+        Ok(balances) => HttpResponse::Ok().json(balances),
+        Err(err) => HttpResponse::BadRequest().json(json!({ "text": err.to_string(), "err": err})),
+    }
+}

+ 47 - 0
src/deposit.rs

@@ -0,0 +1,47 @@
+use crate::{Context, Handler};
+use actix_web::{post, web, HttpResponse, Responder};
+use serde::Deserialize;
+use serde_json::json;
+use verax::{AccountId, AnyAmount, Status, Tag};
+
+#[derive(Deserialize)]
+pub struct Deposit {
+    pub account: AccountId,
+    #[serde(flatten)]
+    pub amount: AnyAmount,
+    pub memo: String,
+    pub tags: Vec<Tag>,
+    pub status: Status,
+}
+
+#[async_trait::async_trait]
+impl Handler for Deposit {
+    type Ok = verax::Transaction;
+    type Err = verax::Error;
+
+    async fn handle(self, ctx: &Context) -> Result<Self::Ok, Self::Err> {
+        ctx.ledger
+            .deposit(
+                &self.account,
+                self.amount.try_into()?,
+                self.status,
+                self.tags,
+                self.memo,
+            )
+            .await
+    }
+}
+
+#[post("/deposit")]
+pub async fn handler(item: web::Json<Deposit>, ledger: web::Data<Context>) -> 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().handle(&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})),
+    }
+}

+ 74 - 0
src/get.rs

@@ -0,0 +1,74 @@
+use crate::Context;
+use actix_web::{get, web, HttpResponse, Responder};
+use serde_json::json;
+use verax::{AnyId, Filter, Type};
+
+#[get("/{id}")]
+async fn handler(info: web::Path<AnyId>, ctx: web::Data<Context>) -> 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
+                .ledger
+                .get_payment_info(&payment_id)
+                .await
+                .map(|tx| HttpResponse::Ok().json(tx));
+
+            todo!()
+        }
+    };
+
+    let limit = filter.limit;
+
+    ctx.ledger
+        .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}))
+        })
+}

+ 22 - 245
src/main.rs

@@ -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()

+ 19 - 13
src/subscribe.rs

@@ -1,4 +1,4 @@
-use crate::Ledger;
+use crate::Context;
 use actix_web::{
     post,
     rt::time::{self, Interval},
@@ -9,27 +9,39 @@ use actix_web::{
 use futures_util::Stream;
 use std::{
     pin::Pin,
-    task::{Context, Poll},
+    task::{Context as TaskContext, Poll},
     time::Duration,
 };
-use tokio::sync::mpsc::Receiver;
+use tokio::sync::mpsc::{Receiver, Sender};
 use verax::Filter;
 
+const DEFAULT_PING_INTERVAL_SECS: u64 = 30;
+
 struct SubscriberStream {
     receiver: Receiver<verax::Transaction>,
     ping_interval: Interval,
     ping: u128,
 }
 
+impl SubscriberStream {
+    pub fn new(subscription: (Sender<verax::Transaction>, Receiver<verax::Transaction>)) -> Self {
+        Self {
+            receiver: subscription.1,
+            ping_interval: time::interval(Duration::from_secs(DEFAULT_PING_INTERVAL_SECS)),
+            ping: 0,
+        }
+    }
+}
+
 impl Stream for SubscriberStream {
     type Item = Result<Bytes, actix_web::Error>;
 
-    fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
+    fn poll_next(mut self: Pin<&mut Self>, cx: &mut TaskContext<'_>) -> Poll<Option<Self::Item>> {
         match Pin::new(&mut self.ping_interval).poll_tick(cx) {
             Poll::Ready(_) => {
                 // Send a heartbeat message
                 self.ping += 1;
-                let message = format!("{}\"ping\": {}{}\n", "{", self.ping, "}");
+                let message = format!("{}\"ping\":{}{}\n", "{", self.ping, "}");
                 let heartbeat_bytes = Bytes::copy_from_slice(&message.as_bytes());
                 return Poll::Ready(Some(Ok(heartbeat_bytes)));
             }
@@ -49,14 +61,8 @@ impl Stream for SubscriberStream {
 }
 
 #[post("/subscribe")]
-pub async fn subscribe_by_tag(tag: web::Json<Filter>, ctx: web::Data<Ledger>) -> impl Responder {
-    let (_, receiver) = ctx._inner.subscribe(tag.0).await;
-
+pub async fn handler(tag: web::Json<Filter>, ctx: web::Data<Context>) -> impl Responder {
     HttpResponse::Ok()
         .content_type("application/json")
-        .streaming(SubscriberStream {
-            receiver,
-            ping_interval: time::interval(Duration::from_secs(30)),
-            ping: 0,
-        })
+        .streaming(SubscriberStream::new(ctx.ledger.subscribe(tag.0).await))
 }

+ 55 - 0
src/tx.rs

@@ -0,0 +1,55 @@
+use crate::{Context, Handler};
+use actix_web::{post, web, HttpResponse, Responder};
+use serde::Deserialize;
+use serde_json::json;
+use verax::{AccountId, AnyAmount, Status};
+
+#[derive(Deserialize)]
+pub struct AccountWithAmount {
+    pub account: AccountId,
+    #[serde(flatten)]
+    pub amount: AnyAmount,
+}
+
+#[derive(Deserialize)]
+pub struct Transaction {
+    pub debit: Vec<AccountWithAmount>,
+    pub credit: Vec<AccountWithAmount>,
+    pub memo: String,
+    pub status: Status,
+}
+
+#[async_trait::async_trait]
+impl Handler for Transaction {
+    type Ok = verax::Transaction;
+    type Err = verax::Error;
+
+    async fn handle(self, ledger: &Context) -> Result<Self::Ok, Self::Err> {
+        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
+            .ledger
+            .new_transaction(self.memo, self.status, from, to)
+            .await
+    }
+}
+
+#[post("/tx")]
+async fn handler(item: web::Json<Transaction>, ledger: web::Data<Context>) -> impl Responder {
+    match item.into_inner().handle(&ledger).await {
+        Ok(tx) => HttpResponse::Accepted().json(tx),
+        Err(err) => {
+            HttpResponse::InternalServerError().json(json!({ "text": err.to_string(), "err": err}))
+        }
+    }
+}

+ 76 - 0
src/update.rs

@@ -0,0 +1,76 @@
+use crate::{Context, Handler};
+use actix_web::{post, web, HttpResponse, Responder};
+use serde::Deserialize;
+use serde_json::json;
+use verax::{RevId, Status, Tag};
+
+#[derive(Deserialize)]
+pub struct UpdateOperation {
+    pub id: RevId,
+    #[serde(default)]
+    pub status: Option<Status>,
+    #[serde(default)]
+    pub tags: Option<Vec<Tag>>,
+    pub memo: String,
+}
+
+struct Update {
+    id: RevId,
+    operation: UpdateOperation,
+}
+
+#[async_trait::async_trait]
+impl Handler for Update {
+    type Ok = verax::Transaction;
+    type Err = verax::Error;
+
+    async fn handle(self, ledger: &Context) -> Result<Self::Ok, Self::Err> {
+        let id = self.id;
+        let memo = self.operation.memo;
+        let id = if let Some(status) = self.operation.status {
+            let transaction = ledger
+                .ledger
+                .change_status(id, status, memo.clone())
+                .await?;
+            transaction.revision_id
+        } else {
+            id
+        };
+
+        let id = if let Some(tags) = self.operation.tags {
+            let transaction = ledger.ledger.set_tags(id, tags, memo).await?;
+            transaction.revision_id
+        } else {
+            id
+        };
+
+        ledger
+            .ledger
+            .get_transactions(verax::Filter {
+                revisions: vec![id],
+                limit: 1,
+                ..Default::default()
+            })
+            .await?
+            .pop()
+            .ok_or(verax::Error::TxNotFound)
+    }
+}
+
+#[post("/{id}")]
+async fn handler(
+    info: web::Path<RevId>,
+    item: web::Json<UpdateOperation>,
+    ctx: web::Data<Context>,
+) -> impl Responder {
+    let update = Update {
+        id: info.into_inner(),
+        operation: item.into_inner(),
+    };
+    match update.handle(&ctx).await {
+        Ok(tx) => HttpResponse::Accepted().json(tx),
+        Err(err) => {
+            HttpResponse::InternalServerError().json(json!({ "text": err.to_string(), "err": err}))
+        }
+    }
+}