@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.
- package/CHANGELOG.md +20 -0
- package/dist/ScmIntegrations.cjs.js +110 -0
- package/dist/ScmIntegrations.cjs.js.map +1 -0
- package/dist/awsCodeCommit/AwsCodeCommitIntegration.cjs.js +58 -0
- package/dist/awsCodeCommit/AwsCodeCommitIntegration.cjs.js.map +1 -0
- package/dist/awsCodeCommit/config.cjs.js +26 -0
- package/dist/awsCodeCommit/config.cjs.js.map +1 -0
- package/dist/awsS3/AwsS3Integration.cjs.js +38 -0
- package/dist/awsS3/AwsS3Integration.cjs.js.map +1 -0
- package/dist/awsS3/config.cjs.js +53 -0
- package/dist/awsS3/config.cjs.js.map +1 -0
- package/dist/azure/AzureIntegration.cjs.js +62 -0
- package/dist/azure/AzureIntegration.cjs.js.map +1 -0
- package/dist/azure/AzureUrl.cjs.js +189 -0
- package/dist/azure/AzureUrl.cjs.js.map +1 -0
- package/dist/azure/CachedAzureDevOpsCredentialsProvider.cjs.js +76 -0
- package/dist/azure/CachedAzureDevOpsCredentialsProvider.cjs.js.map +1 -0
- package/dist/azure/DefaultAzureDevOpsCredentialsProvider.cjs.js +75 -0
- package/dist/azure/DefaultAzureDevOpsCredentialsProvider.cjs.js.map +1 -0
- package/dist/azure/config.cjs.js +153 -0
- package/dist/azure/config.cjs.js.map +1 -0
- package/dist/azure/core.cjs.js +18 -0
- package/dist/azure/core.cjs.js.map +1 -0
- package/dist/azure/deprecated.cjs.js +26 -0
- package/dist/azure/deprecated.cjs.js.map +1 -0
- package/dist/bitbucket/BitbucketIntegration.cjs.js +65 -0
- package/dist/bitbucket/BitbucketIntegration.cjs.js.map +1 -0
- package/dist/bitbucket/config.cjs.js +47 -0
- package/dist/bitbucket/config.cjs.js.map +1 -0
- package/dist/bitbucket/core.cjs.js +95 -0
- package/dist/bitbucket/core.cjs.js.map +1 -0
- package/dist/bitbucketCloud/BitbucketCloudIntegration.cjs.js +54 -0
- package/dist/bitbucketCloud/BitbucketCloudIntegration.cjs.js.map +1 -0
- package/dist/bitbucketCloud/config.cjs.js +30 -0
- package/dist/bitbucketCloud/config.cjs.js.map +1 -0
- package/dist/bitbucketCloud/core.cjs.js +78 -0
- package/dist/bitbucketCloud/core.cjs.js.map +1 -0
- package/dist/bitbucketServer/BitbucketServerIntegration.cjs.js +48 -0
- package/dist/bitbucketServer/BitbucketServerIntegration.cjs.js.map +1 -0
- package/dist/bitbucketServer/config.cjs.js +36 -0
- package/dist/bitbucketServer/config.cjs.js.map +1 -0
- package/dist/bitbucketServer/core.cjs.js +73 -0
- package/dist/bitbucketServer/core.cjs.js.map +1 -0
- package/dist/gerrit/GerritIntegration.cjs.js +52 -0
- package/dist/gerrit/GerritIntegration.cjs.js.map +1 -0
- package/dist/gerrit/config.cjs.js +60 -0
- package/dist/gerrit/config.cjs.js.map +1 -0
- package/dist/gerrit/core.cjs.js +179 -0
- package/dist/gerrit/core.cjs.js.map +1 -0
- package/dist/gitea/GiteaIntegration.cjs.js +34 -0
- package/dist/gitea/GiteaIntegration.cjs.js.map +1 -0
- package/dist/gitea/config.cjs.js +34 -0
- package/dist/gitea/config.cjs.js.map +1 -0
- package/dist/gitea/core.cjs.js +59 -0
- package/dist/gitea/core.cjs.js.map +1 -0
- package/dist/github/DefaultGithubCredentialsProvider.cjs.js +54 -0
- package/dist/github/DefaultGithubCredentialsProvider.cjs.js.map +1 -0
- package/dist/github/GithubIntegration.cjs.js +51 -0
- package/dist/github/GithubIntegration.cjs.js.map +1 -0
- package/dist/github/SingleInstanceGithubCredentialsProvider.cjs.js +211 -0
- package/dist/github/SingleInstanceGithubCredentialsProvider.cjs.js.map +1 -0
- package/dist/github/config.cjs.js +55 -0
- package/dist/github/config.cjs.js.map +1 -0
- package/dist/github/core.cjs.js +46 -0
- package/dist/github/core.cjs.js.map +1 -0
- package/dist/gitlab/DefaultGitlabCredentialsProvider.cjs.js +30 -0
- package/dist/gitlab/DefaultGitlabCredentialsProvider.cjs.js.map +1 -0
- package/dist/gitlab/GitLabIntegration.cjs.js +41 -0
- package/dist/gitlab/GitLabIntegration.cjs.js.map +1 -0
- package/dist/gitlab/SingleInstanceGitlabCredentialsProvider.cjs.js +24 -0
- package/dist/gitlab/SingleInstanceGitlabCredentialsProvider.cjs.js.map +1 -0
- package/dist/gitlab/config.cjs.js +60 -0
- package/dist/gitlab/config.cjs.js.map +1 -0
- package/dist/gitlab/core.cjs.js +84 -0
- package/dist/gitlab/core.cjs.js.map +1 -0
- package/dist/googleGcs/config.cjs.js +16 -0
- package/dist/googleGcs/config.cjs.js.map +1 -0
- package/dist/harness/HarnessIntegration.cjs.js +34 -0
- package/dist/harness/HarnessIntegration.cjs.js.map +1 -0
- package/dist/harness/config.cjs.js +22 -0
- package/dist/harness/config.cjs.js.map +1 -0
- package/dist/harness/core.cjs.js +101 -0
- package/dist/harness/core.cjs.js.map +1 -0
- package/dist/helpers.cjs.js +71 -0
- package/dist/helpers.cjs.js.map +1 -0
- package/dist/index.cjs.js +124 -2451
- package/dist/index.cjs.js.map +1 -1
- package/package.json +6 -6
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var fetch = require('cross-fetch');
|
|
4
|
+
var config = require('./config.cjs.js');
|
|
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
|
+
|
|
10
|
+
async function getGitLabFileFetchUrl(url, config) {
|
|
11
|
+
const projectID = await getProjectId(url, config);
|
|
12
|
+
return buildProjectUrl(url, projectID, config).toString();
|
|
13
|
+
}
|
|
14
|
+
function getGitLabRequestOptions(config, token) {
|
|
15
|
+
if (token) {
|
|
16
|
+
return {
|
|
17
|
+
headers: token.startsWith("gl") ? { "PRIVATE-TOKEN": token } : { Authorization: `Bearer ${token}` }
|
|
18
|
+
// Otherwise, it's a bearer token
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
const { token: configToken = "" } = config;
|
|
22
|
+
return {
|
|
23
|
+
headers: { "PRIVATE-TOKEN": configToken }
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
function buildProjectUrl(target, projectID, config$1) {
|
|
27
|
+
try {
|
|
28
|
+
const url = new URL(target);
|
|
29
|
+
const branchAndFilePath = url.pathname.split("/blob/").slice(1).join("/blob/");
|
|
30
|
+
const [branch, ...filePath] = branchAndFilePath.split("/");
|
|
31
|
+
const relativePath = config.getGitLabIntegrationRelativePath(config$1);
|
|
32
|
+
url.pathname = [
|
|
33
|
+
...relativePath ? [relativePath] : [],
|
|
34
|
+
"api/v4/projects",
|
|
35
|
+
projectID,
|
|
36
|
+
"repository/files",
|
|
37
|
+
encodeURIComponent(decodeURIComponent(filePath.join("/"))),
|
|
38
|
+
"raw"
|
|
39
|
+
].join("/");
|
|
40
|
+
url.search = `?ref=${branch}`;
|
|
41
|
+
return url;
|
|
42
|
+
} catch (e) {
|
|
43
|
+
throw new Error(`Incorrect url: ${target}, ${e}`);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
async function getProjectId(target, config$1) {
|
|
47
|
+
const url = new URL(target);
|
|
48
|
+
if (!url.pathname.includes("/blob/")) {
|
|
49
|
+
throw new Error(
|
|
50
|
+
`Failed converting ${url.pathname} to a project id. Url path must include /blob/.`
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
try {
|
|
54
|
+
let repo = url.pathname.split("/-/blob/")[0].split("/blob/")[0];
|
|
55
|
+
const relativePath = config.getGitLabIntegrationRelativePath(config$1);
|
|
56
|
+
if (relativePath) {
|
|
57
|
+
repo = repo.replace(relativePath, "");
|
|
58
|
+
}
|
|
59
|
+
const repoIDLookup = new URL(
|
|
60
|
+
`${url.origin}${relativePath}/api/v4/projects/${encodeURIComponent(
|
|
61
|
+
repo.replace(/^\//, "")
|
|
62
|
+
)}`
|
|
63
|
+
);
|
|
64
|
+
const response = await fetch__default.default(
|
|
65
|
+
repoIDLookup.toString(),
|
|
66
|
+
getGitLabRequestOptions(config$1)
|
|
67
|
+
);
|
|
68
|
+
const data = await response.json();
|
|
69
|
+
if (!response.ok) {
|
|
70
|
+
throw new Error(
|
|
71
|
+
`GitLab Error '${data.error}', ${data.error_description}`
|
|
72
|
+
);
|
|
73
|
+
}
|
|
74
|
+
return Number(data.id);
|
|
75
|
+
} catch (e) {
|
|
76
|
+
throw new Error(`Could not get GitLab project ID for: ${target}, ${e}`);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
exports.buildProjectUrl = buildProjectUrl;
|
|
81
|
+
exports.getGitLabFileFetchUrl = getGitLabFileFetchUrl;
|
|
82
|
+
exports.getGitLabRequestOptions = getGitLabRequestOptions;
|
|
83
|
+
exports.getProjectId = getProjectId;
|
|
84
|
+
//# sourceMappingURL=core.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"core.cjs.js","sources":["../../src/gitlab/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 {\n getGitLabIntegrationRelativePath,\n GitLabIntegrationConfig,\n} from './config';\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://gitlab.example.com/a/b/blob/master/c.yaml\n * to: https://gitlab.com/api/v4/projects/projectId/repository/c.yaml?ref=master\n * -or-\n * from: https://gitlab.com/groupA/teams/teamA/subgroupA/repoA/-/blob/branch/filepath\n * to: https://gitlab.com/api/v4/projects/projectId/repository/files/filepath?ref=branch\n *\n * @param url - A URL pointing to a file\n * @param config - The relevant provider config\n * @public\n */\nexport async function getGitLabFileFetchUrl(\n url: string,\n config: GitLabIntegrationConfig,\n): Promise<string> {\n const projectID = await getProjectId(url, config);\n return buildProjectUrl(url, projectID, config).toString();\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 getGitLabRequestOptions(\n config: GitLabIntegrationConfig,\n token?: string,\n): { headers: Record<string, string> } {\n if (token) {\n // If token comes from the user and starts with \"gl\", it's a private token (see https://docs.gitlab.com/ee/security/token_overview.html#token-prefixes)\n return {\n headers: token.startsWith('gl')\n ? { 'PRIVATE-TOKEN': token }\n : { Authorization: `Bearer ${token}` }, // Otherwise, it's a bearer token\n };\n }\n\n // If token not provided, fetch the integration token\n const { token: configToken = '' } = config;\n return {\n headers: { 'PRIVATE-TOKEN': configToken },\n };\n}\n\n// Converts\n// from: https://gitlab.com/groupA/teams/teamA/subgroupA/repoA/-/blob/branch/filepath\n// to: https://gitlab.com/api/v4/projects/projectId/repository/files/filepath?ref=branch\nexport function buildProjectUrl(\n target: string,\n projectID: Number,\n config: GitLabIntegrationConfig,\n): URL {\n try {\n const url = new URL(target);\n\n const branchAndFilePath = url.pathname\n .split('/blob/')\n .slice(1)\n .join('/blob/');\n const [branch, ...filePath] = branchAndFilePath.split('/');\n const relativePath = getGitLabIntegrationRelativePath(config);\n\n url.pathname = [\n ...(relativePath ? [relativePath] : []),\n 'api/v4/projects',\n projectID,\n 'repository/files',\n encodeURIComponent(decodeURIComponent(filePath.join('/'))),\n 'raw',\n ].join('/');\n\n url.search = `?ref=${branch}`;\n\n return url;\n } catch (e) {\n throw new Error(`Incorrect url: ${target}, ${e}`);\n }\n}\n\n// Convert\n// from: https://gitlab.com/groupA/teams/teamA/subgroupA/repoA/-/blob/branch/filepath\n// to: The project ID that corresponds to the URL\nexport async function getProjectId(\n target: string,\n config: GitLabIntegrationConfig,\n): Promise<number> {\n const url = new URL(target);\n\n if (!url.pathname.includes('/blob/')) {\n throw new Error(\n `Failed converting ${url.pathname} to a project id. Url path must include /blob/.`,\n );\n }\n\n try {\n let repo = url.pathname.split('/-/blob/')[0].split('/blob/')[0];\n\n // Get gitlab relative path\n const relativePath = getGitLabIntegrationRelativePath(config);\n\n // Check relative path exist and replace it if it's the case.\n if (relativePath) {\n repo = repo.replace(relativePath, '');\n }\n\n // Convert\n // to: https://gitlab.com/api/v4/projects/groupA%2Fteams%2FsubgroupA%2FteamA%2Frepo\n const repoIDLookup = new URL(\n `${url.origin}${relativePath}/api/v4/projects/${encodeURIComponent(\n repo.replace(/^\\//, ''),\n )}`,\n );\n\n const response = await fetch(\n repoIDLookup.toString(),\n getGitLabRequestOptions(config),\n );\n\n const data = await response.json();\n\n if (!response.ok) {\n throw new Error(\n `GitLab Error '${data.error}', ${data.error_description}`,\n );\n }\n\n return Number(data.id);\n } catch (e) {\n throw new Error(`Could not get GitLab project ID for: ${target}, ${e}`);\n }\n}\n"],"names":["config","getGitLabIntegrationRelativePath","fetch"],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
function readGoogleGcsIntegrationConfig(config) {
|
|
4
|
+
if (!config) {
|
|
5
|
+
return {};
|
|
6
|
+
}
|
|
7
|
+
if (!config.has("clientEmail") && !config.has("privateKey")) {
|
|
8
|
+
return {};
|
|
9
|
+
}
|
|
10
|
+
const privateKey = config.getString("privateKey").split("\\n").join("\n");
|
|
11
|
+
const clientEmail = config.getString("clientEmail");
|
|
12
|
+
return { clientEmail, privateKey };
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
exports.readGoogleGcsIntegrationConfig = readGoogleGcsIntegrationConfig;
|
|
16
|
+
//# sourceMappingURL=config.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.cjs.js","sources":["../../src/googleGcs/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\n/**\n * The configuration parameters for a single Google Cloud Storage provider.\n *\n * @public\n */\nexport type GoogleGcsIntegrationConfig = {\n /**\n * Service account email used to authenticate requests.\n */\n clientEmail?: string;\n /**\n * Service account private key used to authenticate requests.\n */\n privateKey?: string;\n};\n\n/**\n * Reads a single Google GCS integration config.\n *\n * @param config - The config object of a single integration\n * @public\n */\nexport function readGoogleGcsIntegrationConfig(\n config: Config,\n): GoogleGcsIntegrationConfig {\n if (!config) {\n return {};\n }\n\n if (!config.has('clientEmail') && !config.has('privateKey')) {\n return {};\n }\n\n const privateKey = config.getString('privateKey').split('\\\\n').join('\\n');\n\n const clientEmail = config.getString('clientEmail');\n return { clientEmail: clientEmail, privateKey: privateKey };\n}\n"],"names":[],"mappings":";;AAwCO,SAAS,+BACd,MAC4B,EAAA;AAC5B,EAAA,IAAI,CAAC,MAAQ,EAAA;AACX,IAAA,OAAO,EAAC,CAAA;AAAA,GACV;AAEA,EAAI,IAAA,CAAC,OAAO,GAAI,CAAA,aAAa,KAAK,CAAC,MAAA,CAAO,GAAI,CAAA,YAAY,CAAG,EAAA;AAC3D,IAAA,OAAO,EAAC,CAAA;AAAA,GACV;AAEA,EAAM,MAAA,UAAA,GAAa,OAAO,SAAU,CAAA,YAAY,EAAE,KAAM,CAAA,KAAK,CAAE,CAAA,IAAA,CAAK,IAAI,CAAA,CAAA;AAExE,EAAM,MAAA,WAAA,GAAc,MAAO,CAAA,SAAA,CAAU,aAAa,CAAA,CAAA;AAClD,EAAO,OAAA,EAAE,aAA0B,UAAuB,EAAA,CAAA;AAC5D;;;;"}
|
|
@@ -0,0 +1,34 @@
|
|
|
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 HarnessIntegration {
|
|
8
|
+
constructor(config) {
|
|
9
|
+
this.config = config;
|
|
10
|
+
}
|
|
11
|
+
static factory = ({ config: config$1 }) => {
|
|
12
|
+
const configs = config$1.getOptionalConfigArray("integrations.harness") ?? [];
|
|
13
|
+
const harnessConfigs = configs.map((c) => config.readHarnessConfig(c));
|
|
14
|
+
return helpers.basicIntegrations(
|
|
15
|
+
harnessConfigs.map((c) => new HarnessIntegration(c)),
|
|
16
|
+
(harness) => harness.config.host
|
|
17
|
+
);
|
|
18
|
+
};
|
|
19
|
+
get type() {
|
|
20
|
+
return "harness";
|
|
21
|
+
}
|
|
22
|
+
get title() {
|
|
23
|
+
return this.config.host;
|
|
24
|
+
}
|
|
25
|
+
resolveUrl(options) {
|
|
26
|
+
return helpers.defaultScmResolveUrl(options);
|
|
27
|
+
}
|
|
28
|
+
resolveEditUrl(url) {
|
|
29
|
+
return core.getHarnessEditContentsUrl(this.config, url);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
exports.HarnessIntegration = HarnessIntegration;
|
|
34
|
+
//# sourceMappingURL=HarnessIntegration.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"HarnessIntegration.cjs.js","sources":["../../src/harness/HarnessIntegration.ts"],"sourcesContent":["/*\n * Copyright 2024 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 */\nimport { basicIntegrations, defaultScmResolveUrl } from '../helpers';\nimport { ScmIntegration, ScmIntegrationsFactory } from '../types';\nimport { HarnessIntegrationConfig, readHarnessConfig } from './config';\nimport { getHarnessEditContentsUrl } from './core';\n\n/**\n * A Harness Code based integration.\n *\n * @public\n */\nexport class HarnessIntegration implements ScmIntegration {\n static factory: ScmIntegrationsFactory<HarnessIntegration> = ({ config }) => {\n const configs = config.getOptionalConfigArray('integrations.harness') ?? [];\n const harnessConfigs = configs.map(c => readHarnessConfig(c));\n\n return basicIntegrations(\n harnessConfigs.map(c => new HarnessIntegration(c)),\n (harness: HarnessIntegration) => harness.config.host,\n );\n };\n\n constructor(readonly config: HarnessIntegrationConfig) {}\n\n get type(): string {\n return 'harness';\n }\n\n get title(): string {\n return this.config.host;\n }\n\n resolveUrl(options: {\n url: string;\n base: string;\n lineNumber?: number | undefined;\n }): string {\n return defaultScmResolveUrl(options);\n }\n\n resolveEditUrl(url: string): string {\n return getHarnessEditContentsUrl(this.config, url);\n }\n}\n"],"names":["config","readHarnessConfig","basicIntegrations","defaultScmResolveUrl","getHarnessEditContentsUrl"],"mappings":";;;;;;AAyBO,MAAM,kBAA6C,CAAA;AAAA,EAWxD,YAAqB,MAAkC,EAAA;AAAlC,IAAA,IAAA,CAAA,MAAA,GAAA,MAAA,CAAA;AAAA,GAAmC;AAAA,EAVxD,OAAO,OAAA,GAAsD,CAAC,UAAEA,UAAa,KAAA;AAC3E,IAAA,MAAM,OAAU,GAAAA,QAAA,CAAO,sBAAuB,CAAA,sBAAsB,KAAK,EAAC,CAAA;AAC1E,IAAA,MAAM,iBAAiB,OAAQ,CAAA,GAAA,CAAI,CAAK,CAAA,KAAAC,wBAAA,CAAkB,CAAC,CAAC,CAAA,CAAA;AAE5D,IAAO,OAAAC,yBAAA;AAAA,MACL,eAAe,GAAI,CAAA,CAAA,CAAA,KAAK,IAAI,kBAAA,CAAmB,CAAC,CAAC,CAAA;AAAA,MACjD,CAAC,OAAgC,KAAA,OAAA,CAAQ,MAAO,CAAA,IAAA;AAAA,KAClD,CAAA;AAAA,GACF,CAAA;AAAA,EAIA,IAAI,IAAe,GAAA;AACjB,IAAO,OAAA,SAAA,CAAA;AAAA,GACT;AAAA,EAEA,IAAI,KAAgB,GAAA;AAClB,IAAA,OAAO,KAAK,MAAO,CAAA,IAAA,CAAA;AAAA,GACrB;AAAA,EAEA,WAAW,OAIA,EAAA;AACT,IAAA,OAAOC,6BAAqB,OAAO,CAAA,CAAA;AAAA,GACrC;AAAA,EAEA,eAAe,GAAqB,EAAA;AAClC,IAAO,OAAAC,8BAAA,CAA0B,IAAK,CAAA,MAAA,EAAQ,GAAG,CAAA,CAAA;AAAA,GACnD;AACF;;;;"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var helpers = require('../helpers.cjs.js');
|
|
4
|
+
|
|
5
|
+
function readHarnessConfig(config) {
|
|
6
|
+
const host = config.getString("host");
|
|
7
|
+
const token = config.getOptionalString("token");
|
|
8
|
+
const apiKey = config.getOptionalString("apiKey");
|
|
9
|
+
if (!helpers.isValidHost(host)) {
|
|
10
|
+
throw new Error(
|
|
11
|
+
`Invalid Harness Code integration config, '${host}' is not a valid host`
|
|
12
|
+
);
|
|
13
|
+
}
|
|
14
|
+
return {
|
|
15
|
+
host,
|
|
16
|
+
apiKey,
|
|
17
|
+
token
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
exports.readHarnessConfig = readHarnessConfig;
|
|
22
|
+
//# sourceMappingURL=config.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.cjs.js","sources":["../../src/harness/config.ts"],"sourcesContent":["/*\n * Copyright 2024 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 { isValidHost } from '../helpers';\n\n/**\n * The configuration for a single Harness integration.\n *\n * @public\n */\nexport type HarnessIntegrationConfig = {\n /**\n * The host of the target that this matches on, e.g. \"app.harness.io\"\n */\n host: string;\n /**\n * The password or http token to use for authentication.\n */\n token?: string;\n /**\n * The API key to use for authentication.\n */\n apiKey?: string;\n};\n\n/**\n * Parses a location config block for use in HarnessIntegration\n *\n * @public\n */\nexport function readHarnessConfig(config: Config): HarnessIntegrationConfig {\n const host = config.getString('host');\n const token = config.getOptionalString('token');\n const apiKey = config.getOptionalString('apiKey');\n\n if (!isValidHost(host)) {\n throw new Error(\n `Invalid Harness Code integration config, '${host}' is not a valid host`,\n );\n }\n\n return {\n host,\n apiKey,\n token,\n };\n}\n"],"names":["isValidHost"],"mappings":";;;;AA4CO,SAAS,kBAAkB,MAA0C,EAAA;AAC1E,EAAM,MAAA,IAAA,GAAO,MAAO,CAAA,SAAA,CAAU,MAAM,CAAA,CAAA;AACpC,EAAM,MAAA,KAAA,GAAQ,MAAO,CAAA,iBAAA,CAAkB,OAAO,CAAA,CAAA;AAC9C,EAAM,MAAA,MAAA,GAAS,MAAO,CAAA,iBAAA,CAAkB,QAAQ,CAAA,CAAA;AAEhD,EAAI,IAAA,CAACA,mBAAY,CAAA,IAAI,CAAG,EAAA;AACtB,IAAA,MAAM,IAAI,KAAA;AAAA,MACR,6CAA6C,IAAI,CAAA,qBAAA,CAAA;AAAA,KACnD,CAAA;AAAA,GACF;AAEA,EAAO,OAAA;AAAA,IACL,IAAA;AAAA,IACA,MAAA;AAAA,IACA,KAAA;AAAA,GACF,CAAA;AACF;;;;"}
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
function getHarnessEditContentsUrl(config, url) {
|
|
4
|
+
const parsedUrl = parseHarnessUrl(config, url);
|
|
5
|
+
return `${parsedUrl.baseUrl}/ng/account/${parsedUrl.accountId}/module/code${parsedUrl.orgName !== "" ? `/orgs/${parsedUrl.orgName}` : ""}${parsedUrl.projectName !== "" ? `/projects/${parsedUrl.projectName}` : ""}/repos/${parsedUrl.repoName}/files/${parsedUrl.branch}/~/${parsedUrl.path}`;
|
|
6
|
+
}
|
|
7
|
+
function getHarnessFileContentsUrl(config, url) {
|
|
8
|
+
const parsedUrl = parseHarnessUrl(config, url);
|
|
9
|
+
let constructedUrl = `${parsedUrl.baseUrl}/gateway/code/api/v1/repos/${parsedUrl.accountId}`;
|
|
10
|
+
if (parsedUrl.orgName) {
|
|
11
|
+
constructedUrl += `/${parsedUrl.orgName}`;
|
|
12
|
+
}
|
|
13
|
+
if (parsedUrl.projectName) {
|
|
14
|
+
constructedUrl += `/${parsedUrl.projectName}`;
|
|
15
|
+
}
|
|
16
|
+
constructedUrl += `/${parsedUrl.repoName}/+/raw/${parsedUrl.path}?routingId=${parsedUrl.accountId}&git_ref=refs/heads/${parsedUrl.refString}`;
|
|
17
|
+
return constructedUrl;
|
|
18
|
+
}
|
|
19
|
+
function getHarnessArchiveUrl(config, url) {
|
|
20
|
+
const parsedUrl = parseHarnessUrl(config, url);
|
|
21
|
+
let constructedUrl = `${parsedUrl.baseUrl}/gateway/code/api/v1/repos/${parsedUrl.accountId}`;
|
|
22
|
+
if (parsedUrl.orgName) {
|
|
23
|
+
constructedUrl += `/${parsedUrl.orgName}`;
|
|
24
|
+
}
|
|
25
|
+
if (parsedUrl.projectName) {
|
|
26
|
+
constructedUrl += `/${parsedUrl.projectName}`;
|
|
27
|
+
}
|
|
28
|
+
constructedUrl += `/${parsedUrl.repoName}/+/archive/${parsedUrl.branch}.zip?routingId=${parsedUrl.accountId}`;
|
|
29
|
+
return constructedUrl;
|
|
30
|
+
}
|
|
31
|
+
function getHarnessLatestCommitUrl(config, url) {
|
|
32
|
+
const parsedUrl = parseHarnessUrl(config, url);
|
|
33
|
+
let constructedUrl = `${parsedUrl.baseUrl}/gateway/code/api/v1/repos/${parsedUrl.accountId}`;
|
|
34
|
+
if (parsedUrl.orgName) {
|
|
35
|
+
constructedUrl += `/${parsedUrl.orgName}`;
|
|
36
|
+
}
|
|
37
|
+
if (parsedUrl.projectName) {
|
|
38
|
+
constructedUrl += `/${parsedUrl.projectName}`;
|
|
39
|
+
}
|
|
40
|
+
constructedUrl += `/${parsedUrl.repoName}/+/content?routingId=${parsedUrl.accountId}&include_commit=true&git_ref=refs/heads/${parsedUrl.branch}`;
|
|
41
|
+
return constructedUrl;
|
|
42
|
+
}
|
|
43
|
+
function getHarnessRequestOptions(config) {
|
|
44
|
+
const headers = {};
|
|
45
|
+
const { token, apiKey } = config;
|
|
46
|
+
if (apiKey) {
|
|
47
|
+
headers["x-api-key"] = apiKey;
|
|
48
|
+
} else if (token) {
|
|
49
|
+
headers.Authorization = `Bearer ${token}`;
|
|
50
|
+
}
|
|
51
|
+
return {
|
|
52
|
+
headers
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
function parseHarnessUrl(config, url) {
|
|
56
|
+
const baseUrl = `https://${config.host}`;
|
|
57
|
+
try {
|
|
58
|
+
const pathUrl = new URL(url);
|
|
59
|
+
const pathSegments = pathUrl.pathname.split("/").filter((segment) => segment !== "");
|
|
60
|
+
const urlParts = pathUrl.pathname.split("/");
|
|
61
|
+
const accountIdIndex = pathSegments.findIndex((segment) => segment === "account") + 1;
|
|
62
|
+
const accountId = pathSegments[accountIdIndex];
|
|
63
|
+
const orgNameIndex = pathSegments.findIndex((segment) => segment === "orgs");
|
|
64
|
+
const orgName = orgNameIndex !== -1 ? pathSegments[orgNameIndex + 1] : "";
|
|
65
|
+
const projectNameIndex = pathSegments.findIndex(
|
|
66
|
+
(segment) => segment === "projects"
|
|
67
|
+
);
|
|
68
|
+
const projectName = projectNameIndex !== -1 ? pathSegments[projectNameIndex + 1] : "";
|
|
69
|
+
const repoNameIndex = pathSegments.findIndex(
|
|
70
|
+
(segment, index) => segment === "repos" && index > Math.max(accountIdIndex, orgNameIndex, projectNameIndex)
|
|
71
|
+
) + 1;
|
|
72
|
+
const repoName = pathSegments[repoNameIndex];
|
|
73
|
+
const refAndPath = urlParts.slice(
|
|
74
|
+
urlParts.findIndex((i) => i === "files" || i === "edit") + 1
|
|
75
|
+
);
|
|
76
|
+
const refIndex = refAndPath.findIndex((item) => item === "~");
|
|
77
|
+
const refString = refAndPath.slice(0, refIndex).join("/");
|
|
78
|
+
const pathWithoutSlash = refIndex !== -1 ? refAndPath.slice(refIndex + 1).join("/").replace(/^\//, "") : "";
|
|
79
|
+
return {
|
|
80
|
+
baseUrl,
|
|
81
|
+
accountId,
|
|
82
|
+
orgName,
|
|
83
|
+
projectName,
|
|
84
|
+
refString,
|
|
85
|
+
path: pathWithoutSlash,
|
|
86
|
+
repoName,
|
|
87
|
+
refDashStr: refAndPath.slice(0, refIndex).join("-"),
|
|
88
|
+
branch: refIndex !== -1 ? refAndPath.slice(0, refIndex).join("/") : refAndPath.join("/")
|
|
89
|
+
};
|
|
90
|
+
} catch (e) {
|
|
91
|
+
throw new Error(`Incorrect URL: ${url}, ${e}`);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
exports.getHarnessArchiveUrl = getHarnessArchiveUrl;
|
|
96
|
+
exports.getHarnessEditContentsUrl = getHarnessEditContentsUrl;
|
|
97
|
+
exports.getHarnessFileContentsUrl = getHarnessFileContentsUrl;
|
|
98
|
+
exports.getHarnessLatestCommitUrl = getHarnessLatestCommitUrl;
|
|
99
|
+
exports.getHarnessRequestOptions = getHarnessRequestOptions;
|
|
100
|
+
exports.parseHarnessUrl = parseHarnessUrl;
|
|
101
|
+
//# sourceMappingURL=core.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"core.cjs.js","sources":["../../src/harness/core.ts"],"sourcesContent":["/*\n * Copyright 2024 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 */\nimport { HarnessIntegrationConfig } from './config';\n\n/**\n * Given a URL pointing to a file, returns a URL\n * for editing the contents of the data.\n *\n * @remarks\n *\n * Converts\n * from: https://app.harness.io/a/b/src/branchname/path/to/c.yaml\n * or: https://app.harness.io/a/b/_edit/branchname/path/to/c.yaml\n *\n * @param url - A URL pointing to a file\n * @param config - The relevant provider config\n * @public\n */\nexport function getHarnessEditContentsUrl(\n config: HarnessIntegrationConfig,\n url: string,\n) {\n const parsedUrl = parseHarnessUrl(config, url);\n\n return `${parsedUrl.baseUrl}/ng/account/${parsedUrl.accountId}/module/code${\n parsedUrl.orgName !== '' ? `/orgs/${parsedUrl.orgName}` : ''\n }${\n parsedUrl.projectName !== '' ? `/projects/${parsedUrl.projectName}` : ''\n }/repos/${parsedUrl.repoName}/files/${parsedUrl.branch}/~/${parsedUrl.path}`;\n}\n\n/**\n * Given a file path URL,\n * it returns an API URL which returns the contents of the file .\n * @remarks\n *\n * Converts\n * from: https://app.harness.io/ng/account/accountId/module/code/orgs/orgName/projects/projName/repos/repoName/files/refMain/~/all-apis.yaml\n * https://qa.harness.io/ng/account/bDCAuAjFSJCLFj_0ug3lCg/module/code/orgs/HiteshTest/repos/impoorter/files/main/~/catalog.yaml\n * to: https://app.harness.io/gateway/code/api/v1/repos/accountId/orgName/projName/repoName/+/content/all-apis.yaml?routingId=accountId&include_commit=false&ref=refMain\n *\n * @param url - A URL pointing to a file\n * @param config - The relevant provider config\n * @public\n */\nexport function getHarnessFileContentsUrl(\n config: HarnessIntegrationConfig,\n url: string,\n) {\n const parsedUrl = parseHarnessUrl(config, url);\n\n let constructedUrl = `${parsedUrl.baseUrl}/gateway/code/api/v1/repos/${parsedUrl.accountId}`;\n\n if (parsedUrl.orgName) {\n constructedUrl += `/${parsedUrl.orgName}`;\n }\n\n if (parsedUrl.projectName) {\n constructedUrl += `/${parsedUrl.projectName}`;\n }\n\n constructedUrl += `/${parsedUrl.repoName}/+/raw/${parsedUrl.path}?routingId=${parsedUrl.accountId}&git_ref=refs/heads/${parsedUrl.refString}`;\n\n return constructedUrl;\n}\n\n/**\n * Given a URL pointing to a repository/path, returns a URL\n * for archive contents of the repository.\n *\n * @remarks\n *\n * Converts\n * from: https://qa.harness.io/ng/account/accountId/module/code/orgs/orgId/projects/projectName/repos/repoName/files/branch/~/fileName\n * to: https://qa.harness.io/gateway/code/api/v1/repos/accountId/orgId/projectName/repoName/+/archive/branch.zip?routingId=accountId\n *\n * @param url - A URL pointing to a repository/path\n * @param config - The relevant provider config\n * @public\n */\nexport function getHarnessArchiveUrl(\n config: HarnessIntegrationConfig,\n url: string,\n) {\n const parsedUrl = parseHarnessUrl(config, url);\n\n let constructedUrl = `${parsedUrl.baseUrl}/gateway/code/api/v1/repos/${parsedUrl.accountId}`;\n\n if (parsedUrl.orgName) {\n constructedUrl += `/${parsedUrl.orgName}`;\n }\n\n if (parsedUrl.projectName) {\n constructedUrl += `/${parsedUrl.projectName}`;\n }\n\n constructedUrl += `/${parsedUrl.repoName}/+/archive/${parsedUrl.branch}.zip?routingId=${parsedUrl.accountId}`;\n\n return constructedUrl;\n}\n\n/**\n * Given a URL pointing to a repository branch, returns a URL\n * for latest commit information.\n *\n * @remarks\n *\n * Converts\n * from: https://app.harness.io/ng/account/accountId/module/code/orgs/orgName/projects/projectName/repos/repoName/files/branchName\n * to: https://app.harness.io/gateway/code/api/v1/repos/accountId/orgName/projectName/repoName/+/content?routingId=accountId&include_commit=true&git_ref=refs/heads/branchName\n *\n * @param url - A URL pointing to a repository branch\n * @param config - The relevant provider config\n * @public\n */\nexport function getHarnessLatestCommitUrl(\n config: HarnessIntegrationConfig,\n url: string,\n) {\n const parsedUrl = parseHarnessUrl(config, url);\n\n let constructedUrl = `${parsedUrl.baseUrl}/gateway/code/api/v1/repos/${parsedUrl.accountId}`;\n\n if (parsedUrl.orgName) {\n constructedUrl += `/${parsedUrl.orgName}`;\n }\n\n if (parsedUrl.projectName) {\n constructedUrl += `/${parsedUrl.projectName}`;\n }\n\n constructedUrl += `/${parsedUrl.repoName}/+/content?routingId=${parsedUrl.accountId}&include_commit=true&git_ref=refs/heads/${parsedUrl.branch}`;\n\n return constructedUrl;\n}\n\n/**\n * Return request headers for a Harness Code provider.\n *\n * @param config - A Harness Code provider config\n * @public\n */\nexport function getHarnessRequestOptions(config: HarnessIntegrationConfig): {\n headers?: Record<string, string>;\n} {\n const headers: Record<string, string> = {};\n const { token, apiKey } = config;\n\n if (apiKey) {\n headers['x-api-key'] = apiKey;\n } else if (token) {\n headers.Authorization = `Bearer ${token}`;\n }\n\n return {\n headers,\n };\n}\n\n/**\n * Return parsed git url properties.\n *\n * @param config - A Harness provider config\n * @param url - A URL pointing to a repository\n * @public\n */\nexport function parseHarnessUrl(\n config: HarnessIntegrationConfig,\n url: string,\n): {\n baseUrl: string;\n accountId: string;\n orgName: string;\n projectName: string;\n refString: string;\n repoName: string;\n path: string;\n refDashStr: string;\n branch: string;\n} {\n const baseUrl = `https://${config.host}`;\n try {\n const pathUrl = new URL(url);\n const pathSegments = pathUrl.pathname\n .split('/')\n .filter(segment => segment !== '');\n const urlParts = pathUrl.pathname.split('/');\n\n const accountIdIndex =\n pathSegments.findIndex(segment => segment === 'account') + 1;\n const accountId = pathSegments[accountIdIndex];\n\n const orgNameIndex = pathSegments.findIndex(segment => segment === 'orgs');\n const orgName = orgNameIndex !== -1 ? pathSegments[orgNameIndex + 1] : '';\n const projectNameIndex = pathSegments.findIndex(\n segment => segment === 'projects',\n );\n\n const projectName =\n projectNameIndex !== -1 ? pathSegments[projectNameIndex + 1] : '';\n // Adjust repoNameIndex to correctly identify the repository name\n const repoNameIndex =\n pathSegments.findIndex(\n (segment, index) =>\n segment === 'repos' &&\n index > Math.max(accountIdIndex, orgNameIndex, projectNameIndex),\n ) + 1;\n const repoName = pathSegments[repoNameIndex];\n const refAndPath = urlParts.slice(\n urlParts.findIndex(i => i === 'files' || i === 'edit') + 1,\n );\n const refIndex = refAndPath.findIndex(item => item === '~');\n\n const refString = refAndPath.slice(0, refIndex).join('/');\n const pathWithoutSlash =\n refIndex !== -1\n ? refAndPath\n .slice(refIndex + 1)\n .join('/')\n .replace(/^\\//, '')\n : '';\n\n return {\n baseUrl: baseUrl,\n accountId: accountId,\n orgName: orgName,\n projectName: projectName,\n refString: refString,\n path: pathWithoutSlash,\n repoName: repoName,\n refDashStr: refAndPath.slice(0, refIndex).join('-'),\n branch:\n refIndex !== -1\n ? refAndPath.slice(0, refIndex).join('/')\n : refAndPath.join('/'),\n };\n } catch (e) {\n throw new Error(`Incorrect URL: ${url}, ${e}`);\n }\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var parseGitUrl = require('git-url-parse');
|
|
4
|
+
var lodash = require('lodash');
|
|
5
|
+
|
|
6
|
+
function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
|
|
7
|
+
|
|
8
|
+
var parseGitUrl__default = /*#__PURE__*/_interopDefaultCompat(parseGitUrl);
|
|
9
|
+
|
|
10
|
+
function isValidHost(host) {
|
|
11
|
+
const check = new URL("http://example.com");
|
|
12
|
+
check.host = host;
|
|
13
|
+
return check.host === host;
|
|
14
|
+
}
|
|
15
|
+
function isValidUrl(url) {
|
|
16
|
+
try {
|
|
17
|
+
new URL(url);
|
|
18
|
+
return true;
|
|
19
|
+
} catch {
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
function basicIntegrations(integrations, getHost) {
|
|
24
|
+
return {
|
|
25
|
+
list() {
|
|
26
|
+
return integrations;
|
|
27
|
+
},
|
|
28
|
+
byUrl(url) {
|
|
29
|
+
try {
|
|
30
|
+
const parsed = typeof url === "string" ? new URL(url) : url;
|
|
31
|
+
return integrations.find((i) => getHost(i) === parsed.host);
|
|
32
|
+
} catch {
|
|
33
|
+
return void 0;
|
|
34
|
+
}
|
|
35
|
+
},
|
|
36
|
+
byHost(host) {
|
|
37
|
+
return integrations.find((i) => getHost(i) === host);
|
|
38
|
+
}
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
function defaultScmResolveUrl(options) {
|
|
42
|
+
const { url, base, lineNumber } = options;
|
|
43
|
+
try {
|
|
44
|
+
new URL(url);
|
|
45
|
+
return url;
|
|
46
|
+
} catch {
|
|
47
|
+
}
|
|
48
|
+
let updated;
|
|
49
|
+
if (url.startsWith("/")) {
|
|
50
|
+
const { href, filepath } = parseGitUrl__default.default(base);
|
|
51
|
+
updated = new URL(href);
|
|
52
|
+
const repoRootPath = lodash.trimEnd(
|
|
53
|
+
updated.pathname.substring(0, updated.pathname.length - filepath.length),
|
|
54
|
+
"/"
|
|
55
|
+
);
|
|
56
|
+
updated.pathname = `${repoRootPath}${url}`;
|
|
57
|
+
} else {
|
|
58
|
+
updated = new URL(url, base);
|
|
59
|
+
}
|
|
60
|
+
updated.search = new URL(base).search;
|
|
61
|
+
if (lineNumber) {
|
|
62
|
+
updated.hash = `L${lineNumber}`;
|
|
63
|
+
}
|
|
64
|
+
return updated.toString();
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
exports.basicIntegrations = basicIntegrations;
|
|
68
|
+
exports.defaultScmResolveUrl = defaultScmResolveUrl;
|
|
69
|
+
exports.isValidHost = isValidHost;
|
|
70
|
+
exports.isValidUrl = isValidUrl;
|
|
71
|
+
//# sourceMappingURL=helpers.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"helpers.cjs.js","sources":["../src/helpers.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 { trimEnd } from 'lodash';\nimport { ScmIntegration, ScmIntegrationsGroup } from './types';\n\n/** Checks whether the given argument is a valid URL hostname */\nexport function isValidHost(host: string): boolean {\n const check = new URL('http://example.com');\n check.host = host;\n return check.host === host;\n}\n\n/** Checks whether the given argument is a valid URL */\nexport function isValidUrl(url: string): boolean {\n try {\n // eslint-disable-next-line no-new\n new URL(url);\n return true;\n } catch {\n return false;\n }\n}\n\nexport function basicIntegrations<T extends ScmIntegration>(\n integrations: T[],\n getHost: (integration: T) => string,\n): ScmIntegrationsGroup<T> {\n return {\n list(): T[] {\n return integrations;\n },\n byUrl(url: string | URL): T | undefined {\n try {\n const parsed = typeof url === 'string' ? new URL(url) : url;\n return integrations.find(i => getHost(i) === parsed.host);\n } catch {\n return undefined;\n }\n },\n byHost(host: string): T | undefined {\n return integrations.find(i => getHost(i) === host);\n },\n };\n}\n\n/**\n * Default implementation of {@link ScmIntegration} `resolveUrl`, that only\n * works with URL pathname based providers.\n *\n * @public\n */\nexport function defaultScmResolveUrl(options: {\n url: string;\n base: string;\n lineNumber?: number;\n}): string {\n const { url, base, lineNumber } = options;\n\n // If it is a fully qualified URL - then return it verbatim\n try {\n // eslint-disable-next-line no-new\n new URL(url);\n return url;\n } catch {\n // ignore intentionally\n }\n\n let updated: URL;\n\n if (url.startsWith('/')) {\n // If it is an absolute path, move relative to the repo root\n const { href, filepath } = parseGitUrl(base);\n\n updated = new URL(href);\n\n const repoRootPath = trimEnd(\n updated.pathname.substring(0, updated.pathname.length - filepath.length),\n '/',\n );\n updated.pathname = `${repoRootPath}${url}`;\n } else {\n // For relative URLs, just let the default URL constructor handle the\n // resolving. Note that this essentially will treat the last segment of the\n // base as a file - NOT a folder - unless the url ends in a slash.\n updated = new URL(url, base);\n }\n\n updated.search = new URL(base).search;\n if (lineNumber) {\n updated.hash = `L${lineNumber}`;\n }\n return updated.toString();\n}\n\n/**\n * Sets up handlers for request mocking\n *\n * Copied from test-utils, as that is a frontend-only package\n *\n * @param worker - service worker\n */\nexport function registerMswTestHooks(worker: {\n listen: (t: any) => void;\n close: () => void;\n resetHandlers: () => void;\n}) {\n beforeAll(() => worker.listen({ onUnhandledRequest: 'error' }));\n afterAll(() => worker.close());\n afterEach(() => worker.resetHandlers());\n}\n"],"names":["parseGitUrl","trimEnd"],"mappings":";;;;;;;;;AAqBO,SAAS,YAAY,IAAuB,EAAA;AACjD,EAAM,MAAA,KAAA,GAAQ,IAAI,GAAA,CAAI,oBAAoB,CAAA,CAAA;AAC1C,EAAA,KAAA,CAAM,IAAO,GAAA,IAAA,CAAA;AACb,EAAA,OAAO,MAAM,IAAS,KAAA,IAAA,CAAA;AACxB,CAAA;AAGO,SAAS,WAAW,GAAsB,EAAA;AAC/C,EAAI,IAAA;AAEF,IAAA,IAAI,IAAI,GAAG,CAAA,CAAA;AACX,IAAO,OAAA,IAAA,CAAA;AAAA,GACD,CAAA,MAAA;AACN,IAAO,OAAA,KAAA,CAAA;AAAA,GACT;AACF,CAAA;AAEgB,SAAA,iBAAA,CACd,cACA,OACyB,EAAA;AACzB,EAAO,OAAA;AAAA,IACL,IAAY,GAAA;AACV,MAAO,OAAA,YAAA,CAAA;AAAA,KACT;AAAA,IACA,MAAM,GAAkC,EAAA;AACtC,MAAI,IAAA;AACF,QAAA,MAAM,SAAS,OAAO,GAAA,KAAQ,WAAW,IAAI,GAAA,CAAI,GAAG,CAAI,GAAA,GAAA,CAAA;AACxD,QAAA,OAAO,aAAa,IAAK,CAAA,CAAA,CAAA,KAAK,QAAQ,CAAC,CAAA,KAAM,OAAO,IAAI,CAAA,CAAA;AAAA,OAClD,CAAA,MAAA;AACN,QAAO,OAAA,KAAA,CAAA,CAAA;AAAA,OACT;AAAA,KACF;AAAA,IACA,OAAO,IAA6B,EAAA;AAClC,MAAA,OAAO,aAAa,IAAK,CAAA,CAAA,CAAA,KAAK,OAAQ,CAAA,CAAC,MAAM,IAAI,CAAA,CAAA;AAAA,KACnD;AAAA,GACF,CAAA;AACF,CAAA;AAQO,SAAS,qBAAqB,OAI1B,EAAA;AACT,EAAA,MAAM,EAAE,GAAA,EAAK,IAAM,EAAA,UAAA,EAAe,GAAA,OAAA,CAAA;AAGlC,EAAI,IAAA;AAEF,IAAA,IAAI,IAAI,GAAG,CAAA,CAAA;AACX,IAAO,OAAA,GAAA,CAAA;AAAA,GACD,CAAA,MAAA;AAAA,GAER;AAEA,EAAI,IAAA,OAAA,CAAA;AAEJ,EAAI,IAAA,GAAA,CAAI,UAAW,CAAA,GAAG,CAAG,EAAA;AAEvB,IAAA,MAAM,EAAE,IAAA,EAAM,QAAS,EAAA,GAAIA,6BAAY,IAAI,CAAA,CAAA;AAE3C,IAAU,OAAA,GAAA,IAAI,IAAI,IAAI,CAAA,CAAA;AAEtB,IAAA,MAAM,YAAe,GAAAC,cAAA;AAAA,MACnB,OAAA,CAAQ,SAAS,SAAU,CAAA,CAAA,EAAG,QAAQ,QAAS,CAAA,MAAA,GAAS,SAAS,MAAM,CAAA;AAAA,MACvE,GAAA;AAAA,KACF,CAAA;AACA,IAAA,OAAA,CAAQ,QAAW,GAAA,CAAA,EAAG,YAAY,CAAA,EAAG,GAAG,CAAA,CAAA,CAAA;AAAA,GACnC,MAAA;AAIL,IAAU,OAAA,GAAA,IAAI,GAAI,CAAA,GAAA,EAAK,IAAI,CAAA,CAAA;AAAA,GAC7B;AAEA,EAAA,OAAA,CAAQ,MAAS,GAAA,IAAI,GAAI,CAAA,IAAI,CAAE,CAAA,MAAA,CAAA;AAC/B,EAAA,IAAI,UAAY,EAAA;AACd,IAAQ,OAAA,CAAA,IAAA,GAAO,IAAI,UAAU,CAAA,CAAA,CAAA;AAAA,GAC/B;AACA,EAAA,OAAO,QAAQ,QAAS,EAAA,CAAA;AAC1B;;;;;;;"}
|