-
Notifications
You must be signed in to change notification settings - Fork 2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for incremental annotation processing in Gradle #1120
Comments
I'm actually 99.9% certain Dagger does not fit. @stephanenicolas? |
Even if not directly possible, would it be possible to split the processor into two parts? e.g. one that handles |
This is what we plan to do for toothpick, an alternative DI lib.
We already have 2 different APs.
To support Incap, we will have an AP that is incremental and another one
that is not, but that can actually be optional. So we plan to only use it
for release and not debug. Hence getting a fully incremental DI lib.
Le mer. 18 avr. 2018 07 h 32, Róbert Papp (TWiStErRob) <
notifications@github.com> a écrit :
… Even if not directly possible, would it be possible to split the processor
into two parts? e.g. one that handles @Inject on constructors (to
generate _Factory, incremental) and another that handles the complex
graph building (full regeneration). I might not have deep enough insight,
but this should help a lot, if people have modules with constructor
injection, and assemble the Dagger graph at top level (:app).
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#1120 (comment)>, or mute
the thread
<https://github.com/notifications/unsubscribe-auth/ABv33dS5c_G6esV1QEck_hqXvdCv2wA9ks5tp06OgaJpZM4S_Nif>
.
|
Why would Dagger not fit, can you please elaborate on that? |
@TWiStErRob it would likely be very difficult to split the processors since they the different steps require ordering that would be hard to ensure if they're split. @stephanenicolas Can I suggest that we add a flag or some other way to test incap without requiring every project in the wild to release a new artifact? I don't think this scales well, and it forces the effort on project maintainers. It would be great if someone could file a bug saying "Hey, I've been using Dagger with Incap and it works great, can you add default support so I don't need to keep this ugly flag in my repo?" I don't love the idea of releasing an artifact with incap support without knowing that it works. |
The effort has to be on the project maintainers, because you really have to think about what your processor does. You can't turn a switch in Gradle and say "Let's just pretend this is incremental". That would lead to broken builds and lost trust with users. Afaik @hungvietnguyen is already planning to work on Dagger after he's done with DataBinding. You shouldn't rely on someone trying it out and saying "it works". There should be automated functional tests ensuring that it works. Functional testing can be done with the Gradle Tooling API. I don't yet see why anything would have to be split. I'd like to understand why Dagger wouldn't fit the isolating or aggregating category. |
Do you need the |
I agree with @oehme that testing is the way to go more than letting users experiments. The amount of complexity to rely on users. And only maintainers can have a good understanding of their AP to declare their behaviour to gradle. I don't think a release for a flag is so much of an issue, but I believe there is much more fine grained understanding of incap necessary to flag the AP properly. Though, about testing, I don't like so much the idea of integration tests, because they can slow down running tests and also use a non standard testing framework. A better solution would be to modify the google testing lib that most APs use for unit tessting to check that the right elements are passed to the filer when creating files. This seems to be doable and would provide very little change to existing tests to make sure incremental behaviors are working fine. I am not too sure about the current status of this issue, but it could be a key to faster tests of incremental annotation processing: google/compile-testing#58 |
Not sure what you mean by "non-standard" testing framework, you can use JUnit just like you would for any other test and call Gradle through the Tooling API. End-to-end tests are key to ensure that this actually works. Unit tests are of course better to test all kinds of corner cases. But if I had to choose a place to start, end-to-end always get my recommendation. |
@oehme @stephanenicolas that's fine and your opinion is totally valid, but I think it's unlikely for project authors to write integration tests for a particular build system's compilation strategy. I don't think we'd ever trust a single voice saying that it's working, but if a larger group gives it a try and can all verify, that's a strong signal. There are many build systems available, and taking away the effort from project maintainers can help features take hold. |
We can't take that effort away from the maintainer, they actually have to know what the processor is doing and ensure that complies with the constraints mentioned in the incremental annotation processing documentation. We put in a lot of work to make it as transparent as possible (i.e. no new APIs), but there are still lots of things a badly behaved processor could do to break it. Hence the opt in which only someone with deep knowledge like the maintainer can do. |
This is the highest voted issue on both our and your issue tracker. For many Android users this is the only thing standing between them and fast incremental builds. How can I help move this forward? |
I have reread the docs over and over... and I still don't understand what would make an annotation processor aggregating. All I know is what it can't do. What happens in this case? class Foo implements Bar {
@Inject Foo() {}
}
interface Bar extends RandomAccess {}
@Module
abstract class FooModule {
@Binds abstract RandomAccess from(Foo foo);
}
@Component(modules = FooModule.class)
interface TestComponent {
RandomAccess r();
} If |
Or, what about this: @Retention(SOURCE)
@interface MaybeScoped {}
@MaybeScoped
class Blue {
@Inject Blue {}
}
@Component
@MaybeScoped
interface BlueComponent {
Blue blue();
} If I add the JSR 330 states that all scopes must be Would this functionality still be considered in "aggregating"? |
I wonder if these question can be answered by setting up a Gradle functional test suite? |
@tasomaniac we care not only that they work now, because that may be by accident. We don't want to rely on aspects of the implementation that aren't in the spec. |
@ronshapiro Dagger doesn't look aggregating to me, but isolating, which is more efficient. As far as I can see, you don't create a module by searching the round environment for possible implementors of an interface (that would be aggregating), but instead the user has to put the implementation class right there in the Module's signature. So the The first example is simple:
The second example is not much harder:
Both of these examples are not specific to annotation processing, they are just about basic correctness of incremental compilation. The incremental processing documentation only mentions what you can't do in processors, because those are the only restrictions. You can do everything else and rely on a correct incremental compiler in Gradle. |
Ok, thanks for that explanation @oehme! I definitely did not get that from the docs; "These look at each annotated element in isolation" didn't give me the sense that these actions could be chained. I think the safest thing is to still probably have a separate artifact that mirrors SGTY? |
The chaining (or transitive) aspect is explained in the chapter on incremental compilation above the one about incremental annotation processing. That's because it's not specific to annotation processing, but necessary for correct incremental compilation in general. Actually you might not even need a separate artifact, because in Gradle 4.8 the registration mechanism was changed to be based on processor options. So your processor can now dynamically answer whether it is incremental or not. That means you could check a system property and tell your users to set that property if they want to try it out. |
@oehme What about the following case: based on @ronshapiro's above (#1120 (comment)), let's assume I have some Nothing yet references that field, will Gradle recompile anything besides For Dagger to work correctly, the |
@tbroyer Ron is correct. You always need to take into account that Gradle already has a fully working incremental compiler. TestComponent transitively references Foo, so it would have to be recompiled even if we take Dagger completely out of the picture. The originatingElement is just to let us know the connection between TestComponent and DaggerTestComponent, so we can delete the latter if the former is deleted. |
What is the status of the issue? Is this something the community can help with, is it still in consideration or it's actively worked on (or in the backlog)? Is there any kind of estimate here? I wanted to bump this up since Dagger is usually one of the relatively heavier annotation processors in Android projects |
I think we just need to have someone implement the switch that @oehme mentioned in #1120 (comment) (as long as gradle/gradle#5498 (comment) is also in, which I presume it is) |
Yes, you could register Dagger as a |
It should be fine to read the options before you say which you support. The latter is only there so javac can warn when an option was not used by anyone. Apart from that the PR looks 👍 |
Fixes #1120 RELNOTES=Add support for gradle's incremental annotation processing. This can be enabled by passing `-Adagger.gradle.incremental` to `javac` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211815034
For certain types (like @ContributesAndroidInjector generated types, which don't refer to their originating element) this can be critical for implementing an incremental processor. See #1120 RELNOTES=Originating elements are now reported to the `Filer` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211809425
Fixes #1120 RELNOTES=Add support for gradle's incremental annotation processing. This can be enabled by passing `-Adagger.gradle.incremental` to `javac` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211815034
For certain types (like @ContributesAndroidInjector generated types, which don't refer to their originating element) this can be critical for implementing an incremental processor. See #1120 RELNOTES=Originating elements are now reported to the `Filer` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211809425
Fixes #1120 RELNOTES=Add support for gradle's incremental annotation processing. This can be enabled by passing `-Adagger.gradle.incremental` to `javac` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211815034
Fixes #1120 RELNOTES=Add support for gradle's incremental annotation processing. This can be enabled by passing `-Adagger.gradle.incremental` to `javac` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211815034
Fixes #1120 RELNOTES=Add support for gradle's incremental annotation processing. This can be enabled by passing `-Adagger.gradle.incremental` to `javac` ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=211815034
@ronshapiro just wondering when we might expect a release with this feature? |
Thanks for the reminder. Just released 2.18. |
Is this the correct way to pass this to Dagger from Gradle?:
|
@wbervoets to support both Kotlin and Java I have it configured (with other options) as: // Enables Dagger fastInit mode, which reduces component building latency by loading less classes,
// see: https://google.github.io/dagger/compiler-options.html#fastinit-mode
def optionFastInitEnabled = '-Adagger.fastInit=enabled'
// Disables Dagger code formatting to speed builds,
// see: https://google.github.io/dagger/compiler-options.html#turning-off-code-formatting
def optionFormattingDisabled = '-Adagger.formatGeneratedSource=disabled'
// Enabled incremental annotation processing support, since Kotlin 1.3.30
// See: https://kotlinlang.org/docs/reference/kapt.html#incremental-annotation-processing-since-1330
def argumentIncremental = 'dagger.gradle.incremental'
subprojects {
pluginManager.withPlugin('kotlin-kapt') {
kapt {
javacOptions {
option(optionFastInitEnabled)
option(optionFormattingDisabled)
}
arguments {
arg(argumentIncremental, 'true')
}
}
}
afterEvaluate {
tasks.withType(JavaCompile.class) {
options.compilerArgs << optionFastInitEnabled << optionFormattingDisabled
}
if (pluginManager.hasPlugin('com.android.library') || pluginManager.hasPlugin('com.android.application')) {
android {
defaultConfig {
javaCompileOptions {
annotationProcessorOptions {
arguments[argumentIncremental] = 'true'
}
}
}
}
}
}
}
|
@eleventigers Is the aforementioned code block necessary for dagger version, 2.23.1? Currently, when scanning for dependencies that do not support incremental annotation processing without the code block, I do not get any warnings. |
It's enabled by default since Dagger 2.24. |
My understanding is that currently, projects that include annotation processors are unable to be incrementally compiled in Gradle; such projects always trigger a full rebuild. The forthcoming Gradle version 4.7 will contain improvements that will allow for incremental annotation processing.
https://docs.gradle.org/nightly/userguide/java_plugin.html#sec:incremental_annotation_processing
Presuming Dagger fits into one of the two categories of annotation processor described in the above link, adding the required meta-data to the META-INF directory would likely significantly improve build times.
The text was updated successfully, but these errors were encountered: