Model topic as causal register instead of linearizable
Sable doesn't do quorum reads or write, so it's not linearizable
This commit is contained in:
@ -16,31 +16,17 @@
|
||||
[tests :as tests]
|
||||
[checker :as checker]
|
||||
[generator :as gen]
|
||||
[independent :as independent]
|
||||
[nemesis :as nemesis]]
|
||||
[jepsen.tests.causal :as causal]
|
||||
[jepsen.os.debian :as debian]
|
||||
[knossos.model :as model]
|
||||
[jable.client :as client]
|
||||
[jable.db :as db])
|
||||
(:import (knossos.model Model)))
|
||||
|
||||
(defn r [_ _] {:type :invoke, :f :read-topic, :value nil})
|
||||
(defn w [_ _] {:type :invoke, :f :write-topic, :value (str (rand-int 5))})
|
||||
|
||||
(defrecord Channel [value]
|
||||
Model
|
||||
(step [c op]
|
||||
(case (:f op)
|
||||
:write-topic (Channel. (:value op))
|
||||
:read-topic (if (= value (:value op))
|
||||
c
|
||||
(model/inconsistent
|
||||
(str (pr-str value) "≠" (pr-str (:value op)))))))
|
||||
|
||||
Object
|
||||
(toString [r] (pr-str value)))
|
||||
|
||||
(defn channel []
|
||||
(Channel. nil))
|
||||
(defn r [_ _] {:type :invoke, :f :read, :value nil})
|
||||
(defn w [_ _] {:type :invoke, :f :write, :value (str (rand-int 5))})
|
||||
|
||||
(defn sable-test
|
||||
"Given an options map from the command line runner (e.g. :nodes, :ssh,
|
||||
@ -54,9 +40,7 @@
|
||||
:db (db/sable (:sable-bin opts) (:nodes opts))
|
||||
:client (client/jable.client.Client. nil nil nil)
|
||||
:nemesis (nemesis/partition-random-halves)
|
||||
:checker (checker/linearizable
|
||||
{:model (channel)
|
||||
:algorithm :linear})
|
||||
:checker (independent/checker (causal/check causal/causal-register))
|
||||
:generator (->> (gen/mix [r w])
|
||||
(gen/stagger 1)
|
||||
(gen/nemesis nil)
|
||||
|
@ -184,41 +184,41 @@
|
||||
|
||||
(invoke! [this test op]
|
||||
(case (:f op)
|
||||
:read-topic (do
|
||||
(send-command this {:tags {"label" "read-topic"}
|
||||
:cmd "TOPIC"
|
||||
:params ["#chan"]})
|
||||
(let [[cmd1 & cmds] (read-response this "read-topic")]
|
||||
(case (:cmd cmd1)
|
||||
; RPL_NOTOPIC
|
||||
"331" (assoc op :type :ok, :value nil)
|
||||
"BATCH" (let [[topic-cmd topicwhotime-cmd batch-close] cmds]
|
||||
; RPL_TOPIC
|
||||
(assert (= (:cmd topic-cmd) "332") (str "expected RPL_TOPIC: " topic-cmd))
|
||||
; RPL_TOPICWHOTIME
|
||||
(assert (= (:cmd topicwhotime-cmd) "333") (str "expected RPL_TOPICWHOTIME: " topicwhotime-cmd))
|
||||
(assert (= (:cmd batch-close) "BATCH") "topic batch close cmd")
|
||||
(let [[my-nick chan topic] (:params topic-cmd)
|
||||
[my-nick chan who time] (:params topicwhotime-cmd)]
|
||||
(let [[nick userhost] (str/split who #"!" 2)
|
||||
time (Integer/parseInt time)]
|
||||
(assoc op :type :ok, :value {:topic topic
|
||||
:who nick
|
||||
:time time})))))))
|
||||
:write-topic (do
|
||||
(send-command this {:tags {"label" "write-topic"}
|
||||
:cmd "TOPIC"
|
||||
:params ["#chan" (:value op)]})
|
||||
(let [[cmd] (read-response this "write-topic")]
|
||||
(case (:cmd cmd)
|
||||
; Sable returns ACK when two clients set the same topic value
|
||||
; at the same time
|
||||
"ACK" (assoc op :type :fail, :error :noop)
|
||||
"TOPIC" (let [[chan topic] (:params cmd)
|
||||
time (/ (.getTime (instant/read-instant-timestamp (get (:tags cmd) "time"))) 1000)]
|
||||
(assoc op :type :ok, :value {:topic topic
|
||||
:who (:nick this)
|
||||
:time time })))))))
|
||||
:read (do
|
||||
(send-command this {:tags {"label" "read-topic"}
|
||||
:cmd "TOPIC"
|
||||
:params ["#chan"]})
|
||||
(let [[cmd1 & cmds] (read-response this "read-topic")]
|
||||
(case (:cmd cmd1)
|
||||
; RPL_NOTOPIC
|
||||
"331" (assoc op :type :ok, :value nil)
|
||||
"BATCH" (let [[topic-cmd topicwhotime-cmd batch-close] cmds]
|
||||
; RPL_TOPIC
|
||||
(assert (= (:cmd topic-cmd) "332") (str "expected RPL_TOPIC: " topic-cmd))
|
||||
; RPL_TOPICWHOTIME
|
||||
(assert (= (:cmd topicwhotime-cmd) "333") (str "expected RPL_TOPICWHOTIME: " topicwhotime-cmd))
|
||||
(assert (= (:cmd batch-close) "BATCH") "topic batch close cmd")
|
||||
(let [[my-nick chan topic] (:params topic-cmd)
|
||||
[my-nick chan who time] (:params topicwhotime-cmd)]
|
||||
(let [[nick userhost] (str/split who #"!" 2)
|
||||
time (Integer/parseInt time)]
|
||||
(assoc op :type :ok, :value {:topic topic
|
||||
:who nick
|
||||
:time time})))))))
|
||||
:write (do
|
||||
(send-command this {:tags {"label" "write-topic"}
|
||||
:cmd "TOPIC"
|
||||
:params ["#chan" (:value op)]})
|
||||
(let [[cmd] (read-response this "write-topic")]
|
||||
(case (:cmd cmd)
|
||||
; Sable returns ACK when two clients set the same topic value
|
||||
; at the same time
|
||||
"ACK" (assoc op :type :fail, :error :noop)
|
||||
"TOPIC" (let [[chan topic] (:params cmd)
|
||||
time (/ (.getTime (instant/read-instant-timestamp (get (:tags cmd) "time"))) 1000)]
|
||||
(assoc op :type :ok, :value {:topic topic
|
||||
:who (:nick this)
|
||||
:time time })))))))
|
||||
|
||||
(teardown! [this test])
|
||||
|
||||
|
Reference in New Issue
Block a user