-
-
Notifications
You must be signed in to change notification settings - Fork 3.4k
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
feat: Custom lint rules #4264
Comments
Hi, a bit off topic here: some of these lints/rules are already in the DCM, which I highly recommend. https://dcm.dev |
I did come across DCM shortly after posting this. Good for inspiration, but my preference would still be if the lints were first-party to the library itself for various reasons as well as open source to allow contributions. Also, all the bloc-based lints appear to be behind the Pro+ plan, which wouldn't be for everyone. Have added it to the related work section. |
nice try w promote |
Because i really like this idea, i've start a project named bloc_lint to purpose something. |
I've upload a first version of the package on pub.dev if you want to test it : https://pub.dev/packages/bloc_lint |
Hi @vmichalak 👋 |
The two options sounds great to me ! |
It’s totally up to you since you did the work if you prefer to maintain it as a separate repo/package that’s totally fine! If you do want to move it into the bloc mono repo and have time to open a PR and add me as a publisher to the package that would be great! There are some things that we’d likely want to discuss for a v0.2.0 (e.g whether we want to keep the custom_lint dependency, lint rules and their names, etc). |
This version is really a first implementation to verify the faisability. It's possible to make a better implementation for sure. I gonna check later today to add it to the main repo, if it's possible i would like to continue to work on it even if it's merge to the main monorepo 🤗 |
Yup makes sense!
Of course! I added you as a collaborator to the bloc repo and welcome all contributions! 💙 |
…bloc monorepo Implements: felangel#4264
Awesome start! I'll be looking to make some contributions this week, for sure! Also appreciate the initiative.
Yeah, my preference is definitely to have it part of the official repo to sort of signify that it's best practices as endorsed by the package itself. |
Dumping the list I have so far (will come back and elaborate on each one shortly)
Also, I'm currently doing research to determine the best approach to building a performant linter. I'm evaluating several options including:
My goal is to pick a solution that is fast, memory efficient, easy to configure, and has minimal dependencies/setup needed. Will report back when I know more 👍 |
@felangel I had a query the other day that might relate to lint rules: Should context extensions avoid being conditional? Or am I getting confused with react hooks. i.e. is this acceptable or should be avoided (and therefore linted)? build() {
if(condition) {
final foo = context.select((FooBloc bloc) => bloc.state.foo));
return Text(foo);
}
return Text('Bar');
}
|
Extensions should be okay to use conditionally unlike hooks. |
Good to know. Another potential suggestion:
To avoid things like class FrogCubit extends Cubit {
getFrog() async {
final frog = await frogService.getFrog();
emit(frog);
return frog;
}
}
// in Widget tree
try {
final frog = await context.read<FrogCubit>.getFrog();
// Use frog
} catch() {
showNotification();
} Obviously |
Good call, I updated the list 👍 |
Since the final objective stay to purpose this as a plugin to the analyzer (perhaps with analyzer plugin v2 in a long long future ?) i think it's important to always have a reference to bloc or cubit in the lint rule name. Also we have to prioritize the rules from the most important to the least to organize the implementation in the next step. |
@felangel just note dart-lang/sdk#50981 and flutter/flutter#121836, aka it seems more than 1 plugin is disabled for Dart now. Sticking with custom_lint over own plugin is therefore advised, since a lot of packages already use custom_lint for some lints. Another option is to use DCM which has a lot of Bloc related rules already, plus hundreds more for other stuff. But I understand that it might not be option for everybody. |
Thanks for the context! The problem with custom_lint is the performance in large codebases and the need to add an additional dependency. The problem with DCM is it's not free to use and only includes a subset of the lint rules we're discussing. Will update the issue when I have a better sense of how to proceed 👍 |
This is the analyzer plugins problem in general (since custom_lint is a plugin). So the only way to make it fast would be to do what we did for DCM (a standalone analysis process) and I'm not sure you want to go this road for just a dozen of rules. We are actually going to add bloc rules to the free plan with the next release, but I doubt it will affect your plans.
And I'll look into that 🙂 |
Added to OP:
❌ Bad class TightlyCoupledBloc extends Bloc {
final OtherBloc otherBloc;
late final StreamSubscription otherBlocSubscription;
TightlyCoupledBloc(this.otherBloc) {
// No matter how much you are tempted to do this, you should not do this!
// Keep reading for better alternatives!
otherBlocSubscription = otherBloc.stream.listen((state) {
add(MyEvent());
});
}
@override
Future<void> close() {
otherBlocSubscription.cancel();
return super.close();
}
} ❌ Bad class FooCubit extends Cubit {
final BarCubit otherCubit;
FooCubit({ required BarCubit otherCubit }): _otherCubit = otherCubit;
}
|
Just wanted to give an update. I have a working p.o.c which doesn't depend on custom_lint and performs very well. I'm planning to clean it up and land it in the next week and then we can start working on lint rules together. I'll also likely need some help with the IntelliJ integration (@vmichalak iirc you mentioned you'd be interested). bloc_lint_demo.mp4 |
It might be cool to add some custom lint rules to enforce consistency and/or good practices when using Bloc.
Some random suggestions and ideas (I haven't thought too long on these, some might be outright wrong, just starting discussion):
State
suffixsealed
context.select()
overBlocBuilder
unless providing abuildWhen
argumentstate
BuildContext
s as arguments to cubit methods / bloc eventscontext.read<T>()
overBlocProvider.of<T>(context)
(or vice/versa) for consistencyon<T>
s in constructorProvider<MyBloc>
orRepositoryProvider<MyBloc>
instead ofBlocProvider<MyBloc>
Rules could be opt-in but it would also be good to have a "recommended" set which are accepted as general best-practices.
Related work:
The text was updated successfully, but these errors were encountered: