Add the simple substitution cipher.
continuous-integration/drone/push Build is passing
Details
continuous-integration/drone/push Build is passing
Details
parent
2db631d8da
commit
178c85b0cb
@ -0,0 +1,47 @@
|
|||||||
|
(ns cipher-analytical-machine.ciphers.simple-substitution
|
||||||
|
(:require [clojure.string :as cs]
|
||||||
|
[clojure.set :as set]
|
||||||
|
[cipher-analytical-machine.ciphers.caesar :as caesar])
|
||||||
|
(:gen-class))
|
||||||
|
|
||||||
|
(defn shuffled-numbers
|
||||||
|
"Generate the shuffled order for integer list."
|
||||||
|
[size]
|
||||||
|
(-> size
|
||||||
|
(take (range))
|
||||||
|
(shuffle)))
|
||||||
|
|
||||||
|
(defn generate-substitution-table
|
||||||
|
"Generate the map (char, int) for the substittion."
|
||||||
|
[symbols]
|
||||||
|
(->> (count symbols)
|
||||||
|
(shuffled-numbers)
|
||||||
|
(zipmap symbols)))
|
||||||
|
|
||||||
|
(defn find-value-in-table
|
||||||
|
"It uses the substitution table to find the value of a char or a number."
|
||||||
|
[char substitution-table symbols]
|
||||||
|
(get substitution-table char))
|
||||||
|
|
||||||
|
(defn encrypt-message
|
||||||
|
"Encrypt a message using the simple substitution cipher. The function is case-insensitive. If a symbol isn't in the symbols list, then it will be removed."
|
||||||
|
[message key substitution-table symbols]
|
||||||
|
(let [max-index (count symbols)]
|
||||||
|
(->> message
|
||||||
|
(cs/lower-case)
|
||||||
|
(caesar/encrypt-message key symbols)
|
||||||
|
(map (fn [char] (find-value-in-table char substitution-table symbols)))
|
||||||
|
(cs/join \,))))
|
||||||
|
|
||||||
|
(defn decrypt-message
|
||||||
|
"Decrypt a message using the simple substitution cipher. The function is case-insensitive."
|
||||||
|
[message key substitution-table symbols]
|
||||||
|
(let [substitution-table (set/map-invert substitution-table)
|
||||||
|
max-index (count symbols)
|
||||||
|
message (cs/split message #",")]
|
||||||
|
(->> message
|
||||||
|
(map #(Integer/parseInt %))
|
||||||
|
(map (fn [char] (find-value-in-table char substitution-table symbols)))
|
||||||
|
(caesar/encrypt-message key symbols)
|
||||||
|
(cs/join))))
|
||||||
|
|
@ -0,0 +1,44 @@
|
|||||||
|
(ns cipher-analytical-machine.ciphers.simple-substitution
|
||||||
|
(:require
|
||||||
|
[clojure.test :refer :all]
|
||||||
|
[cipher-analytical-machine.ciphers.simple-substitution :refer :all]))
|
||||||
|
|
||||||
|
(deftest find-value-in-table-test
|
||||||
|
(let [table {\a 1 \b 2 \c 3}
|
||||||
|
rtable {1 \a 2 \b 3 \c}]
|
||||||
|
(testing "If the symbol is in the table as a key, then the result won't nil."
|
||||||
|
(are [key expected]
|
||||||
|
(= expected (find-value-in-table key table))
|
||||||
|
\a 1
|
||||||
|
\d nil
|
||||||
|
1 nil
|
||||||
|
5 nil))
|
||||||
|
|
||||||
|
(testing "If the digit is in the reversed table as a key, then the result won't nil."
|
||||||
|
(are [key expected]
|
||||||
|
(= expected (find-value-in-table key rtable))
|
||||||
|
\a nil
|
||||||
|
\d nil
|
||||||
|
1 \a
|
||||||
|
5 nil))))
|
||||||
|
|
||||||
|
(deftest encrypt-message-text
|
||||||
|
(let [symbols "abc"
|
||||||
|
table {\a 1 \b 2 \c 3}]
|
||||||
|
(testing "The function must encrypt the message and remove unknown symbols."
|
||||||
|
(are [message key expected]
|
||||||
|
(= expected (encrypt-message message key table symbols))
|
||||||
|
"abc" 0 "1,2,3"
|
||||||
|
"abc" 1 "2,3,1"
|
||||||
|
"aDbdc" 0 "1,2,3"))))
|
||||||
|
|
||||||
|
(deftest decrypt-message-text
|
||||||
|
(let [symbols "abc"
|
||||||
|
table {\a 1 \b 2 \c 3}]
|
||||||
|
(testing "The function must decrypt the message and remove unknown numbers."
|
||||||
|
(are [message key expected]
|
||||||
|
(= expected (decrypt-message message key table symbols))
|
||||||
|
"1,2,3" 0 "abc"
|
||||||
|
"2,3,1" 1 "abc"
|
||||||
|
"1,12,2,3" 0 "abc"))))
|
||||||
|
|
Loading…
Reference in new issue