Skip to content

Latest commit

 

History

History
244 lines (145 loc) · 15.8 KB

CONTRIBUTING.md

File metadata and controls

244 lines (145 loc) · 15.8 KB

Contributing

This page contains instructions and guidelines for anybody contributing code to the eBayUI project.

Table of Contents

Development

This section includes information on system requirements, running the local server, tests and package scripts.

System Requirements

Clone this repo to your local environment then run npm i to install all dependencies.

Local Server

Execute npm start to start the local web server at http://localhost:3000. This is a quick method to run and debug code, but it will only work in modern browsers since it does not include transpilation.

For debugging older browsers, the demo can be run against transpiled source code via npm run start:prod.

The demo page will render the examples/ for each component. To selectively include certain examples for debugging, add .only to their folder names.

Tests

Execute npm test to run all tests and generate coverage reports at ./coverage/index.html.

Most of our server tests use vitest & @marko/testing-library in order to verify html snapshots against the generated html that the current component outputs. If these change the test will fail. In order to update the snapshots, one needs to run npm test -u and then new snapshots will be genrated, which should be checked in.

Marko Lifecycle Functions

Because eBayUI components are written in Marko, the component files may contain many of the Marko lifecycle functions. Within a given component's index.js file you will find the lifecycle functions intermixed with the component functions. Documentation on these functions can be found in the Marko documentation on lifecycle events.

Package Scripts

Execute npm run, or check the "scripts" section of package.json to view all available utility scripts.

Please ensure all scripts in package.json are Unix and Windows friendly.

Continuous Integration (CI)

To ensure that all components function correctly in both Marko v3 and v4, we run our entire test suite against both versions. This requires automated installation of Marko versions, and is best executed in the CI (to avoid conflicting with local environments). The CI runs a specific build:ci task in the NPM scripts for this purpose.

By default when in the CI all browser tests will be ran against the supported list of browsers using BrowserStack. If you would like to manually test with your own BrowserStack account you can do so by adding a .env file to the root of this project with the following environment variables:

BUILD_NUMBER="local"
REPO_SLUG="ebay/ebayui-core"
BROWSERSTACK_USER="YOUR BROWSERSTACK USERNAME"
BROWSERSTACK_ACCESS_KEY="YOUR BROWSERSTACK ACCESS KEY"

Finally run npm run test:browserstack to begin the tests.

Pull Requests

This section contains information and etiquette regarding pull requests (or "PRs" as they are often called).

Before You Start

If you are creating a new component or significantly altering an existing one, please ensure that an issue has been created before you start any work, and that it has been assigned to you. The same goes for any big new additions or changes to the project structure, CI or documentation.

We ask this because:

  • we want to avoid cases where developers build something that does not align with our wants & needs
  • we want to be able to carefully plan our sprint and test cycles with minimal disruption
  • we want to avoid cases where two developers duplicate work

Writing Code

The contents of a pull request should be related to a single issue only (or, at most, perhaps two very closely related issues). The reviewer has the right to reject a pull request that contains anything non-issue related.

Whilst it may be tempting to fix any broken windows that you encounter, we ask you not to because:

  • it can distract the reviewer from the main issue at hand
  • it can add additional time needed for the reviewer
  • it can increase the chance of regressions
  • it can make rollbacks more difficult

So please be a good citizen and create separate issues or pull requests for any broken windows that you find.

Moving Code

Please try and isolate "noisy" commits that only move many files or lines of code, from commits that actually modify code logic. The reviewer has the right to reject a pull request that is difficult to reason with due to too much unnecessary noise in the diff.

For example, assuming moving some code doesn't put the app into a broken state, the move can be considered a safe atomic commit (e.g. "moved functions x, y and z to bottom of file"). The actual fixes to functions x, y and z can then be made in a follow up commit (e.g. "fixed functions x, y and z") which will be easier for the reviewer to diff.

Breaking Changes

Please think very carefully before introducing breaking API changes. Breaking changes may only be introduced in a major version. If you wish to alter the API before a major version, then aliases can often be a good way to achieve this. Using an alias, the old API can be deprecated and will still continue to function.

Deprecating Code

Any deprecated code should be clearly documented in the code comments and the release notes. Deprecated code will typically be scheduled for removal in the next major version.

Avoiding Conflicts

It is your responsibility to ensure that your feature branch has no merge conflicts with its base branch. A pull request created and sent for review containing many conflicts does not inspire confidence in the reviewer.

The best way to ensure there are no conflicts is by keeping your branch up to date with its base branch and resolving any conflicts in your own branch early & often. If you are the only developer working in the feature branch, then it is usually better to rebase. If another developer is sharing your branch, then merging is the safer option (as it doesn't require a force push).

If you are working on a large feature, that takes many days, then there is a good chance that the base branch will have received other commits during that time. If you wait till the very end before syncing your branch with the base branch, then you may have to deal with some difficult merge conflicts.

Commit History

Whilst having multiple "work in progress", "almost done" and "merged branch" type commits in a feature branch is fine, we wish to refrain from those commits polluting the milestone and master branch history. On the flip side, although rarer in comparison, we might also have some atomic commits in our feature branch that we absolutely wish to keep as part of the project history.

Before creating your pull request you have two options regarding what to do with the commit history of your feature branch:

  1. Squash all superfluous commits. If more than 1 commit remains, then clearly label your PR as "do not squash" in order to keep this commit history. Any subsequent commits to the branch (i.e. those made after code review) may also need squashing by you.
  2. Do not squash any commits and clearly mark your PR as "must squash" to indicate to the reviewer that you do not wish to keep any commit history. If you wish to keep any subsequent commits (i.e. after code review), you will now need to switch tracks to option 1 above.

If a pull request is received with a bewildering commit history that is difficult to reason with (e.g. 10+ commits containing obscure commit messages, previously committed code, old branch merges, etc), it will most likely be deemed too risky and therefore closed immediately.

Changesets

Each Pull Request requires at least one changeset as part of the pull requests. Changesets will allow automatic deployments of code when they are merged to master or one of the feature branches.

The general recommendation is to have only one changeset per pull requiest. All pull requests to master need to have at least one changeset. It is recommended to have a changeset for each PR to a feature branch.

In order to add a changeset run the command npm run change. Choose the appropriate version that this changeset will require: patch for bugfixes, minor for features, and major for breaking changes. Enter the summary of the changes. Generally we follow the same format for each of our commits, list the component change, and a brief description of the changes. These will be added in the release notes, so make sure they are descriptive enough for that.

Afterwards, add and commit the file generated under the .changeset directory as part of your PR. Those files will be removed when the release is done as part of the automatic release process.

Merging the Pull Request

An admin will merge or squash the pull request when it receives two approvals. Please try to avoid committing any more code at this point unless it is absolutely necessary, as this may invalidate the prior approvals (depending on branch settings).

The final commit message should include a reference to the pull request number. Commit descriptions are optional/rare and should be kept short. GitHub will by default annotate the description with a list of all commits in the pull request - this description is unnecessary and should be deleted.

Branching Strategy

Members of the eBayUI team must work in the main repository. Non-members must work in their own fork.

Do not attempt to commit feature work directly to the master branch. Pushes to the master branch should only be used for important corrections to the documentation or website. The master branch always reflects the current production facing code.

All other branches are "milestone" branches or "issue" branches.

Every milestone branch must be created from the master branch. The branch name must match the milestone name, For example, when beginning work on the 2.9.0 release, a "2.9.0" milestone branch would be created from the master branch.

Work for every issue must go in it's own branch. The branch name will match the issue number. For example, for issue number 202, a branch named "202" would be created. Remember: more than one developer may work in an issue branch, so please be courteous!

Issue branches must be created from the relevant milestone branch. For example, if issue 202 will go out in the v2.6.7 milestone, then the 202 issue branch must be created from the 2.6.7 milestone branch. This helps to keep any rebasing sane.

When work on an issue branch is complete and committed upstream, the Travis CI job must successfully run on that branch.

When all milestone issues are complete, and merged into the milestone branch, an admin will either perform a pre-release, or they will merge the milestone branch into the master branch in preparation for a final release.

A milestone branch will be deleted after it has been merged into master. There is no need to keep these milestone branches lying around, as we can go back to any point in time using git tags.

With this strategy, there are rarely more than a handful of branches at any one moment in time:

  • master branch (reflects current production code/release)
  • next minor or major milestone branch
  • next patch milestone branch
  • n number of feature/issue branches (usually 1 - 3)

Unit Tests

Each component has two suites of unit tests: server-side tests & client-side tests.

Our unit tests follow the "black-box" approach. We test only the exposed, public API. We do not explicitly test the internal, private methods. Private methods are considered to be tested implicitly whenever we test part of the public API (which can be evidenced via code coverage reports).

Server side testing is predominantly concerned with the testing of rendering. Given an input of a marko tag and it's attributes, we test that the rendered output matches our expected outcome.

Browser side testing is predominantly concerned with the testing of state. For this we follow the Given-When-Then approach (i.e. given a component is in a well-defined state, when a well-defined action is performed on it, then we assert that it meets the expected outcome).

For both types of tests, please observe the following golden rules:

  • A test should only test one thing
    • Watch out for the words 'and' & 'or' in your test description!
  • A test should not contain conditional logic
    • We don't want to have to write tests for our tests!
  • A test description should avoid interpolated strings
    • Test descriptions should be human readable (readable != parsable)

Above all, all aspects of a test should be human readable. A simple litmus test for this is that anybody should be able to read the test and, within a few seconds, be able to tell you what the purpose of the test is. If they are confused, or ask questions, then it is not a well written test. Go back and refactor!

Definition of Done

A component is considered "done", and ready for merge into release branch, when the following 3 checkpoints have been satisfied:

  1. Foundational layer review
  • Visual design (DS6)
  • Markup structure (Bones)
  • Accessibility pattern (MIND)
  • CSS (Skin)
  1. Component layer review
  • Tag API
  • Event API
  • Unit tests
  • Integration test
  • Performance test
  • Documentation
  1. Code review
  • Linter should catch syntax and style issues
  • Prefer performance over readability, but readability over "space saving" code
  • Apply the single-responsibility principle to functions
  • Refactor long functions into several small functions
  • Identify and move common utility functions to libraries or static methods
  • All CSS should come from Skin

Naming Scheme

eBayUI Core is an implementation of the eBay MIND Patterns. The MIND patterns naming scheme follows an accessibility-first mindset; thus aligning our developer speak as closely as possible with existing W3C web standards & conventions.

Releases

Pre-Release

All major and minor releases should be preceded by one or more pre-releases. All pre-releases must be created from a milestone branch.

In order to enter prerelase mode, run npm run prerelase:start. This will generate a file under .changeset directory. Commit and push up that change to the milestone branch. This will generate a pull request targetting the milestone branch. Once that pull request is merged in the milestone branch a prerelease will be published.

Once the prerelases are complete, run npm run prerelase:end. This will generate a file under the .changeset directory. Commit and push up that change to the milestone branch. Another Pull Request will be generated, this PR should be closed and not merged.

Final Release

A final release is always made from the master branch.

  1. Verify that the milestone branch has files in the .changesets directory and that npm run prerelase:end has been run and committed.
  2. Update skin on the milestone branch
  3. Create a GitHub PR to merge the milestone branch into master branch.
  4. Merge the PR after approval (do not squash!)
  5. Once merged and after the CI is run, a PR will be automatically generated. Verify that the version number in the PR is correct.
  6. Squash and merge the automatically generated PR.
  7. The package will be published to NPM.

After every major and minor release, please take the opportunity to upgrade any outdated dependencies and devDependencies (hint: run npm outdated to see outdated dependencies). Except for major version upgrades, the version in package.json should always reflect the last known working version, not the version you are upgrading to.