module Language.CNF.Parse.ParseDIMACS
( parseCNF, CNF(..) ) where
import Text.ParserCombinators.Parsec
data CNF = CNF {numVars :: Int
,numClauses :: Int
,clauses :: [[Int]]}
deriving Show
parseCNF :: String
-> String
-> CNF
parseCNF title input =
case parse cnf title input of
Left parseError -> error $ "Parse error: " ++ show parseError
Right cs -> cs
cnf :: Parser CNF
cnf =
do many comment
char 'p' ; spaces
string "cnf" ; spaces
numVars <- many1 digit ; spaces
numClauses <- many1 digit
space `manyTill` newline
actualClauses <- many1 clause
return $ CNF (read numVars) (read numClauses) actualClauses
comment :: Parser String
comment = do char 'c' ; manyTill anyChar (try newline)
clause :: Parser [Int]
clause = do many (space <|> newline)
lits <- between (string "") (char '0') (many1 intSpaces)
many (space <|> newline)
return lits
intSpaces = do i <- int
many1 (space <|> newline)
return i
int :: Parser Int
int = do parity <- option '+' $ choice $ map char "+-"
first <- posDigit
rest <- many digit
return . read $
case parity of
'+' -> first : rest
'-' -> '-' : first : rest
_ -> error $ "unknown parity syntax: " ++ [parity]
posDigit :: Parser Char
posDigit = oneOf ['1'..'9']