Merge pull request #2 from dgendill/master

Update for 0.11
This commit is contained in:
archaeron 2017-05-07 19:14:44 +02:00 committed by GitHub
commit 29e52559d0
6 changed files with 83 additions and 83 deletions

View File

@ -18,31 +18,32 @@ data GeoObject = GeoObject
## Decode YAML ## Decode YAML
Write `IsForeign` instances for your data structures. Write functions to read your data from foreign values.
```purescript ```purescript
instance pointIsForeign :: IsForeign Point where readPoint :: Foreign -> F Point
read value = do readPoint value = do
x <- readProp "X" value x <- readInt =<< readProp "X" value
y <- readProp "Y" value y <- readInt =<< readProp "Y" value
return $ Point x y pure $ Point x y
instance mobilityIsForeign :: IsForeign Mobility where readMobility :: Foreign -> F Mobility
read value = do readMobility value = do
mob <- readString value mob <- readString value
case mob of case mob of
"Fix" -> return Fix "Fix" -> pure Fix
"Flex" -> return Flex "Flex" -> pure Flex
_ -> Left $ JSONError "Mobility must be either Flex or Fix" _ -> fail $ JSONError "Mobility must be either Flex or Fix"
readGeoObject :: Foreign -> F GeoObject
readGeoObject value = do
name <- readString =<< readProp "Name" value
scale <- readNumber =<< readProp "Scale" value
points <- traverse readPoint =<< readArray =<< readProp "Points" value
mobility <- readMobility =<< readProp "Mobility" value
coverage <- readNumber =<< readProp "Coverage" value
pure $ GeoObject { name, scale, points, mobility, coverage }
instance archiObjectIsForeign :: IsForeign GeoObject where
read value = do
name <- readProp "Name" value
scale <- readProp "Scale" value
points <- readProp "Points" value
mobility <- readProp "Mobility" value
coverage <- readProp "Coverage" value
return $ GeoObject { name, scale, points, mobility, coverage }
``` ```
Read the YAML into your data structures. Read the YAML into your data structures.
@ -74,7 +75,10 @@ yamlInput = """
Coverage: 10 Coverage: 10
""" """
decoded = (readYAML yamlInput) :: F (Array GeoObject) let decoded =
(parseYAML yamlInput) >>=
readArray >>=
traverse readGeoObject
``` ```
## Encode YAML ## Encode YAML

View File

@ -12,14 +12,13 @@
], ],
"dependencies": { "dependencies": {
"js-yaml": "^3.4.6", "js-yaml": "^3.4.6",
"purescript-functions": "^2.0.0", "purescript-functions": "^3.0.0",
"purescript-foreign": "^3.2.0", "purescript-foreign": "^4.0.0",
"purescript-foreign-generic": "^3.0.0", "purescript-foreign-generic": "^4.1.0",
"purescript-unsafe-coerce": "^2.0.0" "purescript-unsafe-coerce": "^3.0.0"
}, },
"devDependencies": { "devDependencies": {
"purescript-console": "^2.0.0", "purescript-console": "^3.0.0",
"purescript-spec": "~0.12.1" "purescript-spec": "^1.0.0"
} }
} }

View File

@ -1,13 +1,12 @@
module Data.YAML.Foreign.Decode (parseYAML, readYAML, readYAMLGeneric) where module Data.YAML.Foreign.Decode (parseYAML, readYAMLGeneric) where
import Data.Foreign (F, Foreign, ForeignError(..), fail) import Data.Foreign (F, Foreign, ForeignError(..), fail)
import Data.Foreign.Class (class IsForeign, read) import Data.Foreign.Generic.Class (class GenericDecode)
import Data.Foreign.Generic (readGeneric) import Data.Foreign.Generic (genericDecode)
import Data.Foreign.Generic.Classes (class GenericDecode)
import Data.Foreign.Generic.Types (Options) import Data.Foreign.Generic.Types (Options)
import Data.Function.Uncurried (Fn3, runFn3) import Data.Function.Uncurried (Fn3, runFn3)
import Data.Generic.Rep (class Generic) import Data.Generic.Rep (class Generic)
import Prelude (pure, (<<<), (>>=), (>=>)) import Prelude ((>=>), (<<<), pure)
foreign import parseYAMLImpl :: forall r. Fn3 (String -> r) (Foreign -> r) String r foreign import parseYAMLImpl :: forall r. Fn3 (String -> r) (Foreign -> r) String r
@ -15,11 +14,6 @@ foreign import parseYAMLImpl :: forall r. Fn3 (String -> r) (Foreign -> r) Strin
parseYAML :: String -> F Foreign parseYAML :: String -> F Foreign
parseYAML yaml = runFn3 parseYAMLImpl (fail <<< JSONError) pure yaml parseYAML yaml = runFn3 parseYAMLImpl (fail <<< JSONError) pure yaml
-- | Attempt to parse a YAML string into the datastructure you want.
readYAML :: forall a. (IsForeign a) => String -> F a
readYAML yaml = parseYAML yaml >>= read
-- | Automatically generate a YAML parser for your data from a generic instance. -- | Automatically generate a YAML parser for your data from a generic instance.
readYAMLGeneric :: forall a rep. (Generic a rep, GenericDecode rep) => Options -> String -> F a readYAMLGeneric :: forall a rep. (Generic a rep) => (GenericDecode rep) => Options -> String -> F a
readYAMLGeneric opts = parseYAML >=> readGeneric opts readYAMLGeneric opts = parseYAML >=> genericDecode opts

View File

@ -1,7 +1,7 @@
module Data.YAML.Foreign.Encode where module Data.YAML.Foreign.Encode where
import Data.Map as M import Data.Map as M
import Data.Array (fromFoldable, toUnfoldable) import Data.Array (toUnfoldable)
import Data.Function.Uncurried (Fn4, runFn4) import Data.Function.Uncurried (Fn4, runFn4)
import Data.List (List) import Data.List (List)
import Data.Maybe (Maybe, maybe) import Data.Maybe (Maybe, maybe)
@ -11,7 +11,7 @@ import Unsafe.Coerce (unsafeCoerce)
type YObject = M.Map String YValue type YObject = M.Map String YValue
type YArray = Array YValue type YArray = Array YValue
foreign import data YAML :: * foreign import data YAML :: Type
data YValue data YValue
= YObject YObject = YObject YObject
@ -87,7 +87,7 @@ foreign import objToHash ::
YAML YAML
valueToYAML :: YValue -> YAML valueToYAML :: YValue -> YAML
valueToYAML (YObject o) = runFn4 objToHash valueToYAML fst snd $ fromFoldable $ M.toList o valueToYAML (YObject o) = runFn4 objToHash valueToYAML fst snd $ M.toUnfoldable o
valueToYAML (YArray a) = unsafeCoerce $ map valueToYAML a valueToYAML (YArray a) = unsafeCoerce $ map valueToYAML a
valueToYAML (YString s) = unsafeCoerce s valueToYAML (YString s) = unsafeCoerce s
valueToYAML (YNumber n) = unsafeCoerce n valueToYAML (YNumber n) = unsafeCoerce n
@ -99,4 +99,3 @@ foreign import toYAMLImpl :: YAML -> String
printYAML :: forall a. (ToYAML a) => a -> String printYAML :: forall a. (ToYAML a) => a -> String
printYAML = toYAMLImpl <<< valueToYAML <<< toYAML printYAML = toYAMLImpl <<< valueToYAML <<< toYAML

View File

@ -1,8 +1,9 @@
module Test.Instances where module Test.Instances where
import Prelude (class Eq, class Show, bind, pure, ($)) import Prelude (class Eq, class Show, bind, pure, ($), (=<<), (<$>), map, (<=<))
import Data.Foreign (ForeignError(..), fail, readString) import Data.Traversable (traverse)
import Data.Foreign.Class (class IsForeign, readProp) import Data.Foreign (readArray, readNumber, readString, readInt, F, Foreign, ForeignError(..), fail, readString)
import Data.Foreign.Index (readProp)
import Data.Generic (class Generic, gShow, gEq) import Data.Generic (class Generic, gShow, gEq)
import Data.YAML.Foreign.Encode import Data.YAML.Foreign.Encode
@ -32,28 +33,28 @@ derive instance genericMobility :: Generic Mobility
instance showMobility :: Show Mobility where show = gShow instance showMobility :: Show Mobility where show = gShow
instance eqMobility :: Eq Mobility where eq = gEq instance eqMobility :: Eq Mobility where eq = gEq
instance archiObjectIsForeign :: IsForeign GeoObject where readGeoObject :: Foreign -> F GeoObject
read value = do readGeoObject value = do
name <- readProp "Name" value name <- readString =<< readProp "Name" value
scale <- readProp "Scale" value scale <- readNumber =<< readProp "Scale" value
points <- readProp "Points" value points <- traverse readPoint =<< readArray =<< readProp "Points" value
mobility <- readProp "Mobility" value mobility <- readMobility =<< readProp "Mobility" value
coverage <- readProp "Coverage" value coverage <- readNumber =<< readProp "Coverage" value
pure $ GeoObject { name, scale, points, mobility, coverage } pure $ GeoObject { name, scale, points, mobility, coverage }
instance pointIsForeign :: IsForeign Point where readPoint :: Foreign -> F Point
read value = do readPoint value = do
x <- readProp "X" value x <- readInt =<< readProp "X" value
y <- readProp "Y" value y <- readInt =<< readProp "Y" value
pure $ Point x y pure $ Point x y
instance mobilityIsForeign :: IsForeign Mobility where readMobility :: Foreign -> F Mobility
read value = do readMobility value = do
mob <- readString value mob <- readString value
case mob of case mob of
"Fix" -> pure Fix "Fix" -> pure Fix
"Flex" -> pure Flex "Flex" -> pure Flex
_ -> fail $ JSONError "Mobility must be either Flex or Fix" _ -> fail $ JSONError "Mobility must be either Flex or Fix"
instance pointToYAML :: ToYAML Point where instance pointToYAML :: ToYAML Point where
toYAML (Point x y) = toYAML (Point x y) =
@ -75,4 +76,3 @@ instance archiObjectToYAML :: ToYAML GeoObject where
, "Mobility" := o.mobility , "Mobility" := o.mobility
, "Coverage" := o.coverage , "Coverage" := o.coverage
] ]

View File

@ -3,15 +3,17 @@ module Test.Main where
import Control.Monad.Eff (Eff) import Control.Monad.Eff (Eff)
import Control.Monad.Except (runExcept) import Control.Monad.Except (runExcept)
import Data.Either (Either(..)) import Data.Either (Either(..))
import Data.Foreign (F) import Data.Foreign (F, readArray)
import Data.YAML.Foreign.Decode (readYAML) import Data.YAML.Foreign.Decode (parseYAML)
import Data.YAML.Foreign.Encode (printYAML) import Data.YAML.Foreign.Encode (printYAML)
import Prelude (Unit, bind, ($)) import Data.Traversable (traverse)
import Test.Instances (GeoObject(..), Mobility(..), Point(..)) import Prelude (Unit, bind, ($), void, discard, (>>=))
import Test.Instances (readGeoObject, readMobility, readPoint, GeoObject(..), Mobility(..), Point(..))
import Test.Spec (describe, it) import Test.Spec (describe, it)
import Test.Spec.Assertions (shouldEqual) import Test.Spec.Assertions (shouldEqual)
import Test.Spec.Reporter.Console (consoleReporter) import Test.Spec.Reporter.Console (consoleReporter)
import Test.Spec.Runner (RunnerEffects, run) import Test.Spec.Runner (RunnerEffects, run)
import Control.Monad.Eff.Console (log, CONSOLE)
yamlInput :: String yamlInput :: String
yamlInput = """ yamlInput = """
@ -41,9 +43,7 @@ yamlInput = """
""" """
yamlOutput :: String yamlOutput :: String
yamlOutput = """- Coverage: 10 yamlOutput = """- Mobility: Fix
Mobility: Fix
Name: House
Points: Points:
- X: 10 - X: 10
'Y': 10 'Y': 10
@ -51,10 +51,10 @@ yamlOutput = """- Coverage: 10
'Y': 10 'Y': 10
- X: 5 - X: 5
'Y': 5 'Y': 5
Coverage: 10
Name: House
Scale: 9.5 Scale: 9.5
- Coverage: 10 - Mobility: Fix
Mobility: Fix
Name: Tree
Points: Points:
- X: 1 - X: 1
'Y': 1 'Y': 1
@ -62,6 +62,8 @@ yamlOutput = """- Coverage: 10
'Y': 2 'Y': 2
- X: 0 - X: 0
'Y': 0 'Y': 0
Coverage: 10
Name: Tree
Scale: 1 Scale: 1
""" """
@ -85,13 +87,15 @@ parsedData =
main :: Eff (RunnerEffects ()) Unit main :: Eff (RunnerEffects ()) Unit
main = run [consoleReporter] do main = run [consoleReporter] do
describe "purescript-yaml" do void $ describe "purescript-yaml" do
describe "decode" do describe "decode" do
it "Decodes YAML" do it "Decodes YAML" do
let decoded = (readYAML yamlInput) :: F (Array GeoObject) let decoded =
(parseYAML yamlInput) >>=
readArray >>=
traverse readGeoObject
(runExcept decoded) `shouldEqual` (Right parsedData) (runExcept decoded) `shouldEqual` (Right parsedData)
describe "encode" do void $ describe "encode" do
it "Encodes YAML" $ do it "Encodes YAML" $ do
let encoded = printYAML parsedData let encoded = printYAML parsedData
encoded `shouldEqual` yamlOutput encoded `shouldEqual` yamlOutput