From 4c8477d25e382337e86dff077b03d8fc418d4e7b Mon Sep 17 00:00:00 2001 From: Brent Yorgey Date: Mon, 30 Dec 2024 19:53:00 -0600 Subject: [PATCH] Add missing cases to var usage analysis (#2251) Fixes #2250. Some term constructors (records, type annotations) were missing, but no one ever realized since there was a catch-all `_ -> mempty` case at the end. This PR: - adds the missing cases - adds some associated tests - explicitly enumerates the other non-variable cases so that in the future, when we add a new term constructor we will get a warning. --- .../warnings/unused-vars/lambda-with-annot.sw | 1 + .../unused-vars/lambda-with-record-unused.sw | 1 + .../lambda-with-record-used-abbrev.sw | 1 + .../unused-vars/lambda-with-record-used.sw | 1 + src/swarm-lang/Swarm/Language/LSP/VarUsage.hs | 22 ++++++++++++++++++- test/unit/TestLSP.hs | 16 ++++++++++++++ 6 files changed, 41 insertions(+), 1 deletion(-) create mode 100644 data/test/language-snippets/warnings/unused-vars/lambda-with-annot.sw create mode 100644 data/test/language-snippets/warnings/unused-vars/lambda-with-record-unused.sw create mode 100644 data/test/language-snippets/warnings/unused-vars/lambda-with-record-used-abbrev.sw create mode 100644 data/test/language-snippets/warnings/unused-vars/lambda-with-record-used.sw diff --git a/data/test/language-snippets/warnings/unused-vars/lambda-with-annot.sw b/data/test/language-snippets/warnings/unused-vars/lambda-with-annot.sw new file mode 100644 index 000000000..45dd0ce31 --- /dev/null +++ b/data/test/language-snippets/warnings/unused-vars/lambda-with-annot.sw @@ -0,0 +1 @@ +def put = \y. place (y : Text); end; diff --git a/data/test/language-snippets/warnings/unused-vars/lambda-with-record-unused.sw b/data/test/language-snippets/warnings/unused-vars/lambda-with-record-unused.sw new file mode 100644 index 000000000..47ed29e95 --- /dev/null +++ b/data/test/language-snippets/warnings/unused-vars/lambda-with-record-unused.sw @@ -0,0 +1 @@ +def put = \y. [x = 3, z = 7]; end; diff --git a/data/test/language-snippets/warnings/unused-vars/lambda-with-record-used-abbrev.sw b/data/test/language-snippets/warnings/unused-vars/lambda-with-record-used-abbrev.sw new file mode 100644 index 000000000..dcd1de552 --- /dev/null +++ b/data/test/language-snippets/warnings/unused-vars/lambda-with-record-used-abbrev.sw @@ -0,0 +1 @@ +def put = \y. [x = 3, y]; end; diff --git a/data/test/language-snippets/warnings/unused-vars/lambda-with-record-used.sw b/data/test/language-snippets/warnings/unused-vars/lambda-with-record-used.sw new file mode 100644 index 000000000..520bc603c --- /dev/null +++ b/data/test/language-snippets/warnings/unused-vars/lambda-with-record-used.sw @@ -0,0 +1 @@ +def put = \y. [x = 3, z = y + 2]; end; diff --git a/src/swarm-lang/Swarm/Language/LSP/VarUsage.hs b/src/swarm-lang/Swarm/Language/LSP/VarUsage.hs index 868770ab6..09045c6bc 100644 --- a/src/swarm-lang/Swarm/Language/LSP/VarUsage.hs +++ b/src/swarm-lang/Swarm/Language/LSP/VarUsage.hs @@ -110,4 +110,24 @@ getUsage bindings (CSyntax _pos t _comments) = case t of Just v -> checkOccurrences bindings v Bind [s1, s2] Nothing -> getUsage bindings s1 <> getUsage bindings s2 SDelay s -> getUsage bindings s - _ -> mempty + SRcd m -> M.foldMapWithKey (\x -> maybe (getUsage bindings (STerm (TVar x))) (getUsage bindings)) m + SProj s _ -> getUsage bindings s + SAnnotate s _ -> getUsage bindings s + SSuspend s -> getUsage bindings s + -- Explicitly enumerate the cases with no variables, instead of a + -- catch-all, so that we get a warning when adding new constructors. + TUnit {} -> mempty + TConst {} -> mempty + TDir {} -> mempty + TInt {} -> mempty + TAntiInt {} -> mempty + TText {} -> mempty + TAntiText {} -> mempty + TBool {} -> mempty + TRobot {} -> mempty + TRef {} -> mempty + TRequireDevice {} -> mempty + TRequire {} -> mempty + SRequirements {} -> mempty + STydef {} -> mempty + TType {} -> mempty diff --git a/test/unit/TestLSP.hs b/test/unit/TestLSP.hs index 92bfe38ca..b252760b8 100644 --- a/test/unit/TestLSP.hs +++ b/test/unit/TestLSP.hs @@ -72,6 +72,22 @@ testLSP = checkFile "single-bind-used.sw" [] + , testCase "lambda with var used inside annotation" $ + checkFile + "lambda-with-annot.sw" + [] + , testCase "record with used var" $ + checkFile + "lambda-with-record-used.sw" + [] + , testCase "record with used var abbrev" $ + checkFile + "lambda-with-record-used-abbrev.sw" + [] + , testCase "record with unused var" $ + checkFile + "lambda-with-record-unused.sw" + [UnusedVar "y" VU.Lambda] ] where checkFile :: FilePath -> [UnusedVar] -> IO ()