Contributions start with community conversations that lead to positive change. Open source provides a flexible collaboration model that facilitates change, even among perfect strangers. Contributions therefore:
- Begin with Issues,
- Occur in Pull Requests, and
- End with Merges.
- 1. Issues
- 2. Git
- 2.1. Rules
- 2.1.1. Makes changes in a topic branch.
- 2.1.2. Favor the topic branch naming recommendation
type/issue-change-name
. - 2.1.3. Branch out from
master
. - 2.1.4. Never push into the
master
branch. Always submit a Pull Request. - 2.1.5. Submit a Pull Request as soon as possible.
- 2.1.6. Rebase your local
master
branch before you ask for PR approvals. - 2.1.7. Resolve rebase conflicts before Pull Request reviews.
- 2.1.8. Add reviewers and the label
Status: Needs Review
when the topic branch is ready. - 2.1.9. Delete local and remote topic branches after merging.
- 2.1.10. Protect your
master
branch.
- 2.2. Feature-branch-workflow
- 2.2.1. Initialize a Git repository in the product directory (for new repositories only).
- 2.2.2. Checkout a new
feat
ure orfix
branch. - 2.2.3. Make Changes.
- 2.2.4. Follow the Conventional Commits Specification for commit messages.
- 2.2.5. Sync with remote to get changes you’ve missed.
- 2.2.6. Update your topic branch with the latest changes from
master
by interactive rebase. - 2.2.7. Resolve conflicts (if any occur), and continue rebase.
- 2.2.8. Push your branch with the
-f
flag (if necessary). - 2.2.9. Submit a Pull Request.
- 2.2.10. Once accepted, the Pull request will be merged, closed, and deleted by an administrator.
- 2.2.11. Remove your local topic branch if you're done.
- 2.3. Tell your boss how Git enables collaborative process models.
- 2.1. Rules
- 3. Code standards
- 4. Unit testing
- 5. Directory structure
- 6. Logging
- 7. Dependencies
- 8. APIs
- 9. Licensing
-
Collaboration starts with Issues. Changes happen through Pull Requests.
View
eslint-plugin-crc's
collaboration and contribution flowcharts:
-
Why:
⌦
eslint-plugin-crc
follows an issue-driven product delivery model. Before any work is done, create an Issue, first. This starts a conversation about features, defects ("bugs"), refactoring, product delivery improvements, etc.Go ahead! Get started now:
-
Why:
⌦
type
categorizes product changes. Valid types are:build
: Changes that affect the build system or external dependencies.ci
: Changes related to continuous integration, delivery, and deployment tasks.docs
: Documentation changes.feat
: A new feature.fix
: Defect (bug) repair.perf
: Performance enhancements.refactor
: Source code design improvements that don't affect product behavior.style
: Changes involving graphics, typography, etc., as well as source code beautification.test
: Tests added to increase code coverage, or corrected due to errors.
-
Why:
⌦It keeps communication consistent and unambiguous.
-
Why:
⌦ We use
git-labelmaker
to categorize Issues (and Pull Requests) consistently. There are four label categories:Type
: the "kind" of product change.Status
: the state of a change.Priority
: the importance and value of a change.Points
: the size/complexity of a change.
Toggle view of the Label definitions table.
Label 🏷 Definition Type: Feature
A distinguished or expected characteristic of a product that either differentiates the product from competitors, or whose absence would be diminish the product’s value.
Note thatType: Feature
andType: Defect
are mutually exclusive: an Issue cannot be both a feature and a defect.Type: Defect
A flaw, fault, or abnormality that deviates from or prevents the product’s expected behavior.
Note thatType: Feature
andType: Defect
are mutually exclusive: an Issue cannot be both a feature and a defect.CLA: Signed
The person who submitted a product change has signed your Contributor License Agreement.
Remove this label if your product owner does not require a CLA.CLA: Unsigned
The person who submitted a product change has **not**signed your Contributor License Agreement.
Remove this label if your product owner does not require a CLA.Priority: Critical
Type: Feature
: The proposed enhancement is essential to the success of your product.Type: Defect
: Your product no longer functions due to internal,FATAL
errors, and must be addressed immediately in order to maintain consumer loyalty.Priority: High
Type: Feature
: The proposed enhancement is central to product’s value proposition, and should be implemented as soon as possible.Type: Defect
: The product functions overall, but with an issue that risks consumer abandonment.Priority: Medium
Type: Feature
orType: Defect
: The proposed change should be implemented as long as noPriority: Critical
orPriority: High
issues exists.Priority: Low
Type: Feature
: A proposal that minimally affects the product’s value.Type: Defect
: Represents “cosmetic” problems like misspelled words or misaligned text that do not affect branding and marketing strategy.Status: Abandoned
Type: Feature
orType: Defect
: The team and community have neglected, forgotten, discarded, or ignored resolving a Issue.Status: Accepted
Type: Feature
orType: Defect
: The product owner or maintainers agreed to a product change proposal.Status: Available
Type: Feature
andType: Defect
: The change proposal is ready for team and community members to work on.Status: Blocked
Type: Feature
andType: Defect
: The proposed change cannot be addressed until another issue has been resolved.
Note that the Issue blocking the proposed change SHOULD be referenced in theBlocked
Issue’s description field.Status: Completed
Type: Feature
andType: Defect
: The issue has been resolved and all acceptance criteria validated.Status: In Progress
Type: Feature
andType: Defect
: The team or community is actively working on the Issue’s resolution.Status: On Hold
Type: Feature
andType: Defect
: The Product Owner has (temporarily) postponed Issue resolution.
Note that the reason for postponement should be stated in the Issue’s description field.Status: Pending
Type: Feature
andType: Defect
: product change or resolution is either awaiting the Product Owner’s decision. Ideally, the Product Owner should declare why they’re undecided somewhere in the Issue thread.Status: Rejected
Type: Feature
andType: Defect
: The Product Owner has declined a change proposal.
Note that the Product Owner should politely explain why they dismissed the change request.Status: Review Needed
Type: Feature
andType: Defect
: The person working on an Issue has requested help or discussion. When applied to a Pull Request,Status: Review Needed
signifies that the PR is ready for evaluation (and potentially, approval).Status: Revision Needed
Type: Feature
andType: Defect
: The Issue is not ready for evaluation because of incomplete or insufficient information.Type: Breaking Change
The change introduces backward incompatibility with previous product versions. Type: Breaking Change
MUST be recorded with a- Git commit message,
- An increment (+1) in the product’s Semantic Version’s MAJOR version,
CHANGELOG
entry, and- Updated API documentation.
Type: Build
Changes to the process that convert source code into a stand-alone form that can be run on a computer or to the form itself. Type: Chore
Miscellaneous non-functional changes such as typographical fixes or source code repository initialization, e.g., chore(scm): scaffold product directory structure
Type: CI
Continuous Integration (CI) changes, i.e., automated build, test, an quality assurance tasks. Type: Docs
The introduction of or revisions to natural language documents or source code comments. Type: Duplicate
An Issue that shares the same characteristics as a previously reported issue.
Note that product maintainers should reference the original Issue and close theType: Duplicate
Issue.Type: Feedback
A response to a Type: Question
or voluntary information used as a basis for improvement.Type: Fix
A change intended to repair a Type: Defect
Issue.Type: Performance
A change intended to reduce product latency. Type: Question
A request for information. Type: Refactor
Source code design improvements that do not affect product behavior. Type: Revert
Changes that return the product’s source code to previous Git commit hash. Type: Spike
A technical or design experiment that investigates a possible solution.
Note that spike solutions are, by definition, throwaway solutions that should NEVER be added to a product release.Type: Style
Issues that address code standard or brand compliance. Type: Test
Issues that prove intended behavior or substantiate “definitions of done.” Type: Test
can also refer to changes that result in broader code coverage.
-
Why:
⌦ The team might need more clarification.
-
Why:
⌦ It's up to the Product Owner to agree to proposed changes. If they believe your issue add value, the issue will be approved, and we'll ask someone to volunteer to do the work.
Otherwise, your issue will be politely declined.
-
eslint-plugin-crc
manages contributions with the feature-branch-workflow. -
Why:
⌦ Use an isolated topic branch for parallel product development. Topic branches allow you to submit multiple pull requests without confusion. You can iterate without polluting the master branch with potentially unstable, unfinished code. The
eslint-plugin-crc
team uses:- Feature-branch-workflow for small-ish codebases, or
- Gitflow Workflow for large applications and monoliths
-
Why:
⌦ Although not required, our team prefixes branches with the type of change being introduced, followed by a forward slash and the issue id.
Pattern:
type/issueId-subject
Icon legend: Bitbucket GitHubbugfix/
- Defect (bug) repair issues.
build/
- Issues related to product builds.
ci/
- Issues related to continuous integration, delivery, and deployment tasks.
docs/
- Issues related to documentation.
feat/
- New feature or enhancement requests.
feature/
- New feature or enhancement requests.
fix/
- Defect (bug) repair issues.
- `hotfix/`
perf/
- Performance improvement issues.
refactor/
- Source code design **improvements that do not affect product behavior**.
revert/
- Tasks that revert to a previous commit hash.
spike/
- Issues related in solution investigation.
style/
- Issues related to style guideline compliance, especially `ESLint` errors and warnings.
test/
- Test coverage tasks.
-
Why:
⌦
eslint-plugin-crc
follows the feature-branch-workflow. -
Why:
⌦ It notifies team members whenever changes occur and allows the community to review your changes at any time..
It also enables easy peer-review of the code and dedicates forum for discussing the proposed feature.
-
Why:
⌦ Pull Requests declare work in progress. Frequent pushes to a Pull Request notify your team members about change, and gives them the opportunity to provide feedback more often.
Pull Request pushes also trigger automated CI-services, which help you fail fast and assess quality.
-
Why:
⌦ Rebasing will merge in the requested branch (
master
ordevelop
) and apply the commits that you have made locally to the top of the history without creating a merge commit (assuming there were no conflicts). This results in a nice and clean history. -
Why:
⌦ Rebasing will merge in the
master
branch and apply the commits that you have made locally to the top of it. -
Why:
⌦ When you add a Reviewer, GitHub (or Bitbucket) notifies teammates that your topic branch meets all Acceptance Criteria and is ready to be merged into
master
.Add the label "Status: Review Needed" formally declares the status of your topic branch, and helps teams filter through issues.
-
Why:
⌦ Topic branches should only exist while work is in-progress. Merged topic branches clutter up your list of branches with dead branches. Topic branch deletion also insures that you only ever merge back into
master
. -
Why:
⌦ Branch protection prevents production-ready branches from incorporating unexpected and irreversible changes. Learn more about
-
We use the feature-branch-workflow. We recommend interactive rebasing, too, but that's not required.
-
For subsequent features and changes, this step should be ignored.
cd <product-repo-directory> git init
-
# For a new feature branch: git checkout -b feat/<issueId>-scope-of-change # For branches that address defects: git checkout -b fix/<issueId>-scope-of-change
-
git add git commit -a
Why:
⌦
git commit -a
will start an editor which lets you separate the subject from the body. Read more about it in section 1.3. -
This project enforces AngularJS Git Commit Guidelines (which is now an extension of the Conventional Commits Specfication) with
commitplease
pre-commit hooks.Why:
Consistent, legible Git logs not only facilitate communication, but also enable automated
CHANGELOG
generation and semantic versioning withstandard-version
.-
build
commit messagesIssues related to product builds.
build(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
chore
commit messagesIssues related to miscellaneous non-functional changes (usually "maintenance" changes).
chore(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
ci
commit messagesIssues related to continuous integration, delivery, and deployment tasks.
ci(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
docs
commit messagesIssues related to documentation.
docs(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
feat
(feature) commit messagesNew feature or enhancement requests.
feat(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
fix
commit messagesDefect (bug) repair issues.
fix(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
perf
(performance) commit messagesPerformance improvement issues.
perf(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
refactor
commit messagesSource code design improvements that do not affect product behavior.
refactor(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
revert
commit messagesTasks that revert to a previous commit hash. Your message should begin with
revert:
, followed by the header of the reverted commit.In the body it should say:
This reverts commit <hash>.
, where the hash is the SHA of the commit being reverted.revert: <hash> <BLANK LINE> This reverts commit <hash>. <BLANK LINE> <footer>
-
style
commit messagesIssues related to style guideline compliance, especially
ESLint
errors and warnings.style(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
test
commit messagesTest coverage tasks.
test(<scope>): <subject> <BLANK LINE> <[body]> <BLANK LINE> <footer>
-
-
git checkout master git pull
Why:
⌦ This will give you a chance to deal with conflicts on your machine while rebasing(later) rather than creating a Pull Request that contains conflicts.
-
git checkout <branchname> git rebase -i --autosquash master
Why:
⌦ You can use
--autosquash
to squash all your commits to a single commit. Nobody wants many commits for a single feature in develop branch. -
git add <file1> <file2> ... git rebase --continue
-
Rebase changes history, so you might need to force changes into the
remote
branch with the-f
flag. If someone else is working on your branch, use the less destructive--force-with-lease
.git push -f
Why:
⌦ When you do a rebase, you are changing the history on your topic branch. As a result, Git will reject normal
git push
. Instead, you'll need to use the -f or --force flag. -
git branch -d <branchname>
to remove all branches which are no longer on remote
git fetch -p && \ for branch in `git branch -vv | grep ': gone]' | awk '{print $1}'`; \ do git branch -D $branch; \ done
-
Git enables collaboration with structured (business) process models. Git is a collaboration tool that grants people the opportunity to contribute to software products using simple and consistent process models called workflows.
Toggle view of a workflow example.
⌦ Git is a collaboration tool that grants people the opportunity to contribute to software products using simple and consistent process models called _workflows_.
- Git **repositories** store software products, as well as the historical changes to these products.
- Git workflows begin with either **cloning** or **forking** a repository, which **pulls** a duplicate of the entire codebase onto a local computer.
- Contributors then create a local **topic branch** in which they introduce changes—-features, fixes, documentation, tests, and design improvements—-which allows others to develop in parallel.
- Contributors **push** their topic branch to others (usually another `remote` repository called `origin`) and submit **pull requests (PRs)** to their announce their changes. **Pull requests are central to almost all Git workflows, since PRs allow Product Owners and their community to review suggested modifications. Product owners can approve or reject PRs at any time.**
- Other contributors answer the request to **pull** the changes so they can review and test them.
- The changes in the PR are approved.
- The topic branch is either **merged** or **rebased** into the the software product.
- The topic branch gets deleted, and (ultimately) its changes are delivered to the world.
Why:
⌦ Git can be intimidating, and The Boss (even if that's you!) needs to know why Git is valuable in business terms.
- Git **repositories** store software products, as well as the historical changes to these products.
-
Why:
⌦ Git is a collaboration tool that grants people the opportunity to contribute to software products using simple and consistent process models called workflows.
-
Why:
⌦ Many engineers can discuss Git's technicalities, but cannot articulate Git's as an applied technology for collaboration.
-
eslint-plugin-crc
follows the Standard JS Style. -
Why:
⌦ ESLint evaluates JavaScript code (and
--fix
es what it can) whenevernpm test
runs. You can run ESLint directly with:npm run lint:js
View
eslint-plugin-crc's
ESLint rules and their enforcement.
-
Why:
⌦ Behavior-driven development specifications are executable documentation.
-
Put test files in the
/tests/
directory. -
Use the
.spec.js
suffix for all tests.
Tip:
💡 Test continuously during development.
Run
npm run watch:test
to execute all specs every time you save a file. -
-
Why:
⌦ Full coverage makes automated dependency drift updates easier to address, and makes accepting contributions safer.
-
A test coverage summary report appears after each test run in the Terminal:
18 passing (76ms) 11 pending -------------------------------------|----------|----------|----------|----------|----------------| File | % Stmts | % Branch | % Funcs | % Lines |Uncovered Lines | -------------------------------------|----------|----------|----------|----------|----------------| All files | 100 | 100 | 100 | 100 | | crc | 100 | 100 | 100 | 100 | | ast-config.js | 100 | 100 | 100 | 100 | | crc-class.js | 100 | 100 | 100 | 100 | | crc-model.js | 100 | 100 | 100 | 100 | | crc-reporter.js | 100 | 100 | 100 | 100 | | eslint-context-factory.js | 100 | 100 | 100 | 100 | | crc/descriptors | 100 | 100 | 100 | 100 | | class-matcher-crc-class.js | 100 | 100 | 100 | 100 | | index.js | 100 | 100 | 100 | 100 | | new-expression-crc-class.js | 100 | 100 | 100 | 100 | | object-expression-crc-class.js | 100 | 100 | 100 | 100 | | prototype-constructor-crc-class.js | 100 | 100 | 100 | 100 | | formatters/crc | 100 | 100 | 100 | 100 | | index.js | 100 | 100 | 100 | 100 | | -------------------------------------|----------|----------|----------|----------|----------------| > [email protected] posttest /Users/swindle/Projects/github/gregswindle/eslint-plugin-crc > nsp check --output summary (+) No known vulnerabilities found
-
Open
/coverage/lcov-report/index.html
in a Web browser to view detailed coverage reports.
-
.
├── CHANGELOG.md
├── LICENSE
├── NOTICE.md
├── README.md
├── appveyor.yml
├── coverage/
├── docs
│ ├── analytics
│ │ └── readme.md
│ ├── api
│ │ └── eslint-plugin-crc
│ │ └── 0.2.0/
│ ├── diagrams/
│ └── img/
├── inch.json
├── jsdoc.conf.json
├── lib
│ ├── crc
│ │ ├── ast-config.js
│ │ ├── class-matcher-crc-class.js
│ │ ├── class-expression-crc-class.js
│ │ ├── crc-class.js
│ │ ├── crc-model.js
│ │ ├── crc-reporter.js
│ │ ├── crc-responsibilities-builder.js
│ │ ├── descriptors
│ │ │ ├── class-matcher-crc-class.js
│ │ │ ├── class-expression-crc-class.js
│ │ │ ├── index.js
│ │ │ ├── new-expression-crc-class.js
│ │ │ ├── object-expression-crc-class.js
│ │ │ └── prototype-constructor-crc-class.js
│ │ ├── eslint-context-factory.js
│ │ ├── index.js
│ │ ├── new-expression-crc-class.js
│ │ ├── object-expression-crc-class.js
│ │ └── prototype-constructor-crc-class.js
│ ├── formatters
│ │ └── crc
│ │ ├── crc-model-template.md
│ │ ├── crc-page-template.md
│ │ ├── crc-summary-template.md
│ │ ├── icon-javascript-filled-25.png
│ │ └── index.js
│ └── index.js
├── markdown.config.js
├── node-context-logs.log
├── package-lock.json
├── package.json
├── sonar-project.properties
└── tests
├── fixtures
│ ├── crc
│ │ ├── class-declaration
│ │ │ ├── context.json
│ │ │ ├── polygon.js
│ │ │ └── square.js
│ │ ├── eslint-contexts.json
│ │ └── function-declarations
│ │ ├── assignment-expression-constructor
│ │ │ ├── context.json
│ │ │ └── square.js
│ │ ├── assignment-expression-object-create
│ │ │ ├── context.json
│ │ │ └── square.js
│ │ ├── new-expression
│ │ │ ├── context.json
│ │ │ └── square.js
│ │ └── polygon.js
│ └── formatters
│ ├── crc-report.md
│ ├── es5-object-identification.js
│ ├── es5-object-prototypes.js
│ ├── eslint-results.json
│ ├── shape-crc-model-list.json
│ └── square-crc-model.json
└── lib
├── crc
│ ├── crc-class.spec.js
│ ├── crc-model.spec.js
│ ├── crc-reporter.spec.js
│ ├── crc-responsibilities-builder.spec.js
│ ├── descriptors
│ │ ├── class-matcher-crc-class.spec.js
│ │ ├── class-expression-crc-class.spec.js
│ │ ├── new-expression-crc-class.spec.js
│ │ ├── object-expression-crc-class.spec.js
│ │ └── prototype-constructor-crc-class.spec.js
│ └── eslint-context-factory.spec.js
└── formatters
└── markdown
└── markdown.spec.js
eslint-plugin-crc
requires the following tech stack to either run, build, test, or deploy:
Dependency | Description | Version | License | Type |
---|---|---|---|---|
[email protected] | a JSON logging library for node.js services | 1.8.12 | MIT | production |
[email protected] | JSDoc parser | 2.1.0 | Apache-2.0 | production |
[email protected] | An AST-based pattern checker for JavaScript. | 4.19.1 | MIT | production |
[email protected] | An Esprima-compatible JavaScript parser built on Acorn | 3.5.4 | BSD-2-Clause | production |
[email protected] | A query library for ECMAScript AST using a CSS selector like query language. | 1.0.1 | BSD-3-Clause | production |
[email protected] | Lodash modular utilities. | 4.17.10 | MIT | production |
[email protected] | Write minimal node index.js files that require and export siblings by file basename | 1.2.0 | MIT | production |
[email protected] | babel require hook | 6.26.0 | MIT | dev |
[email protected] | BDD/TDD assertion library for node.js and the browser. Test framework agnostic. | 4.1.2 | MIT | dev |
[email protected] | takes json-cov output into stdin and POSTs to coveralls.io | 3.0.1 | BSD-2-Clause | dev |
[email protected] | A clean, responsive documentation template theme for JSDoc 3 inspired by lodash and minami | 0.4.0 | Apache-2.0 | dev |
[email protected] | ESLint shareable config for XO with 2-space indent | 0.18.0 | MIT | dev |
[email protected] | Import with sanity. | 2.11.0 | MIT | dev |
[email protected] | JSDoc linting rules for ESLint. | 3.6.3 | BSD-3-Clause | dev |
[email protected] | custom ESLint rule to disallows unsafe innerHTML, outerHTML and insertAdjacentHTML | 1.0.16 | MPL-2.0 | dev |
[email protected] | ESLint rule to disallow unsanitized code | 3.0.0 | MPL-2.0 | dev |
[email protected] | Additional ESLint's rules for Node.js | 6.0.1 | MIT | dev |
[email protected] | Enforce best practices for JavaScript promises | 3.7.0 | ISC | dev |
[email protected] | ESLint plugin that contains ScanJS rules | 0.2.1 | MPL-2.0 | dev |
[email protected] | Security rules for eslint | 1.4.0 | Apache-2.0 | dev |
[email protected] | ESlint Plugin for the Standard Linter | 3.1.0 | MIT | dev |
[email protected] | Validates XSS related issues of mixing HTML and non-HTML content in variables. | 0.1.9 | ISC | dev |
[email protected] | An API documentation generator for JavaScript. | 3.5.5 | Apache-2.0 | dev |
[email protected] | Automatically update markdown files with content from external sources | 0.1.21 | MIT | dev |
[email protected] | Generate table of information about dependencies automatically in markdown | 1.3.2 | MIT | dev |
[email protected] | Print list of scripts in package.json with descriptions | 1.2.1 | MIT | dev |
[email protected] | simple, flexible, fun test framework | 5.1.1 | MIT | dev |
[email protected] | Simple monitor script for use during development of a node.js app. | 1.17.3 | MIT | dev |
[email protected] | The Node Security (nodesecurity.io) command line interface | 3.2.1 | Apache-2.0 | dev |
[email protected] | the Istanbul command line interface | 11.7.1 | ISC | dev |
[email protected] | A parser for .properties files written in javascript | 0.3.1 | MIT | dev |
[email protected] | Automated semver compliant package publishing | 15.2.0 | MIT | dev |
[email protected] | snyk library and cli utility | 1.78.0 | Apache-2.0 | dev |
[email protected] | replacement for npm version with automatic CHANGELOG generation |
4.3.0 | ISC | dev |
-
Why:
⌦ Because we try to enforce development of sanely constructed RESTful interfaces, which team members and clients can consume simply and consistently.
Why:
⌦ Lack of consistency and simplicity can massively increase integration and maintenance costs. Which is why
API design
is included in this document.-
We mostly follow resource-oriented design. It has three main factors: resources, collection, and URLs.
- A resource has data, gets nested, and there are methods that operate against it.
- A group of resources is called a collection.
- URL identifies the online location of resource or collection.
Why:
⌦ This is a very well-known design to developers (your main API consumers). Apart from readability and ease of use, it allows us to write generic libraries and connectors without even knowing what the API is about.
-
Use kebab-case for URLs.
-
Use camelCase for parameters in the query string or resource fields.
-
Use plural kebab-case for resource names in URLs.
-
Always use a plural nouns for naming a url pointing to a collection:
/users
.Why:
⌦ Basically, it reads better and keeps URLs consistent. read more...
-
In the source code convert plurals to variables and properties with a List suffix.
Why:
⌦ Plural is nice in the URL but in the source code, it’s just too subtle and error-prone.
-
Always use a singular concept that starts with a collection and ends to an identifier:
/students/245743 /airports/kjfk
-
Avoid URLs like this:
GET /blogs/:blogId/posts/:postId/summary
Why:
⌦ This is not pointing to a resource but to a property instead. You can pass the property as a parameter to trim your response.
-
Keep verbs out of your resource URLs.
Why:
⌦ Because if you use a verb for each resource operation you soon will have a huge list of URLs and no consistent pattern which makes it difficult for developers to learn. Plus we use verbs for something else.
-
Use verbs for non-resources. In this case, your API doesn't return any resources. Instead, you execute an operation and return the result. These are not CRUD (create, retrieve, update, and delete) operations:
/translate?text=Hallo
Why:
⌦ Because for CRUD we use HTTP methods on
resource
orcollection
URLs. The verbs we were talking about are actuallyControllers
. You usually don't develop many of these. read more... -
The request body or response type is JSON then please follow
camelCase
forJSON
property names to maintain the consistency.Why:
⌦ This is a JavaScript project guideline, Where Programming language for generating JSON as well as Programming language for parsing JSON are assumed to be JavaScript.
-
Even though a resource is a singular concept that is similar to an object instance or database record, you should not use your
table_name
for a resource name andcolumn_name
resource property.Why:
⌦ Because your intention is to expose Resources, not your database schema details.
-
Again, only use nouns in your URL when naming your resources and don’t try to explain their functionality.
Why:
⌦ Only use nouns in your resource URLs, avoid endpoints like
/addNewUser
or/updateUser
. Also avoid sending resource operations as a parameter. -
Explain the CRUD functionalities using HTTP methods:
How:
GET
: Retrieve a representation of a resource.POST
: Create new resources and sub-resources.PUT
: Replace existing resources.PATCH
: Update existing resources. It only updates the fields that were supplied, leaving the others alone.DELETE
: Delete existing resources. -
For nested resources, use the relation between them in the URL. For instance, using
id
to relate an employee to a company.Why:
⌦ This is a natural way to make resources explorable.
How:
GET /schools/2/students
, should get the list of all students from school 2.GET /schools/2/students/31
, should get the details of student 31, which belongs to school 2.DELETE /schools/2/students/31
, should delete student 31, which belongs to school 2.PUT /schools/2/students/31
, should update info of student 31, Use PUT on resource-URL only, not collection.POST /schools
, should create a new school and return the details of the new school created. Use POST on collection-URLs. -
Use a simple ordinal number for a version with a
v
prefix (v1, v2). Move it all the way to the left in the URL so that it has the highest scope:http://api.domain.com/v1/schools/3/students
Why:
⌦ When your APIs are public for other third parties, upgrading the APIs with some breaking change would also lead to breaking the existing products or services using your APIs. Using versions in your URL can prevent that from happening. read more...
-
Response messages must be self-descriptive. A good error message response might look something like this:
{ "code" : 404, "level" : "ERROR", "logger" : "[http-logger]", "message" : "No resource found at URL /archetypes/v1/locales/iso-country-codes/BS", "timestamp": 1504878062000 }
or for validation errors:
{ "code" : 400, "logger" : "[registration-form-logger]", "level" : "ERROR", "timestamp": 1504878062000, "message" : "Validation Failed", "stack" :[ { "code" : 1233, "field" : "email", "message": "Invalid email" }, { "code" : 1234, "field" : "password", "message": "No password provided" } ] }
Why:
⌦ Developers depend on well-designed errors at the critical times when they are troubleshooting and resolving issues after the applications they've built using your APIs are in the hands of their users.
Note: Keep security exception messages as generic as possible. For instance, Instead of saying ‘incorrect password’, you can reply back saying ‘invalid username or password’ so that we don’t unknowingly inform user that username was indeed correct and only the password was incorrect.
-
Use only these 8 status codes to send with you response to describe whether everything worked, The client app did something wrong or The API did something wrong.
Which ones:
200 OK
response represents success forGET
,PUT
orPOST
requests.201 Created
for when new instance is created. Creating a new instance, usingPOST
method returns201
status code.304 Not Modified
response is to minimize information transfer when the recipient already has cached representations.400 Bad Request
for when the request was not processed, as the server could not understand what the client is asking for.401 Unauthorized
for when the request lacks valid credentials and it should re-request with the required credentials.403 Forbidden
means the server understood the request but refuses to authorize it.404 Not Found
indicates that the requested resource was not found.500 Internal Server Error
indicates that the request is valid, but the server could not fulfill it due to some unexpected condition.Why:
⌦ Most API providers use a small subset HTTP status codes. For example, the Google GData API uses only 10 status codes, Netflix uses 9, and Digg, only 8. Of course, these responses contain a body with additional information.There are over 70 HTTP status codes. However, most developers don't have all 70 memorized. So if you choose status codes that are not very common you will force application developers away from building their apps and over to wikipedia to figure out what you're trying to tell them. read more...
-
Provide total numbers of resources in your response.
-
Accept
limit
andoffset
parameters. -
The amount of data the resource exposes should also be taken into account. The API consumer doesn't always need the full representation of a resource.Use a fields query parameter that takes a comma separated list of fields to include:
GET /student?fields=id,name,age,class
-
Pagination, filtering, and sorting don’t need to be supported from start for all resources. Document those resources that offer filtering and sorting.
-
-
These are some basic security best practices:
-
Don't use basic authentication unless over a secure connection (HTTPS). Authentication tokens must not be transmitted in the URL:
GET /users/123?token=asdf....
Why:
⌦ Because Token, or user ID and password are passed over the network as clear text (it is base64 encoded, but base64 is a reversible encoding), the basic authentication scheme is not secure. read more...
-
Tokens must be transmitted using the Authorization header on every request:
Authorization: Bearer xxxxxx, Extra yyyyy
. -
Authorization Code should be short-lived.
-
Reject any non-TLS requests by not responding to any HTTP request to avoid any insecure data exchange. Respond to HTTP requests by
403 Forbidden
. -
Consider using Rate Limiting.
Why:
⌦ To protect your APIs from bot threats that call your API thousands of times per hour. You should consider implementing rate limit early on.
-
Setting HTTP headers appropriately can help to lock down and secure your web application. read more...
-
Your API should convert the received data to their canonical form or reject them. Return 400 Bad Request with details about any errors from bad or missing data.
-
All the data exchanged with the ReST API must be validated by the API.
-
Serialize your JSON.
Why:
⌦ A key concern with JSON encoders is preventing arbitrary JavaScript remote code execution within the browser... or, if you're using node.js, on the server. It's vital that you use a proper JSON serializer to encode user-supplied data properly to prevent the execution of user-supplied input on the browser.
-
Validate the content-type and mostly use
application/*json
(Content-Type header).Why:
⌦ For instance, accepting the
application/x-www-form-urlencoded
mime type allows the attacker to create a form and trigger a simple POST request. The server should never assume the Content-Type. A lack of Content-Type header or an unexpected Content-Type header should result in the server rejecting the content with a4XX
response.
-
-
- Fill the
API Reference
section in README.md template for API. - Describe API authentication methods with a code sample.
- Explaining The URL Structure (path only, no root URL) including The request type (Method).
For each endpoint explain:
-
URL Params If URL Params exist, specify them in accordance with name mentioned in URL section:
Required: id=[integer] Optional: photo_id=[alphanumeric]
-
If the request type is POST, provide working examples. URL Params rules apply here too. Separate the section into Optional and Required.
-
Success Responses. What should be the status code and is there any return data? This is useful when people need to know what their callbacks should expect:
Code: 200 Content: { id : 12 }
-
Error Responses. Most endpoints have many ways to fail. From unauthorized access to wrongful parameters etc. All of those should be listed here. It might seem repetitive, but it helps prevent assumptions from being made. For example:
-
HTTP status code
404 Not Found
-
Response body
{ "code" : 404, "level" : "ERROR", "logger" : "[http-logger]", "message" : "No resource found at URL /archetypes/v1/locales/iso-country-codes/BS", "timestamp": 1504878062000 }
-
Response headers
accept-ranges: bytes access-control-allow-headers: Authorization access-control-allow-methods: GET, HEAD, OPTIONS access-control-allow-origin: * cache-control: public, no-transform, must-revalidate connection: keep-alive content-encoding: gzip content-language: en-US content-length: 149 content-type: application/json date: Fri, 08 Sep 2017 06:41:02 GMT last-modified: Tue, 1 Oct 2014 10:10:10 GMT server: nginx/1.12.1 vary: Accept-Encoding
-
-
Use API design tools, There are lots of open source tools for good documentation such as API Blueprint and Swagger.
- Fill the
Make sure you use resources that you have the rights to use. If you use libraries, remember to look for MIT, Apache or BSD but if you modify them, then take a look into license details. Copyrighted images and videos may cause legal problems.