From 79e0d033677504546e00c961889ee51a3ef60e2f Mon Sep 17 00:00:00 2001 From: Sergei Shmakov Date: Sun, 24 Nov 2024 20:05:23 +0100 Subject: [PATCH] Groups repo+prID pairs by provider to perform the more precise search (#3788, #3795) --- .../__tests__/pullRequest.utils.test.ts | 61 +++------ src/git/models/pullRequest.ts | 3 +- src/git/models/pullRequest.utils.ts | 27 +--- src/plus/integrations/integration.ts | 3 + .../providers/__tests__/github.utils.test.ts | 80 ++++++++++++ .../providers/__tests__/gitlab.utils.test.ts | 107 ++++++++++++++++ src/plus/integrations/providers/github.ts | 6 + .../integrations/providers/github.utils.ts | 28 +++++ .../integrations/providers/github/models.ts | 8 +- src/plus/integrations/providers/gitlab.ts | 6 + .../integrations/providers/gitlab.utils.ts | 28 +++++ .../integrations/providers/gitlab/models.ts | 8 +- src/plus/launchpad/launchpadProvider.ts | 118 +++++++++++------- src/plus/launchpad/utils.ts | 8 +- 14 files changed, 365 insertions(+), 126 deletions(-) create mode 100644 src/plus/integrations/providers/__tests__/github.utils.test.ts create mode 100644 src/plus/integrations/providers/__tests__/gitlab.utils.test.ts create mode 100644 src/plus/integrations/providers/github.utils.ts create mode 100644 src/plus/integrations/providers/gitlab.utils.ts diff --git a/src/git/models/__tests__/pullRequest.utils.test.ts b/src/git/models/__tests__/pullRequest.utils.test.ts index adeaec4f240b8..59beccf302a4c 100644 --- a/src/git/models/__tests__/pullRequest.utils.test.ts +++ b/src/git/models/__tests__/pullRequest.utils.test.ts @@ -2,61 +2,28 @@ import * as assert from 'assert'; import { suite, test } from 'mocha'; import { getPullRequestIdentityValuesFromSearch } from '../pullRequest.utils'; -suite('Test GitHub PR URL parsing to identity: getPullRequestIdentityValuesFromSearch()', () => { +suite('Test PR URL parsing to identity: getPullRequestIdentityValuesFromSearch()', () => { function t(message: string, query: string, prNumber: string | undefined, ownerAndRepo?: string) { assert.deepStrictEqual( getPullRequestIdentityValuesFromSearch(query), - { - ownerAndRepo: ownerAndRepo, - prNumber: prNumber, - }, + prNumber == null + ? undefined + : { + ownerAndRepo: ownerAndRepo, + prNumber: prNumber, + }, `${message} (${JSON.stringify(query)})`, ); } - test('full URL or without protocol but with domain, should parse to ownerAndRepo and prNumber', () => { - t('full URL', 'https://github.com/eamodio/vscode-gitlens/pull/1', '1', 'eamodio/vscode-gitlens'); - t( - 'with suffix', - 'https://github.com/eamodio/vscode-gitlens/pull/1/files?diff=unified#hello', - '1', - 'eamodio/vscode-gitlens', - ); - t( - 'with query', - 'https://github.com/eamodio/vscode-gitlens/pull/1?diff=unified#hello', - '1', - 'eamodio/vscode-gitlens', - ); - - t('with anchor', 'https://github.com/eamodio/vscode-gitlens/pull/1#hello', '1', 'eamodio/vscode-gitlens'); - t('a weird suffix', 'https://github.com/eamodio/vscode-gitlens/pull/1-files', '1', 'eamodio/vscode-gitlens'); - t('numeric repo name', 'https://github.com/sergeibbb/1/pull/16', '16', 'sergeibbb/1'); - - t('no protocol with leading slash', '/github.com/sergeibbb/1/pull/16?diff=unified', '16', 'sergeibbb/1'); - t('no protocol without leading slash', 'github.com/sergeibbb/1/pull/16/files', '16', 'sergeibbb/1'); - }); - - test('no domain, should parse to ownerAndRepo and prNumber', () => { - t('with leading slash', '/sergeibbb/1/pull/16#hello', '16', 'sergeibbb/1'); - t('words in repo name', 'eamodio/vscode-gitlens/pull/1?diff=unified#hello', '1', 'eamodio/vscode-gitlens'); - t('numeric repo name', 'sergeibbb/1/pull/16/files', '16', 'sergeibbb/1'); - }); - - test('domain vs. no domain', () => { - t( - 'with anchor', - 'https://github.com/eamodio/vscode-gitlens/pull/1#hello/sergeibbb/1/pull/16', - '1', - 'eamodio/vscode-gitlens', - ); - }); + test('cannot recognize GitHub or GitLab URLs, sees only numbers', () => { + t('full URL', 'https://github.com/eamodio/vscode-gitlens/pull/16', '16'); + t('numeric repo name', 'https://github.com/sergeibbb/1/pull/16', '1'); - test('has "pull/" fragment', () => { - t('with leading slash', '/pull/16/files#hello', '16'); - t('without leading slash', 'pull/16?diff=unified#hello', '16'); - t('with numeric repo name', '1/pull/16?diff=unified#hello', '16'); - t('with double slash', '1//pull/16?diff=unified#hello', '16'); + t('no protocol', '/github.com/sergeibbb/1/pull/16?diff=unified', '1'); + t('no domain', '/sergeibbb/1/pull/16#hello', '1'); + t('domain vs. no domain', 'https://github.com/eamodio/vscode-gitlens/pull/1#hello/sergeibbb/2/pull/16', '1'); + t('has "pull/" fragment', '/pull/16/files#hello', '16'); }); test('has "/" fragment', () => { diff --git a/src/git/models/pullRequest.ts b/src/git/models/pullRequest.ts index 365b6255e44ac..384c037545653 100644 --- a/src/git/models/pullRequest.ts +++ b/src/git/models/pullRequest.ts @@ -420,11 +420,12 @@ export async function getOpenedPullRequestRepo( export function doesPullRequestSatisfyRepositoryURLIdentity( pr: EnrichablePullRequest | undefined, - { ownerAndRepo, prNumber }: PullRequestUrlIdentity, + prUrlIdentity: { [key in string]?: PullRequestUrlIdentity }, ): boolean { if (pr == null) { return false; } + const { ownerAndRepo, prNumber } = prUrlIdentity[pr.provider.id] ?? {}; const satisfiesPrNumber = prNumber != null && pr.number === parseInt(prNumber, 10); if (!satisfiesPrNumber) { return false; diff --git a/src/git/models/pullRequest.utils.ts b/src/git/models/pullRequest.utils.ts index 51744f239d39e..71cdc7f47b776 100644 --- a/src/git/models/pullRequest.utils.ts +++ b/src/git/models/pullRequest.utils.ts @@ -8,31 +8,16 @@ export interface PullRequestUrlIdentity { provider?: HostingIntegrationId; ownerAndRepo?: string; - prNumber?: string; + prNumber: string; } -export function getPullRequestIdentityValuesFromSearch(search: string): PullRequestUrlIdentity { - let ownerAndRepo: string | undefined = undefined; +export function getPullRequestIdentityValuesFromSearch(search: string): PullRequestUrlIdentity | undefined { + const ownerAndRepo: string | undefined = undefined; let prNumber: string | undefined = undefined; - let match = search.match(/([^/]+\/[^/]+)\/(?:pull|-\/merge_requests)\/(\d+)/); // with org and rep name + let match = search.match(/(?:\/)(\d+)/); // any number starting with "/" if (match != null) { - ownerAndRepo = match[1]; - prNumber = match[2]; - } - - if (prNumber == null) { - match = search.match(/(?:\/|^)(?:pull|-\/merge_requests)\/(\d+)/); // without repo name - if (match != null) { - prNumber = match[1]; - } - } - - if (prNumber == null) { - match = search.match(/(?:\/)(\d+)/); // any number starting with "/" - if (match != null) { - prNumber = match[1]; - } + prNumber = match[1]; } if (prNumber == null) { @@ -42,5 +27,5 @@ export function getPullRequestIdentityValuesFromSearch(search: string): PullRequ } } - return { ownerAndRepo: ownerAndRepo, prNumber: prNumber }; + return prNumber == null ? undefined : { ownerAndRepo: ownerAndRepo, prNumber: prNumber }; } diff --git a/src/plus/integrations/integration.ts b/src/plus/integrations/integration.ts index c7aabf3a2c54a..e8099375fc646 100644 --- a/src/plus/integrations/integration.ts +++ b/src/plus/integrations/integration.ts @@ -16,6 +16,7 @@ import type { PullRequestState, SearchedPullRequest, } from '../../git/models/pullRequest'; +import type { PullRequestUrlIdentity } from '../../git/models/pullRequest.utils'; import type { RepositoryMetadata } from '../../git/models/repositoryMetadata'; import { showIntegrationDisconnectedTooManyFailedRequestsWarningMessage } from '../../messages'; import { gate } from '../../system/decorators/gate'; @@ -1361,4 +1362,6 @@ export abstract class HostingIntegration< repos?: T[], cancellation?: CancellationToken, ): Promise; + + getPullRequestIdentityValuesFromSearch?(search: string): PullRequestUrlIdentity | undefined; } diff --git a/src/plus/integrations/providers/__tests__/github.utils.test.ts b/src/plus/integrations/providers/__tests__/github.utils.test.ts new file mode 100644 index 0000000000000..2042148886229 --- /dev/null +++ b/src/plus/integrations/providers/__tests__/github.utils.test.ts @@ -0,0 +1,80 @@ +import * as assert from 'assert'; +import { suite, test } from 'mocha'; +import { getGitHubPullRequestIdentityValuesFromSearch } from '../github.utils'; + +suite('Test GitHub PR URL parsing to identity: getPullRequestIdentityValuesFromSearch()', () => { + function t(message: string, query: string, prNumber: string | undefined, ownerAndRepo?: string) { + assert.deepStrictEqual( + getGitHubPullRequestIdentityValuesFromSearch(query), + prNumber == null + ? undefined + : { + ownerAndRepo: ownerAndRepo, + prNumber: prNumber, + }, + `${message} (${JSON.stringify(query)})`, + ); + } + + test('full URL or without protocol but with domain, should parse to ownerAndRepo and prNumber', () => { + t('full URL', 'https://github.com/eamodio/vscode-gitlens/pull/1', '1', 'eamodio/vscode-gitlens'); + t( + 'with suffix', + 'https://github.com/eamodio/vscode-gitlens/pull/1/files?diff=unified#hello', + '1', + 'eamodio/vscode-gitlens', + ); + t( + 'with query', + 'https://github.com/eamodio/vscode-gitlens/pull/1?diff=unified#hello', + '1', + 'eamodio/vscode-gitlens', + ); + + t('with anchor', 'https://github.com/eamodio/vscode-gitlens/pull/1#hello', '1', 'eamodio/vscode-gitlens'); + t('a weird suffix', 'https://github.com/eamodio/vscode-gitlens/pull/1-files', '1', 'eamodio/vscode-gitlens'); + t('numeric repo name', 'https://github.com/sergeibbb/1/pull/16', '16', 'sergeibbb/1'); + + t('no protocol with leading slash', '/github.com/sergeibbb/1/pull/16?diff=unified', '16', 'sergeibbb/1'); + t('no protocol without leading slash', 'github.com/sergeibbb/1/pull/16/files', '16', 'sergeibbb/1'); + }); + + test('no domain, should parse to ownerAndRepo and prNumber', () => { + t('with leading slash', '/sergeibbb/1/pull/16#hello', '16', 'sergeibbb/1'); + t('words in repo name', 'eamodio/vscode-gitlens/pull/1?diff=unified#hello', '1', 'eamodio/vscode-gitlens'); + t('numeric repo name', 'sergeibbb/1/pull/16/files', '16', 'sergeibbb/1'); + }); + + test('domain vs. no domain', () => { + t( + 'with anchor', + 'https://github.com/eamodio/vscode-gitlens/pull/1#hello/sergeibbb/1/pull/16', + '1', + 'eamodio/vscode-gitlens', + ); + }); + + test('numbers', () => { + t('has "pull/" fragment', '/pull/16/files#hello', '16'); + t('has "pull/" fragment with double slash', '1//pull/16?diff=unified#hello', '16'); + t('with leading slash', '/16/files#hello', '16'); + t('just a number', '16', '16'); + t('with a hash', '#16', '16'); + }); + + test('does not match', () => { + t('without leading slash', '16?diff=unified#hello', undefined); + t('with leading hash', '/#16/files#hello', undefined); + t('number is a part of a word', 'hello16', undefined); + t('number is a part of a word', '16hello', undefined); + + t('GitLab', 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/16', '16'); + + t('with a number', '1/16?diff=unified#hello', '16'); + t('with a number and slash', '/1/16?diff=unified#hello', '1'); + t('with a word', 'anything/16?diff=unified#hello', '16'); + + t('with a wrong character leading to pull', 'sergeibbb/1/-pull/16?diff=unified#hello', '1'); + t('with a wrong character leading to pull', 'sergeibbb/1-pull/16?diff=unified#hello', '1'); + }); +}); diff --git a/src/plus/integrations/providers/__tests__/gitlab.utils.test.ts b/src/plus/integrations/providers/__tests__/gitlab.utils.test.ts new file mode 100644 index 0000000000000..81cbca0ab92c1 --- /dev/null +++ b/src/plus/integrations/providers/__tests__/gitlab.utils.test.ts @@ -0,0 +1,107 @@ +import * as assert from 'assert'; +import { suite, test } from 'mocha'; +import { getGitLabPullRequestIdentityValuesFromSearch } from '../gitlab.utils'; + +suite('Test GitLab PR URL parsing to identity: getPullRequestIdentityValuesFromSearch()', () => { + function t(message: string, query: string, prNumber: string | undefined, ownerAndRepo?: string) { + assert.deepStrictEqual( + getGitLabPullRequestIdentityValuesFromSearch(query), + prNumber == null + ? undefined + : { + ownerAndRepo: ownerAndRepo, + prNumber: prNumber, + }, + `${message} (${JSON.stringify(query)})`, + ); + } + + test('full URL or without protocol but with domain, should parse to ownerAndRepo and prNumber', () => { + t('full URL', 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/1', '1', 'eamodio/vscode-gitlens'); + t( + 'with suffix', + 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/1/files?diff=unified#hello', + '1', + 'eamodio/vscode-gitlens', + ); + t( + 'with query', + 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/1?diff=unified#hello', + '1', + 'eamodio/vscode-gitlens', + ); + + t( + 'with anchor', + 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/1#hello', + '1', + 'eamodio/vscode-gitlens', + ); + t( + 'a weird suffix', + 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/1-files', + '1', + 'eamodio/vscode-gitlens', + ); + t('numeric repo name', 'https://gitlab.com/sergeibbb/1/-/merge_requests/16', '16', 'sergeibbb/1'); + + t( + 'no protocol with leading slash', + '/gitlab.com/sergeibbb/1/-/merge_requests/16?diff=unified', + '16', + 'sergeibbb/1', + ); + t('no protocol without leading slash', 'gitlab.com/sergeibbb/1/-/merge_requests/16/files', '16', 'sergeibbb/1'); + }); + test('no domain, should parse to ownerAndRepo and prNumber', () => { + t('with leading slash', '/sergeibbb/1/-/merge_requests/16#hello', '16', 'sergeibbb/1'); + t( + 'words in repo name', + 'eamodio/vscode-gitlens/-/merge_requests/1?diff=unified#hello', + '1', + 'eamodio/vscode-gitlens', + ); + t('numeric repo name', 'sergeibbb/1/-/merge_requests/16/files', '16', 'sergeibbb/1'); + }); + + test('domain vs. no domain', () => { + t( + 'with anchor', + 'https://gitlab.com/eamodio/vscode-gitlens/-/merge_requests/1#hello/sergeibbb/1/-/merge_requests/16', + '1', + 'eamodio/vscode-gitlens', + ); + }); + + test('numbers', () => { + t('has "-/merge_requests/" fragment', '/-/merge_requests/16/files#hello', '16'); + t('has "-/merge_requests/" fragment with double slash', '1//-/merge_requests/16?diff=unified#hello', '16'); + t('with leading slash', '/16/files#hello', '16'); + t('just a number', '16', '16'); + t('with a hash', '#16', '16'); + }); + + test('does not match', () => { + t('without leading slash', '16?diff=unified#hello', undefined); + t('with leading hash', '/#16/files#hello', undefined); + t('number is a part of a word', 'hello16', undefined); + t('number is a part of a word', '16hello', undefined); + + t('GitHub', 'https://github.com/eamodio/vscode-gitlens/pull/16', '16'); + + t('with a number', '1/16?diff=unified#hello', '16'); + t('with a number and slash', '/1/16?diff=unified#hello', '1'); + t('with a word', 'anything/16?diff=unified#hello', '16'); + + t( + 'with a wrong character leading to "-/merge_requests/"', + 'sergeibbb/1/--/merge_requests/16?diff=unified#hello', + '1', + ); + t( + 'with a wrong character leading to "-/merge_requests/"', + 'sergeibbb/1--/merge_requests/16?diff=unified#hello', + '1', + ); + }); +}); diff --git a/src/plus/integrations/providers/github.ts b/src/plus/integrations/providers/github.ts index a5d73cd3402b8..c73996826245a 100644 --- a/src/plus/integrations/providers/github.ts +++ b/src/plus/integrations/providers/github.ts @@ -11,6 +11,7 @@ import type { PullRequestState, SearchedPullRequest, } from '../../../git/models/pullRequest'; +import type { PullRequestUrlIdentity } from '../../../git/models/pullRequest.utils'; import type { RepositoryMetadata } from '../../../git/models/repositoryMetadata'; import { log } from '../../../system/decorators/log'; import { ensurePaidPlan } from '../../utils'; @@ -20,6 +21,7 @@ import type { } from '../authentication/integrationAuthentication'; import type { RepositoryDescriptor, SupportedIntegrationIds } from '../integration'; import { HostingIntegration } from '../integration'; +import { getGitHubPullRequestIdentityValuesFromSearch } from './github.utils'; import { providersMetadata } from './models'; import type { ProvidersApi } from './providersApi'; @@ -292,6 +294,10 @@ export class GitHubIntegration extends GitHubIntegrationBase { diff --git a/src/plus/integrations/providers/github.utils.ts b/src/plus/integrations/providers/github.utils.ts new file mode 100644 index 0000000000000..ed6854e69cbbb --- /dev/null +++ b/src/plus/integrations/providers/github.utils.ts @@ -0,0 +1,28 @@ +// GitHub provider: github.ts pulls many dependencies through Container and some of them break the unit tests. +// That's why this file has been created that can collect more simple functions which +// don't require Container and can be tested. + +import type { PullRequestUrlIdentity } from '../../../git/models/pullRequest.utils'; +import { getPullRequestIdentityValuesFromSearch } from '../../../git/models/pullRequest.utils'; + +export function getGitHubPullRequestIdentityValuesFromSearch(search: string): PullRequestUrlIdentity | undefined { + let ownerAndRepo: string | undefined = undefined; + let prNumber: string | undefined = undefined; + + let match = search.match(/([^/]+\/[^/]+)\/(?:pull)\/(\d+)/); // with org and rep name + if (match != null) { + ownerAndRepo = match[1]; + prNumber = match[2]; + } + + if (prNumber == null) { + match = search.match(/(?:\/|^)(?:pull)\/(\d+)/); // without repo name + if (match != null) { + prNumber = match[1]; + } + } + + return prNumber != null + ? { ownerAndRepo: ownerAndRepo, prNumber: prNumber } + : getPullRequestIdentityValuesFromSearch(search); +} diff --git a/src/plus/integrations/providers/github/models.ts b/src/plus/integrations/providers/github/models.ts index 99b3f55c849ac..76022bd898755 100644 --- a/src/plus/integrations/providers/github/models.ts +++ b/src/plus/integrations/providers/github/models.ts @@ -519,11 +519,13 @@ export function isMaybeGitHubPullRequestUrl(url: string): boolean { return prUrlRegex.test(url); } -export function getGitHubPullRequestIdentityFromMaybeUrl(url: string): RequireSome { - if (url == null) return { prNumber: undefined, ownerAndRepo: undefined, provider: HostingIntegrationId.GitHub }; +export function getGitHubPullRequestIdentityFromMaybeUrl( + url: string, +): undefined | RequireSome { + if (url == null) return undefined; const match = prUrlRegex.exec(url); - if (match == null) return { prNumber: undefined, ownerAndRepo: undefined, provider: HostingIntegrationId.GitHub }; + if (match == null) return undefined; return { prNumber: match[2], ownerAndRepo: match[1], provider: HostingIntegrationId.GitHub }; } diff --git a/src/plus/integrations/providers/gitlab.ts b/src/plus/integrations/providers/gitlab.ts index 3eb7a8404e026..6339773e21c26 100644 --- a/src/plus/integrations/providers/gitlab.ts +++ b/src/plus/integrations/providers/gitlab.ts @@ -12,6 +12,7 @@ import type { PullRequestState, SearchedPullRequest, } from '../../../git/models/pullRequest'; +import type { PullRequestUrlIdentity } from '../../../git/models/pullRequest.utils'; import type { RepositoryMetadata } from '../../../git/models/repositoryMetadata'; import { log } from '../../../system/decorators/log'; import { uniqueBy } from '../../../system/iterable'; @@ -23,6 +24,7 @@ import type { import type { RepositoryDescriptor } from '../integration'; import { HostingIntegration } from '../integration'; import { fromGitLabMergeRequestProvidersApi } from './gitlab/models'; +import { getGitLabPullRequestIdentityValuesFromSearch } from './gitlab.utils'; import type { ProviderPullRequest } from './models'; import { ProviderPullRequestReviewState, providersMetadata, toSearchedIssue } from './models'; import type { ProvidersApi } from './providersApi'; @@ -393,6 +395,10 @@ abstract class GitLabIntegrationBase< username: currentUser.username || undefined, }; } + + override getPullRequestIdentityValuesFromSearch(search: string): PullRequestUrlIdentity | undefined { + return getGitLabPullRequestIdentityValuesFromSearch(search); + } } export class GitLabIntegration extends GitLabIntegrationBase { diff --git a/src/plus/integrations/providers/gitlab.utils.ts b/src/plus/integrations/providers/gitlab.utils.ts new file mode 100644 index 0000000000000..bcd8711de82bf --- /dev/null +++ b/src/plus/integrations/providers/gitlab.utils.ts @@ -0,0 +1,28 @@ +// GitLab provider: gitlab.ts pulls many dependencies through Container and some of them break the unit tests. +// That's why this file has been created that can collect more simple functions which +// don't require Container and can be tested. + +import type { PullRequestUrlIdentity } from '../../../git/models/pullRequest.utils'; +import { getPullRequestIdentityValuesFromSearch } from '../../../git/models/pullRequest.utils'; + +export function getGitLabPullRequestIdentityValuesFromSearch(search: string): PullRequestUrlIdentity | undefined { + let ownerAndRepo: string | undefined = undefined; + let prNumber: string | undefined = undefined; + + let match = search.match(/([^/]+\/[^/]+)\/(?:-\/merge_requests)\/(\d+)/); // with org and rep name + if (match != null) { + ownerAndRepo = match[1]; + prNumber = match[2]; + } + + if (prNumber == null) { + match = search.match(/(?:\/|^)(?:-\/merge_requests)\/(\d+)/); // without repo name + if (match != null) { + prNumber = match[1]; + } + } + + return prNumber != null + ? { ownerAndRepo: ownerAndRepo, prNumber: prNumber } + : getPullRequestIdentityValuesFromSearch(search); +} diff --git a/src/plus/integrations/providers/gitlab/models.ts b/src/plus/integrations/providers/gitlab/models.ts index be86578ff073c..75a53605558d5 100644 --- a/src/plus/integrations/providers/gitlab/models.ts +++ b/src/plus/integrations/providers/gitlab/models.ts @@ -169,11 +169,13 @@ export function isMaybeGitLabPullRequestUrl(url: string): boolean { return prUrlRegex.test(url); } -export function getGitLabPullRequestIdentityFromMaybeUrl(url: string): RequireSome { - if (url == null) return { prNumber: undefined, ownerAndRepo: undefined, provider: HostingIntegrationId.GitLab }; +export function getGitLabPullRequestIdentityFromMaybeUrl( + url: string, +): undefined | RequireSome { + if (url == null) return undefined; const match = prUrlRegex.exec(url); - if (match == null) return { prNumber: undefined, ownerAndRepo: undefined, provider: HostingIntegrationId.GitLab }; + if (match == null) return undefined; return { prNumber: match[2], ownerAndRepo: match[1], provider: HostingIntegrationId.GitLab }; } diff --git a/src/plus/launchpad/launchpadProvider.ts b/src/plus/launchpad/launchpadProvider.ts index 3e87a0c31af38..ccf7db8c1aa22 100644 --- a/src/plus/launchpad/launchpadProvider.ts +++ b/src/plus/launchpad/launchpadProvider.ts @@ -328,7 +328,8 @@ export class LaunchpadProvider implements Disposable { // The current idea is that we should iterate the connected integrations and apply their parsing. // Probably we even want to build a map like this: { integrationId: identity } // Then we iterate connected integrations and search in each of them with the corresponding identity. - const prUrlIdentity = getPullRequestIdentityFromMaybeUrl(search); + const prUrlIdentity: { [key in IntegrationId]?: PullRequestUrlIdentity } = + await this.getPullRequestIdentityFromMaybeUrl(search); const result: { readonly value: SearchedPullRequest[]; duration: number } = { value: [], duration: 0, @@ -336,20 +337,56 @@ export class LaunchpadProvider implements Disposable { const connectedIntegrations = await this.getConnectedIntegrations(); + const findByUrlIdentity = async ( + integration: HostingIntegration, + ): Promise> => { + const { provider, ownerAndRepo, prNumber } = prUrlIdentity[integration.id] ?? {}; + if (provider != null && prNumber != null && ownerAndRepo != null) { + const [owner, repo] = ownerAndRepo.split('/', 2); + // TODO: This needs to be generalized to work outside of GitHub/GitLab + const descriptor: GitHubRepositoryDescriptor | GitLabRepositoryDescriptor = { + key: ownerAndRepo, + owner: owner, + name: repo, + }; + const pr = await withDurationAndSlowEventOnTimeout( + integration?.getPullRequest(descriptor, prNumber), + 'getPullRequest', + this.container, + ); + if (pr?.value != null) { + return { value: [{ pullRequest: pr.value, reasons: [] }], duration: pr.duration }; + } + } + return undefined; + }; + + const findByQuery = async ( + integration: HostingIntegration, + ): Promise> => { + const prs = await withDurationAndSlowEventOnTimeout( + integration?.searchPullRequests(search, undefined, cancellation), // + 'searchPullRequests', + this.container, + ); + if (prs != null) { + return { value: prs.value?.map(pr => ({ pullRequest: pr, reasons: [] })), duration: prs.duration }; + } + return undefined; + }; + + const hasUrlIdentity = Object.keys(prUrlIdentity).length > 0; + const searchIntegrationPRs = hasUrlIdentity ? findByUrlIdentity : findByQuery; + await Promise.allSettled( [...connectedIntegrations.keys()] .filter( (id: IntegrationId): id is SupportedLaunchpadIntegrationIds => (connectedIntegrations.get(id) && isSupportedLaunchpadIntegrationId(id)) ?? false, ) - .map(async (id: HostingIntegrationId) => { + .map(async (id: SupportedLaunchpadIntegrationIds) => { const integration = await this.container.integrations.get(id); - const searchResult = await this.searchIntegrationPRs( - search, - prUrlIdentity, - integration, - cancellation, - ); + const searchResult = await searchIntegrationPRs(integration); const prs = searchResult?.value; if (prs) { result.value?.push(...prs); @@ -363,45 +400,6 @@ export class LaunchpadProvider implements Disposable { }; } - private async searchIntegrationPRs( - search: string, - { ownerAndRepo, prNumber, provider }: PullRequestUrlIdentity, - integration: HostingIntegration, - cancellation: CancellationToken | undefined, - ): Promise> { - let result: TimedResult | undefined; - if (provider != null && prNumber != null && ownerAndRepo != null) { - // TODO: This needs to be generalized to work outside of GitHub/GitLab - const integration = await this.container.integrations.get(provider); - const [owner, repo] = ownerAndRepo.split('/', 2); - const descriptor: GitHubRepositoryDescriptor | GitLabRepositoryDescriptor = { - key: ownerAndRepo, - owner: owner, - name: repo, - }; - const pr = await withDurationAndSlowEventOnTimeout( - integration?.getPullRequest(descriptor, prNumber), - 'getPullRequest', - this.container, - ); - if (pr?.value != null) { - result = { value: [{ pullRequest: pr.value, reasons: [] }], duration: pr.duration }; - return result; - } - } else { - const prs = await withDurationAndSlowEventOnTimeout( - integration?.searchPullRequests(search, undefined, cancellation), // - 'searchPullRequests', - this.container, - ); - if (prs != null) { - result = { value: prs.value?.map(pr => ({ pullRequest: pr, reasons: [] })), duration: prs.duration }; - return result; - } - } - return undefined; - } - private _enrichedItems: CachedLaunchpadPromise> | undefined; @debug({ args: { 0: o => `force=${o?.force}` } }) private async getEnrichedItems(options?: { cancellation?: CancellationToken; force?: boolean }) { @@ -699,6 +697,32 @@ export class LaunchpadProvider implements Disposable { return repoRemotes; } + async getPullRequestIdentityFromMaybeUrl(search: string): Promise<{ + [key in SupportedLaunchpadIntegrationIds]?: PullRequestUrlIdentity; + }> { + const result: Partial> = {}; + const fullResult: Partial> = {}; + let hasFullResult = false; + + const connectedIntegrations = await this.getConnectedIntegrations(); + + for (const integrationId of supportedLaunchpadIntegrations) { + if (connectedIntegrations.get(integrationId)) { + const integration = await this.container.integrations.get(integrationId); + const prIdentity = integration.getPullRequestIdentityValuesFromSearch?.(search); + if (prIdentity) { + result[integrationId] = prIdentity; + } + if (prIdentity?.ownerAndRepo != null) { + fullResult[integrationId] = prIdentity; + hasFullResult = true; + } + } + } + + return hasFullResult ? fullResult : result; + } + @gate(o => `${o?.force ?? false}`) @log({ args: { 0: o => `force=${o?.force}`, 1: false } }) async getCategorizedItems( diff --git a/src/plus/launchpad/utils.ts b/src/plus/launchpad/utils.ts index 35f9c2b249ca5..c3dbaa7ea66d3 100644 --- a/src/plus/launchpad/utils.ts +++ b/src/plus/launchpad/utils.ts @@ -31,12 +31,12 @@ export function isMaybeSupportedLaunchpadPullRequestSearchUrl(search: string): b } // TODO: Needs to be generalized for other providers -export function getPullRequestIdentityFromMaybeUrl(url: string): PullRequestUrlIdentity { +export function getPullRequestIdentityFromMaybeUrl(url: string): PullRequestUrlIdentity | undefined { const github = getGitHubPullRequestIdentityFromMaybeUrl(url); - if (github.prNumber != null) return github; + if (github?.prNumber != null) return github; const gitlab = getGitLabPullRequestIdentityFromMaybeUrl(url); - if (gitlab.prNumber != null) return gitlab; + if (gitlab?.prNumber != null) return gitlab; - return { prNumber: undefined, ownerAndRepo: undefined, provider: undefined }; + return undefined; }