-
Notifications
You must be signed in to change notification settings - Fork 146
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
Tree Shaking Capability #181
Comments
@JAAvander To be honest there aren't any prior considerations of any kind, I'll have to do something similar to your scenario in my day job real soon and I still have no idea where to start. it seems like you've already put some work into this so maybe you can help add Tree Shaking support to injectable? |
It would take some time until I could finish a proper pull request. A lot of work is piling up for me currently. While I've known the topic for a long time, I'm not an expert on the topic of tree shaking in dart. On the risk of going into too much detail, here is what I'm aware of: Tree shaking summed up ...In generell, if something is not imported, it clearly would not be included. Furthermore, even if a library is imported, if classes and/or constants within it are not directly depended on, their code does not need to be included. It then comes down to how the compiler can tell that something is not being depended on. Say we are depending on some large library, and we have something like this:if(constantBool){
// Code that uses the library
} else {
// Code that does not use the library
} then given that the value of if( constantString == "some string") ...
if( constantInt == 5 ) ... would make the code tree shakable as well. Comparisons between all primitives can be resolved at compile time. For classes and functions it depends on the exact situation. As soon as classes and functions are involved, the aforementioned unclarity can be seen. Some functions can be resolved at compile time, given that all their arguments are consts, even though function calls can never count as const expressions. Looking at injectable specifically, I think the problem that prevents tree shaking from happening is the use of sets to register things for specific environments. I have tried to see to what extend I would have to modify the generated code, such that the app that is the result of the compilation has dependencies from unused environments removed. This is just an example, assuming that only an environment string is given, and only the default behavior is desired. const String _prod = 'prod';
const String _dev = 'dev';
const String _test = 'test';
GetIt $initGetIt(GetIt get,
{String environment, EnvironmentFilter environmentFilter}) {
final gh = GetItHelper(get, environment, environmentFilter);
gh.factory<IChatService>(() => NetworkChatService(),
registerFor: {_prod, _test});
gh.factory<IChatService>(() => TestChatService(),
registerFor: {_dev});
return get;
} to const String _prod = 'prod';
const String _dev = 'dev';
const String _test = 'test';
GetIt $initGetIt(GetIt get,
{String environment, EnvironmentFilter environmentFilter}) {
final gh = GetItHelper(get, environment, environmentFilter);
if (environment == _prod ||
environment == _test)
gh.factory<IChatService>(() => NetworkChatService(),
registerFor: {_prod, _test});
if (_dev == environment)
gh.factory<IChatService>(() => TestChatService(),
registerFor: {_dev});
return get;
} Of course this only imitates one specific filter. I'm still thinking of solutions, but it might not be possible to have the full power of custom filters, and allow tree shaking at the same time. |
Ok, I've been thinking, and here are some ideas for enabling tree shaking:
|
I also consider filters usage is the critical factor here. I would like to add another alternative: generating separated This is related with @JAAvander's third idea, adding more config resolution regarding initialization setup. |
I also think enable tree shaking is important and here is my issue: #191 |
Hi everyone, did anyone manage to come up with a solution? If not, any pointers on where to look at is welcome, and perhaps I could write a PR for it. |
This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions |
btw, can we choose specific environment at build runner phase? not in runtime. maybe it correlate with tree shaking |
I am building a fairly large project, using
injectable
for dependency injection. Currently I thinkinjectable
is the best dependency injection package for dart.In this project, we at some point made the assumption that darts tree shaking would take care of taking out parts of code that would not be used in the production environment.
A while ago I was searching through issues on here, and found nothing about the topic, so I was suddenly uncertain of this assumption, and yesterday I took the time to validate it.
I set up a small project, plugged in some relatively large dependencies, as well as
injectable
andget_it
. After that, I built some apks while running a code-size-analysis under the following conditions:injectable
v, and unusedinjectable
, but usedAside from the first one, all of them had the same analysis result. The first was clearly smaller then the rest. Since it was a small project, this is hardly a problem, but on a larger scale a lot of technically dead code would be shipped.
I presume that the reason for why dart can not tree shake the dependencies out when they are in a different environment is the complexity of the environment filter. and
GetItHelper
.Maybe there is a simple solution that I have not thought of yet, where a custom implementation of the environment filter makes tree shaking possible again.
Are there any prior considerations about tree shaking that went into
injectable
?The text was updated successfully, but these errors were encountered: