-- | The CSV (comma-separated value) format is defined by RFC 4180,
--   \"Common Format and MIME Type for Comma-Separated Values (CSV) Files\",
--   <http://www.rfc-editor.org/rfc/rfc4180.txt>
--
--   This lazy parser can report all CSV formatting errors, whilst also
--   returning all the valid data, so the user can choose whether to
--   continue, to show warnings, or to halt on error.
--
--   Valid fields retain information about their original location in the
--   input, so a secondary parser from textual fields to typed values
--   can give intelligent error messages.
--
--   In a valid CSV file, all rows must have the same number of columns.
--   This parser will flag a row with the wrong number of columns as a error.
--   (But the error type contains the actual data, so the user can recover
--   it if desired.)  Completely blank lines are also treated as errors,
--   and again the user is free either to filter these out or convert them
--   to a row of actual null fields.

module Text.CSV.Lazy.String
  ( -- * CSV types
    CSVTable
  , CSVRow
  , CSVField(..)
    -- * CSV parsing
  , CSVError(..)
  , CSVResult
  , csvErrors
  , csvTable
  , parseCSV
  , parseDSV
    -- * Pretty-printing
  , ppCSVError
  , ppCSVField
  , ppCSVTable
  , ppDSVTable
    -- * Conversion between standard and simple representations
  , fromCSVTable
  , toCSVTable
    -- * Selection, validation, and algebra of CSV tables
  , selectFields
  , expectFields
  , mkEmptyColumn
  , joinCSV
  ) where

import Data.List (groupBy, partition, elemIndex, intercalate, takeWhile)
import Data.Function (on)
import Data.Maybe (fromJust)

-- | A CSV table is a sequence of rows.  All rows have the same number
--   of fields.
type CSVTable   = [CSVRow]

-- | A CSV row is just a sequence of fields.
type CSVRow     = [CSVField]

-- | A CSV field's content is stored with its logical row and column number,
--   as well as its textual extent.  This information is necessary if you
--   want to generate good error messages in a secondary parsing stage,
--   should you choose to convert the textual fields to typed data values.
data CSVField   = CSVField       { csvRowNum        :: !Int
                                 , csvColNum        :: !Int
                                 , csvTextStart     :: !(Int,Int)
                                 , csvTextEnd       :: !(Int,Int)
                                 , csvFieldContent  :: !String
                                 , csvFieldQuoted   :: !Bool }
                | CSVFieldError  { csvRowNum        :: !Int
                                 , csvColNum        :: !Int
                                 , csvTextStart     :: !(Int,Int)
                                 , csvTextEnd       :: !(Int,Int)
                                 , csvFieldError    :: !String }
                                                    deriving (Eq,Show)

-- | A structured error type for CSV formatting mistakes.
data CSVError   = IncorrectRow   { csvRow           :: !Int
                                 , csvColsExpected  :: !Int
                                 , csvColsActual    :: !Int
                                 , csvFields        :: [CSVField] }
                | BlankLine      { csvRow           :: !Int
                                 , csvColsExpected  :: !Int
                                 , csvColsActual    :: !Int
                                 , csvField         :: CSVField }
                | FieldError     { csvField         :: CSVField }
                | NoData
                                                    deriving (Eq,Show)

-- | The result of parsing a CSV input is a mixed collection of errors
--   and valid rows.  This way of representing things is crucial to the
--   ability to parse lazily whilst still catching format errors.
type CSVResult  = [Either [CSVError] CSVRow]

-- | Extract just the valid portions of a CSV parse.
csvTable    :: CSVResult -> CSVTable
csvTable  r  = [ v | Right v <- r ]
-- | Extract just the errors from a CSV parse.
csvErrors   :: CSVResult -> [CSVError]
csvErrors r  = concat [ v | Left  v <- r ]


-- | A first-stage parser for CSV (comma-separated values) data.
--   The individual fields remain as text, but errors in CSV formatting
--   are reported.  Errors (containing unrecognisable rows/fields) are
--   interspersed with the valid rows/fields.
parseCSV :: String -> CSVResult
parseCSV = parseDSV True ','

-- | Sometimes CSV is not comma-separated, but delimiter-separated
--   values (DSV).  The choice of delimiter is arbitrary, but semi-colon
--   is common in locales where comma is used as a decimal point, and tab
--   is also common.  The Boolean argument is
--   whether newlines should be accepted within quoted fields.  The CSV RFC
--   says newlines can occur in quotes, but other DSV formats might say
--   otherwise.  You can often get better error messages if newlines are
--   disallowed.
parseDSV :: Bool -> Char -> String -> CSVResult
parseDSV qn delim = validate
                    . groupBy ((==)`on`csvRowNum)
                    . lexCSV qn delim

validate          :: [CSVRow] -> CSVResult
validate []        = [Left [NoData]]
validate xs@(x:_)  = map (extractErrs (length x)) xs

extractErrs       :: Int -> CSVRow -> Either [CSVError] CSVRow
extractErrs size row
    | length row0 == size && null errs0    = Right row0
    | length row0 == 1    && empty field0  = Left [blankLine field0]
    | otherwise                            = Left (map convert errs0
                                                   ++ validateColumns row0)
  where
  (row0,errs0)   = partition isField row
  (field0:_)     = row0

  isField (CSVField{})      = True
  isField (CSVFieldError{}) = False

  empty   f@(CSVField{})    = null (csvFieldContent f)
  empty   _                 = False

  convert err = FieldError {csvField = err}

  validateColumns r  =
      if length r == size then []
      else [ IncorrectRow{ csvRow  = if null r then 0 else csvRowNum (head r)
                         , csvColsExpected  = size
                         , csvColsActual    = length r
                         , csvFields        = r } ]
  blankLine f = BlankLine{ csvRow           = csvRowNum f
                         , csvColsExpected  = size
                         , csvColsActual    = 1
                         , csvField         = f }


-- Reading CSV data is essentially lexical, and can be implemented with a
-- simple finite state machine.  We keep track of logical row number,
-- logical column number (in tabular terms), and textual position (row,col)
-- to enable good error messages.
-- Positional data is retained even after successful lexing, in case a
-- second-stage field parser wants to complain.
--
-- A double-quoted CSV field may contain commas, newlines, and double quotes.

data CSVState  = CSVState  { tableRow, tableCol  :: !Int
                           , textRow,  textCol   :: !Int }

incTableRow, incTableCol, incTextRow, incTextCol :: CSVState -> CSVState
incTableRow  st = st { tableRow  = tableRow  st + 1 }
incTableCol  st = st { tableCol  = tableCol  st + 1 }
incTextRow   st = st { textRow   = textRow   st + 1 }
incTextCol   st = st { textCol   = textCol   st + 1 }

-- Lexer is a small finite state machine.
lexCSV :: Bool -> Char -> [Char] -> [CSVField]
lexCSV quotedNewline delim =
    simple (CSVState{tableRow=1,tableCol=1,textRow=1,textCol=1}) (1,1) []
  where
  -- 'simple' recognises an unquoted field, and delimiter char as separator
  simple :: CSVState -> (Int,Int) -> String -> String -> [CSVField]
  simple _ _     []         []    = []
  simple s begin acc        []    = mkField s begin acc False : []
  simple s begin acc     (c:cs)
            | not (interesting c) = simple (incTextCol $! s) begin (c:acc) cs
  simple s begin acc  (c:'"':cs)
                      | c==delim  = mkField s begin acc False :
                                    string s' (textRow s',textCol s') [] cs
                                    where s' = incTextCol . incTextCol .
                                               incTableCol $! s
  simple s begin acc  (c:cs)
                      | c==delim  = mkField s begin acc False :
                                    simple s' (textRow s',textCol s') [] cs
                                    where s' = incTableCol . incTextCol $! s
  simple s begin acc  ('\r':'\n':cs)
                                  = mkField s begin acc False :
                                    simple s' (textRow s',1) [] cs
                                    where s' = incTableRow . incTextRow $!
                                               s {tableCol=1, textCol=1}
  simple s begin acc  ('\n' :cs)  = mkField s begin acc False :
                                    simple s' (textRow s',1) [] cs
                                    where s' = incTableRow . incTextRow $!
                                               s {tableCol=1, textCol=1}
  simple s begin []   ('"'  :cs)  = string (incTextCol $! s) begin [] cs
  simple s begin acc  ('"'  :cs)  = mkError s begin
                                            "Start-quote not next to comma":
                                    string (incTextCol $! s) begin acc cs

  -- 'string' recognises a double-quoted field containing commas and newlines
  string :: CSVState -> (Int,Int) -> String -> String -> [CSVField]
  string s begin []   []          = mkError s begin "Data ends at start-quote":
                                    []
  string s begin acc  []          = mkError s begin "Data ends in quoted field":
                                    []
  string s begin acc   (c:cs)
    | not (interestingInString c) = string (incTextCol $! s) begin (c:acc) cs
  string s begin acc ('"':'"':cs) = string s' begin ('"':acc) cs
                                    where s' = incTextCol . incTextCol $! s
  string s begin acc ('"':c:'"':cs)
                       | c==delim = mkField s begin acc True :
                                    string s' (textRow s',textCol s') [] cs
                                    where s' = incTextCol . incTextCol .
                                               incTextCol . incTableCol $! s
  string s begin acc ('"':c:cs)
                       | c==delim = mkField s begin acc True :
                                    simple s' (textRow s',textCol s') [] cs
                                    where s' = incTextCol . incTextCol .
                                               incTableCol $! s
  string s begin acc ('"':'\n':cs)= mkField s begin acc True :
                                    simple s' (textRow s',1) [] cs
                                    where s' = incTableRow . incTextRow $!
                                               s {tableCol=1, textCol=1}
  string s begin acc ('"':'\r':'\n':cs)
                                  = mkField s begin acc True :
                                    simple s' (textRow s',1) [] cs
                                    where s' = incTableRow . incTextRow $!
                                               s {tableCol=1, textCol=1}
  string s begin acc ('"':cs)     = mkError s begin
                                            "End-quote not followed by comma":
                                    simple (incTextCol $! s) begin acc cs
  string s begin acc ('\r':'\n':cs)
                  | quotedNewline = string s' begin ('\n':acc) cs
                  | otherwise     = mkError s begin
                                            "Found newline within quoted field":
                                    simple s'' (textRow s'',textCol s'') [] cs
                                    where s'  = incTextRow $! s {textCol=1}
                                          s'' = incTableRow . incTextRow $!
                                                s {textCol=1, tableCol=1}
  string s begin acc ('\n' :cs)
                  | quotedNewline = string s' begin ('\n':acc) cs
                  | otherwise     = mkError s begin
                                            "Found newline within quoted field":
                                    simple s'' (textRow s'',textCol s'') [] cs
                                    where s'  = incTextRow $! s {textCol=1}
                                          s'' = incTableRow . incTextRow $!
                                                s {textCol=1, tableCol=1}

  interesting :: Char -> Bool
  interesting '\n' = True
  interesting '\r' = True
  interesting '"'  = True
  interesting c    = c==delim

  interestingInString :: Char -> Bool
  interestingInString '\n' = True
  interestingInString '\r' = True
  interestingInString '"'  = True
  interestingInString _    = False

  -- generate the lexical tokens representing either a field or an error
  mkField st begin f q =    CSVField { csvRowNum       = tableRow st
                                     , csvColNum       = tableCol st
                                     , csvTextStart    = begin
                                     , csvTextEnd      = (textRow st,textCol st)
                                     , csvFieldContent = reverse f
                                     , csvFieldQuoted  = q }
  mkError st begin e = CSVFieldError { csvRowNum       = tableRow st
                                     , csvColNum       = tableCol st
                                     , csvTextStart    = begin
                                     , csvTextEnd      = (textRow st,textCol st)
                                     , csvFieldError   = e }

-- | Some pretty-printing for structured CSV errors.
ppCSVError :: CSVError -> String
ppCSVError (err@IncorrectRow{}) =
        "\nRow "++show (csvRow err)++" has wrong number of fields."++
        "\n    Expected "++show (csvColsExpected err)++" but got "++
        show (csvColsActual err)++"."++
        "\n    The fields are:"++
        indent 8 (concatMap ppCSVField (csvFields err))
ppCSVError (err@BlankLine{})  =
        "\nRow "++show (csvRow err)++" is blank."++
        "\n    Expected "++show (csvColsExpected err)++" fields."
ppCSVError (err@FieldError{}) = ppCSVField (csvField err)
ppCSVError (err@NoData{})     =
        "\nNo usable data (after accounting for any other errors)."

-- | Pretty-printing for CSV fields, shows positional information in addition
--   to the textual content.
ppCSVField :: CSVField -> String
ppCSVField (f@CSVField{}) =
        "\n"++quoted (csvFieldQuoted f) (csvFieldContent f)++
        "\nin row "++show (csvRowNum f)++" at column "++show (csvColNum f)++
        " (textually from "++show (csvTextStart f)++" to "++
        show (csvTextEnd f)++")"
ppCSVField (f@CSVFieldError{}) =
        "\n"++csvFieldError f++
        "\nin row "++show (csvRowNum f)++" at column "++show (csvColNum f)++
        " (textually from "++show (csvTextStart f)++" to "++
        show (csvTextEnd f)++")"

-- | Turn a full CSV table back into text, as much like the original
--   input as possible,  e.g. preserving quoted/unquoted format of fields.
ppCSVTable :: CSVTable -> String
ppCSVTable = unlines . map (intercalate "," . map ppField)
  where ppField f = quoted (csvFieldQuoted f) (csvFieldContent f)

-- | Turn a full CSV table back into text, using the given delimiter
--   character.  Quoted/unquoted formatting of the original is preserved.
ppDSVTable :: Char -> CSVTable -> String
ppDSVTable delim = unlines . map (intercalate [delim] . map ppField)
  where ppField f = quoted (csvFieldQuoted f) (csvFieldContent f)


-- Some pp helpers - indent and quoted - should live elsewhere, in a
-- pretty-printing package.

indent :: Int -> String -> String
indent n = unlines . map (replicate n ' ' ++) . lines

quoted :: Bool -> String -> String
quoted False  s  = s
quoted True   s  = '"': escape s ++"\""
  where escape ('"':cs) = '"':'"': escape cs
        escape (c:cs)   = c: escape cs
        escape  []      = []


-- | Convert a CSV table to a simpler representation, by dropping all
--   the original location information.
fromCSVTable :: CSVTable -> [[String]]
fromCSVTable = map (map csvFieldContent)

-- | Convert a simple list of lists into a CSVTable by the addition of
--   logical locations.  (Textual locations are not so useful.)
--   Rows of varying lengths generate errors.  Fields that need
--   quotation marks are automatically marked as such.
toCSVTable   :: [[String]] -> ([CSVError], CSVTable)
toCSVTable []         = ([NoData], [])
toCSVTable rows@(r:_) = (\ (a,b)-> (concat a, b)) $
                        unzip (zipWith walk [1..] rows)
  where
    n            = length r
    walk        :: Int -> [String] -> ([CSVError], CSVRow)
    walk rnum [] = ( [blank rnum]
                   , map (\c-> newField rnum c "") [1..n])
    walk rnum cs = ( if length cs /= n then [bad rnum cs] else []
                   , zipWith (newField rnum) [1..n] cs )

    blank rnum =  BlankLine{ csvRow          = rnum
                           , csvColsExpected = n
                           , csvColsActual   = 0
                           , csvField        = newField rnum 0 ""
                           }
    bad r cs = IncorrectRow{ csvRow          = r
                           , csvColsExpected = n
                           , csvColsActual   = length cs
                           , csvFields       = zipWith (newField r) [1..] cs
                           }

-- | Select and/or re-arrange columns from a CSV table, based on names in the
--   header row of the table.  The original header row is re-arranged too.
--   The result is either a list of column names that were not present, or
--   the (possibly re-arranged) sub-table.
selectFields :: [String] -> CSVTable -> Either [String] CSVTable
selectFields names table
    | null table          = Left names
    | not (null missing)  = Left missing
    | otherwise           = Right (map select table)
  where
    header         = map csvFieldContent (head table)
    missing        = filter (`notElem` header) names
    reordering     = map (fromJust . (\n-> elemIndex n header)) names
    select fields  = map (fields!!) reordering

-- | Validate that the named columns of a table have exactly the names and
--   ordering given in the argument.
expectFields :: [String] -> CSVTable -> Either [String] CSVTable
expectFields names table
    | null table          = Left ["CSV table is empty"]
    | not (null missing)  = Left (map ("CSV table is missing field: "++)
                                       missing)
    | header /= names     = Left ["CSV columns are in the wrong order"
                                 ,"Expected: "++intercalate ", " names
                                 ,"Found:    "++intercalate ", " header]
    | otherwise           = Right table
  where
    header         = map csvFieldContent (head table)
    missing        = filter (`notElem` header) names

-- | A join operator, adds the columns of two tables together.
--   Precondition: the tables have the same number of rows.
joinCSV :: CSVTable -> CSVTable -> CSVTable
joinCSV = zipWith (++)

-- | A generator for a new CSV column, of arbitrary length.
--   The result can be joined to an existing table if desired.
mkEmptyColumn :: String -> CSVTable
mkEmptyColumn header = [newField 1 0 header] :
                       map (\n-> [newField n 0 ""]) [2..]

-- | Generate a fresh field with the given textual content.
--   The quoting flag is set automatically based on the text.
--   Textual extents are not particularly useful, since there was no original
--   input to refer to.
newField :: Int -> Int -> String -> CSVField
newField n c text = CSVField { csvRowNum       = n
                             , csvColNum       = c
                             , csvTextStart    = (0,0)
                             , csvTextEnd      = (length (filter (=='\n') text)
                                                 ,length . takeWhile (/='\n')
                                                         . reverse $ text )
                             , csvFieldContent = text
                             , csvFieldQuoted  = any (`elem`"\",\n\r") text
                             }