跳到主内容
paste
bin
.ca
type · paste · share
⌘
K
文档
登录
?
← 返回文本
›
编辑 / 分支
Hew v0.5 — redis-style KV server (Session machine + Store actor, MULTI/EXEC)
#72Qe5sQSGf
public / public
新版本
由 @slepp
已创建 4 days ago
永不过期
2.9 KB
语法:
hew
你的更改会创建一个链接到此文本的新文本 — 原始文本不变。
新版本
你的更改会创建一个链接到此文本的新文本 — 原始文本不变。
标题(可选)
文件名
语法
hew
text
bash
c
cpp
css
diff
dockerfile
go
html
ini
java
javascript
json
kotlin
lua
makefile
markdown
nginx
php
python
ruby
rust
shellscript
sql
swift
toml
typescript
xml
yaml
可见性
公开动态
访问
public
过期
7 天
10 分钟
1 小时
1 天
7 天
30 天
90 天
自定义…
自定义过期
变更备注
(可选)
此文本会显示在公开动态中。如果只想通过链接分享,请更改可见性。
创建新版本
取消
粘贴或输入…
// A redis-style key/value server: a Store actor owns the data, a Session // machine drives MULTI/EXEC transactions on the client side. // // Invariants: // * All key/value state lives in the Store actor and is mutated only inside // its receive fn, so every command is serialized through one mailbox. // * A Session is always in exactly one state. MULTI moves Ready -> Buffering; // each command then appends to the buffer; EXEC replays the buffer in order // and returns to Ready; DISCARD drops the buffer and returns to Ready. // * exec applies exactly one command and returns its RESP-style reply. machine Session { events { Begin; Enqueue { cmd: string; } Commit; Abort; } state Ready; state Buffering { pending: Vec<string>; } on Begin: Ready => Buffering { Buffering { pending: Vec::new() } } on Enqueue(cmd): Buffering => Buffering reenter { self.pending.push(cmd); Buffering { pending: self.pending } } on Commit: Buffering => Ready { Ready } on Abort: Buffering => Ready { Ready } default { state } } actor Store { let data: HashMap<string, string>; receive fn exec(line: string) -> string { let parts = line.trim().split(" "); let verb = parts.get(0); match verb { "PING" => "+PONG", "DBSIZE" => f":{data.len()}", "SET" => { data.insert(parts.get(1), parts.get(2)); "+OK" }, "GET" => match data.get(parts.get(1)) { Some(value) => f"+{value}", None => "-NOT_FOUND", }, "DEL" => { let key = parts.get(1); if data.contains_key(key) { data.remove(key); "+OK" } else { "-NOT_FOUND" } }, "EXISTS" => if data.contains_key(parts.get(1)) { ":1" } else { ":0" }, _ => "-ERR unknown command", } } } fn apply(store: LocalPid<Store>, line: string) { let reply = match await store.exec(line.clone()) { Ok(r) => r, Err(_) => "-ERR no reply", }; println(f"{line} -> {reply}"); } fn main() { let store = spawn Store(data: HashMap::new()); apply(store, "PING"); apply(store, "SET greeting hello"); apply(store, "GET greeting"); apply(store, "EXISTS greeting"); // A transaction: MULTI buffers, EXEC replays the buffer in order. var session = Ready; session.step(Begin); session.step(Enqueue { cmd: "SET a 1" }); session.step(Enqueue { cmd: "SET b 2" }); match session { Buffering { pending } => { for i in 0 .. pending.len() { apply(store, pending.get(i)); } }, Ready => {}, } session.step(Commit); apply(store, "DBSIZE"); apply(store, "GET a"); }