The mono-repository management is handled by Yarn v3 (Berry) workspaces.
Those workspaces are defined in the root package.json
file.
By default it defines the packages
folder as the root of the
workspaces.
You may edit this configuration to add new workspaces.
Every action you take at the root of the mono-repository (e.g. yarn install
)
will have an effect at the root of the mono-repository, i.e. modify the root
package.json
file.
If you need to take an action in a specific workspace, you should use the following syntax:
yarn workspace <workspace name> <command>
You can also use the following syntax to take an action in every workspace:
yarn workspaces foreach [--all|--recursive|--since|--worktree] [--parallel] [--topological|--topological-dev] <command>
Dependencies are kept up to date using Renovate.
The core Renovate configuration is offered as
an NPM package
in the Glacier mono-repository.
Of course, you can configure/disable/extend the basic configuration at the whole
repository level
(using the renovate.json5
file at the root).
Renovate is a (better) alternative to Dependabot. It brings a lot of new features and improvements to dependency management such as:
- Renovate handles WAY more ecosystems than Dependabot
- It supports regexes to be able to be able to parse any string as a dependency version and indicate to Renovate where to look for newer versions (look here for instance)
- It supports WAY more options and extensible rules (schedules, auto-merges, custom labels, … seriously, look at all those options!)
- Is supports reusable configurations, so you can store a common configuration in a repository (or several if you want), and can re-use it across multiple repositories (to avoid duplication and ensure that everybody has the same base config)
- And a lot more 😄
This tools helps us keep our dependencies up to date and avoid security issues by automatically generating Pull Requests to update the dependencies. It even auto-merge patches update to speed up the process.
Discussions about some dependencies
This chapter serves as an annotation to package.json
and the dependencies of
the project.
The goal is not to explain the ins and outs of every dependency, but rather to serve as useful history and background to some of the choices made — and why we have some of the dependencies in the project.
About root dependencies
Some dependencies are available at the root of the mono-repository. This means that you don't have to add them in your package/application.But if you want to use those without having to add it in your own package/application, you will have to run it this way:
yarn run -T < dependency-name > [...parameters]
The dependencies that are available at the root level are:
multi-semantic-release
andsemantic-release
: to release multiple packagesis-ci
: to check if you are in a CI environmenteslint
: to lint your codeprettier
to format your code
Note that despite the fact that
typescript
is available at the root level, you have to install it as a development dependency in your own package/application.
Lodash
Using lodash-es
had a severe performance penalty on the Jest tests, since
lodash-es
uses an index.js file which contains references to all of the
operators this had to be compiled for every test file. There is also the initial
performance penalty of having to transform from ESM to CJS.
The performance of lodash-es
is significantly worse, and only becomes worse as
more tests are run.
Also, lodash-es
is not tree-shakeable, so it will always be included in the
final bundle.
That's why the preferred way of using Lodash is by importing functions through their dedicated export file, e.g.:
import isEmpty from 'lodash/fp/isEmpty'
Note that if you are using TypeScript and want to enjoying typeguards in some Lodash functions (like
isEmpty
orisString
) you should consider importing it from thefp
submodule as demonstrated above.
This repository uses its own packages to enforce conventions and best practices.
Those package offers out of the box:
- Linting based on ESLint with
@snowball-tech/eslint-snowball-config
. It automatically detects your dependencies and activate rules accordingly.
Of course, you can still configure/disable/extend the linting configuration at the whole repository level (using theeslint.config.mjs
file at the root) or at each package/app level (by adding aeslint.config.mjs
file in the appropriate folder).
See the package documentation for more information. - Formatting based on Prettier with
@snowball-tech/prettier-config
. Of course, you can still configure/disable/extend the formatting configuration at the whole repository level (using the.prettierrc.js
and.prettierignore
files at the root) or at each package/app level (by adding a.prettierrc.js
and/or a.prettierignore
file in the appropriate folder).
You can also add plugins in each of your package/app according to your needs. See the package documentation for more information.
We also have those conventions enforced:
- as a pre-commit hook, that check the format, errors and warning before committing the code (see .husky/pre-commit),
- during Continuous Integration, by running a non-changing linting step
(
yarn run -T lint
andyarn run -T format
).
Note: while these steps ensure our codebase follows our coding standards, it is recommended to enable automatic fixing in your IDE, to reduce friction during commit.
Why no `lint-staged`
We tried to integrate lint-staged
in
the repository to automatically fix linting and formatting errors and warning as
a pre-commit hooks.
However this does not behave really nicely with mono-repository and the dependency detection of our linter, making it skip some errors or reports some false positive.
So instead, we really on the basic husky
hooks to run the linter and the
formatter during pre-commit.
The .vscode/extensions.json
file contains a list of
recommended plugins for you to use. When you'll load the repository for the
first time in VSCode, your IDE will offer you to install the one you don't have.
- Install Prettier extension.
- Enable Editor: Format on Save in your Workspace settings.
- Make sure that the proper version (> 2.0) of Prettier is used.
To do so, you may have to enforce the usage of the
@snowball-tech/prettier-config
Prettier binary:
"prettier.prettierPath": "./node_modules/@snowball-tech/prettier-config/node_modules/prettier",
- Test: edit a
.md
,.js
,.tsx
or any other supported file (ex: jump multiple lines), and save your file.
- Install ESLint extension.
- Enable auto-fix on save by adding the following to
Editor: Code Actions on Save
in your workspace settings:
{
"editor.codeActionsOnSave": {
"source.fixAll.eslint": true
}
}
There is also a .vscode/settings.json
that is used to
shared the recommended setting for VSCode.
Please be carefull to not add anything specific to your personal use case in this file.
- Test: edit a
.js
,.vue
(ex: add unwanted spaces), and press ⌘ + S.
{
"eslint.validate": ["javascript", "typescript"],
"files.insertFinalNewline": true,
"files.trimFinalNewlines": true,
"files.trimTrailingWhitespace": true,
"javascript.preferences.quoteStyle": "single",
"editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true,
"editor.codeActionsOnSave": {
"source.fixAll.eslint": true
}
}