-
Notifications
You must be signed in to change notification settings - Fork 4.9k
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
Build services can only be registered if the set of applied plugins is the same on all projects #17559
Comments
This works around a bug in Gradle which prevents the service provider from being registered on the native image tasks. See gradle/gradle#17559
Workaround is to use a |
This works around a bug in Gradle which prevents the service provider from being registered on the native image tasks. See gradle/gradle#17559
@gradle/configuration-cache Could you take a look at this one? |
@gradle/bt-configuration-cache I encountered this earlier today and used the workaround from @melix (thank you!) which worked well. Now encountering something that looks the same, but for Extension objects:
Edit:
Edit 2: |
Hi team! Do you plan to investigate this one? It is quite problematic as a "well designed" plugin can be facing this issue independently in a multi-project build, so testing wouldn't highlight the problem. |
@melix if you add the plugin to the root i.e. in the plugins {
id 'my.plugin' apply false
} |
There's not a single week where I'm not pinged about this bug. Can this be prioritized, please 🙏
|
As a workaround if you make your service return serializable objects you can use reflection combined with serializing/deserializing to get around class cast exceptions.
|
- Re-register the BuildService if it fails. Mark the first successfully registered BuildService as 'primary'. Only 'primary' BuildServices will log user-facing messages. - Update MigrationMessagesTest to test BuildService classloader issues. - Rename flag properties for consistency. - Add `.nowarn` and `.noWarn` variants, because I really don't want to be constantly nagged by spellcheck. - use `lazy(SYNCHRONIZED) {}` to better control logging, only log when necessary, and to avoid potential parallel issues.
- Re-write the Dokka Gradle Plugin v1 & v2 messages. - Update the opt-in flag for v2. - Create a BuildService, so the messages are only logged once per project. - re-implement how `gradle.properties` are set for Gradle tests - Create test for V1 & V2 migration messages - rename 'suppressV2Message' to 'enableV2.noWarn' * re-run tasks in some tests now that Build Cache is enabled by default * update message: V1 will be removed in 2.1.0 * update `getFlag()` to lazily evaluate the properties * Workaround buggy BuildService behaviour gradle/gradle#17559 - Re-register the BuildService if it fails. Mark the first successfully registered BuildService as 'primary'. Only 'primary' BuildServices will log user-facing messages. - Update MigrationMessagesTest to test BuildService classloader issues. - Rename flag properties for consistency. - Add `.nowarn` and `.noWarn` variants, because I really don't want to be constantly nagged by spellcheck. - use `lazy(SYNCHRONIZED) {}` to better control logging, only log when necessary, and to avoid potential parallel issues. * fix gradle properties - Avoid using `Properties()` because it needlessly escapes values, doesn't order values, and adds a timestamp.
I'm running into this issue as well, when writing a custom plugin that delegates to another one that has a This was very tricky to diagnose and identify, and we may wind up going with either a pattern of maintaining the same order for our plugins or requiring downstream consumers to supply the plugin, which defeats the purpose of mine. For those of us who are consumers of plugins with |
If sharing the service between subprojects is not mandatory and have an instance per project is acceptable, one can prefix the service name with the project name when registering: class MyPlugin: Plugin<Project> {
override fun apply(project: Project) {
val downloadClientProvider = project.gradle.sharedServices.registerIfAbsent("${project.name}_${DownloadClient.SERVICE_NAME}", DownloadClient::class.java)
project.tasks.register("download", Download::class.java) { task ->
task.downloadClient.set(downloadClientProvider)
task.usesService(downloadClientProvider)
}
}
} |
@tamaracha the whole purpose of a "shared build service" is to share state between all projects in a build. What you've proposed is functionally just a project extension. |
Actually, it would be nice to have something like this "shared build service" to share state within a single project. The docs aren't clear at all about how to do that. DSL extensions (if you mean them) are user-facing, and do not feel like the right place where to put implementation details. |
Why shouldn't there be a general DI-usable custom service concept where the scope can be selected (build-wide singletons, project-wide singletons, instances)? |
A plugin is applied to a project, so instantiating the class via constructor is enough as it will not be shared between projects. Anyway, that is out scope of this issue, as build services are created to share the state between multiple projects that are using your plugin. |
…ng plugins on all projects (#485) * gradle/gradle#17559 run-task/issues/88 * Fix issue with different versions. --------- Co-authored-by: Rollczi <[email protected]>
Are there any thoughts about disabling/deprecating per-project classloaders (which seems to be the underlying issue here)? Almost every project I come into has plugins loaded in the root build script with Plugin implementers who need a separated classloader can always use workers or create their own ClassLoader. It won't happen overnight but clear guidance would be welcome. |
That sounds like the scope of this issue: |
Expected Behavior
Given a build service:
a plugin which registers the build service:
and a task which uses the build service:
then the build service should be registered in any case.
Current Behavior
Build service registration fails in multi-project builds, if the projects using the build service have a different set of plugins applied:
The error is confusing, but hints at a classloading issue: the services seem to be cached by classloader, which is by set of applied plugins. Therefore there are 2 different instances of the
MyService
class.Context
Initially reported at graalvm/native-build-tools#70
Steps to Reproduce
Using the attached project attached project, cd into
consumer
then run:./gradlew hello
Now edit
lib/build.gradle
and comment out the jmh plugin. Now both projectscore
andlib
have the same set of plugins. Run./gradlew hello
again and see the build pass.Your Environment
Build scan URL: https://scans.gradle.com/s/fjgx3mlcahdfu
JMH plugin commented out: https://scans.gradle.com/s/nj3oohuwdyx2q
The text was updated successfully, but these errors were encountered: