-
Notifications
You must be signed in to change notification settings - Fork 354
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
Improve performance of XmlDependencies IsDependenciesFile #602
base: master
Are you sure you want to change the base?
Improve performance of XmlDependencies IsDependenciesFile #602
Conversation
CHANGELOG.md
Outdated
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Revert this.
Please don't make a release for us. 😆
We have an internal pipeline to release EDM4U and make sure the release is available for Github, Google APIs for Unity and every other Google products which is using EDM4U.
build.gradle
Outdated
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Revert this.
Same. Please don't make a release for us. :)
return false; | ||
internal static bool IsDependenciesFile(string filename) { | ||
bool isInEditorFolder = filename.Contains("/Editor/") || filename.Contains(@"\Editor\"); | ||
return isInEditorFolder && filename.EndsWith("Dependencies.xml"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is string parsing for 3 times (worst cast)
Also, I expect there are far more files to be under Editor
folder than a file name ended with Dependencies.xml
Here is my recommendation:
- Early out when the filename does not end with
Dependencies.xml
- Otherwise, use regex to check if the file is under
Editor
folder. sometime like".*[/\\]Editor[/\\].*"
Regex is suppose to match at O(n)
time so that should be better than parsing strings twice just to know if it is under Editor
folder.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, the early-out definitely makes it even faster.
However, as for the regex, that's the point of the PR: while profiling my large company project I noticed that the regex was taking considerable time compared to the more simple string comparisons. In theory, they should be the same, but the practical overhead appears to be quite substantial, even if pre-compiling the regex instruction.
@@ -0,0 +1,3 @@ | |||
namespace GooglePlayServices { | |||
public delegate bool FileMatchPattern(string filename); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am not entirely sure the benefit to introduce this delegate and the change in PlayServicesResolver.cs
. I do not think there would be any performance gain, is there?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree, removed it!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Kudos to you to add a Unit test for this!
Could you move the entire test to source/AndroidResolver/unit_tests
for consistency sake. Thank you!
Also, please add the following lines to build.gradle
, like right...here.
// TODO: Build and run NUnit tests for #602
Currently there are some issue with this build script to run NUnit tests from Unity 2019. While we do not want to burden you with the effort to fix it, adding a line here as a reminder would be nice.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could you move this to source/AndroidResolver/unit_tests/src
folder for consistency sake.
<Reference Include="System.Data" /> | ||
<Reference Include="System.Xml" /> | ||
<Reference Include="nunit.framework, Version=3.5.0.0, Culture=neutral, PublicKeyToken=2638cd05610744eb"> | ||
<HintPath>..\packages\NUnit.3.5.0\lib\net45\nunit.framework.dll</HintPath> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is quite specify about where you find NUnit library.
Which Unity version are you using? Is this one of the package downloaded from Unity Package Manager?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I copied this from a previous version of JarResolverTests, basically just added a nuget package reference to NUnit to run the tests in the Rider IDE outside of Unity. I'm not sure what the "packages" directory here is, but removing the HintPath doesn't prevent me from running the tests.
Please advise if there's a way to run the test that's better aligned with the rest of the project. My local project on macOS in Rider doesn't really compile all too well, and I'm used to much newer SDK versions of dotnet, so I'm not sure what the best way would be in this project. :)
@chkuang-g Hi. What's the status of this? When will it be released? |
Hi @chkuang-g, when do you estimate this fix to land? Thanks. |
@chkuang-g Hi, when will this PR land? |
This commit records the current behaviour of the method before I make changes to try and optimize it.
This simpler string manipulation checks the same logic as before but much faster in the context of a large project. If the OnPostprocessAllAssets callback is invoked multiple times with thousands of files, as can happen with non-trivial Unity projects, the regex call becomes quite expensive. For profiling data see: googlesamples#601
Internally, only Dependencies.xml files are matched, so we can use the more performant file-matching pattern here. The public API remains unchanged and only supports regex for now.
This was only added because I initially thought it would read better, but it also obscures the in and out types. Since there's no functional difference, I removed it again just to reduce the amount of custom code.
…earlier one already exists in the project
bdd2e8e
to
167a3ce
Compare
I finally got some time to apply the review feedback. I retested the changes in my large company project and the custom changes still show some good improvements in Unity. |
Why this change?
This implements the improvement for issue: #601
The
XmlDependencies.IsDependenciesFile
method is called thousands of times in large projects and is now much faster than before:The previous slowdown was causing long waiting times during content iteration in Unity projects with more than a few hundred assets.
What did I do?
I added a new test project to add a test which fixates the previous logic of file pattern matches with regex. I then replaced the regex with simpler string manipulation.
To avoid breaking API changes, I kept the public method in PlayServicesResolver the same, but internally, the new code uses a delegate which accepts a file path and returns true if it's a match. This way we can have both the Regex expressions passed from existing user code and a faster version for the internal default which only searches for the Dependencies.xml files in editor folders.
Review & Release
The review should be easiest commit-by-commit.
Feedback is greatly appreciated and please let me know which steps I need to take to make my change release-ready or if any of the maintainers will take over. Thank you! :)
Tasks
./gradlew release
./gradlew gitCreateReleaseCommit
./gradlew gitTagRelease
andgit push --tag REMOTE HEAD:master