mirror of
				https://github.com/koalaman/shellcheck.git
				synced 2025-10-25 18:20:41 +08:00 
			
		
		
		
	Separate out command specific checks.
The checks use a better interface and give an overall speed boost of 10%.
This commit is contained in:
		| @@ -52,7 +52,9 @@ library | ||||
|       ShellCheck.ASTLib | ||||
|       ShellCheck.Analytics | ||||
|       ShellCheck.Analyzer | ||||
|       ShellCheck.AnalyzerLib | ||||
|       ShellCheck.Checker | ||||
|       ShellCheck.Checks.Commands | ||||
|       ShellCheck.Data | ||||
|       ShellCheck.Formatter.Format | ||||
|       ShellCheck.Formatter.CheckStyle | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -19,9 +19,18 @@ | ||||
| -} | ||||
| module ShellCheck.Analyzer (analyzeScript) where | ||||
|  | ||||
| import ShellCheck.Interface | ||||
| import ShellCheck.Analytics | ||||
| import ShellCheck.AnalyzerLib | ||||
| import ShellCheck.Interface | ||||
| import Data.List | ||||
| import qualified ShellCheck.Checks.Commands | ||||
|  | ||||
|  | ||||
| -- TODO: Clean up the cruft this is layered on | ||||
| analyzeScript :: AnalysisSpec -> AnalysisResult | ||||
| analyzeScript = runAnalytics | ||||
| analyzeScript spec = AnalysisResult { | ||||
|     arComments = | ||||
|         filterByAnnotation (asScript spec) . nub $ | ||||
|             runAnalytics spec | ||||
|             ++ ShellCheck.Checks.Commands.runChecks spec | ||||
| } | ||||
|   | ||||
							
								
								
									
										624
									
								
								ShellCheck/AnalyzerLib.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										624
									
								
								ShellCheck/AnalyzerLib.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,624 @@ | ||||
| {- | ||||
|     Copyright 2012-2015 Vidar Holen | ||||
|  | ||||
|     This file is part of ShellCheck. | ||||
|     http://www.vidarholen.net/contents/shellcheck | ||||
|  | ||||
|     ShellCheck is free software: you can redistribute it and/or modify | ||||
|     it under the terms of the GNU General Public License as published by | ||||
|     the Free Software Foundation, either version 3 of the License, or | ||||
|     (at your option) any later version. | ||||
|  | ||||
|     ShellCheck is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public License | ||||
|     along with this program.  If not, see <http://www.gnu.org/licenses/>. | ||||
| -} | ||||
| {-# LANGUAGE TemplateHaskell #-} -- prop_testing | ||||
| {-# LANGUAGE FlexibleContexts #-} | ||||
| module ShellCheck.AnalyzerLib where | ||||
| import ShellCheck.AST | ||||
| import ShellCheck.ASTLib | ||||
| import ShellCheck.Data | ||||
| import ShellCheck.Interface | ||||
| import ShellCheck.Parser | ||||
| import ShellCheck.Regex | ||||
|  | ||||
| import Control.Arrow (first) | ||||
| import Control.Monad.Identity | ||||
| import Control.Monad.Reader | ||||
| import Control.Monad.State | ||||
| import Control.Monad.Writer | ||||
| import Data.Char | ||||
| import Data.List | ||||
| import Data.Maybe | ||||
| import qualified Data.Map as Map | ||||
|  | ||||
| import Test.QuickCheck.All (forAllProperties) -- prop_testing | ||||
| import Test.QuickCheck.Test (quickCheckWithResult, stdArgs, maxSuccess) --prop_testing | ||||
|  | ||||
| type Analysis = ReaderT Parameters (Writer [TokenComment]) () | ||||
|  | ||||
|  | ||||
| data Parameters = Parameters { | ||||
|     variableFlow :: [StackData], | ||||
|     parentMap :: Map.Map Id Token, | ||||
|     shellType :: Shell, | ||||
|     shellTypeSpecified :: Bool | ||||
|     } | ||||
|  | ||||
| data Scope = SubshellScope String | NoneScope deriving (Show, Eq) | ||||
| data StackData = | ||||
|     StackScope Scope | ||||
|     | StackScopeEnd | ||||
|     -- (Base expression, specific position, var name, assigned values) | ||||
|     | Assignment (Token, Token, String, DataType) | ||||
|     | Reference (Token, Token, String) | ||||
|   deriving (Show) | ||||
|  | ||||
| data DataType = DataString DataSource | DataArray DataSource | ||||
|   deriving (Show) | ||||
|  | ||||
| data DataSource = SourceFrom [Token] | SourceExternal | SourceDeclaration | SourceInteger | ||||
|   deriving (Show) | ||||
|  | ||||
| data VariableState = Dead Token String | Alive deriving (Show) | ||||
|  | ||||
| defaultSpec root = AnalysisSpec { | ||||
|     asScript = root, | ||||
|     asShellType = Nothing, | ||||
|     asExecutionMode = Executed | ||||
| } | ||||
|  | ||||
| pScript s = | ||||
|   let | ||||
|     pSpec = ParseSpec { | ||||
|         psFilename = "script", | ||||
|         psScript = s | ||||
|     } | ||||
|   in prRoot . runIdentity $ parseScript (mockedSystemInterface []) pSpec | ||||
|  | ||||
| makeComment :: Severity -> Id -> Code -> String -> TokenComment | ||||
| makeComment severity id code note = | ||||
|     TokenComment id $ Comment severity code note | ||||
|  | ||||
| addComment note = tell [note] | ||||
|  | ||||
| warn :: MonadWriter [TokenComment] m => Id -> Code -> String -> m () | ||||
| warn  id code str = addComment $ makeComment WarningC id code str | ||||
| err   id code str = addComment $ makeComment ErrorC id code str | ||||
| info  id code str = addComment $ makeComment InfoC id code str | ||||
| style id code str = addComment $ makeComment StyleC id code str | ||||
|  | ||||
| makeParameters spec = | ||||
|     let params = Parameters { | ||||
|         shellType = fromMaybe (determineShell root) $ asShellType spec, | ||||
|         shellTypeSpecified = isJust $ asShellType spec, | ||||
|         parentMap = getParentTree root, | ||||
|         variableFlow = | ||||
|             getVariableFlow (shellType params) (parentMap params) root | ||||
|     } in params | ||||
|   where root = asScript spec | ||||
|  | ||||
| prop_determineShell0 = determineShell (fromJust $ pScript "#!/bin/sh") == Sh | ||||
| prop_determineShell1 = determineShell (fromJust $ pScript "#!/usr/bin/env ksh") == Ksh | ||||
| prop_determineShell2 = determineShell (fromJust $ pScript "") == Bash | ||||
| prop_determineShell3 = determineShell (fromJust $ pScript "#!/bin/sh -e") == Sh | ||||
| prop_determineShell4 = determineShell (fromJust $ pScript | ||||
|     "#!/bin/ksh\n#shellcheck shell=sh\nfoo") == Sh | ||||
| prop_determineShell5 = determineShell (fromJust $ pScript | ||||
|     "#shellcheck shell=sh\nfoo") == Sh | ||||
| determineShell t = fromMaybe Bash $ do | ||||
|     shellString <- foldl mplus Nothing $ getCandidates t | ||||
|     shellForExecutable shellString | ||||
|   where | ||||
|     forAnnotation t = | ||||
|         case t of | ||||
|             (ShellOverride s) -> return s | ||||
|             _ -> fail "" | ||||
|     getCandidates :: Token -> [Maybe String] | ||||
|     getCandidates t@(T_Script {}) = [Just $ fromShebang t] | ||||
|     getCandidates (T_Annotation _ annotations s) = | ||||
|         map forAnnotation annotations ++ | ||||
|            [Just $ fromShebang s] | ||||
|     fromShebang (T_Script _ s t) = shellFor s | ||||
|  | ||||
|     shellFor s | "/env " `isInfixOf` s = head (drop 1 (words s)++[""]) | ||||
|     shellFor s | ' ' `elem` s = shellFor $ takeWhile (/= ' ') s | ||||
|     shellFor s = reverse . takeWhile (/= '/') . reverse $ s | ||||
|  | ||||
|  | ||||
| --- Context seeking | ||||
|  | ||||
| getParentTree t = | ||||
|     snd . snd $ runState (doStackAnalysis pre post t) ([], Map.empty) | ||||
|   where | ||||
|     pre t = modify (first ((:) t)) | ||||
|     post t = do | ||||
|         (_:rest, map) <- get | ||||
|         case rest of [] -> put (rest, map) | ||||
|                      (x:_) -> put (rest, Map.insert (getId t) x map) | ||||
|  | ||||
| getTokenMap t = | ||||
|     execState (doAnalysis f t) Map.empty | ||||
|   where | ||||
|     f t = modify (Map.insert (getId t) t) | ||||
|  | ||||
|  | ||||
| -- Is this node self quoting for a regular element? | ||||
| isQuoteFree = isQuoteFreeNode False | ||||
|  | ||||
| -- Is this node striclty self quoting, for array expansions | ||||
| isStrictlyQuoteFree = isQuoteFreeNode True | ||||
|  | ||||
|  | ||||
| isQuoteFreeNode strict tree t = | ||||
|     (isQuoteFreeElement t == Just True) || | ||||
|         head (mapMaybe isQuoteFreeContext (drop 1 $ getPath tree t) ++ [False]) | ||||
|   where | ||||
|     -- Is this node self-quoting in itself? | ||||
|     isQuoteFreeElement t = | ||||
|         case t of | ||||
|             T_Assignment {} -> return True | ||||
|             T_FdRedirect {} -> return True | ||||
|             _ -> Nothing | ||||
|  | ||||
|     -- Are any subnodes inherently self-quoting? | ||||
|     isQuoteFreeContext t = | ||||
|         case t of | ||||
|             TC_Noary _ DoubleBracket _ -> return True | ||||
|             TC_Unary _ DoubleBracket _ _ -> return True | ||||
|             TC_Binary _ DoubleBracket _ _ _ -> return True | ||||
|             TA_Sequence {} -> return True | ||||
|             T_Arithmetic {} -> return True | ||||
|             T_Assignment {} -> return True | ||||
|             T_Redirecting {} -> return $ | ||||
|                 if strict then False else | ||||
|                     -- Not true, just a hack to prevent warning about non-expansion refs | ||||
|                     any (isCommand t) ["local", "declare", "typeset", "export", "trap", "readonly"] | ||||
|             T_DoubleQuoted _ _ -> return True | ||||
|             T_DollarDoubleQuoted _ _ -> return True | ||||
|             T_CaseExpression {} -> return True | ||||
|             T_HereDoc {} -> return True | ||||
|             T_DollarBraced {} -> return True | ||||
|             -- When non-strict, pragmatically assume it's desirable to split here | ||||
|             T_ForIn {} -> return (not strict) | ||||
|             T_SelectIn {} -> return (not strict) | ||||
|             _ -> Nothing | ||||
|  | ||||
| isParamTo tree cmd = | ||||
|     go | ||||
|   where | ||||
|     go x = case Map.lookup (getId x) tree of | ||||
|                 Nothing -> False | ||||
|                 Just parent -> check parent | ||||
|     check t = | ||||
|         case t of | ||||
|             T_SingleQuoted _ _ -> go t | ||||
|             T_DoubleQuoted _ _ -> go t | ||||
|             T_NormalWord _ _ -> go t | ||||
|             T_SimpleCommand {} -> isCommand t cmd | ||||
|             T_Redirecting {} -> isCommand t cmd | ||||
|             _ -> False | ||||
|  | ||||
| getClosestCommand tree t = | ||||
|     msum . map getCommand $ getPath tree t | ||||
|   where | ||||
|     getCommand t@(T_Redirecting {}) = return t | ||||
|     getCommand _ = Nothing | ||||
|  | ||||
| usedAsCommandName tree token = go (getId token) (tail $ getPath tree token) | ||||
|   where | ||||
|     go currentId (T_NormalWord id [word]:rest) | ||||
|         | currentId == getId word = go id rest | ||||
|     go currentId (T_DoubleQuoted id [word]:rest) | ||||
|         | currentId == getId word = go id rest | ||||
|     go currentId (T_SimpleCommand _ _ (word:_):_) | ||||
|         | currentId == getId word = True | ||||
|     go _ _ = False | ||||
|  | ||||
| -- A list of the element and all its parents | ||||
| getPath tree t = t : | ||||
|     case Map.lookup (getId t) tree of | ||||
|         Nothing -> [] | ||||
|         Just parent -> getPath tree parent | ||||
|  | ||||
| isParentOf tree parent child = | ||||
|     elem (getId parent) . map getId $ getPath tree child | ||||
|  | ||||
| parents params = getPath (parentMap params) | ||||
|  | ||||
| pathTo t = do | ||||
|     parents <- reader parentMap | ||||
|     return $ getPath parents t | ||||
|  | ||||
| -- Check whether a word is entirely output from a single command | ||||
| tokenIsJustCommandOutput t = case t of | ||||
|     T_NormalWord id [T_DollarExpansion _ cmds] -> check cmds | ||||
|     T_NormalWord id [T_DoubleQuoted _ [T_DollarExpansion _ cmds]] -> check cmds | ||||
|     T_NormalWord id [T_Backticked _ cmds] -> check cmds | ||||
|     T_NormalWord id [T_DoubleQuoted _ [T_Backticked _ cmds]] -> check cmds | ||||
|     _ -> False | ||||
|   where | ||||
|     check [x] = not $ isOnlyRedirection x | ||||
|     check _ = False | ||||
|  | ||||
| -- TODO: Replace this with a proper Control Flow Graph | ||||
| getVariableFlow shell parents t = | ||||
|     let (_, stack) = runState (doStackAnalysis startScope endScope t) [] | ||||
|     in reverse stack | ||||
|   where | ||||
|     startScope t = | ||||
|         let scopeType = leadType shell parents t | ||||
|         in do | ||||
|             when (scopeType /= NoneScope) $ modify (StackScope scopeType:) | ||||
|             when (assignFirst t) $ setWritten t | ||||
|  | ||||
|     endScope t = | ||||
|         let scopeType = leadType shell parents t | ||||
|         in do | ||||
|             setRead t | ||||
|             unless (assignFirst t) $ setWritten t | ||||
|             when (scopeType /= NoneScope) $ modify (StackScopeEnd:) | ||||
|  | ||||
|     assignFirst (T_ForIn {}) = True | ||||
|     assignFirst (T_SelectIn {}) = True | ||||
|     assignFirst _ = False | ||||
|  | ||||
|     setRead t = | ||||
|         let read    = getReferencedVariables t | ||||
|         in mapM_ (\v -> modify (Reference v:)) read | ||||
|  | ||||
|     setWritten t = | ||||
|         let written = getModifiedVariables t | ||||
|         in mapM_ (\v -> modify (Assignment v:)) written | ||||
|  | ||||
|  | ||||
| leadType shell parents t = | ||||
|     case t of | ||||
|         T_DollarExpansion _ _  -> SubshellScope "$(..) expansion" | ||||
|         T_Backticked _ _  -> SubshellScope "`..` expansion" | ||||
|         T_Backgrounded _ _  -> SubshellScope "backgrounding &" | ||||
|         T_Subshell _ _  -> SubshellScope "(..) group" | ||||
|         T_CoProcBody _ _  -> SubshellScope "coproc" | ||||
|         T_Redirecting {}  -> | ||||
|             if fromMaybe False causesSubshell | ||||
|             then SubshellScope "pipeline" | ||||
|             else NoneScope | ||||
|         _ -> NoneScope | ||||
|   where | ||||
|     parentPipeline = do | ||||
|         parent <- Map.lookup (getId t) parents | ||||
|         case parent of | ||||
|             T_Pipeline {} -> return parent | ||||
|             _ -> Nothing | ||||
|  | ||||
|     causesSubshell = do | ||||
|         (T_Pipeline _ _ list) <- parentPipeline | ||||
|         if length list <= 1 | ||||
|             then return False | ||||
|             else if lastCreatesSubshell | ||||
|                 then return True | ||||
|                 else return . not $ (getId . head $ reverse list) == getId t | ||||
|  | ||||
|     lastCreatesSubshell = | ||||
|         case shell of | ||||
|             Bash -> True | ||||
|             Dash -> True | ||||
|             Sh -> True | ||||
|             Ksh -> False | ||||
|  | ||||
| getModifiedVariables t = | ||||
|     case t of | ||||
|         T_SimpleCommand _ vars [] -> | ||||
|             concatMap (\x -> case x of | ||||
|                                 T_Assignment id _ name _ w  -> | ||||
|                                     [(x, x, name, dataTypeFrom DataString w)] | ||||
|                                 _ -> [] | ||||
|                       ) vars | ||||
|         c@(T_SimpleCommand {}) -> | ||||
|             getModifiedVariableCommand c | ||||
|  | ||||
|         TA_Unary _ "++|" var -> maybeToList $ do | ||||
|             name <- getLiteralString var | ||||
|             return (t, t, name, DataString $ SourceFrom [t]) | ||||
|         TA_Unary _ "|++" var -> maybeToList $ do | ||||
|             name <- getLiteralString var | ||||
|             return (t, t, name, DataString $ SourceFrom [t]) | ||||
|         TA_Binary _ op lhs rhs -> maybeToList $ do | ||||
|             guard $ op `elem` ["=", "*=", "/=", "%=", "+=", "-=", "<<=", ">>=", "&=", "^=", "|="] | ||||
|             name <- getLiteralString lhs | ||||
|             return (t, t, name, DataString $ SourceFrom [rhs]) | ||||
|  | ||||
|         t@(T_FdRedirect _ ('{':var) op) -> -- {foo}>&2 modifies foo | ||||
|             [(t, t, takeWhile (/= '}') var, DataString SourceInteger) | not $ isClosingFileOp op] | ||||
|  | ||||
|         t@(T_CoProc _ name _) -> | ||||
|             [(t, t, fromMaybe "COPROC" name, DataArray SourceInteger)] | ||||
|  | ||||
|         --Points to 'for' rather than variable | ||||
|         T_ForIn id str words _ -> [(t, t, str, DataString $ SourceFrom words)] | ||||
|         T_SelectIn id str words _ -> [(t, t, str, DataString $ SourceFrom words)] | ||||
|         _ -> [] | ||||
|  | ||||
| isClosingFileOp op = | ||||
|     case op of | ||||
|         T_IoFile _ (T_GREATAND _) (T_NormalWord _ [T_Literal _ "-"]) -> True | ||||
|         T_IoFile _ (T_LESSAND  _) (T_NormalWord _ [T_Literal _ "-"]) -> True | ||||
|         _ -> False | ||||
|  | ||||
|  | ||||
| -- Consider 'export/declare -x' a reference, since it makes the var available | ||||
| getReferencedVariableCommand base@(T_SimpleCommand _ _ (T_NormalWord _ (T_Literal _ x:_):rest)) = | ||||
|     case x of | ||||
|         "export" -> if "f" `elem` flags | ||||
|             then [] | ||||
|             else concatMap getReference rest | ||||
|         "declare" -> if any (`elem` flags) ["x", "p"] | ||||
|             then concatMap getReference rest | ||||
|             else [] | ||||
|         "readonly" -> concatMap getReference rest | ||||
|         "trap" -> | ||||
|             case rest of | ||||
|                 head:_ -> map (\x -> (head, head, x)) $ getVariablesFromLiteralToken head | ||||
|                 _ -> [] | ||||
|         _ -> [] | ||||
|   where | ||||
|     getReference t@(T_Assignment _ _ name _ value) = [(t, t, name)] | ||||
|     getReference t@(T_NormalWord _ [T_Literal _ name]) | not ("-" `isPrefixOf` name) = [(t, t, name)] | ||||
|     getReference _ = [] | ||||
|     flags = map snd $ getAllFlags base | ||||
|  | ||||
| getReferencedVariableCommand _ = [] | ||||
|  | ||||
| getModifiedVariableCommand base@(T_SimpleCommand _ _ (T_NormalWord _ (T_Literal _ x:_):rest)) = | ||||
|    filter (\(_,_,s,_) -> not ("-" `isPrefixOf` s)) $ | ||||
|     case x of | ||||
|         "read" -> | ||||
|             let params = map getLiteral rest in | ||||
|                 catMaybes . takeWhile isJust . reverse $ params | ||||
|         "getopts" -> | ||||
|             case rest of | ||||
|                 opts:var:_ -> maybeToList $ getLiteral var | ||||
|                 _ -> [] | ||||
|  | ||||
|         "let" -> concatMap letParamToLiteral rest | ||||
|  | ||||
|         "export" -> | ||||
|             if "f" `elem` flags then [] else concatMap getModifierParamString rest | ||||
|  | ||||
|         "declare" -> if any (`elem` flags) ["F", "f", "p"] then [] else declaredVars | ||||
|         "typeset" -> declaredVars | ||||
|  | ||||
|         "local" -> concatMap getModifierParamString rest | ||||
|         "readonly" -> concatMap getModifierParamString rest | ||||
|         "set" -> maybeToList $ do | ||||
|             params <- getSetParams rest | ||||
|             return (base, base, "@", DataString $ SourceFrom params) | ||||
|  | ||||
|         "printf" -> maybeToList $ getPrintfVariable rest | ||||
|  | ||||
|         "mapfile" -> maybeToList $ getMapfileArray base rest | ||||
|         "readarray" -> maybeToList $ getMapfileArray base rest | ||||
|  | ||||
|         _ -> [] | ||||
|   where | ||||
|     flags = map snd $ getAllFlags base | ||||
|     stripEquals s = let rest = dropWhile (/= '=') s in | ||||
|         if rest == "" then "" else tail rest | ||||
|     stripEqualsFrom (T_NormalWord id1 (T_Literal id2 s:rs)) = | ||||
|         T_NormalWord id1 (T_Literal id2 (stripEquals s):rs) | ||||
|     stripEqualsFrom (T_NormalWord id1 [T_DoubleQuoted id2 [T_Literal id3 s]]) = | ||||
|         T_NormalWord id1 [T_DoubleQuoted id2 [T_Literal id3 (stripEquals s)]] | ||||
|     stripEqualsFrom t = t | ||||
|  | ||||
|     declaredVars = concatMap (getModifierParam defaultType) rest | ||||
|       where | ||||
|         defaultType = if any (`elem` flags) ["a", "A"] then DataArray else DataString | ||||
|  | ||||
|     getLiteral t = do | ||||
|         s <- getLiteralString t | ||||
|         when ("-" `isPrefixOf` s) $ fail "argument" | ||||
|         return (base, t, s, DataString SourceExternal) | ||||
|  | ||||
|     getModifierParamString = getModifierParam DataString | ||||
|  | ||||
|     getModifierParam def t@(T_Assignment _ _ name _ value) = | ||||
|         [(base, t, name, dataTypeFrom def value)] | ||||
|     getModifierParam def t@(T_NormalWord {}) = maybeToList $ do | ||||
|         name <- getLiteralString t | ||||
|         guard $ isVariableName name | ||||
|         return (base, t, name, def SourceDeclaration) | ||||
|     getModifierParam _ _ = [] | ||||
|  | ||||
|     letParamToLiteral token = | ||||
|           if var == "" | ||||
|             then [] | ||||
|             else [(base, token, var, DataString $ SourceFrom [stripEqualsFrom token])] | ||||
|         where var = takeWhile isVariableChar $ dropWhile (`elem` "+-") $ concat $ oversimplify token | ||||
|  | ||||
|     getSetParams (t:_:rest) | getLiteralString t == Just "-o" = getSetParams rest | ||||
|     getSetParams (t:rest) = | ||||
|         let s = getLiteralString t in | ||||
|             case s of | ||||
|                 Just "--" -> return rest | ||||
|                 Just ('-':_) -> getSetParams rest | ||||
|                 _ -> return (t:fromMaybe [] (getSetParams rest)) | ||||
|     getSetParams [] = Nothing | ||||
|  | ||||
|     getPrintfVariable list = f $ map (\x -> (x, getLiteralString x)) list | ||||
|       where | ||||
|         f ((_, Just "-v") : (t, Just var) : _) = return (base, t, var, DataString $ SourceFrom list) | ||||
|         f (_:rest) = f rest | ||||
|         f [] = fail "not found" | ||||
|  | ||||
|     -- mapfile has some curious syntax allowing flags plus 0..n variable names | ||||
|     -- where only the first non-option one is used if any. Here we cheat and | ||||
|     -- just get the last one, if it's a variable name. | ||||
|     getMapfileArray base arguments = do | ||||
|         lastArg <- listToMaybe (reverse arguments) | ||||
|         name <- getLiteralString lastArg | ||||
|         guard $ isVariableName name | ||||
|         return (base, lastArg, name, DataArray SourceExternal) | ||||
|  | ||||
| getModifiedVariableCommand _ = [] | ||||
|  | ||||
| getIndexReferences s = fromMaybe [] $ do | ||||
|     match <- matchRegex re s | ||||
|     index <- match !!! 0 | ||||
|     return $ matchAllStrings variableNameRegex index | ||||
|   where | ||||
|     re = mkRegex "(\\[.*\\])" | ||||
|  | ||||
| getReferencedVariables t = | ||||
|     case t of | ||||
|         T_DollarBraced id l -> let str = bracedString t in | ||||
|             (t, t, getBracedReference str) : | ||||
|                 map (\x -> (l, l, x)) (getIndexReferences str) | ||||
|         TA_Expansion id _ -> getIfReference t t | ||||
|         T_Assignment id mode str _ word -> | ||||
|             [(t, t, str) | mode == Append] ++ specialReferences str t word | ||||
|  | ||||
|         TC_Unary id _ "-v" token -> getIfReference t token | ||||
|         TC_Unary id _ "-R" token -> getIfReference t token | ||||
|         TC_Binary id DoubleBracket op lhs rhs -> | ||||
|             if isDereferencing op | ||||
|             then concatMap (getIfReference t) [lhs, rhs] | ||||
|             else [] | ||||
|  | ||||
|         t@(T_FdRedirect _ ('{':var) op) -> -- {foo}>&- references and closes foo | ||||
|             [(t, t, takeWhile (/= '}') var) | isClosingFileOp op] | ||||
|         x -> getReferencedVariableCommand x | ||||
|   where | ||||
|     -- Try to reduce false positives for unused vars only referenced from evaluated vars | ||||
|     specialReferences name base word = | ||||
|         if name `elem` [ | ||||
|             "PS1", "PS2", "PS3", "PS4", | ||||
|             "PROMPT_COMMAND" | ||||
|           ] | ||||
|         then | ||||
|             map (\x -> (base, base, x)) $ | ||||
|                 getVariablesFromLiteralToken word | ||||
|         else [] | ||||
|  | ||||
|     literalizer (TA_Index {}) = return ""  -- x[0] becomes a reference of x | ||||
|     literalizer _ = Nothing | ||||
|  | ||||
|     getIfReference context token = maybeToList $ do | ||||
|             str <- getLiteralStringExt literalizer token | ||||
|             guard . not $ null str | ||||
|             when (isDigit $ head str) $ fail "is a number" | ||||
|             return (context, token, getBracedReference str) | ||||
|  | ||||
|     isDereferencing = (`elem` ["-eq", "-ne", "-lt", "-le", "-gt", "-ge"]) | ||||
|  | ||||
| dataTypeFrom defaultType v = (case v of T_Array {} -> DataArray; _ -> defaultType) $ SourceFrom [v] | ||||
|  | ||||
|  | ||||
| --- Command specific checks | ||||
|  | ||||
| isCommand token str = isCommandMatch token (\cmd -> cmd  == str || ('/' : str) `isSuffixOf` cmd) | ||||
| isUnqualifiedCommand token str = isCommandMatch token (== str) | ||||
|  | ||||
| isCommandMatch token matcher = fromMaybe False $ do | ||||
|     cmd <- getCommandName token | ||||
|     return $ matcher cmd | ||||
|  | ||||
| isConfusedGlobRegex ('*':_) = True | ||||
| isConfusedGlobRegex [x,'*'] | x /= '\\' = True | ||||
| isConfusedGlobRegex _ = False | ||||
|  | ||||
| isVariableStartChar x = x == '_' || isAsciiLower x || isAsciiUpper x | ||||
| isVariableChar x = isVariableStartChar x || isDigit x | ||||
| variableNameRegex = mkRegex "[_a-zA-Z][_a-zA-Z0-9]*" | ||||
|  | ||||
| prop_isVariableName1 = isVariableName "_fo123" | ||||
| prop_isVariableName2 = not $ isVariableName "4" | ||||
| prop_isVariableName3 = not $ isVariableName "test: " | ||||
| isVariableName (x:r) = isVariableStartChar x && all isVariableChar r | ||||
| isVariableName _ = False | ||||
|  | ||||
| getVariablesFromLiteralToken token = | ||||
|     getVariablesFromLiteral (fromJust $ getLiteralStringExt (const $ return " ") token) | ||||
|  | ||||
| -- Try to get referenced variables from a literal string like "$foo" | ||||
| -- Ignores tons of cases like arithmetic evaluation and array indices. | ||||
| prop_getVariablesFromLiteral1 = | ||||
|     getVariablesFromLiteral "$foo${bar//a/b}$BAZ" == ["foo", "bar", "BAZ"] | ||||
| getVariablesFromLiteral string = | ||||
|     map (!! 0) $ matchAllSubgroups variableRegex string | ||||
|   where | ||||
|     variableRegex = mkRegex "\\$\\{?([A-Za-z0-9_]+)" | ||||
|  | ||||
| prop_getBracedReference1 = getBracedReference "foo" == "foo" | ||||
| prop_getBracedReference2 = getBracedReference "#foo" == "foo" | ||||
| prop_getBracedReference3 = getBracedReference "#" == "#" | ||||
| prop_getBracedReference4 = getBracedReference "##" == "#" | ||||
| prop_getBracedReference5 = getBracedReference "#!" == "!" | ||||
| prop_getBracedReference6 = getBracedReference "!#" == "#" | ||||
| prop_getBracedReference7 = getBracedReference "!foo#?" == "foo" | ||||
| prop_getBracedReference8 = getBracedReference "foo-bar" == "foo" | ||||
| prop_getBracedReference9 = getBracedReference "foo:-bar" == "foo" | ||||
| prop_getBracedReference10= getBracedReference "foo: -1" == "foo" | ||||
| prop_getBracedReference11= getBracedReference "!os*" == "" | ||||
| prop_getBracedReference12= getBracedReference "!os?bar**" == "" | ||||
| getBracedReference s = fromMaybe s $ | ||||
|     nameExpansion s `mplus` takeName noPrefix `mplus` getSpecial noPrefix `mplus` getSpecial s | ||||
|   where | ||||
|     noPrefix = dropPrefix s | ||||
|     dropPrefix (c:rest) = if c `elem` "!#" then rest else c:rest | ||||
|     dropPrefix "" = "" | ||||
|     takeName s = do | ||||
|         let name = takeWhile isVariableChar s | ||||
|         guard . not $ null name | ||||
|         return name | ||||
|     getSpecial (c:_) = | ||||
|         if c `elem` "*@#?-$!" then return [c] else fail "not special" | ||||
|     getSpecial _ = fail "empty" | ||||
|  | ||||
|     nameExpansion ('!':rest) = do -- e.g. ${!foo*bar*} | ||||
|         let suffix = dropWhile isVariableChar rest | ||||
|         guard $ suffix /= rest -- e.g. ${!@} | ||||
|         first <- suffix !!! 0 | ||||
|         guard $ first `elem` "*?" | ||||
|         return "" | ||||
|     nameExpansion _ = Nothing | ||||
|  | ||||
|  | ||||
| -- Useful generic functions | ||||
| potentially :: Monad m => Maybe (m ()) -> m () | ||||
| potentially = fromMaybe (return ()) | ||||
|  | ||||
| headOrDefault _ (a:_) = a | ||||
| headOrDefault def _ = def | ||||
|  | ||||
| (!!!) list i = | ||||
|     case drop i list of | ||||
|         [] -> Nothing | ||||
|         (r:_) -> Just r | ||||
|  | ||||
|  | ||||
|  | ||||
| filterByAnnotation token = | ||||
|     filter (not . shouldIgnore) | ||||
|   where | ||||
|     idFor (TokenComment id _) = id | ||||
|     shouldIgnore note = | ||||
|         any (shouldIgnoreFor (getCode note)) $ | ||||
|             getPath parents (T_Bang $ idFor note) | ||||
|     shouldIgnoreFor num (T_Annotation _ anns _) = | ||||
|         any hasNum anns | ||||
|       where | ||||
|         hasNum (DisableComment ts) = num == ts | ||||
|         hasNum _ = False | ||||
|     shouldIgnoreFor _ (T_Include {}) = True -- Ignore included files | ||||
|     shouldIgnoreFor _ _ = False | ||||
|     parents = getParentTree token | ||||
|     getCode (TokenComment _ (Comment _ c _)) = c | ||||
|  | ||||
|  | ||||
| return []  | ||||
| runTests =  $( [| $(forAllProperties) (quickCheckWithResult (stdArgs { maxSuccess = 1 }) ) |]) -- prop_testing | ||||
							
								
								
									
										560
									
								
								ShellCheck/Checks/Commands.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										560
									
								
								ShellCheck/Checks/Commands.hs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,560 @@ | ||||
| {- | ||||
|     Copyright 2012-2015 Vidar Holen | ||||
|  | ||||
|     This file is part of ShellCheck. | ||||
|     http://www.vidarholen.net/contents/shellcheck | ||||
|  | ||||
|     ShellCheck is free software: you can redistribute it and/or modify | ||||
|     it under the terms of the GNU General Public License as published by | ||||
|     the Free Software Foundation, either version 3 of the License, or | ||||
|     (at your option) any later version. | ||||
|  | ||||
|     ShellCheck is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public License | ||||
|     along with this program.  If not, see <http://www.gnu.org/licenses/>. | ||||
| -} | ||||
| {-# LANGUAGE TemplateHaskell #-} | ||||
| {-# LANGUAGE FlexibleContexts #-} | ||||
|  | ||||
| -- This module contains checks that examine specific commands by name. | ||||
| module ShellCheck.Checks.Commands (runChecks | ||||
|     , ShellCheck.Checks.Commands.runTests | ||||
| ) where | ||||
|  | ||||
| import ShellCheck.AST | ||||
| import ShellCheck.ASTLib | ||||
| import ShellCheck.AnalyzerLib | ||||
| import ShellCheck.Data | ||||
| import ShellCheck.Interface | ||||
| import ShellCheck.Parser | ||||
| import ShellCheck.Regex | ||||
|  | ||||
| import Control.Monad | ||||
| import Control.Monad.Reader | ||||
| import Control.Monad.Writer | ||||
| import Data.Char | ||||
| import Data.List | ||||
| import Data.Maybe | ||||
| import qualified Data.Map as Map | ||||
| import Test.QuickCheck.All (forAllProperties) | ||||
| import Test.QuickCheck.Test (quickCheckWithResult, stdArgs, maxSuccess) | ||||
|  | ||||
| data CommandName = Exactly String | Basename String | ||||
|     deriving (Eq, Ord) | ||||
|  | ||||
| data CommandCheck = | ||||
|     CommandCheck CommandName (Token -> Analysis) | ||||
|  | ||||
| nullCheck :: Token -> Analysis | ||||
| nullCheck _ = return () | ||||
|  | ||||
|  | ||||
| verify :: CommandCheck -> String -> Bool | ||||
| verify f s = producesComments f s == Just True | ||||
| verifyNot f s = producesComments f s == Just False | ||||
|  | ||||
| producesComments :: CommandCheck -> String -> Maybe Bool | ||||
| producesComments f s = do | ||||
|         root <- pScript s | ||||
|         return . not . null $ runList (defaultSpec root) [f] | ||||
|  | ||||
| composeChecks f g t = do | ||||
|     f t | ||||
|     g t | ||||
|  | ||||
| arguments (T_SimpleCommand _ _ (cmd:args)) = args | ||||
|  | ||||
| commandChecks :: [CommandCheck] | ||||
| commandChecks = [ | ||||
|     checkTr | ||||
|     ,checkFindNameGlob | ||||
|     ,checkNeedlessExpr | ||||
|     ,checkGrepRe | ||||
|     ,checkTrapQuotes | ||||
|     ,checkReturn | ||||
|     ,checkFindExecWithSingleArgument | ||||
|     ,checkUnusedEchoEscapes | ||||
|     ,checkInjectableFindSh | ||||
|     ,checkFindActionPrecedence | ||||
|     ,checkMkdirDashPM | ||||
|     ,checkNonportableSignals | ||||
|     ,checkInteractiveSu | ||||
|     ,checkSshCommandString | ||||
|     ,checkPrintfVar | ||||
|     ,checkUuoeCmd | ||||
|     ,checkSetAssignment | ||||
|     ,checkExportedExpansions | ||||
|     ,checkAliasesUsesArgs | ||||
|     ,checkAliasesExpandEarly | ||||
|     ] | ||||
|  | ||||
| buildCommandMap :: [CommandCheck] -> Map.Map CommandName (Token -> Analysis) | ||||
| buildCommandMap = foldl' addCheck Map.empty | ||||
|   where | ||||
|     addCheck map (CommandCheck name function) = | ||||
|         Map.insertWith' composeChecks name function map | ||||
|  | ||||
|  | ||||
| checkCommand :: Map.Map CommandName (Token -> Analysis) -> Token -> Analysis | ||||
| checkCommand map t@(T_SimpleCommand id _ (cmd:rest)) = fromMaybe (return ()) $ do | ||||
|     name <- getLiteralString cmd | ||||
|     return $ | ||||
|         if '/' `elem` name | ||||
|         then | ||||
|             Map.findWithDefault nullCheck (Basename $ basename name) map t | ||||
|         else do | ||||
|             Map.findWithDefault nullCheck (Exactly name) map t | ||||
|             Map.findWithDefault nullCheck (Basename name) map t | ||||
|  | ||||
|   where | ||||
|     basename = reverse . takeWhile (/= '/') . reverse | ||||
| checkCommand _ _ = return () | ||||
|  | ||||
| runList spec list = notes | ||||
|     where | ||||
|         root = asScript spec | ||||
|         params = makeParameters spec | ||||
|         notes = execWriter $ runReaderT (doAnalysis (checkCommand map) root) params | ||||
|         map = buildCommandMap list | ||||
|  | ||||
| runChecks spec = runList spec commandChecks | ||||
|  | ||||
|  | ||||
| prop_checkTr1 = verify checkTr "tr [a-f] [A-F]" | ||||
| prop_checkTr2 = verify checkTr "tr 'a-z' 'A-Z'" | ||||
| prop_checkTr2a= verify checkTr "tr '[a-z]' '[A-Z]'" | ||||
| prop_checkTr3 = verifyNot checkTr "tr -d '[:lower:]'" | ||||
| prop_checkTr3a= verifyNot checkTr "tr -d '[:upper:]'" | ||||
| prop_checkTr3b= verifyNot checkTr "tr -d '|/_[:upper:]'" | ||||
| prop_checkTr4 = verifyNot checkTr "ls [a-z]" | ||||
| prop_checkTr5 = verify checkTr "tr foo bar" | ||||
| prop_checkTr6 = verify checkTr "tr 'hello' 'world'" | ||||
| prop_checkTr8 = verifyNot checkTr "tr aeiou _____" | ||||
| prop_checkTr9 = verifyNot checkTr "a-z n-za-m" | ||||
| prop_checkTr10= verifyNot checkTr "tr --squeeze-repeats rl lr" | ||||
| prop_checkTr11= verifyNot checkTr "tr abc '[d*]'" | ||||
| checkTr = CommandCheck (Basename "tr") (mapM_ f . arguments) | ||||
|   where | ||||
|     f w | isGlob w = -- The user will go [ab] -> '[ab]' -> 'ab'. Fixme? | ||||
|         warn (getId w) 2060 "Quote parameters to tr to prevent glob expansion." | ||||
|     f word = | ||||
|       case getLiteralString word of | ||||
|         Just "a-z" -> info (getId word) 2018 "Use '[:lower:]' to support accents and foreign alphabets." | ||||
|         Just "A-Z" -> info (getId word) 2019 "Use '[:upper:]' to support accents and foreign alphabets." | ||||
|         Just s -> do  -- Eliminate false positives by only looking for dupes in SET2? | ||||
|           when (not ("-" `isPrefixOf` s || "[:" `isInfixOf` s) && duplicated s) $ | ||||
|             info (getId word) 2020 "tr replaces sets of chars, not words (mentioned due to duplicates)." | ||||
|           unless ("[:" `isPrefixOf` s) $ | ||||
|             when ("[" `isPrefixOf` s && "]" `isSuffixOf` s && (length s > 2) && ('*' `notElem` s)) $ | ||||
|               info (getId word) 2021 "Don't use [] around ranges in tr, it replaces literal square brackets." | ||||
|         Nothing -> return () | ||||
|  | ||||
|     duplicated s = | ||||
|         let relevant = filter isAlpha s | ||||
|         in relevant /= nub relevant | ||||
|  | ||||
| prop_checkFindNameGlob1 = verify checkFindNameGlob "find / -name *.php" | ||||
| prop_checkFindNameGlob2 = verify checkFindNameGlob "find / -type f -ipath *(foo)" | ||||
| prop_checkFindNameGlob3 = verifyNot checkFindNameGlob "find * -name '*.php'" | ||||
| checkFindNameGlob = CommandCheck (Basename "find") (f . arguments)  where | ||||
|     acceptsGlob (Just s) = s `elem` [ "-ilname", "-iname", "-ipath", "-iregex", "-iwholename", "-lname", "-name", "-path", "-regex", "-wholename" ] | ||||
|     acceptsGlob _ = False | ||||
|     f [] = return () | ||||
|     f [x] = return () | ||||
|     f (a:b:r) = do | ||||
|         when (acceptsGlob (getLiteralString a) && isGlob b) $ do | ||||
|             let (Just s) = getLiteralString a | ||||
|             warn (getId b) 2061 $ "Quote the parameter to " ++ s ++ " so the shell won't interpret it." | ||||
|         f (b:r) | ||||
|  | ||||
|  | ||||
| prop_checkNeedlessExpr = verify checkNeedlessExpr "foo=$(expr 3 + 2)" | ||||
| prop_checkNeedlessExpr2 = verify checkNeedlessExpr "foo=`echo \\`expr 3 + 2\\``" | ||||
| prop_checkNeedlessExpr3 = verifyNot checkNeedlessExpr "foo=$(expr foo : regex)" | ||||
| prop_checkNeedlessExpr4 = verifyNot checkNeedlessExpr "foo=$(expr foo \\< regex)" | ||||
| checkNeedlessExpr = CommandCheck (Basename "expr") f where | ||||
|     f t = | ||||
|         when (all (`notElem` exceptions) (words $ arguments t)) $ | ||||
|             style (getId t) 2003 | ||||
|                 "expr is antiquated. Consider rewriting this using $((..)), ${} or [[ ]]." | ||||
|     -- These operators are hard to replicate in POSIX | ||||
|     exceptions = [ ":", "<", ">", "<=", ">=" ] | ||||
|     words = mapMaybe getLiteralString | ||||
|  | ||||
|  | ||||
| prop_checkGrepRe1 = verify checkGrepRe "cat foo | grep *.mp3" | ||||
| prop_checkGrepRe2 = verify checkGrepRe "grep -Ev cow*test *.mp3" | ||||
| prop_checkGrepRe3 = verify checkGrepRe "grep --regex=*.mp3 file" | ||||
| prop_checkGrepRe4 = verifyNot checkGrepRe "grep foo *.mp3" | ||||
| prop_checkGrepRe5 = verifyNot checkGrepRe "grep-v  --regex=moo *" | ||||
| prop_checkGrepRe6 = verifyNot checkGrepRe "grep foo \\*.mp3" | ||||
| prop_checkGrepRe7 = verify checkGrepRe "grep *foo* file" | ||||
| prop_checkGrepRe8 = verify checkGrepRe "ls | grep foo*.jpg" | ||||
| prop_checkGrepRe9 = verifyNot checkGrepRe "grep '[0-9]*' file" | ||||
| prop_checkGrepRe10= verifyNot checkGrepRe "grep '^aa*' file" | ||||
| prop_checkGrepRe11= verifyNot checkGrepRe "grep --include=*.png foo" | ||||
|  | ||||
| checkGrepRe = CommandCheck (Basename "grep") (f . arguments) where | ||||
|     -- --regex=*(extglob) doesn't work. Fixme? | ||||
|     skippable (Just s) = not ("--regex=" `isPrefixOf` s) && "-" `isPrefixOf` s | ||||
|     skippable _ = False | ||||
|     f [] = return () | ||||
|     f (x:r) | skippable (getLiteralStringExt (const $ return "_") x) = f r | ||||
|     f (re:_) = do | ||||
|         when (isGlob re) $ | ||||
|             warn (getId re) 2062 "Quote the grep pattern so the shell won't interpret it." | ||||
|         let string = concat $ oversimplify re | ||||
|         if isConfusedGlobRegex string then | ||||
|             warn (getId re) 2063 "Grep uses regex, but this looks like a glob." | ||||
|           else potentially $ do | ||||
|             char <- getSuspiciousRegexWildcard string | ||||
|             return $ info (getId re) 2022 $ | ||||
|                 "Note that unlike globs, " ++ [char] ++ "* here matches '" ++ [char, char, char] ++ "' but not '" ++ wordStartingWith char ++ "'." | ||||
|  | ||||
|     wordStartingWith c = | ||||
|         head . filter ([c] `isPrefixOf`) $ candidates | ||||
|       where | ||||
|         candidates = | ||||
|             sampleWords ++ map (\(x:r) -> toUpper x : r) sampleWords ++ [c:"test"] | ||||
|  | ||||
|     getSuspiciousRegexWildcard str = | ||||
|         if not $ str `matches` contra | ||||
|         then do | ||||
|             match <- matchRegex suspicious str | ||||
|             str <- match !!! 0 | ||||
|             str !!! 0 | ||||
|         else | ||||
|             fail "looks good" | ||||
|       where | ||||
|         suspicious = mkRegex "([A-Za-z1-9])\\*" | ||||
|         contra = mkRegex "[^a-zA-Z1-9]\\*|[][^$+\\\\]" | ||||
|  | ||||
|  | ||||
| prop_checkTrapQuotes1 = verify checkTrapQuotes "trap \"echo $num\" INT" | ||||
| prop_checkTrapQuotes1a= verify checkTrapQuotes "trap \"echo `ls`\" INT" | ||||
| prop_checkTrapQuotes2 = verifyNot checkTrapQuotes "trap 'echo $num' INT" | ||||
| prop_checkTrapQuotes3 = verify checkTrapQuotes "trap \"echo $((1+num))\" EXIT DEBUG" | ||||
| checkTrapQuotes = CommandCheck (Exactly "trap") (f . arguments) where | ||||
|     f (x:_) = checkTrap x | ||||
|     f _ = return () | ||||
|     checkTrap (T_NormalWord _ [T_DoubleQuoted _ rs]) = mapM_ checkExpansions rs | ||||
|     checkTrap _ = return () | ||||
|     warning id = warn id 2064 "Use single quotes, otherwise this expands now rather than when signalled." | ||||
|     checkExpansions (T_DollarExpansion id _) = warning id | ||||
|     checkExpansions (T_Backticked id _) = warning id | ||||
|     checkExpansions (T_DollarBraced id _) = warning id | ||||
|     checkExpansions (T_DollarArithmetic id _) = warning id | ||||
|     checkExpansions _ = return () | ||||
|  | ||||
|  | ||||
| prop_checkReturn1 = verifyNot checkReturn "return" | ||||
| prop_checkReturn2 = verifyNot checkReturn "return 1" | ||||
| prop_checkReturn3 = verifyNot checkReturn "return $var" | ||||
| prop_checkReturn4 = verifyNot checkReturn "return $((a|b))" | ||||
| prop_checkReturn5 = verify checkReturn "return -1" | ||||
| prop_checkReturn6 = verify checkReturn "return 1000" | ||||
| prop_checkReturn7 = verify checkReturn "return 'hello world'" | ||||
| checkReturn = CommandCheck (Exactly "return") (f . arguments) | ||||
|   where | ||||
|     f (first:second:_) = | ||||
|         err (getId second) 2151 | ||||
|             "Only one integer 0-255 can be returned. Use stdout for other data." | ||||
|     f [value] = | ||||
|         when (isInvalid $ literal value) $ | ||||
|             err (getId value) 2152 | ||||
|                 "Can only return 0-255. Other data should be written to stdout." | ||||
|     f _ = return () | ||||
|  | ||||
|     isInvalid s = s == "" || any (not . isDigit) s || length s > 5 | ||||
|         || let value = (read s :: Integer) in value > 255 | ||||
|  | ||||
|     literal token = fromJust $ getLiteralStringExt lit token | ||||
|     lit (T_DollarBraced {}) = return "0" | ||||
|     lit (T_DollarArithmetic {}) = return "0" | ||||
|     lit (T_DollarExpansion {}) = return "0" | ||||
|     lit (T_Backticked {}) = return "0" | ||||
|     lit _ = return "WTF" | ||||
|  | ||||
|  | ||||
| prop_checkFindExecWithSingleArgument1 = verify checkFindExecWithSingleArgument "find . -exec 'cat {} | wc -l' \\;" | ||||
| prop_checkFindExecWithSingleArgument2 = verify checkFindExecWithSingleArgument "find . -execdir 'cat {} | wc -l' +" | ||||
| prop_checkFindExecWithSingleArgument3 = verifyNot checkFindExecWithSingleArgument "find . -exec wc -l {} \\;" | ||||
| checkFindExecWithSingleArgument = CommandCheck (Basename "find") (f . arguments) | ||||
|   where | ||||
|     f = void . sequence . mapMaybe check . tails | ||||
|     check (exec:arg:term:_) = do | ||||
|         execS <- getLiteralString exec | ||||
|         termS <- getLiteralString term | ||||
|         cmdS <- getLiteralStringExt (const $ return " ") arg | ||||
|  | ||||
|         guard $ execS `elem` ["-exec", "-execdir"] && termS `elem` [";", "+"] | ||||
|         guard $ cmdS `matches` commandRegex | ||||
|         return $ warn (getId exec) 2150 "-exec does not invoke a shell. Rewrite or use -exec sh -c .. ." | ||||
|     check _ = Nothing | ||||
|     commandRegex = mkRegex "[ |;]" | ||||
|  | ||||
|  | ||||
| prop_checkUnusedEchoEscapes1 = verify checkUnusedEchoEscapes "echo 'foo\\nbar\\n'" | ||||
| prop_checkUnusedEchoEscapes2 = verifyNot checkUnusedEchoEscapes "echo -e 'foi\\nbar'" | ||||
| prop_checkUnusedEchoEscapes3 = verify checkUnusedEchoEscapes "echo \"n:\\t42\"" | ||||
| prop_checkUnusedEchoEscapes4 = verifyNot checkUnusedEchoEscapes "echo lol" | ||||
| prop_checkUnusedEchoEscapes5 = verifyNot checkUnusedEchoEscapes "echo -n -e '\n'" | ||||
| checkUnusedEchoEscapes = CommandCheck (Basename "echo") (f . arguments) | ||||
|   where | ||||
|     isDashE = mkRegex "^-.*e" | ||||
|     hasEscapes = mkRegex "\\\\[rnt]" | ||||
|     f args | concat (concatMap oversimplify allButLast) `matches` isDashE = | ||||
|         return () | ||||
|       where allButLast = reverse . drop 1 . reverse $ args | ||||
|     f args = mapM_ checkEscapes args | ||||
|  | ||||
|     checkEscapes (T_NormalWord _ args) = | ||||
|         mapM_ checkEscapes args | ||||
|     checkEscapes (T_DoubleQuoted id args) = | ||||
|         mapM_ checkEscapes args | ||||
|     checkEscapes (T_Literal id str) = examine id str | ||||
|     checkEscapes (T_SingleQuoted id str) = examine id str | ||||
|     checkEscapes _ = return () | ||||
|  | ||||
|     examine id str = | ||||
|         when (str `matches` hasEscapes) $ | ||||
|             info id 2028 "echo won't expand escape sequences. Consider printf." | ||||
|  | ||||
|  | ||||
| prop_checkInjectableFindSh1 = verify checkInjectableFindSh "find . -exec sh -c 'echo {}' \\;" | ||||
| prop_checkInjectableFindSh2 = verify checkInjectableFindSh "find . -execdir bash -c 'rm \"{}\"' ';'" | ||||
| prop_checkInjectableFindSh3 = verifyNot checkInjectableFindSh "find . -exec sh -c 'rm \"$@\"' _ {} \\;" | ||||
| checkInjectableFindSh = CommandCheck (Basename "find") (check . arguments) | ||||
|   where | ||||
|     check args = do | ||||
|         let idStrings = map (\x -> (getId x, onlyLiteralString x)) args | ||||
|         match pattern idStrings | ||||
|  | ||||
|     match _ [] = return () | ||||
|     match [] (next:_) = action next | ||||
|     match (p:tests) ((id, arg):args) = do | ||||
|         when (p arg) $ match tests args | ||||
|         match (p:tests) args | ||||
|  | ||||
|     pattern = [ | ||||
|         (`elem` ["-exec", "-execdir"]), | ||||
|         (`elem` ["sh", "bash", "ksh"]), | ||||
|         (== "-c") | ||||
|         ] | ||||
|     action (id, arg) = | ||||
|         when ("{}" `isInfixOf` arg) $ | ||||
|             warn id 2156 "Injecting filenames is fragile and insecure. Use parameters." | ||||
|  | ||||
|  | ||||
| prop_checkFindActionPrecedence1 = verify checkFindActionPrecedence "find . -name '*.wav' -o -name '*.au' -exec rm {} +" | ||||
| prop_checkFindActionPrecedence2 = verifyNot checkFindActionPrecedence "find . -name '*.wav' -o \\( -name '*.au' -exec rm {} + \\)" | ||||
| prop_checkFindActionPrecedence3 = verifyNot checkFindActionPrecedence "find . -name '*.wav' -o -name '*.au'" | ||||
| checkFindActionPrecedence = CommandCheck (Basename "find") (f . arguments) | ||||
|   where | ||||
|     pattern = [isMatch, const True, isParam ["-o", "-or"], isMatch, const True, isAction] | ||||
|     f list | length list < length pattern = return () | ||||
|     f list@(_:rest) = | ||||
|         if and (zipWith ($) pattern list) | ||||
|         then warnFor (list !! (length pattern - 1)) | ||||
|         else f rest | ||||
|     isMatch = isParam [ "-name", "-regex", "-iname", "-iregex", "-wholename", "-iwholename" ] | ||||
|     isAction = isParam [ "-exec", "-execdir", "-delete", "-print", "-print0" ] | ||||
|     isParam strs t = fromMaybe False $ do | ||||
|         param <- getLiteralString t | ||||
|         return $ param `elem` strs | ||||
|     warnFor t = warn (getId t) 2146 "This action ignores everything before the -o. Use \\( \\) to group." | ||||
|  | ||||
|  | ||||
| prop_checkMkdirDashPM0 = verify checkMkdirDashPM "mkdir -p -m 0755 a/b" | ||||
| prop_checkMkdirDashPM1 = verify checkMkdirDashPM "mkdir -pm 0755 $dir" | ||||
| prop_checkMkdirDashPM2 = verify checkMkdirDashPM "mkdir -vpm 0755 a/b" | ||||
| prop_checkMkdirDashPM3 = verify checkMkdirDashPM "mkdir -pm 0755 -v a/b" | ||||
| prop_checkMkdirDashPM4 = verify checkMkdirDashPM "mkdir --parents --mode=0755 a/b" | ||||
| prop_checkMkdirDashPM5 = verify checkMkdirDashPM "mkdir --parents --mode 0755 a/b" | ||||
| prop_checkMkdirDashPM6 = verify checkMkdirDashPM "mkdir -p --mode=0755 a/b" | ||||
| prop_checkMkdirDashPM7 = verify checkMkdirDashPM "mkdir --parents -m 0755 a/b" | ||||
| prop_checkMkdirDashPM8 = verifyNot checkMkdirDashPM "mkdir -p a/b" | ||||
| prop_checkMkdirDashPM9 = verifyNot checkMkdirDashPM "mkdir -m 0755 a/b" | ||||
| prop_checkMkdirDashPM10 = verifyNot checkMkdirDashPM "mkdir a/b" | ||||
| prop_checkMkdirDashPM11 = verifyNot checkMkdirDashPM "mkdir --parents a/b" | ||||
| prop_checkMkdirDashPM12 = verifyNot checkMkdirDashPM "mkdir --mode=0755 a/b" | ||||
| prop_checkMkdirDashPM13 = verifyNot checkMkdirDashPM "mkdir_func -pm 0755 a/b" | ||||
| prop_checkMkdirDashPM14 = verifyNot checkMkdirDashPM "mkdir -p -m 0755 singlelevel" | ||||
| checkMkdirDashPM = CommandCheck (Basename "mkdir") check | ||||
|   where | ||||
|     check t = potentially $ do | ||||
|         let flags = getAllFlags t | ||||
|         dashP <- find ((\f -> f == "p" || f == "parents") . snd) flags | ||||
|         dashM <- find ((\f -> f == "m" || f == "mode") . snd) flags | ||||
|         guard $ any couldHaveSubdirs (drop 1 $ arguments t) -- mkdir -pm 0700 dir  is fine, but dir/subdir is not. | ||||
|         return $ warn (getId $ fst dashM) 2174 "When used with -p, -m only applies to the deepest directory." | ||||
|     couldHaveSubdirs t = fromMaybe True $ do | ||||
|         name <- getLiteralString t | ||||
|         return $ '/' `elem` name | ||||
|  | ||||
|  | ||||
| prop_checkNonportableSignals1 = verify checkNonportableSignals "trap f 8" | ||||
| prop_checkNonportableSignals2 = verifyNot checkNonportableSignals "trap f 0" | ||||
| prop_checkNonportableSignals3 = verifyNot checkNonportableSignals "trap f 14" | ||||
| prop_checkNonportableSignals4 = verify checkNonportableSignals "trap f SIGKILL" | ||||
| prop_checkNonportableSignals5 = verify checkNonportableSignals "trap f 9" | ||||
| prop_checkNonportableSignals6 = verify checkNonportableSignals "trap f stop" | ||||
| checkNonportableSignals = CommandCheck (Exactly "trap") (f . arguments) | ||||
|   where | ||||
|     f = mapM_ check | ||||
|     check param = potentially $ do | ||||
|         str <- getLiteralString param | ||||
|         let id = getId param | ||||
|         return $ sequence_ $ mapMaybe (\f -> f id str) [ | ||||
|             checkNumeric, | ||||
|             checkUntrappable | ||||
|             ] | ||||
|  | ||||
|     checkNumeric id str = do | ||||
|         guard $ not (null str) | ||||
|         guard $ all isDigit str | ||||
|         guard $ str /= "0" -- POSIX exit trap | ||||
|         guard $ str `notElem` ["1", "2", "3", "6", "9", "14", "15" ] -- XSI | ||||
|         return $ warn id 2172 | ||||
|             "Trapping signals by number is not well defined. Prefer signal names." | ||||
|  | ||||
|     checkUntrappable id str = do | ||||
|         guard $ map toLower str `elem` ["kill", "9", "sigkill", "stop", "sigstop"] | ||||
|         return $ err id 2173 | ||||
|             "SIGKILL/SIGSTOP can not be trapped." | ||||
|  | ||||
|  | ||||
| prop_checkInteractiveSu1 = verify checkInteractiveSu "su; rm file; su $USER" | ||||
| prop_checkInteractiveSu2 = verify checkInteractiveSu "su foo; something; exit" | ||||
| prop_checkInteractiveSu3 = verifyNot checkInteractiveSu "echo rm | su foo" | ||||
| prop_checkInteractiveSu4 = verifyNot checkInteractiveSu "su root < script" | ||||
| checkInteractiveSu = CommandCheck (Basename "su") f | ||||
|   where | ||||
|     f cmd = when (length (arguments cmd) <= 1) $ do | ||||
|         path <- pathTo cmd | ||||
|         when (all undirected path) $ | ||||
|             info (getId cmd) 2117 | ||||
|                 "To run commands as another user, use su -c or sudo." | ||||
|  | ||||
|     undirected (T_Pipeline _ _ l) = length l <= 1 | ||||
|     -- This should really just be modifications to stdin, but meh | ||||
|     undirected (T_Redirecting _ list _) = null list | ||||
|     undirected _ = True | ||||
|  | ||||
|  | ||||
| -- This is hard to get right without properly parsing ssh args | ||||
| prop_checkSshCmdStr1 = verify checkSshCommandString "ssh host \"echo $PS1\"" | ||||
| prop_checkSshCmdStr2 = verifyNot checkSshCommandString "ssh host \"ls foo\"" | ||||
| prop_checkSshCmdStr3 = verifyNot checkSshCommandString "ssh \"$host\"" | ||||
| checkSshCommandString = CommandCheck (Basename "ssh") (f . arguments) | ||||
|   where | ||||
|     nonOptions = | ||||
|         filter (\x -> not $ "-" `isPrefixOf` concat (oversimplify x)) | ||||
|     f args = | ||||
|         case nonOptions args of | ||||
|             (hostport:r@(_:_)) -> checkArg $ last r | ||||
|             _ -> return () | ||||
|     checkArg (T_NormalWord _ [T_DoubleQuoted id parts]) = | ||||
|         case filter (not . isConstant) parts of | ||||
|             [] -> return () | ||||
|             (x:_) -> info (getId x) 2029 | ||||
|                 "Note that, unescaped, this expands on the client side." | ||||
|     checkArg _ = return () | ||||
|  | ||||
|  | ||||
| prop_checkPrintfVar1 = verify checkPrintfVar "printf \"Lol: $s\"" | ||||
| prop_checkPrintfVar2 = verifyNot checkPrintfVar "printf 'Lol: $s'" | ||||
| prop_checkPrintfVar3 = verify checkPrintfVar "printf -v cow $(cmd)" | ||||
| prop_checkPrintfVar4 = verifyNot checkPrintfVar "printf \"%${count}s\" var" | ||||
| checkPrintfVar = CommandCheck (Exactly "printf") (f . arguments) where | ||||
|     f (dashv:var:rest) | getLiteralString dashv == Just "-v" = f rest | ||||
|     f (format:params) = check format | ||||
|     f _ = return () | ||||
|     check format = | ||||
|         unless ('%' `elem` concat (oversimplify format) || isLiteral format) $ | ||||
|           warn (getId format) 2059 | ||||
|               "Don't use variables in the printf format string. Use printf \"..%s..\" \"$foo\"." | ||||
|  | ||||
|  | ||||
| prop_checkUuoeCmd1 = verify checkUuoeCmd "echo $(date)" | ||||
| prop_checkUuoeCmd2 = verify checkUuoeCmd "echo `date`" | ||||
| prop_checkUuoeCmd3 = verify checkUuoeCmd "echo \"$(date)\"" | ||||
| prop_checkUuoeCmd4 = verify checkUuoeCmd "echo \"`date`\"" | ||||
| prop_checkUuoeCmd5 = verifyNot checkUuoeCmd "echo \"The time is $(date)\"" | ||||
| prop_checkUuoeCmd6 = verifyNot checkUuoeCmd "echo \"$(<file)\"" | ||||
| checkUuoeCmd = CommandCheck (Exactly "echo") (f . arguments) where | ||||
|     msg id = style id 2005 "Useless echo? Instead of 'echo $(cmd)', just use 'cmd'." | ||||
|     f [token] = when (tokenIsJustCommandOutput token) $ msg (getId token) | ||||
|     f _ = return () | ||||
|  | ||||
|  | ||||
| prop_checkSetAssignment1 = verify checkSetAssignment "set foo 42" | ||||
| prop_checkSetAssignment2 = verify checkSetAssignment "set foo = 42" | ||||
| prop_checkSetAssignment3 = verify checkSetAssignment "set foo=42" | ||||
| prop_checkSetAssignment4 = verifyNot checkSetAssignment "set -- if=/dev/null" | ||||
| prop_checkSetAssignment5 = verifyNot checkSetAssignment "set 'a=5'" | ||||
| prop_checkSetAssignment6 = verifyNot checkSetAssignment "set" | ||||
| checkSetAssignment = CommandCheck (Exactly "set") (f . arguments) | ||||
|   where | ||||
|     f (var:value:rest) = | ||||
|         let str = literal var in | ||||
|             when (isVariableName str || isAssignment str) $ | ||||
|                 msg (getId var) | ||||
|     f (var:_) = | ||||
|         when (isAssignment $ literal var) $ | ||||
|             msg (getId var) | ||||
|     f _ = return () | ||||
|  | ||||
|     msg id = warn id 2121 "To assign a variable, use just 'var=value', no 'set ..'." | ||||
|  | ||||
|     isAssignment str = '=' `elem` str | ||||
|     literal (T_NormalWord _ l) = concatMap literal l | ||||
|     literal (T_Literal _ str) = str | ||||
|     literal _ = "*" | ||||
|  | ||||
|  | ||||
| prop_checkExportedExpansions1 = verify checkExportedExpansions "export $foo" | ||||
| prop_checkExportedExpansions2 = verify checkExportedExpansions "export \"$foo\"" | ||||
| prop_checkExportedExpansions3 = verifyNot checkExportedExpansions "export foo" | ||||
| checkExportedExpansions = CommandCheck (Exactly "export") (check . arguments) | ||||
|   where | ||||
|     check = mapM_ checkForVariables | ||||
|     checkForVariables f = | ||||
|         case getWordParts f of | ||||
|             [t@(T_DollarBraced {})] -> | ||||
|                 warn (getId t) 2163 "Exporting an expansion rather than a variable." | ||||
|             _ -> return () | ||||
|  | ||||
|  | ||||
| prop_checkAliasesUsesArgs1 = verify checkAliasesUsesArgs "alias a='cp $1 /a'" | ||||
| prop_checkAliasesUsesArgs2 = verifyNot checkAliasesUsesArgs "alias $1='foo'" | ||||
| prop_checkAliasesUsesArgs3 = verify checkAliasesUsesArgs "alias a=\"echo \\${@}\"" | ||||
| checkAliasesUsesArgs = CommandCheck (Exactly "alias") (f . arguments) | ||||
|   where | ||||
|     re = mkRegex "\\$\\{?[0-9*@]" | ||||
|     f = mapM_ checkArg | ||||
|     checkArg arg = | ||||
|         let string = fromJust $ getLiteralStringExt (const $ return "_") arg in | ||||
|             when ('=' `elem` string && string `matches` re) $ | ||||
|                 err (getId arg) 2142 | ||||
|                     "Aliases can't use positional parameters. Use a function." | ||||
|  | ||||
|  | ||||
| prop_checkAliasesExpandEarly1 = verify checkAliasesExpandEarly "alias foo=\"echo $PWD\"" | ||||
| prop_checkAliasesExpandEarly2 = verifyNot checkAliasesExpandEarly "alias -p" | ||||
| prop_checkAliasesExpandEarly3 = verifyNot checkAliasesExpandEarly "alias foo='echo {1..10}'" | ||||
| checkAliasesExpandEarly = CommandCheck (Exactly "alias") (f . arguments) | ||||
|   where | ||||
|     f = mapM_ checkArg | ||||
|     checkArg arg | '=' `elem` concat (oversimplify arg) = | ||||
|         forM_ (take 1 $ filter (not . isLiteral) $ getWordParts arg) $ | ||||
|             \x -> warn (getId x) 2139 "This expands when defined, not when used. Consider escaping." | ||||
|     checkArg _ = return () | ||||
|  | ||||
|  | ||||
| return [] | ||||
| runTests =  $( [| $(forAllProperties) (quickCheckWithResult (stdArgs { maxSuccess = 1 }) ) |]) | ||||
| @@ -4,7 +4,12 @@ | ||||
| # 'cabal test' remains the source of truth. | ||||
|  | ||||
| ( | ||||
|  var=$(echo 'liftM and $ sequence [ShellCheck.Analytics.runTests, ShellCheck.Parser.runTests, ShellCheck.Checker.runTests]' | cabal repl 2>&1 | tee /dev/stderr) | ||||
|  var=$(echo 'liftM and $ sequence [ | ||||
|   ShellCheck.Analytics.runTests | ||||
|   ,ShellCheck.Parser.runTests | ||||
|   ,ShellCheck.Checker.runTests | ||||
|   ,ShellCheck.Checks.Commands.runTests | ||||
|     ]' | tr -d '\n' | cabal repl 2>&1 | tee /dev/stderr) | ||||
| if [[ $var == *$'\nTrue'* ]] | ||||
| then | ||||
|   exit 0 | ||||
|   | ||||
| @@ -5,11 +5,13 @@ import System.Exit | ||||
| import qualified ShellCheck.Checker | ||||
| import qualified ShellCheck.Analytics | ||||
| import qualified ShellCheck.Parser | ||||
| import qualified ShellCheck.Checks.Commands | ||||
|  | ||||
| main = do | ||||
|     putStrLn "Running ShellCheck tests..." | ||||
|     results <- sequence [ | ||||
|         ShellCheck.Checker.runTests, | ||||
|         ShellCheck.Checks.Commands.runTests, | ||||
|         ShellCheck.Analytics.runTests, | ||||
|         ShellCheck.Parser.runTests | ||||
|       ] | ||||
|   | ||||
		Reference in New Issue
	
	Block a user