Haskell Quiz/The Solitaire Cipher/Solution Mike McClurg

From HaskellWiki


module Main where

import Char (chr, ord, toUpper)
import Data.List (delete, elemIndices, findIndices)
import System.Environment (getArgs, getProgName)

-- Driver for the program
main :: IO ()
main = getArgs >>= parse

parse :: [String] -> IO ()
parse (key:[msg])      = putStrLn $ encodeWithKey key msg
parse ("-d":key:[msg]) = putStrLn $ decodeWithKey key msg
parse ["-h"]           = usage
parse _                = usage

usage :: IO ()
usage = do prog <- getProgName
           putStrLn $ "usage: " ++ prog ++ " [-d] <keyphrase> <message>"

-- Define the deck we'll be using
type Deck = [Card]

data Card = Card !Int
          | JokerA
          | JokerB
            deriving (Eq, Show)


-- Standard deck, in bridge order (0==Ace Spades, 51==King Clubs), jokers at end
mkStdDeck :: Deck
mkStdDeck = map Card [0..51] ++ [JokerA, JokerB]

-- Encode and decode
encode :: Deck -> String -> String
encode d msg = formatOutput $ map (uncurry add) $ zip (formatInput msg) $ keystream d

decode :: Deck -> String -> String
decode d cph = formatOutput $ map (uncurry add) $ zip (formatInput cph) $ map (0-) $ keystream d

encodeWithKey :: String -> String -> String
encodeWithKey key msg = encode (keyDeck key mkStdDeck) msg

decodeWithKey :: String -> String -> String
decodeWithKey key msg = decode (keyDeck key mkStdDeck) msg

-- Initialize the deck with the given key
keyDeck :: String -> Deck -> Deck
keyDeck []     = id
keyDeck (k:ks) = keyDeck ks . countCut' (charVal k) . countCut . tripleCut . moveJokers
    where charVal c = (ord $ toUpper c) - (ord 'A') + 1

-- Generate an infinite keystream
keystream :: Deck -> [Int]
keystream d = let d' = step d in
              case getOutput d' of
                Nothing -> keystream d' -- skip jokers
                Just i  -> i : keystream d'
    where step = countCut . tripleCut . moveJokers

-- Plaintext must have no spaces, and be padded to multiple of five
formatInput :: String -> String
formatInput s = pad $ map toUpper $ filter (/=' ') s
    where pad s = if (length s `mod` 5 == 0)
                  then s
                  else pad $ s ++ "X"

formatOutput :: String -> String
formatOutput [] = []
formatOutput cs = (take 5 cs) ++ " " ++ (formatOutput $ drop 5 cs)

-- Move jokers
moveJokers :: Deck -> Deck
moveJokers = moveJoker JokerB 2 . moveJoker JokerA 1

moveJoker :: Card -> Int -> Deck -> Deck
moveJoker j n d = a ++ [j] ++ b
    where n' = let i = (elemIndices j d)!!0 in
               if (i + n == 54) -- special case where joker would end up as top card; need to move under top one or two
                  then (i + n + 1) `mod` 54
                  else (i + n) `mod` 54
          (a,b) = splitAt n' $ delete j d

-- Triple cut: swap cards above first joker with cards below second joker
tripleCut :: Deck -> Deck
tripleCut d = let a = take j1 d                      -- first 'third'
                  b = take (j2 - j1 + 1) $ drop j1 d -- second 'third' (drop first third, then take up to next joker)
                  c = drop (j2 + 1) d in             -- third 'third'
              c ++ b ++ a
    where is = findIndices (\e -> (e==JokerA) || (e==JokerB)) d
          j1 = is!!0
          j2 = is!!1

-- Count cut: cut deck at n cards, where n is value of last card, leave last card in place
countCut :: Deck -> Deck
countCut d = countCut' i d
    where i = (cardVal $ last d)

countCut' :: Int -> Deck -> Deck
countCut' i d = (drop i d') ++ (take i d') ++ [l]
    where d' = take 53 d
          l  = last d

-- Return value of output card, or Nothing if joker
getOutput :: Deck -> Maybe Int
getOutput []     = Nothing
getOutput (c:cs) = let i = (cardVal c) in
                   case (c:cs)!!i of
                     JokerA   -> Nothing
                     JokerB   -> Nothing
                     (Card a) -> Just $ a+1

-- Int value of Card
cardVal :: Card -> Int
cardVal JokerA = 53
cardVal JokerB = 53
cardVal (Card c) = c + 1

-- Add Chars and Ints, modulo 26
add :: Char -> Int -> Char
add c i = intToChar $ i + charToInt c
    where charToInt c = (ord $ toUpper c) - ord 'A'
          intToChar i = chr $ i `mod` 26 + ord 'A'