@backstage/integration 1.15.1-next.0 → 1.15.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (88) hide show
  1. package/CHANGELOG.md +20 -0
  2. package/dist/ScmIntegrations.cjs.js +110 -0
  3. package/dist/ScmIntegrations.cjs.js.map +1 -0
  4. package/dist/awsCodeCommit/AwsCodeCommitIntegration.cjs.js +58 -0
  5. package/dist/awsCodeCommit/AwsCodeCommitIntegration.cjs.js.map +1 -0
  6. package/dist/awsCodeCommit/config.cjs.js +26 -0
  7. package/dist/awsCodeCommit/config.cjs.js.map +1 -0
  8. package/dist/awsS3/AwsS3Integration.cjs.js +38 -0
  9. package/dist/awsS3/AwsS3Integration.cjs.js.map +1 -0
  10. package/dist/awsS3/config.cjs.js +53 -0
  11. package/dist/awsS3/config.cjs.js.map +1 -0
  12. package/dist/azure/AzureIntegration.cjs.js +62 -0
  13. package/dist/azure/AzureIntegration.cjs.js.map +1 -0
  14. package/dist/azure/AzureUrl.cjs.js +189 -0
  15. package/dist/azure/AzureUrl.cjs.js.map +1 -0
  16. package/dist/azure/CachedAzureDevOpsCredentialsProvider.cjs.js +76 -0
  17. package/dist/azure/CachedAzureDevOpsCredentialsProvider.cjs.js.map +1 -0
  18. package/dist/azure/DefaultAzureDevOpsCredentialsProvider.cjs.js +75 -0
  19. package/dist/azure/DefaultAzureDevOpsCredentialsProvider.cjs.js.map +1 -0
  20. package/dist/azure/config.cjs.js +153 -0
  21. package/dist/azure/config.cjs.js.map +1 -0
  22. package/dist/azure/core.cjs.js +18 -0
  23. package/dist/azure/core.cjs.js.map +1 -0
  24. package/dist/azure/deprecated.cjs.js +26 -0
  25. package/dist/azure/deprecated.cjs.js.map +1 -0
  26. package/dist/bitbucket/BitbucketIntegration.cjs.js +65 -0
  27. package/dist/bitbucket/BitbucketIntegration.cjs.js.map +1 -0
  28. package/dist/bitbucket/config.cjs.js +47 -0
  29. package/dist/bitbucket/config.cjs.js.map +1 -0
  30. package/dist/bitbucket/core.cjs.js +95 -0
  31. package/dist/bitbucket/core.cjs.js.map +1 -0
  32. package/dist/bitbucketCloud/BitbucketCloudIntegration.cjs.js +54 -0
  33. package/dist/bitbucketCloud/BitbucketCloudIntegration.cjs.js.map +1 -0
  34. package/dist/bitbucketCloud/config.cjs.js +30 -0
  35. package/dist/bitbucketCloud/config.cjs.js.map +1 -0
  36. package/dist/bitbucketCloud/core.cjs.js +78 -0
  37. package/dist/bitbucketCloud/core.cjs.js.map +1 -0
  38. package/dist/bitbucketServer/BitbucketServerIntegration.cjs.js +48 -0
  39. package/dist/bitbucketServer/BitbucketServerIntegration.cjs.js.map +1 -0
  40. package/dist/bitbucketServer/config.cjs.js +36 -0
  41. package/dist/bitbucketServer/config.cjs.js.map +1 -0
  42. package/dist/bitbucketServer/core.cjs.js +73 -0
  43. package/dist/bitbucketServer/core.cjs.js.map +1 -0
  44. package/dist/gerrit/GerritIntegration.cjs.js +52 -0
  45. package/dist/gerrit/GerritIntegration.cjs.js.map +1 -0
  46. package/dist/gerrit/config.cjs.js +60 -0
  47. package/dist/gerrit/config.cjs.js.map +1 -0
  48. package/dist/gerrit/core.cjs.js +179 -0
  49. package/dist/gerrit/core.cjs.js.map +1 -0
  50. package/dist/gitea/GiteaIntegration.cjs.js +34 -0
  51. package/dist/gitea/GiteaIntegration.cjs.js.map +1 -0
  52. package/dist/gitea/config.cjs.js +34 -0
  53. package/dist/gitea/config.cjs.js.map +1 -0
  54. package/dist/gitea/core.cjs.js +59 -0
  55. package/dist/gitea/core.cjs.js.map +1 -0
  56. package/dist/github/DefaultGithubCredentialsProvider.cjs.js +54 -0
  57. package/dist/github/DefaultGithubCredentialsProvider.cjs.js.map +1 -0
  58. package/dist/github/GithubIntegration.cjs.js +51 -0
  59. package/dist/github/GithubIntegration.cjs.js.map +1 -0
  60. package/dist/github/SingleInstanceGithubCredentialsProvider.cjs.js +211 -0
  61. package/dist/github/SingleInstanceGithubCredentialsProvider.cjs.js.map +1 -0
  62. package/dist/github/config.cjs.js +55 -0
  63. package/dist/github/config.cjs.js.map +1 -0
  64. package/dist/github/core.cjs.js +46 -0
  65. package/dist/github/core.cjs.js.map +1 -0
  66. package/dist/gitlab/DefaultGitlabCredentialsProvider.cjs.js +30 -0
  67. package/dist/gitlab/DefaultGitlabCredentialsProvider.cjs.js.map +1 -0
  68. package/dist/gitlab/GitLabIntegration.cjs.js +41 -0
  69. package/dist/gitlab/GitLabIntegration.cjs.js.map +1 -0
  70. package/dist/gitlab/SingleInstanceGitlabCredentialsProvider.cjs.js +24 -0
  71. package/dist/gitlab/SingleInstanceGitlabCredentialsProvider.cjs.js.map +1 -0
  72. package/dist/gitlab/config.cjs.js +60 -0
  73. package/dist/gitlab/config.cjs.js.map +1 -0
  74. package/dist/gitlab/core.cjs.js +84 -0
  75. package/dist/gitlab/core.cjs.js.map +1 -0
  76. package/dist/googleGcs/config.cjs.js +16 -0
  77. package/dist/googleGcs/config.cjs.js.map +1 -0
  78. package/dist/harness/HarnessIntegration.cjs.js +34 -0
  79. package/dist/harness/HarnessIntegration.cjs.js.map +1 -0
  80. package/dist/harness/config.cjs.js +22 -0
  81. package/dist/harness/config.cjs.js.map +1 -0
  82. package/dist/harness/core.cjs.js +101 -0
  83. package/dist/harness/core.cjs.js.map +1 -0
  84. package/dist/helpers.cjs.js +71 -0
  85. package/dist/helpers.cjs.js.map +1 -0
  86. package/dist/index.cjs.js +124 -2451
  87. package/dist/index.cjs.js.map +1 -1
  88. package/package.json +6 -6
@@ -0,0 +1,95 @@
1
+ 'use strict';
2
+
3
+ var fetch = require('cross-fetch');
4
+ var parseGitUrl = require('git-url-parse');
5
+
6
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
7
+
8
+ var fetch__default = /*#__PURE__*/_interopDefaultCompat(fetch);
9
+ var parseGitUrl__default = /*#__PURE__*/_interopDefaultCompat(parseGitUrl);
10
+
11
+ async function getBitbucketDefaultBranch(url, config) {
12
+ const { name: repoName, owner: project, resource } = parseGitUrl__default.default(url);
13
+ const isHosted = resource === "bitbucket.org";
14
+ let branchUrl = isHosted ? `${config.apiBaseUrl}/repositories/${project}/${repoName}` : `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/default-branch`;
15
+ let response = await fetch__default.default(branchUrl, getBitbucketRequestOptions(config));
16
+ if (response.status === 404 && !isHosted) {
17
+ branchUrl = `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/branches/default`;
18
+ response = await fetch__default.default(branchUrl, getBitbucketRequestOptions(config));
19
+ }
20
+ if (!response.ok) {
21
+ const message = `Failed to retrieve default branch from ${branchUrl}, ${response.status} ${response.statusText}`;
22
+ throw new Error(message);
23
+ }
24
+ let defaultBranch;
25
+ if (isHosted) {
26
+ const repoInfo = await response.json();
27
+ defaultBranch = repoInfo.mainbranch.name;
28
+ } else {
29
+ const { displayId } = await response.json();
30
+ defaultBranch = displayId;
31
+ }
32
+ if (!defaultBranch) {
33
+ throw new Error(
34
+ `Failed to read default branch from ${branchUrl}. Response ${response.status} ${response.json()}`
35
+ );
36
+ }
37
+ return defaultBranch;
38
+ }
39
+ async function getBitbucketDownloadUrl(url, config) {
40
+ const {
41
+ name: repoName,
42
+ owner: project,
43
+ ref,
44
+ protocol,
45
+ resource,
46
+ filepath
47
+ } = parseGitUrl__default.default(url);
48
+ const isHosted = resource === "bitbucket.org";
49
+ let branch = ref;
50
+ if (!branch) {
51
+ branch = await getBitbucketDefaultBranch(url, config);
52
+ }
53
+ const path = filepath ? `&path=${encodeURIComponent(decodeURIComponent(filepath))}` : "";
54
+ const archiveUrl = isHosted ? `${protocol}://${resource}/${project}/${repoName}/get/${branch}.tar.gz` : `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/archive?format=tgz&at=${branch}&prefix=${project}-${repoName}${path}`;
55
+ return archiveUrl;
56
+ }
57
+ function getBitbucketFileFetchUrl(url, config) {
58
+ try {
59
+ const { owner, name, ref, filepathtype, filepath } = parseGitUrl__default.default(url);
60
+ if (!owner || !name || filepathtype !== "browse" && filepathtype !== "raw" && filepathtype !== "src") {
61
+ throw new Error("Invalid Bitbucket URL or file path");
62
+ }
63
+ const pathWithoutSlash = filepath.replace(/^\//, "");
64
+ if (config.host === "bitbucket.org") {
65
+ if (!ref) {
66
+ throw new Error("Invalid Bitbucket URL or file path");
67
+ }
68
+ return `${config.apiBaseUrl}/repositories/${owner}/${name}/src/${ref}/${pathWithoutSlash}`;
69
+ }
70
+ return `${config.apiBaseUrl}/projects/${owner}/repos/${name}/raw/${pathWithoutSlash}?at=${ref}`;
71
+ } catch (e) {
72
+ throw new Error(`Incorrect URL: ${url}, ${e}`);
73
+ }
74
+ }
75
+ function getBitbucketRequestOptions(config) {
76
+ const headers = {};
77
+ if (config.token) {
78
+ headers.Authorization = `Bearer ${config.token}`;
79
+ } else if (config.username && config.appPassword) {
80
+ const buffer = Buffer.from(
81
+ `${config.username}:${config.appPassword}`,
82
+ "utf8"
83
+ );
84
+ headers.Authorization = `Basic ${buffer.toString("base64")}`;
85
+ }
86
+ return {
87
+ headers
88
+ };
89
+ }
90
+
91
+ exports.getBitbucketDefaultBranch = getBitbucketDefaultBranch;
92
+ exports.getBitbucketDownloadUrl = getBitbucketDownloadUrl;
93
+ exports.getBitbucketFileFetchUrl = getBitbucketFileFetchUrl;
94
+ exports.getBitbucketRequestOptions = getBitbucketRequestOptions;
95
+ //# sourceMappingURL=core.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"core.cjs.js","sources":["../../src/bitbucket/core.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'cross-fetch';\nimport parseGitUrl from 'git-url-parse';\nimport { BitbucketIntegrationConfig } from './config';\n\n/**\n * Given a URL pointing to a path on a provider, returns the default branch.\n *\n * @param url - A URL pointing to a path\n * @param config - The relevant provider config\n * @public\n * @deprecated no longer in use, bitbucket integration replaced by integrations bitbucketCloud and bitbucketServer.\n */\nexport async function getBitbucketDefaultBranch(\n url: string,\n config: BitbucketIntegrationConfig,\n): Promise<string> {\n const { name: repoName, owner: project, resource } = parseGitUrl(url);\n\n const isHosted = resource === 'bitbucket.org';\n // Bitbucket Server https://docs.atlassian.com/bitbucket-server/rest/7.9.0/bitbucket-rest.html#idp184\n let branchUrl = isHosted\n ? `${config.apiBaseUrl}/repositories/${project}/${repoName}`\n : `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/default-branch`;\n\n let response = await fetch(branchUrl, getBitbucketRequestOptions(config));\n\n if (response.status === 404 && !isHosted) {\n // First try the new format, and then if it gets specifically a 404 it should try the old format\n // (to support old Atlassian Bitbucket v5.11.1 format )\n branchUrl = `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/branches/default`;\n response = await fetch(branchUrl, getBitbucketRequestOptions(config));\n }\n\n if (!response.ok) {\n const message = `Failed to retrieve default branch from ${branchUrl}, ${response.status} ${response.statusText}`;\n throw new Error(message);\n }\n\n let defaultBranch;\n if (isHosted) {\n const repoInfo = await response.json();\n defaultBranch = repoInfo.mainbranch.name;\n } else {\n const { displayId } = await response.json();\n defaultBranch = displayId;\n }\n if (!defaultBranch) {\n throw new Error(\n `Failed to read default branch from ${branchUrl}. ` +\n `Response ${response.status} ${response.json()}`,\n );\n }\n return defaultBranch;\n}\n\n/**\n * Given a URL pointing to a path on a provider, returns a URL that is suitable\n * for downloading the subtree.\n *\n * @param url - A URL pointing to a path\n * @param config - The relevant provider config\n * @public\n * @deprecated no longer in use, bitbucket integration replaced by integrations bitbucketCloud and bitbucketServer.\n */\nexport async function getBitbucketDownloadUrl(\n url: string,\n config: BitbucketIntegrationConfig,\n): Promise<string> {\n const {\n name: repoName,\n owner: project,\n ref,\n protocol,\n resource,\n filepath,\n } = parseGitUrl(url);\n\n const isHosted = resource === 'bitbucket.org';\n\n let branch = ref;\n if (!branch) {\n branch = await getBitbucketDefaultBranch(url, config);\n }\n // path will limit the downloaded content\n // /docs will only download the docs folder and everything below it\n // /docs/index.md will download the docs folder and everything below it\n const path = filepath\n ? `&path=${encodeURIComponent(decodeURIComponent(filepath))}`\n : '';\n const archiveUrl = isHosted\n ? `${protocol}://${resource}/${project}/${repoName}/get/${branch}.tar.gz`\n : `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/archive?format=tgz&at=${branch}&prefix=${project}-${repoName}${path}`;\n\n return archiveUrl;\n}\n\n/**\n * Given a URL pointing to a file on a provider, returns a URL that is suitable\n * for fetching the contents of the data.\n *\n * @remarks\n *\n * Converts\n * from: https://bitbucket.org/orgname/reponame/src/master/file.yaml\n * to: https://api.bitbucket.org/2.0/repositories/orgname/reponame/src/master/file.yaml\n *\n * @param url - A URL pointing to a file\n * @param config - The relevant provider config\n * @public\n * @deprecated no longer in use, bitbucket integration replaced by integrations bitbucketCloud and bitbucketServer.\n */\nexport function getBitbucketFileFetchUrl(\n url: string,\n config: BitbucketIntegrationConfig,\n): string {\n try {\n const { owner, name, ref, filepathtype, filepath } = parseGitUrl(url);\n if (\n !owner ||\n !name ||\n (filepathtype !== 'browse' &&\n filepathtype !== 'raw' &&\n filepathtype !== 'src')\n ) {\n throw new Error('Invalid Bitbucket URL or file path');\n }\n\n const pathWithoutSlash = filepath.replace(/^\\//, '');\n\n if (config.host === 'bitbucket.org') {\n if (!ref) {\n throw new Error('Invalid Bitbucket URL or file path');\n }\n return `${config.apiBaseUrl}/repositories/${owner}/${name}/src/${ref}/${pathWithoutSlash}`;\n }\n return `${config.apiBaseUrl}/projects/${owner}/repos/${name}/raw/${pathWithoutSlash}?at=${ref}`;\n } catch (e) {\n throw new Error(`Incorrect URL: ${url}, ${e}`);\n }\n}\n\n/**\n * Gets the request options necessary to make requests to a given provider.\n *\n * @param config - The relevant provider config\n * @public\n * @deprecated no longer in use, bitbucket integration replaced by integrations bitbucketCloud and bitbucketServer.\n */\nexport function getBitbucketRequestOptions(\n config: BitbucketIntegrationConfig,\n): { headers: Record<string, string> } {\n const headers: Record<string, string> = {};\n\n if (config.token) {\n headers.Authorization = `Bearer ${config.token}`;\n } else if (config.username && config.appPassword) {\n const buffer = Buffer.from(\n `${config.username}:${config.appPassword}`,\n 'utf8',\n );\n headers.Authorization = `Basic ${buffer.toString('base64')}`;\n }\n\n return {\n headers,\n };\n}\n"],"names":["parseGitUrl","fetch"],"mappings":"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"}
@@ -0,0 +1,54 @@
1
+ 'use strict';
2
+
3
+ var parseGitUrl = require('git-url-parse');
4
+ var helpers = require('../helpers.cjs.js');
5
+ var config = require('./config.cjs.js');
6
+
7
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
8
+
9
+ var parseGitUrl__default = /*#__PURE__*/_interopDefaultCompat(parseGitUrl);
10
+
11
+ class BitbucketCloudIntegration {
12
+ constructor(integrationConfig) {
13
+ this.integrationConfig = integrationConfig;
14
+ }
15
+ static factory = ({
16
+ config: config$1
17
+ }) => {
18
+ const configs = config.readBitbucketCloudIntegrationConfigs(
19
+ config$1.getOptionalConfigArray("integrations.bitbucketCloud") ?? []
20
+ );
21
+ return helpers.basicIntegrations(
22
+ configs.map((c) => new BitbucketCloudIntegration(c)),
23
+ (i) => i.config.host
24
+ );
25
+ };
26
+ get type() {
27
+ return "bitbucketCloud";
28
+ }
29
+ get title() {
30
+ return this.integrationConfig.host;
31
+ }
32
+ get config() {
33
+ return this.integrationConfig;
34
+ }
35
+ resolveUrl(options) {
36
+ const resolved = helpers.defaultScmResolveUrl(options);
37
+ if (options.lineNumber) {
38
+ const url = new URL(resolved);
39
+ url.hash = `lines-${options.lineNumber}`;
40
+ return url.toString();
41
+ }
42
+ return resolved;
43
+ }
44
+ resolveEditUrl(url) {
45
+ const urlData = parseGitUrl__default.default(url);
46
+ const editUrl = new URL(url);
47
+ editUrl.searchParams.set("mode", "edit");
48
+ editUrl.searchParams.set("at", urlData.ref);
49
+ return editUrl.toString();
50
+ }
51
+ }
52
+
53
+ exports.BitbucketCloudIntegration = BitbucketCloudIntegration;
54
+ //# sourceMappingURL=BitbucketCloudIntegration.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"BitbucketCloudIntegration.cjs.js","sources":["../../src/bitbucketCloud/BitbucketCloudIntegration.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport parseGitUrl from 'git-url-parse';\nimport { basicIntegrations, defaultScmResolveUrl } from '../helpers';\nimport { ScmIntegration, ScmIntegrationsFactory } from '../types';\nimport {\n BitbucketCloudIntegrationConfig,\n readBitbucketCloudIntegrationConfigs,\n} from './config';\n\n/**\n * A Bitbucket Cloud based integration.\n *\n * @public\n */\nexport class BitbucketCloudIntegration implements ScmIntegration {\n static factory: ScmIntegrationsFactory<BitbucketCloudIntegration> = ({\n config,\n }) => {\n const configs = readBitbucketCloudIntegrationConfigs(\n config.getOptionalConfigArray('integrations.bitbucketCloud') ?? [],\n );\n return basicIntegrations(\n configs.map(c => new BitbucketCloudIntegration(c)),\n i => i.config.host,\n );\n };\n\n constructor(\n private readonly integrationConfig: BitbucketCloudIntegrationConfig,\n ) {}\n\n get type(): string {\n return 'bitbucketCloud';\n }\n\n get title(): string {\n return this.integrationConfig.host;\n }\n\n get config(): BitbucketCloudIntegrationConfig {\n return this.integrationConfig;\n }\n\n resolveUrl(options: {\n url: string;\n base: string;\n lineNumber?: number;\n }): string {\n const resolved = defaultScmResolveUrl(options);\n\n // Bitbucket Cloud line numbers use the syntax #lines-42, rather than #L42\n if (options.lineNumber) {\n const url = new URL(resolved);\n\n url.hash = `lines-${options.lineNumber}`;\n return url.toString();\n }\n\n return resolved;\n }\n\n resolveEditUrl(url: string): string {\n const urlData = parseGitUrl(url);\n const editUrl = new URL(url);\n\n editUrl.searchParams.set('mode', 'edit');\n editUrl.searchParams.set('at', urlData.ref);\n return editUrl.toString();\n }\n}\n"],"names":["config","readBitbucketCloudIntegrationConfigs","basicIntegrations","defaultScmResolveUrl","parseGitUrl"],"mappings":";;;;;;;;;;AA6BO,MAAM,yBAAoD,CAAA;AAAA,EAa/D,YACmB,iBACjB,EAAA;AADiB,IAAA,IAAA,CAAA,iBAAA,GAAA,iBAAA,CAAA;AAAA,GAChB;AAAA,EAdH,OAAO,UAA6D,CAAC;AAAA,YACnEA,QAAA;AAAA,GACI,KAAA;AACJ,IAAA,MAAM,OAAU,GAAAC,2CAAA;AAAA,MACdD,QAAO,CAAA,sBAAA,CAAuB,6BAA6B,CAAA,IAAK,EAAC;AAAA,KACnE,CAAA;AACA,IAAO,OAAAE,yBAAA;AAAA,MACL,QAAQ,GAAI,CAAA,CAAA,CAAA,KAAK,IAAI,yBAAA,CAA0B,CAAC,CAAC,CAAA;AAAA,MACjD,CAAA,CAAA,KAAK,EAAE,MAAO,CAAA,IAAA;AAAA,KAChB,CAAA;AAAA,GACF,CAAA;AAAA,EAMA,IAAI,IAAe,GAAA;AACjB,IAAO,OAAA,gBAAA,CAAA;AAAA,GACT;AAAA,EAEA,IAAI,KAAgB,GAAA;AAClB,IAAA,OAAO,KAAK,iBAAkB,CAAA,IAAA,CAAA;AAAA,GAChC;AAAA,EAEA,IAAI,MAA0C,GAAA;AAC5C,IAAA,OAAO,IAAK,CAAA,iBAAA,CAAA;AAAA,GACd;AAAA,EAEA,WAAW,OAIA,EAAA;AACT,IAAM,MAAA,QAAA,GAAWC,6BAAqB,OAAO,CAAA,CAAA;AAG7C,IAAA,IAAI,QAAQ,UAAY,EAAA;AACtB,MAAM,MAAA,GAAA,GAAM,IAAI,GAAA,CAAI,QAAQ,CAAA,CAAA;AAE5B,MAAI,GAAA,CAAA,IAAA,GAAO,CAAS,MAAA,EAAA,OAAA,CAAQ,UAAU,CAAA,CAAA,CAAA;AACtC,MAAA,OAAO,IAAI,QAAS,EAAA,CAAA;AAAA,KACtB;AAEA,IAAO,OAAA,QAAA,CAAA;AAAA,GACT;AAAA,EAEA,eAAe,GAAqB,EAAA;AAClC,IAAM,MAAA,OAAA,GAAUC,6BAAY,GAAG,CAAA,CAAA;AAC/B,IAAM,MAAA,OAAA,GAAU,IAAI,GAAA,CAAI,GAAG,CAAA,CAAA;AAE3B,IAAQ,OAAA,CAAA,YAAA,CAAa,GAAI,CAAA,MAAA,EAAQ,MAAM,CAAA,CAAA;AACvC,IAAA,OAAA,CAAQ,YAAa,CAAA,GAAA,CAAI,IAAM,EAAA,OAAA,CAAQ,GAAG,CAAA,CAAA;AAC1C,IAAA,OAAO,QAAQ,QAAS,EAAA,CAAA;AAAA,GAC1B;AACF;;;;"}
@@ -0,0 +1,30 @@
1
+ 'use strict';
2
+
3
+ const BITBUCKET_CLOUD_HOST = "bitbucket.org";
4
+ const BITBUCKET_CLOUD_API_BASE_URL = "https://api.bitbucket.org/2.0";
5
+ function readBitbucketCloudIntegrationConfig(config) {
6
+ const host = BITBUCKET_CLOUD_HOST;
7
+ const apiBaseUrl = BITBUCKET_CLOUD_API_BASE_URL;
8
+ const username = config.getString("username");
9
+ const appPassword = config.getString("appPassword")?.trim();
10
+ return {
11
+ host,
12
+ apiBaseUrl,
13
+ username,
14
+ appPassword
15
+ };
16
+ }
17
+ function readBitbucketCloudIntegrationConfigs(configs) {
18
+ const result = configs.map(readBitbucketCloudIntegrationConfig);
19
+ if (result.length === 0) {
20
+ result.push({
21
+ host: BITBUCKET_CLOUD_HOST,
22
+ apiBaseUrl: BITBUCKET_CLOUD_API_BASE_URL
23
+ });
24
+ }
25
+ return result;
26
+ }
27
+
28
+ exports.readBitbucketCloudIntegrationConfig = readBitbucketCloudIntegrationConfig;
29
+ exports.readBitbucketCloudIntegrationConfigs = readBitbucketCloudIntegrationConfigs;
30
+ //# sourceMappingURL=config.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.cjs.js","sources":["../../src/bitbucketCloud/config.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { Config } from '@backstage/config';\n\nconst BITBUCKET_CLOUD_HOST = 'bitbucket.org';\nconst BITBUCKET_CLOUD_API_BASE_URL = 'https://api.bitbucket.org/2.0';\n\n/**\n * The configuration parameters for a single Bitbucket Cloud API provider.\n *\n * @public\n */\nexport type BitbucketCloudIntegrationConfig = {\n /**\n * Constant. bitbucket.org\n */\n host: string;\n\n /**\n * Constant. https://api.bitbucket.org/2.0\n */\n apiBaseUrl: string;\n\n /**\n * The username to use for requests to Bitbucket Cloud (bitbucket.org).\n */\n username?: string;\n\n /**\n * Authentication with Bitbucket Cloud (bitbucket.org) is done using app passwords.\n *\n * See https://support.atlassian.com/bitbucket-cloud/docs/app-passwords/\n */\n appPassword?: string;\n\n /**\n * The access token to use for requests to Bitbucket Cloud (bitbucket.org).\n */\n token?: string;\n};\n\n/**\n * Reads a single Bitbucket Cloud integration config.\n *\n * @param config - The config object of a single integration\n * @public\n */\nexport function readBitbucketCloudIntegrationConfig(\n config: Config,\n): BitbucketCloudIntegrationConfig {\n const host = BITBUCKET_CLOUD_HOST;\n const apiBaseUrl = BITBUCKET_CLOUD_API_BASE_URL;\n // If config is provided, we assume authenticated access is desired\n // (as the anonymous one is provided by default).\n const username = config.getString('username');\n const appPassword = config.getString('appPassword')?.trim();\n\n return {\n host,\n apiBaseUrl,\n username,\n appPassword,\n };\n}\n\n/**\n * Reads a set of Bitbucket Cloud integration configs,\n * and inserts one for public Bitbucket Cloud if none specified.\n *\n * @param configs - All of the integration config objects\n * @public\n */\nexport function readBitbucketCloudIntegrationConfigs(\n configs: Config[],\n): BitbucketCloudIntegrationConfig[] {\n // First read all the explicit integrations\n const result = configs.map(readBitbucketCloudIntegrationConfig);\n\n // If no explicit bitbucket.org integration was added,\n // put one in the list as a convenience\n if (result.length === 0) {\n result.push({\n host: BITBUCKET_CLOUD_HOST,\n apiBaseUrl: BITBUCKET_CLOUD_API_BASE_URL,\n });\n }\n\n return result;\n}\n"],"names":[],"mappings":";;AAkBA,MAAM,oBAAuB,GAAA,eAAA,CAAA;AAC7B,MAAM,4BAA+B,GAAA,+BAAA,CAAA;AA0C9B,SAAS,oCACd,MACiC,EAAA;AACjC,EAAA,MAAM,IAAO,GAAA,oBAAA,CAAA;AACb,EAAA,MAAM,UAAa,GAAA,4BAAA,CAAA;AAGnB,EAAM,MAAA,QAAA,GAAW,MAAO,CAAA,SAAA,CAAU,UAAU,CAAA,CAAA;AAC5C,EAAA,MAAM,WAAc,GAAA,MAAA,CAAO,SAAU,CAAA,aAAa,GAAG,IAAK,EAAA,CAAA;AAE1D,EAAO,OAAA;AAAA,IACL,IAAA;AAAA,IACA,UAAA;AAAA,IACA,QAAA;AAAA,IACA,WAAA;AAAA,GACF,CAAA;AACF,CAAA;AASO,SAAS,qCACd,OACmC,EAAA;AAEnC,EAAM,MAAA,MAAA,GAAS,OAAQ,CAAA,GAAA,CAAI,mCAAmC,CAAA,CAAA;AAI9D,EAAI,IAAA,MAAA,CAAO,WAAW,CAAG,EAAA;AACvB,IAAA,MAAA,CAAO,IAAK,CAAA;AAAA,MACV,IAAM,EAAA,oBAAA;AAAA,MACN,UAAY,EAAA,4BAAA;AAAA,KACb,CAAA,CAAA;AAAA,GACH;AAEA,EAAO,OAAA,MAAA,CAAA;AACT;;;;;"}
@@ -0,0 +1,78 @@
1
+ 'use strict';
2
+
3
+ var fetch = require('cross-fetch');
4
+ var parseGitUrl = require('git-url-parse');
5
+
6
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
7
+
8
+ var fetch__default = /*#__PURE__*/_interopDefaultCompat(fetch);
9
+ var parseGitUrl__default = /*#__PURE__*/_interopDefaultCompat(parseGitUrl);
10
+
11
+ async function getBitbucketCloudDefaultBranch(url, config) {
12
+ const { name: repoName, owner: project } = parseGitUrl__default.default(url);
13
+ const branchUrl = `${config.apiBaseUrl}/repositories/${project}/${repoName}`;
14
+ const response = await fetch__default.default(
15
+ branchUrl,
16
+ getBitbucketCloudRequestOptions(config)
17
+ );
18
+ if (!response.ok) {
19
+ const message = `Failed to retrieve default branch from ${branchUrl}, ${response.status} ${response.statusText}`;
20
+ throw new Error(message);
21
+ }
22
+ const repoInfo = await response.json();
23
+ const defaultBranch = repoInfo.mainbranch.name;
24
+ if (!defaultBranch) {
25
+ throw new Error(
26
+ `Failed to read default branch from ${branchUrl}. Response ${response.status} ${response.json()}`
27
+ );
28
+ }
29
+ return defaultBranch;
30
+ }
31
+ async function getBitbucketCloudDownloadUrl(url, config) {
32
+ const {
33
+ name: repoName,
34
+ owner: project,
35
+ ref,
36
+ protocol,
37
+ resource
38
+ } = parseGitUrl__default.default(url);
39
+ let branch = ref;
40
+ if (!branch) {
41
+ branch = await getBitbucketCloudDefaultBranch(url, config);
42
+ }
43
+ return `${protocol}://${resource}/${project}/${repoName}/get/${branch}.tar.gz`;
44
+ }
45
+ function getBitbucketCloudFileFetchUrl(url, config) {
46
+ try {
47
+ const { owner, name, ref, filepathtype, filepath } = parseGitUrl__default.default(url);
48
+ if (!owner || !name || filepathtype !== "src" && filepathtype !== "raw") {
49
+ throw new Error("Invalid Bitbucket Cloud URL or file path");
50
+ }
51
+ const pathWithoutSlash = filepath.replace(/^\//, "");
52
+ if (!ref) {
53
+ throw new Error("Invalid Bitbucket Cloud URL or file path");
54
+ }
55
+ return `${config.apiBaseUrl}/repositories/${owner}/${name}/src/${ref}/${pathWithoutSlash}`;
56
+ } catch (e) {
57
+ throw new Error(`Incorrect URL: ${url}, ${e}`);
58
+ }
59
+ }
60
+ function getBitbucketCloudRequestOptions(config) {
61
+ const headers = {};
62
+ if (config.username && config.appPassword) {
63
+ const buffer = Buffer.from(
64
+ `${config.username}:${config.appPassword}`,
65
+ "utf8"
66
+ );
67
+ headers.Authorization = `Basic ${buffer.toString("base64")}`;
68
+ }
69
+ return {
70
+ headers
71
+ };
72
+ }
73
+
74
+ exports.getBitbucketCloudDefaultBranch = getBitbucketCloudDefaultBranch;
75
+ exports.getBitbucketCloudDownloadUrl = getBitbucketCloudDownloadUrl;
76
+ exports.getBitbucketCloudFileFetchUrl = getBitbucketCloudFileFetchUrl;
77
+ exports.getBitbucketCloudRequestOptions = getBitbucketCloudRequestOptions;
78
+ //# sourceMappingURL=core.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"core.cjs.js","sources":["../../src/bitbucketCloud/core.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'cross-fetch';\nimport parseGitUrl from 'git-url-parse';\nimport { BitbucketCloudIntegrationConfig } from './config';\n\n/**\n * Given a URL pointing to a path on a provider, returns the default branch.\n *\n * @param url - A URL pointing to a path\n * @param config - The relevant provider config\n * @public\n */\nexport async function getBitbucketCloudDefaultBranch(\n url: string,\n config: BitbucketCloudIntegrationConfig,\n): Promise<string> {\n const { name: repoName, owner: project } = parseGitUrl(url);\n\n const branchUrl = `${config.apiBaseUrl}/repositories/${project}/${repoName}`;\n const response = await fetch(\n branchUrl,\n getBitbucketCloudRequestOptions(config),\n );\n\n if (!response.ok) {\n const message = `Failed to retrieve default branch from ${branchUrl}, ${response.status} ${response.statusText}`;\n throw new Error(message);\n }\n\n const repoInfo = await response.json();\n const defaultBranch = repoInfo.mainbranch.name;\n if (!defaultBranch) {\n throw new Error(\n `Failed to read default branch from ${branchUrl}. ` +\n `Response ${response.status} ${response.json()}`,\n );\n }\n return defaultBranch;\n}\n\n/**\n * Given a URL pointing to a path on a provider, returns a URL that is suitable\n * for downloading the subtree.\n *\n * @param url - A URL pointing to a path\n * @param config - The relevant provider config\n * @public\n */\nexport async function getBitbucketCloudDownloadUrl(\n url: string,\n config: BitbucketCloudIntegrationConfig,\n): Promise<string> {\n const {\n name: repoName,\n owner: project,\n ref,\n protocol,\n resource,\n } = parseGitUrl(url);\n\n let branch = ref;\n if (!branch) {\n branch = await getBitbucketCloudDefaultBranch(url, config);\n }\n return `${protocol}://${resource}/${project}/${repoName}/get/${branch}.tar.gz`;\n}\n\n/**\n * Given a URL pointing to a file on a provider, returns a URL that is suitable\n * for fetching the contents of the data.\n *\n * @remarks\n *\n * Converts\n * from: https://bitbucket.org/orgname/reponame/src/master/file.yaml\n * to: https://api.bitbucket.org/2.0/repositories/orgname/reponame/src/master/file.yaml\n *\n * @param url - A URL pointing to a file\n * @param config - The relevant provider config\n * @public\n */\nexport function getBitbucketCloudFileFetchUrl(\n url: string,\n config: BitbucketCloudIntegrationConfig,\n): string {\n try {\n const { owner, name, ref, filepathtype, filepath } = parseGitUrl(url);\n if (!owner || !name || (filepathtype !== 'src' && filepathtype !== 'raw')) {\n throw new Error('Invalid Bitbucket Cloud URL or file path');\n }\n\n const pathWithoutSlash = filepath.replace(/^\\//, '');\n\n if (!ref) {\n throw new Error('Invalid Bitbucket Cloud URL or file path');\n }\n return `${config.apiBaseUrl}/repositories/${owner}/${name}/src/${ref}/${pathWithoutSlash}`;\n } catch (e) {\n throw new Error(`Incorrect URL: ${url}, ${e}`);\n }\n}\n\n/**\n * Gets the request options necessary to make requests to a given provider.\n *\n * @param config - The relevant provider config\n * @public\n */\nexport function getBitbucketCloudRequestOptions(\n config: BitbucketCloudIntegrationConfig,\n): { headers: Record<string, string> } {\n const headers: Record<string, string> = {};\n\n if (config.username && config.appPassword) {\n const buffer = Buffer.from(\n `${config.username}:${config.appPassword}`,\n 'utf8',\n );\n headers.Authorization = `Basic ${buffer.toString('base64')}`;\n }\n\n return {\n headers,\n };\n}\n"],"names":["parseGitUrl","fetch"],"mappings":"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"}
@@ -0,0 +1,48 @@
1
+ 'use strict';
2
+
3
+ var helpers = require('../helpers.cjs.js');
4
+ var config = require('./config.cjs.js');
5
+
6
+ class BitbucketServerIntegration {
7
+ constructor(integrationConfig) {
8
+ this.integrationConfig = integrationConfig;
9
+ }
10
+ static factory = ({
11
+ config: config$1
12
+ }) => {
13
+ const configs = config.readBitbucketServerIntegrationConfigs(
14
+ config$1.getOptionalConfigArray("integrations.bitbucketServer") ?? []
15
+ );
16
+ return helpers.basicIntegrations(
17
+ configs.map((c) => new BitbucketServerIntegration(c)),
18
+ (i) => i.config.host
19
+ );
20
+ };
21
+ get type() {
22
+ return "bitbucketServer";
23
+ }
24
+ get title() {
25
+ return this.integrationConfig.host;
26
+ }
27
+ get config() {
28
+ return this.integrationConfig;
29
+ }
30
+ resolveUrl(options) {
31
+ const resolved = helpers.defaultScmResolveUrl(options);
32
+ if (options.lineNumber) {
33
+ const url = new URL(resolved);
34
+ url.hash = options.lineNumber.toString();
35
+ return url.toString();
36
+ }
37
+ return resolved;
38
+ }
39
+ resolveEditUrl(url) {
40
+ if (url.includes("?")) {
41
+ return url.substring(0, url.indexOf("?"));
42
+ }
43
+ return url;
44
+ }
45
+ }
46
+
47
+ exports.BitbucketServerIntegration = BitbucketServerIntegration;
48
+ //# sourceMappingURL=BitbucketServerIntegration.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"BitbucketServerIntegration.cjs.js","sources":["../../src/bitbucketServer/BitbucketServerIntegration.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { basicIntegrations, defaultScmResolveUrl } from '../helpers';\nimport { ScmIntegration, ScmIntegrationsFactory } from '../types';\nimport {\n BitbucketServerIntegrationConfig,\n readBitbucketServerIntegrationConfigs,\n} from './config';\n\n/**\n * A Bitbucket Server based integration.\n *\n * @public\n */\nexport class BitbucketServerIntegration implements ScmIntegration {\n static factory: ScmIntegrationsFactory<BitbucketServerIntegration> = ({\n config,\n }) => {\n const configs = readBitbucketServerIntegrationConfigs(\n config.getOptionalConfigArray('integrations.bitbucketServer') ?? [],\n );\n return basicIntegrations(\n configs.map(c => new BitbucketServerIntegration(c)),\n i => i.config.host,\n );\n };\n\n constructor(\n private readonly integrationConfig: BitbucketServerIntegrationConfig,\n ) {}\n\n get type(): string {\n return 'bitbucketServer';\n }\n\n get title(): string {\n return this.integrationConfig.host;\n }\n\n get config(): BitbucketServerIntegrationConfig {\n return this.integrationConfig;\n }\n\n resolveUrl(options: {\n url: string;\n base: string;\n lineNumber?: number;\n }): string {\n const resolved = defaultScmResolveUrl(options);\n\n // Bitbucket Server line numbers use the syntax #42, rather than #L42\n if (options.lineNumber) {\n const url = new URL(resolved);\n\n url.hash = options.lineNumber.toString();\n return url.toString();\n }\n\n return resolved;\n }\n\n resolveEditUrl(url: string): string {\n // Bitbucket Server doesn't support deep linking to edit mode, therefore there's nothing to do here.\n // We just remove query parameters since they cause issues with TechDocs edit button.\n if (url.includes('?')) {\n return url.substring(0, url.indexOf('?'));\n }\n return url;\n }\n}\n"],"names":["config","readBitbucketServerIntegrationConfigs","basicIntegrations","defaultScmResolveUrl"],"mappings":";;;;;AA4BO,MAAM,0BAAqD,CAAA;AAAA,EAahE,YACmB,iBACjB,EAAA;AADiB,IAAA,IAAA,CAAA,iBAAA,GAAA,iBAAA,CAAA;AAAA,GAChB;AAAA,EAdH,OAAO,UAA8D,CAAC;AAAA,YACpEA,QAAA;AAAA,GACI,KAAA;AACJ,IAAA,MAAM,OAAU,GAAAC,4CAAA;AAAA,MACdD,QAAO,CAAA,sBAAA,CAAuB,8BAA8B,CAAA,IAAK,EAAC;AAAA,KACpE,CAAA;AACA,IAAO,OAAAE,yBAAA;AAAA,MACL,QAAQ,GAAI,CAAA,CAAA,CAAA,KAAK,IAAI,0BAAA,CAA2B,CAAC,CAAC,CAAA;AAAA,MAClD,CAAA,CAAA,KAAK,EAAE,MAAO,CAAA,IAAA;AAAA,KAChB,CAAA;AAAA,GACF,CAAA;AAAA,EAMA,IAAI,IAAe,GAAA;AACjB,IAAO,OAAA,iBAAA,CAAA;AAAA,GACT;AAAA,EAEA,IAAI,KAAgB,GAAA;AAClB,IAAA,OAAO,KAAK,iBAAkB,CAAA,IAAA,CAAA;AAAA,GAChC;AAAA,EAEA,IAAI,MAA2C,GAAA;AAC7C,IAAA,OAAO,IAAK,CAAA,iBAAA,CAAA;AAAA,GACd;AAAA,EAEA,WAAW,OAIA,EAAA;AACT,IAAM,MAAA,QAAA,GAAWC,6BAAqB,OAAO,CAAA,CAAA;AAG7C,IAAA,IAAI,QAAQ,UAAY,EAAA;AACtB,MAAM,MAAA,GAAA,GAAM,IAAI,GAAA,CAAI,QAAQ,CAAA,CAAA;AAE5B,MAAI,GAAA,CAAA,IAAA,GAAO,OAAQ,CAAA,UAAA,CAAW,QAAS,EAAA,CAAA;AACvC,MAAA,OAAO,IAAI,QAAS,EAAA,CAAA;AAAA,KACtB;AAEA,IAAO,OAAA,QAAA,CAAA;AAAA,GACT;AAAA,EAEA,eAAe,GAAqB,EAAA;AAGlC,IAAI,IAAA,GAAA,CAAI,QAAS,CAAA,GAAG,CAAG,EAAA;AACrB,MAAA,OAAO,IAAI,SAAU,CAAA,CAAA,EAAG,GAAI,CAAA,OAAA,CAAQ,GAAG,CAAC,CAAA,CAAA;AAAA,KAC1C;AACA,IAAO,OAAA,GAAA,CAAA;AAAA,GACT;AACF;;;;"}
@@ -0,0 +1,36 @@
1
+ 'use strict';
2
+
3
+ var lodash = require('lodash');
4
+ var helpers = require('../helpers.cjs.js');
5
+
6
+ function readBitbucketServerIntegrationConfig(config) {
7
+ const host = config.getString("host");
8
+ let apiBaseUrl = config.getOptionalString("apiBaseUrl");
9
+ const token = config.getOptionalString("token")?.trim();
10
+ const username = config.getOptionalString("username");
11
+ const password = config.getOptionalString("password");
12
+ if (!helpers.isValidHost(host)) {
13
+ throw new Error(
14
+ `Invalid Bitbucket Server integration config, '${host}' is not a valid host`
15
+ );
16
+ }
17
+ if (apiBaseUrl) {
18
+ apiBaseUrl = lodash.trimEnd(apiBaseUrl, "/");
19
+ } else {
20
+ apiBaseUrl = `https://${host}/rest/api/1.0`;
21
+ }
22
+ return {
23
+ host,
24
+ apiBaseUrl,
25
+ token,
26
+ username,
27
+ password
28
+ };
29
+ }
30
+ function readBitbucketServerIntegrationConfigs(configs) {
31
+ return configs.map(readBitbucketServerIntegrationConfig);
32
+ }
33
+
34
+ exports.readBitbucketServerIntegrationConfig = readBitbucketServerIntegrationConfig;
35
+ exports.readBitbucketServerIntegrationConfigs = readBitbucketServerIntegrationConfigs;
36
+ //# sourceMappingURL=config.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.cjs.js","sources":["../../src/bitbucketServer/config.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { Config } from '@backstage/config';\nimport { trimEnd } from 'lodash';\nimport { isValidHost } from '../helpers';\n\n/**\n * The configuration parameters for a single Bitbucket Server API provider.\n *\n * @public\n */\nexport type BitbucketServerIntegrationConfig = {\n /**\n * The host of the target that this matches on, e.g. \"bitbucket.company.com\"\n */\n host: string;\n\n /**\n * The base URL of the API of this provider, e.g. \"https://<host>/rest/api/1.0\",\n * with no trailing slash.\n *\n * The API will always be preferred if both its base URL and a token are\n * present.\n */\n apiBaseUrl: string;\n\n /**\n * The authorization token to use for requests to a Bitbucket Server provider.\n *\n * See https://confluence.atlassian.com/bitbucketserver/personal-access-tokens-939515499.html\n *\n * If no token is specified, anonymous access is used.\n */\n token?: string;\n\n /**\n * The credentials for Basic Authentication for requests to a Bitbucket Server provider.\n *\n * If `token` was provided, it will be preferred.\n *\n * See https://developer.atlassian.com/server/bitbucket/how-tos/command-line-rest/#authentication\n */\n username?: string;\n\n /**\n * The credentials for Basic Authentication for requests to a Bitbucket Server provider.\n *\n * If `token` was provided, it will be preferred.\n *\n * See https://developer.atlassian.com/server/bitbucket/how-tos/command-line-rest/#authentication\n */\n password?: string;\n};\n\n/**\n * Reads a single Bitbucket Server integration config.\n *\n * @param config - The config object of a single integration\n * @public\n */\nexport function readBitbucketServerIntegrationConfig(\n config: Config,\n): BitbucketServerIntegrationConfig {\n const host = config.getString('host');\n let apiBaseUrl = config.getOptionalString('apiBaseUrl');\n const token = config.getOptionalString('token')?.trim();\n const username = config.getOptionalString('username');\n const password = config.getOptionalString('password');\n\n if (!isValidHost(host)) {\n throw new Error(\n `Invalid Bitbucket Server integration config, '${host}' is not a valid host`,\n );\n }\n\n if (apiBaseUrl) {\n apiBaseUrl = trimEnd(apiBaseUrl, '/');\n } else {\n apiBaseUrl = `https://${host}/rest/api/1.0`;\n }\n\n return {\n host,\n apiBaseUrl,\n token,\n username,\n password,\n };\n}\n\n/**\n * Reads a set of Bitbucket Server integration configs.\n *\n * @param configs - All of the integration config objects\n * @public\n */\nexport function readBitbucketServerIntegrationConfigs(\n configs: Config[],\n): BitbucketServerIntegrationConfig[] {\n // Read all the explicit integrations\n // No default integration will be added\n return configs.map(readBitbucketServerIntegrationConfig);\n}\n"],"names":["isValidHost","trimEnd"],"mappings":";;;;;AA0EO,SAAS,qCACd,MACkC,EAAA;AAClC,EAAM,MAAA,IAAA,GAAO,MAAO,CAAA,SAAA,CAAU,MAAM,CAAA,CAAA;AACpC,EAAI,IAAA,UAAA,GAAa,MAAO,CAAA,iBAAA,CAAkB,YAAY,CAAA,CAAA;AACtD,EAAA,MAAM,KAAQ,GAAA,MAAA,CAAO,iBAAkB,CAAA,OAAO,GAAG,IAAK,EAAA,CAAA;AACtD,EAAM,MAAA,QAAA,GAAW,MAAO,CAAA,iBAAA,CAAkB,UAAU,CAAA,CAAA;AACpD,EAAM,MAAA,QAAA,GAAW,MAAO,CAAA,iBAAA,CAAkB,UAAU,CAAA,CAAA;AAEpD,EAAI,IAAA,CAACA,mBAAY,CAAA,IAAI,CAAG,EAAA;AACtB,IAAA,MAAM,IAAI,KAAA;AAAA,MACR,iDAAiD,IAAI,CAAA,qBAAA,CAAA;AAAA,KACvD,CAAA;AAAA,GACF;AAEA,EAAA,IAAI,UAAY,EAAA;AACd,IAAa,UAAA,GAAAC,cAAA,CAAQ,YAAY,GAAG,CAAA,CAAA;AAAA,GAC/B,MAAA;AACL,IAAA,UAAA,GAAa,WAAW,IAAI,CAAA,aAAA,CAAA,CAAA;AAAA,GAC9B;AAEA,EAAO,OAAA;AAAA,IACL,IAAA;AAAA,IACA,UAAA;AAAA,IACA,KAAA;AAAA,IACA,QAAA;AAAA,IACA,QAAA;AAAA,GACF,CAAA;AACF,CAAA;AAQO,SAAS,sCACd,OACoC,EAAA;AAGpC,EAAO,OAAA,OAAA,CAAQ,IAAI,oCAAoC,CAAA,CAAA;AACzD;;;;;"}
@@ -0,0 +1,73 @@
1
+ 'use strict';
2
+
3
+ var fetch = require('cross-fetch');
4
+ var parseGitUrl = require('git-url-parse');
5
+
6
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
7
+
8
+ var fetch__default = /*#__PURE__*/_interopDefaultCompat(fetch);
9
+ var parseGitUrl__default = /*#__PURE__*/_interopDefaultCompat(parseGitUrl);
10
+
11
+ async function getBitbucketServerDefaultBranch(url, config) {
12
+ const { name: repoName, owner: project } = parseGitUrl__default.default(url);
13
+ let branchUrl = `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/default-branch`;
14
+ let response = await fetch__default.default(
15
+ branchUrl,
16
+ getBitbucketServerRequestOptions(config)
17
+ );
18
+ if (response.status === 404) {
19
+ branchUrl = `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/branches/default`;
20
+ response = await fetch__default.default(branchUrl, getBitbucketServerRequestOptions(config));
21
+ }
22
+ if (!response.ok) {
23
+ const message = `Failed to retrieve default branch from ${branchUrl}, ${response.status} ${response.statusText}`;
24
+ throw new Error(message);
25
+ }
26
+ const { displayId } = await response.json();
27
+ const defaultBranch = displayId;
28
+ if (!defaultBranch) {
29
+ throw new Error(
30
+ `Failed to read default branch from ${branchUrl}. Response ${response.status} ${response.json()}`
31
+ );
32
+ }
33
+ return defaultBranch;
34
+ }
35
+ async function getBitbucketServerDownloadUrl(url, config) {
36
+ const { name: repoName, owner: project, ref, filepath } = parseGitUrl__default.default(url);
37
+ let branch = ref;
38
+ if (!branch) {
39
+ branch = await getBitbucketServerDefaultBranch(url, config);
40
+ }
41
+ const path = filepath ? `&path=${encodeURIComponent(decodeURIComponent(filepath))}` : "";
42
+ return `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/archive?format=tgz&at=${branch}&prefix=${project}-${repoName}${path}`;
43
+ }
44
+ function getBitbucketServerFileFetchUrl(url, config) {
45
+ try {
46
+ const { owner, name, ref, filepathtype, filepath } = parseGitUrl__default.default(url);
47
+ if (!owner || !name || filepathtype !== "browse" && filepathtype !== "raw" && filepathtype !== "src") {
48
+ throw new Error("Invalid Bitbucket Server URL or file path");
49
+ }
50
+ const pathWithoutSlash = filepath.replace(/^\//, "");
51
+ return `${config.apiBaseUrl}/projects/${owner}/repos/${name}/raw/${pathWithoutSlash}?at=${ref}`;
52
+ } catch (e) {
53
+ throw new Error(`Incorrect URL: ${url}, ${e}`);
54
+ }
55
+ }
56
+ function getBitbucketServerRequestOptions(config) {
57
+ const headers = {};
58
+ if (config.token) {
59
+ headers.Authorization = `Bearer ${config.token}`;
60
+ } else if (config.username && config.password) {
61
+ const buffer = Buffer.from(`${config.username}:${config.password}`, "utf8");
62
+ headers.Authorization = `Basic ${buffer.toString("base64")}`;
63
+ }
64
+ return {
65
+ headers
66
+ };
67
+ }
68
+
69
+ exports.getBitbucketServerDefaultBranch = getBitbucketServerDefaultBranch;
70
+ exports.getBitbucketServerDownloadUrl = getBitbucketServerDownloadUrl;
71
+ exports.getBitbucketServerFileFetchUrl = getBitbucketServerFileFetchUrl;
72
+ exports.getBitbucketServerRequestOptions = getBitbucketServerRequestOptions;
73
+ //# sourceMappingURL=core.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"core.cjs.js","sources":["../../src/bitbucketServer/core.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'cross-fetch';\nimport parseGitUrl from 'git-url-parse';\nimport { BitbucketServerIntegrationConfig } from './config';\n\n/**\n * Given a URL pointing to a path on a provider, returns the default branch.\n *\n * @param url - A URL pointing to a path\n * @param config - The relevant provider config\n * @public\n */\nexport async function getBitbucketServerDefaultBranch(\n url: string,\n config: BitbucketServerIntegrationConfig,\n): Promise<string> {\n const { name: repoName, owner: project } = parseGitUrl(url);\n\n // Bitbucket Server https://docs.atlassian.com/bitbucket-server/rest/7.9.0/bitbucket-rest.html#idp184\n let branchUrl = `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/default-branch`;\n\n let response = await fetch(\n branchUrl,\n getBitbucketServerRequestOptions(config),\n );\n\n if (response.status === 404) {\n // First try the new format, and then if it gets specifically a 404 it should try the old format\n // (to support old Atlassian Bitbucket Server v5.11.1 format )\n branchUrl = `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/branches/default`;\n response = await fetch(branchUrl, getBitbucketServerRequestOptions(config));\n }\n\n if (!response.ok) {\n const message = `Failed to retrieve default branch from ${branchUrl}, ${response.status} ${response.statusText}`;\n throw new Error(message);\n }\n\n const { displayId } = await response.json();\n const defaultBranch = displayId;\n if (!defaultBranch) {\n throw new Error(\n `Failed to read default branch from ${branchUrl}. ` +\n `Response ${response.status} ${response.json()}`,\n );\n }\n return defaultBranch;\n}\n\n/**\n * Given a URL pointing to a path on a provider, returns a URL that is suitable\n * for downloading the subtree.\n *\n * @param url - A URL pointing to a path\n * @param config - The relevant provider config\n * @public\n */\nexport async function getBitbucketServerDownloadUrl(\n url: string,\n config: BitbucketServerIntegrationConfig,\n): Promise<string> {\n const { name: repoName, owner: project, ref, filepath } = parseGitUrl(url);\n\n let branch = ref;\n if (!branch) {\n branch = await getBitbucketServerDefaultBranch(url, config);\n }\n // path will limit the downloaded content\n // /docs will only download the docs folder and everything below it\n // /docs/index.md will download the docs folder and everything below it\n const path = filepath\n ? `&path=${encodeURIComponent(decodeURIComponent(filepath))}`\n : '';\n return `${config.apiBaseUrl}/projects/${project}/repos/${repoName}/archive?format=tgz&at=${branch}&prefix=${project}-${repoName}${path}`;\n}\n\n/**\n * Given a URL pointing to a file on a provider, returns a URL that is suitable\n * for fetching the contents of the data.\n *\n * @remarks\n *\n * Converts\n * from: https://bitbucket.company.com/projectname/reponame/src/main/file.yaml\n * to: https://bitbucket.company.com/rest/api/1.0/project/projectname/reponame/raw/file.yaml?at=main\n *\n * @param url - A URL pointing to a file\n * @param config - The relevant provider config\n * @public\n */\nexport function getBitbucketServerFileFetchUrl(\n url: string,\n config: BitbucketServerIntegrationConfig,\n): string {\n try {\n const { owner, name, ref, filepathtype, filepath } = parseGitUrl(url);\n if (\n !owner ||\n !name ||\n (filepathtype !== 'browse' &&\n filepathtype !== 'raw' &&\n filepathtype !== 'src')\n ) {\n throw new Error('Invalid Bitbucket Server URL or file path');\n }\n\n const pathWithoutSlash = filepath.replace(/^\\//, '');\n return `${config.apiBaseUrl}/projects/${owner}/repos/${name}/raw/${pathWithoutSlash}?at=${ref}`;\n } catch (e) {\n throw new Error(`Incorrect URL: ${url}, ${e}`);\n }\n}\n\n/**\n * Gets the request options necessary to make requests to a given provider.\n *\n * @param config - The relevant provider config\n * @public\n */\nexport function getBitbucketServerRequestOptions(\n config: BitbucketServerIntegrationConfig,\n): { headers: Record<string, string> } {\n const headers: Record<string, string> = {};\n\n if (config.token) {\n headers.Authorization = `Bearer ${config.token}`;\n } else if (config.username && config.password) {\n const buffer = Buffer.from(`${config.username}:${config.password}`, 'utf8');\n headers.Authorization = `Basic ${buffer.toString('base64')}`;\n }\n\n return {\n headers,\n };\n}\n"],"names":["parseGitUrl","fetch"],"mappings":"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"}
@@ -0,0 +1,52 @@
1
+ 'use strict';
2
+
3
+ var helpers = require('../helpers.cjs.js');
4
+ var config = require('./config.cjs.js');
5
+ var core = require('./core.cjs.js');
6
+
7
+ class GerritIntegration {
8
+ constructor(integrationConfig) {
9
+ this.integrationConfig = integrationConfig;
10
+ }
11
+ static factory = ({ config: config$1 }) => {
12
+ const configs = config.readGerritIntegrationConfigs(
13
+ config$1.getOptionalConfigArray("integrations.gerrit") ?? []
14
+ );
15
+ return helpers.basicIntegrations(
16
+ configs.map((c) => new GerritIntegration(c)),
17
+ (i) => i.config.host
18
+ );
19
+ };
20
+ get type() {
21
+ return "gerrit";
22
+ }
23
+ get title() {
24
+ return this.integrationConfig.host;
25
+ }
26
+ get config() {
27
+ return this.integrationConfig;
28
+ }
29
+ resolveUrl(options) {
30
+ const { url, base, lineNumber } = options;
31
+ let updated;
32
+ if (url.startsWith("/")) {
33
+ const { basePath } = core.parseGitilesUrlRef(this.config, base);
34
+ return basePath + url;
35
+ }
36
+ if (url) {
37
+ updated = new URL(url, base);
38
+ } else {
39
+ updated = new URL(base);
40
+ }
41
+ if (lineNumber) {
42
+ updated.hash = lineNumber.toString();
43
+ }
44
+ return updated.toString();
45
+ }
46
+ resolveEditUrl(url) {
47
+ return url;
48
+ }
49
+ }
50
+
51
+ exports.GerritIntegration = GerritIntegration;
52
+ //# sourceMappingURL=GerritIntegration.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GerritIntegration.cjs.js","sources":["../../src/gerrit/GerritIntegration.ts"],"sourcesContent":["/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { basicIntegrations } from '../helpers';\nimport { ScmIntegration, ScmIntegrationsFactory } from '../types';\nimport {\n GerritIntegrationConfig,\n readGerritIntegrationConfigs,\n} from './config';\nimport { parseGitilesUrlRef } from './core';\n\n/**\n * A Gerrit based integration.\n *\n * @public\n */\nexport class GerritIntegration implements ScmIntegration {\n static factory: ScmIntegrationsFactory<GerritIntegration> = ({ config }) => {\n const configs = readGerritIntegrationConfigs(\n config.getOptionalConfigArray('integrations.gerrit') ?? [],\n );\n return basicIntegrations(\n configs.map(c => new GerritIntegration(c)),\n i => i.config.host,\n );\n };\n\n constructor(private readonly integrationConfig: GerritIntegrationConfig) {}\n\n get type(): string {\n return 'gerrit';\n }\n\n get title(): string {\n return this.integrationConfig.host;\n }\n\n get config(): GerritIntegrationConfig {\n return this.integrationConfig;\n }\n\n resolveUrl(options: {\n url: string;\n base: string;\n lineNumber?: number;\n }): string {\n const { url, base, lineNumber } = options;\n let updated;\n if (url.startsWith('/')) {\n const { basePath } = parseGitilesUrlRef(this.config, base);\n return basePath + url;\n }\n if (url) {\n updated = new URL(url, base);\n } else {\n updated = new URL(base);\n }\n if (lineNumber) {\n updated.hash = lineNumber.toString();\n }\n return updated.toString();\n }\n\n resolveEditUrl(url: string): string {\n // Not applicable for gerrit.\n return url;\n }\n}\n"],"names":["config","readGerritIntegrationConfigs","basicIntegrations","parseGitilesUrlRef"],"mappings":";;;;;;AA6BO,MAAM,iBAA4C,CAAA;AAAA,EAWvD,YAA6B,iBAA4C,EAAA;AAA5C,IAAA,IAAA,CAAA,iBAAA,GAAA,iBAAA,CAAA;AAAA,GAA6C;AAAA,EAV1E,OAAO,OAAA,GAAqD,CAAC,UAAEA,UAAa,KAAA;AAC1E,IAAA,MAAM,OAAU,GAAAC,mCAAA;AAAA,MACdD,QAAO,CAAA,sBAAA,CAAuB,qBAAqB,CAAA,IAAK,EAAC;AAAA,KAC3D,CAAA;AACA,IAAO,OAAAE,yBAAA;AAAA,MACL,QAAQ,GAAI,CAAA,CAAA,CAAA,KAAK,IAAI,iBAAA,CAAkB,CAAC,CAAC,CAAA;AAAA,MACzC,CAAA,CAAA,KAAK,EAAE,MAAO,CAAA,IAAA;AAAA,KAChB,CAAA;AAAA,GACF,CAAA;AAAA,EAIA,IAAI,IAAe,GAAA;AACjB,IAAO,OAAA,QAAA,CAAA;AAAA,GACT;AAAA,EAEA,IAAI,KAAgB,GAAA;AAClB,IAAA,OAAO,KAAK,iBAAkB,CAAA,IAAA,CAAA;AAAA,GAChC;AAAA,EAEA,IAAI,MAAkC,GAAA;AACpC,IAAA,OAAO,IAAK,CAAA,iBAAA,CAAA;AAAA,GACd;AAAA,EAEA,WAAW,OAIA,EAAA;AACT,IAAA,MAAM,EAAE,GAAA,EAAK,IAAM,EAAA,UAAA,EAAe,GAAA,OAAA,CAAA;AAClC,IAAI,IAAA,OAAA,CAAA;AACJ,IAAI,IAAA,GAAA,CAAI,UAAW,CAAA,GAAG,CAAG,EAAA;AACvB,MAAA,MAAM,EAAE,QAAS,EAAA,GAAIC,uBAAmB,CAAA,IAAA,CAAK,QAAQ,IAAI,CAAA,CAAA;AACzD,MAAA,OAAO,QAAW,GAAA,GAAA,CAAA;AAAA,KACpB;AACA,IAAA,IAAI,GAAK,EAAA;AACP,MAAU,OAAA,GAAA,IAAI,GAAI,CAAA,GAAA,EAAK,IAAI,CAAA,CAAA;AAAA,KACtB,MAAA;AACL,MAAU,OAAA,GAAA,IAAI,IAAI,IAAI,CAAA,CAAA;AAAA,KACxB;AACA,IAAA,IAAI,UAAY,EAAA;AACd,MAAQ,OAAA,CAAA,IAAA,GAAO,WAAW,QAAS,EAAA,CAAA;AAAA,KACrC;AACA,IAAA,OAAO,QAAQ,QAAS,EAAA,CAAA;AAAA,GAC1B;AAAA,EAEA,eAAe,GAAqB,EAAA;AAElC,IAAO,OAAA,GAAA,CAAA;AAAA,GACT;AACF;;;;"}