Skip to content

Latest commit

 

History

History
580 lines (355 loc) · 24.8 KB

CONTRIBUTING.md

File metadata and controls

580 lines (355 loc) · 24.8 KB

Contributing

Hey 👋 Glad you're here 😊 Contributions are very welcome ❤️

Tip

GitHub provides a full table of contents. Right now it can be enabled clicking the list icon button.

Code of conduct

To keep the project open and inclusive, please read and follow our Code of Conduct (TL;DR: don't be an a**hole)

Submission Guidelines

Discussions

If you want to discuss an idea, improvement, feature request or something related to one of the libraries hosted in this repository, please do by visiting the GitHub Discussions page. And remember to follow the code of conduct

  • Do use GitHub Discussions to talk about ideas, enhancement, improvements, feature requests or anything else
  • Do not use GitHub Issues to suggest new features or enhancements.

Issues

Before submitting an issue, please do search the GitHub Issues tracker first. An issue may exist already.

Issues should be about bugs or existent features that do not work as expected. For other topics, do use discussions.

Provide a minimal reproduction where the wrong behaviour is observed to speed up the reproduction and confirmation of bugs.

If you know the bug and can provide a bug fix, your pull request is more than welcome ❤️

  • Do search first existing GitHub Issues before posting a new one
  • Do provide a minimal reproduction example to ease bug reproduction and confirmation.
  • ❤️ Do open a pull request with a bug fix, if you are able to provide one
  • Do not use GitHub Issues to talk about non-issues. Use GitHub Discussions instead.

Pull requests

Happy to see you want to provide some code or docs 🥰

Search first there is no open (or closed) pull request already in GitHub Pull Requests page to avoid duplicating efforts. If there's none, feel free to open yours.

Whilst developing the code, remember to follow conventions stated in developing section. Don't worry, there aren't any quirky ones apart from the regular ones you'd expect in a regular project.

Add a little description describing your change. Then, open a pull request.

You can check GitHub's documentation about creating pull requests for more information. You'll probably need to fork the repository first.

A reviewer will check your code and provide feedback. If everything's ready, they may merge your pull request. That's it 🚀

You can then proceed to remove your branch and repository fork if you want.

  • ❤️ Do contribute with a pull request. They're very welcome 🥰
  • Do search GitHub Pull Requests to find an existing pull request first. Even if it's in closed state.
  • Do describe your changes with a description

Developing

You can go and just start writing code. The CI/CD will tell you if you fail to comply with a required code style / convention. However, if you want to know them in advance:

  • Follow conventions when developing some code and committing it with git
  • See how to perform usual tasks such as building, testing, ...
  • Take a look at tools recommended to help you develop
  • Something strange happens? Quirks may help

Conventions

Commit messages

Conventional Commits convention is used to format commit messages. This means commit messages must be in the shape of:

<type>(optional scope): <description>

[optional body]

[optional footer(s)]

Pretty similar to Angular's commit message format

If you don't provide a commit message with proper conventions, CI/CD will fail.

To help you start writing commit messages if you have never used the convention, you can run

pnpm run commit

To create your first commit message. Type a wrong type and the tool will list you the types available.

There is no formal convention on scopes. Though for ngx-meta, metadata module name in kebab case is used. For instance, to add an Open Graph related feature, commit message could be: feat(open-graph): add article vertical

See linting section about commit messages to know how to ensure commit conventions are followed.

TSDoc comments

Every exported member must be documented using TSDoc. Specifically, API Extractor subset of TSDoc

At least, it must provide the visibility of the exported member. Mainly either @alpha, @beta, @internal or @public

Check out API Report section for more information

Testing

Do add an E2E test when adding some metadata to ensure it's add and removed as expected.

Avoid unit tests for setting a specific metadata if the logic to set the metadata is straight forward (ie: no ifs or loops).

Documentation

Remember to add some docs to the comparison page when adding new metadata.

If adding a whole metadata module, add it to the built-in modules section too.

CI/CD jobs

To easily reproduce locally CI/CD jobs, commands run by CI/CD which are different from commands ran locally are stored in .ci/Makefile file.

For instance, you can run make install to run the same command used in CI/CD for installing.

Release and versioning

Every release is identified by a version number that follows Semantic Versioning 2.0 specification

In order to automatically detect if a new release is needed and what kind of version number bump is needed (major, minor or patch), commit messages since last release are analyzed. That's why commit messages follow a convention. So that they can be analyzed later to automate the version bump and release process.

Every push to main branch analyzes all commits since latest release. Based on commit messages there's enough information to decide if a new release is needed. If a new version is needed the release process associates the commit with a tag, creates a release in GitHub with notes generated from commit messages and publishes a new package version to NPM.

See release task for more information about how to operate releases.

Tasks

Package management

As you can see in package.json's packageManager field, pnpm is used to manage packages. corepack is used to manage the package manager 🤯

To install packages needed to develop, do

pnpm install

Build

After installing dependencies as seen in package management, just run

pnpm run build

To build all libraries

Test

Unit tests

Unit tests for libraries are run with Angular's default test runner Karma and default framework Jasmine

Run them with

pnpm run test:unit:libs

Unit tests for libraries' schematics are run with Jest. They use a different tool given they need to be run by an engine like Node.js, not a browser environment.

Run them with

pnpm run test:unit:schematics

Or to run them all in parallel

pnpm run test:unit

Tip

There are also WebStorm/JetBrains IDEs run configurations to run unit tests and report results inside the IDE:

  • Unit tests: libraries
  • Unit tests: schematics
With coverage

To enable coverage reporting, use the previous run scripts, but with :coverage suffix.

For instance:

pnpm run test:unit:libs:coverage
pnpm run test:unit:schematics:coverage
pnpm run test:unit:coverage # both in parallel

Reports will be generated in JSON, lcov and HTML format. JSON report file name is unit-test.json for libraries and jest/coverage-final.json for Jest schematic tests.

See coverage section for more details.

Note

You can also use WebStorm's Run with coverage option. However, seems that it will use a custom Karma config. So the coverage report configuration may be different. At the moment of writing this document, for instance, the report directory and format is not the same.

E2E tests

Refer to ngx-meta's E2E infra for more information

Coverage

Tests are configured to output coverage reports to the coverage/<projectName> directory. See each kind of tests to see which formats will be output there.

Codecov is used to keep track of code coverage over time. It is configured in CI/CD to ensure each PR introduces tests for new code introduced and to maintain a good overall project coverage. All coverage results from different tests (unit, E2E) are uploaded there to have a holistic code coverage view.

In local, beware that reports may overwrite each other. Except for JSON reports. For instance if running unit tests coverage and then E2E tests coverage, the lcov and HTML report will take int account E2E tests only.

Global coverage

In order to get the coverage for all kinds of tests, nyc CLI can be used to merge several JSON code coverage reports.

First, run all tests with coverage you want to merge to emit those JSON code coverage reports.

Then, you can use the coverage:report:all run script in package.json to merge reports.

pnpm run coverage:report:all

The global coverage report will be in lcov(.info) and HTML format.

Format

Prettier is used for consistent code formatting.

You can run

pnpm run format

To format all files in the repository. Or if just wanting to check if files are properly formatted, run:

pnpm run format:check

Git hooks are provided to automatically ensure code is formatted before committing it.

Lint

To lint everything, run

pnpm run lint

Commit messages

commitlint is used to enforce the commit messages conventions.

In order to lint the last commit message tracked in git (.git/COMMIT_EDITMSG), run

pnpm run lint:commit-message

To lint the last commit message in git history:

pnpm run commitlint-last

Git hooks are provided to automatically ensure your message follows the convention when committing.

Most code: TS, JS, HTML...

ESLint is used to lint the project amongst Angular ESLint. As recommended by Angular when doing ng lint.

To lint the whole project:

pnpm run lint:code

Git hooks are provided to automatically lint changed files when committing. Thanks to lint-staged

GitHub Actions code (optional)

actionlint provides linting for GitHub Actions workflows & actions. Checkout the repository's instructions on how to install the tool in your machine.

Can be run by:

pnpm run lint:gh-actions

If the command is not installed, linting will be skipped and a warning will be shown instead. This is to allow operating the repository with as minimal tools as possible. CI/CD will run the linter anyway.

It is also included to run as part of Git hooks.

Note

actionlint uses shellcheck to lint shell scripts inside run blocks in GitHub Actions workflows. In CI/CD the Docker image includes shellcheck and pyflakes so those checks will run. If locally you don't have those commands available, actionlint won't run those checks. This means different results may be obtained from running in CI/CD vs running locally. Install those (mainly shellcheck) to have the same behaviour in CI/CD and locally.

API Report

API Extractor generates a report of the exposed API of a package in order to keep track of public API signatures. In order to for instance help detecting breaking changes.

The report must be generated every time the API surface of a package changes.

To do so, first compile the project using tsc:

pnpm run ngx-meta:tsc

Then, run API Extractor locally

pnpm run ngx-meta:api-extractor:local

It will generate the API Report file for you to commit it.

CI/CD will ensure the API Report is up-to-date with current code.

API Reference docs

API Extractor also generates the base to generate API Reference docs. Run it first as explained in previous step before generating API Reference docs.

To generate docs, API Documenter sibling package is used. Run

pnpm run ngx-meta:api-documenter

To generate API Reference docs, which will end up in the documentation directory.

Docs

Check out ngx-meta docs README.md to operate docs project

Release

Semantic Release is used to create releases by analyzing commit messages' semantic information. Here you'll find how to simulate a release.

Working on a branch

Semantic Release behaves differently depending on the checked out branch.

If you're working on a branch and want to force a release as if that branch was the main branch, you can export an environment variable:

export LOCAL_SEMANTIC_RELEASE_BRANCH=true

This way, configuration will use the local branch as if it was the main branch. Keep reading to see the commands to run to actually simulate a release.

[!NOTE] > Branch needs to be pushed. Otherwise, it does not exist for semantic release. Same for commits on that branch. If not pushed, it's like they don't exist. Unless you use . as repository URL as can be seen below in dry run: publish. Though beware of that.

Dry run: minimal

To have a minimal idea of what would happen when running the release process, you can run Semantic Release in dry run mode. This includes: authentication checks, commit analysis to see if a new release is needed and the release notes for it in case it's needed.

Warning

Push permissions to the repository are needed to pass the git auth check. That's a hardcoded check. So there's unfortunately no way to bypass this If you don't have permissions, check out the next dry run approach

First, some env vars need to be defined. Check out sample release env file to see which ones with current configuration.

Tip

You can copy the sample file into a .env file and set the credentials there. The file is ignored in git, won't be commited by accident. Then, use a tool like ohmyzsh's dotenv plugin or use direnv to automatically load the env vars into your shell.

Tip

To simulate a release, you can (and indeed should) use read-only tokens. Check out the sample release env file for more details

With env vars set, run

# 👇 --dry-run is the default mode when not on CI, but just to be explicit & moar sure
pnpm semantic-release --dry-run

This is useful to know the version bumped and release notes generated, but prepare and publish NPM steps are skipped. Keep reading to know how to dry run them.

"Dry run": publish

What if you want to simulate what would be published to the NPM registry? Or how the generated release notes would look in GitHub? Or how the generated CHANGELOG.md would look like? Keep reading to see how to safely test that.

However, bear in mind that it won't be a dry run (hence the quotes). Semantic Release will be called as if actually performing a release, but with a different registry and repository. There's no way of dry running a release with semantic release that includes prepare, publish, success or failure steps.

Private, local registry

A private registry can be used to simulate packaging and publishing steps. Verdaccio is a popular choice at this time.

Install Verdaccio and run it:

pnpm i -g verdaccio # you may be instructed to run a setup step
verdaccio

Now let's configure this project to use Verdaccio. Semantic Release's NPM plugin will also read that configuration in order to publish there and not to the official NPM registry.

npm config set registry http://localhost:4873/ --location project

Note

The --location project flag is added so just NPM configuration for this project is changed. Instead of the default behaviour which changes the user's configuration. A .npmrc file will be created in the repo if it doesn't exist. Don't worry, it's git ignored. It won't end up in version control.

Time for authentication! Create a user in the fresh Verdaccio private registry.

Note

Skip this step if you already have a user for the private registry. Ensure you're logged in instead by running npm login

npm adduser ngx-meta
# 👆 Registry should be Verdaccio's localhost URL
#    Any username, password email can be set in there.

Note

Skip this step if you already have a token for the user.

Tip

If you forgot the token for the user, you can use npm token ls to list existing tokens. However the full token won't be listed there. Copy the id. Then remove it by running npm token revoke <tokenId>. Create a new one now :)

Now, we need a token. Let's create one:

npm token create

Take note of it. This token will be used by Semantic Release to authenticate against the registry as seen in the minimal dry run

Different repository

Unfortunately Semantic Release hardcodes pushing to the git repository. So there's no way to dry-run that part :(

Then what it can be done is to use another git repository for this purpose. Options:

🔒 SAFEST (but a bit cumbersome): Create a fork/copy of the repository

To copy all existing tags needed to simulate a release at the current state. Or copy it if you can't fork it as you own it.

Then, when using Semantic Release, set the repository URL to this new repository:

export LOCAL_SEMANTIC_RELEASE_REPOSITORY_URL=https://github.com/user/ngx.git

⚡️ TRICKY (but fastest): Use a noop git remote

[!CAUTION] > It's tricky and dangerous. If not setup properly, or Semantic Release implementation changes, this could end up actually performing actions on the actual repository. Also, you'll need to later clear the objects created by semantic release which will be in your local git repository. Unless you know what you're doing and are very careful, skip this approach.

A trick that can be done is to create a dummy remote that does nothing by setting . as the repository URL.

This way Semantic Release will push everything to this no-op repository URL / remote URL.

export LOCAL_SEMANTIC_RELEASE_REPOSITORY_URL=.

Note

You could use . as the URL for the origin git remote. Semantic Release would grab that and use it for the repository URL if not set anywhere else. But it would end up trying to be converted to a URL anyway. Plus the repository URL may be grabbed from somewhere else therefore using the real repository to operate, which would end up making real changes in there. Also the remote to change would have to be origin, which is hardcoded in the implementation. Finally, pushURL is not respected by semantic release so the origin URL has to be changed.

Cleanup

After running semantic release with this approach, you'll end up with the git objects that semantic release created in your local repository. You'll need to clean them up to avoid weird scenarios in the future (like fetching a tag that already exists as was created during a "dry-run")

This means currently:

git tag -d <createdTag> # you can use `git tag -l` to list them
git notes prune -v # to clear notes about non-existing tags
# 👆 Seems tag deletion already deletes the note though

If using Semantic Release's git plugin (or another that writes to git), you'll need to remove the created commit by it too.

Running the release

With the private registry setup and the new repository created, you can run the release.

Bear in mind to configure if working on a branch. Set up the needed env vars as seen in minimal dry run. With following differences:

  • GitHub: if using . as repository URL, you can skip configuring it. If just interested in NPM publish, you can skip that part by commenting out the plugin in the configuration. Otherwise, provide a token that can write to the repository created for running the release simulation.
  • NPM: provide the auth token instead obtained in the private local registry step

Finally, run the release rehearsal. It is a script wrapping a Semantic Release command that will ensure to publish to the local private registry.

pnpm semantic-release:local

Note

To try again releasing the same release, you can unpublish it from Verdaccio:

npm unpublish <pkg>@<version> --registry http://localhost:4873

Be careful of specifying the registry. You don't want the package to be actually removed from NPM registry!

Tools

IDE

JetBrains WebStorm is used to develop this project. The repository contains many run configurations and settings to help you. However, feel free to use whatever tool you prefer :)

Git hooks

Some Git hooks are provided to help you:

  • Follow commit conventions
  • Format code automatically before committing
  • Lint code automatically before committing

thanks to husky.

They should be installed by default. If not, do run

pnpm husky

Quirks

Can't run unit tests: TestBed error

If you see this error:

Error: Need to call TestBed.initTestEnvironment() first
    at TestBedImpl.compiler (node_modules/.pnpm/@[email protected][email protected][email protected]/node_modules/@angular/core/fesm2022/testing.mjs:1970:19)

Maybe you accidentally ran pnpm install inside a project's source code (ie: projects/ngx-meta/src). Remove the node_modules inside that directory and error should go away.

See https://stackoverflow.com/a/74903074/3263250

Remember to stop the Karma server before running tests again.