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 math-ops.operations | |
(:require | |
[math-ops.game-levels :as levels])) | |
(defrecord Operation [op1 operator op2 res]) | |
(defn- hide-sth [operation] | |
(assoc operation (rand-nth [:op1 :op2 :res]) :?)) | |
(defn- random-0-9 [] | |
(rand-int 10)) | |
(defn invertible? [level {:keys [op1 operator]}] | |
(and (levels/allowed-operator? level operator) | |
(not (and (= operator *) (= op1 0))))) | |
(defn- invert [level operation] | |
(if (invertible? level operation) | |
(->Operation (:res operation) | |
(levels/inverse-operator level (:operator operation)) | |
(:op1 operation) | |
(:op2 operation)) | |
operation)) | |
(defn- invert-may-be [level operation] | |
(if (rand-nth [true false]) | |
(invert level operation) | |
operation)) | |
(defn make [level] | |
(let [op1 (random-0-9) | |
op2 (random-0-9) | |
operator (levels/random-operator level) | |
res (operator op1 op2)] | |
(->> (->Operation op1 operator op2 res) | |
(invert-may-be level) | |
(hide-sth)))) | |
(defn- find-unknown [operation] | |
(ffirst (filter #(= :? (second %)) operation))) | |
(defn- substitute-unknown [operation guess] | |
(assoc operation (find-unknown operation) guess)) | |
(defn- correct? [{:keys [res op1 op2 operator]}] | |
(= res (operator op1 op2))) | |
(defn correct-guess? [operation guess] | |
(correct? (substitute-unknown operation guess))) |
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 doo node unit-tests | |
Building ... | |
Analyzing jar:file:/home/trikitrok/.m2/repository/org/clojure/clojurescript/1.7.170/clojurescript-1.7.170.jar!/cljs/core.cljs | |
... done. Elapsed 2.32174747 seconds | |
;; ====================================================================== | |
;; Testing with Node: | |
Testing math-ops.operations-test | |
FAIL in (operations-tests) (math_ops/operations_test.js:130:4) | |
expected: (= some-non-invertible-operation (invert :max-level some-non-invertible-operation)) | |
actual: (not (= #math-ops.operations.Operation{:op1 1, :operator #object[cljs$core$_STAR_ "function cljs$core$_STAR_(var_args){ | |
var args6895 = []; | |
var len__5318__auto___6901 = arguments.length; | |
var i__5319__auto___6902 = (0); | |
while(true){ | |
if((i__5319__auto___6902 < len__5318__auto___6901)){ | |
args6895.push((arguments[i__5319__auto___6902])); | |
var G__6903 = (i__5319__auto___6902 + (1)); | |
i__5319__auto___6902 = G__6903; | |
continue; | |
} else { | |
} | |
break; | |
} | |
var G__6900 = args6895.length; | |
switch (G__6900) { | |
case 0: | |
return cljs.core._STAR_.cljs$core$IFn$_invoke$arity$0(); | |
break; | |
case 1: | |
return cljs.core._STAR_.cljs$core$IFn$_invoke$arity$1((arguments[(0)])); | |
break; | |
case 2: | |
return cljs.core._STAR_.cljs$core$IFn$_invoke$arity$2((arguments[(0)]),(arguments[(1)])); | |
break; | |
default: | |
var argseq__5337__auto__ = (new cljs.core.IndexedSeq(args6895.slice((2)),(0))); | |
return cljs.core._STAR_.cljs$core$IFn$_invoke$arity$variadic((arguments[(0)]),(arguments[(1)]),argseq__5337__auto__); | |
} | |
}"], :op2 0, :res 0} #math-ops.operations.Operation{:op1 0, :operator #object[cljs$core$_SLASH_ "function cljs$core$_SLASH_(var_args){ | |
var args6905 = []; | |
var len__5318__auto___6911 = arguments.length; | |
var i__5319__auto___6912 = (0); | |
while(true){ | |
if((i__5319__auto___6912 < len__5318__auto___6911)){ | |
args6905.push((arguments[i__5319__auto___6912])); | |
var G__6913 = (i__5319__auto___6912 + (1)); | |
i__5319__auto___6912 = G__6913; | |
continue; | |
} else { | |
} | |
break; | |
} | |
var G__6910 = args6905.length; | |
switch (G__6910) { | |
case 1: | |
return cljs.core._SLASH_.cljs$core$IFn$_invoke$arity$1((arguments[(0)])); | |
break; | |
case 2: | |
return cljs.core._SLASH_.cljs$core$IFn$_invoke$arity$2((arguments[(0)]),(arguments[(1)])); | |
break; | |
default: | |
var argseq__5337__auto__ = (new cljs.core.IndexedSeq(args6905.slice((2)),(0))); | |
return cljs.core._SLASH_.cljs$core$IFn$_invoke$arity$variadic((arguments[(0)]),(arguments[(1)]),argseq__5337__auto__); | |
} | |
}"], :op2 1, :res 0})) | |
Testing math-ops.operations-guessing-test | |
Ran 2 tests containing 18 assertions. | |
1 failures, 0 errors. | |
Watching paths: /home/trikitrok/Clojure/0_MiClojure/math-ops/src/cljs, /home/trikitrok/Clojure/0_MiClojure/math-ops/test |
So we decided to extend the IPrintWithWriter protocol to improve the test failures feedback.
First, we tried to do it in the record declaration as we were doing for the Object protocol:
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
(defrecord Operation [op1 operator op2 res] | |
Object | |
(toString [this] (operation->string this)) | |
IPrintWithWriter | |
(-pr-writer [this writer _] | |
(-write writer (operation->string this)))) |
We were also getting this warning:
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
WARNING: Protocol IPrintWithWriter implemented multiple times at line 17 /home/trikitrok/Clojure/0_MiClojure/operations.cljs |
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 math-ops.operations | |
(:require | |
[math-ops.game-levels :as levels] | |
[clojure.string :as string])) | |
(def symbols-description | |
{+ "+" | |
- "-" | |
* "x" | |
/ "/" | |
:? "?"}) | |
(defn- operation->string [{:keys [op1 operator op2 res]}] | |
(string/join " " ["#Operation [" op1 (symbols-description operator) op2 "=" res "]"])) | |
(defrecord Operation [op1 operator op2 res] | |
Object | |
(toString [this] (operation->string this))) | |
(extend-protocol IPrintWithWriter | |
Operation | |
(-pr-writer [this writer _] | |
(-write writer (operation->string this)))) | |
... | |
... |
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 doo node unit-tests | |
Building ... | |
Analyzing jar:file:/home/trikitrok/.m2/repository/org/clojure/clojurescript/1.7.170/clojurescript-1.7.170.jar!/cljs/core.cljs | |
... done. Elapsed 2.309172112 seconds | |
;; ====================================================================== | |
;; Testing with Node: | |
Testing math-ops.operations-test | |
FAIL in (operations-tests) (math_ops/operations_test.js:130:4) | |
expected: (= some-non-invertible-operation (invert :max-level some-non-invertible-operation)) | |
actual: (not (= #Operation [ 1 x 0 = 0 ] #Operation [ 0 / 1 = 0 ])) | |
Testing math-ops.operations-guessing-test | |
Ran 2 tests containing 18 assertions. | |
1 failures, 0 errors. | |
Watching paths: /home/trikitrok/Clojure/0_MiClojure/math-ops/src/cljs, /home/trikitrok/Clojure/0_MiClojure/math-ops/test |
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
... | |
... | |
(extend-type Operation | |
IPrintWithWriter | |
(-pr-writer [this writer _] | |
(-write writer (operation->string this)))) | |
... | |
... |
So we decided to ask it on the Clojurians slack.
There thanks to Alejandro Gómez and Nicolás Berger we understood what was happening. As Nicolás said:
That's the reason why it wasn't working and why we were getting that warning.
Check the code Nicolas is talking about in ClojureScript repository.
In the end, it was a nice problem to have, because we not only got a nicer and more useful failure feedback which was our initial goal, but also, learned more about extending core protocols in ClojureScript.
PS: Thank you very much to Alejandro Gómez and Nicolás Berger for their help.
Manuel, if you take a look at the transpiled js code for `defrecord`, you'll discover that `defecord` implements cljs.core.IPrintWithWriter. See it here: http://tinyurl.com/defrecord-IPrintWithWriter
ReplyDeleteHello Yeonathan
ReplyDeleteThanks for your comment Yeonathan.
Yes, it does implement it. We just wanted to override that implementation providing our own, so that the operator function didn't get printed.
Thanks again Yeonathan, Klipse is really great to see the transpiled JS code.