-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path.hlint.yaml
179 lines (149 loc) · 6.59 KB
/
.hlint.yaml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
- warn:
name: "Use explicit module export list"
- warn:
name: "Use module export list"
- ignore:
name: "Use explicit module export list"
within: Tests.**
- ignore:
name: "Use module export list"
within: Tests.**
- warn:
lhs: anyOf o (== x)
rhs: elemOf o x
name: "Use elemOf"
- warn:
lhs: noneOf o (== x)
rhs: notElemOf o x
name: "Use notElemOf"
- warn:
lhs: foldl
rhs: foldl'
name: Strict fold
note: DecreasesLaziness
- warn:
lhs: foldM
rhs: foldlM
name: foldM
note: Be explicit about the direction of the fold
- group:
name: future
enabled: true
- group:
name: monomorphic
enabled: true
# Haskell's Dangerous Functions
# For a how-to-use and the latest version of this file go to:
# https://github.com/NorfairKing/haskell-dangerous-functions/
# This copy was from commit f8f3e511dc08b69d5164beea8ded7292e3cc0c1d,
# with a few removals where we disagree
- functions:
- {name: unsafeDupablePerformIO, within: []} # Unsafe
- {name: unsafeInterleaveIO, within: []} # Unsafe
- {name: unsafeFixIO, within: []} # Unsafe
- {name: unsafePerformIO, within: []} # Unsafe
# _VERY_ hard to get right, use the async library instead.
# See also https://github.com/informatikr/hedis/issues/165
- {name: forkIO, within: []}
# Mostly impossible to get right, rethink what you're doing entirely.
# See also https://www.reddit.com/r/haskell/comments/jsap9r/how_dangerous_is_forkprocess/
- {name: forkProcess, within: []}
- {name: undefined, within: []} # Purposely fails. Deal with errors appropriately instead.
- {name: throw, within: []} # Don't throw from pure code, use throwIO instead.
# TODO See https://github.com/hackworthltd/primer/issues/148
# - {name: Prelude.error, within: [Primer.Gen.**, Primer.**.Test.** , Tests.**]}
- {name: Data.List.head, within: []} # Partial, use `listToMaybe` instead.
- {name: Data.List.tail, within: []} # Partial
- {name: Data.List.init, within: []} # Partial
- {name: Data.List.last, within: []} # Partial
- {name: 'Data.List.!!', within: []} # Partial
- {name: Data.List.genericIndex, within: []} # Partial
- {name: Data.List.genericLength, within: []}
# Same, but for Data.Text
- {name: Data.Text.head, within: []}
- {name: Data.Text.tail, within: []}
- {name: Data.Text.init, within: []}
- {name: Data.Text.last, within: []}
- {name: Data.Foldable.minimum, within: []} # Partial
- {name: Data.Foldable.minimumBy, within: []} # Partial
- {name: Data.Foldable.maximum, within: []} # Partial
- {name: Data.Foldable.maximumBy, within: []} # Partial
- {name: Data.List.minimum, within: []} # Partial
- {name: Data.List.minimumBy, within: []} # Partial
- {name: Data.List.maximum, within: []} # Partial
- {name: Data.List.maximumBy, within: []} # Partial
# Same, but for Data.Text
- {name: Data.Text.maximum, within: []}
- {name: Data.Text.minimum, within: []}
- {name: GHC.Enum.pred, within: []} # Partial
- {name: GHC.Enum.succ, within: []} # Partial
- {name: GHC.Enum.toEnum, within: []} # Partial
- {name: GHC.Enum.fromEnum, within: []} # Does not do what you think it does.
- {name: GHC.Enum.enumFrom, within: []} # Does not do what you think it does, depending on the type.
- {name: GHC.Enum.enumFromThen, within: []} # Does not do what you think it does, depending on the type.
- {name: GHC.Enum.enumFromTo, within: []} # Does not do what you think it does, depending on the type.
- {name: GHC.Enum.enumFromThenTo, within: []} # Does not do what you think it does, depending on the type.
- {name: nub, within: []} # O(n^2)
- {name: Data.Foldable.foldl, within: []} # Lazy accumulator. Use foldl' instead.
- {name: Data.Foldable.foldMap, within: []} # Lazy accumulator. Use foldMap' instead.
- {name: Data.Foldable.sum, within: []} # Lazy accumulator
- {name: Data.Foldable.product, within: []} # Lazy accumulator
# Functions involving division
# TODO These are commented out as we would like more safety here in the future,
# but are unsure how to do so ergonomically.
# We have some instances of 'div', 'mod' and 'quotRem'
# where the denominator is obviously non-zero
# (and thus fine, but hlint cannot tell).
# We also have many instances of using 'mod' as a local variable
# (standing for "module"), which hlint gets confused by.
# - {name: Prelude.quot, within: []} # Partial, see https://github.com/NorfairKing/haskell-WAT#num-int
# - {name: Prelude.div, within: []}
# - {name: Prelude.rem, within: []}
# - {name: Prelude.mod, within: []}
# - {name: Prelude.quotRem, within: []}
# - {name: Prelude.divMod, within: []}
# Does unexpected things, see
# https://github.com/NorfairKing/haskell-WAT#real-double
- {name: realToFrac, within: []}
# Constructs rationals, which is either wrong or a bad idea.
- {name: 'Data.Ratio.%', within: []}
# Don't use string for command-line output.
- {name: System.IO.putChar, within: []}
- {name: System.IO.putStr, within: []}
- {name: System.IO.putStrLn, within: []}
- {name: System.IO.print, within: []}
# Don't use string for command-line input either.
- {name: System.IO.getChar, within: []}
- {name: System.IO.getLine, within: []}
- {name: System.IO.getContents, within: []} # Does lazy IO.
- {name: System.IO.interact, within: []}
- {name: System.IO.readIO, within: []}
- {name: System.IO.readLn, within: []}
# Don't use strings to interact with files
- {name: System.IO.readFile, within: []}
- {name: System.IO.writeFile, within: []}
- {name: System.IO.appendFile, within: []}
# Can succeed in dev, but fail in prod, because of encoding guessing
# It's also Lazy IO.
# See https://www.snoyman.com/blog/2016/12/beware-of-readfile/ for more info.
- {name: Data.Text.IO.readFile, within: []}
- {name: Data.Text.IO.Lazy.readFile, within: []}
- {name: Data.Text.Encoding.decodeUtf8, within: []} # Throws on invalid UTF8
- {name: fromJust, within: [Tests.**], message: 'Partial'} # Partial
# Does silent truncation:
# > fromIntegral (300 :: Word) :: Word8
# 44
# TODO: review occurrences of this in the codebase,
# see https://github.com/hackworthltd/primer/issues/848
# - {name: fromIntegral, within: []}
# - {name: fromInteger, within: []}
- {name: 'read', within: []} # Partial, use `Text.Read.readMaybe` instead.
# Deprecated, use `pure` instead.
# See https://gitlab.haskell.org/ghc/ghc/-/wikis/proposal/monad-of-no-return
- {name: 'return', within: []}
- modules:
# We prefer to use Optics, rather than Lens
- { name: Control.Lens, within: [] }
- extensions:
- { name: NamedFieldPuns, within: [] }
- { name: TupleSections, within: [] }