-
Notifications
You must be signed in to change notification settings - Fork 129
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix: add gitlab support #545
base: main
Are you sure you want to change the base?
Changes from all commits
a0e4d64
360c7c7
c5e8659
43f348d
b5cc493
e56f894
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,16 @@ | ||
{ | ||
"proxyUrl": "https://github.com", | ||
"proxyConfig": [ | ||
{ | ||
"path": "/github.com", | ||
"url": "https://github.com", | ||
"enabled": true | ||
}, | ||
{ | ||
"path": "/gitlab.com", | ||
"url": "https://gitlab.com", | ||
"enabled": true | ||
Comment on lines
+4
to
+11
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I like your proposed configuration. Would it not be better to recommend |
||
} | ||
], | ||
"cookieSecret": "cookie secret", | ||
"sessionMaxAgeHours": 12, | ||
"tempPassword": { | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -94,8 +94,8 @@ | |
const canUserCancelPush = async (id, user) => { | ||
return new Promise(async (resolve) => { | ||
const pushDetail = await getPush(id); | ||
const repoName = pushDetail.repoName.replace('.git', ''); | ||
const isAllowed = await repo.isUserPushAllowed(repoName, user); | ||
const repoUrl = pushDetail.repo.url; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What about pre-existing repositories using the |
||
const isAllowed = await repo.isUserPushAllowed(repoUrl, user); | ||
|
||
if (isAllowed) { | ||
resolve(true); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -34,6 +34,22 @@ | |
}); | ||
}; | ||
|
||
exports.getRepoByUrl = async (url) => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Keep me honest, but can we not enforce uniqueness with Can we take advantage of the fact that both GitHub and GitLab follow the same naming convention for repositories and enforce uniqueness this way? GitHub: There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. While Gitlab and GitHub follow similar patterns, does everyone? What if there's no provider and just a URL (e.g. a project or firm running their own git server)? The current model does need to change as it appears to just be the repo name and that's not workable alone (two different forks of the same project can't exist nor can two different projects with the same name from different orgs). |
||
return new Promise((resolve, reject) => { | ||
db.findOne({ url: url }, (err, doc) => { | ||
if (err) { | ||
reject(err); | ||
} else { | ||
if (!doc) { | ||
resolve(null); | ||
} else { | ||
resolve(doc); | ||
} | ||
} | ||
}); | ||
}); | ||
}; | ||
|
||
exports.createRepo = async (repo) => { | ||
repo.users = { | ||
canPush: [], | ||
|
@@ -140,10 +156,9 @@ | |
}); | ||
}; | ||
|
||
exports.isUserPushAllowed = async (name, user) => { | ||
name = name.toLowerCase(); | ||
exports.isUserPushAllowed = async (url, user) => { | ||
return new Promise(async (resolve, reject) => { | ||
const repo = await exports.getRepo(name); | ||
const repo = await exports.getRepoByUrl(url); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Again, same point here - this is fine moving forward for updated data properties but what about previous stored data? Do we need a transformer? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @JamieSlome I find myself looking at the similar changes to Miklos before me. Do you mean something specific by a transformer? I.e. is there something in place for updating the DB models or do we need to invent/adopt/implement something to do so? |
||
console.log(repo.users.canPush); | ||
console.log(repo.users.canAuthorise); | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -102,8 +102,8 @@ const canUserApproveRejectPush = async (id, user) => { | |
const canUserCancelPush = async (id, user) => { | ||
return new Promise(async (resolve) => { | ||
const pushDetail = await getPush(id); | ||
const repoName = pushDetail.repoName.replace('.git', ''); | ||
const isAllowed = await repo.isUserPushAllowed(repoName, user); | ||
const repoUrl = pushDetail.repo.url; | ||
const isAllowed = await repo.isUserPushAllowed(repoUrl, user); | ||
Comment on lines
+105
to
+106
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Same point as the above comments here. |
||
|
||
if (isAllowed) { | ||
resolve(true); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -15,6 +15,11 @@ exports.getRepo = async (name) => { | |
return collection.findOne({ name: { $eq: name } }); | ||
}; | ||
|
||
exports.getRepoByUrl = async (url) => { | ||
const collection = await connect(cnName); | ||
return collection.findOne({ url: { $eq: url}}); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Thoughts? We should probably step back a little and define a data model for repository together that will serve us well moving forward and isn't too opinionated. I am not sure about URL as only as this is quite un-atomic in its nature and includes content that we don't need in DB, i.e. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. As per previous comments, I'm making another attempt at this (and only just discovered this PR addressing the same thing). I had independently run into the same issue when testing with two forks of the same project and realized that changes applied in the UI where being stored on the wrong project as Git Proxy couldn't differentiate them by name alone. Regarding breaking the URL up into the component date (platform/org/repo), I foresee (an arguably small) problem in going beyond GitHub/GitLab - in any ideal world Git proxy should be applicable to any Git repository. However, that will limit us to concepts in git protocol only and there you only really have the repository URL to work with AFAIK (orgs are contstructs that the platforms apply on top of the repos); you can't break out an org and platform which may not exist. I'm torn as to whether you do need the protocol details in the DB (i.e. the https://) as there are multiple protocols (ssh + git://) although neither are supported (and are unlikely to be). You could also host the repo with out using SSL - but that would be daft. Hence, that just leaves simplicity. It is simpler to store and use the whole valid URL as the identity of the project/repo as it avoids the need to process it (and to process it conditionally based on which host/type of host it is). This doesn't affect the end user, only the implementation, where simplicity is a benefit. |
||
} | ||
|
||
exports.createRepo = async (repo) => { | ||
console.log(`creating new repo ${JSON.stringify(repo)}`); | ||
|
||
|
@@ -67,10 +72,9 @@ exports.deleteRepo = async (name) => { | |
await collection.deleteMany({ name: name }); | ||
}; | ||
|
||
exports.isUserPushAllowed = async (name, user) => { | ||
name = name.toLowerCase(); | ||
exports.isUserPushAllowed = async (url, user) => { | ||
return new Promise(async (resolve, reject) => { | ||
const repo = await exports.getRepo(name); | ||
const repo = await exports.getRepoByUrl(url); | ||
console.log(repo.users.canPush); | ||
console.log(repo.users.canAuthorise); | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
// Regex inspector - https://www.debuggex.com/ | ||
const GIT_URL_REGEX = new RegExp("^(https)://(github\\.com|gitlab\\.com)/([a-zA-Z0-9\\-\\.]+)/([a-zA-Z0-9\\-]+)(\\.git)(/)?$"); | ||
|
||
/** Class representing a Repo. */ | ||
class Repo { | ||
url; | ||
protocol; | ||
host; | ||
project; | ||
name; | ||
|
||
/** | ||
* | ||
* @param {string} url The url for the repo. | ||
*/ | ||
constructor(url) { | ||
const parsedUrl = url?.match(GIT_URL_REGEX); | ||
if (parsedUrl) { | ||
this.url = url; | ||
this.protocol = parsedUrl[1]; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why is this required? |
||
this.host = parsedUrl[2]; | ||
this.project = parsedUrl[3]; | ||
this.name = parsedUrl[4] + parsedUrl[5]; // repo name + .git | ||
return; | ||
} | ||
throw new Error(`Invalid repo url: "${url}"`); | ||
} | ||
} | ||
|
||
exports.Repo = Repo; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
exports.Repo = require('./Repo').Repo; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,4 @@ | ||
/** Class representing a Push. */ | ||
const config = require('../../config'); | ||
|
||
/** | ||
* Create a new action | ||
|
@@ -35,9 +34,9 @@ class Action { | |
this.type = type; | ||
this.method = method; | ||
this.timestamp = timestamp; | ||
this.project = repo.split('/')[0]; | ||
this.repoName = repo.split('/')[1]; | ||
this.url = `${config.getProxyUrl()}/${repo}`; | ||
this.project = repo.project; | ||
this.repoName = repo.name; | ||
this.url = repo.url; | ||
Comment on lines
+37
to
+39
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Nice 👍 |
||
this.repo = repo; | ||
} | ||
|
||
|
@@ -97,7 +96,7 @@ class Action { | |
} | ||
|
||
/** | ||
*` | ||
* | ||
*/ | ||
setAllowPush() { | ||
this.allowPush = true; | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we have type
url
?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nope - but there is a format assertion for URIs