From da98ae1dfee7b2a4d7cef006b432c521dfe80de4 Mon Sep 17 00:00:00 2001 From: Ryan Nowak Date: Mon, 11 Sep 2023 11:43:54 -0700 Subject: [PATCH] Add code review documentation --- .../contributing-code-reviewing/README.md | 189 ++++++++++++++++++ .../contributing-pull-requests/README.md | 4 +- docs/contributing/how-to.md | 3 + 3 files changed, 195 insertions(+), 1 deletion(-) create mode 100644 docs/contributing/contributing-code/contributing-code-reviewing/README.md diff --git a/docs/contributing/contributing-code/contributing-code-reviewing/README.md b/docs/contributing/contributing-code/contributing-code-reviewing/README.md new file mode 100644 index 0000000000..f4a5426e44 --- /dev/null +++ b/docs/contributing/contributing-code/contributing-code-reviewing/README.md @@ -0,0 +1,189 @@ +# Code reviews + +We welcome **any contributor or community-member** to engage with any **any pull-request** on our repository as a reviewer. + +This page some contains guidance for: + +- Our philosophy as maintainers for code-reviewing +- How to give effective code review feedback +- Responsibilities for the **maintainer** and **reviewer** roles + +This is recommended reading for anyone participating in pull-requests either as a submitter or reviewer. + +## About code reviews + +Code reviews for Radius take place as part of the pull-request process. See the [documentation on pull-requests](../../contributing-pull-requests/) for information and tips about submitting a pull-request. This means that code reviews take place on [Github](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-pull-requests) using the pull-request UI and comment system. [Github's documentation](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/reviewing-changes-in-pull-requests/commenting-on-a-pull-request) covers the basics of using the comment system. + +Contributors and reviewers might communicate about a pull-request outside Github, for example on a video call or in chat. We encourage this when it is helpful, and we ask that contributors update Github with the important feedback and conclusions reached so that the information is visible to others. + +## Philosophy + +As maintainers we like the [code review pyramid](https://www.morling.dev/blog/the-code-review-pyramid/) as a guiding principle. + +The most valuable feedback comes from a deep understanding of the project goals and design, and should be applied early in the pull-request process. Surface-level feedback like style suggestions is still valuable, but is most relevant later in the pull-request lifecycle. + +Another observation is that the further towards the top of the pyramid the more people are capable of giving feedback. Any Go programmer or software developer should be able to give meaningful feedback on test and documentation quality. In contrast, those most qualified to give deep feedback about a change are the set of people with most experience and knowledge of the project. + +We also highly value automation for trivial matters, and use automation to turn debates into trivial matters. You can't argue with the CI. + +### Application to Radius + +We can tweak the code review pyramid's descriptions to be more applicable to Radius (from top to bottom): + +- **Style**: Does the code match good Go style? Is the style consistent with the decisions in the surrounding code? +- **Correctness**: Does the code work as advertised? Are tests in place to make sure it will continue to work? +- **Design**: Is the design testable? Is the design straightforward? Is the design consistent with the decisions in the surrounding code? +- **Behavior**: Does the user-visible behavior makes sense to the user? Has the feature design been fully considered, edge cases and all? +- **Scope**: Does the feature belong in Radius? Is this feature something that the maintainers can support long-term? + +> :lightbulb: Open questions about behavior and scope should ideally be resolved before an issue becomes a pull-request, but things don't always go to plan. If a code review identifies a misunderstanding or incomplete design for a feature's behavior, it's a good idea to put the pull-request *"on pause"* until we can resolve the ambiguity. + +Since we automate most matters of Style, and try to resolve matters of Behavior and Scope before a pull-request then that means the focus of most reviews should be on the Correctness and Design. + +## How to give good feedback + +Good code review feedback: + +- Is on-topic and related to the change being discussed +- Is polite and uses clear and simple language +- Is well-reasoned and explains why a change should be made +- Is appropriate for the maturity of the pull-request + +We want code-review feedback to be on-topic and related to the change being made. As a reviewer, if you are unsure about the scope of a change feel free to ask, it is the submitter's job to make the scope clear. + +We want code-review feedback to be polite and use relatively simple language. In a public project your comments can be seen by the general public, and will be seen by people for whom English is not their first language. Try to phrase your feedback and questions clearly and simply. If you personally know the person who submitted the pull-request, then avoid being overly casual with your comments. Github is a public space. + +We want code-review feedback to be well-reasoned and self-contained. If it helps explain the point you are making, feel free to link to external resources. Avoid making suggestions that provide an alternative without a justification - there are many ways to solve a problem with code, alternatives without a good reasoning are rarely helpful. + +We want code-review feedback to be appropriate based on the maturity of the pull-request. Large and complex pull-requests usually go through several iterations, with the first few focused on design. If you are unsure about the maturity of a pull-request please ask the submitter and then give appropriate feedback based on their response. + +### Where are you on the pyramid? + +As individuals, we're all approaching code reviews with differing levels of expertise, knowledge, and free time. Try to contribute where you can add the most value and have the most impact. + +Most of what's needed in code reviews is *feedback on correctness and the tests that ensure correctness over time*. If you're new to Radius or new to code reviewing in general, this is definitely where to apply yourself. If you're new here and trying to learn more about the codebase, opening some code reviews and then diving deep on the relevant topics is a great way to focus your learning. + +### Giving praise + +It's great to leave positive code-review feedback too! When we talk about code-review feedback the discussion is often focused on the requirements, and ways to give criticism politely. Positive feedback is valuable too and lets contributors know they are on the right track. + +### Good feedback: Style + +Our expectations for code style and linting are described in our [coding documentation](../contributing-code-writing/README.md#coding-style--linting). You do not need to spend your time commenting on issues that will be caught by our automation. The set of rules we follow is very minimal and is enforced by tooling already, exceptions are documented. As an example, we expect that all contributors can run `go fmt` on their code. If a submitter has not done this, then please point them to the `go fmt` documentation instead of personally pointing out every problem. + +Beyond this we value consistency with the decisions we've already made. Ideally the code we write is consistent everywhere. Failing that, it should be consistent with the surrounding components. + +Outside of these expectations, please feel free to make style suggestions if you think the code could be clearer. If the submitter has a good reason for doing it the way they are, and rejects your feedback then please consider: *"Is someone credibly going to make a mistake in the future based on this decision?"* before continuing to push the issue. We want the project to be approachable for new contributors (both new to Go and new to Radius). We do not want to frustrate people with unneccesary ceremony. + +The bar is **very high** for adopting new rules or changing style guidance for the project. Please start a discussion **outside** of the context of a pull-request if you think we need to adopt a new rule. + +### Good feedback: Correctness + +Review feedback for correctness should focus on the following points: + +- Does the code work as advertised? +- Do the tests being added ensure that the code will continue to work as advertised? + +--- + +If you're new to code-reviewing, a good way to figure this out is to look at the areas of most complexity. Focus on the most complex first. Mentally, try to describe the behavior (or read the documentation) of a *"unit"* and then walk through the code and see if your mental model matches. Are there any edge cases? Are there better ways to do it? + +Now that you've walked through a *"unit"* of functionality, look at the tests for that *"unit"*. Is there sufficient coverage of the common cases? What about negative cases. + +This is a basic recipe to follow so you can give good feedback without familiarity with the codebase or overall architecture. + +---- + +As you're reviewing, here's a checklist of things to look for in most code: + +- What errors can occur and how are they reported? +- What are the dependencies? How are errors from dependencies handled? +- What standard library or dependency functions are being used, are there more appropriate alternatives? +- What are the preconditions and postconditions of a function? What is the failure mode when the preconditions are violated? +- If the code is highly algorithmic is there are a simpler alternative? +- What kinds of tests should be written? + - All code should have unit tests. New code should have better-than-average unit test coverage. + - Code that interacts with external systems should have integration tests or functional tests that verify the interactions between systems. + - User-visible behaviors and features should have E2E tests if they are significant. + +### Good feedback: Design + +Review feedback for design should focus on the following points: + +- Is this design consistent with the feature's current requirements? +- Is this design consistent with the choices we have made for similar functionality? (does not reinvent the wheel) +- Is there a simpler design that would achieve the same thing? +- Do the names and concepts make sense? Are they consistent? + +--- + +Go is a pragmatic and simple language and our designs should leverage its simplicity. In general we want to avoid complex hierarchies or object-orientied abstractions where possible. The flip-side of this is that because Go is a simple language, the type system and what it can express is limited. + +Where possible we want to optimize for correctness, testability, and simplicity in that order. + +--- + +We include *naming* in the design category because an understanding of the requirements, concepts, and design of a feature is required knowledge for most good naming feedback. For each feature or feature-area we want to define a limited set of named concepts, and then use those concepts to choose the names of software constructs like struct and variable names. + +It is much better for us to leverage user-facing concepts in code than to make up artificial constructs. This way someone with a knowledge of the features can navigate the code based on their background knowledge. + +--- + +As you're reviewing, here's a checklist of things to look for in most code: + +- Can the design satisfy all of the functional requirements? +- What are the data-access patterns? How are we limiting the scope and number of data-accesses necessary? +- What are the dependencies? What are the interaction patterns with dependencies? +- Do the names reflect the concepts? Have we created artificial concepts? +- How long are the functions (in terms of code)? +- Are there new Go APIs? Do they follow *effective Go*? +- How testable is the design? + +### Good feedback: Behavior + +*The decision of how a feature will behave should occur before a pull-request is submitted. This section is guidance for Approvers and Maintainers who have official decision-making responsibility over the project.* + +If new feature does not have a clearly-defined behavior that you are **confident** in, please raise your concerns respectfully and ask the submitter to put the pull-request *"on pause"* while the discussion plays out. When doing this please avoid giving other kinds of feedback, as this can send a mixed-signal to the submitter. Apply the `do not merge` label to make the status clear. + +The appropriate avenue for discussion is usually an issue. If a large change is submitted without an issue, then it should be paused until we can clarify the expectations. + +As a Maintainer, you are responsible for the documentation, quality, and usefulness of the project. If you cannot understand clearly how a feature behaves, then you cannot make a recommendation to users about how and when to use it, and so you cannot support it. + +### Good feedback: Scope + +*The decision of whether a feature or change is in-scope for the project should occur before a pull-request is submitted. This section is guidance for Approvers and Maintainers who have official decision-making responsibility over the project.* + +If you're uncomfortable accepting a pull-request due to its scope or complexity, please raise your concerns respectfully and ask the submitter to put the pull-request *"on pause"* while the discussion plays out. When doing this please avoid giving other kinds of feedback, as this can send a mixed-signal to the submitter. Apply the `do not merge` label to make the status clear. + +The appropriate avenue for discussion is usually an issue. If a large change is submitted without an issue, then it should be paused until we can clarify the expectations. + +As a Maintainer, you are responsible for the long-term maintenance and stewardship of the project. Don't accept changes you are uncomfortable with. We have limited time and capacity, so everything we say *"yes"* to means that we're saying *"no"* to something else. Saying *"no"* today can preserve your ability to say *"yes"* in the future. + +## Roles + +Within the Radius project we recognize individuals with a history of contributions to the project as **approvers** - they have the right to approve pull-requests. Those with high quality judgement and leadership are also recognized as **maintainers**. Maintainers are the decision makers for the technical roadmap of the project and have the final decision over each change. + +More details about those roles can be found in the [community repo](https://github.com/radius-project/community). + +### Responsibilities of an approver + +Approvers are expected to display good judgement, responsiblity, and competance over the *top* of the code review pyramid (Style, Correctness, Design). + +It's likely that for for small changes a maintainer will delegate responsibility for approval to a maintainer working in the relevant area. + +Approvers are responsible for enforcing the **completeness** of a change. Each pull-request that is merged to `main` must stand on its own merits. We do not allow a submitter to merge and then "follow up" to complete the work (for example merging a change without tests). If you encounter this situation, **DO NOT** merge the pull-request, ask the submitter to complete the work. If they are unable to do so, then it is appropriate to close the pull-request without merging. + +### Responsibilities of a maintainer + +Maintainers are expected to display good judgement, responsiblity, and competance over **all** of the code review pyramid, with special emphasis on Behavior and Scope. Maintainers are responsible for the technical oversight of the project, and this often means saying **"no"** when a feature request is out of scope, or a change is too complex to maintain. + +Maintainers are explicitly required to review and approve new dependencies to the project. + +### Code of conduct enforcement + +We encourage anyone to report a [code of conduct](https://github.com/radius-project/community/blob/main/CODE-OF-CONDUCT.md) violation if they see one occur. Any code of conduct violation should be reported as per instructions in the link above. + +It is **explicitly** a responsibility of approvers and maintainers to report a violation if they suspect one. If a code of conduct violation occurs in a pull-request, follow the instructions [here](https://docs.github.com/en/communities/moderating-comments-and-conversations/managing-disruptive-comments) to minimize the comment and report the violation. + +Approvers and maintainers (or anyone else with write-access) **MUST NOT** edit any *other* contributor's comments to maintain trust and transparency. If you do this by accident, please restore the original content and apologize with your own comment. + diff --git a/docs/contributing/contributing-pull-requests/README.md b/docs/contributing/contributing-pull-requests/README.md index 694310972e..50784fca4c 100644 --- a/docs/contributing/contributing-pull-requests/README.md +++ b/docs/contributing/contributing-pull-requests/README.md @@ -105,4 +105,6 @@ If you are the code reviewer, it's your responsibility to follow up (politely) i ### Participating in code review -We welcome any contributor or community member to engage with any pull request on our repository. Feel free to make suggestions for improvements and ask questions that are relevant. If you're asking questions for your learning, please make it clear that your questions are "non-blocking" for the pull request. \ No newline at end of file +We welcome **any contributor or community member** to engage with **any pull request** on our repository. Feel free to make suggestions for improvements and ask questions that are relevant. If you're asking questions for your learning, please make it clear that your questions are "non-blocking" for the pull request. + +See the [code reviewing documentation](../contributing-code/contributing-code-reviewing/README.md) for guidance on code reviewing. \ No newline at end of file diff --git a/docs/contributing/how-to.md b/docs/contributing/how-to.md index 306ca0dbb7..9eb796687a 100644 --- a/docs/contributing/how-to.md +++ b/docs/contributing/how-to.md @@ -12,6 +12,9 @@ We welcome and encourage users to contribute to the Radius open-source project i * Submit pull requests: Once you've made changes or improvements to your forked repository, [submit a pull request](./contributing-pull-requests/) to the main project. Provide clear descriptions of the purpose and benefits of your changes. +* Help with code reviews: Give feedback and help with changes made by others +* through [code reviews](./contributing-code/contributing-code-reviewing/). + * Engage with the community: Participate in discussions, forums, and chat channels related to Radius. Seek guidance, offer help to others, and build connections within the community. * Test and report bugs: Assist in testing Radius by [identifying and reporting bugs](./contributing-issues/). Detailed bug reports help us address issues promptly and enhance the software's quality.