- Code Standards & Contributing Guidelines
- Follow guidelines and local formatting and linting settings (like prettier) for style and formatting.
- Write self-documenting code and minimize comments.
- Ensure comprehensive test coverage including happy and error paths.
- Provide meaningful and constructive code reviews.
- Adhere to Conventional Commits for commit messages and PR naming.
- Document every feature adequately, especially for open-source projects.
- Keep documentation clear, concise, up-to-date, and accessible.
- Branching - choose consistent naming conventions, include issue number, delete branches after merging.
- Google JavaScript Style Guide - Google's coding standards for JavaScript.
- Google TypeScript Style Guide - Google's coding standards for TypeScript.
- Thinking in React - good for beginners to start thinking in terms of React Components.
- Node.js best practices - summary and curation of the top-ranked content on Node.js best practices.
- Mozilla's Guidelines for JavaScript - Mozilla's guidelines for JavaScript with code examples.
- When possible, use default values of a given type rather than
null
orundefined
values.
const person = {
firstName: "Bill",
lastName: null, // 🟥
lastName: "", // ✅
}
- If for some reason a default value cannot be assigned or it is impractical to assign a default value, assign a
null
value rather than anundefined
value.
const person = {
firstName: "Bill",
address: undefined, // 🟥
address: null, // ✅
}
- When code you have no control over (external library) or JavaScript itself may return undefined, convert it to null (or preferably to a default value if possible).
const found = arr.find((item) => item > 5); // 🟥
const found = arr.find((item) => item > 5) ?? null; // ✅
- Whenever writing TypeScript code, avoid using
any
and always annotate types for Props passed to a Component.
interface MyComponentProps {
setName: any // 🟥
setName: React.Dispatch<React.SetStateAction<string>> // ✅
}
const MyComponent = (props: any) => {} // 🟥
const MyComponent: FC<MyComponentProps> = ({setName}) => {} // ✅
- Use curly braces
{}
instead ofnew Object()
.
const newObject = new Object() // 🟥
const newObject = {} // ✅
- Use brackets
[]
instead ofnew Array()
.
const newArray = new Array() // 🟥
const newArray = [] // ✅
- Use
===
and!==
instead of==
and!=
.
if (oneObject == anotherObject) {} // 🟥
if (oneObject === anotherObject) {} // ✅
- When writing html/jsx/tsx, use proper semantic html tags, suitable for a given component.
return (
<>
<navbar>
<MyNavBarComponent />
</navbar>
<main>
<MyMainSectionComponent />
</main>
<footer>
<MyFooterComponent />
</footer>
</>)
List of all categorized html tags with short description: HTML Elements Reference
- When an import needs to go to more than one directory above, use full-path imports.
import { MyComponent } from "../../../MyComponent" // 🟥
import { MyComponent } from "/src/components/MyComponent" // ✅
Every component should have a folder with the name of that component. In that folder, you should keep an index.{js/ts} file, to export that component, a component itself and all the files that only this component refrences: style files, custom hooks, etc.
Style files should take the name of the component with the suffix .style.{js/ts}. For example: MyComponent.style.js
.
Custom hooks can take the name of use<componentName>.{js/tx}. For example: useMyComponent.js
.
Example:
|-- index.html
|-- index.js
|-- package.json
|-- package-lock.json
|-- README.md
`-- src/
|-- app.js
`-- MyComponent/
|-- MyComponent.js
|-- MyComponent.style.js
|-- useMyComponent.js
- Refactoring tasks should be strategically undertaken; not every piece of code warrants modification. If a segment of code, despite comments, remains untouched and unproblematic, it is likely fulfilling its purpose effectively.
- When embarking on feature additions or bug fixes, and encountering commented code, consider extracting functions and creating a PR before proceeding with the primary task. This helps maintain code clarity and function.
- Should you come across a superfluous comment during your changes, do take the initiative to remove it and create a PR for that. This practice contributes to keeping the repository neat and well-maintained.
- Endeavor to minimize the addition of comments when making any changes to the code.
- Be vigilant of newly added comments during reviews. If a comment appears unnecessary, uninformative, or could be replaced with a function, do not hesitate to highlight this.
- Assess the meaningfulness and clarity of function names, ensuring they contribute to self-documenting code.
Developers are required to diligently cover their changes with tests and organize these tests with caution. A well-structured set of tests serves as a safety net, facilitating the swift identification and resolution of issues.
-
Readability: Maintain test readability through the use of descriptive test names. When possible, group cases into table-driven tests, exercising discretion to avoid excessiveness. Avoid code branching in tests; instead, split differing scenarios into separate test cases.
-
Comprehensive Coverage: Ensure comprehensive test coverage, including error paths. Prioritize adding use cases for well-known errors that could be returned from your code.
-
Structural Consistency: Adopt the "Given-When-Then" structure in automatic tests to enhance readability and consistency. This structure assists team members in following the test flow and understanding its purpose, thus facilitating smoother onboarding for new members and maintaining uniformity across tests.
- Given: Set the stage by preparing inputs, mocks, and application state.
- When: Trigger the action or function under test.
- Then: Verify if the outcomes match the expectations.
import { importantFunction } from "./index.js" test("Test Something Very Useful Is Happening", () => { // given const functionInput = "importantInput" const expectedResult = "importantResult" // when const result = importantFunction(functionInput) // then expect(result).toBe(expectedResult) })
-
Test Isolation: Ensure test isolation by avoiding the use of global variables and shared state. Each test should be independent and not rely on the execution of other tests. If a test requires a shared state, use a setup function to create the state before each test.
-
Test Data: Avoid random data in tests. Instead, use predefined data to ensure test consistency and reproducibility. If random data is required, use a seed to ensure the same data is generated each time the test is run.
-
Test Cases: If you are writing a public functions - it should be covered by tests. We should have test cases for all possible scenarios. That means that we should have tests for all possible errors that can be returned from the function. Of course not only error paths should be covered - we should highlight the happy path as well.
-
Testing private (unexported) functions: When testing private functions, we should test them through the public (exported) functions that use them. The exception is when the private function is too complex to be tested through the public function. Good example is for example a function that is implementing a complex algorithm. In this case we should test the private function directly.
-
Constructive Feedback: Reviewers should provide constructive feedback, highlighting both strengths and areas of improvement. Comments should be clear, concise, and related to code structure, functionality, or style.
-
Coding Standards: Both authors and reviewers should ensure the code adheres to established coding standards, including formatting, naming conventions, and best practices as outlined in official Golang guidelines.
-
Testing: Reviewers should ensure that the submitted code is accompanied by adequate tests, covering both happy paths and error paths. Check the readability, descriptiveness, and structure of tests.
-
Error Handling: Pay special attention to error handling within the code. Ensure that errors are not ignored, logged appropriately, and presented to the user in a user-friendly format when applicable.
-
Performance: Review the code for any potential performance issues, such as inefficient loops, unnecessary allocations, or misuse of concurrency.
-
Dependency Management: Ensure that any new dependencies are necessary, appropriately versioned, and have been vetted for performance and security.
-
Security Practices: Ensure that the code follows secure coding practices and avoids common vulnerabilities. Good checklist for security practices can be found here.
-
Documentation: Confirm that the code is well-documented, including comments, function/method descriptions, and module-level documentation as necessary.
-
Efficiency and Readability: The code should be efficient and readable. Reviewers should look for any code smells, overly complex functions, and ensure the use of idiomatic Go patterns.
-
Responsiveness: Both authors and reviewers should be timely in their responses. Authors should address all review comments, and reviewers should re-review changes promptly.
- Does the code adhere to the project’s coding standards?
- Are there sufficient tests, and do they cover a variety of cases?
- Is error handling comprehensive and user-friendly?
- Are there any performance concerns in the code?
- Have new dependencies been appropriately vetted?
- Does the code follow secure coding practices and avoid common vulnerabilities?
- Is the code well(self)-documented, with clear variable, function naming?
- Is the code efficient, readable, and free of code smells?
- Have all review comments been addressed in a timely manner?
- Do you understand the code and it's purpose?
This checklist serves as a guide to both authors and reviewers to ensure a thorough and effective code review process.
We have separate templates for Pull Requests and Issues. Please use them when creating a new PR or Issue.
In an effort to maintain clarity and coherence in our commit history, we are adopting the Conventional Commits style for all commit messages across our repositories. This uniform format not only enhances the readability of our commit history but also facilitates automated tools in generating changelogs and extracting valuable information effectively.
Conventional Commits follow a structured format: type(scope): description
, where:
type
: Represents the nature of the commit (e.g., feat, fix, chore).scope
: Denotes the relevant module or issue.description
: Provides a brief explanation of the change.
When introducing breaking changes, an !
should be appended after the type/scope
:
feat(#123)!: introduce a breaking change
.
feat
: Utilized when introducing a new feature to the codebase.fix
: Employed when resolving a bug or issue in the code.docs
: Designated for commits involving documentation changes, such as updating README files or adding comments.style
: Applied to commits focusing on code style and formatting, without altering the code's functionality.refactor
: Used for code changes that neither introduce new features nor fix bugs, but improve the code structure or design.test
: Assigned to commits pertaining to the addition, modification, or refactoring of tests.chore
: For changes related to build processes, local development, or other maintenance tasks.perf
: Employed when enhancing the performance of the codebase.revert
: Marked for commits that revert a previous change.ci
: Applied to changes concerning the Continuous Integration (CI) configuration or scripts.deps
: Used when updating or modifying dependencies.
In our repositories, we use Conventional Commits to automatically generate the version number for our releases.
It works like this:
fix: which represents bug fixes, and correlates to a SemVer patch.
feat: which represents a new feature, and correlates to a SemVer minor.
feat!:, or fix!:, refactor!:, etc., which represent a breaking change (indicated by the !) and will result in a SemVer major.
Real life example:
feat(#123)!: introduce breaking change - 1.0.0 -> 2.0.0
feat(#124): introduce new feature - 2.0.0 -> 2.1.0
fix(#125): fix a bug - 2.1.0 -> 2.1.1
Given a version number MAJOR.MINOR.PATCH, increment the:
MAJOR version when you make incompatible API changes MINOR version when you add functionality in a backward compatible manner PATCH version when you make backward compatible bug fixes Additional labels for pre-release and build metadata are available as extensions to the MAJOR.MINOR.PATCH format.
More about Semantic Versioning can be found here.
We have standardized the use of JIRA/GitHub issue numbers as the scope
in commits within our team. This practice aids in easily tracing the origin of changes.
In the absence of an existing issue for your changes, please create one in the client’s JIRA system. If the change is not client-related, establish a GitHub issue in the repository.
Additional information and guidelines on Conventional Commits can be found here.
Good example:
feat(#123): add possibility to create a new user by admin
Bad example:
debugo feature - checkpoint full work
feat(#123): add new feature
- Choose consistent naming conventions. Common practices include:
feature/feature-name
bugfix/issue-or-bug-name
hotfix/hotfix-name
chore/task-name
refactor/refactor-name
- Branch names should be descriptive and represent the task/feature at hand.
- Use hyphens to separate words for readability, e.g.,
feature/add-login-button
.
- If applicable, include the issue number in the branch name for easy tracking, e.g.,
feature/123-add-login-button
.
- Once a PR has been merged, delete the remote branch to keep the repository clean.
- GitHub provides a button to delete the branch once the PR is merged.
- Regularly prune local branches that have been deleted remotely with
git fetch -p && git branch -vv | grep 'origin/.*: gone]' | awk '{print $1}' | xargs git branch -d
.
A well-documented codebase is pivotal for both internal development and external contributions, especially for open-source projects that expose functionalities for public use. Comprehensive documentation, supplemented with examples where necessary, ensures that every feature is easily understandable, usable, and maintainable.
- Clarity and Conciseness: Documentation should be clear, concise, and focused, providing necessary information without unnecessary complexity or verbosity.
- Accessibility: Documentation should be accessible to developers of varying skill levels, enabling both novices and experts to understand the codebase and its features.
- Up-to-Date: Documentation must be kept current, reflecting the latest changes and developments in the codebase to avoid misinformation and confusion.
Every feature developed should be accompanied by adequate documentation. The necessity for documentation becomes even more pronounced for open-source projects, where clear instructions and examples facilitate easier adoption and contribution from the community.
- Inclusion of Examples: Where applicable, documentation should include practical examples demonstrating the feature’s usage and benefits. Examples act as a practical guide, aiding developers in understanding and implementing the feature correctly.
- Clarity of Examples: Examples should be clear, concise, and relevant, illustrating the functionality of the feature effectively.
For projects exposing external features:
- Comprehensive Guides: Ensure the creation of comprehensive guides detailing the utilization of exposed features, their benefits, and any potential configurations or customizations.
- Community Engagement: Encourage community members to contribute to documentation by providing feedback, suggestions, and improvements. This collaborative approach enriches the documentation quality and breadth.
We should write documentation in Markdown format. It allows us to write documentation in a simple and readable way. It's also easy to convert Markdown to HTML or PDF or create a website from it.
Markdown Guide - Comprehensive guide to Markdown syntax.
Adhering to documentation code standards is integral for maintaining a healthy, understandable, and contributable codebase. By ensuring every feature is well-documented, with the inclusion of clear examples where necessary, we foster a conducive environment for development and community engagement, particularly in open-source projects.