@backstage/plugin-scaffolder-backend-module-gitlab 0.10.1-next.0 → 0.11.0-next.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 CHANGED
@@ -1,5 +1,20 @@
1
1
  # @backstage/plugin-scaffolder-backend-module-gitlab
2
2
 
3
+ ## 0.11.0-next.1
4
+
5
+ ### Minor Changes
6
+
7
+ - f2d034b: In the `gitlabRepoPush` action, add 'auto' possibility for `commitAction` input.
8
+
9
+ ### Patch Changes
10
+
11
+ - Updated dependencies
12
+ - @backstage/integration@1.18.3-next.1
13
+ - @backstage/backend-plugin-api@1.6.0-next.1
14
+ - @backstage/config@1.3.6
15
+ - @backstage/errors@1.2.7
16
+ - @backstage/plugin-scaffolder-node@0.12.2-next.1
17
+
3
18
  ## 0.10.1-next.0
4
19
 
5
20
  ### Patch Changes
@@ -6,41 +6,12 @@ var errors = require('@backstage/errors');
6
6
  var backendPluginApi = require('@backstage/backend-plugin-api');
7
7
  var helpers = require('./helpers.cjs.js');
8
8
  var gitlabMergeRequest_examples = require('./gitlabMergeRequest.examples.cjs.js');
9
- var crypto = require('crypto');
9
+ var util = require('../util.cjs.js');
10
10
 
11
11
  function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
12
12
 
13
13
  var path__default = /*#__PURE__*/_interopDefaultCompat(path);
14
14
 
15
- function computeSha256(file) {
16
- const hash = crypto.createHash("sha256");
17
- hash.update(file.content);
18
- return hash.digest("hex");
19
- }
20
- async function getFileAction(fileInfo, target, api, logger, remoteFiles, defaultCommitAction = "auto") {
21
- if (defaultCommitAction === "auto") {
22
- const filePath = path__default.default.join(fileInfo.targetPath ?? "", fileInfo.file.path);
23
- if (remoteFiles?.some((remoteFile) => remoteFile.path === filePath)) {
24
- try {
25
- const targetFile = await api.RepositoryFiles.show(
26
- target.repoID,
27
- filePath,
28
- target.branch
29
- );
30
- if (computeSha256(fileInfo.file) === targetFile.content_sha256) {
31
- return "skip";
32
- }
33
- } catch (error) {
34
- logger.warn(
35
- `Unable to retrieve detailed information for remote file ${filePath}`
36
- );
37
- }
38
- return "update";
39
- }
40
- return "create";
41
- }
42
- return defaultCommitAction;
43
- }
44
15
  async function getReviewersFromApprovalRules(api, mergerequestIId, repoID, ctx) {
45
16
  try {
46
17
  let mergeRequest = await api.MergeRequests.show(
@@ -206,7 +177,7 @@ _deprecated_: \`projectid\` passed as query parameters in the \`repoUrl\``
206
177
  }
207
178
  const actions = ctx.input.commitAction === "skip" ? [] : (await Promise.all(
208
179
  fileContents.map(async (file) => {
209
- const action = await getFileAction(
180
+ const action = await util.getFileAction(
210
181
  { file, targetPath },
211
182
  { repoID, branch: targetBranch },
212
183
  api,
@@ -1 +1 @@
1
- {"version":3,"file":"gitlabMergeRequest.cjs.js","sources":["../../src/actions/gitlabMergeRequest.ts"],"sourcesContent":["/*\n * Copyright 2021 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 {\n createTemplateAction,\n parseRepoUrl,\n SerializedFile,\n serializeDirectoryContents,\n} from '@backstage/plugin-scaffolder-node';\nimport {\n Camelize,\n CommitAction,\n ExpandedMergeRequestSchema,\n Gitlab,\n RepositoryTreeSchema,\n SimpleUserSchema,\n} from '@gitbeaker/rest';\nimport path from 'path';\nimport { ScmIntegrationRegistry } from '@backstage/integration';\nimport { InputError } from '@backstage/errors';\nimport {\n LoggerService,\n resolveSafeChildPath,\n} from '@backstage/backend-plugin-api';\nimport { createGitlabApi, getErrorMessage } from './helpers';\nimport { examples } from './gitlabMergeRequest.examples';\nimport { createHash } from 'crypto';\n\nfunction computeSha256(file: SerializedFile): string {\n const hash = createHash('sha256');\n hash.update(file.content);\n return hash.digest('hex');\n}\n\nasync function getFileAction(\n fileInfo: { file: SerializedFile; targetPath?: string },\n target: { repoID: string; branch: string },\n api: InstanceType<typeof Gitlab>,\n logger: LoggerService,\n remoteFiles: RepositoryTreeSchema[],\n defaultCommitAction:\n | 'create'\n | 'delete'\n | 'update'\n | 'skip'\n | 'auto' = 'auto',\n): Promise<'create' | 'delete' | 'update' | 'skip'> {\n if (defaultCommitAction === 'auto') {\n const filePath = path.join(fileInfo.targetPath ?? '', fileInfo.file.path);\n\n if (remoteFiles?.some(remoteFile => remoteFile.path === filePath)) {\n try {\n const targetFile = await api.RepositoryFiles.show(\n target.repoID,\n filePath,\n target.branch,\n );\n if (computeSha256(fileInfo.file) === targetFile.content_sha256) {\n return 'skip';\n }\n } catch (error) {\n logger.warn(\n `Unable to retrieve detailed information for remote file ${filePath}`,\n );\n }\n return 'update';\n }\n return 'create';\n }\n return defaultCommitAction;\n}\n\nasync function getReviewersFromApprovalRules(\n api: InstanceType<typeof Gitlab>,\n mergerequestIId: number,\n repoID: string,\n ctx: any,\n): Promise<number[]> {\n try {\n // Because we don't know the code owners before the MR is created, we can't check the approval rules beforehand.\n // Getting the approval rules beforehand is very difficult, especially, because of the inheritance rules for groups.\n // Code owners take a moment to be processed and added to the approval rules after the MR is created.\n\n let mergeRequest:\n | ExpandedMergeRequestSchema\n | Camelize<ExpandedMergeRequestSchema> = await api.MergeRequests.show(\n repoID,\n mergerequestIId,\n );\n\n while (\n mergeRequest.detailed_merge_status === 'preparing' ||\n mergeRequest.detailed_merge_status === 'approvals_syncing' ||\n mergeRequest.detailed_merge_status === 'checking'\n ) {\n mergeRequest = await api.MergeRequests.show(repoID, mergeRequest.iid);\n ctx.logger.info(`${mergeRequest.detailed_merge_status}`);\n }\n\n const approvalRules = await api.MergeRequestApprovals.allApprovalRules(\n repoID,\n {\n mergerequestIId: mergeRequest.iid,\n },\n );\n\n return approvalRules\n .filter(rule => rule.eligible_approvers !== undefined)\n .map(rule => {\n return rule.eligible_approvers as SimpleUserSchema[];\n })\n .flat()\n .map(user => user.id);\n } catch (e) {\n ctx.logger.warn(\n `Failed to retrieve approval rules for MR ${mergerequestIId}: ${getErrorMessage(\n e,\n )}. Proceeding with MR creation without reviewers from approval rules.`,\n );\n return [];\n }\n}\n\nconst commitActions = ['create', 'delete', 'update', 'skip', 'auto'] as const;\n\n/**\n * Create a new action that creates a GitLab merge request.\n *\n * @public\n */\nexport const createPublishGitlabMergeRequestAction = (options: {\n integrations: ScmIntegrationRegistry;\n}) => {\n const { integrations } = options;\n\n return createTemplateAction({\n id: 'publish:gitlab:merge-request',\n examples,\n schema: {\n input: {\n repoUrl: z =>\n z.string().describe(`\\\nAccepts the format \\`gitlab.com?repo=project_name&owner=group_name\\` where \\\n\\`project_name\\` is the repository name and \\`group_name\\` is a group or username`),\n title: z => z.string().describe('The name for the merge request'),\n description: z =>\n z\n .string()\n .optional()\n .describe('The description of the merge request'),\n branchName: z =>\n z.string().describe('The source branch name of the merge request'),\n targetBranchName: z =>\n z\n .string()\n .optional()\n .describe(\n 'The target branch name of the merge request (defaults to _default branch of repository_)',\n ),\n sourcePath: z =>\n z.string().optional().describe(`\\\nSubdirectory of working directory to copy changes from. \\\nFor reasons of backward compatibility, any specified \\`targetPath\\` input will \\\nbe applied in place of an absent/falsy value for this input. \\\nCircumvent this behavior using \\`.\\``),\n targetPath: z =>\n z\n .string()\n .optional()\n .describe('Subdirectory of repository to apply changes to'),\n token: z =>\n z\n .string()\n .optional()\n .describe('The token to use for authorization to GitLab'),\n commitAction: z =>\n z.enum(commitActions).optional().describe(`\\\nThe action to be used for \\`git\\` commit. Defaults to the custom \\`auto\\` action provided by Backstage,\nwhich uses additional API calls in order to detect whether to \\`create\\`, \\`update\\` or \\`skip\\` each source file.`),\n /** @deprecated projectID passed as query parameters in the repoUrl */\n projectid: z =>\n z\n .string()\n .optional()\n .describe(\n `\\\nProject ID/Name(slug) of the GitLab Project\n_deprecated_: \\`projectid\\` passed as query parameters in the \\`repoUrl\\``,\n ),\n removeSourceBranch: z =>\n z\n .boolean()\n .optional()\n .describe(\n 'Option to delete source branch once the MR has been merged. Default: `false`',\n ),\n assignee: z =>\n z\n .string()\n .optional()\n .describe('User this merge request will be assigned to'),\n reviewers: z =>\n z\n .string()\n .array()\n .optional()\n .describe('Users that will be assigned as reviewers'),\n assignReviewersFromApprovalRules: z =>\n z\n .boolean()\n .optional()\n .describe(\n 'Automatically assign reviewers from the approval rules of the MR. Includes `CODEOWNERS`',\n ),\n labels: z =>\n z\n .string()\n .or(z.string().array())\n .optional()\n .describe('Labels with which to tag the created merge request'),\n },\n output: {\n targetBranchName: z =>\n z.string().describe('Target branch name of the merge request'),\n projectid: z => z.string().describe('GitLab Project id/Name(slug)'),\n projectPath: z => z.string().describe('GitLab Project path'),\n mergeRequestUrl: z =>\n z.string().describe('Link to the merge request in GitLab'),\n },\n },\n async handler(ctx) {\n const {\n assignee,\n reviewers,\n branchName,\n targetBranchName,\n description,\n repoUrl,\n removeSourceBranch,\n targetPath,\n sourcePath,\n title,\n token,\n labels,\n } = ctx.input;\n\n const { owner, repo, project } = parseRepoUrl(repoUrl, integrations);\n const repoID = project ? project : `${owner}/${repo}`;\n\n const api = createGitlabApi({\n integrations,\n token,\n repoUrl,\n });\n\n let assigneeId: number | undefined = undefined;\n\n if (assignee !== undefined) {\n try {\n const assigneeUser = await api.Users.all({ username: assignee });\n assigneeId = assigneeUser[0].id;\n } catch (e) {\n ctx.logger.warn(\n `Failed to find gitlab user id for ${assignee}: ${getErrorMessage(\n e,\n )}. Proceeding with MR creation without an assignee.`,\n );\n }\n }\n\n let reviewerIds: number[] | undefined = undefined; // Explicitly set to undefined. Strangely, passing an empty array to the API will result the other options being undefined also being explicitly passed to the Gitlab API call (e.g. assigneeId)\n if (reviewers !== undefined) {\n reviewerIds = (\n await Promise.all(\n reviewers.map(async reviewer => {\n try {\n const reviewerUser = await api.Users.all({\n username: reviewer,\n });\n return reviewerUser[0].id;\n } catch (e) {\n ctx.logger.warn(\n `Failed to find gitlab user id for ${reviewer}: ${e}. Proceeding with MR creation without reviewer.`,\n );\n return undefined;\n }\n }),\n )\n ).filter(id => id !== undefined) as number[];\n }\n\n let fileRoot: string;\n if (sourcePath) {\n fileRoot = resolveSafeChildPath(ctx.workspacePath, sourcePath);\n } else if (targetPath) {\n // for backward compatibility\n fileRoot = resolveSafeChildPath(ctx.workspacePath, targetPath);\n } else {\n fileRoot = ctx.workspacePath;\n }\n\n const fileContents = await serializeDirectoryContents(fileRoot, {\n gitignore: true,\n });\n\n let targetBranch = targetBranchName;\n if (!targetBranch) {\n const projects = await api.Projects.show(repoID);\n const defaultBranch = projects.default_branch ?? projects.defaultBranch;\n if (typeof defaultBranch !== 'string' || !defaultBranch) {\n throw new InputError(\n `The branch creation failed. Target branch was not provided, and could not find default branch from project settings. Project: ${JSON.stringify(\n project,\n )}`,\n );\n }\n targetBranch = defaultBranch;\n }\n\n let remoteFiles: RepositoryTreeSchema[] = [];\n if ((ctx.input.commitAction ?? 'auto') === 'auto') {\n try {\n remoteFiles = await api.Repositories.allRepositoryTrees(repoID, {\n ref: targetBranch,\n recursive: true,\n path: targetPath ?? undefined,\n });\n } catch (e) {\n ctx.logger.warn(\n `Could not retrieve the list of files for ${repoID} (branch: ${targetBranch}) : ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n const actions: CommitAction[] =\n ctx.input.commitAction === 'skip'\n ? []\n : (\n (\n await Promise.all(\n fileContents.map(async file => {\n const action = await getFileAction(\n { file, targetPath },\n { repoID, branch: targetBranch! },\n api,\n ctx.logger,\n remoteFiles,\n ctx.input.commitAction,\n );\n return { file, action };\n }),\n )\n ).filter(o => o.action !== 'skip') as {\n file: SerializedFile;\n action: CommitAction['action'];\n }[]\n ).map(({ file, action }) => ({\n action,\n filePath: targetPath\n ? path.posix.join(targetPath, file.path)\n : file.path,\n encoding: 'base64',\n content: file.content.toString('base64'),\n execute_filemode: file.executable,\n }));\n\n let createBranch = actions.length > 0;\n\n try {\n const branch = await api.Branches.show(repoID, branchName);\n if (createBranch) {\n const mergeRequests = await api.MergeRequests.all({\n projectId: repoID,\n source_branch: branchName,\n });\n\n if (mergeRequests.length > 0) {\n // If an open MR exists, include the MR link in the error message\n throw new InputError(\n `The branch creation failed because the branch already exists at: ${branch.web_url}. Additionally, there is a Merge Request for this branch: ${mergeRequests[0].web_url}`,\n );\n } else {\n // If no open MR, just notify about the existing branch\n throw new InputError(\n `The branch creation failed because the branch already exists at: ${branch.web_url}.`,\n );\n }\n }\n\n ctx.logger.info(\n `Using existing branch ${branchName} without modification.`,\n );\n } catch (e) {\n if (e instanceof InputError) {\n throw e;\n }\n createBranch = true;\n }\n\n if (createBranch) {\n try {\n await api.Branches.create(repoID, branchName, String(targetBranch));\n } catch (e) {\n throw new InputError(\n `The branch creation failed. Please check that your repo does not already contain a branch named '${branchName}'. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n\n await ctx.checkpoint({\n key: `commit.to.${repoID}.${branchName}`,\n fn: async () => {\n if (actions.length) {\n try {\n const commit = await api.Commits.create(\n repoID,\n branchName,\n title,\n actions,\n );\n return commit.id;\n } catch (e) {\n throw new InputError(\n `Committing the changes to ${branchName} failed. Please check that none of the files created by the template already exists. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n return null;\n },\n });\n\n const { mrId, mrWebUrl } = await ctx.checkpoint({\n key: `create.mr.${repoID}.${branchName}`,\n fn: async () => {\n try {\n const mergeRequest = await api.MergeRequests.create(\n repoID,\n branchName,\n String(targetBranch),\n title,\n {\n description,\n removeSourceBranch: removeSourceBranch\n ? removeSourceBranch\n : false,\n assigneeId,\n reviewerIds,\n labels,\n },\n );\n return {\n mrId: mergeRequest.iid,\n mrWebUrl: mergeRequest.web_url ?? mergeRequest.webUrl,\n };\n } catch (e) {\n throw new InputError(\n `Merge request creation failed. ${getErrorMessage(e)}`,\n );\n }\n },\n });\n\n await ctx.checkpoint({\n key: `create.mr.assign.reviewers.${repoID}.${branchName}`,\n fn: async () => {\n if (ctx.input.assignReviewersFromApprovalRules) {\n try {\n const reviewersFromApprovalRules =\n await getReviewersFromApprovalRules(api, mrId, repoID, ctx);\n if (reviewersFromApprovalRules.length > 0) {\n const eligibleUserIds = new Set([\n ...reviewersFromApprovalRules,\n ...(reviewerIds ?? []),\n ]);\n\n const mergeRequest = await api.MergeRequests.edit(\n repoID,\n mrId,\n {\n reviewerIds: Array.from(eligibleUserIds),\n },\n );\n return {\n mrWebUrl: mergeRequest.web_url ?? mergeRequest.webUrl,\n };\n }\n } catch (e) {\n ctx.logger.warn(\n `Failed to assign reviewers from approval rules: ${getErrorMessage(\n e,\n )}.`,\n );\n }\n }\n return { mrWebUrl };\n },\n });\n\n ctx.output('projectid', repoID);\n ctx.output('targetBranchName', targetBranch);\n ctx.output('projectPath', repoID);\n ctx.output('mergeRequestUrl', mrWebUrl as string);\n },\n });\n};\n"],"names":["createHash","path","getErrorMessage","createTemplateAction","examples","parseRepoUrl","createGitlabApi","resolveSafeChildPath","serializeDirectoryContents","InputError"],"mappings":"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"}
1
+ {"version":3,"file":"gitlabMergeRequest.cjs.js","sources":["../../src/actions/gitlabMergeRequest.ts"],"sourcesContent":["/*\n * Copyright 2021 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 {\n createTemplateAction,\n parseRepoUrl,\n SerializedFile,\n serializeDirectoryContents,\n} from '@backstage/plugin-scaffolder-node';\nimport {\n Camelize,\n CommitAction,\n ExpandedMergeRequestSchema,\n Gitlab,\n RepositoryTreeSchema,\n SimpleUserSchema,\n} from '@gitbeaker/rest';\nimport path from 'path';\nimport { ScmIntegrationRegistry } from '@backstage/integration';\nimport { InputError } from '@backstage/errors';\nimport { resolveSafeChildPath } from '@backstage/backend-plugin-api';\nimport { createGitlabApi, getErrorMessage } from './helpers';\nimport { examples } from './gitlabMergeRequest.examples';\n\nimport { getFileAction } from '../util';\n\nasync function getReviewersFromApprovalRules(\n api: InstanceType<typeof Gitlab>,\n mergerequestIId: number,\n repoID: string,\n ctx: any,\n): Promise<number[]> {\n try {\n // Because we don't know the code owners before the MR is created, we can't check the approval rules beforehand.\n // Getting the approval rules beforehand is very difficult, especially, because of the inheritance rules for groups.\n // Code owners take a moment to be processed and added to the approval rules after the MR is created.\n\n let mergeRequest:\n | ExpandedMergeRequestSchema\n | Camelize<ExpandedMergeRequestSchema> = await api.MergeRequests.show(\n repoID,\n mergerequestIId,\n );\n\n while (\n mergeRequest.detailed_merge_status === 'preparing' ||\n mergeRequest.detailed_merge_status === 'approvals_syncing' ||\n mergeRequest.detailed_merge_status === 'checking'\n ) {\n mergeRequest = await api.MergeRequests.show(repoID, mergeRequest.iid);\n ctx.logger.info(`${mergeRequest.detailed_merge_status}`);\n }\n\n const approvalRules = await api.MergeRequestApprovals.allApprovalRules(\n repoID,\n {\n mergerequestIId: mergeRequest.iid,\n },\n );\n\n return approvalRules\n .filter(rule => rule.eligible_approvers !== undefined)\n .map(rule => {\n return rule.eligible_approvers as SimpleUserSchema[];\n })\n .flat()\n .map(user => user.id);\n } catch (e) {\n ctx.logger.warn(\n `Failed to retrieve approval rules for MR ${mergerequestIId}: ${getErrorMessage(\n e,\n )}. Proceeding with MR creation without reviewers from approval rules.`,\n );\n return [];\n }\n}\n\nconst commitActions = ['create', 'delete', 'update', 'skip', 'auto'] as const;\n\n/**\n * Create a new action that creates a GitLab merge request.\n *\n * @public\n */\nexport const createPublishGitlabMergeRequestAction = (options: {\n integrations: ScmIntegrationRegistry;\n}) => {\n const { integrations } = options;\n\n return createTemplateAction({\n id: 'publish:gitlab:merge-request',\n examples,\n schema: {\n input: {\n repoUrl: z =>\n z.string().describe(`\\\nAccepts the format \\`gitlab.com?repo=project_name&owner=group_name\\` where \\\n\\`project_name\\` is the repository name and \\`group_name\\` is a group or username`),\n title: z => z.string().describe('The name for the merge request'),\n description: z =>\n z\n .string()\n .optional()\n .describe('The description of the merge request'),\n branchName: z =>\n z.string().describe('The source branch name of the merge request'),\n targetBranchName: z =>\n z\n .string()\n .optional()\n .describe(\n 'The target branch name of the merge request (defaults to _default branch of repository_)',\n ),\n sourcePath: z =>\n z.string().optional().describe(`\\\nSubdirectory of working directory to copy changes from. \\\nFor reasons of backward compatibility, any specified \\`targetPath\\` input will \\\nbe applied in place of an absent/falsy value for this input. \\\nCircumvent this behavior using \\`.\\``),\n targetPath: z =>\n z\n .string()\n .optional()\n .describe('Subdirectory of repository to apply changes to'),\n token: z =>\n z\n .string()\n .optional()\n .describe('The token to use for authorization to GitLab'),\n commitAction: z =>\n z.enum(commitActions).optional().describe(`\\\nThe action to be used for \\`git\\` commit. Defaults to the custom \\`auto\\` action provided by Backstage,\nwhich uses additional API calls in order to detect whether to \\`create\\`, \\`update\\` or \\`skip\\` each source file.`),\n /** @deprecated projectID passed as query parameters in the repoUrl */\n projectid: z =>\n z\n .string()\n .optional()\n .describe(\n `\\\nProject ID/Name(slug) of the GitLab Project\n_deprecated_: \\`projectid\\` passed as query parameters in the \\`repoUrl\\``,\n ),\n removeSourceBranch: z =>\n z\n .boolean()\n .optional()\n .describe(\n 'Option to delete source branch once the MR has been merged. Default: `false`',\n ),\n assignee: z =>\n z\n .string()\n .optional()\n .describe('User this merge request will be assigned to'),\n reviewers: z =>\n z\n .string()\n .array()\n .optional()\n .describe('Users that will be assigned as reviewers'),\n assignReviewersFromApprovalRules: z =>\n z\n .boolean()\n .optional()\n .describe(\n 'Automatically assign reviewers from the approval rules of the MR. Includes `CODEOWNERS`',\n ),\n labels: z =>\n z\n .string()\n .or(z.string().array())\n .optional()\n .describe('Labels with which to tag the created merge request'),\n },\n output: {\n targetBranchName: z =>\n z.string().describe('Target branch name of the merge request'),\n projectid: z => z.string().describe('GitLab Project id/Name(slug)'),\n projectPath: z => z.string().describe('GitLab Project path'),\n mergeRequestUrl: z =>\n z.string().describe('Link to the merge request in GitLab'),\n },\n },\n async handler(ctx) {\n const {\n assignee,\n reviewers,\n branchName,\n targetBranchName,\n description,\n repoUrl,\n removeSourceBranch,\n targetPath,\n sourcePath,\n title,\n token,\n labels,\n } = ctx.input;\n\n const { owner, repo, project } = parseRepoUrl(repoUrl, integrations);\n const repoID = project ? project : `${owner}/${repo}`;\n\n const api = createGitlabApi({\n integrations,\n token,\n repoUrl,\n });\n\n let assigneeId: number | undefined = undefined;\n\n if (assignee !== undefined) {\n try {\n const assigneeUser = await api.Users.all({ username: assignee });\n assigneeId = assigneeUser[0].id;\n } catch (e) {\n ctx.logger.warn(\n `Failed to find gitlab user id for ${assignee}: ${getErrorMessage(\n e,\n )}. Proceeding with MR creation without an assignee.`,\n );\n }\n }\n\n let reviewerIds: number[] | undefined = undefined; // Explicitly set to undefined. Strangely, passing an empty array to the API will result the other options being undefined also being explicitly passed to the Gitlab API call (e.g. assigneeId)\n if (reviewers !== undefined) {\n reviewerIds = (\n await Promise.all(\n reviewers.map(async reviewer => {\n try {\n const reviewerUser = await api.Users.all({\n username: reviewer,\n });\n return reviewerUser[0].id;\n } catch (e) {\n ctx.logger.warn(\n `Failed to find gitlab user id for ${reviewer}: ${e}. Proceeding with MR creation without reviewer.`,\n );\n return undefined;\n }\n }),\n )\n ).filter(id => id !== undefined) as number[];\n }\n\n let fileRoot: string;\n if (sourcePath) {\n fileRoot = resolveSafeChildPath(ctx.workspacePath, sourcePath);\n } else if (targetPath) {\n // for backward compatibility\n fileRoot = resolveSafeChildPath(ctx.workspacePath, targetPath);\n } else {\n fileRoot = ctx.workspacePath;\n }\n\n const fileContents = await serializeDirectoryContents(fileRoot, {\n gitignore: true,\n });\n\n let targetBranch = targetBranchName;\n if (!targetBranch) {\n const projects = await api.Projects.show(repoID);\n const defaultBranch = projects.default_branch ?? projects.defaultBranch;\n if (typeof defaultBranch !== 'string' || !defaultBranch) {\n throw new InputError(\n `The branch creation failed. Target branch was not provided, and could not find default branch from project settings. Project: ${JSON.stringify(\n project,\n )}`,\n );\n }\n targetBranch = defaultBranch;\n }\n\n let remoteFiles: RepositoryTreeSchema[] = [];\n if ((ctx.input.commitAction ?? 'auto') === 'auto') {\n try {\n remoteFiles = await api.Repositories.allRepositoryTrees(repoID, {\n ref: targetBranch,\n recursive: true,\n path: targetPath ?? undefined,\n });\n } catch (e) {\n ctx.logger.warn(\n `Could not retrieve the list of files for ${repoID} (branch: ${targetBranch}) : ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n const actions: CommitAction[] =\n ctx.input.commitAction === 'skip'\n ? []\n : (\n (\n await Promise.all(\n fileContents.map(async file => {\n const action = await getFileAction(\n { file, targetPath },\n { repoID, branch: targetBranch! },\n api,\n ctx.logger,\n remoteFiles,\n ctx.input.commitAction,\n );\n return { file, action };\n }),\n )\n ).filter(o => o.action !== 'skip') as {\n file: SerializedFile;\n action: CommitAction['action'];\n }[]\n ).map(({ file, action }) => ({\n action,\n filePath: targetPath\n ? path.posix.join(targetPath, file.path)\n : file.path,\n encoding: 'base64',\n content: file.content.toString('base64'),\n execute_filemode: file.executable,\n }));\n\n let createBranch = actions.length > 0;\n\n try {\n const branch = await api.Branches.show(repoID, branchName);\n if (createBranch) {\n const mergeRequests = await api.MergeRequests.all({\n projectId: repoID,\n source_branch: branchName,\n });\n\n if (mergeRequests.length > 0) {\n // If an open MR exists, include the MR link in the error message\n throw new InputError(\n `The branch creation failed because the branch already exists at: ${branch.web_url}. Additionally, there is a Merge Request for this branch: ${mergeRequests[0].web_url}`,\n );\n } else {\n // If no open MR, just notify about the existing branch\n throw new InputError(\n `The branch creation failed because the branch already exists at: ${branch.web_url}.`,\n );\n }\n }\n\n ctx.logger.info(\n `Using existing branch ${branchName} without modification.`,\n );\n } catch (e) {\n if (e instanceof InputError) {\n throw e;\n }\n createBranch = true;\n }\n\n if (createBranch) {\n try {\n await api.Branches.create(repoID, branchName, String(targetBranch));\n } catch (e) {\n throw new InputError(\n `The branch creation failed. Please check that your repo does not already contain a branch named '${branchName}'. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n\n await ctx.checkpoint({\n key: `commit.to.${repoID}.${branchName}`,\n fn: async () => {\n if (actions.length) {\n try {\n const commit = await api.Commits.create(\n repoID,\n branchName,\n title,\n actions,\n );\n return commit.id;\n } catch (e) {\n throw new InputError(\n `Committing the changes to ${branchName} failed. Please check that none of the files created by the template already exists. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n return null;\n },\n });\n\n const { mrId, mrWebUrl } = await ctx.checkpoint({\n key: `create.mr.${repoID}.${branchName}`,\n fn: async () => {\n try {\n const mergeRequest = await api.MergeRequests.create(\n repoID,\n branchName,\n String(targetBranch),\n title,\n {\n description,\n removeSourceBranch: removeSourceBranch\n ? removeSourceBranch\n : false,\n assigneeId,\n reviewerIds,\n labels,\n },\n );\n return {\n mrId: mergeRequest.iid,\n mrWebUrl: mergeRequest.web_url ?? mergeRequest.webUrl,\n };\n } catch (e) {\n throw new InputError(\n `Merge request creation failed. ${getErrorMessage(e)}`,\n );\n }\n },\n });\n\n await ctx.checkpoint({\n key: `create.mr.assign.reviewers.${repoID}.${branchName}`,\n fn: async () => {\n if (ctx.input.assignReviewersFromApprovalRules) {\n try {\n const reviewersFromApprovalRules =\n await getReviewersFromApprovalRules(api, mrId, repoID, ctx);\n if (reviewersFromApprovalRules.length > 0) {\n const eligibleUserIds = new Set([\n ...reviewersFromApprovalRules,\n ...(reviewerIds ?? []),\n ]);\n\n const mergeRequest = await api.MergeRequests.edit(\n repoID,\n mrId,\n {\n reviewerIds: Array.from(eligibleUserIds),\n },\n );\n return {\n mrWebUrl: mergeRequest.web_url ?? mergeRequest.webUrl,\n };\n }\n } catch (e) {\n ctx.logger.warn(\n `Failed to assign reviewers from approval rules: ${getErrorMessage(\n e,\n )}.`,\n );\n }\n }\n return { mrWebUrl };\n },\n });\n\n ctx.output('projectid', repoID);\n ctx.output('targetBranchName', targetBranch);\n ctx.output('projectPath', repoID);\n ctx.output('mergeRequestUrl', mrWebUrl as string);\n },\n });\n};\n"],"names":["getErrorMessage","createTemplateAction","examples","parseRepoUrl","createGitlabApi","resolveSafeChildPath","serializeDirectoryContents","InputError","getFileAction","path"],"mappings":"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"}
@@ -6,6 +6,7 @@ var backendPluginApi = require('@backstage/backend-plugin-api');
6
6
  var pluginScaffolderNode = require('@backstage/plugin-scaffolder-node');
7
7
  var helpers = require('./helpers.cjs.js');
8
8
  var gitlabRepoPush_examples = require('./gitlabRepoPush.examples.cjs.js');
9
+ var util = require('../util.cjs.js');
9
10
 
10
11
  function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
11
12
 
@@ -36,7 +37,7 @@ const createGitlabRepoPushAction = (options) => {
36
37
  token: (z) => z.string({
37
38
  description: "The token to use for authorization to GitLab"
38
39
  }).optional(),
39
- commitAction: (z) => z.enum(["create", "update", "delete"], {
40
+ commitAction: (z) => z.enum(["create", "update", "delete", "auto"], {
40
41
  description: "The action to be used for git commit. Defaults to create, but can be set to update or delete"
41
42
  }).optional()
42
43
  },
@@ -77,8 +78,36 @@ const createGitlabRepoPushAction = (options) => {
77
78
  const fileContents = await pluginScaffolderNode.serializeDirectoryContents(fileRoot, {
78
79
  gitignore: true
79
80
  });
80
- const actions = fileContents.map((file) => ({
81
- action: commitAction ?? "create",
81
+ let remoteFiles = [];
82
+ if ((ctx.input.commitAction ?? "auto") === "auto") {
83
+ try {
84
+ remoteFiles = await api.Repositories.allRepositoryTrees(repoID, {
85
+ ref: branchName,
86
+ recursive: true,
87
+ path: targetPath ?? void 0
88
+ });
89
+ } catch (e) {
90
+ ctx.logger.warn(
91
+ `Could not retrieve the list of files for ${repoID} (branch: ${branchName}) : ${helpers.getErrorMessage(
92
+ e
93
+ )}`
94
+ );
95
+ }
96
+ }
97
+ const fileActionMap = [];
98
+ for (const file of fileContents) {
99
+ const action = await util.getFileAction(
100
+ { file, targetPath },
101
+ { repoID, branch: branchName },
102
+ api,
103
+ ctx.logger,
104
+ remoteFiles,
105
+ ctx.input.commitAction
106
+ );
107
+ fileActionMap.push({ file, action });
108
+ }
109
+ const actions = fileActionMap.filter((o) => o.action !== "skip").map(({ file, action }) => ({
110
+ action,
82
111
  filePath: targetPath ? path__default.default.posix.join(targetPath, file.path) : file.path,
83
112
  encoding: "base64",
84
113
  content: file.content.toString("base64"),
@@ -1 +1 @@
1
- {"version":3,"file":"gitlabRepoPush.cjs.js","sources":["../../src/actions/gitlabRepoPush.ts"],"sourcesContent":["/*\n * Copyright 2023 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 path from 'path';\nimport { ScmIntegrationRegistry } from '@backstage/integration';\nimport { InputError } from '@backstage/errors';\nimport { resolveSafeChildPath } from '@backstage/backend-plugin-api';\nimport {\n createTemplateAction,\n parseRepoUrl,\n serializeDirectoryContents,\n} from '@backstage/plugin-scaffolder-node';\nimport { CommitAction } from '@gitbeaker/rest';\nimport { createGitlabApi, getErrorMessage } from './helpers';\nimport { examples } from './gitlabRepoPush.examples';\n\n/**\n * Create a new action that commits into a gitlab repository.\n *\n * @public\n */\nexport const createGitlabRepoPushAction = (options: {\n integrations: ScmIntegrationRegistry;\n}) => {\n const { integrations } = options;\n\n return createTemplateAction({\n id: 'gitlab:repo:push',\n examples,\n schema: {\n input: {\n repoUrl: z =>\n z.string({\n description: `Accepts the format 'gitlab.com?repo=project_name&owner=group_name' where 'project_name' is the repository name and 'group_name' is a group or username`,\n }),\n branchName: z =>\n z.string({\n description: 'The branch name for the commit',\n }),\n commitMessage: z =>\n z.string({\n description: `The commit message`,\n }),\n sourcePath: z =>\n z\n .string({\n description:\n 'Subdirectory of working directory to copy changes from',\n })\n .optional(),\n targetPath: z =>\n z\n .string({\n description: 'Subdirectory of repository to apply changes to',\n })\n .optional(),\n token: z =>\n z\n .string({\n description: 'The token to use for authorization to GitLab',\n })\n .optional(),\n commitAction: z =>\n z\n .enum(['create', 'update', 'delete'], {\n description:\n 'The action to be used for git commit. Defaults to create, but can be set to update or delete',\n })\n .optional(),\n },\n output: {\n projectid: z =>\n z.string({\n description: 'Gitlab Project id/Name(slug)',\n }),\n projectPath: z =>\n z.string({\n description: 'Gitlab Project path',\n }),\n commitHash: z =>\n z.string({\n description: 'The git commit hash of the commit',\n }),\n },\n },\n async handler(ctx) {\n const {\n branchName,\n repoUrl,\n targetPath,\n sourcePath,\n token,\n commitAction,\n } = ctx.input;\n\n const { owner, repo, project } = parseRepoUrl(repoUrl, integrations);\n const repoID = project ? project : `${owner}/${repo}`;\n\n const api = createGitlabApi({\n integrations,\n token,\n repoUrl,\n });\n\n let fileRoot: string;\n if (sourcePath) {\n fileRoot = resolveSafeChildPath(ctx.workspacePath, sourcePath);\n } else {\n fileRoot = ctx.workspacePath;\n }\n\n const fileContents = await serializeDirectoryContents(fileRoot, {\n gitignore: true,\n });\n\n const actions: CommitAction[] = fileContents.map(file => ({\n action: commitAction ?? 'create',\n filePath: targetPath\n ? path.posix.join(targetPath, file.path)\n : file.path,\n encoding: 'base64',\n content: file.content.toString('base64'),\n execute_filemode: file.executable,\n }));\n\n const branchExists = await ctx.checkpoint({\n key: `branch.exists.${repoID}.${branchName}`,\n fn: async () => {\n try {\n await api.Branches.show(repoID, branchName);\n return true;\n } catch (e: any) {\n if (e.cause?.response?.status !== 404) {\n throw new InputError(\n `Failed to check status of branch '${branchName}'. Please make sure that branch already exists or Backstage has permissions to create one. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n return false;\n },\n });\n\n if (!branchExists) {\n // create a branch using the default branch as ref\n try {\n const projects = await api.Projects.show(repoID);\n const { default_branch: defaultBranch } = projects;\n await api.Branches.create(repoID, branchName, String(defaultBranch));\n } catch (e) {\n throw new InputError(\n `The branch '${branchName}' was not found and creation failed with error. Please make sure that branch already exists or Backstage has permissions to create one. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n\n try {\n const commitId = await ctx.checkpoint({\n key: `commit.create.${repoID}.${branchName}`,\n fn: async () => {\n const commit = await api.Commits.create(\n repoID,\n branchName,\n ctx.input.commitMessage,\n actions,\n );\n return commit.id;\n },\n });\n\n ctx.output('projectid', repoID);\n ctx.output('projectPath', repoID);\n ctx.output('commitHash', commitId);\n } catch (e) {\n if (commitAction !== 'create') {\n throw new InputError(\n `Committing the changes to ${branchName} failed. Please verify that all files you're trying to modify exist in the repository. ${getErrorMessage(\n e,\n )}`,\n );\n }\n throw new InputError(\n `Committing the changes to ${branchName} failed. Please check that none of the files created by the template already exists. ${getErrorMessage(\n e,\n )}`,\n );\n }\n },\n });\n};\n"],"names":["createTemplateAction","examples","parseRepoUrl","createGitlabApi","resolveSafeChildPath","serializeDirectoryContents","path","InputError","getErrorMessage"],"mappings":"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"}
1
+ {"version":3,"file":"gitlabRepoPush.cjs.js","sources":["../../src/actions/gitlabRepoPush.ts"],"sourcesContent":["/*\n * Copyright 2023 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 path from 'path';\nimport { ScmIntegrationRegistry } from '@backstage/integration';\nimport { InputError } from '@backstage/errors';\nimport { resolveSafeChildPath } from '@backstage/backend-plugin-api';\nimport {\n createTemplateAction,\n parseRepoUrl,\n serializeDirectoryContents,\n} from '@backstage/plugin-scaffolder-node';\nimport { CommitAction } from '@gitbeaker/rest';\nimport { createGitlabApi, getErrorMessage } from './helpers';\nimport { examples } from './gitlabRepoPush.examples';\nimport { getFileAction } from '../util';\nimport { SerializedFile } from '@backstage/plugin-scaffolder-node';\nimport { RepositoryTreeSchema } from '@gitbeaker/rest';\n\n/**\n * Create a new action that commits into a gitlab repository.\n *\n * @public\n */\nexport const createGitlabRepoPushAction = (options: {\n integrations: ScmIntegrationRegistry;\n}) => {\n const { integrations } = options;\n\n return createTemplateAction({\n id: 'gitlab:repo:push',\n examples,\n schema: {\n input: {\n repoUrl: z =>\n z.string({\n description: `Accepts the format 'gitlab.com?repo=project_name&owner=group_name' where 'project_name' is the repository name and 'group_name' is a group or username`,\n }),\n branchName: z =>\n z.string({\n description: 'The branch name for the commit',\n }),\n commitMessage: z =>\n z.string({\n description: `The commit message`,\n }),\n sourcePath: z =>\n z\n .string({\n description:\n 'Subdirectory of working directory to copy changes from',\n })\n .optional(),\n targetPath: z =>\n z\n .string({\n description: 'Subdirectory of repository to apply changes to',\n })\n .optional(),\n token: z =>\n z\n .string({\n description: 'The token to use for authorization to GitLab',\n })\n .optional(),\n commitAction: z =>\n z\n .enum(['create', 'update', 'delete', 'auto'], {\n description:\n 'The action to be used for git commit. Defaults to create, but can be set to update or delete',\n })\n .optional(),\n },\n output: {\n projectid: z =>\n z.string({\n description: 'Gitlab Project id/Name(slug)',\n }),\n projectPath: z =>\n z.string({\n description: 'Gitlab Project path',\n }),\n commitHash: z =>\n z.string({\n description: 'The git commit hash of the commit',\n }),\n },\n },\n async handler(ctx) {\n const {\n branchName,\n repoUrl,\n targetPath,\n sourcePath,\n token,\n commitAction,\n } = ctx.input;\n\n const { owner, repo, project } = parseRepoUrl(repoUrl, integrations);\n const repoID = project ? project : `${owner}/${repo}`;\n\n const api = createGitlabApi({\n integrations,\n token,\n repoUrl,\n });\n\n let fileRoot: string;\n if (sourcePath) {\n fileRoot = resolveSafeChildPath(ctx.workspacePath, sourcePath);\n } else {\n fileRoot = ctx.workspacePath;\n }\n\n const fileContents = await serializeDirectoryContents(fileRoot, {\n gitignore: true,\n });\n\n let remoteFiles: RepositoryTreeSchema[] = [];\n if ((ctx.input.commitAction ?? 'auto') === 'auto') {\n try {\n remoteFiles = await api.Repositories.allRepositoryTrees(repoID, {\n ref: branchName,\n recursive: true,\n path: targetPath ?? undefined,\n });\n } catch (e) {\n ctx.logger.warn(\n `Could not retrieve the list of files for ${repoID} (branch: ${branchName}) : ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n\n const fileActionMap: {\n file: SerializedFile;\n action: 'create' | 'delete' | 'update' | 'skip';\n }[] = [];\n for (const file of fileContents) {\n const action = await getFileAction(\n { file, targetPath },\n { repoID, branch: branchName },\n api,\n ctx.logger,\n remoteFiles,\n ctx.input.commitAction,\n );\n fileActionMap.push({ file, action });\n }\n\n const actions: CommitAction[] = fileActionMap\n .filter(o => o.action !== 'skip')\n .map(({ file, action }) => ({\n action: action as CommitAction['action'],\n filePath: targetPath\n ? path.posix.join(targetPath, file.path)\n : file.path,\n encoding: 'base64',\n content: file.content.toString('base64'),\n execute_filemode: file.executable,\n }));\n\n const branchExists = await ctx.checkpoint({\n key: `branch.exists.${repoID}.${branchName}`,\n fn: async () => {\n try {\n await api.Branches.show(repoID, branchName);\n return true;\n } catch (e: any) {\n if (e.cause?.response?.status !== 404) {\n throw new InputError(\n `Failed to check status of branch '${branchName}'. Please make sure that branch already exists or Backstage has permissions to create one. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n return false;\n },\n });\n\n if (!branchExists) {\n // create a branch using the default branch as ref\n try {\n const projects = await api.Projects.show(repoID);\n const { default_branch: defaultBranch } = projects;\n await api.Branches.create(repoID, branchName, String(defaultBranch));\n } catch (e) {\n throw new InputError(\n `The branch '${branchName}' was not found and creation failed with error. Please make sure that branch already exists or Backstage has permissions to create one. ${getErrorMessage(\n e,\n )}`,\n );\n }\n }\n\n try {\n const commitId = await ctx.checkpoint({\n key: `commit.create.${repoID}.${branchName}`,\n fn: async () => {\n const commit = await api.Commits.create(\n repoID,\n branchName,\n ctx.input.commitMessage,\n actions,\n );\n return commit.id;\n },\n });\n\n ctx.output('projectid', repoID);\n ctx.output('projectPath', repoID);\n ctx.output('commitHash', commitId);\n } catch (e) {\n if (commitAction !== 'create') {\n throw new InputError(\n `Committing the changes to ${branchName} failed. Please verify that all files you're trying to modify exist in the repository. ${getErrorMessage(\n e,\n )}`,\n );\n }\n throw new InputError(\n `Committing the changes to ${branchName} failed. Please check that none of the files created by the template already exists. ${getErrorMessage(\n e,\n )}`,\n );\n }\n },\n });\n};\n"],"names":["createTemplateAction","examples","parseRepoUrl","createGitlabApi","resolveSafeChildPath","serializeDirectoryContents","getErrorMessage","getFileAction","path","InputError"],"mappings":"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"}
package/dist/index.d.ts CHANGED
@@ -275,7 +275,7 @@ declare const createGitlabRepoPushAction: (options: {
275
275
  sourcePath?: string | undefined;
276
276
  targetPath?: string | undefined;
277
277
  token?: string | undefined;
278
- commitAction?: "update" | "delete" | "create" | undefined;
278
+ commitAction?: "auto" | "update" | "delete" | "create" | undefined;
279
279
  }, {
280
280
  projectid: string;
281
281
  projectPath: string;
package/dist/util.cjs.js CHANGED
@@ -2,6 +2,12 @@
2
2
 
3
3
  var errors = require('@backstage/errors');
4
4
  var rest = require('@gitbeaker/rest');
5
+ var crypto = require('crypto');
6
+ var path = require('path');
7
+
8
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
9
+
10
+ var path__default = /*#__PURE__*/_interopDefaultCompat(path);
5
11
 
6
12
  const parseRepoHost = (repoUrl) => {
7
13
  let parsed;
@@ -115,10 +121,40 @@ async function checkEpicScope(client, projectId, epicId) {
115
121
  throw new errors.InputError(`Could not find epic scope: ${error.message}`);
116
122
  }
117
123
  }
124
+ function computeSha256(file) {
125
+ const hash = crypto.createHash("sha256");
126
+ hash.update(file.content);
127
+ return hash.digest("hex");
128
+ }
129
+ async function getFileAction(fileInfo, target, api, logger, remoteFiles, defaultCommitAction = "auto") {
130
+ if (defaultCommitAction === "auto") {
131
+ const filePath = path__default.default.join(fileInfo.targetPath ?? "", fileInfo.file.path);
132
+ if (remoteFiles?.some((remoteFile) => remoteFile.path === filePath)) {
133
+ try {
134
+ const targetFile = await api.RepositoryFiles.show(
135
+ target.repoID,
136
+ filePath,
137
+ target.branch
138
+ );
139
+ if (computeSha256(fileInfo.file) === targetFile.content_sha256) {
140
+ return "skip";
141
+ }
142
+ } catch (error) {
143
+ logger.warn(
144
+ `Unable to retrieve detailed information for remote file ${filePath}`
145
+ );
146
+ }
147
+ return "update";
148
+ }
149
+ return "create";
150
+ }
151
+ return defaultCommitAction;
152
+ }
118
153
 
119
154
  exports.checkEpicScope = checkEpicScope;
120
155
  exports.convertDate = convertDate;
121
156
  exports.getClient = getClient;
157
+ exports.getFileAction = getFileAction;
122
158
  exports.getToken = getToken;
123
159
  exports.getTopLevelParentGroup = getTopLevelParentGroup;
124
160
  exports.parseRepoHost = parseRepoHost;
@@ -1 +1 @@
1
- {"version":3,"file":"util.cjs.js","sources":["../src/util.ts"],"sourcesContent":["/*\n * Copyright 2021 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 { InputError } from '@backstage/errors';\nimport {\n GitLabIntegration,\n ScmIntegrationRegistry,\n} from '@backstage/integration';\nimport { Gitlab, GroupSchema } from '@gitbeaker/rest';\nimport { z } from 'zod';\nimport commonGitlabConfig from './commonGitlabConfig';\n\nexport const parseRepoHost = (repoUrl: string): string => {\n let parsed;\n try {\n parsed = new URL(`https://${repoUrl}`);\n } catch (error) {\n throw new InputError(\n `Invalid repo URL passed to publisher, got ${repoUrl}, ${error}`,\n );\n }\n return parsed.host;\n};\n\nexport const getToken = (\n config: z.infer<typeof commonGitlabConfig>,\n integrations: ScmIntegrationRegistry,\n): { token: string; integrationConfig: GitLabIntegration } => {\n const host = parseRepoHost(config.repoUrl);\n const integrationConfig = integrations.gitlab.byHost(host);\n\n if (!integrationConfig) {\n throw new InputError(\n `No matching integration configuration for host ${host}, please check your integrations config`,\n );\n }\n\n const token = config.token || integrationConfig.config.token!;\n\n return { token: token, integrationConfig: integrationConfig };\n};\n\nexport type RepoSpec = {\n repo: string;\n host: string;\n owner?: string;\n};\n\nexport const parseRepoUrl = (\n repoUrl: string,\n integrations: ScmIntegrationRegistry,\n): RepoSpec => {\n let parsed;\n try {\n parsed = new URL(`https://${repoUrl}`);\n } catch (error) {\n throw new InputError(\n `Invalid repo URL passed to publisher, got ${repoUrl}, ${error}`,\n );\n }\n const host = parsed.host;\n const owner = parsed.searchParams.get('owner') ?? undefined;\n const repo: string = parsed.searchParams.get('repo')!;\n\n const type = integrations.byHost(host)?.type;\n\n if (!type) {\n throw new InputError(\n `No matching integration configuration for host ${host}, please check your integrations config`,\n );\n }\n\n return { host, owner, repo };\n};\n\nexport function getClient(props: {\n host: string;\n token?: string;\n integrations: ScmIntegrationRegistry;\n}): InstanceType<typeof Gitlab> {\n const { host, token, integrations } = props;\n const integrationConfig = integrations.gitlab.byHost(host);\n\n if (!integrationConfig) {\n throw new InputError(\n `No matching integration configuration for host ${host}, please check your integrations config`,\n );\n }\n\n const { config } = integrationConfig;\n\n if (!config.token && !token) {\n throw new InputError(`No token available for host ${host}`);\n }\n\n const requestToken = token || config.token!;\n const tokenType = token ? 'oauthToken' : 'token';\n\n const gitlabOptions: any = {\n host: config.baseUrl,\n };\n\n gitlabOptions[tokenType] = requestToken;\n return new Gitlab(gitlabOptions);\n}\n\nexport function convertDate(\n inputDate: string | undefined,\n defaultDate: string,\n) {\n try {\n return inputDate\n ? new Date(inputDate).toISOString()\n : new Date(defaultDate).toISOString();\n } catch (error) {\n throw new InputError(`Error converting input date - ${error}`);\n }\n}\n\nexport async function getTopLevelParentGroup(\n client: InstanceType<typeof Gitlab>,\n groupId: number,\n): Promise<GroupSchema> {\n try {\n const topParentGroup = await client.Groups.show(groupId);\n if (topParentGroup.parent_id) {\n return getTopLevelParentGroup(client, topParentGroup.parent_id as number);\n }\n return topParentGroup as GroupSchema;\n } catch (error: any) {\n throw new InputError(\n `Error finding top-level parent group ID: ${error.message}`,\n );\n }\n}\n\nexport async function checkEpicScope(\n client: InstanceType<typeof Gitlab>,\n projectId: number,\n epicId: number,\n) {\n try {\n // If project exists, get the top level group id\n const project = await client.Projects.show(projectId);\n if (!project) {\n throw new InputError(\n `Project with id ${projectId} not found. Check your GitLab instance.`,\n );\n }\n const topParentGroup = await getTopLevelParentGroup(\n client,\n project.namespace.id,\n );\n if (!topParentGroup) {\n throw new InputError(`Couldn't find a suitable top-level parent group.`);\n }\n // Get the epic\n const epic = (await client.Epics.all(topParentGroup.id)).find(\n (x: any) => x.id === epicId,\n );\n if (!epic) {\n throw new InputError(\n `Epic with id ${epicId} not found in the top-level parent group ${topParentGroup.name}.`,\n );\n }\n\n const epicGroup = await client.Groups.show(epic.group_id as number);\n const projectNamespace: string = project.path_with_namespace as string;\n return projectNamespace.startsWith(epicGroup.full_path as string);\n } catch (error: any) {\n throw new InputError(`Could not find epic scope: ${error.message}`);\n }\n}\n"],"names":["InputError","Gitlab"],"mappings":"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"}
1
+ {"version":3,"file":"util.cjs.js","sources":["../src/util.ts"],"sourcesContent":["/*\n * Copyright 2021 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 { LoggerService } from '@backstage/backend-plugin-api';\nimport { InputError } from '@backstage/errors';\nimport {\n GitLabIntegration,\n ScmIntegrationRegistry,\n} from '@backstage/integration';\nimport { Gitlab, GroupSchema, RepositoryTreeSchema } from '@gitbeaker/rest';\nimport { z } from 'zod';\nimport commonGitlabConfig from './commonGitlabConfig';\n\nimport { SerializedFile } from '@backstage/plugin-scaffolder-node';\n\nimport { createHash } from 'crypto';\nimport path from 'path';\n\nexport const parseRepoHost = (repoUrl: string): string => {\n let parsed;\n try {\n parsed = new URL(`https://${repoUrl}`);\n } catch (error) {\n throw new InputError(\n `Invalid repo URL passed to publisher, got ${repoUrl}, ${error}`,\n );\n }\n return parsed.host;\n};\n\nexport const getToken = (\n config: z.infer<typeof commonGitlabConfig>,\n integrations: ScmIntegrationRegistry,\n): { token: string; integrationConfig: GitLabIntegration } => {\n const host = parseRepoHost(config.repoUrl);\n const integrationConfig = integrations.gitlab.byHost(host);\n\n if (!integrationConfig) {\n throw new InputError(\n `No matching integration configuration for host ${host}, please check your integrations config`,\n );\n }\n\n const token = config.token || integrationConfig.config.token!;\n\n return { token: token, integrationConfig: integrationConfig };\n};\n\nexport type RepoSpec = {\n repo: string;\n host: string;\n owner?: string;\n};\n\nexport const parseRepoUrl = (\n repoUrl: string,\n integrations: ScmIntegrationRegistry,\n): RepoSpec => {\n let parsed;\n try {\n parsed = new URL(`https://${repoUrl}`);\n } catch (error) {\n throw new InputError(\n `Invalid repo URL passed to publisher, got ${repoUrl}, ${error}`,\n );\n }\n const host = parsed.host;\n const owner = parsed.searchParams.get('owner') ?? undefined;\n const repo: string = parsed.searchParams.get('repo')!;\n\n const type = integrations.byHost(host)?.type;\n\n if (!type) {\n throw new InputError(\n `No matching integration configuration for host ${host}, please check your integrations config`,\n );\n }\n\n return { host, owner, repo };\n};\n\nexport function getClient(props: {\n host: string;\n token?: string;\n integrations: ScmIntegrationRegistry;\n}): InstanceType<typeof Gitlab> {\n const { host, token, integrations } = props;\n const integrationConfig = integrations.gitlab.byHost(host);\n\n if (!integrationConfig) {\n throw new InputError(\n `No matching integration configuration for host ${host}, please check your integrations config`,\n );\n }\n\n const { config } = integrationConfig;\n\n if (!config.token && !token) {\n throw new InputError(`No token available for host ${host}`);\n }\n\n const requestToken = token || config.token!;\n const tokenType = token ? 'oauthToken' : 'token';\n\n const gitlabOptions: any = {\n host: config.baseUrl,\n };\n\n gitlabOptions[tokenType] = requestToken;\n return new Gitlab(gitlabOptions);\n}\n\nexport function convertDate(\n inputDate: string | undefined,\n defaultDate: string,\n) {\n try {\n return inputDate\n ? new Date(inputDate).toISOString()\n : new Date(defaultDate).toISOString();\n } catch (error) {\n throw new InputError(`Error converting input date - ${error}`);\n }\n}\n\nexport async function getTopLevelParentGroup(\n client: InstanceType<typeof Gitlab>,\n groupId: number,\n): Promise<GroupSchema> {\n try {\n const topParentGroup = await client.Groups.show(groupId);\n if (topParentGroup.parent_id) {\n return getTopLevelParentGroup(client, topParentGroup.parent_id as number);\n }\n return topParentGroup as GroupSchema;\n } catch (error: any) {\n throw new InputError(\n `Error finding top-level parent group ID: ${error.message}`,\n );\n }\n}\n\nexport async function checkEpicScope(\n client: InstanceType<typeof Gitlab>,\n projectId: number,\n epicId: number,\n) {\n try {\n // If project exists, get the top level group id\n const project = await client.Projects.show(projectId);\n if (!project) {\n throw new InputError(\n `Project with id ${projectId} not found. Check your GitLab instance.`,\n );\n }\n const topParentGroup = await getTopLevelParentGroup(\n client,\n project.namespace.id,\n );\n if (!topParentGroup) {\n throw new InputError(`Couldn't find a suitable top-level parent group.`);\n }\n // Get the epic\n const epic = (await client.Epics.all(topParentGroup.id)).find(\n (x: any) => x.id === epicId,\n );\n if (!epic) {\n throw new InputError(\n `Epic with id ${epicId} not found in the top-level parent group ${topParentGroup.name}.`,\n );\n }\n\n const epicGroup = await client.Groups.show(epic.group_id as number);\n const projectNamespace: string = project.path_with_namespace as string;\n return projectNamespace.startsWith(epicGroup.full_path as string);\n } catch (error: any) {\n throw new InputError(`Could not find epic scope: ${error.message}`);\n }\n}\n\nfunction computeSha256(file: SerializedFile): string {\n const hash = createHash('sha256');\n hash.update(file.content);\n return hash.digest('hex');\n}\n\nexport async function getFileAction(\n fileInfo: { file: SerializedFile; targetPath?: string },\n target: { repoID: string; branch: string },\n api: InstanceType<typeof Gitlab>,\n logger: LoggerService,\n remoteFiles: RepositoryTreeSchema[],\n defaultCommitAction:\n | 'create'\n | 'delete'\n | 'update'\n | 'skip'\n | 'auto' = 'auto',\n): Promise<'create' | 'delete' | 'update' | 'skip'> {\n if (defaultCommitAction === 'auto') {\n const filePath = path.join(fileInfo.targetPath ?? '', fileInfo.file.path);\n\n if (remoteFiles?.some(remoteFile => remoteFile.path === filePath)) {\n try {\n const targetFile = await api.RepositoryFiles.show(\n target.repoID,\n filePath,\n target.branch,\n );\n if (computeSha256(fileInfo.file) === targetFile.content_sha256) {\n return 'skip';\n }\n } catch (error) {\n logger.warn(\n `Unable to retrieve detailed information for remote file ${filePath}`,\n );\n }\n return 'update';\n }\n return 'create';\n }\n return defaultCommitAction;\n}\n"],"names":["InputError","Gitlab","createHash","path"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-scaffolder-backend-module-gitlab",
3
- "version": "0.10.1-next.0",
3
+ "version": "0.11.0-next.1",
4
4
  "backstage": {
5
5
  "role": "backend-plugin-module",
6
6
  "pluginId": "scaffolder",
@@ -53,11 +53,11 @@
53
53
  "test": "backstage-cli package test"
54
54
  },
55
55
  "dependencies": {
56
- "@backstage/backend-plugin-api": "1.5.1-next.0",
56
+ "@backstage/backend-plugin-api": "1.6.0-next.1",
57
57
  "@backstage/config": "1.3.6",
58
58
  "@backstage/errors": "1.2.7",
59
- "@backstage/integration": "1.18.3-next.0",
60
- "@backstage/plugin-scaffolder-node": "0.12.2-next.0",
59
+ "@backstage/integration": "1.18.3-next.1",
60
+ "@backstage/plugin-scaffolder-node": "0.12.2-next.1",
61
61
  "@gitbeaker/requester-utils": "^41.2.0",
62
62
  "@gitbeaker/rest": "^41.2.0",
63
63
  "luxon": "^3.0.0",
@@ -65,8 +65,8 @@
65
65
  "zod": "^3.22.4"
66
66
  },
67
67
  "devDependencies": {
68
- "@backstage/backend-test-utils": "1.10.1-next.0",
69
- "@backstage/cli": "0.34.6-next.0",
70
- "@backstage/plugin-scaffolder-node-test-utils": "0.3.6-next.0"
68
+ "@backstage/backend-test-utils": "1.10.2-next.1",
69
+ "@backstage/cli": "0.35.0-next.2",
70
+ "@backstage/plugin-scaffolder-node-test-utils": "0.3.6-next.1"
71
71
  }
72
72
  }