I am kind of blocked with the following (macro annotation)
situation. Suppose I have an annotation called @factory
which
aims to generate an apply
method for the annotated trait in the corresponding companion object. For instance, given the trait A
:
@factory
trait A {
val a1: Int
}
the expected code to be generated is the following one:
object A extends Factory[A] {
def apply(_a1: Int) = new A {
val a1 = _a1
}
}
Now suppose we have a trait B
which inherits from A
:
@factory
trait B extends A {
val b1: String
}
which is supposed to generate:
object B extends Factory[B] {
def apply(_a1: Int, _b1: String) = new B {
val a1 = _a1
val b1 = _b1
}
}
In the latter case, I need to know which are the attributes existing at A
, but I don't know how to get any information about them. While dealing with macro annotations I have only access to the B
trait AST (as a ClassDef
). Although its template
contains references to the parents (as TypeTrees
), both fields tpe
and symbol
are empty.
It would be great for me to get access to the A
AST. However, I think that's not feasible. Therefore, any symbol or type (pointing either the parent or the current type) would be good enough.
If you want to see more implementation details, I have uploaded the project to https://github.com/jesuslopez-gonzalez/cool-factory. It can generate the apply
for the local values.
Trees that go into macro annotation arguments are purposefully untyped. However running
c.typeCheck(q"(??? : <tree that represents the parent>)").tpe
will provide the missing information. Don't forget toduplicate
that tree before typechecking, becausec.typeCheck
mutates the tree in place, which might be undesireable.In case when both parent and child are declared in the same non-toplevel scope, there will be a problem of typeCheck not seeing the parent, as c.typeCheck's in macro annotations are performed in parent lexical scope, so that annotations don't get to see half-constructed scopes. Something similar has been reported here: https://github.com/aztek/scala-workflow/issues/2#issuecomment-23947943.
The decision to exclude current scope from typechecking is not a final one. This week I'll be thinking a bit more about how macro annotations should interact with enclosing scopes, and will probably change it to do what you would like it to do. I'd do the change right now, but I need to make sure there won't be any insane behaviour arising from that change.