I started by writing a nearly end-to-end test for the scenario of running ohce during the morning:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.acceptance-test | |
(:require [midje.sweet :refer :all] | |
[ohce.core :refer :all])) | |
(unfinished hour-fn) | |
(unfinished read-input) | |
(future-facts | |
"about running ohce" | |
(fact | |
"during the morning" | |
(clojure.string/split | |
(with-out-str | |
(ohce hour-fn read-input "Pedro")) | |
#"\n") => ["¡Buenos días Pedro!" | |
"aloh" | |
"oto" | |
"¡Bonita palabra!" | |
"pots" | |
"Adios Pedro"] | |
(provided | |
(hour-fn) => 8 | |
(read-input) =streams=> ["hola" "oto" "stop" "Stop!"]))) |
Using Midje's future-facts macro I avoided seeing this acceptance test failing every time the tests were run, instead I saw a reminder message like this one:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
$ lein midje | |
WORK TO DO "about running ohce" at (acceptance_test.clj:8) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.unit-tests.ohce-test | |
(:require | |
[midje.sweet :refer :all] | |
[ohce.ohce :refer :all] | |
[ohce.test-helpers :as helpers])) | |
(unfinished select-greeting) | |
(unfinished read-input) | |
(facts | |
"about ohce" | |
(fact | |
"it greets the user" | |
(let [notifier (helpers/fake-notifier) | |
stop-word "Stop!"] | |
(ohce select-greeting notifier read-input stop-word ...username...) => irrelevant | |
(provided | |
(read-input) => "Stop!" | |
(select-greeting ...username...) => ...greeting...) | |
(helpers/args-of-call | |
:greet :notifications notifier) => [[...greeting...]])) | |
(fact | |
"it reverses the user inputs" | |
(let [notifier (helpers/fake-notifier) | |
stop-word "Stop!"] | |
(ohce select-greeting notifier read-input stop-word ...username...) => irrelevant | |
(provided | |
(select-greeting ...username...) => irrelevant | |
(read-input) =streams=> ["hola" "lolo" "Stop!"]) | |
(helpers/args-of-call | |
:echo :notifications notifier) => [["aloh"] ["olol"]])) | |
(fact | |
"it ignores inputs that are blank" | |
(let [notifier (helpers/fake-notifier) | |
stop-word "Stop!"] | |
(ohce select-greeting notifier read-input stop-word ...username...) => irrelevant | |
(provided | |
(select-greeting ...username...) => irrelevant | |
(read-input) =streams=> ["memo" "" "moko" "Stop!"]) | |
(helpers/args-of-call | |
:echo :notifications notifier) => [["omem"] ["okom"]])) | |
(fact | |
"it identifies palindromes" | |
(let [notifier (helpers/fake-notifier) | |
stop-word "Stop!"] | |
(ohce select-greeting notifier read-input stop-word ...username...) => irrelevant | |
(provided | |
(select-greeting ...username...) => irrelevant | |
(read-input) =streams=> ["oto" "ana" "Stop!"]) | |
(helpers/args-of-call | |
:echo :notifications notifier) => [["oto"] ["ana"]] | |
(helpers/args-of-call | |
:palindromes-rock :notifications notifier) => [:no-args :no-args])) | |
(fact | |
"it knows when to stop" | |
(let [notifier (helpers/fake-notifier) | |
stop-word "Stop!"] | |
(ohce select-greeting notifier read-input stop-word ...username...) => irrelevant | |
(provided | |
(select-greeting ...username...) => irrelevant | |
(read-input) =streams=> ["Stop!"]) | |
(helpers/args-of-call | |
:bye-user :notifications notifier) => [[...username...]]))) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(defprotocol Notifier | |
(greet [this greeting]) | |
(echo [this reversed-phrase]) | |
(palindromes-rock [this]) | |
(bye-user [this name])) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.test-helpers | |
(:require | |
[ohce.notifications :as notifications])) | |
(defn- register-call [func-keyword an-atom & args] | |
(let [args (if (nil? args) :no-args args) | |
calls (vec (func-keyword @an-atom))] | |
(swap! an-atom assoc func-keyword (conj calls args)))) | |
(defn args-of-call | |
[func-keyword atom-keyword protocol-implementation] | |
(func-keyword @(atom-keyword protocol-implementation))) | |
(defrecord FakeNotifier [notifications] | |
notifications/Notifier | |
(greet [_ greeting] | |
(register-call :greet notifications greeting)) | |
(echo [_ reversed-phrase] | |
(register-call :echo notifications reversed-phrase)) | |
(palindromes-rock [_] | |
(register-call :palindromes-rock notifications)) | |
(bye-user [_ name] | |
(register-call :bye-user notifications name))) | |
(defn fake-notifier [] | |
(->FakeNotifier (atom {}))) | |
(defn output-lines [f & args] | |
(clojure.string/split (with-out-str (apply f args)) #"\n")) |
Update:
The statement above is wrong. It is not necessary at all to use defrecord-openly in your production code in order to mock a protocol. See how it's done in this other post: Revisited Kata: Using Midje's defrecord-openly to mock a protocol in Ohce.
--------------------------
Another thing to notice in that test is the use of Midje's metaconstants to avoid the facts knowing about the "shape" of data (where it was possible to do it).
Well, those facts lead to the following code:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.ohce | |
(:require | |
[ohce.notifications :as notifications])) | |
(defn- reverse-str [input] | |
(apply str (reverse input))) | |
(defn- palindrome? [input] | |
(= input (reverse-str input))) | |
(defn- should-stop? [input stop-word] | |
(= input stop-word)) | |
(defn- respond-to-input [notifier input] | |
(notifications/echo notifier (reverse-str input)) | |
(when (palindrome? input) | |
(notifications/palindromes-rock notifier))) | |
(defn- process-input [notifier input] | |
(when-not (clojure.string/blank? input) | |
(respond-to-input notifier input))) | |
(defn- interact-with-user [notifier read-input stop-word] | |
(loop [input (read-input)] | |
(when-not (should-stop? input stop-word) | |
(do (process-input notifier input) | |
(recur (read-input)))))) | |
(defn ohce [select-greeting notifier read-input stop-word name] | |
(notifications/greet notifier (select-greeting name)) | |
(interact-with-user notifier read-input stop-word) | |
(notifications/bye-user notifier name)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.unit-tests.console-notifications-test | |
(:require | |
[midje.sweet :refer :all] | |
[ohce.test-helpers :as test-helpers] | |
[ohce.notifications :refer :all])) | |
(facts | |
"about console notifications" | |
(let [notifier (console-notifier {:bye-word "Adios" | |
:celebration "¡Bonita palabra!"})] | |
(fact | |
"greeting user" | |
(test-helpers/output-lines | |
greet notifier "greeting") => ["greeting"]) | |
(fact | |
"greeting user" | |
(test-helpers/output-lines | |
bye-user notifier "Juan") => ["Adios Juan"]) | |
(fact | |
"echoing word" | |
(test-helpers/output-lines | |
echo notifier "moko") => ["moko"]) | |
(fact | |
"celebrating palindromes" | |
(test-helpers/output-lines | |
palindromes-rock notifier) => ["¡Bonita palabra!"]))) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.unit-tests.day-period-greeting-selection-test | |
(:require | |
[midje.sweet :refer :all] | |
[ohce.greet-selectors :refer [select-by-day-period]])) | |
(unfinished hour-fn) | |
(facts | |
"about hour greeter" | |
(fact | |
"during the morning" | |
(select-by-day-period hour-fn "Koko") => "¡Buenos días Koko!" | |
(provided (hour-fn) => 6) | |
(select-by-day-period hour-fn "Koko") => "¡Buenos días Koko!" | |
(provided (hour-fn) => 8) | |
(select-by-day-period hour-fn "Koko") => "¡Buenos días Koko!" | |
(provided (hour-fn) => 11)) | |
(fact | |
"during the afternoon" | |
(select-by-day-period hour-fn "Koko") => "¡Buenas tardes Koko!" | |
(provided (hour-fn) => 12) | |
(select-by-day-period hour-fn "Koko") => "¡Buenas tardes Koko!" | |
(provided (hour-fn) => 16) | |
(select-by-day-period hour-fn "Koko") => "¡Buenas tardes Koko!" | |
(provided (hour-fn) => 19)) | |
(fact | |
"during the afternoon" | |
(select-by-day-period hour-fn "Koko") => "¡Buenas noches Koko!" | |
(provided (hour-fn) => 20) | |
(select-by-day-period hour-fn "Koko") => "¡Buenas noches Koko!" | |
(provided (hour-fn) => 24) | |
(select-by-day-period hour-fn "Koko") => "¡Buenas noches Koko!" | |
(provided (hour-fn) => 5))) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.acceptance-test | |
(:require | |
[midje.sweet :refer :all] | |
[ohce.ohce :refer :all] | |
[ohce.greet-selectors :refer [select-by-day-period]] | |
[ohce.notifications :refer [console-notifier]] | |
[ohce.test-helpers :as helpers])) | |
(unfinished read-input) | |
(unfinished hour-fn) | |
(let [notifications-config {:bye-word "Adios" | |
:celebration "¡Bonita palabra!"}] | |
(facts | |
"about running ohce" | |
(fact | |
"during the morning" | |
(let [stop-word "Stop!" | |
any-hour-during-morning 9 | |
notifier (console-notifier notifications-config) | |
select-greeting (fn [name] (select-by-day-period hour-fn name)) | |
ohce (partial ohce select-greeting notifier #(read-input) stop-word)] | |
(helpers/output-lines | |
ohce "Pedro") => ["¡Buenos días Pedro!" | |
"aloh" | |
"oto" | |
"¡Bonita palabra!" | |
"pots" | |
"Adios Pedro"] | |
(provided | |
(hour-fn) => any-hour-during-morning | |
(read-input) =streams=> ["hola" "oto" "stop" "Stop!"]))) | |
(fact | |
"during the afternoon" | |
(let [stop-word "Stop!" | |
any-hour-during-afternoon 16 | |
notifier (console-notifier notifications-config) | |
select-greeting (fn [name] (select-by-day-period hour-fn name)) | |
ohce (partial ohce select-greeting notifier #(read-input) stop-word)] | |
(helpers/output-lines | |
ohce "Lolo") => ["¡Buenas tardes Lolo!" | |
"opip" | |
"Adios Lolo"] | |
(provided | |
(hour-fn) => any-hour-during-afternoon | |
(read-input) =streams=> ["pipo" "Stop!"]))) | |
(fact | |
"during the afternoon" | |
(let [stop-word "Stop!" | |
any-hour-during-night 1 | |
notifier (console-notifier notifications-config) | |
select-greeting (fn [name] (select-by-day-period hour-fn name)) | |
ohce (partial ohce select-greeting notifier #(read-input) stop-word)] | |
(helpers/output-lines | |
ohce "Juan") => ["¡Buenas noches Juan!" | |
"oko" | |
"¡Bonita palabra!" | |
"Adios Juan"] | |
(provided | |
(hour-fn) => any-hour-during-night | |
(read-input) =streams=> ["oko" "Stop!"]))))) |
A curious thing to notice is that I couldn't use partial with select-greeting if I wanted the provided macro to work with hour-fn, that's why I used a lambda instead. Due to the same problem I also had to wrap the read-input function inside a lambda.
Then to be able to really run ohce on the console I just had to write a real hour-fn function (tested on the REPL):
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.clock | |
(:import (java.util Calendar))) | |
(defn hour [] | |
(.get (Calendar/getInstance) Calendar/HOUR_OF_DAY)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns ohce.core | |
(:require | |
[ohce.ohce :refer :all] | |
[ohce.clock :as clock] | |
[ohce.notifications :refer [console-notifier]] | |
[ohce.greet-selectors :as greet-selectors])) | |
(defn -main [& args] | |
(let [select-greeting (partial greet-selectors/select-by-day-period clock/hour) | |
notifier (console-notifier {:bye-word "Adios" | |
:celebration "¡Bonita palabra!"}) | |
ohce (partial ohce select-greeting notifier read-line "Stop!")] | |
(ohce (first args)))) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
$ lein run Lolo | |
¡Buenas tardes Lolo! | |
hola | |
aloh | |
vino | |
oniv | |
akka | |
akka | |
¡Bonita palabra! | |
Stop! | |
Adios Lolo | |
$ |
You can find the resulting code in this GitHub repository.
Doing this kata helped me to learn a lot of new things about Midje.
No comments:
Post a Comment