From fecfce4880a1489b0bc7857c1636822b33fb5587 Mon Sep 17 00:00:00 2001 From: Dominick Gendill Date: Sat, 6 May 2017 22:25:58 -0600 Subject: [PATCH 1/2] Migrate for purescript 0.11 --- bower.json | 13 +++++----- src/Data/YAML/Foreign/Decode.purs | 18 +++++--------- src/Data/YAML/Foreign/Encode.purs | 7 +++--- test/Instances.purs | 40 +++++++++++++++++-------------- test/Main.purs | 32 ++++++++++++++----------- 5 files changed, 55 insertions(+), 55 deletions(-) diff --git a/bower.json b/bower.json index bc9d3de..ce1fabc 100644 --- a/bower.json +++ b/bower.json @@ -12,14 +12,13 @@ ], "dependencies": { "js-yaml": "^3.4.6", - "purescript-functions": "^2.0.0", - "purescript-foreign": "^3.2.0", - "purescript-foreign-generic": "^3.0.0", - "purescript-unsafe-coerce": "^2.0.0" + "purescript-functions": "^3.0.0", + "purescript-foreign": "^4.0.0", + "purescript-foreign-generic": "^4.1.0", + "purescript-unsafe-coerce": "^3.0.0" }, "devDependencies": { - "purescript-console": "^2.0.0", - "purescript-spec": "~0.12.1" + "purescript-console": "^3.0.0", + "purescript-spec": "^1.0.0" } } - diff --git a/src/Data/YAML/Foreign/Decode.purs b/src/Data/YAML/Foreign/Decode.purs index b3cd750..9765b30 100644 --- a/src/Data/YAML/Foreign/Decode.purs +++ b/src/Data/YAML/Foreign/Decode.purs @@ -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.Class (class IsForeign, read) -import Data.Foreign.Generic (readGeneric) -import Data.Foreign.Generic.Classes (class GenericDecode) +import Data.Foreign.Generic.Class (class GenericDecode) +import Data.Foreign.Generic (genericDecode) import Data.Foreign.Generic.Types (Options) import Data.Function.Uncurried (Fn3, runFn3) import Data.Generic.Rep (class Generic) -import Prelude (pure, (<<<), (>>=), (>=>)) +import Prelude ((>=>), (<<<), pure) 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 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. -readYAMLGeneric :: forall a rep. (Generic a rep, GenericDecode rep) => Options -> String -> F a -readYAMLGeneric opts = parseYAML >=> readGeneric opts - +readYAMLGeneric :: forall a rep. (Generic a rep) => (GenericDecode rep) => Options -> String -> F a +readYAMLGeneric opts = parseYAML >=> genericDecode opts diff --git a/src/Data/YAML/Foreign/Encode.purs b/src/Data/YAML/Foreign/Encode.purs index 9c2db48..8ff7400 100644 --- a/src/Data/YAML/Foreign/Encode.purs +++ b/src/Data/YAML/Foreign/Encode.purs @@ -1,7 +1,7 @@ module Data.YAML.Foreign.Encode where import Data.Map as M -import Data.Array (fromFoldable, toUnfoldable) +import Data.Array (toUnfoldable) import Data.Function.Uncurried (Fn4, runFn4) import Data.List (List) import Data.Maybe (Maybe, maybe) @@ -11,7 +11,7 @@ import Unsafe.Coerce (unsafeCoerce) type YObject = M.Map String YValue type YArray = Array YValue -foreign import data YAML :: * +foreign import data YAML :: Type data YValue = YObject YObject @@ -87,7 +87,7 @@ foreign import objToHash :: 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 (YString s) = unsafeCoerce s valueToYAML (YNumber n) = unsafeCoerce n @@ -99,4 +99,3 @@ foreign import toYAMLImpl :: YAML -> String printYAML :: forall a. (ToYAML a) => a -> String printYAML = toYAMLImpl <<< valueToYAML <<< toYAML - diff --git a/test/Instances.purs b/test/Instances.purs index cb261c6..363c189 100644 --- a/test/Instances.purs +++ b/test/Instances.purs @@ -1,8 +1,9 @@ module Test.Instances where -import Prelude (class Eq, class Show, bind, pure, ($)) -import Data.Foreign (ForeignError(..), fail, readString) -import Data.Foreign.Class (class IsForeign, readProp) +import Prelude (class Eq, class Show, bind, pure, ($), (=<<), (<$>), map, (<=<)) +import Data.Traversable (traverse) +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.YAML.Foreign.Encode @@ -32,23 +33,27 @@ derive instance genericMobility :: Generic Mobility instance showMobility :: Show Mobility where show = gShow instance eqMobility :: Eq Mobility where eq = gEq -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 - pure $ GeoObject { name, scale, points, mobility, coverage } +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 pointIsForeign :: IsForeign Point where - read value = do - x <- readProp "X" value - y <- readProp "Y" value +readPoint :: Foreign -> F Point +readPoint value = do +-- instance pointIsForeign :: IsForeign Point where +-- read value = do + x <- readInt =<< readProp "X" value + y <- readInt =<< readProp "Y" value pure $ Point x y -instance mobilityIsForeign :: IsForeign Mobility where - read value = do +readMobility :: Foreign -> F Mobility +readMobility value = do +-- instance mobilityIsForeign :: IsForeign Mobility where + -- read value = do mob <- readString value case mob of "Fix" -> pure Fix @@ -75,4 +80,3 @@ instance archiObjectToYAML :: ToYAML GeoObject where , "Mobility" := o.mobility , "Coverage" := o.coverage ] - diff --git a/test/Main.purs b/test/Main.purs index 035e87a..8fc9a19 100644 --- a/test/Main.purs +++ b/test/Main.purs @@ -3,15 +3,17 @@ module Test.Main where import Control.Monad.Eff (Eff) import Control.Monad.Except (runExcept) import Data.Either (Either(..)) -import Data.Foreign (F) -import Data.YAML.Foreign.Decode (readYAML) +import Data.Foreign (F, readArray) +import Data.YAML.Foreign.Decode (parseYAML) import Data.YAML.Foreign.Encode (printYAML) -import Prelude (Unit, bind, ($)) -import Test.Instances (GeoObject(..), Mobility(..), Point(..)) +import Data.Traversable (traverse) +import Prelude (Unit, bind, ($), void, discard, (>>=)) +import Test.Instances (readGeoObject, readMobility, readPoint, GeoObject(..), Mobility(..), Point(..)) import Test.Spec (describe, it) import Test.Spec.Assertions (shouldEqual) import Test.Spec.Reporter.Console (consoleReporter) import Test.Spec.Runner (RunnerEffects, run) +import Control.Monad.Eff.Console (log, CONSOLE) yamlInput :: String yamlInput = """ @@ -41,9 +43,7 @@ yamlInput = """ """ yamlOutput :: String -yamlOutput = """- Coverage: 10 - Mobility: Fix - Name: House +yamlOutput = """- Mobility: Fix Points: - X: 10 'Y': 10 @@ -51,10 +51,10 @@ yamlOutput = """- Coverage: 10 'Y': 10 - X: 5 'Y': 5 + Coverage: 10 + Name: House Scale: 9.5 -- Coverage: 10 - Mobility: Fix - Name: Tree +- Mobility: Fix Points: - X: 1 'Y': 1 @@ -62,6 +62,8 @@ yamlOutput = """- Coverage: 10 'Y': 2 - X: 0 'Y': 0 + Coverage: 10 + Name: Tree Scale: 1 """ @@ -85,13 +87,15 @@ parsedData = main :: Eff (RunnerEffects ()) Unit main = run [consoleReporter] do - describe "purescript-yaml" do + void $ describe "purescript-yaml" do describe "decode" do it "Decodes YAML" do - let decoded = (readYAML yamlInput) :: F (Array GeoObject) + let decoded = + (parseYAML yamlInput) >>= + readArray >>= + traverse readGeoObject (runExcept decoded) `shouldEqual` (Right parsedData) - describe "encode" do + void $ describe "encode" do it "Encodes YAML" $ do let encoded = printYAML parsedData encoded `shouldEqual` yamlOutput - From fab2d38e08174dfaffd2bfe4932bcf342ed94d81 Mon Sep 17 00:00:00 2001 From: Dominick Gendill Date: Sun, 7 May 2017 09:19:50 -0600 Subject: [PATCH 2/2] Code cleanup and readme update. --- README.md | 48 ++++++++++++++++++++++++--------------------- test/Instances.purs | 20 ++++++++----------- 2 files changed, 34 insertions(+), 34 deletions(-) diff --git a/README.md b/README.md index f1469f6..2ced9b3 100644 --- a/README.md +++ b/README.md @@ -18,31 +18,32 @@ data GeoObject = GeoObject ## Decode YAML -Write `IsForeign` instances for your data structures. +Write functions to read your data from foreign values. ```purescript -instance pointIsForeign :: IsForeign Point where - read value = do - x <- readProp "X" value - y <- readProp "Y" value - return $ Point x y +readPoint :: Foreign -> F Point +readPoint value = do + x <- readInt =<< readProp "X" value + y <- readInt =<< readProp "Y" value + pure $ Point x y -instance mobilityIsForeign :: IsForeign Mobility where - read value = do - mob <- readString value - case mob of - "Fix" -> return Fix - "Flex" -> return Flex - _ -> Left $ JSONError "Mobility must be either Flex or Fix" +readMobility :: Foreign -> F Mobility +readMobility value = do + mob <- readString value + case mob of + "Fix" -> pure Fix + "Flex" -> pure Flex + _ -> 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. @@ -74,7 +75,10 @@ yamlInput = """ Coverage: 10 """ -decoded = (readYAML yamlInput) :: F (Array GeoObject) +let decoded = + (parseYAML yamlInput) >>= + readArray >>= + traverse readGeoObject ``` ## Encode YAML diff --git a/test/Instances.purs b/test/Instances.purs index 363c189..208351c 100644 --- a/test/Instances.purs +++ b/test/Instances.purs @@ -44,21 +44,17 @@ readGeoObject value = do readPoint :: Foreign -> F Point readPoint value = do --- instance pointIsForeign :: IsForeign Point where --- read value = do - x <- readInt =<< readProp "X" value - y <- readInt =<< readProp "Y" value - pure $ Point x y + x <- readInt =<< readProp "X" value + y <- readInt =<< readProp "Y" value + pure $ Point x y readMobility :: Foreign -> F Mobility readMobility value = do --- instance mobilityIsForeign :: IsForeign Mobility where - -- read value = do - mob <- readString value - case mob of - "Fix" -> pure Fix - "Flex" -> pure Flex - _ -> fail $ JSONError "Mobility must be either Flex or Fix" + mob <- readString value + case mob of + "Fix" -> pure Fix + "Flex" -> pure Flex + _ -> fail $ JSONError "Mobility must be either Flex or Fix" instance pointToYAML :: ToYAML Point where toYAML (Point x y) =