From 68ab0b4dd3d627756e10adb55cb16845b08d09d9 Mon Sep 17 00:00:00 2001 From: Nikolai Kudasov Date: Sat, 16 Dec 2023 13:39:11 +0300 Subject: [PATCH] Stop typechecking after a parse error in some file (avoid invalid cache) --- rzk/src/Language/Rzk/VSCode/Handlers.hs | 2 +- rzk/src/Language/Rzk/VSCode/Lsp.hs | 23 +++++++++++++++++------ 2 files changed, 18 insertions(+), 7 deletions(-) diff --git a/rzk/src/Language/Rzk/VSCode/Handlers.hs b/rzk/src/Language/Rzk/VSCode/Handlers.hs index 2744dcb85..5af6bb276 100644 --- a/rzk/src/Language/Rzk/VSCode/Handlers.hs +++ b/rzk/src/Language/Rzk/VSCode/Handlers.hs @@ -61,7 +61,7 @@ collectErrors :: [(FilePath, Either String Module)] -> ([(FilePath, String)], [( collectErrors [] = ([], []) collectErrors ((path, result) : paths) = case result of - Left err -> ((path, err) : errors, modules) + Left err -> ((path, err) : errors, []) Right module_ -> (errors, (path, module_) : modules) where (errors, modules) = collectErrors paths diff --git a/rzk/src/Language/Rzk/VSCode/Lsp.hs b/rzk/src/Language/Rzk/VSCode/Lsp.hs index 026b00d31..3f2ee860d 100644 --- a/rzk/src/Language/Rzk/VSCode/Lsp.hs +++ b/rzk/src/Language/Rzk/VSCode/Lsp.hs @@ -39,26 +39,37 @@ import Rzk.TypeCheck (defaultTypeCheck, maxDiagnosticCount :: Int maxDiagnosticCount = 100 +data IsChanged + = HasChanged + | NotChanged + -- | Detects if the given path has changes in its declaration compared to what's in the cache -hasNotChanged :: RzkTypecheckCache -> FilePath -> LSP Bool -hasNotChanged cache path = toBool $ do +isChanged :: RzkTypecheckCache -> FilePath -> LSP IsChanged +isChanged cache path = toIsChanged $ do cachedDecls <- maybeToEitherLSP $ lookup path cache module' <- toExceptTLifted $ parseModuleFile path e <- toExceptTLifted $ try @SomeException $ evaluate $ - defaultTypeCheck (typecheckModulesWithLocationIncremental (filter ((/= path) . fst) cache) [(path, module')]) + defaultTypeCheck (typecheckModulesWithLocationIncremental (takeWhile ((/= path) . fst) cache) [(path, module')]) (checkedModules, _errors) <- toExceptT $ return e decls' <- maybeToEitherLSP $ lookup path checkedModules - return (decls' == cachedDecls) + return $ if decls' == cachedDecls + then NotChanged + else HasChanged where toExceptT = modifyError (const ()) . ExceptT toExceptTLifted = toExceptT . liftIO maybeToEitherLSP = \case Nothing -> throwError () Just x -> return x - toBool m = runExceptT m >>= \case - Left _ -> return False + toIsChanged m = runExceptT m >>= \case + Left _ -> return HasChanged -- in case of error consider the file has changed Right x -> return x +hasNotChanged :: RzkTypecheckCache -> FilePath -> LSP Bool +hasNotChanged cache path = isChanged cache path >>= \case + HasChanged -> return False + NotChanged -> return True + -- | Monadic 'dropWhile' dropWhileM :: (Monad m) => (a -> m Bool) -> [a] -> m [a] dropWhileM _ [] = return []