-
Notifications
You must be signed in to change notification settings - Fork 8
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
Implement the complete solution for processing the flat configuration structure #95
Implement the complete solution for processing the flat configuration structure #95
Conversation
Additionally remove rambda and install ramda
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 for this! I haven't finished reviewing yet, I just wanted to drop some early comments about some runtime specifics that I think are important. I'm also not sure I understand what a RuleRunEnvironment
is, could you add a docstring that describes what this object represents?
packages/steiger/src/models/config/process-configuration/create-rule-instructions.test.ts
Outdated
Show resolved
Hide resolved
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 also have one high-level architectural concern for this PR. It introduces a lot of functionality and responsibility to the config
model, and it's hard for me to wrap my head around it all. I'd prefer if the config model was only responsible for reading the config and answering questions about it in a format that's convenient for Steiger itself. Then the logic of applying globs could be extracted to a feature, like it used to be before this PR.
I tried to imagine the flow of communication that would keep the config model as slim as possible, and here's what I came up with. This diagram is an imaginary dialog between Steiger (the app.ts
file) and the config
model, going from top to bottom. Underlined are the inputs that Steiger supplies to the queries/commands to the config:
However, you've done some great work here, so ideally we would come to a solution somewhere in the middle. Do you think that what I've described a good idea or would you prefer to keep the code as it is?
packages/steiger/src/models/config/prepare-rule-run-envs/mark-severities.ts
Outdated
Show resolved
Hide resolved
I agree with your concern, the config model has become too bloated with responsibilities and code in general. I will try to fix this problem tomorrow, and yes, your diagram offers an interesting idea and will come in handy, thanks! |
Hi @illright |
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 for the changes, it is indeed better. However, I still feel like the solution is more complicated that I'd like it to be, which might impact maintainability in the long run. The concrete points of complexity, in my opinion, are:
- The
apply-severity-globs-to-vfs
feature. From the perspective of Steiger's business domain, and to use the FSD understanding of a feature, this isn't a feature because it doesn't do anything meaningful to the user by itself. That's why I find this feature and its abstractionsVfsWithSeverity
andSeverityMarked*
hard to understand. - The flat representation of the VFS (record of path to VFS object). The fact that there are several representations for the same thing makes it hard for me to understand what parts of code are compatible with what other parts.
- Stepping out of Effector philosophy in certain places. I noticed that in some places in code we have
.getState()
calls and getter functions, and I believe that Effector is meant to work a bit differently, instead with derived stores and meaningful events (for example, "load this config") and effects. - Consistency of error production and handling. Currently, errors are thrown in many places, for example, when building a validation schema, when creating rule instructions, when accessing some of the getters, when parsing a config with Zod. I'm getting lost in these places because they also include human-readable error messages, which ideally I'd prefer to keep all in one place, somewhere higher up, like in
app.ts
, to keep a close look over the amount, variety, and clarity of possible error messages. There are also ways to handle errors in Effector using effects, which might also be a good solution if we isolate this functionality well.
I really wish we had a more real-time communication channel than simply back-and-forth with reviews. I can think of Telegram, Discord, or even maybe a video call to do some brainstorming. What do you think?
const config = [
...fsd.configs.recommended,
{
files: ['./src/shared/**'],
rules: {
'fsd/public-api': 'off',
},
},
{
files: ['./src/shared/**'],
rules: {
'fsd/public-api': 'error',
},
},
],
And yes, I think it would significantly increase productivity and improve coordination. In fact, I was thinking of suggesting the same. Telegram would be the most convenient option. I will send you an invite on Linkedin and share my contact.
|
And yes, about the visualizer, looks like it could be highly reusable and be useful not just for globs but also in some examples in the docs and learning materials |
Hi @illright I pushed my progress partially, could you take a look at /shared/globs when you have time? It looks like it’s aligned with your idea. P.S. I know that there are some cross-imports in /shared now. I’ll consider moving some utils inside shared/globs if they are used only there or moving |
Yes, this is quite nice! One minor thing — could you please keep index files as simple re-exports, and move logic in new adjacent files? I usually use index files to quickly get an understanding of what a certain module is capable of, and having logic there gets in the way of that a bit. |
Hi @illright A few notes on my changes:
|
packages/steiger/src/features/calculate-diagnostic-severities/calculate-final-severity.ts
Outdated
Show resolved
Hide resolved
packages/steiger/src/features/remove-global-ignores-from-vfs/remove-global-ignores-from-vfs.ts
Outdated
Show resolved
Hide resolved
I'm not done reviewing yet, thought I'd drop a few early comments |
packages/steiger/src/features/calculate-diagnostic-severities/calculate-final-severity.spec.ts
Outdated
Show resolved
Hide resolved
packages/steiger/src/features/run-rule/prepare-vfs-for-rule-run.spec.ts
Outdated
Show resolved
Hide resolved
Refactor the logic to distinguish between empty glob arrays and the absence of a glob array in a glob group
…ting with ** Allows **/ui/** => /projects/my-project/src/**/ui/** conversions as it works completely the same but keeps the codebase more clean
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.
One little bug to fix
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.
Okay, I think this is almost good to go! Could you please run https://knip.dev to get rid of some unused code and exports? The solution itself looks good :)
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.
Thank you so much for sticking through to the end! I ran some tests now, I'm quite happy with how it works, let's merge it :)
Resolves #82
Hi @illright
I know it's huge, but it didn't make any sense to partition this work, because it was kind of bringing all that we've done together.
To make it a little easier for you to review, I've created a diagram of the config parsing and application flow:
(you can find types that the processes pass to each other in
steiger/src/models/config/types.ts
)Notes
The function that I wrote for glob matching did not fully suit our needs because it removed nodes from the tree. And as I saw we needed more of a thing that would mark nodes instead of removing them because we’ll need to run globs on a tree several times e.g. for the following config. We need to disable /shared first and then re-enable it (yeah, we discussed that it’s a weird config, but it still could be specified). So, I decided that we needed to mark/remark nodes for every set of globs provided for a rule, and only when all sets are processed we can safely remove nodes that remained “off”
I installed
ramda
library. It has a lot of cool functional utils. One of the most useful functions is pipe which allows you to run operations on an array with one traversal, like filtering and mapping in the example below. There aren't many places where it's used now, but we can keep adding more in the future. It should improve the performance a bit by removing the need to re-traverse arrays with chaining array methods