platform-specific after_prepare hooks don't run after restoring
Bug Report
Problem
Include cordova-plugin-androidx-adapter in package.json, along with cordova-ios and cordova-android.
With no platforms or plugins directories, run cordova prepare.
What is expected to happen?
The after_prepare hook of cordova-plugin-androidx-adapter should be invoked at the end of the prepare, because the Android platform and the plugin have just been restored.
What does actually happen?
The after_prepare hook never fires, cordova-lib verbose output indicates that no hook scripts were found.
Information
What happens is that the ScriptsFinder tries to look up scripts in plugins for the provided list of platforms: https://github.com/apache/cordova-lib/blob/db5c8c014e917004108a440563b1aad7fc6b3140/src/hooks/scriptsFinder.js#L87
That list of platforms comes from the options of the HooksRunner: https://github.com/apache/cordova-lib/blob/db5c8c014e917004108a440563b1aad7fc6b3140/src/hooks/scriptsFinder.js#L114
The HooksRunner sets its list of platforms to a provided option, or falls back to querying for the installed platforms: https://github.com/apache/cordova-lib/blob/db5c8c014e917004108a440563b1aad7fc6b3140/src/hooks/HooksRunner.js#L68-L69
In the case of prepare, if a platform is not manually specified, we set that list of platforms to an empty array:
https://github.com/apache/cordova-lib/blob/db5c8c014e917004108a440563b1aad7fc6b3140/src/cordova/prepare.js#L32
The end result is that after restoring all the platforms and plugins, the HooksRunner still tries to find hook scripts with an empty platforms list, and the Android-specific hook never runs.
Suggested fix
HooksRunner should maybe check if the platforms array is empty and then still fallback to querying the installed platforms, rather than taking the empty input and running with it.
Version information
Was seeing this in an older project using Cordova CLI 10, but having looked at the code it seems like it's probably still an issue today.
Checklist
- [x] I searched for existing GitHub issues
- [x] I updated all Cordova tooling to most recent version
- [x] I included all the necessary information above
Maybe try the following.
const platformList = cordova_util.listPlatforms(projectRoot);
options = options || { verbose: false, platforms: platformList, options: {} };
cordova_util is already required and projectRoot is also already defined.
If you mean doing that in prepare.js, I think it will be too early. That line to set the initial options runs before any platforms are restored, so it would still end up being an empty array.
Now I'm confused because util.preProcessOptions looks like it should be adding the platforms if the array is empty... https://github.com/apache/cordova-lib/blob/db5c8c014e917004108a440563b1aad7fc6b3140/src/cordova/util.js#L279-L281
I'll try to do some more testing tomorrow to see what going on with that...
Okay, the root cause is this line: https://github.com/apache/cordova-lib/blob/db5c8c014e917004108a440563b1aad7fc6b3140/src/hooks/HooksRunner.js#L68
When it initially runs, options.platforms is an empty array, so it sets options.cordova.platforms to an empty array. Then, preProcessOptions ends up filling options.platforms with the right values, but because options.cordova.platforms is already set to a non-falsey value (an empty array), it never gets updated to reflect the updated list of platforms.
I'm not sure why the hooks runner needs a separate list of platforms and plugins nested under a cordova key, but we should probably update that assignment to update that array every time if it is empty.