File: ToFromPersistValuesSpec.hs

package info (click to toggle)
haskell-persistent 2.17.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 1,196 kB
  • sloc: haskell: 14,076; makefile: 3
file content (274 lines) | stat: -rw-r--r-- 8,728 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
{-# LANGUAGE DataKinds, ScopedTypeVariables #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE QuasiQuotes #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE UndecidableInstances #-}
--
-- DeriveAnyClass is not actually used by persistent-template
-- But a long standing bug was that if it was enabled, it was used to derive instead of GeneralizedNewtypeDeriving
-- This was fixed by using DerivingStrategies to specify newtype deriving should be used.
-- This pragma is left here as a "test" that deriving works when DeriveAnyClass is enabled.
-- See https://github.com/yesodweb/persistent/issues/578
{-# LANGUAGE DeriveAnyClass #-}

module Database.Persist.TH.ToFromPersistValuesSpec where

import TemplateTestImports

import Database.Persist.Sql.Util
import Database.Persist.Class.PersistEntity
import Data.List.NonEmpty (NonEmpty(..))
import qualified Data.List.NonEmpty as NEL

instance PersistFieldSql a => PersistFieldSql (NonEmpty a) where
    sqlType _ = SqlString

instance PersistField a => PersistField (NonEmpty a) where
    toPersistValue = toPersistValue . NEL.toList
    fromPersistValue pv = do
        xs <- fromPersistValue pv
        case xs of
            [] -> Left "PersistField: NonEmpty found unexpected Empty List"
            (l:ls) -> Right (l:|ls)

mkPersist sqlSettings [persistLowerCase|

NormalModel
    name Text
    age  Int
    deriving Eq Show

PrimaryModel
    name Text
    age Int
    Primary name age
    deriving Eq Show

IsMigrationOnly
    name Text
    age Int
    blargh Int MigrationOnly
    deriving Eq Show

HasListField
    names [Text]
    deriving Eq Show

HasNonEmptyListField
    names (NonEmpty Text)
    deriving Eq Show

HasNonEmptyListKeyField
    names (NonEmpty (Key NormalModel))
    deriving Eq Show
|]

spec :: Spec
spec = describe "{to,from}PersistValues" $ do
    let
        toPersistValues
            :: PersistEntity rec => rec -> [PersistValue]
        toPersistValues =
            map toPersistValue . toPersistFields

        subject
            :: (PersistEntity rec, Show rec, Eq rec)
            => rec
            -> [PersistValue]
            -> Spec
        subject model fields = do
            it "toPersistValues" $ do
                toPersistValues model
                    `shouldBe`
                        fields
            it "fromPersistValues" $ do
                fromPersistValues fields
                    `shouldBe`
                        Right model
    describe "NormalModel" $ do
        subject
            (NormalModel "hello" 30)
            [ PersistText "hello"
            , PersistInt64 30
            ]

    describe "PrimaryModel" $ do
        subject
            (PrimaryModel "hello" 30)
            [ PersistText "hello"
            , PersistInt64 30
            ]

    describe "IsMigrationOnly" $ do
        subject
            (IsMigrationOnly "hello" 30)
            [ PersistText "hello"
            , PersistInt64 30
            ]

    describe "mkInsertValues" $ do
        describe "NormalModel" $ do
            it "has all values" $ do
                mkInsertValues (NormalModel "hello" 30)
                    `shouldBe`
                        [ PersistText "hello"
                        , PersistInt64 30
                        ]
        describe "PrimaryModel" $ do
            it "has all values" $ do
                mkInsertValues (PrimaryModel "hello" 30)
                    `shouldBe`
                        [ PersistText "hello"
                        , PersistInt64 30
                        ]
        describe "IsMigrationOnly" $ do
            it "has all values" $ do
                mkInsertValues (IsMigrationOnly "hello" 30)
                    `shouldBe`
                        [ PersistText "hello"
                        , PersistInt64 30
                        ]
    describe "parseEntityValues" $ do
        let
            subject
                :: forall rec. (PersistEntity rec, Show rec, Eq rec)
                => [PersistValue]
                -> Entity rec
                -> Spec
            subject pvs rec =
                it "parses" $ do
                    parseEntityValues (entityDef (Proxy @rec)) pvs
                        `shouldBe`
                            Right rec
        describe "NormalModel" $ do
            subject
                [ PersistInt64 20
                , PersistText "hello"
                , PersistInt64 30
                ]
                Entity
                    { entityKey =
                        NormalModelKey 20
                    , entityVal =
                        NormalModel "hello" 30
                    }
        describe "PrimaryModel" $ do
            subject
                [ PersistText "hey"
                , PersistInt64 30
                ]
                Entity
                    { entityKey =
                        PrimaryModelKey "hey" 30
                    , entityVal =
                        PrimaryModel "hey" 30
                    }
        describe "IsMigrationOnly" $ do
            subject
                [ PersistInt64 20
                , PersistText "hello"
                , PersistInt64 30
                ]
                Entity
                    { entityKey =
                        IsMigrationOnlyKey 20
                    , entityVal =
                        IsMigrationOnly "hello" 30
                    }
    describe "entityValues" $ do
        let
            subject
                :: forall rec. (PersistEntity rec, Show rec, Eq rec)
                => [PersistValue]
                -> Entity rec
                -> Spec
            subject pvals entity = do
                it "renders as you would expect"$ do
                    entityValues entity
                        `shouldBe`
                            pvals
                it "round trips with parseEntityValues" $ do
                    parseEntityValues
                        (entityDef $ Proxy @rec)
                        (entityValues entity)
                        `shouldBe`
                            Right entity
        describe "NormalModel" $ do
            subject
                [ PersistInt64 10
                , PersistText "hello"
                , PersistInt64 20
                ]
                Entity
                    { entityKey =
                        NormalModelKey 10
                    , entityVal =
                        NormalModel "hello" 20
                    }
        describe "PrimaryModel" $ do
            subject
                [ PersistText "hello"
                , PersistInt64 20
                ]
                Entity
                    { entityKey =
                        PrimaryModelKey "hello" 20
                    , entityVal =
                        PrimaryModel "hello" 20
                    }
        describe "IsMigrationOnly" $ do
            subject
                [ PersistInt64 20
                , PersistText "hello"
                , PersistInt64 20
                ]
                Entity
                    { entityKey =
                        IsMigrationOnlyKey 20
                    , entityVal =
                        IsMigrationOnly "hello" 20
                    }

        describe "HasListField" $ do
            subject
                [ PersistInt64 10
                , PersistList [PersistText "hello"]
                ]
                Entity
                    { entityKey =
                        HasListFieldKey 10
                    , entityVal =
                        HasListField ["hello"]
                    }
        describe "HasNonEmptyListField" $ do
            subject
                [ PersistInt64 10
                , PersistList [PersistText "hello"]
                ]
                Entity
                    { entityKey =
                        HasNonEmptyListFieldKey 10
                    , entityVal =
                        HasNonEmptyListField (pure "hello")
                    }
        describe "HasNonEmptyListKeyField" $ do
            subject
                [ PersistInt64 5
                , PersistList [PersistInt64 4]
                ]
                Entity
                    { entityKey =
                        HasNonEmptyListKeyFieldKey 5
                    , entityVal =
                        HasNonEmptyListKeyField (pure (NormalModelKey 4))
                    }