How to Do a Code Review in Bitbucket: Best Practices from Our Team

November 28, 2023
#How To#Project management#Bitbucket#Collaboration
21 min

In this article, we will spill the beans and share how our team conducts code review in Bitbucket to ensure smooth feature delivery and mitigate risks.

At Stiltsoft, we have several development teams dedicated to creating apps that enhance the functionality of Atlassian products. Our development team consists of 4 full-stack engineers with accumulated software development experience of over 30 years and focuses on several apps for Jira and Bitbucket. One of them is Awesome Graphs for Bitbucket, which is used by over 1,600 companies, including Apple, Oracle, and Mastercard.

Table of Contents

Software Development Workflow

The illustration of the code review process and its main principles would only be complete by delving into the software development workflow. In our team, this process is structured around the following key milestones:

  1. Solution Proposal
  2. Code writing
  3. Code review and Testing
  4. Release and Deployment

software development workflow

Let’s look closer at the first three stages and their peculiarities.

Solution Proposal

Before making significant changes to the code, we prepare a Solution Proposal to outline the planned changes and their implementation to our teammates. Significant changes include:

  • adding a completely new feature
  • making substantial changes to UI or changes that impact many files
  • changing backend components
  • integrating new tools.

Our solution proposal consists of 4 main parts:

  1. Motivation – Why did we decide to work on this issue?
  2. Context of the problem – What is happening now?
  3. Suggested Solution – How exactly are we going to implement this?
  4. Consequences – What changes will this solution entail?

This is extremely useful, particularly for newcomers who could spend a lot of time implementing a solution that either does not meet our internal standards or could be achieved much easier and faster as we have already resolved such issues. Apart from this, it helps the whole team to:

  • validate the solution with colleagues and avoid “tunnel vision” or reinventing the wheel
  • discuss the implementation of complex issues that are not entirely clear how to solve
  • facilitate the later code review process as teammates are already familiar with the proposed solution
  • check if the suggested solution adheres to the business requirements.

When the Solution Proposal is ready, the author shares it with the product and development teams and organizes a call in Slack to discuss questions if necessary. These documents are stored in the Confluence product space, ensuring that our team or colleagues can access them whenever needed.

Code Writing

Here, we won’t dwell on how to write the code but will describe how the repositories are structured, using the example of one of our apps, Awesome Graphs for Bitbucket. Since the apps for Bitbucket Cloud and Data Center differ in functionality, we have two separate repositories for them. However, unlike the backend, the frontend for both app editions is almost identical. To avoid duplicating features, we allocated the shared part of the code into a separate third repository and connected it as a Git submodule.

code writing

It allows us to:

  • maintain consistency across both versions of the app
  • avoid code duplicating and implementing shared features twice
  • reduce redundancy and development efforts.

However, this solution has its drawbacks, such as:

  1. Complexity in reviewing changes as reviewers might need to navigate between the parent repository and the submodule’s repository to understand the full context of the changes.
  2. Merge conflicts as adding a new feature to the shared code for one parent repository blocks changes for the second repo until we update the submodule.
  3. Submodule updates challenges as we don’t create a separate pull request in the subdirectory. Thus, after merging the pull request into the main repository, the author needs to remember to merge the feature branch of the shared repo to the dev branch.

How the team tackles these challenges:

  1. We use Submodule Changes for Bitbucket to avoid complexity in reviewing changes. This tool allows us to review code changes in the subdirectory as a part of the main repository. We can leave comments and tasks on the submodule’s code in the pull request of the parent repo.
  2. Possible merge conflicts and the impact of a feature on another parent repo are considered within the Solution Proposal.
  3. We use the internal app to automate the process of merging the feature branch to the dev branch of the shared repository.

Code Review and Testing 

Since the details of the code review will be given in the next section, the only thing to mention here is that we create pull requests to facilitate this process in Bitbucket Data Center.

As for testing, an important point to note is that we don’t have QA engineers. The development team carries out testing to check the code for bugs and errors, and the product team to verify compliance with business requirements. We can use tools like Playwright to automate the process if necessary. The main idea behind this approach is that if a developer writes code conscientiously and pays attention to the quality, there is no need to involve testers. The code owner will understand its behavior and find deviations much better than anyone else. The developer creates tests during code writing to cover at least 80% of the code in the pull request. This is tracked for each commit using SonarQube.

Now that you know the main milestones of our development process, let’s move on to the main principles of the team’s code review process.

Code Review Guidelines of Our Team

When to Create a Pull Request

We agreed to create a pull request only if it contains code ready for deployment so the app will remain working after its merge. When an author needs feedback on functionality that is not ready for production, alternative ways, such as comparing two branches or requesting assistance in Slack, are used.

When there are dependent pull requests, the subsequent pull request is created only when the previous one was merged. For example, we recently enhanced one of the reports for the Awesome Graphs app. To avoid making one large pull request, we split the code changes into two parts: frontend and backend. Initially, the author created a pull request to review changes in the backend. Only after its merge did they submit the second one with frontend modifications. This is done to ensure that reviewers don’t get puzzled over determining which pull request to review first and have a clear understanding of priorities.

In this case, two versions of one functionality may appear in the master branch. As in the example above, we had two backend versions for the same report. That’s why, to indicate our intention to remove the old functionality soon and ensure that another developer won’t start using it, we annotate methods and classes slated for removal in future pull requests with “Deprecated”. This annotation should contain a link to the feature, after which completion, the annotated element should be deleted.

Pull Request Title and Description

Creating a pull request is always accompanied by giving it a title and adding a description. The title should be clear and reflect the essence of code changes, e.g., Pie Report Optimization: Backend.

The description is added so that each reviewer understands what changes will be made and why. It is similar to a short summary of the Solution Proposal and consists of 3 parts:

  • Context. An overview of the initial problem and associated constraints.
  • Solution. Key points of how we resolve the issue.
  • ResultsA brief description of what we will get after implementing this solution.

code review guidelines

Pull Request Reviewers

As our development team is relatively small, we add all members to review a pull request by default. This is done to share knowledge and ensure everyone is aware of code changes. However, if a reviewer cannot participate in code review for any reason, e.g., vacation, illness, or heavy workload, they remove themselves from the list of reviewers. Later, they can rejoin code review if circumstances have changed.

The main principle is that at least one reviewer should always be available to assess the pull request. Typically, code reviews involve 2-3 developers.

Once all developers have approved the pull request, the author adds a product team member. This step is essential as the product team tests changes and checks their compliance with the requirements and expected results.

PR Pickup Time & Review Rounds

During code review, one of the crucial aspects our team considers is pull request pickup time. We define it as the time from when a reviewer is added to the pull request until their first action.

The Service Level Agreement (SLA) for pickup time is one working day. The pull request author monitors this, and if no response is received within this time frame, they notify the team in the Slack channel. The second and subsequent reminders the author sends once every 4 hours.

We clearly define these time frames to avoid making the author guess when they will receive feedback and to enable them to make plans and work on other tasks.

The same rules are applied to reviewers’ pickup time after changes are made to the pull request. In this case, the author notifies reviewers by adding a comment such as “All comments addressed, please review again.”

Reviewers, in turn, notify the author of the pull request that they finished the code review by adding “Approved” or “Needs work” status.

We do not limit the number of review rounds but track the number of activities in each pull request, as many comments might indicate some issues in the workflow. To perform this, we have an internal app for Bitbucket, which counts comments and tasks left in PRs. If this number exceeds 30, this tool asks the author and the reviewers to answer four questions:

  1. What went well in this pull request?
  2. Why do you think there are more than 30 comments and tasks?
  3. What can be done next time to avoid this?
  4. What caused the most trouble in this pull request personally to you?

how to do a code review

We review these answers every two months to reflect on the processes, find bottlenecks, and optimize the workflow.

Pull Request Size

In pull requests, we look not only at the number of comments but also at their size. Our team believes that a great pull request should strike a balance between being small enough to be reviewed quickly by teammates and comprehensive enough to cover the planned changes or feature additions. That’s why an author of a pull request always strives to make it as simple and clean as possible.

Moreover, we are considering working on a new report for the Awesome Graphs app. It will enable users to see pull request size distribution and track changes in PR size over time. This functionality is not currently available to our clients, but we’ve made its prototype using the team’s data. Pretty good, right?

If this report sparked your interest, feel free to contact us. We will be glad to share the details and hear your thoughts.

pull request size

Code Refactoring

When developing a new feature, the author sometimes comes across some parts in the code that they’d like to rewrite and update. This is inevitable as technologies evolve, new tools emerge, and our expertise expands. However, we agreed that refactoring is permitted within the scope of a pull request only if it influences the feature we are working on. Besides, reviewers can request to create a separate pull request for changes related to refactoring or to roll back these changes in case they distract from the review of the main functionality. This is extremely useful for large-scale refactoring as the developer will work on a particular task without investing time in non-priority ones.

Refactoring unrelated to the current feature is postponed or scheduled for R&D Day.


Every Friday, instead of working on regular tasks, we focus on enhancing the Developer Experience (DX) and exploring new tools to streamline and improve our processes. Here also come tasks related to refactoring and technical debt.


Tools for Code Review in Bitbucket

We’ve already mentioned some of these tools, but let’s put them together.

Submodule Changes for Bitbucket – a tool that helps review the changes made to Git submodules in the Diff tab in Bitbucket. We use it for two primary purposes:

  1. To review changes in the submodule as if they are a part of the parental repository. We can leave comments and tasks on the submodule’s code within the code review process of the parental repository.
  2. To check a new merge to ensure that the submodule commit can be merged to the dev branch in fast-forward mode.

tools for code review

SonarQube – a tool for static code analysis and calculating the percentage of code covered by tests. It helps us analyze the code of a single pull request. We have installed the Include Code Quality for Bitbucket plugin, which displays the SonarQube code quality check results within the pull request. It prevents us from merging a pull request if the code quality does not meet our team’s standards:

  • There are no errors or vulnerabilities in the new code identified by SonarQube.
  • Test coverage for new code is at least 80%. This requirement encourages us to gradually increase the test coverage of the codebase, as modifying previously written code, we also write tests for it if none exist.

TeamCity Integration for Bitbucket – a tool to run TeamCity builds from Bitbucket. We use it for:

  • Running tests and initiating SonarQube checks after each commit pushed to Bitbucket.
  • Managing the deployment of our projects to the Atlassian Marketplace and production environments. Although this is not directly related to code review in Bitbucket, it is essential to the overall development process.

Following these code review guidelines allows us to maintain high-quality code and foster efficient collaboration, leading to better, more reliable software and a more productive development environment. Apart from this, we always leave room for improvement and discuss the processes every two months to make the development of our products even more robust and effective.

Related posts

    Remote Teams Management: How to Turn Data into Code Review Insights

    March 17, 2020
    #How To#Bitbucket#Reporting#Analytics
    12 min

    Remote teams management can be quite challenging. Distance makes it more difficult to monitor your distributed team activity. So, you have to either rely on them, hoping it won’t end up as a disaster, or annoy developers with overcontrol.

    Fortunately, there’s no need to go blindly as you can go data-driven and use the statistics of developers’ activity to your advantage.

    In this article, we focus on pull requests metrics as a way to identify bottlenecks in the code review process and check if it’s healthy or not. You will find the answers to the following questions:

    • What metrics to choose as a subject to analyze and how to organize them.
    • How to retrieve them using Bitbucket REST API.
    • How to interpret metrics to get as many insights as possible.

    Pull Requests status dashboard

    If you want to keep in touch with the pull requests on your project status, you may find it useful to create a dashboard that visualizes the statuses of pull requests. This helps you see:

    • the number of pull requests your team created for the last week/month/sprint
    • how many pull requests are already merged and which of them are still open
    • if there are any pull requests open long ago and are not merged yet.

    You can retrieve this data using this Bitbucket REST API request:

    https://your-bitbucket-url/rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests?state=all

    Get the following information from the returned results: pull request ID, state, and the date it was created and merged in the Unix time format.

    {
        "size": 25,
        "limit": 25,
        "isLastPage": false,
        "values": [
            {
                "id": 13,
                "version": 13,
                "title": "Looking for Sarah Connor",
                "description": "Trying to save the Future",
                "state": "MERGED",
                "open": false,
                "closed": true,
                "createdDate": 1583837364846,
                "updatedDate": 1583843660283,
                "closedDate": 1583843660283,
                "fromRef": {},
                "toRef": {},
                "locked": false,
                "author": {
                    "user": {
                        "name": "arnie",
                        "emailAddress": "arnie@stiltsoft.com",
                        "id": 3,
                        "displayName": "Arnie",
                        "active": true,
                        "slug": "arnie",
                        "type": "NORMAL",
                        "links": {},
                    "role": "AUTHOR",
                    "approved": false,
                    "status": "UNAPPROVED"
                },
                "reviewers": [
                    {
                        "user": {
                            "name": "sarahconnor",
                            "emailAddress": "sarahconnor@stiltsoft.com",
                            "id": 4,
                            "displayName": "Sarah Connor",
                            "active": true,
                            "slug": "sarahconnor",
                            "type": "NORMAL",
                            "links": {},
                        "lastReviewedCommit": "",
                        "role": "REVIEWER",
                        "approved": true,
                        "status": "APPROVED"
                    }
                ],
                "participants": [],
                "properties": {
                    "resolvedTaskCount": 3,
                    "commentCount": 1,
                    "openTaskCount": 0
                },
                "links": {}
                    ]
                }
            },

    Then create a script that parses these data and turns it into a pull requests status dashboard. 

    There are some possible difficulties you may face due to the REST API restrictions:

    • The number of returned results is limited, so you’ll have to make several requests to retrieve all data.
    • It’s only possible to run the query on a repository level, so the more repositories you have, the more requests you’ll need to make.
    • The Bitbucket performance may be affected in large instances.

    The same can be done using the Pie Chart report in Awesome Graphs for Bitbucket. Grouped by state, it shows the number of pull requests created during the chosen time span with their current statuses. 

    Pull request metrics - check status of pull requests

    Tip: find pull requests created long ago and not merged yet by selecting a previous week or month as a period to analyze.

    Reviewers’ involvement

    Another pattern to watch is whether all team members participate in the code review. If not, there are some possible consequences:

    • knowledge sharing in a team is poor, and junior developers miss the opportunity to grow their skills without watching the examples of more experienced programmers’ work
    • pull requests resolution may be slow if there are more pull requests created than people who can review it
    • those who review all pull requests may have no time to write new code.

    The same REST API request from the previous paragraph retrieves the data about the participants of pull requests. Organize these data as a list of reviewers and the number of pull requests they reviewed to get insights about the developers’ involvement in the peer review. 

    The Pie Chart report grouped by reviewer serves the same purposes and gives you a breakdown of pull requests reviewed by all members of your team.

    Analyze how your team is involved in the code review process

    Tip: pay attention to pull requests that are merged without reviewing. Those are the areas where you may have a risk as the code was not checked as appropriate. 

    Reviewers’ activity chart

    The difference between just being assigned as a reviewer and truly reviewing the new code and suggesting improvements to it is huge. Luckily, there are a couple of metrics to watch that may help you get a clear view of the thoroughness with which developers check each other’s work.

    Use this REST API request and retrieve the data about the number of tasks, comments, needs work, or approved actions per pull request and group it by user:

    https://your-bitbucket-url/rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/activities

    The example of the returned results with the data that needs to be parsed:

    {
        "size": 5,
        "limit": 25,
        "isLastPage": true,
        "values": [
            {
                "id": 46766,
                "createdDate": 1574762295000,
                "user": {
                    "name": "sarahconnor",
                    "emailAddress": "sarahconnor@stiltsoft.com",
                    "id": 3,
                    "displayName": "Sarah Connor",
                    "active": true,
                    "slug": "sarahconnor",
                    "type": "NORMAL",
                    "links": {}
                },
                "action": "MERGED",
                "commit": {}
            },
            {
                "id": 46756,
                "createdDate": 1574690608000,
                "user": {
                    "name": "johnconnor",
                    "emailAddress": "johnconnor@stiltsoft.com",
                    "id": 652,
                    "displayName": "John Connor",
                    "active": true,
                    "slug": "johnconnor",
                    "type": "NORMAL",
                    "links": {}
                },
                "action": "APPROVED"
            },
            {
                "id": 46688,
                "createdDate": 1574431693000,
                "user": {
                    "name": "arnie",
                    "emailAddress": "arnie@stiltsoft.com",
                    "id": 4,
                    "displayName": "Arnie",
                    "active": true,
                    "slug": "arnie",
                    "type": "NORMAL",
                    "links": {}
                },
                "action": "COMMENTED",
                "commentAction": "ADDED",
                "comment": {
                    "properties": {},
                    "id": 6569,
                    "version": 0,
                    "text": "i need your jacket, boots, and motorcycle",
                    "author": {}
                    },
            {
                "id": 46668,
                "createdDate": 1574418254000,
                "user": {
                    "name": "sarahconnor",
                    "emailAddress": "sarahconnor@stiltsoft.com",
                    "id": 3,
                    "displayName": "Sarah Connor",
                    "active": true,
                    "slug": "sarahconnor",
                    "type": "NORMAL",
                    "links": {}
                },
                "action": "OPENED"
            }
        ],
        "start": 0
    }

    The visualization of the developers’ activity in pull requests provides you with the following insights:

    • Most active reviewers in your team. These people check the code thoroughly. On the other hand, this pattern may have negative sides if the perfectionism slows down the feature delivery too much.
    • Infrequent participants in the code review. These developers are not involved in the knowledge sharing process for some reason. Do they skip this part of work intentionally, or are they afraid to criticize their colleagues’ work? 
    • Periods with unusual activity. Check the values that go beyond the trend to find the answer to why the reviewer had to leave too many or too little comments. For example, there can be bad code or the pull request was too large, and nobody wanted to check it. These are the patterns that you’d definitely like to avoid.

    The Contributions report provides the same view across projects and repositories with the swift access to each pull request in Bitbucket so you could check its content.

    Apply a data-driven approach in remote teams management

    Tip: group this report by author to see who receives the largest number of tasks and comments in their pull requests. This may be a sign of a code that took lots of re-writing and may lead to bugs in the future. Besides, looking at the dynamics of these numbers makes it obvious if developers are improving their working patterns or not.

    Try the data-driven approach to bring transparency into your code review processes

    Tracking developers’ activity may be especially important in remote or distributed teams where it’s impossible to reach each other physically. 

    However, the benefits that software development metrics bring are valuable for any company as they help:

    • find areas to improve in code review processes
    • optimize peer review practices
    • avoid slowdowns and poor quality code. 

    Create your dashboard using the Bitbucket REST API or try Awesome Graphs for Bitbucket for free to discover how you can benefit from it!

    Related posts

      How to Perform Code Review and Track Team Activity in Bitbucket

      August 8, 2019
      #Case Study#How To#Bitbucket
      6 min

      Resolution Time Distribution Reports have helped a lot during the development of a high priority module to increase the reviewers, review frequency and also track team activity in Bitbucket.

      Eduard Pal, Innoface

      Innoface GmbH, one of our customers, is one of the leading providers of interfaces between PDM and ERP systems worldwide. Its applications help ensure a direct and reliable flow of information between engineering and logistics. They also offer additional products and services in order to facilitate seamless integration of the involved systems.

      Innoface product development team uses Awesome Graphs for Bitbucket mostly for making data-driven management decisions. They find it very useful to:

      • track the activity of each developer and a team in general;
      • check if code review goes according to the plans;
      • predict time to resolve pull requests.

      Let’s see how Innoface benefits from using Awesome Graphs for Bitbucket.

      Tracking team activity in Bitbucket

      Activity graph helps Innoface have a high-level overview of the total number of commits and contributors in a specific period, for example, a week. It also provides an opportunity to track the number of open pull requests from all the contributors and the repositories.

      Using this graph, the manager can assess the activeness of a contributor. This gives the manager an idea of whether an employee is following the company’s development policy or not as well.

      tracking team activity in Bitbucket

      Based on the number of open pull requests, the manager in Innoface can actually plan time to review the code and merge them. This graph also gives an opportunity to process the pull requests from the contributors that are working on the high-priority projects.

      Analyzing code review practices

      Innoface management finds Resolution Time Report very helpful when it comes to tracking the frequency of processing the pull requests.

      analyze code review in Bitbucket

      The manager uses this report in order to see the frequency with which the reviewers are processing the pull requests. Based on the data that this report provides, it’s easier to make a reasonable decision and to assign either more reviewers or increase the frequency of the reviewers, especially for the high-priority projects.

      Predicting pull requests resolution time

      Resolution Time Distribution report helps Innoface estimate resolution time for future pull requests to make more accurate release dates planning.

      pull request resolution time in Bitbucket

      The manager of a product team uses this report in addition to the Resolution Time Report in order to estimate the resolution time for future pull requests.

      Conclusion

      Awesome Graphs greatly enriches the Bitbucket user experience for Innoface. With its help, it becomes much easier to make decisions based on data and keep an eye on how efficient their development practices are.

      Read more case studies to see how our customers benefit from using Awesome Graphs for Bitbucket in their work:

      Start a free trial of Awesome Graphs

      Related posts