mirror of
https://gitlab.com/openstapps/openstapps.git
synced 2026-01-22 01:22:54 +00:00
feat: add command to copy issues
This commit is contained in:
17
src/api.ts
17
src/api.ts
@@ -28,6 +28,7 @@ import {
|
|||||||
MergeRequestApproval,
|
MergeRequestApproval,
|
||||||
MergeRequestState,
|
MergeRequestState,
|
||||||
Milestone,
|
Milestone,
|
||||||
|
Note,
|
||||||
Project,
|
Project,
|
||||||
Scope,
|
Scope,
|
||||||
Tag,
|
Tag,
|
||||||
@@ -425,6 +426,16 @@ export class Api {
|
|||||||
return this.makeGitLabAPIRequest(`projects/${projectId}/milestones`) as Promise<Milestone[]>;
|
return this.makeGitLabAPIRequest(`projects/${projectId}/milestones`) as Promise<Milestone[]>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get notes for issue
|
||||||
|
*
|
||||||
|
* @param projectId Project ID of issue to get notes for
|
||||||
|
* @param issue Issue to get notes for
|
||||||
|
*/
|
||||||
|
public async getNotes(projectId: number, issue: Issue) {
|
||||||
|
return this.makeGitLabAPIRequest(`/projects/${projectId}/issues/${issue.iid}/notes?sort=asc`) as Promise<Note[]>;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get projects for a group
|
* Get projects for a group
|
||||||
*
|
*
|
||||||
@@ -503,6 +514,7 @@ export class Api {
|
|||||||
json: true,
|
json: true,
|
||||||
method: _options.method,
|
method: _options.method,
|
||||||
timeout: 60000,
|
timeout: 60000,
|
||||||
|
followAllRedirects: true,
|
||||||
transform: (bodyToTransform, response) => {
|
transform: (bodyToTransform, response) => {
|
||||||
const xTotalPages = response.headers['x-total-pages'];
|
const xTotalPages = response.headers['x-total-pages'];
|
||||||
|
|
||||||
@@ -514,7 +526,7 @@ export class Api {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error.error.message.includes('not responding') || _options.retryOnAnyError) {
|
if (error.message.includes('not responding') || _options.retryOnAnyError) {
|
||||||
const seconds = 5;
|
const seconds = 5;
|
||||||
|
|
||||||
Logger.warn(`GitLab was not responding. Waiting ${seconds}s and retrying...`);
|
Logger.warn(`GitLab was not responding. Waiting ${seconds}s and retrying...`);
|
||||||
@@ -525,6 +537,9 @@ export class Api {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Logger.log(url);
|
||||||
|
Logger.log(JSON.stringify(options));
|
||||||
|
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
76
src/cli.ts
76
src/cli.ts
@@ -20,7 +20,7 @@ import {Command} from 'commander';
|
|||||||
import {readFileSync} from 'fs';
|
import {readFileSync} from 'fs';
|
||||||
import {join} from 'path';
|
import {join} from 'path';
|
||||||
import {Api, ApiRequestOptions} from './api';
|
import {Api, ApiRequestOptions} from './api';
|
||||||
import {Issue} from './types';
|
import {Issue, IssueState, MembershipScope, Scope} from './types';
|
||||||
|
|
||||||
Logger.setTransformations([
|
Logger.setTransformations([
|
||||||
new AddLogLevel(),
|
new AddLogLevel(),
|
||||||
@@ -97,6 +97,80 @@ commander
|
|||||||
Logger.ok('Processed all issues.');
|
Logger.ok('Processed all issues.');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
commander
|
||||||
|
.command('copy <projectId> <targetUrl> <targetToken> <targetProjectId>')
|
||||||
|
.action(async (projectId, targetUrl, targetToken, targetProjectId) => {
|
||||||
|
const api = new Api(commander.url, commander.token);
|
||||||
|
const targetApi = new Api(targetUrl, targetToken);
|
||||||
|
|
||||||
|
// get all issues from project
|
||||||
|
const issues = await api.makeGitLabAPIRequest(`/projects/${projectId}/issues`, {
|
||||||
|
retryOnAnyError: true,
|
||||||
|
tries: 10,
|
||||||
|
}) as Issue[];
|
||||||
|
|
||||||
|
// sort issues by their project specific ids
|
||||||
|
issues.sort((a, b) => {
|
||||||
|
return a.iid - b.iid;
|
||||||
|
});
|
||||||
|
|
||||||
|
// get members of target project
|
||||||
|
const members = await targetApi.getMembers(MembershipScope.PROJECTS, targetProjectId);
|
||||||
|
|
||||||
|
let idx = 0;
|
||||||
|
|
||||||
|
// tslint:disable-next-line:no-magic-numbers
|
||||||
|
await asyncPool(2, issues, async (issue) => {
|
||||||
|
// get notes of old issue
|
||||||
|
const notes = await api.getNotes(projectId, issue);
|
||||||
|
|
||||||
|
// create new issue
|
||||||
|
const newIssue = await targetApi.createIssue(targetProjectId, issue.title, issue.description === null ? '---' : `${issue.web_url}
|
||||||
|
|
||||||
|
${issue.description}`);
|
||||||
|
|
||||||
|
for (const note of notes) {
|
||||||
|
// skip system notes
|
||||||
|
if (note.system) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// create new note in new issue for every note in issue
|
||||||
|
await targetApi.createNote(targetProjectId, Scope.ISSUES, newIssue.iid, `**${note.author.name} (@${note.author.username}):**
|
||||||
|
|
||||||
|
${note.body}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// close newly created issue if original is closed to
|
||||||
|
if (issue.state === IssueState.CLOSED) {
|
||||||
|
await targetApi.makeGitLabAPIRequest(`/projects/${targetProjectId}/issues/${newIssue.iid}`, {
|
||||||
|
data: {
|
||||||
|
state_event: 'close',
|
||||||
|
},
|
||||||
|
method: 'PUT',
|
||||||
|
retryOnAnyError: true,
|
||||||
|
tries: 10,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// search for member in target group with same username
|
||||||
|
const assignee = members.find((member) => {
|
||||||
|
if (issue.assignee === null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return member.username === issue.assignee.username;
|
||||||
|
});
|
||||||
|
|
||||||
|
// set assignee if usernames match
|
||||||
|
if (typeof assignee !== 'undefined') {
|
||||||
|
await targetApi.setAssigneeForIssue(newIssue, assignee.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger.log(`Finished issue ${++idx} of ${issues.length}.`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
commander
|
commander
|
||||||
.parse(process.argv);
|
.parse(process.argv);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user