0

我正在尝试使用 PostgreSQL 在 Persistent 中做一些看起来应该很简单的事情:给定一个Attendance记录列表,用相同的唯一键覆盖任何现有行,或者如果它们不存在则插入。以下是类型:

share [mkPersist sqlSettings, mkDeleteCascade sqlSettings, mkMigrate "migrateAllEvents"] [persistLowerCase|
Event json sql=events
    description Text
    date UTCTime
    UniqueEvent date
Attendance json
    attending Bool
    eventId EventId
    user UserId
    UniqueAttendance eventId user
|]

我没有钥匙,所以我不能使用repsert,但我认为upsert这是我需要的。对我来说,文档upsert看起来有点模糊:如果记录存在并且我将更新留空,我知道它将保留实体原样。但这并不等同于 的行为repsert,正如所声称的那样,是吗?它也没有说明不存在唯一性约束时会发生什么。

我尝试的是

runDb (mapM_ ups atts)
 where 
  -- ups rec = upsert rec [AttendanceAttending =. True] also doesn't work
  ups rec = upsert rec [AttendanceAttending =. (attendanceAttending rec)]

但这会导致我不太了解的错误:

• Illegal equational constraint BaseBackend backend ~ SqlBackend
  (Use GADTs or TypeFamilies to permit this)
• When checking the inferred type
    ups :: forall (m :: * -> *) backend.
           (BaseBackend backend ~ SqlBackend, PersistUniqueWrite backend,
            MonadIO m) =>
           Attendance -> ReaderT backend m (Entity Attendance)

知道是什么导致了这个错误吗?upsertBy也是一样的。我已经在同一个文件中导入了 Esqueleto,但是hiding ((=.)),所以它应该是纯持久的。

4

1 回答 1

1

尴尬的是,它真的就像添加一样简单{-# LANGUAGE TypeFamilies #-}。我没有尝试过,因为我误读了错误消息,并且我的其他模块都不需要该扩展。谢谢,费奥多尔!

于 2018-01-20T10:45:35.060 回答