4

我正在 Haskell 中构建一个 Web 应用程序。我正在使用持久库连接到 postgresql 数据库。

我正在使用标准模式定义文件系统,其中模板 Haskell 用于从模式生成类型。

share [mkPersist sqlSettings, mkMigrate "migrateAll"]
  $(persistFileWith lowerCaseSettings "schema")

我的模式文件中定义了许多数据类型,看起来像(假例子):

User json
  email Text

Post json
  owner UserId
  name  Text
  body  Text

“Post”旁边的“json”表示ToJSON/FromJSON将由框架自动生成。

我一直在为我的许多类型使用这种自动实例生成,以便我可以通过网络对它们进行序列化。

我的问题:我想为 Keys 提供一个自定义的 ToJSON 实例。例如在上面的“UserId”将是一个“关键用户”。每次都会在“PostId”作为“Key Post”等时自动为其生成这样一个实例。

当“Post”被序列化时,它会将“owner”从键转换为索引号,例如“52”。

我想将所有数据库键序列化为不同的样式。例如,不是生成数字“52”,而是生成字符串“fiftytwo”(只是一个示例)。

如果我不使用代码生成,我可以做类似的事情

instance ToJSON (Key record) where
  toJSON _ = Data.Aeson.String "placeholder"

但这需要不使用自动代码生成,因为重叠的实例错误。也许有一种方法可以告诉代码生成器不要为所有数据类型生成“ToJSON(Key Post)”等实例?

我也可以简单地为每种类型编写自定义实例声明,但这将是非常多余的。

我熟悉使用 newtypes 来拥有类型类的多个实例,但是这不太适合这种情况。

谢谢!

4

1 回答 1

0

根据 haoformayor 评论中的建议,此问题现已解决。答案是 fork 持久模板库并修改该mkKeyTypeDec部分。

于 2017-06-21T20:15:30.027 回答