feat(xanthous): Track the volume and density of item types

Allow the itemType raw to have density and volume fields, both of which
represent *intervals* of both density and volume (because both can
hypothetically vary a bit). The idea here is that when we're making
an *instance* of one of these items, we pick a random value in the
range.

Lots of stuff in this commit is datatype and typeclass instances to
support things like intervals being fields on datatypes that get
serialized to saved games - including a manual definition of Ord for
Item since Ord isn't well-defined for intervals

Change-Id: Ia088f2f75cdce9d00560297e5c269e3310b85bc3
Reviewed-on: https://cl.tvl.fyi/c/depot/+/3225
Reviewed-by: grfn <grfn@gws.fyi>
Tested-by: BuildkiteCI
This commit is contained in:
Griffin Smith 2021-06-19 10:42:32 -04:00 committed by grfn
parent 638b355aa6
commit 8b97683f6e
12 changed files with 255 additions and 33 deletions

View file

@ -94,6 +94,7 @@ default-extensions:
- GADTSyntax - GADTSyntax
- GeneralizedNewtypeDeriving - GeneralizedNewtypeDeriving
- KindSignatures - KindSignatures
- StandaloneKindSignatures
- LambdaCase - LambdaCase
- MultiWayIf - MultiWayIf
- NoImplicitPrelude - NoImplicitPrelude

View file

@ -89,7 +89,8 @@ type TagSingleConstructors = 'TagSingleConstructors
class Demotable (a :: k) where class Demotable (a :: k) where
demote :: proxy a -> Demoted k demote :: proxy a -> Demoted k
type family All (p :: Type -> Constraint) (xs :: [k]) :: Constraint where type All :: (Type -> Constraint) -> [Type] -> Constraint
type family All p xs where
All p '[] = () All p '[] = ()
All p (x ': xs) = (p x, All p xs) All p (x ': xs) = (p x, All p xs)

View file

@ -32,6 +32,7 @@ import Xanthous.Data
, position , position
, Position , Position
, (|*|) , (|*|)
, Tiles(..)
) )
import Xanthous.Data.App (ResourceName, Panel(..), AppEvent(..)) import Xanthous.Data.App (ResourceName, Panel(..), AppEvent(..))
import qualified Xanthous.Data.EntityMap as EntityMap import qualified Xanthous.Data.EntityMap as EntityMap
@ -127,7 +128,7 @@ handleCommand (Move dir) = do
collisionAt newPos >>= \case collisionAt newPos >>= \case
Nothing -> do Nothing -> do
characterPosition .= newPos characterPosition .= newPos
stepGameBy =<< uses (character . speed) (|*| 1) stepGameBy =<< uses (character . speed) (|*| Tiles 1)
describeEntitiesAt newPos describeEntitiesAt newPos
Just Combat -> attackAt newPos Just Combat -> attackAt newPos
Just Stop -> pure () Just Stop -> pure ()

View file

@ -30,7 +30,7 @@ autoStep (AutoMove dir) = do
collisionAt newPos >>= \case collisionAt newPos >>= \case
Nothing -> do Nothing -> do
characterPosition .= newPos characterPosition .= newPos
stepGameBy =<< uses (character . speed) (|*| 1) stepGameBy =<< uses (character . speed) (|*| (1 :: Tiles))
describeEntitiesAt newPos describeEntitiesAt newPos
cancelIfDanger cancelIfDanger
Just _ -> cancelAutocommand Just _ -> cancelAutocommand

View file

@ -3,8 +3,6 @@
{-# LANGUAGE RoleAnnotations #-} {-# LANGUAGE RoleAnnotations #-}
{-# LANGUAGE RecordWildCards #-} {-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE DeriveTraversable #-} {-# LANGUAGE DeriveTraversable #-}
{-# LANGUAGE DeriveFoldable #-}
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE NoTypeSynonymInstances #-} {-# LANGUAGE NoTypeSynonymInstances #-}
{-# LANGUAGE DuplicateRecordFields #-} {-# LANGUAGE DuplicateRecordFields #-}
@ -56,6 +54,10 @@ module Xanthous.Data
, TicksPerTile , TicksPerTile
, TilesPerTick , TilesPerTick
, timesTiles , timesTiles
, Square(..)
, Cubic(..)
, Grams
, Meters
-- * -- *
, Dimensions'(..) , Dimensions'(..)
@ -490,9 +492,9 @@ rotations orig@(Neighbors tl t tr l r bl b br) = V4
newtype Per a b = Rate Double newtype Per a b = Rate Double
deriving stock (Show, Eq, Generic) deriving stock (Show, Eq, Generic)
deriving anyclass (NFData, CoArbitrary, Function) deriving anyclass (NFData, CoArbitrary, Function)
deriving (Num, Ord, Enum, Real, Fractional, ToJSON, FromJSON) via Double deriving (Arbitrary, Num, Ord, Enum, Real, Fractional, ToJSON, FromJSON)
via Double
deriving (Semigroup, Monoid) via Product Double deriving (Semigroup, Monoid) via Product Double
instance Arbitrary (Per a b) where arbitrary = genericArbitrary
invertRate :: a `Per` b -> b `Per` a invertRate :: a `Per` b -> b `Per` a
invertRate (Rate p) = Rate $ 1 / p invertRate (Rate p) = Rate $ 1 / p
@ -500,9 +502,42 @@ invertRate (Rate p) = Rate $ 1 / p
invertedRate :: Iso (a `Per` b) (b' `Per` a') (b `Per` a) (a' `Per` b') invertedRate :: Iso (a `Per` b) (b' `Per` a') (b `Per` a) (a' `Per` b')
invertedRate = iso invertRate invertRate invertedRate = iso invertRate invertRate
type (:*:) :: Type -> Type -> Type
type family (:*:) a b where
(a `Per` b) :*: b = a
(Square a) :*: a = Cubic a
a :*: a = Square a
infixl 7 |*| infixl 7 |*|
(|*|) :: (Scalar a, Scalar b) => a `Per` b -> b -> a class MulUnit a b where
(|*|) (Rate rate) b = fromScalar $ rate * scalar b (|*|) :: a -> b -> a :*: b
instance (Scalar a, Scalar b) => MulUnit (a `Per` b) b where
(Rate rate) |*| b = fromScalar $ rate * scalar b
instance forall a. (Scalar a, a :*: a ~ Square a) => MulUnit a a where
x' |*| y' = Square @a . fromScalar $ scalar x' * scalar y'
instance forall a. (Scalar a) => MulUnit (Square a) a where
x' |*| y' = Cubic @a . fromScalar $ scalar x' * scalar y'
newtype Square a = Square a
deriving stock (Show, Eq, Generic)
deriving anyclass (NFData, CoArbitrary, Function)
deriving ( Arbitrary, Num, Ord, Enum, Real, Fractional, ToJSON, FromJSON
, Scalar
)
via a
newtype Cubic a = Cubic a
deriving stock (Show, Eq, Generic)
deriving anyclass (NFData, CoArbitrary, Function)
deriving ( Arbitrary, Num, Ord, Enum, Real, Fractional, ToJSON, FromJSON
, Scalar
)
via a
--------------------------------------------------------------------------------
newtype Ticks = Ticks Word newtype Ticks = Ticks Word
deriving stock (Show, Eq, Generic) deriving stock (Show, Eq, Generic)
@ -510,14 +545,14 @@ newtype Ticks = Ticks Word
deriving (Num, Ord, Bounded, Enum, Integral, Real, ToJSON, FromJSON) via Word deriving (Num, Ord, Bounded, Enum, Integral, Real, ToJSON, FromJSON) via Word
deriving (Semigroup, Monoid) via (Sum Word) deriving (Semigroup, Monoid) via (Sum Word)
deriving Scalar via ScalarIntegral Ticks deriving Scalar via ScalarIntegral Ticks
instance Arbitrary Ticks where arbitrary = genericArbitrary deriving Arbitrary via GenericArbitrary Ticks
newtype Tiles = Tiles Double newtype Tiles = Tiles Double
deriving stock (Show, Eq, Generic) deriving stock (Show, Eq, Generic)
deriving anyclass (NFData, CoArbitrary, Function) deriving anyclass (NFData, CoArbitrary, Function)
deriving (Num, Ord, Enum, Real, ToJSON, FromJSON, Scalar) via Double deriving (Num, Ord, Enum, Real, ToJSON, FromJSON, Scalar) via Double
deriving (Semigroup, Monoid) via (Sum Double) deriving (Semigroup, Monoid) via (Sum Double)
instance Arbitrary Tiles where arbitrary = genericArbitrary deriving Arbitrary via GenericArbitrary Tiles
type TicksPerTile = Ticks `Per` Tiles type TicksPerTile = Ticks `Per` Tiles
type TilesPerTick = Tiles `Per` Ticks type TilesPerTick = Tiles `Per` Ticks
@ -534,6 +569,23 @@ newtype Hitpoints = Hitpoints Word
via Word via Word
deriving (Semigroup, Monoid) via Sum Word deriving (Semigroup, Monoid) via Sum Word
--------------------------------------------------------------------------------
-- | Grams, the fundamental measure of weight in Xanthous.
newtype Grams = Grams Double
deriving stock (Show, Eq, Generic)
deriving anyclass (NFData, CoArbitrary, Function)
deriving ( Arbitrary, Num, Ord, Enum, Real, Floating, Fractional, RealFloat
, RealFrac, Scalar, ToJSON, FromJSON
)
via Double
deriving (Semigroup, Monoid) via Sum Double
-- | Every tile is 1 meter
type Meters = Tiles
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
data Box a = Box data Box a = Box

View file

@ -51,11 +51,12 @@ import Data.Aeson.Generic.DerivingVia
import Data.Aeson (ToJSON, FromJSON) import Data.Aeson (ToJSON, FromJSON)
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
import Xanthous.Messages (Message(..)) import Xanthous.Messages (Message(..))
import Xanthous.Data (TicksPerTile, Hitpoints) import Xanthous.Data (TicksPerTile, Hitpoints, Per, Grams, Cubic, Meters)
import Xanthous.Data.EntityChar import Xanthous.Data.EntityChar
import Xanthous.Util.QuickCheck import Xanthous.Util.QuickCheck
import Xanthous.Generators.Speech (Language, gormlak, english) import Xanthous.Generators.Speech (Language, gormlak, english)
import Xanthous.Orphans () import Xanthous.Orphans ()
import Data.Interval (Interval, lowerBound', upperBound')
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
-- | Identifiers for languages that creatures can speak. -- | Identifiers for languages that creatures can speak.
@ -153,10 +154,12 @@ data ItemType = ItemType
, _description :: !Text , _description :: !Text
, _longDescription :: !Text , _longDescription :: !Text
, _char :: !EntityChar , _char :: !EntityChar
, _density :: !(Interval (Grams `Per` Cubic Meters))
, _volume :: !(Interval (Cubic Meters))
, _edible :: !(Maybe EdibleItem) , _edible :: !(Maybe EdibleItem)
, _wieldable :: !(Maybe WieldableItem) , _wieldable :: !(Maybe WieldableItem)
} }
deriving stock (Show, Eq, Ord, Generic) deriving stock (Show, Eq, Generic)
deriving anyclass (NFData, CoArbitrary, Function) deriving anyclass (NFData, CoArbitrary, Function)
deriving Arbitrary via GenericArbitrary ItemType deriving Arbitrary via GenericArbitrary ItemType
deriving (ToJSON, FromJSON) deriving (ToJSON, FromJSON)
@ -164,6 +167,20 @@ data ItemType = ItemType
ItemType ItemType
makeFieldsNoPrefix ''ItemType makeFieldsNoPrefix ''ItemType
instance Ord ItemType where
compare x y
= compareOf name x y
<> compareOf description x y
<> compareOf longDescription x y
<> compareOf char x y
<> compareOf (density . to extractInterval) x y
<> compareOf (volume . to extractInterval) x y
<> compareOf edible x y
<> compareOf wieldable x y
where
compareOf l = comparing (view l)
extractInterval = lowerBound' &&& upperBound'
-- | Can this item be eaten? -- | Can this item be eaten?
isEdible :: ItemType -> Bool isEdible :: ItemType -> Bool
isEdible = has $ edible . _Just isEdible = has $ edible . _Just

View file

@ -10,3 +10,5 @@ Item:
hitpointsHealed: 2 hitpointsHealed: 2
eatMessage: eatMessage:
- You slurp up the noodles. Yumm! - You slurp up the noodles. Yumm!
density: 500000
volume: 0.001

View file

@ -12,3 +12,7 @@ Item:
- You bonk the {{creature.creatureType.name}} over the head with your stick. - You bonk the {{creature.creatureType.name}} over the head with your stick.
- You bash the {{creature.creatureType.name}} on the noggin with your stick. - You bash the {{creature.creatureType.name}} on the noggin with your stick.
- You whack the {{creature.creatureType.name}} with your stick. - You whack the {{creature.creatureType.name}} with your stick.
# https://www.sciencedirect.com/topics/agricultural-and-biological-sciences/wood-density
# it's a hard stick. so it's dense wood.
density: 890000 # g/m³
volume: [ 0.003, 0.006 ] # ≈3.5 cm radius × ≈1m length

View file

@ -1,10 +1,10 @@
{-# LANGUAGE RecordWildCards #-} {-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE StandaloneDeriving #-} {-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE UndecidableInstances #-} {-# LANGUAGE UndecidableInstances #-}
{-# LANGUAGE PackageImports #-} {-# LANGUAGE PackageImports #-}
{-# OPTIONS_GHC -Wno-orphans #-} {-# OPTIONS_GHC -Wno-orphans #-}
{-# OPTIONS_GHC -Wno-type-defaults #-}
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
{-# OPTIONS_GHC -Wno-deferred-type-errors #-}
module Xanthous.Orphans module Xanthous.Orphans
( ppTemplate ( ppTemplate
) where ) where
@ -28,11 +28,15 @@ import Text.Mustache
import Text.Mustache.Type ( showKey ) import Text.Mustache.Type ( showKey )
import Control.Monad.State import Control.Monad.State
import Linear import Linear
import qualified Data.Interval as Interval
import Data.Interval ( Interval, Extended (..), Boundary (..)
, lowerBound', upperBound', (<=..<), (<=..<=)
, interval)
import Test.QuickCheck.Checkers (EqProp ((=-=)))
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
import Xanthous.Util.JSON import Xanthous.Util.JSON
import Xanthous.Util.QuickCheck import Xanthous.Util.QuickCheck
import qualified Data.Interval as Interval import Xanthous.Util (EqEqProp(EqEqProp))
import Data.Interval (Interval, Extended (..))
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
instance forall s a. instance forall s a.
@ -241,6 +245,8 @@ instance CoArbitrary a => CoArbitrary (MaybeDefault a) where
instance (Eq a, Show a, Read a, Function a) => Function (MaybeDefault a) where instance (Eq a, Show a, Read a, Function a) => Function (MaybeDefault a) where
function = functionShow function = functionShow
deriving via (EqEqProp Attr) instance EqProp Attr
instance Arbitrary Attr where instance Arbitrary Attr where
arbitrary = do arbitrary = do
attrStyle <- arbitrary attrStyle <- arbitrary
@ -367,12 +373,46 @@ instance Function a => Function (V2 a)
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
instance Arbitrary r => Arbitrary (Extended r) where instance CoArbitrary Boundary
instance Function Boundary
instance Arbitrary a => Arbitrary (Extended a) where
arbitrary = oneof [ pure NegInf arbitrary = oneof [ pure NegInf
, pure PosInf , pure PosInf
, Finite <$> arbitrary , Finite <$> arbitrary
] ]
instance CoArbitrary a => CoArbitrary (Extended a) where
coarbitrary NegInf = variant 1
coarbitrary PosInf = variant 2
coarbitrary (Finite x) = variant 3 . coarbitrary x
instance (Function a) => Function (Extended a) where
function = functionMap g h
where
g NegInf = Left True
g (Finite a) = Right a
g PosInf = Left False
h (Left False) = PosInf
h (Left True) = NegInf
h (Right a) = Finite a
instance ToJSON a => ToJSON (Extended a) where
toJSON NegInf = String "NegInf"
toJSON PosInf = String "PosInf"
toJSON (Finite x) = toJSON x
instance FromJSON a => FromJSON (Extended a) where
parseJSON (String "NegInf") = pure NegInf
parseJSON (String "PosInf") = pure PosInf
parseJSON val = Finite <$> parseJSON val
instance (EqProp a, Show a) => EqProp (Extended a) where
NegInf =-= NegInf = property True
PosInf =-= PosInf = property True
(Finite x) =-= (Finite y) = x =-= y
x =-= y = counterexample (show x <> " /= " <> show y) False
instance Arbitrary Interval.Boundary where instance Arbitrary Interval.Boundary where
arbitrary = elements [ Interval.Open , Interval.Closed ] arbitrary = elements [ Interval.Open , Interval.Closed ]
@ -384,3 +424,60 @@ instance (Ord r, Arbitrary r) => Arbitrary (Interval r) where
Interval.interval Interval.interval
lower lower
upper upper
instance CoArbitrary a => CoArbitrary (Interval a) where
coarbitrary int = coarbitrary (lowerBound' int) . coarbitrary (upperBound' int)
instance (Function a, Ord a) => Function (Interval a) where
function = functionMap g h
where
g = lowerBound' &&& upperBound'
h = uncurry interval
deriving via (EqEqProp (Interval a)) instance Eq a => (EqProp (Interval a))
instance ToJSON a => ToJSON (Interval a) where
toJSON x = Array . fromList $
[ object [ lowerKey .= lowerVal ]
, object [ upperKey .= upperVal ]
]
where
(lowerVal, lowerBoundary) = lowerBound' x
(upperVal, upperBoundary) = upperBound' x
upperKey = boundaryToKey upperBoundary
lowerKey = boundaryToKey lowerBoundary
boundaryToKey Open = "Excluded"
boundaryToKey Closed = "Included"
instance forall a. (FromJSON a, Ord a) => FromJSON (Interval a) where
parseJSON x =
boundPairWithBoundary x
<|> boundPairWithoutBoundary x
<|> singleVal x
where
boundPairWithBoundary = withArray "Bound pair" $ \arr -> do
checkLength arr
lower <- parseBound $ arr ^?! ix 0
upper <- parseBound $ arr ^?! ix 1
pure $ interval lower upper
parseBound = withObject "Bound" $ \obj -> do
when (length obj /= 1) $ fail "Expected an object with a single key"
let [(k, v)] = obj ^@.. ifolded
boundary <- case k of
"Excluded" -> pure Open
"Open" -> pure Open
"Included" -> pure Closed
"Closed" -> pure Closed
_ -> fail "Invalid boundary specification"
val <- parseJSON v
pure (val, boundary)
boundPairWithoutBoundary = withArray "Bound pair" $ \arr -> do
checkLength arr
lower <- parseJSON $ arr ^?! ix 0
upper <- parseJSON $ arr ^?! ix 1
pure $ lower <=..< upper
singleVal v = do
val <- parseJSON v
pure $ val <=..<= val
checkLength arr =
when (length arr /= 2) $ fail "Expected array of length 2"

View file

@ -1,3 +1,5 @@
{-# LANGUAGE AllowAmbiguousTypes #-}
--------------------------------------------------------------------------------
module Test.Prelude module Test.Prelude
( module Xanthous.Prelude ( module Xanthous.Prelude
, module Test.Tasty , module Test.Tasty
@ -5,15 +7,26 @@ module Test.Prelude
, module Test.Tasty.QuickCheck , module Test.Tasty.QuickCheck
, module Test.QuickCheck.Classes , module Test.QuickCheck.Classes
, testBatch , testBatch
, jsonRoundTrip
) where ) where
--------------------------------------------------------------------------------
import Xanthous.Prelude hiding (assert, elements) import Xanthous.Prelude hiding (assert, elements)
import Test.Tasty --------------------------------------------------------------------------------
import Test.Tasty.QuickCheck import Test.Tasty
import Test.Tasty.HUnit import Test.Tasty.QuickCheck
import Test.QuickCheck.Classes import Test.Tasty.HUnit
import Test.QuickCheck.Checkers (TestBatch) import Test.QuickCheck.Classes
import Test.QuickCheck.Instances.ByteString () import Test.QuickCheck.Checkers (TestBatch, EqProp ((=-=)))
import Test.QuickCheck.Instances.ByteString ()
--------------------------------------------------------------------------------
import qualified Data.Aeson as JSON
import Data.Aeson (ToJSON, FromJSON)
--------------------------------------------------------------------------------
testBatch :: TestBatch -> TestTree testBatch :: TestBatch -> TestTree
testBatch (name, tests) = testGroup name $ uncurry testProperty <$> tests testBatch (name, tests) = testGroup name $ uncurry testProperty <$> tests
jsonRoundTrip
:: forall a. (ToJSON a, FromJSON a, EqProp a, Arbitrary a, Show a) => TestTree
jsonRoundTrip = testProperty "JSON round trip" $ \(x :: a) ->
JSON.decode (JSON.encode x) =-= Just x

View file

@ -1,4 +1,5 @@
{-# LANGUAGE BlockArguments #-} {-# LANGUAGE BlockArguments #-}
{-# LANGUAGE OverloadedLists #-}
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
module Xanthous.OrphansSpec where module Xanthous.OrphansSpec where
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -8,6 +9,10 @@ import Text.Mustache
import Text.Megaparsec (errorBundlePretty) import Text.Megaparsec (errorBundlePretty)
import Graphics.Vty.Attributes import Graphics.Vty.Attributes
import qualified Data.Aeson as JSON import qualified Data.Aeson as JSON
import Data.Interval (Interval, (<=..<=), (<=..<), (<..<=))
import Data.Aeson ( ToJSON(toJSON), object, Value(Array) )
import Data.Aeson.Types (fromJSON)
import Data.IntegerInterval (Extended(Finite))
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
import Xanthous.Orphans import Xanthous.Orphans
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -36,7 +41,32 @@ test = testGroup "Xanthous.Orphans"
$ JSON.decode (JSON.encode tpl) === Just tpl $ JSON.decode (JSON.encode tpl) === Just tpl
] ]
, testGroup "Attr" , testGroup "Attr"
[ testProperty "JSON round trip" $ \(attr :: Attr) -> [ jsonRoundTrip @Attr ]
JSON.decode (JSON.encode attr) === Just attr , testGroup "Extended"
[ jsonRoundTrip @(Extended Int) ]
, testGroup "Interval"
[ testGroup "JSON"
[ jsonRoundTrip @(Interval Int)
, testCase "parses a single value as a length-1 interval" $
getSuccess (fromJSON $ toJSON (1 :: Int))
@?= Just (Finite (1 :: Int) <=..<= Finite 1)
, testCase "parses a pair of values as a single-ended interval" $
getSuccess (fromJSON $ toJSON ([1, 2] :: [Int]))
@?= Just (Finite (1 :: Int) <=..< Finite (2 :: Int))
, testCase "parses the full included/excluded syntax" $
getSuccess (fromJSON $ Array [ object [ "Excluded" JSON..= (1 :: Int) ]
, object [ "Included" JSON..= (4 :: Int) ]
])
@?= Just (Finite (1 :: Int) <..<= Finite (4 :: Int))
, testCase "parses open/closed as aliases" $
getSuccess (fromJSON $ Array [ object [ "Open" JSON..= (1 :: Int) ]
, object [ "Closed" JSON..= (4 :: Int) ]
])
@?= Just (Finite (1 :: Int) <..<= Finite (4 :: Int))
]
] ]
] ]
where
getSuccess :: JSON.Result a -> Maybe a
getSuccess (JSON.Error _) = Nothing
getSuccess (JSON.Success r) = Just r

View file

@ -4,7 +4,7 @@ cabal-version: 1.12
-- --
-- see: https://github.com/sol/hpack -- see: https://github.com/sol/hpack
-- --
-- hash: b39d4dd906804ca42f8965c2dbe519434e80622fef7fafce1dca0d211a1c6663 -- hash: fdfa821ad291b11a2d7a7ee9cc38d7980a9b1f494b77216b141d3424168d621d
name: xanthous name: xanthous
version: 0.1.0.0 version: 0.1.0.0
@ -104,6 +104,7 @@ library
GADTSyntax GADTSyntax
GeneralizedNewtypeDeriving GeneralizedNewtypeDeriving
KindSignatures KindSignatures
StandaloneKindSignatures
LambdaCase LambdaCase
MultiWayIf MultiWayIf
NoImplicitPrelude NoImplicitPrelude
@ -261,6 +262,7 @@ executable xanthous
GADTSyntax GADTSyntax
GeneralizedNewtypeDeriving GeneralizedNewtypeDeriving
KindSignatures KindSignatures
StandaloneKindSignatures
LambdaCase LambdaCase
MultiWayIf MultiWayIf
NoImplicitPrelude NoImplicitPrelude
@ -385,6 +387,7 @@ test-suite test
GADTSyntax GADTSyntax
GeneralizedNewtypeDeriving GeneralizedNewtypeDeriving
KindSignatures KindSignatures
StandaloneKindSignatures
LambdaCase LambdaCase
MultiWayIf MultiWayIf
NoImplicitPrelude NoImplicitPrelude
@ -494,6 +497,7 @@ benchmark benchmark
GADTSyntax GADTSyntax
GeneralizedNewtypeDeriving GeneralizedNewtypeDeriving
KindSignatures KindSignatures
StandaloneKindSignatures
LambdaCase LambdaCase
MultiWayIf MultiWayIf
NoImplicitPrelude NoImplicitPrelude