-
Notifications
You must be signed in to change notification settings - Fork 1.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'feat/add-gitlab-mr-context-provider' of https://github.…
…com/theBenForce/continue into theBenForce-feat/add-gitlab-mr-context-provider
- Loading branch information
Showing
10 changed files
with
459 additions
and
28 deletions.
There are no files selected for viewing
277 changes: 277 additions & 0 deletions
277
core/context/providers/GitLabMergeRequestContextProvider.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,277 @@ | ||
import { AxiosInstance, AxiosError } from "axios"; | ||
import {BaseContextProvider} from ".."; | ||
import { ContextProviderExtras, ContextItem, ContextProviderDescription } from "../.."; | ||
|
||
interface RemoteBranchInfo { | ||
branch: string | null; | ||
project: string | null; | ||
} | ||
|
||
interface GitLabUser { | ||
id: number; | ||
username: string; | ||
name: string; | ||
state: "active"; | ||
locked: boolean; | ||
avatar_url: string; | ||
web_url: string; | ||
} | ||
|
||
interface GitLabMergeRequest { | ||
iid: number; | ||
project_id: number; | ||
title: string; | ||
description: string; | ||
} | ||
|
||
interface GitLabComment { | ||
type: null | "DiffNote"; | ||
resolvable: boolean; | ||
resolved?: boolean; | ||
body: string; | ||
created_at: string; | ||
author: GitLabUser; | ||
position?: { | ||
new_path: string; | ||
new_line: number; | ||
head_sha: string; | ||
line_range: { | ||
start: { | ||
line_code: string; | ||
type: "new"; | ||
old_line: null; | ||
new_line: number; | ||
}; | ||
end: { | ||
line_code: string; | ||
type: "new"; | ||
old_line: null; | ||
new_line: number; | ||
}; | ||
}; | ||
}; | ||
} | ||
|
||
const trimFirstElement = (args: Array<string>): string => { | ||
return args[0].trim(); | ||
}; | ||
|
||
const getSubprocess = async (extras: ContextProviderExtras) => { | ||
const workingDir = await extras.ide | ||
.getWorkspaceDirs() | ||
.then(trimFirstElement); | ||
|
||
return (command: string) => | ||
extras.ide | ||
.subprocess(`cd ${workingDir}; ${command}`) | ||
.then(trimFirstElement); | ||
}; | ||
|
||
|
||
class GitLabMergeRequestContextProvider extends BaseContextProvider { | ||
static description: ContextProviderDescription = { | ||
title: 'gitlab-mr', | ||
displayTitle: 'GitLab Merge Request', | ||
description: 'Reference comments in a GitLab Merge Request', | ||
type: 'normal' | ||
}; | ||
|
||
private async getApi(): Promise<AxiosInstance> { | ||
const { default: Axios } = await import("axios"); | ||
|
||
const domain = this.options.domain ?? "gitlab.com"; | ||
const token = this.options.token; | ||
|
||
if(!token) { | ||
throw new Error(`GitLab Private Token is required!`); | ||
} | ||
|
||
return Axios.create({ | ||
baseURL: `https://${domain ?? "gitlab.com"}/api/v4`, | ||
headers: { | ||
"PRIVATE-TOKEN": token, | ||
}, | ||
}); | ||
}; | ||
|
||
|
||
private async getRemoteBranchName(extras: ContextProviderExtras): Promise<RemoteBranchInfo> { | ||
|
||
const subprocess = await getSubprocess(extras); | ||
|
||
const branchName = await subprocess(`git branch --show-current`); | ||
|
||
const branchRemote = await subprocess( | ||
`git config branch.${branchName}.remote` | ||
); | ||
|
||
const branchInfo = await subprocess(`git branch -vv`); | ||
|
||
const currentBranchInfo = branchInfo | ||
.split("\n") | ||
.find((line) => line.startsWith("*")); | ||
|
||
const remoteMatches = RegExp( | ||
`\\[${branchRemote}/(?<remote_branch>[^\\]]+)\\]` | ||
).exec(currentBranchInfo!); | ||
|
||
console.dir({ remoteMatches }); | ||
|
||
const remoteBranch = remoteMatches?.groups?.["remote_branch"] ?? null; | ||
|
||
const remoteUrl = await subprocess(`git remote get-url ${branchRemote}`); | ||
|
||
const urlMatches = RegExp(`:(?<project>.*).git`).exec(remoteUrl); | ||
|
||
const project = urlMatches?.groups?.["project"] ?? null; | ||
|
||
return { | ||
branch: remoteBranch, | ||
project, | ||
}; | ||
}; | ||
|
||
async getContextItems(query: string, extras: ContextProviderExtras): Promise<ContextItem[]> { | ||
|
||
const { branch, project } = await this.getRemoteBranchName(extras); | ||
|
||
const api = await this.getApi(); | ||
|
||
const result = [] as Array<ContextItem>; | ||
|
||
try { | ||
const mergeRequests = await api | ||
.get<Array<GitLabMergeRequest>>( | ||
`/projects/${encodeURIComponent(project!)}/merge_requests`, | ||
{ | ||
params: { | ||
source_branch: branch, | ||
state: "opened", | ||
}, | ||
} | ||
) | ||
.then((x) => x.data); | ||
|
||
|
||
const subprocess = await getSubprocess(extras); | ||
|
||
for (const mergeRequest of mergeRequests) { | ||
const parts = [ | ||
`# GitLab Merge Request\ntitle: "${mergeRequest.title}"\ndescription: "${mergeRequest.description ?? 'None'}"`, | ||
`## Comments`, | ||
]; | ||
|
||
const comments = await api.get<Array<GitLabComment>>( | ||
`/projects/${mergeRequest.project_id}/merge_requests/${mergeRequest.iid}/notes`, | ||
{ | ||
params: { | ||
sort: "asc", | ||
order_by: "created_at", | ||
}, | ||
} | ||
); | ||
|
||
const filteredComments = comments.data.filter( | ||
(x) => x.type === "DiffNote" | ||
); | ||
|
||
const locations = {} as Record<string, Array<GitLabComment>>; | ||
|
||
for (const comment of filteredComments) { | ||
const filename = comment.position?.new_path ?? "general"; | ||
|
||
if (!locations[filename]) { | ||
locations[filename] = []; | ||
} | ||
|
||
locations[filename].push(comment); | ||
} | ||
|
||
if (extras.selectedCode.length && this.options.filterComments) { | ||
const toRemove = Object.keys(locations).filter(filename => !extras.selectedCode.find(selection => selection.filepath.endsWith(filename)) && filename !== "general"); | ||
for (const filepath of toRemove) { | ||
delete locations[filepath]; | ||
} | ||
} | ||
|
||
const commentFormatter = async (comment: GitLabComment) => { | ||
const commentLabel = comment.body.includes("```suggestion") ? 'Code Suggestion' : 'Comment'; | ||
let result = `#### ${commentLabel}\nauthor: "${comment.author.name}"\ndate: "${comment.created_at}"\nresolved: ${ | ||
comment.resolved ? "Yes" : "No" | ||
}`; | ||
|
||
if (comment.position?.new_line) { | ||
result += `\nline: ${comment.position.new_line}`; | ||
|
||
if (comment.position.head_sha) { | ||
const sourceLines = await subprocess(`git show ${comment.position.head_sha}:${comment.position.new_path}`).then(result => result.split("\n")).catch(ex => []); | ||
|
||
const line = comment.position.new_line <= sourceLines.length ? sourceLines[comment.position.new_line - 1] : null; | ||
|
||
if (line) { | ||
result += `\nsource: \`${line}\``; | ||
} | ||
} | ||
} | ||
|
||
result += `\n\n${comment.body}`; | ||
|
||
return result; | ||
}; | ||
|
||
for (const [filename, locationComments] of Object.entries(locations)) { | ||
if (filename !== "general") { | ||
parts.push(`### File ${filename}`); | ||
locationComments.sort( | ||
(a, b) => a.position!.new_line - b.position!.new_line | ||
); | ||
} else { | ||
parts.push("### General"); | ||
} | ||
|
||
const commentSections = await Promise.all(locationComments.map(commentFormatter)); | ||
parts.push(...commentSections); | ||
} | ||
|
||
|
||
const content = parts.join("\n\n"); | ||
|
||
result.push( | ||
{ | ||
name: mergeRequest.title, | ||
content, | ||
description: `Comments from the Merge Request for this branch.`, | ||
}, | ||
); | ||
} | ||
|
||
} catch (ex) { | ||
let content = `# GitLab Merge Request\n\nError getting merge request. `; | ||
if (ex instanceof AxiosError) { | ||
if (ex.response) { | ||
const errorMessage = ex.response?.data ? ex.response.data.message ?? JSON.stringify(ex.response?.data) : `${ex.response.status}: ${ex.response.statusText}`; | ||
content += `GitLab Error: ${errorMessage}`; | ||
} else { | ||
content += `GitLab Request Error ${ex.request}`; | ||
} | ||
} else { | ||
// @ts-ignore | ||
content += `Unknown error: ${ex.message ?? JSON.stringify(ex)}`; | ||
} | ||
|
||
|
||
result.push( | ||
{ | ||
name: `GitLab Merge Request`, | ||
content, | ||
description: `Error getting the Merge Request for this branch.`, | ||
}, | ||
); | ||
} | ||
|
||
return result; | ||
} | ||
|
||
} | ||
|
||
export default GitLabMergeRequestContextProvider; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.