Skip to content

Instantly share code, notes, and snippets.

@pete-murphy
Last active January 26, 2019 03:01
Show Gist options
  • Save pete-murphy/146fc723a2d2df9395811dfad030f70f to your computer and use it in GitHub Desktop.
Save pete-murphy/146fc723a2d2df9395811dfad030f70f to your computer and use it in GitHub Desktop.
module NewVigenere where
import Data.Char
import Test.Hspec
import Test.QuickCheck
caesar :: Int -> Char -> Char
caesar _ ' ' = ' '
caesar n c = chr $ ordAlpha c + mod (ord c - ordAlpha c + n) 26
unCaesar :: Int -> Char -> Char
unCaesar _ ' ' = ' '
unCaesar n c = chr $ ordAlpha c + mod (ord c - ordAlpha c - n) 26
ordAlpha :: Char -> Int
ordAlpha c
| isUpper c = ord 'A'
| otherwise = ord 'a'
vigenere :: String -> String -> String
vigenere "" raw = raw
vigenere key raw = go "" (cycle key) raw
where
go _ "" raw' = raw'
go acc _ "" = reverse acc
go acc key' (' ':rs) = go (' ' : acc) key' rs
go acc (k:ks) (r:rs) = go (c : acc) ks rs
where
c = caesar (ord k - ordAlpha k) r
unVigenere :: String -> String -> String
unVigenere "" enc = enc
unVigenere key enc = go "" (cycle key) enc
where
go _ "" enc' = enc'
go acc _ "" = reverse acc
go acc key' (' ':rs) = go (' ' : acc) key' rs
go acc (k:ks) (r:rs) = go (c : acc) ks rs
where
c = unCaesar (ord k - ordAlpha k) r
prop_caesarRoundTrip :: Property
prop_caesarRoundTrip =
forAll (arbitrary `suchThat` (> 0) :: Gen Int) $ \n ->
forAll validCharGen $ \c -> unCaesar n (caesar n c) == c
validStringGen :: Gen String
validStringGen = arbitrary `suchThat` all ((&&) <$> isLetter <*> isAscii)
validCharGen :: Gen Char
validCharGen = arbitrary `suchThat` ((&&) <$> isLetter <*> isAscii)
vigenereSpec :: Spec
vigenereSpec = do
describe "Spec test" $ do
it "vigenere encodes" $ do
vigenere "ALLY" "MEET AT DAWN" `shouldBe` "MPPR AE OYWY"
it "unVigenere decodes" $ do
unVigenere "ALLY" "MPPR AE OYWY" `shouldBe` "MEET AT DAWN"
prop_vigenereRoundTrip :: Property
prop_vigenereRoundTrip =
forAll validStringGen $ \ks ->
forAll validStringGen $ \xs -> unVigenere ks (vigenere ks xs) == xs
main :: IO ()
main = do
hspec vigenereSpec
quickCheck prop_caesarRoundTrip
quickCheck prop_vigenereRoundTrip
module NewVigenereIO where
import NewVigenere (unVigenere, vigenere)
import System.Environment (getArgs)
import System.IO (hGetLine, hPutStrLn, stdin, stdout)
main :: IO ()
main = do
[opt, key] <- getArgs
str <- hGetLine stdin
let out
| opt == "-e" = vigenere key str
| opt == "-d" = unVigenere key str
| otherwise = "Need to provide valid flag"
hPutStrLn stdout out
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment