Add KV impl.

use kv in counter, still doesn't pass.
This commit is contained in:
Joe Ardent 2024-06-04 11:32:00 -07:00
parent e92af7cf92
commit 77316fa927
4 changed files with 63 additions and 76 deletions

View File

@ -1,4 +1,4 @@
use nebkor_maelstrom::{mk_payload, Body, Message, Node, Runner}; use nebkor_maelstrom::{kv::Kv, mk_payload, Body, Message, Node, Runner};
const KEY: &str = "COUNTER"; const KEY: &str = "COUNTER";
@ -11,14 +11,8 @@ fn main() {
let runner = Runner::new(node); let runner = Runner::new(node);
let on_init = |rnr: &Runner| { let on_init = |rnr: &Runner| {
let payload = mk_payload(&[ let kv = Kv::seq();
("key", KEY.into()), let _ = kv.cas(rnr, KEY, 0i64.into(), 0i64.into(), true);
("from", 0i64.into()),
("to", 0i64.into()),
("create_if_not_exists", true.into()),
]);
let body = Body::from_type("cas").with_payload(payload);
rnr.send("seq-kv", body);
}; };
let on_init = Box::new(on_init); let on_init = Box::new(on_init);
@ -28,49 +22,22 @@ fn main() {
impl Node for Counter { impl Node for Counter {
fn handle<'slf>(&'slf mut self, runner: &'slf Runner, req: Message) { fn handle<'slf>(&'slf mut self, runner: &'slf Runner, req: Message) {
let read_payload = mk_payload(&[("key", KEY.into())]);
let read_body = Body::from_type("read").with_payload(read_payload);
let typ = req.body.typ.as_str(); let typ = req.body.typ.as_str();
//let frm = req.src.clone(); let kv = Kv::seq();
//let msg_id = req.body.msg_id.to_owned();
match typ { match typ {
"add" => { "add" => {
let delta = req.body.payload.get("delta").unwrap().as_i64().unwrap(); let delta = req.body.payload.get("delta").unwrap().as_i64().unwrap();
let cur = runner loop {
.rpc("seq-kv", read_body) let cur = kv.read(runner, KEY).unwrap().as_i64().unwrap();
.recv() match kv.cas(runner, KEY, cur.into(), (cur + delta).into(), false) {
.unwrap() Err(_) => {}
.body Ok(_) => break,
.payload }
.get("value") }
.cloned()
.unwrap()
.as_i64()
.unwrap();
let cas_payload = mk_payload(&[
("key", KEY.into()),
("from", cur.into()),
("to", (cur + delta).into()),
]);
// ERRORS BE HERE
runner
.rpc("seq-kv", Body::from_type("cas").with_payload(cas_payload))
.recv()
.unwrap();
runner.reply(&req, Body::from_type("add_ok")); runner.reply(&req, Body::from_type("add_ok"));
} }
"read" => { "read" => {
let payload = mk_payload(&[("key", KEY.into())]); let val = kv.read(runner, KEY).unwrap();
let body = Body::from_type("read").with_payload(payload);
let val = runner
.rpc("seq-kv", body)
.recv()
.unwrap()
.body
.payload
.get("value")
.cloned()
.unwrap();
let body = Body::from_type("read_ok").with_payload(mk_payload(&[("value", val)])); let body = Body::from_type("read_ok").with_payload(mk_payload(&[("value", val)]));
runner.reply(&req, body); runner.reply(&req, body);
} }

View File

@ -1,40 +1,42 @@
/*
use std::{rc::Rc, sync::Arc};
use serde_json::Value; use serde_json::Value;
use crate::{mk_payload, Body, Message, Result, Runner}; use crate::{check_err, mk_payload, protocol::ErrorCode, Body, RpcResult, Runner};
pub type ReadResult = std::result::Result<Value, ErrorCode>;
#[derive(Debug, Default, Clone)] #[derive(Debug, Default, Clone)]
pub struct Kv { pub struct Kv {
pub service: String, pub service: &'static str,
} }
impl Kv { impl Kv {
pub fn seq() -> Self { pub fn seq() -> Self {
Kv { Kv { service: "seq-kv" }
service: "seq-kv".to_string(),
}
} }
pub fn lin() -> Self { pub fn lin() -> Self {
Kv { Kv { service: "lin-kv" }
service: "lin-kv".to_string(),
}
} }
pub fn lww() -> Self { pub fn lww() -> Self {
Kv { Kv { service: "lww-kv" }
service: "lww-kv".to_string(),
}
} }
pub fn read(&self, runner: &Runner, key: &str) -> Result<Value> { pub fn read(&self, runner: &Runner, key: &str) -> ReadResult {
todo!() let payload = mk_payload(&[("key", key.into())]);
let body = Body::from_type("read").with_payload(payload);
let rx = runner.rpc(self.service, body);
let msg = rx.recv().unwrap();
check_err(&msg)?;
Ok(msg.body.payload.get("value").unwrap().to_owned())
} }
pub fn write(&self, runner: &Runner, key: &str, val: Value, create: bool) -> Result<()> { pub fn write(&self, runner: &Runner, key: &str, val: Value) -> RpcResult {
todo!() let payload = mk_payload(&[("key", key.into()), ("value", val)]);
let body = Body::from_type("write").with_payload(payload);
let msg = runner.rpc(self.service, body).recv().unwrap();
check_err(&msg)?;
Ok(())
} }
pub fn cas( pub fn cas(
@ -44,8 +46,16 @@ impl Kv {
from: Value, from: Value,
to: Value, to: Value,
create: bool, create: bool,
) -> Result<()> { ) -> RpcResult {
todo!() let payload = mk_payload(&[
("key", key.into()),
("from", from),
("to", to),
("create_if_not_exists", create.into()),
]);
let body = Body::from_type("cas").with_payload(payload);
let msg = runner.rpc(self.service, body).recv().unwrap();
check_err(&msg)?;
Ok(())
} }
} }
*/

View File

@ -18,8 +18,8 @@ pub mod kv;
pub type NodeyNodeFace = Arc<Mutex<dyn Node>>; pub type NodeyNodeFace = Arc<Mutex<dyn Node>>;
pub type OnInit = Box<dyn Fn(&Runner)>; pub type OnInit = Box<dyn Fn(&Runner)>;
pub type Result<T> = std::result::Result<T, ErrorCode>;
pub type RpcPromise = Receiver<Message>; pub type RpcPromise = Receiver<Message>;
pub type RpcResult = std::result::Result<(), ErrorCode>;
pub trait Node { pub trait Node {
fn handle(&mut self, runner: &Runner, msg: Message); fn handle(&mut self, runner: &Runner, msg: Message);
@ -198,6 +198,14 @@ impl Runner {
} }
} }
pub fn check_err(msg: &Message) -> RpcResult {
if msg.body.typ.as_str() == "error" {
let error = msg.body.code.unwrap();
return Err(error);
}
Ok(())
}
pub fn mk_payload(payload: &[(&str, Value)]) -> Payload { pub fn mk_payload(payload: &[(&str, Value)]) -> Payload {
payload payload
.iter() .iter()

View File

@ -21,6 +21,11 @@ pub struct Body {
pub in_reply_to: u64, pub in_reply_to: u64,
#[serde(flatten)] #[serde(flatten)]
pub payload: Payload, pub payload: Payload,
// the following are for the case of errors
#[serde(default, skip_serializing_if = "Option::is_none")]
pub code: Option<ErrorCode>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub text: Option<String>,
} }
impl Body { impl Body {
@ -56,14 +61,14 @@ pub fn init_ok(msg_id: u64, in_reply_to: u64) -> Body {
.with_in_reply_to(in_reply_to) .with_in_reply_to(in_reply_to)
} }
#[derive(Debug, Clone, Serialize, Deserialize)] #[derive(Debug, Clone, Copy, Serialize, Deserialize)]
#[serde(untagged)] #[serde(untagged)]
pub enum ErrorCode { pub enum ErrorCode {
Definite(DefiniteError), Definite(DefiniteError),
Indefinite(IndefiniteError), Indefinite(IndefiniteError),
} }
#[derive(Debug, Clone, Serialize_repr, Deserialize_repr)] #[derive(Debug, Clone, Copy, Serialize_repr, Deserialize_repr)]
#[repr(u64)] #[repr(u64)]
pub enum DefiniteError { pub enum DefiniteError {
NodeNotFound = 2, NodeNotFound = 2,
@ -77,7 +82,7 @@ pub enum DefiniteError {
TxnConflict = 30, TxnConflict = 30,
} }
#[derive(Debug, Clone, Serialize_repr, Deserialize_repr)] #[derive(Debug, Clone, Copy, Serialize_repr, Deserialize_repr)]
#[repr(u64)] #[repr(u64)]
pub enum IndefiniteError { pub enum IndefiniteError {
Timeout = 0, Timeout = 0,
@ -89,12 +94,9 @@ pub fn error(msg_id: u64, in_reply_to: u64, code: ErrorCode, text: Option<&str>)
typ: "error".to_string(), typ: "error".to_string(),
msg_id, msg_id,
in_reply_to, in_reply_to,
payload: [ code: Some(code),
("code".to_string(), serde_json::to_value(code).unwrap()), text: text.map(|t| t.to_string()),
("text".to_string(), serde_json::to_value(text).unwrap()), payload: Default::default(),
]
.into_iter()
.collect(),
} }
} }