Skip to content

Commit

Permalink
Improve docs: Gen.filter[T], Gen.mapMaybe[T], Tree.prune.
Browse files Browse the repository at this point in the history
`prune`'s documentation was incorrect. `filter`'s was at best
misleading, and in any case not detailed enough for me to avoid pitfalls
(hedgehogqa#484). The others were missing entirely.
  • Loading branch information
ChickenProp committed Apr 13, 2023
1 parent a4ceeea commit a14b5f1
Show file tree
Hide file tree
Showing 2 changed files with 62 additions and 8 deletions.
68 changes: 61 additions & 7 deletions hedgehog/src/Hedgehog/Internal/Gen.hs
Original file line number Diff line number Diff line change
Expand Up @@ -1328,19 +1328,33 @@ fromPred p a = a <$ guard (p a)

-- | Generates a value that satisfies a predicate.
--
-- This is essentially:
--
-- @
-- filter p gen = 'mfilter' p gen '<|>' filter p gen
-- @
-- Shrinks of the generated value will also satisfy the predicate. From the
-- original generator's shrink tree, any values that fail the predicate will
-- be removed, but any subsequent shrinks that satisfy it will be retained.
-- Compared to 'filter', shrinking may be slower but will be optimal.
--
-- It differs from the above in that we keep some state to avoid looping
-- forever. If we trigger these limits then the whole generator is discarded.
-- It's possible that the predicate will never pass, or will only pass at a
-- larger size than we're currently running at. To avoid looping forever, we
-- limit the number of retries, and grow the size with each retry. If we retry
-- too many times then the whole generator is discarded.
--
filter :: (MonadGen m, GenBase m ~ Identity) => (a -> Bool) -> m a -> m a
filter p =
mapMaybe (fromPred p)

-- | Generates a value which is the result of the given function returning a
-- 'Just'.
--
-- The original generator's shrink tree will be retained, with values
-- returning 'Nothing' removed. Subsequent shrinks of those values will be
-- retained. Compared to 'mapMaybeT', shrinking may be slower but will be
-- optimal.
--
-- It's possible that the function will never return 'Just', or will only do
-- so a larger size than we're currently running at. To avoid looping forever,
-- we limit the number of retries, and grow the size with each retry. If we
-- retry too many times then the whole generator is discarded.
--
mapMaybe :: (MonadGen m, GenBase m ~ Identity) => (a -> Maybe b) -> m a -> m b
mapMaybe p gen0 =
let
Expand All @@ -1358,10 +1372,50 @@ mapMaybe p gen0 =
in
try 0

-- | Generates a value that satisfies a predicate.
--
-- Shrinks of the generated value will also satisfy the predicate. From the
-- original generator's shrink tree, any values that fail the predicate will
-- be removed, along with their subsequent shrinks. Compared to 'filter',
-- shrinking may be faster but may also be less optimal.
--
-- The type is also more general, because the shrink behavior from 'filter'
-- would force the entire shrink tree to be evaluated when applied to an
-- impure tree.
--
-- This is essentially:
--
-- @
-- filterT p gen = 'mfilter' p gen '<|>' filterT p gen
-- @
--
-- But that could loop forever, if the predicate will never pass or will only
-- pass at a larger size than we're currently running at. We differ from the
-- above in keeping some state to avoid that. We limit the number of retries,
-- and grow the size with each retry. If we retry too many times then the
-- whole generator is discarded.
--
filterT :: MonadGen m => (a -> Bool) -> m a -> m a
filterT p =
mapMaybeT (fromPred p)

-- | Generates a value which is the result of the given function returning a
-- 'Just'.
--
-- The original generator's shrink tree will be retained, with values
-- returning 'Nothing' removed. Subsequent shrinks of those values will be
-- retained. Compared to 'mapMaybeT', shrinking may be slower but will be
-- optimal.
--
-- The type is also more general, because the shrink behavior from 'mapMaybe'
-- would force the entire shrink tree to be evaluated when applied to an
-- impure tree.
--
-- It's possible that the function will never return 'Just', or will only do
-- so a larger size than we're currently running at. To avoid looping forever,
-- we limit the number of retries, and grow the size with each retry. If we
-- retry too many times then the whole generator is discarded.
--
mapMaybeT :: MonadGen m => (a -> Maybe b) -> m a -> m b
mapMaybeT p gen0 =
let
Expand Down
2 changes: 1 addition & 1 deletion hedgehog/src/Hedgehog/Internal/Tree.hs
Original file line number Diff line number Diff line change
Expand Up @@ -188,7 +188,7 @@ expand f m =
pure . NodeT x $
fmap (expand f) xs ++ unfoldForest f x

-- | Throw away @n@ levels of a tree's children.
-- | Throw away all but the top @n@ levels of a tree's children.
--
-- /@prune 0@ will throw away all of a tree's children./
--
Expand Down

0 comments on commit a14b5f1

Please sign in to comment.