mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-13 19:33:55 +03:00
3ffc3c8166
GitOrigin-RevId: 7bd0ef402a559b0bf8f34cb1f33b517da4eac227
313 lines
13 KiB
YAML
313 lines
13 KiB
YAML
# HLint configuration file
|
|
# https://github.com/ndmitchell/hlint
|
|
|
|
################################################################################
|
|
# Global hlint configuration
|
|
|
|
- arguments: [-XQuasiQuotes]
|
|
|
|
|
|
################################################################################
|
|
# Restrictions on functions
|
|
|
|
- functions:
|
|
|
|
# Hspec restrictions
|
|
- name: "Test.Hspec.focus"
|
|
within: []
|
|
message: "Specs with focusing should only be used during development"
|
|
|
|
- name: "Test.Hspec.fit"
|
|
within: []
|
|
message: "Specs with focusing should only be used during development"
|
|
|
|
- name: "Test.Hspec.fspecify"
|
|
within: []
|
|
message: "Specs with focusing should only be used during development"
|
|
|
|
- name: "Test.Hspec.fdescribe"
|
|
within: []
|
|
message: "Specs with focusing should only be used during development"
|
|
|
|
- name: "Test.Hspec.fcontext"
|
|
within: []
|
|
message: "Specs with focusing should only be used during development"
|
|
|
|
- name: "Harness.TestEnvironment.testLogHarness"
|
|
within: []
|
|
message: "testLogHarness should only be ad-hoc during development. Please define proper log message data types for published code"
|
|
|
|
# Harness restrictions
|
|
- name: "Harness.Logging.LogHarness"
|
|
within: []
|
|
message: "LogHarness should only be ad-hoc during development. Please define proper log message data types for published code"
|
|
|
|
- name: "Harness.Logging.Messages.LogHarness"
|
|
within: []
|
|
message: "LogHarness should only be ad-hoc during development. Please define proper log message data types for published code"
|
|
|
|
- name: "Harness.TestEnvironment.testLogTrace"
|
|
within: ["Test.**"]
|
|
message: "testLogTrace should only be used outside the test-harness package"
|
|
|
|
- name: "Harness.Logging.LogTrace"
|
|
within: ["Test.**"]
|
|
message: "LogTrace should only be used outside the test-harness package"
|
|
|
|
- name: "Harness.Logging.Messages.LogTrace"
|
|
within: ["Test.**"]
|
|
message: "LogTrace should only be used outside the test-harness package"
|
|
|
|
- name: "Harness.Logging.logTrace"
|
|
within: ["Test.**"]
|
|
message: "logTrace should only be used outside the test-harness package"
|
|
|
|
- name: "Harness.Logging.Messages.logTrace"
|
|
within: ["Test.**"]
|
|
message: "logTrace should only be used outside the test-harness package"
|
|
|
|
# General restrictions
|
|
- name: "undefined"
|
|
within: []
|
|
message: "undefined isn't allowed"
|
|
|
|
## Trace debugging
|
|
- name: "ltrace"
|
|
within: []
|
|
message: "ltrace left in code"
|
|
- name: "ltraceM"
|
|
within: []
|
|
message: "ltraceM left in code"
|
|
- name: "traceToFile"
|
|
within: []
|
|
message: "traceToFile left in code"
|
|
- name: "traceToFileM"
|
|
within: []
|
|
message: "traceToFileM left in code"
|
|
|
|
################################################################################
|
|
# Ignore some builtin hints
|
|
|
|
- ignore: {name: Reduce duplication}
|
|
- ignore: {name: Redundant $}
|
|
- ignore: {name: Redundant bracket}
|
|
- ignore: {name: Use fmap}
|
|
- ignore: {name: Use first}
|
|
- ignore: {name: Use if}
|
|
- ignore: {name: Redundant <$>}
|
|
- ignore: {name: Use ++}
|
|
- ignore: {name: Use const}
|
|
- ignore: {name: Eta reduce}
|
|
- ignore: {name: Redundant multi-way if}
|
|
- ignore: {name: Use newtype instead of data}
|
|
- ignore: {name: Use bimap}
|
|
- ignore: {name: Use fromRight}
|
|
- ignore: {name: Use $>}
|
|
- ignore: {name: Use <$>}
|
|
- ignore: {name: "Use ?~"}
|
|
- ignore: {name: Use <=<}
|
|
- ignore: {name: Use sequenceA}
|
|
- ignore: {name: Use camelCase}
|
|
- ignore: {name: Redundant return}
|
|
- ignore: {name: Use <$>, within: [Hasura.RQL.DDL.Metadata, Hasura.Backends.MSSQL.Types.Instances]}
|
|
- ignore: {name: Functor law, within: Hasura.Server.AuthSpec}
|
|
- ignore: {name: Use underscore} # suggests to format port numbers
|
|
# These three need to be disabled under 'simplified subsumption' unfortunately.
|
|
# If we want to use -XDeepSubsumption these can be reenabled and lints fixed
|
|
# See: https://github.com/mpickering/ghc-proposals/blob/deep-subsumption/proposals/0000-deep-subsumption.rst
|
|
- ignore: {name: Avoid lambda}
|
|
- ignore: {name: Redundant lambda}
|
|
- ignore: {name: Avoid lambda using `infix`}
|
|
|
|
|
|
################################################################################
|
|
# Custom rules
|
|
|
|
- suggest: {lhs: onJust, rhs: for_}
|
|
|
|
- group:
|
|
name: hasura-prelude
|
|
enabled: true
|
|
imports:
|
|
- package base
|
|
rules:
|
|
- error: {lhs: "a == []", rhs: "null a"}
|
|
- error: {lhs: "a /= []", rhs: "not (null a)"}
|
|
- error: {lhs: "maybe b return a", rhs: "onNothing a b"}
|
|
- error: {lhs: "maybe b pure a", rhs: "onNothing a b"}
|
|
- error: {lhs: "maybe (return ()) b a", rhs: "for_ a b"}
|
|
- error: {lhs: "maybe (pure ()) b a", rhs: "for_ a b"}
|
|
- error: {lhs: "either b return a", rhs: "onLeft a b"}
|
|
- error: {lhs: "either b pure a", rhs: "onLeft a b"}
|
|
- error: {lhs: "maybe (f x) f", rhs: "f . fromMaybe x"}
|
|
- error: {lhs: "maybe (f x) f", rhs: "f . fromMaybe x"}
|
|
- error: {lhs: "either (f . g) (f . h)", rhs: "f . either g h"}
|
|
- error: {lhs: "either (f . g) (f . h)", rhs: "f . either g h"}
|
|
- error: {lhs: "onNothing x (return y)", rhs: "return (fromMaybe y x)"}
|
|
- error: {lhs: "onNothing x (pure y)", rhs: "pure (fromMaybe y x)"}
|
|
- error: {lhs: "onLeft x (return . f)", rhs: "return (either f id x)"}
|
|
- error: {lhs: "onLeft x (pure . f)", rhs: "pure (either f id x)"}
|
|
- error: {lhs: "case x of {Right a -> pure a; Left c -> d}", rhs: "onLeft x (\\ c -> d)"}
|
|
- error: {lhs: "case x of {Left c -> d; Right a -> pure a}", rhs: "onLeft x (\\ c -> d)"}
|
|
- error: {lhs: "case x of {Right a -> return a; Left c -> d}", rhs: "onLeft x (\\ c -> d)"}
|
|
- error: {lhs: "case x of {Left c -> d; Right a -> return a}", rhs: "onLeft x (\\ c -> d)"}
|
|
- error: {lhs: "case x of {Nothing -> a; Just b -> pure b}", rhs: "onNothing x a"}
|
|
- error: {lhs: "case x of {Just b -> pure b; Nothing -> a}", rhs: "onNothing x a"}
|
|
- error: {lhs: "case x of {Nothing -> a; Just b -> return b}", rhs: "onNothing x a"}
|
|
- error: {lhs: "case x of {Just b -> return b; Nothing -> a}", rhs: "onNothing x a"}
|
|
- error: {lhs: "if x then pure (); else y", rhs: "unless x y"}
|
|
- error: {lhs: "if x then return (); else y", rhs: "unless x y"}
|
|
- error: {lhs: "if x then y; else pure ()", rhs: "when x y"}
|
|
- error: {lhs: "if x then y; else return ()", rhs: "when x y"}
|
|
- error: {lhs: "unless (not x)", rhs: "when x"}
|
|
- error: {lhs: "when (not x)", rhs: "unless x"}
|
|
- error: {lhs: "arr id", rhs: "returnA"}
|
|
- suggest: {lhs: "not x && not y", rhs: "not (x || y)"}
|
|
- suggest: {lhs: "not x || not y", rhs: "not (x && y)"}
|
|
- error: {lhs: "Data.Text.pack (Prelude.show x)", rhs: "Hasura.Prelude.tshow x"}
|
|
- error: {lhs: "map f xs == map f ys", rhs: "liftEq ((==) `on` f)", note: "This is liable to be faster"}
|
|
- suggest: {lhs: "fmap f xs == fmap f ys", rhs: "liftEq ((==) `on` f)", note: "This is liable to be faster"}
|
|
# mapKeys:
|
|
- error: {lhs: "Data.HashMap.Strict.Extended.fromList . map (first f) . Data.HashMap.Strict.Extended.toList", rhs: "mapKeys f"}
|
|
- error: {lhs: "Data.HashMap.Strict.fromList . map (first f) . Data.HashMap.Strict.toList", rhs: "mapKeys f"}
|
|
- error: {lhs: "Data.HashMap.Strict.Extended.fromList $ map (first f) $ Data.HashMap.Strict.Extended.toList x", rhs: "mapKeys f x"}
|
|
- error: {lhs: "Data.HashMap.Strict.fromList $ map (first f) $ Data.HashMap.Strict.toList x", rhs: "mapKeys f x"}
|
|
# MaybeT
|
|
- error: {lhs: "MaybeT (pure x)", rhs: "hoistMaybe x"}
|
|
- error: {lhs: "MaybeT (return x)", rhs: "hoistMaybe x"}
|
|
- error: {lhs: "MaybeT (pure Nothing)", rhs: "Control.Applicative.empty"}
|
|
- error: {lhs: "MaybeT (return Nothing)", rhs: "Control.Applicative.empty"}
|
|
# catMaybes
|
|
- error: {lhs: "Data.HashMap.Strict.mapMaybe id", rhs: "Data.HashMap.Strict.Extended.catMaybes"}
|
|
- error: {lhs: "Data.HashMap.Strict.Extended.mapMaybe id", rhs: "Data.HashMap.Strict.Extended.catMaybes"}
|
|
- error: {lhs: "Data.HashMap.Strict.InsOrd.mapMaybe id", rhs: "Data.HashMap.Strict.InsOrd.Extended.catMaybes"}
|
|
- error: {lhs: "Data.HashMap.Strict.InsOrd.Extended.mapMaybe id", rhs: "Data.HashMap.Strict.InsOrd.Extended.catMaybes"}
|
|
- error: {lhs: "Data.IntMap.Strict.mapMaybe id", rhs: "Data.HashMap.Strict.Extended.catMaybes"}
|
|
- error: {lhs: "Data.IntMap.Strict.Extended.mapMaybe id", rhs: "Data.HashMap.Strict.Extended.catMaybes"}
|
|
|
|
- group:
|
|
name: data-text-extended
|
|
enabled: true
|
|
imports:
|
|
- package base
|
|
rules:
|
|
- error: {lhs: "Data.Text.intercalate \", \" x", rhs: "commaSeparated x", note: "From Data.Text.Extended"}
|
|
- error: {lhs: "Data.Text.Extended.intercalate \", \" x", rhs: "commaSeparated x"}
|
|
|
|
- group:
|
|
name: mssql-transaction
|
|
enabled: true
|
|
imports:
|
|
- package base
|
|
rules:
|
|
- error: {lhs: "Database.MSSQL.runTxE id tx pool", rhs: "Database.MSSQL.runTx tx pool"}
|
|
- error: {lhs: "Database.MSSQL.unitQueryE id q", rhs: "Database.MSSQL.unitQuery q"}
|
|
- error: {lhs: "Database.MSSQL.singleRowQueryE id tx pool", rhs: "Database.MSSQL.singleRowQuery tx pool"}
|
|
- error: {lhs: "Database.MSSQL.multiRowQueryE id tx pool", rhs: "Database.MSSQL.multiRowQuery tx pool"}
|
|
|
|
- group:
|
|
name: unsafe
|
|
enabled: true
|
|
imports:
|
|
- package base
|
|
rules:
|
|
- error:
|
|
name: "avoid Language.GraphQL.Draft.Syntax.unsafeMkName"
|
|
lhs: "Language.GraphQL.Draft.Syntax.unsafeMkName"
|
|
rhs: "Language.GraphQL.Draft.Syntax.mkName"
|
|
note: "avoid unsafeMkName if possible"
|
|
- error:
|
|
name: "avoid Control.Concurrent.threadDelay"
|
|
lhs: "Control.Concurrent.threadDelay"
|
|
rhs: "Control.Concurrent.Extended.sleep"
|
|
note: "Please use `sleep` instead (and read the docs!)"
|
|
- error:
|
|
name: "avoid InsecurelyShowDetails"
|
|
lhs: "Hasura.Backends.BigQuery.Execute.InsecurelyShowDetails"
|
|
rhs: "Hasura.Backends.BigQuery.Execute.HideDetails"
|
|
note: "Using 'InsecurelyShowDetails' could potentially expose sensitive information."
|
|
- suggest:
|
|
name: "avoid Control.Concurrent.Async.Lifted.Safe.async"
|
|
lhs: "Control.Concurrent.Async.Lifted.Safe.async"
|
|
rhs: "Control.Concurrent.Async.Lifted.Safe.withAsync"
|
|
note: "Please use 'Control.Control.Concurrent.Async.Lifted.Safe.withAsync' or our 'forkImmortal' instead for more robust threading."
|
|
- suggest:
|
|
name: "avoid Control.Concurrent.forkIO"
|
|
lhs: "Control.Concurrent.forkIO"
|
|
rhs: "Control.Concurrent.Async.Lifted.Safe.withAsync"
|
|
note: "Please use 'Control.Control.Concurrent.Async.Lifted.Safe.withAsync' or our 'forkImmortal' instead for more robust threading."
|
|
- suggest:
|
|
name: "avoid getEnvironment"
|
|
lhs: "System.Environment.getEnvironment"
|
|
rhs: "env_from_function_argument"
|
|
note: "Instead of 'getEnvironment', use the environment from the server context, typically accessible as a function argument"
|
|
- suggest:
|
|
name: "avoid getEnvironment"
|
|
lhs: "Data.Environment.getEnvironment"
|
|
rhs: "env_from_function_argument"
|
|
note: "Instead of 'getEnvironment', use the environment from the server context, typically accessible as a function argument"
|
|
|
|
|
|
################################################################################
|
|
# Exceptions to our own rules
|
|
|
|
- ignore:
|
|
within: ["Harness.**", "Spec", "Spec.**", "SpecHook", "Test.**", "Hasura.QuickCheck.**"]
|
|
name:
|
|
- "avoid Control.Concurrent.threadDelay"
|
|
- "avoid Control.Concurrent.Async.Lifted.Safe.async"
|
|
- "avoid Control.Concurrent.forkIO"
|
|
- "avoid getEnvironment"
|
|
- "avoid Language.GraphQL.Draft.Syntax.unsafeMkName"
|
|
|
|
- ignore:
|
|
within: ["Language.GraphQL.Draft.**"]
|
|
name: "avoid Language.GraphQL.Draft.Syntax.unsafeMkName"
|
|
|
|
|
|
################################################################################
|
|
# Import conventions
|
|
|
|
- modules:
|
|
- name:
|
|
- Data.Aeson
|
|
- Data.Aeson.Extended
|
|
as: J
|
|
- name:
|
|
- Data.HashMap.Strict
|
|
- Data.HashMap.Strict.Extended
|
|
as: HashMap
|
|
- name:
|
|
- Data.HashMap.Strict.InsOrd
|
|
- Data.HashMap.Strict.InsOrd.Extended
|
|
as: InsOrdHashMap
|
|
|
|
################################################################################
|
|
# Import restrictions
|
|
|
|
# We limit the place where the module `Hasura.RQL.Types.Instances` can be
|
|
# imported in order to make it easier to avoid module cycles.
|
|
#
|
|
# This check is initially quite limited, and will need to be extended to cover
|
|
# other instance-defining modules as well:
|
|
#
|
|
# * RQL.Types.Metadata.Instances
|
|
# * Hasura.GraphQL.Transport.Instances
|
|
# * Hasura.GraphQL.Execute.Instances
|
|
# * Hasura.GraphQL.Schema.Instances
|
|
# * Hasura.Server.API.Instances
|
|
# * Hasura.RQL.Types.SchemaCache.Instances
|
|
|
|
- modules:
|
|
- name: Hasura.RQL.Types.Instances
|
|
within: [
|
|
Hasura.RQL.IR.Select,
|
|
Hasura.RQL.Types.Metadata.Object,
|
|
Hasura.RQL.Types.Relationships.Remote,
|
|
Hasura.RQL.Types.SchemaCacheTypes,
|
|
Hasura.RQL.Types.Source,
|
|
Hasura.GraphQL.Schema.Build.UpdateSpec,
|
|
Test.Parser.Delete,
|
|
Test.Parser.Internal
|
|
]
|