-
Notifications
You must be signed in to change notification settings - Fork 41.1k
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
Provide a possibility to extend/override configuration properties binding process #42361
Comments
There's not really enough information here for us to be able to take any action. Could you describe what optimizations you're trying to apply and what changes you'd need to do them? A sample application that shows the issue would also be very helpful. Perhaps if we profile it there might be some general optimizations we can apply for everyone. |
I'm mostly interested in extending About optimisation. I'm not sure it would work generally, but the idea is the following:
And I have thousands of such files that are bond to the map. And I don't need a property I'm using spring boot 2.7.18, but I've tested on spring boot 3.2.7 and performance is the same. |
@dmitriytsokolov thanks for the additional details. Can you please share a sample application that uses Spring Boot 3.2.x or later that demonstrates what you've described above? |
@wilkinsona sure. Should I just create an example with couple of config files, or with hundreds to emulate performance issues? |
Thanks. Something that replicates the performance issue would be ideal. Perhaps you could use a script to generate a sufficient volume of synthetic test data? |
attaching example of spring app with 1000 configs that is started up for 40-50s locally. In my real app I have more fields and more spring placeholders, so app start time/hot refresh takes 5-6 minutes One more thing: initially I though that I could cache results of resolved spring placeholders (because I have a lot of them) and get some performance benefits, so I've made the following change in Binder class locally:
but I've got approximately minus 20 seconds out of 5-6 minutes of binding time. It's not much, but still with the ability to override/extend some binding logic I could reduce it even more knowing some specifics of my serivce Anyway, will be waiting for your input, thanks a lot |
The other option I'm thinking of: |
I am experiencing similar issues with a spring-boot application that I am working on. It is a multi-tenant application where configuration for each tenant is provided via environment variables (~6000 keys in total atm). The binding of these properties takes more than 10 minutes. This happens on startup and whenever the configuration is refreshed using Spring Cloud as suggested by @dmitriytsokolov. I also suspect this to be the cause for the issue I commented on here (at least in my case). I have done some profiling for this and was able to identify the following performance hotspots:
Above evaluation was done using Spring Boot 3.1.6. However, I checked the latest Spring Boot sources and there seems to have been no relevant changes to the code sections mentioned above. |
Unfortunately the relaxed binding code continues to be a source of frustration. @Mobe91 have you tried using the |
Thank you for the helpful pointer @philwebb , I did not know about the By enabling caching globally using the Unfortunately, the workaround for (1) is still needed to yield acceptable performance due to the lack of descendent caching for Some numbers:
|
Could it be disabled by default since next major release? |
@dmitriytsokolov We're a small team with many different priorities. Rest assured, we will look at the sample when we have time to do so. That may take a little while as we work on higher priority issues for 3.4.0-RC1 that releases next week. Thank you for your patience in the meantime. |
Of course, thanks a lot @wilkinsona! |
Sure, this is what it comes down to: /**
* Mostly a copy of {@link SpringConfigurationPropertySource} to pass different {@link PropertyMapper}s to the
* constructed {@link ConfigurationPropertySource}. Sadly, at the time of writing Spring does not provide a way
* to do this differently.
*/
public class CustomConfigurationPropertySourceFactory {
private static final PropertyMapper[] DEFAULT_MAPPERS = { DefaultPropertyMapper.INSTANCE };
private static final PropertyMapper[] SYSTEM_ENVIRONMENT_MAPPERS = {
NonLegacySystemEnvironmentPropertyMapper.INSTANCE,
DefaultPropertyMapper.INSTANCE };
public static ConfigurationPropertySource from(PropertySource<?> source) {
Assert.notNull(source, "Source must not be null");
PropertyMapper[] mappers = getPropertyMappers(source);
if (isFullEnumerable(source)) {
return new SpringIterableConfigurationPropertySource((EnumerablePropertySource<?>) source,
mappers);
}
return new SpringConfigurationPropertySource(source, mappers);
}
// need to copy some more static methods here - skipped for brevity
} /**
* Copy of {@link SystemEnvironmentPropertyMapper} modified to remove the handling of legacy property naming
* <a href="https://github.com/spring-projects/spring-boot/issues/42361#issuecomment-2399941504">for performance
* reasons</a>.
*/
class NonLegacySystemEnvironmentPropertyMapper implements PropertyMapper {
// Get rid of isLegacyAncestorOf and all uses of it
} Use it: // Make a copy of the environment's PropertySources to wrap them in ConfigurationPropertySources created
// using our CustomConfigurationPropertySourceFactory.
Iterable<ConfigurationPropertySource> configurationPropertySources = env.getPropertySources().stream()
.filter(Predicate.not(ConfigurationPropertySources::isAttachedConfigurationPropertySource))
.map(CustomConfigurationPropertySourceFactory::from)
.toList();
// Enabling caching for the copied ConfigurationPropertySources at this point is important for the binding
// performance.
ConfigurationPropertyCaching.get(configurationPropertySources).enable();
Binder binder = new Binder(configurationPropertySources, new PropertySourcesPlaceholdersResolver(env)); |
I have a possible optimisation for this situation that significantly improves the startup time of the provided sample. On my machine, using Spring Boot 3.3.7, the sample starts in around 30 seconds:
With the optimisation in place it's less than 7 seconds:
The tests all pass with the optimisation in place so I am hopeful that it doesn't break anything functionally. I am, however, a little wary that it may regress the binder's performance in other scenarios so I'd like to discuss it with the rest of the team before we proceed. @dmitriytsokolov If you'd like to try things out in your real-world app, it would be very interesting to know the impact that the change has. You should be able to copy the modified |
Hi @philwebb . Thanks a lot for the feedback! I just tried 3.5.0-M3 and it works the same as 2.7/3.1 versions:
are you sure you haven't deleted/changed app config or file(s)? |
Damn, you're too quick! The SNAPSHOTs for the updates haven't yet been pushed to artifactory. When this CI job passes can you please try again (running |
haha, sure, I will try later 🤣 |
I'm afraid the changes are too risky to backport. 2.7 is also out of OSS support so you should really be thinking about upgrading anyway. |
@philwebb I just checked with the new SNAPSHOT and still see +- the same results. I'm attaching a screenshot where you can see that I'm using the recent version with your latest commit. ![]() What I'm doing wrong? I'm using Mac M1 |
Is this the same demo application? Can you attach another zip if so. I'll check it again when I'm back at my desk. |
Yes, the same (as in this message). Here it is: |
I need to add the following to make things import correctly, but after that I still see the improvement: <repositories>
<repository>
<id>spring-milestones</id>
<name>Spring Milestones</name>
<url>https://repo.spring.io/milestone</url>
<snapshots>
<enabled>false</enabled>
</snapshots>
</repository>
<repository>
<id>spring-snapshots</id>
<name>Spring Snapshots</name>
<url>https://repo.spring.io/snapshot</url>
<releases>
<enabled>false</enabled>
</releases>
</repository>
</repositories>
<pluginRepositories>
<pluginRepository>
<id>spring-milestones</id>
<name>Spring Milestones</name>
<url>https://repo.spring.io/milestone</url>
<snapshots>
<enabled>false</enabled>
</snapshots>
</pluginRepository>
<pluginRepository>
<id>spring-snapshots</id>
<name>Spring Snapshots</name>
<url>https://repo.spring.io/snapshot</url>
<releases>
<enabled>false</enabled>
</releases>
</pluginRepository>
</pluginRepositories> Can you double check that your jar has a |
Is that 67 seconds figure accurate as well? Even before these changes I can run the demo application in about 4-5 seconds. Is there anything specific to your mac that might slow the application down? Antivirus for example? |
Running the original demo zip I also see some improvements, although not as dramatic as Andy's
vs
|
yes, it has both Numbers should be accurate. Also I don't see this slowness in other apps/services. Initially, I provided a slightly different version of the app and Andy Wilkinson got the same results as me (here) |
Oh, it might be me. I don't have Lombok installed and I think it's not binding anything because of that. Let me dig a bit more and get someone else from the team to also try the demo. |
Well that's disappointing. It was Lombok missing from my IDE. Back to the drawing board :( |
I can imagine how frustrating that must be 😢 |
I've pushed 2 more updates with #44867 and #44868, with those the app (delomboked) starts in I can't find much more to optimize and I think the sheer volume of entries is probably overwhelming the @dmitriytsokolov Can you let me know if the updates make any difference for you? |
Wow! I've got ~2.5 times better performance! Amazing work, @philwebb. Thanks a lot! Answering to your previous question:
Ideally, I'd like to have a possibility to extend the binder and/or related beans, so that I can apply Andy's patch, slightly update our code, and get even better performance because this part is important for a couple of our services. |
Also, I will try to test the new version with the real app to confirm performance improvement, but it will take some time because I will need to bump spring in a couple of starters and services |
It's a bit of a tricky balancing act because we've found that sometimes when we open things up it becomes very hard to make changes to that area of the code. The binder is certainly one of the more complex areas of the codebase and can be a frequent source of bugs. I have a branch that applied Andy's patch on top of mine, but I didn't find it made a lot of difference with the sample I was running. Let me try again and run the original sample. Perhaps we can merge that one as well. |
Another option if we don't open up the classes is for you to maintain a local copy of the binder and use it directly. You should be able to register or replace the You might even be able to do that with a 2.7 application, depending on the version of Java that you run. |
@philwebb I see, thanks for the explanation. |
I think I looked into this class back then, but as far as I remember it will be a tricky "fork". As for me it's worth dedicating that effort to updating the spring. But anyway thanks for the hint, I will take a look one more time |
@philwebb I've updated Spring to the latest 3.5.0-SNAPSHOT in one of my real services and got a 30-40% performance improvement. It's not as massive as in the test app because there is complex business logic that is executed during hot refresh, but the improvement in the binding process itself is impressive. Thanks a lot for your work! |
We faced an issue where a property source for 700+- objects (consists of 100+- fields) binds to java objects slowly: around 6 minutes. Our services rely on spring cloud hot refresh and because of slow binding users are struggling. We could optimize this process for our use case, but most of the binding classes are closed for extension.
The text was updated successfully, but these errors were encountered: