2018-06-12 00:10:29 +03:00
|
|
|
{-# LANGUAGE TypeApplications, ScopedTypeVariables #-}
|
2018-06-05 19:07:47 +03:00
|
|
|
|
|
|
|
module Proto3.Roundtrip (spec) where
|
|
|
|
|
|
|
|
import SpecHelpers
|
|
|
|
|
|
|
|
import Data.Span
|
2018-06-05 19:32:24 +03:00
|
|
|
import qualified Data.ByteString.Lazy as L
|
2018-06-05 19:07:47 +03:00
|
|
|
import Data.Source
|
|
|
|
import Proto3.Suite
|
2018-06-05 19:32:24 +03:00
|
|
|
import qualified Proto3.Wire.Encode as E
|
2018-06-12 00:10:29 +03:00
|
|
|
import qualified Data.Syntax.Literal as Literal
|
|
|
|
import Data.Term (Term)
|
|
|
|
import Data.Sum
|
|
|
|
import Language.JSON.Assignment (Syntax)
|
|
|
|
import Data.Functor.Classes
|
2018-06-05 19:07:47 +03:00
|
|
|
|
|
|
|
shouldRoundtrip :: (Eq a, Show a, Message a) => a -> Expectation
|
|
|
|
shouldRoundtrip a = go a `shouldBe` Right a
|
2018-06-05 19:32:24 +03:00
|
|
|
where go = fromByteString . L.toStrict . toLazyByteString
|
2018-06-05 19:07:47 +03:00
|
|
|
|
2018-06-12 00:10:29 +03:00
|
|
|
shouldRoundtrip' :: forall f a. (Show (f a), Eq (f a), Show1 f, Eq1 f, Eq a, Show a, Message1 f, Message a) => f a -> Expectation
|
|
|
|
shouldRoundtrip' a = go a `shouldBe` Right a
|
|
|
|
where go = fromByteString1 . L.toStrict . toLazyByteString1
|
|
|
|
|
2018-06-05 19:07:47 +03:00
|
|
|
spec :: Spec
|
|
|
|
spec = parallel $ do
|
|
|
|
describe "spans" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip @Span sp
|
|
|
|
|
2018-06-12 00:10:29 +03:00
|
|
|
describe "nulls" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip' @Literal.Null @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
|
|
|
describe "text elements" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip' @Literal.TextElement @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
|
|
|
describe "floats" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip' @Literal.Float @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
|
|
|
describe "booleans" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip' @Literal.Boolean @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
2018-06-18 23:57:22 +03:00
|
|
|
describe "terms of syntax" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
2018-06-12 00:10:29 +03:00
|
|
|
describe "arrays" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip' @Literal.Array @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
|
|
|
describe "key values" $
|
|
|
|
prop "roundtrips" $
|
|
|
|
\sp -> shouldRoundtrip' @Literal.KeyValue @(Term (Sum Syntax) ()) (unListableF sp)
|
|
|
|
|
2018-06-05 19:07:47 +03:00
|
|
|
describe "blobs" $ do
|
|
|
|
it "should roundtrip given a Message instance" $ do
|
|
|
|
let bl = Blob (fromUTF8 "puts 'hi'") "example.rb" Ruby
|
|
|
|
shouldRoundtrip bl
|
2018-06-05 19:32:24 +03:00
|
|
|
|
|
|
|
describe "languages" $ do
|
2018-06-05 19:55:32 +03:00
|
|
|
-- If this test broke, it means you've probably added another 'Language'.
|
|
|
|
-- Add it to the list of languages below and everything should be good,
|
|
|
|
-- as long as you added it in a way that doesn't break prior Enum encodings.
|
2018-06-05 19:32:24 +03:00
|
|
|
it "should match up with Enum declarations" $ do
|
|
|
|
let go :: (Primitive f, MessageField f) => [f] -> [L.ByteString]
|
|
|
|
go x = E.toLazyByteString . encodePrimitive (fieldNumber 0) <$> x
|
2018-06-05 19:55:32 +03:00
|
|
|
let ints = [0..fromEnum (maxBound @Language)]
|
2018-06-05 19:32:24 +03:00
|
|
|
let langs = [Unknown, Go, Haskell, Java, JavaScript, JSON,
|
|
|
|
JSX, Markdown, Python, Ruby, TypeScript, PHP]
|
|
|
|
go ints `shouldBe` go langs
|