diff options
author | Marvin Borner | 2024-09-05 15:34:20 +0200 |
---|---|---|
committer | Marvin Borner | 2024-09-06 14:46:03 +0200 |
commit | c95688c2fa63ba91df518ddf0d97261d6bd02426 (patch) | |
tree | 0c548056289d7551243dd73cb585f54d179403d1 /src/Error.hs | |
parent | b565350fb5f44f57dcb02a66ae99bab3b27313d3 (diff) |
Refactored Helper.hs
Diffstat (limited to 'src/Error.hs')
-rw-r--r-- | src/Error.hs | 94 |
1 files changed, 94 insertions, 0 deletions
diff --git a/src/Error.hs b/src/Error.hs new file mode 100644 index 0000000..0260aa1 --- /dev/null +++ b/src/Error.hs @@ -0,0 +1,94 @@ +-- MIT License, Copyright (c) 2022 Marvin Borner +{-# LANGUAGE BangPatterns #-} +{-# LANGUAGE RecordWildCards #-} +{-# LANGUAGE ScopedTypeVariables #-} + +module Error where + +import Data.List ( intercalate ) +import Text.Megaparsec + +import Helper + +errPrefix :: String +errPrefix = "\ESC[101m\ESC[30mERROR\ESC[0m " + +okPrefix :: String +okPrefix = "\ESC[102m\ESC[30m OK \ESC[0m " + +data Error = SyntaxError String + | UndefinedIdentifier Identifier + | UnmatchedMixfix [MixfixIdentifierKind] [Mixfix] + | InvalidIndex Int + | FailedTest Expression Expression Expression Expression + | PassedTest Expression Expression + | ContextualError Error Context + | SuggestSolution Error String + | ImportError String + | OptimizerError String + +instance Show Error where + show (ContextualError err ctx) = show err <> "\n" <> printContext ctx + show (SuggestSolution err sol) = + show err <> "\n\ESC[102m\ESC[30msuggestion\ESC[0m Perhaps you meant " <> sol + show (SyntaxError err) = + errPrefix <> "invalid syntax\n\ESC[105m\ESC[30mnear\ESC[0m " <> err + show (UndefinedIdentifier ident) = + errPrefix <> "undefined identifier " <> show ident + show (UnmatchedMixfix ks ms) = + errPrefix + <> "couldn't find matching mixfix for " + <> intercalate "" (map show ks) + <> "\n\ESC[105m\ESC[30mnear\ESC[0m " + <> unwords (map show ms) + show (InvalidIndex err) = errPrefix <> "invalid index " <> show err + show (PassedTest exp1 exp2) = + okPrefix <> "test passed: " <> show exp1 <> " = " <> show exp2 + show (FailedTest exp1 exp2 red1 red2) = + errPrefix + <> "test failed: " + <> show exp1 + <> " = " + <> show exp2 + <> "\n reduced to " + <> show red1 + <> " = " + <> show red2 + show (ImportError path) = errPrefix <> "invalid import " <> show path + show (OptimizerError msg ) = errPrefix <> "optimizer failed: " <> msg + +type Failable = Either Error + +-- Modified from megaparsec's errorBundlePretty +printBundle + :: forall s e + . (VisualStream s, TraversableStream s, ShowErrorComponent e) + => ParseErrorBundle s e + -> String +printBundle ParseErrorBundle {..} = + let (r, _) = foldl f (id, bundlePosState) bundleErrors in drop 1 (r "") + where + f :: (ShowS, PosState s) -> ParseError s e -> (ShowS, PosState s) + f (o, !pst) e = (o . (outChunk ++), pst') + where + (msline, pst') = reachOffset (errorOffset e) pst + epos = pstateSourcePos pst' + outChunk = "\n\n" <> offendingLine <> init (parseErrorTextPretty e) + offendingLine = case msline of + Nothing -> "" + Just sline -> + let pointer = "^" + rpadding = replicate rpshift ' ' + rpshift = unPos (sourceColumn epos) - 2 + lineNumber = (show . unPos . sourceLine) epos + padding = replicate (length lineNumber + 1) ' ' + in padding + <> "|\n" + <> " | " + <> sline + <> "\n" + <> padding + <> "| " + <> rpadding + <> pointer + <> "\n" |