import Data.Array.IArray
import Data.Array.Unboxed
import ArrayStuff
-import Text.CSV
+import TSV
-- Command-line interface with simple tab-separated input/output formats.
-- ./match <example.in
--- pretty silly but it does the job
-swapTabCommaIn s = map (\c -> if c == '\t' then ',' else if c == ',' then '\t' else c) s
-removeQuotes s = filter (\c -> not (c == '"')) s
-
--- Some versions of Text.CSV do not use or accept a trailing newline; compensate for that.
-removeTrailingNewline s = if not (null s) && last s == '\n' then init s else s
-addTrailingNewline s = if not (null s) && last s /= '\n' then s ++ ['\n'] else s
-
-parseTSV fname str = case parseCSV fname (swapTabCommaIn $ removeTrailingNewline str) of
- Left pe -> Left pe
- Right ll -> Right $ map (map swapTabCommaIn) ll
-printTSV ll = addTrailingNewline $ removeQuotes $ swapTabCommaIn $ printCSV $ map (map swapTabCommaIn) ll
-
main = do
incsv <- hGetContents stdin
-- handle errors another day, or let the platform do it
- let Right inll = parseTSV "standard input" incsv
+ let inll = parseTSV incsv
let loadList = head inll
let numRvrs = length loadList
let loadA = listArray (0, numRvrs-1) (map read loadList)
let prefA = funcArray ((0,0), (numRvrs-1,numProps-1)) (\(i,j) -> read (pxarr ! (j, i)))
let theInst = PMInstance numRvrs numProps loadA prefA
let PMatching theMatching = doMatching pmDefaults theInst
- hPutStr stdout $ printTSV $ map (\(i, j) -> map show [i, j]) theMatching
+ hPutStr stdout $ formatTSV $ map (\(i, j) -> map show [i, j]) theMatching
--- /dev/null
+module TSV (
+ parseTSV, formatTSV
+) where
+import Data.List
+
+-- Simple implementation of tab-separated values with a trailing newline and no
+-- quoting support.
+
+splitList :: Eq a => a -> Bool -> [a] -> [[a]]
+splitList delim terminated =
+ let sl [] | terminated = []
+ sl l = case break (== delim) l of
+ (hh, []) -> [hh]
+ (hh, {-delim-} _ : t) -> hh : sl t
+ in sl
+
+-- We use the Eq only to check the validity...
+joinList :: Eq a => a -> Bool -> [[a]] -> [a]
+joinList delim terminated l =
+ if any (any (== delim)) l then error "joinList: item contains delimiter"
+ else if terminated
+ then concatMap (++ [delim]) l
+ else intercalate [delim] l
+
+type TSV = [[String]]
+
+formatTSV :: TSV -> String
+formatTSV = joinList '\n' True . map (joinList '\t' False)
+
+parseTSV :: String -> TSV
+parseTSV = map (splitList '\t' False) . splitList '\n' True