We are happy that you are considering contribution to GoodData.UI SDK. Before proceeding, please familiarize yourself with the technical aspects of the contribution. This document gives overview how to build and test the SDK and perform other typical tasks related to contributing.
The development process is tested on macOS and Linux and uses bash for some of the scripting. This means that some parts of the development process will not work on Windows. You can use Windows Subsystem for Linux but note that this mode of development is not tested and not supported by our teams.
Please check out How to setup IntelliJ guide to find out about recommended settings for IntelliJ IDEA or How to setup VS Code guide to find out about recommended settings for VS Code.
Rush is an opinionated monorepo management tool that comes with batteries included. We strongly encourage you to read the official documentation.
Long story short, here are facts and commands you need to know:
-
lock file (shrinkwrap file) is stored in pnpm-lock.yaml.
-
Rush maintains internal registry of packages in common/temp; all package dependencies are symlinked; projects within the monorepo are also symlinked.
-
the common directory contains Rush and project configuration including:
- lockfile
- git hooks scripts which will be automatically installed by Rush
- custom scripts that can be integrated into
-
rush install
- installs dependencies according to the lock file. -
rush update
- conservatively install dependencies describes in package.json files, update lockfile.This will only install / update dependencies that you have modified.
-
rush update --full
- install and update all dependencies according to the semver, update lockfile.This is a more aggressive mode in which Rush will install and use latest applicable versions of packages, possibly resulting in a massive change to the lockfile. There is no need to run this as part of typical feature development.
-
rush build
- builds all the projects' ESM and CommonJS builds, in the right order, possibly skipping those that have no changes since the last build. -
rush link
andrush link --force
- builds or rebuilds symlinks between projects in the repository.This is typically not needed as Rush will re-link during update. Use in case you encounter strange dependency errors during build.
-
rush add
- adds a new dependency to a project.
Note: Rush by default tries to use all cores available on the machine. It is possible to override this using the
--parallelism
option on the CLI or using theRUSH_PARALLELISM
environment variable.
On top of Rush built-in commands, we have added our own custom commands (see command-line.json):
Command | Description |
---|---|
rush audit |
Performs security audit of all packages listed in global lock file |
rush clean |
Cleans up artifacts created by build and tests. Cleans Jest caches. Full 'rebuild' required after cleaning. |
rush validate |
Validates code in all projects. |
rush validate-ci |
Validates code in all projects in CI mode. |
rush test-once |
Tests code in all projects. |
rush test-ci |
Tests code in all projects in CI mode with coverage reporting. |
rush dep-cruiser |
Runs dependency-cruiser in all projects. |
rush dep-cruiser-ci |
Runs dependency-cruiser in CI mode in all projects. This creates HTML reports |
rush prettier-check |
Verifies code formatting in all projects. |
rush prettier-write |
Formats code in all projects. |
rush populate-ref |
Makes projects populate reference workspace with recording definitions. |
You must use Rush to add a new dependency to a project and use the make-consistent
parameter. This parameter
will make Rush update all other projects that already use the package to also use the desired version.
Navigate to the subproject and invoke:
rush add -p <package>@^<version> --make-consistent
to add a new production dependency. Add the --dev
flag to add the package as dev dependency instead.
With rare exceptions such as typescript package, all our dependencies use caret. Dependencies must be kept consistent across subprojects.
Note: never add new packages by running package manager directly (e.g.
yarn add ...
). This will break the node_modules.
Remove the dependencies from package.json and then run rush update
.
To build everything, run rush build
; this will trigger parallel execution of build scripts defined in each subproject's
package.json
, starting them in the correct order - honoring the inter-package dependencies.
Rush build is incremental and will only build projects which have changed since the last build. This is
safe and preferred way to build. In rare cases such as large refactors or refactors of SCSS styles you should
perform clean & rebuild: rush clean && rush rebuild
To build a single subproject with its dependencies:
rush build -t @gooddata/sdk-ui
If you want to build just single subproject, you can navigate to the subproject directory and invoke build using
rushx
:
rushx build
Hint: starting package.json scripts using npm/yarn/pnpm also works and there is nothing wrong with it.
Hint: if you find that the
rush build
makes your system unresponsive, you can try tweaking (lowering) the parallelism, either by using the-p
parameter ofrush build
, or by setting theRUSH_PARALLELISM
environment variable.
You should write imperative commit messages with well-defined title and body (for more context, please see this blog post).
To give some examples of commit messages we'd like to see from this repo:
Clarify attribute area sort
- the attribute sort aggregation switches the attribute sort into
area sort
- having a 'flag' or 'aggregation' on a newAttributeSort factory seemed
confusing while I tried to explain this in docs
- it is more clear if the area sort has separate factory - even if it
creates the same type.
- but this way, the intent is cleaner
- also, the aggregation: boolean was not good. this is indeed a function that may one day support more.
JIRA: RAIL-2175
Improve dashboard updating logic
The order of operations matters here:
1. delete any alerts for to-be-deleted widgets
2. update dashboard
3. delete to-be-deleted widgets
If we do it in any other order, the backend will complain
that we are removing an object referenced by something else.
JIRA: RAIL-2537
Note the imperative, present tense voice in the title. A good rule of thumb is to imagine the commit title as the ending to "If applied, this commit will [commit title]".
Also note how to add JIRA ticket reference: using a "JIRA: [ticket-id]" to the end of the commit body. We do not include the JIRA ticket in the first line because it wastes valuable space (the first line is visible in git log etc.) and the name of the change is more immediately useful.
The PR title should follow this general template:
RELATED: RAIL-1234 add [feature name] to [package]
This is to keep the PR title structure in line with our other frontend repositories.
In the PR body, please follow the checklist and really try to explain the changes happening in the PR (for single commit PR this will be pre-filled from your well described commit already). All the communication about the PR should happen in the PR via comments so that the process is transparent and traceable.
Run rush change
and follow the instructions. Run this after any significant block of work (one or more commits) that you want to mention in the changelog. Think of this as a condensed commit message. You should probably run this after you have your PR ready for review and you have squashed your commits. Run the rush change
and amend your final commit with the results. This will create files in common/changes
. Commit these files, they will be used during release to generate CHANGELOG automatically.
As for how to write the changes, please refer to the Rush recommendations. Moreover, if your changes are only in one package, please prefix the message with the name of the package like:
sdk-ui-pivot: Add automatic column resizing
There will probably be a check step in the future that will make sure you ran the rush change
command.
For detailed description of the release process, please see Release process.
Please go ahead and familiarize yourself with the GoodData.UI SDK architecture, layering and packaging design which are all documented in our Developer's Guide.
If you are still in doubt, ask.
Before proceeding, first ask yourself:
- Why does the contribution not fit into an existing package? (see Developer's Guide)
- Which of the existing packages is closest fit for the contribution? Why does the contribution not fit in there? (see Developer's Guide)
Once you have answers to these questions and still feel a new package is needed, get in touch for approval before proceeding.
After that, we have a couple of skeleton projects to bootstrap development of new SDK packages. See skel directory for more information. Bear in mind the naming conventions described in the developer's guide.
There might be situations when you want to quickly test changes made in the SDK packages in your own app. There are two ways you can do it: using links or using rsync
.
Let's try sdk-ui
as an example (this guide is applicable to other packages – sdk-model
, sdk-backend-spi
, etc. as well). Use the following steps (we assume your app uses yarn
as a dependency manager, you should be able to replace yarn
by pnpm
or npm
depending on your app's setup and achieve the same results):
- Run
yarn link
in thesdk-ui
folder. - Run
yarn link "@gooddata/sdk-ui"
in your app root folder.
You only have to do the linking once. After you linked the package, you can run the compilation:
- Run
rushx dev
in thesdk-ui
folder. This will start the compilation in watch mode and will rebuildsdk-ui
on every change - Run your app (you can use watch mode if applicable). You will see the up-to-date version of
sdk-ui
in your app and it will refresh as long aspnpm run dev
is running.
Alternatively, if you want to avoid the potential problems with links, you can use the rsync
utility to copy the dist files of your version of the SDK packages to your app's node_modules
. For example:
cd gooddata-ui-sdk
rush build
rsync -rptgD --no-links --include="/*/esm/*" ./libs/ ~/your-app/node_modules/@gooddata/
This will make sure that the SDK8 files in your app are from your local SDK8 version. To revert the changes, run yarn install --force
or equivalent in your app.
Hint: it is better to run the first
rsync
before starting your application, as the first run copies a lot of files and might make your app dev-server recompile multiple times unnecessarily. So the typical flow would be
- run
rsync
as described above- after it finishes, run your app's dev server
- change the SDK code and run
rush build -t [the package you make changes in]
and then runrsync
without stopping your app
Every pull-request can be merged by adding merge
label. This triggers test scripts and once they pass, the pull-request is automatically merged. All related scripts run in docker, see ./common/scripts/ci/
for individual scripts being run on jenkins slaves.
You can run all checks which are executed in CI pipelines locally:
rush rebuild
rush validate-ci
rush test-ci
cd libs/sdk-ui-tests; npm run backstop-test
Note: the pipelines WILL NOT run automatically for contributors outside of the GoodData organization. Please ping maintainers to trigger the pipeline execution.
This might be caused by corrupted symlinks. Try running rush link --force
to recreate them. After that, everything should work fine.
rush install
gives me "Error: Cannot find module '~/.rush/node-v10.16.3/rush-5.14.0/node_modules/@microsoft/rush-lib/lib/index'"
Try to run the following sequence of commands npm un -g @microsoft/rush pnpm && rm -rf ~/.rush && npm i -g @microsoft/rush pnpm
to fix the issue.