使用 Scalameta 提升绳索
Lifting string with Scalameta
我想注入一个方法 'toCSV' 来将 class 解析为 CSV 字符串。我的函数有两个参数:
- 序列[字符串]:header
- 序列[序列[任意]]:字段
我的宏:
class model extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case cls @ Defn.Class(_, _, _, Ctor.Primary(_, _, paramss), template) =>
def createToCSV(paramss: Seq[Seq[Term.Param]]): Defn.Def = {
val names = paramss.flatten.map(param => param.name.syntax)
val fields = paramss.map(_.map(param => Term.Name(param.name.value)))
q"""def toCSV =
CSVTools.toCSV(Seq(...$names), Seq(Seq(...$fields)))"""
}
val toCSVImpl = createToCSV(paramss)
val templateStats: Seq[Stat] = toCSVImpl +: template.stats.getOrElse(Nil)
cls.copy(templ = template.copy(stats = Some(templateStats)))
case _ =>
println(defn.structure)
abort("@model must annotate a class.")
}
}
}
变量 Term 是一个 Seq[String] 并且 quasiquotes 语法只接受一个 Term。
因此,出现以下错误:
[error] /home/xxxxx/intellij/workspace/heroes-scala-meta/macros/src/main/scala/examples/model.scala:18: type mismatch when unquoting;
[error] found : scala.collection.immutable.Seq[String]
[error] required: scala.collection.immutable.Seq[scala.collection.immutable.Seq[scala.meta.Term.Arg]]
[error] CSVTools.toCSV(Seq(...$names), Seq(Seq(...$fields)))"""
[error] ^
[error] one error found
[error] (macros/compile:compileIncremental) Compilation failed
你有解决办法吗?
提前致谢,
这可能有点 hacky,但它似乎有效:明确地将您的 names
映射到字符串文字
class model extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case cls@Defn.Class(_, _, _, Ctor.Primary(_, _, paramss), template) =>
def createToCSV(paramss: Seq[Seq[Term.Param]]): Defn.Def = {
val names: Seq[String] = paramss.flatten.map(param => param.name.syntax)
val fields = paramss.map(_.map(param => Term.Name(param.name.value)))
val nameLiterals: Seq[Lit.String] = names.map(n => q"$n".asInstanceOf[Lit.String])
q"""def toCSV =
CSVTools.toCSV(scala.collection.immutable.Seq(..$nameLiterals),
scala.collection.immutable.Seq(scala.collection.immutable.Seq(...$fields)))"""
}
val toCSVImpl = createToCSV(paramss)
val templateStats: Seq[Stat] = toCSVImpl +: template.stats.getOrElse(Nil)
cls.copy(templ = template.copy(stats = Some(templateStats)))
case _ =>
println(defn.structure)
abort("@model must annotate a class.")
}
}
}
我想注入一个方法 'toCSV' 来将 class 解析为 CSV 字符串。我的函数有两个参数:
- 序列[字符串]:header
- 序列[序列[任意]]:字段
我的宏:
class model extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case cls @ Defn.Class(_, _, _, Ctor.Primary(_, _, paramss), template) =>
def createToCSV(paramss: Seq[Seq[Term.Param]]): Defn.Def = {
val names = paramss.flatten.map(param => param.name.syntax)
val fields = paramss.map(_.map(param => Term.Name(param.name.value)))
q"""def toCSV =
CSVTools.toCSV(Seq(...$names), Seq(Seq(...$fields)))"""
}
val toCSVImpl = createToCSV(paramss)
val templateStats: Seq[Stat] = toCSVImpl +: template.stats.getOrElse(Nil)
cls.copy(templ = template.copy(stats = Some(templateStats)))
case _ =>
println(defn.structure)
abort("@model must annotate a class.")
}
}
}
变量 Term 是一个 Seq[String] 并且 quasiquotes 语法只接受一个 Term。 因此,出现以下错误:
[error] /home/xxxxx/intellij/workspace/heroes-scala-meta/macros/src/main/scala/examples/model.scala:18: type mismatch when unquoting;
[error] found : scala.collection.immutable.Seq[String]
[error] required: scala.collection.immutable.Seq[scala.collection.immutable.Seq[scala.meta.Term.Arg]]
[error] CSVTools.toCSV(Seq(...$names), Seq(Seq(...$fields)))"""
[error] ^
[error] one error found
[error] (macros/compile:compileIncremental) Compilation failed
你有解决办法吗?
提前致谢,
这可能有点 hacky,但它似乎有效:明确地将您的 names
映射到字符串文字
class model extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case cls@Defn.Class(_, _, _, Ctor.Primary(_, _, paramss), template) =>
def createToCSV(paramss: Seq[Seq[Term.Param]]): Defn.Def = {
val names: Seq[String] = paramss.flatten.map(param => param.name.syntax)
val fields = paramss.map(_.map(param => Term.Name(param.name.value)))
val nameLiterals: Seq[Lit.String] = names.map(n => q"$n".asInstanceOf[Lit.String])
q"""def toCSV =
CSVTools.toCSV(scala.collection.immutable.Seq(..$nameLiterals),
scala.collection.immutable.Seq(scala.collection.immutable.Seq(...$fields)))"""
}
val toCSVImpl = createToCSV(paramss)
val templateStats: Seq[Stat] = toCSVImpl +: template.stats.getOrElse(Nil)
cls.copy(templ = template.copy(stats = Some(templateStats)))
case _ =>
println(defn.structure)
abort("@model must annotate a class.")
}
}
}