-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
5a15e2b
commit 928ca62
Showing
9 changed files
with
216 additions
and
19 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
# CI | ||
|
||
With Nx our CI: | ||
|
||
- Runs only tasks affected by a PR. | ||
- Shares the task cache between CI and local development machines (Nx Replay). | ||
- Distributes task execution across multiple agent machines (Nx Agents). | ||
- Automatically split long e2e tasks into smaller tasks (Atomizer). | ||
- Identifies and rerun flaky tasks. | ||
- Not limited to CI, but also works for e2e tests. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,177 @@ | ||
# What the heck is monorepo? | ||
|
||
In [VCS](https://en.wikipedia.org/wiki/Version_control), a monorepo is: | ||
|
||
- A software-development strategy. | ||
- The code for a number of projects is stored in the same repository. | ||
|
||
## What does Nx for me? | ||
|
||
- A powerful open-source build system. | ||
- Provides tools and techniques for: | ||
- Enhancing developer productivity. | ||
- [Optimizing CI performance](./CI.md). | ||
- Maintaining code quality. | ||
- Run tasks in parallel based on the dependencies between them. | ||
- Caches tasks. | ||
- Works the same way with any JS or non-JS project. | ||
|
||
### Create a workspace in Nx | ||
|
||
- Workspace is a monorepo in which we are gonna have our codebase. | ||
- `npx create-nx-workspace --pm pnpm` | ||
- Will ask you a bunch of question about what kind of workspace do you need. | ||
|
||
### `Plugins` | ||
|
||
- NPM packages. | ||
- Build on top of the fundamental capabilities provided by the Nx. | ||
- Contain code generators, executors (to abstract lower-level build tooling) and automated code migrations for keeping your tools up to date. | ||
- Are usually technology specific. | ||
- Increases productive by removing any friction of integrating different tools with each other and by providing utilities to keep them up to date. | ||
- Use the `@nx/plugin` package to easily scaffold a new plugin or even just automate your local workspace. | ||
|
||
### Devkit | ||
|
||
- A set of utilities for building Nx [plugins](#plugins). | ||
|
||
### [Tasks](https://nx.dev/features/run-tasks) | ||
|
||
data:image/s3,"s3://crabby-images/e0194/e019453737a039270f86726111b627b73b10b634" alt="Run a task" | ||
|
||
- Run multiple tasks: `npx nx --run-many targetName1 targetName2` | ||
- <a href="#executorDefinitionInNx" id="executorDefinitionInNx">#</a> Each task has what we call it "executor": it is the tool or script used to run that target. | ||
- Nx picks up tasks from: | ||
|
||
1. `package.json` scripts. | ||
2. Defined inside `project.json`. | ||
|
||
And their config: | ||
|
||
- It infer/deduce them from configuration files (e.g. `jest.config.ts`). | ||
|
||
- Run tasks affected by your PR: `npx nx affected -t test`. | ||
|
||
data:image/s3,"s3://crabby-images/92ce2/92ce25b4bb6802665a71e8f4c1c866bedc74b45f" alt="nx.json" | ||
|
||
- <table> | ||
<caption> | ||
<strong>targetDefaults</strong> | ||
<dl> | ||
<dt>Here we assume Same Target Name, Same Executor.</dt> | ||
<dd> | ||
Multiple targets (tasks like <code>build</code>, <code>test</code>) with the same name across different projects, they all use the same executor. | ||
</dd> | ||
</dl> | ||
<p>What did we get out of this?</p> | ||
<ul> | ||
<li>DRY principle.</li> | ||
<li>Ease of maintenance and development.</li> | ||
<li>Easer time to reason about each project's configuration.</li> | ||
</ul> | ||
</caption> | ||
<thead> | ||
<tr> | ||
<th>Folder structure</th> | ||
<th>Before operation <code>libs/lib1/project.json</code></th> | ||
<th><code>nx.json</code></th> | ||
<th>After operation <code>libs/lib1/project.json</code></th> | ||
</tr> | ||
<thead> | ||
<tbody> | ||
<tr> | ||
<td><img width="340" src="./project-directory-structure.png" alt="Directory structure" /></td> | ||
<td> | ||
https://github.com/kasir-barati/you-say/blob/182d7607249bd3371cf8f39e0290dccd94ecdbdd/.github/docs/monorepo/libs-lib1-project.json#L1-L40 | ||
</td> | ||
<td> | ||
https://github.com/kasir-barati/you-say/blob/182d7607249bd3371cf8f39e0290dccd94ecdbdd/.github/docs/monorepo/root-nx-for-libs.json#L1-L35 | ||
</td> | ||
<td> | ||
https://github.com/kasir-barati/you-say/blob/91a2759180482b5e961bc01555c09c02bea09c9d/.github/docs/monorepo/reduced-project.json#L1-L16 | ||
</td> | ||
</tr> | ||
</tbody> | ||
</table> | ||
|
||
- Two `build` targets with different executors: | ||
|
||
Do **NOT** use `build` as key inside the `targetDefaults`. But instead use use executor name as key: | ||
|
||
<table> | ||
<thead> | ||
<tr> | ||
<th><code>nx.json</code></th> | ||
<th><code>project.json</code></th> | ||
</tr> | ||
</thead> | ||
<tbody> | ||
<tr> | ||
<td> | ||
https://github.com/kasir-barati/you-say/blob/0324448d5446cd2a7e8bd8736e8ea1a9ab07182d/.github/docs/monorepo/executor-as-key.js#L15-L33 | ||
</td> | ||
<td> | ||
https://github.com/kasir-barati/you-say/blob/0324448d5446cd2a7e8bd8736e8ea1a9ab07182d/.github/docs/monorepo/executor-as-key.js#L2-L12 | ||
</td> | ||
</tr> | ||
</tbody> | ||
</table> | ||
|
||
- If you have to make exceptions for most of the projects in your repository, then that setting probably should not be a default. | ||
|
||
data:image/s3,"s3://crabby-images/a5807/a5807e4082433dc4bc6f13bc3fc526aec7eacc55" alt="project.json" | ||
|
||
### Caching | ||
|
||
- Battle-tested computation caching system. | ||
- Code is never rebuilt twice. | ||
- Can be don in either: | ||
|
||
| `libs/lib1/project.json` | `nx.json` | | ||
| -------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------- | | ||
| https://github.com/kasir-barati/you-say/blob/aea61a68e1d9898b29dd6cb519867582522e27e5/.github/docs/monorepo/cache.js#L2-L8 | https://github.com/kasir-barati/you-say/blob/aea61a68e1d9898b29dd6cb519867582522e27e5/.github/docs/monorepo/cache.js#L11-L17 | | ||
|
||
- The operation that you wanna cache need to be pure, no side effect is allowed. E.g. you cannot cache your e2e tests since something might have changed on your backend. | ||
- `inputs`: defines what gets included as part of the calculated hash (e.g. files, environment variables, etc.). | ||
|
||
Exclude all `*.md` files from the cache so that whenever we change the any markdown file, it does not invalidate the build cache. | ||
|
||
| `libs/lib1/project.json` | `nx.json` | | ||
| --------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------- | | ||
| https://github.com/kasir-barati/you-say/blob/5a15e2bd131f8bfbb79e022a8b08ae5b5c2dd62d/.github/docs/monorepo/cache.js#L9-L16 | https://github.com/kasir-barati/you-say/blob/5a15e2bd131f8bfbb79e022a8b08ae5b5c2dd62d/.github/docs/monorepo/cache.js#L26-L33 | | ||
|
||
- `outputs`: defines where the result of a task is placed after task execution (e.g. where is our coverage report). | ||
|
||
Define output locations if they differ from the usual `dist` or `build` directory since Nx by default recognizes those. | ||
|
||
> [!NOTE] | ||
> | ||
> Learn about what are `projectName`, and they other place holders in [glossary](./glossary.md#projectRootDefinitionInNx). | ||
- Nx plugin automatically configure caching based on your configuration files and also the executor. | ||
|
||
data:image/s3,"s3://crabby-images/dc435/dc4352c2f77effec04e0596cd70401277b09af67" alt="Nx plugins can automatically infer tasks and configure caching based on your underlying tooling configuration files." | ||
|
||
### Important notes about `Nx` | ||
|
||
1. Make sure to install the `@nx/whatever` version that matches the version of `nx` in your repository. If the version numbers get out of sync, you can encounter some difficult to debug errors. You can [fix Nx version mismatches with this recipe](https://nx.dev/recipes/tips-n-tricks/keep-nx-versions-in-sync). | ||
2. Nx plugins lift the burden of things like scaffolding a new app (e.g. NestJS, NextJS), testing, building, etc | ||
3. You can use `nx graph` and `nx show projects` to see what is going on in your monorepo. | ||
|
||
### Upgrade dependencies -- `nx migrate latest` | ||
|
||
This command is **NOT** gonna upgrade your devDeps and deps to the latest version. It is only upgrading `nx` and its belongings. So we can use other approaches to upgrade our dependencies: | ||
|
||
```cmd | ||
npm outdated | ||
npm update | ||
``` | ||
|
||
But most of the times `npm update` will NOT do the trick. So in those cases we need to rely on tools such as [npm-check-updates](https://www.npmjs.com/package/npm-check-updates) | ||
|
||
```cmd | ||
npm install -g npm-check-updates | ||
ncu | ||
``` | ||
|
||
if do not need `migrations.json` (scenarios such as applying same migration in other branches) remove it. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
# Glossary | ||
|
||
<dl> | ||
<dt id="projectRootDefinitionInNx"> | ||
<a href="#projectRootDefinitionInNx">#</a> | ||
<code>projectRoot</code> | ||
</dt> | ||
<dd> | ||
A key word that is replaced by the path to the current project's root directory. | ||
</dd> | ||
<dd>Where <code>project.json</code> is located.</dd> | ||
<dd> | ||
In standalone repos (where you have only one project in your repo) your <code>projectRoot</code> and <code>workspaceRoot</code> points to the same location. | ||
</dd> | ||
<dt id="projectNameDefinitionInNx"> | ||
<a href="#projectNameDefinitionInNx">#</a> | ||
<code>projectName</code> | ||
</dt> | ||
<dd>The name of your project</dd> | ||
<dt id="workspaceRoot"> | ||
<a href="#workspaceRootDefinitionInNx">#</a> | ||
<code>workspaceRoot</code> | ||
</dt> | ||
<dd>A key word that is replaced by the root path of your workspace.</dd> | ||
<dd>The entire repo is your Nx workspace.</dd> | ||
<dd>Where <code>nx.json</code> is located.</dd> | ||
</dl> |
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.