Skip to content
Snippets Groups Projects
Select Git revision
  • 4b83298f752e7677cf75f047c61779e0ce3fbd87
  • main default protected
  • next
  • renovate/main-ghcr.io-renovatebot-base-image-10.x
  • renovate/main-ghcr.io-containerbase-devcontainer-13.x
  • revert-31645-feat/rename-gradle-wrapper-validation-action
  • renovate/main-redis-5.x
  • fix/36615b-branch-reuse-no-cache
  • chore/punycode
  • fix/36615-branch-reuse-bug
  • refactor/pin-new-value
  • feat/36219--git-x509-signing
  • feat/structured-logger
  • hotfix/39.264.1
  • feat/skip-dangling
  • gh-readonly-queue/next/pr-36034-7a061c4ca1024a19e2c295d773d9642625d1c2be
  • hotfix/39.238.3
  • refactor/gitlab-auto-approve
  • feat/template-strings
  • gh-readonly-queue/next/pr-35654-137d934242c784e0c45d4b957362214f0eade1d7
  • fix/32307-global-extends-merging
  • 41.32.1
  • 41.32.0
  • 41.31.1
  • 41.31.0
  • 41.30.5
  • 41.30.4
  • 41.30.3
  • 41.30.2
  • 41.30.1
  • 41.30.0
  • 41.29.1
  • 41.29.0
  • 41.28.2
  • 41.28.1
  • 41.28.0
  • 41.27.1
  • 41.27.0
  • 41.26.2
  • 41.26.1
  • 41.26.0
41 results

dependency-dashboard.ts

  • dependency-dashboard.ts 11.81 KiB
    import is from '@sindresorhus/is';
    import { nameFromLevel } from 'bunyan';
    import { GlobalConfig } from '../../config/global';
    import type { RenovateConfig } from '../../config/types';
    import { getProblems, logger } from '../../logger';
    import { platform } from '../../platform';
    import { regEx } from '../../util/regex';
    import * as template from '../../util/template';
    import { BranchConfig, BranchResult } from '../types';
    
    interface DependencyDashboard {
      dependencyDashboardChecks: Record<string, string>;
      dependencyDashboardRebaseAllOpen: boolean;
    }
    
    function parseDashboardIssue(issueBody: string): DependencyDashboard {
      const checkMatch = ' - \\[x\\] <!-- ([a-zA-Z]+)-branch=([^\\s]+) -->';
      const checked = issueBody.match(regEx(checkMatch, 'g'));
      const dependencyDashboardChecks: Record<string, string> = {};
      if (checked?.length) {
        const re = regEx(checkMatch);
        checked.forEach((check) => {
          const [, type, branchName] = re.exec(check);
          dependencyDashboardChecks[branchName] = type;
        });
      }
      const checkedRebaseAll = issueBody.includes(
        ' - [x] <!-- rebase-all-open-prs -->'
      );
      let dependencyDashboardRebaseAllOpen = false;
      if (checkedRebaseAll) {
        dependencyDashboardRebaseAllOpen = true;
      }
      return { dependencyDashboardChecks, dependencyDashboardRebaseAllOpen };
    }
    
    export async function readDashboardBody(config: RenovateConfig): Promise<void> {
      config.dependencyDashboardChecks = {};
      const stringifiedConfig = JSON.stringify(config);
      if (
        config.dependencyDashboard ||
        stringifiedConfig.includes('"dependencyDashboardApproval":true') ||
        stringifiedConfig.includes('"prCreation":"approval"')
      ) {
        config.dependencyDashboardTitle =
          config.dependencyDashboardTitle || `Dependency Dashboard`;
        const issue = await platform.findIssue(config.dependencyDashboardTitle);
        if (issue) {
          config.dependencyDashboardIssue = issue.number;
          Object.assign(config, parseDashboardIssue(issue.body));
        }
      }
    }
    
    function getListItem(branch: BranchConfig, type: string): string {
      let item = ' - [ ] ';
      item += `<!-- ${type}-branch=${branch.branchName} -->`;
      if (branch.prNo) {
        item += `[${branch.prTitle}](../pull/${branch.prNo})`;
      } else {
        item += branch.prTitle;
      }
      const uniquePackages = [
        ...new Set(branch.upgrades.map((upgrade) => '`' + upgrade.depName + '`')),
      ];
      if (uniquePackages.length < 2) {
        return item + '\n';
      }
      return item + ' (' + uniquePackages.join(', ') + ')\n';
    }
    
    function appendRepoProblems(config: RenovateConfig, issueBody: string): string {
      let newIssueBody = issueBody;
      const repoProblems = new Set(
        getProblems()
          .filter(
            (problem) =>
              problem.repository === config.repository && !problem.artifactErrors
          )
          .map(
            (problem) =>
              `${nameFromLevel[problem.level].toUpperCase()}: ${problem.msg}`
          )
      );
      if (repoProblems.size) {
        newIssueBody += '## Repository problems\n\n';
        newIssueBody +=
          'These problems occurred while renovating this repository.\n\n';
        for (const repoProblem of repoProblems) {
          newIssueBody += ` - ${repoProblem}\n`;
        }
        newIssueBody += '\n';
      }
      return newIssueBody;
    }
    
    export async function ensureDependencyDashboard(
      config: RenovateConfig,
      branches: BranchConfig[]
    ): Promise<void> {
      // legacy/migrated issue
      const reuseTitle = 'Update Dependencies (Renovate Bot)';
      if (
        !(
          config.dependencyDashboard ||
          config.dependencyDashboardApproval ||
          config.packageRules?.some((rule) => rule.dependencyDashboardApproval) ||
          branches.some(
            (branch) =>
              branch.dependencyDashboardApproval ||
              branch.dependencyDashboardPrApproval
          )
        )
      ) {
        if (GlobalConfig.get('dryRun')) {
          logger.info(
            { title: config.dependencyDashboardTitle },
            'DRY-RUN: Would close Dependency Dashboard'
          );
        } else {
          logger.debug('Closing Dependency Dashboard');
          await platform.ensureIssueClosing(config.dependencyDashboardTitle);
        }
        return;
      }
      // istanbul ignore if
      if (config.repoIsOnboarded === false) {
        logger.debug('Repo is onboarding - skipping dependency dashboard');
        return;
      }
      logger.debug('Ensuring Dependency Dashboard');
      const hasBranches =
        is.nonEmptyArray(branches) &&
        branches.some((branch) => branch.result !== BranchResult.Automerged);
      if (config.dependencyDashboardAutoclose && !hasBranches) {
        if (GlobalConfig.get('dryRun')) {
          logger.info(
            { title: config.dependencyDashboardTitle },
            'DRY-RUN: Would close Dependency Dashboard'
          );
        } else {
          logger.debug('Closing Dependency Dashboard');
          await platform.ensureIssueClosing(config.dependencyDashboardTitle);
        }
        return;
      }
      let issueBody = '';
      if (config.dependencyDashboardHeader?.length) {
        issueBody +=
          template.compile(config.dependencyDashboardHeader, config) + '\n\n';
      }
    
      issueBody = appendRepoProblems(config, issueBody);
    
      const pendingApprovals = branches.filter(
        (branch) => branch.result === BranchResult.NeedsApproval
      );
      if (pendingApprovals.length) {
        issueBody += '## Pending Approval\n\n';
        issueBody += `These branches will be created by Renovate only once you click their checkbox below.\n\n`;
        for (const branch of pendingApprovals) {
          issueBody += getListItem(branch, 'approve');
        }
        issueBody += '\n';
      }
      const awaitingSchedule = branches.filter(
        (branch) => branch.result === BranchResult.NotScheduled
      );
      if (awaitingSchedule.length) {
        issueBody += '## Awaiting Schedule\n\n';
        issueBody +=
          'These updates are awaiting their schedule. Click on a checkbox to get an update now.\n';
        for (const branch of awaitingSchedule) {
          issueBody += getListItem(branch, 'unschedule');
        }
        issueBody += '\n';
      }
      const rateLimited = branches.filter(
        (branch) =>
          branch.result === BranchResult.BranchLimitReached ||
          branch.result === BranchResult.PrLimitReached ||
          branch.result === BranchResult.CommitLimitReached
      );
      if (rateLimited.length) {
        issueBody += '## Rate Limited\n\n';
        issueBody +=
          'These updates are currently rate limited. Click on a checkbox below to force their creation now.\n\n';
        for (const branch of rateLimited) {
          issueBody += getListItem(branch, 'unlimit');
        }
        issueBody += '\n';
      }
      const errorList = branches.filter(
        (branch) => branch.result === BranchResult.Error
      );
      if (errorList.length) {
        issueBody += '## Errored\n\n';
        issueBody +=
          'These updates encountered an error and will be retried. Click on a checkbox below to force a retry now.\n\n';
        for (const branch of errorList) {
          issueBody += getListItem(branch, 'retry');
        }
        issueBody += '\n';
      }
      const awaitingPr = branches.filter(
        (branch) => branch.result === BranchResult.NeedsPrApproval
      );
      if (awaitingPr.length) {
        issueBody += '## PR Creation Approval Required\n\n';
        issueBody +=
          "These branches exist but PRs won't be created until you approve them by clicking on a checkbox.\n\n";
        for (const branch of awaitingPr) {
          issueBody += getListItem(branch, 'approvePr');
        }
        issueBody += '\n';
      }
      const prEdited = branches.filter(
        (branch) => branch.result === BranchResult.PrEdited
      );
      if (prEdited.length) {
        issueBody += '## Edited/Blocked\n\n';
        issueBody += `These updates have been manually edited so Renovate will no longer make changes. To discard all commits and start over, click on a checkbox.\n\n`;
        for (const branch of prEdited) {
          issueBody += getListItem(branch, 'rebase');
        }
        issueBody += '\n';
      }
      const prPending = branches.filter(
        (branch) => branch.result === BranchResult.Pending
      );
      if (prPending.length) {
        issueBody += '## Pending Status Checks\n\n';
        issueBody += `These updates await pending status checks. To force their creation now, click the checkbox below.\n\n`;
        for (const branch of prPending) {
          issueBody += getListItem(branch, 'approvePr');
        }
        issueBody += '\n';
      }
      const prPendingBranchAutomerge = branches.filter(
        (branch) => branch.prBlockedBy === 'BranchAutomerge'
      );
      if (prPendingBranchAutomerge.length) {
        issueBody += '## Pending Branch Automerge\n\n';
        issueBody += `These updates await pending status checks before automerging. Click on a checkbox to abort the branch automerge, and create a PR instead.\n\n`;
        for (const branch of prPendingBranchAutomerge) {
          issueBody += getListItem(branch, 'approvePr');
        }
        issueBody += '\n';
      }
      const otherRes = [
        BranchResult.Pending,
        BranchResult.NeedsApproval,
        BranchResult.NeedsPrApproval,
        BranchResult.NotScheduled,
        BranchResult.PrLimitReached,
        BranchResult.CommitLimitReached,
        BranchResult.BranchLimitReached,
        BranchResult.AlreadyExisted,
        BranchResult.Error,
        BranchResult.Automerged,
        BranchResult.PrEdited,
      ];
      let inProgress = branches.filter(
        (branch) =>
          !otherRes.includes(branch.result) &&
          branch.prBlockedBy !== 'BranchAutomerge'
      );
      const otherBranches = inProgress.filter(
        (branch) => branch.prBlockedBy || !branch.prNo
      );
      // istanbul ignore if
      if (otherBranches.length) {
        issueBody += '## Other Branches\n\n';
        issueBody += `These updates are pending. To force PRs open, click the checkbox below.\n\n`;
        for (const branch of otherBranches) {
          issueBody += getListItem(branch, 'other');
        }
        issueBody += '\n';
      }
      inProgress = inProgress.filter(
        (branch) => branch.prNo && !branch.prBlockedBy
      );
      if (inProgress.length) {
        issueBody += '## Open\n\n';
        issueBody +=
          'These updates have all been created already. Click a checkbox below to force a retry/rebase of any.\n\n';
        for (const branch of inProgress) {
          issueBody += getListItem(branch, 'rebase');
        }
        if (inProgress.length > 2) {
          issueBody += ' - [ ] ';
          issueBody += '<!-- rebase-all-open-prs -->';
          issueBody += '**Click on this checkbox to rebase all open PRs at once**';
          issueBody += '\n';
        }
        issueBody += '\n';
      }
      const alreadyExisted = branches.filter(
        (branch) => branch.result === BranchResult.AlreadyExisted
      );
      if (alreadyExisted.length) {
        issueBody += '## Ignored or Blocked\n\n';
        issueBody +=
          'These are blocked by an existing closed PR and will not be recreated unless you click a checkbox below.\n\n';
        for (const branch of alreadyExisted) {
          issueBody += getListItem(branch, 'recreate');
        }
        issueBody += '\n';
      }
    
      if (!hasBranches) {
        issueBody +=
          'This repository currently has no open or pending branches.\n\n';
      }
    
      if (config.dependencyDashboardFooter?.length) {
        issueBody +=
          '---\n' +
          template.compile(config.dependencyDashboardFooter, config) +
          '\n';
      }
    
      if (config.dependencyDashboardIssue) {
        const updatedIssue = await platform.getIssue?.(
          config.dependencyDashboardIssue,
          false
        );
        if (updatedIssue) {
          const { dependencyDashboardChecks } = parseDashboardIssue(
            updatedIssue.body
          );
          for (const branchName of Object.keys(config.dependencyDashboardChecks)) {
            delete dependencyDashboardChecks[branchName];
          }
          for (const branchName of Object.keys(dependencyDashboardChecks)) {
            const checkText = `- [ ] <!-- ${dependencyDashboardChecks[branchName]}-branch=${branchName} -->`;
            issueBody = issueBody.replace(
              checkText,
              checkText.replace('[ ]', '[x]')
            );
          }
        }
      }
    
      if (GlobalConfig.get('dryRun')) {
        logger.info(
          { title: config.dependencyDashboardTitle },
          'DRY-RUN: Would ensure Dependency Dashboard'
        );
      } else {
        await platform.ensureIssue({
          title: config.dependencyDashboardTitle,
          reuseTitle,
          body: issueBody,
          labels: config.dependencyDashboardLabels,
          confidential: config.confidential,
        });
      }
    }