|
Pierre-Yves Chibon |
33b534 |
# -*- coding: utf-8 -*-
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
"""
|
|
Pierre-Yves Chibon |
fac0b1 |
(c) 2014 - Copyright Red Hat Inc
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
Authors:
|
|
Pierre-Yves Chibon |
fac0b1 |
Pierre-Yves Chibon <pingou@pingoured.fr></pingou@pingoured.fr>
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
"""
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
import flask
|
|
Pierre-Yves Chibon |
fac0b1 |
import os
|
|
Pierre-Yves Chibon |
be7090 |
import shutil
|
|
Pierre-Yves Chibon |
41558c |
import tempfile
|
|
Pierre-Yves Chibon |
fac0b1 |
from math import ceil
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
import pygit2
|
|
Pierre-Yves Chibon |
fac0b1 |
from sqlalchemy.exc import SQLAlchemyError
|
|
Pierre-Yves Chibon |
fac0b1 |
from pygments import highlight
|
|
Pierre-Yves Chibon |
fac0b1 |
from pygments.lexers import guess_lexer
|
|
Pierre-Yves Chibon |
fac0b1 |
from pygments.lexers.text import DiffLexer
|
|
Pierre-Yves Chibon |
fac0b1 |
from pygments.formatters import HtmlFormatter
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
6bf4e3 |
import progit.doc_utils
|
|
Pierre-Yves Chibon |
80bba3 |
import progit.lib
|
|
Pierre-Yves Chibon |
118968 |
import progit.lib.git
|
|
Pierre-Yves Chibon |
ac8023 |
import progit.forms
|
|
Pierre-Yves Chibon |
1e5fe9 |
from progit import (APP, SESSION, LOG, __get_file_in_tree, cla_required,
|
|
Pierre-Yves Chibon |
59be40 |
is_repo_admin, generate_gitolite_acls)
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
8df273 |
def _get_repo_path(repo):
|
|
Pierre-Yves Chibon |
8df273 |
""" Return the pat of the git repository corresponding to the provided
|
|
Pierre-Yves Chibon |
8df273 |
Repository object from the DB.
|
|
Pierre-Yves Chibon |
8df273 |
"""
|
|
Pierre-Yves Chibon |
8df273 |
if repo.is_fork:
|
|
Pierre-Yves Chibon |
8df273 |
repopath = os.path.join(APP.config['FORK_FOLDER'], repo.path)
|
|
Pierre-Yves Chibon |
8df273 |
else:
|
|
Pierre-Yves Chibon |
8df273 |
repopath = os.path.join(APP.config['GIT_FOLDER'], repo.path)
|
|
Pierre-Yves Chibon |
8df273 |
return repopath
|
|
Pierre-Yves Chibon |
8df273 |
|
|
Pierre-Yves Chibon |
8df273 |
|
|
Pierre-Yves Chibon |
8df273 |
def _get_parent_repo_path(repo):
|
|
Pierre-Yves Chibon |
8df273 |
""" Return the path of the parent git repository corresponding to the
|
|
Pierre-Yves Chibon |
8df273 |
provided Repository object from the DB.
|
|
Pierre-Yves Chibon |
8df273 |
"""
|
|
Pierre-Yves Chibon |
8df273 |
if repo.parent:
|
|
Pierre-Yves Chibon |
8df273 |
parentpath = os.path.join(APP.config['GIT_FOLDER'], repo.parent.path)
|
|
Pierre-Yves Chibon |
8df273 |
else:
|
|
Pierre-Yves Chibon |
8df273 |
parentpath = os.path.join(APP.config['GIT_FOLDER'], repo.path)
|
|
Pierre-Yves Chibon |
8df273 |
return parentpath
|
|
Pierre-Yves Chibon |
8df273 |
|
|
Pierre-Yves Chibon |
8df273 |
|
|
Pierre-Yves Chibon |
b31474 |
@APP.route('/<repo>/request-pulls')</repo>
|
|
Pierre-Yves Chibon |
b31474 |
@APP.route('/fork/<username>/<repo>/request-pulls')</repo></username>
|
|
Pierre-Yves Chibon |
b31474 |
def request_pulls(repo, username=None):
|
|
Pierre-Yves Chibon |
b31474 |
""" Request pulling the changes from the fork into the project.
|
|
Pierre-Yves Chibon |
47950c |
"""
|
|
Pierre-Yves Chibon |
b31474 |
status = flask.request.args.get('status', True)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
if not repo:
|
|
Pierre-Yves Chibon |
47950c |
flask.abort(404, 'Project not found')
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
if status is False or str(status).lower() == 'closed':
|
|
Pierre-Yves Chibon |
ff8c27 |
requests = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
b31474 |
SESSION, project_id=repo.id, status=False)
|
|
Pierre-Yves Chibon |
47950c |
else:
|
|
Pierre-Yves Chibon |
ff8c27 |
requests = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
b31474 |
SESSION, project_id=repo.id, status=status)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
return flask.render_template(
|
|
Pierre-Yves Chibon |
b31474 |
'requests.html',
|
|
Pierre-Yves Chibon |
b31474 |
select='requests',
|
|
Pierre-Yves Chibon |
b31474 |
repo=repo,
|
|
Pierre-Yves Chibon |
b31474 |
username=username,
|
|
Pierre-Yves Chibon |
b31474 |
requests=requests,
|
|
Pierre-Yves Chibon |
b31474 |
status=status,
|
|
Pierre-Yves Chibon |
b31474 |
)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/<repo>/request-pull/<int:requestid>')</int:requestid></repo>
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/fork/<username>/<repo>/request-pull/<int:requestid>')</int:requestid></repo></username>
|
|
Pierre-Yves Chibon |
b31474 |
def request_pull(repo, requestid, username=None):
|
|
Pierre-Yves Chibon |
47950c |
""" Request pulling the changes from the fork into the project.
|
|
Pierre-Yves Chibon |
47950c |
"""
|
|
Pierre-Yves Chibon |
b31474 |
|
|
Pierre-Yves Chibon |
47950c |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
if not repo:
|
|
Pierre-Yves Chibon |
47950c |
flask.abort(404, 'Project not found')
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
ff8c27 |
request = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
47950c |
SESSION, project_id=repo.id, requestid=requestid)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
if not request:
|
|
Pierre-Yves Chibon |
47950c |
flask.abort(404, 'Pull-request not found')
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
0784c9 |
repo_from = request.repo_from
|
|
Pierre-Yves Chibon |
8df273 |
repopath = _get_repo_path(repo_from)
|
|
Pierre-Yves Chibon |
47950c |
repo_obj = pygit2.Repository(repopath)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
8df273 |
parentpath = _get_parent_repo_path(repo_from)
|
|
Pierre-Yves Chibon |
8df273 |
orig_repo = pygit2.Repository(parentpath)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
e31131 |
branch = repo_obj.lookup_branch(request.branch_from)
|
|
Pierre-Yves Chibon |
e31131 |
commitid = branch.get_object().hex
|
|
Pierre-Yves Chibon |
e31131 |
|
|
Pierre-Yves Chibon |
47950c |
diff_commits = []
|
|
Pierre-Yves Chibon |
e31131 |
diff = None
|
|
Pierre-Yves Chibon |
47950c |
if not repo_obj.is_empty and not orig_repo.is_empty:
|
|
Pierre-Yves Chibon |
2c9e9f |
orig_commit = orig_repo[
|
|
Pierre-Yves Chibon |
e31131 |
orig_repo.lookup_branch(request.branch).get_object().hex]
|
|
Pierre-Yves Chibon |
2c9e9f |
|
|
Pierre-Yves Chibon |
afd10d |
# Closed pull-request
|
|
Pierre-Yves Chibon |
f986ce |
if request.status is False:
|
|
Pierre-Yves Chibon |
f986ce |
commitid = request.commit_stop
|
|
Pierre-Yves Chibon |
afd10d |
for commit in repo_obj.walk(commitid, pygit2.GIT_SORT_TIME):
|
|
Pierre-Yves Chibon |
afd10d |
diff_commits.append(commit)
|
|
Pierre-Yves Chibon |
afd10d |
if commit.oid.hex == request.commit_start:
|
|
Pierre-Yves Chibon |
afd10d |
break
|
|
Pierre-Yves Chibon |
afd10d |
# Pull-request open
|
|
Pierre-Yves Chibon |
afd10d |
else:
|
|
Pierre-Yves Chibon |
afd10d |
master_commits = [
|
|
Pierre-Yves Chibon |
afd10d |
commit.oid.hex
|
|
Pierre-Yves Chibon |
afd10d |
for commit in orig_repo.walk(
|
|
Pierre-Yves Chibon |
afd10d |
orig_repo.lookup_branch(request.branch).get_object().hex,
|
|
Pierre-Yves Chibon |
afd10d |
pygit2.GIT_SORT_TIME)
|
|
Pierre-Yves Chibon |
afd10d |
]
|
|
Pierre-Yves Chibon |
afd10d |
for commit in repo_obj.walk(commitid, pygit2.GIT_SORT_TIME):
|
|
Pierre-Yves Chibon |
afd10d |
if request.status and commit.oid.hex in master_commits:
|
|
Pierre-Yves Chibon |
afd10d |
break
|
|
Pierre-Yves Chibon |
afd10d |
diff_commits.append(commit)
|
|
Pierre-Yves Chibon |
afd10d |
|
|
Pierre-Yves Chibon |
afd10d |
if request.status:
|
|
Pierre-Yves Chibon |
afd10d |
first_commit = repo_obj[diff_commits[-1].oid.hex]
|
|
Pierre-Yves Chibon |
afd10d |
request.commit_start = first_commit.oid.hex
|
|
Pierre-Yves Chibon |
afd10d |
request.commit_stop = diff_commits[0].oid.hex
|
|
Pierre-Yves Chibon |
afd10d |
SESSION.add(request)
|
|
Pierre-Yves Chibon |
1a6537 |
try:
|
|
Pierre-Yves Chibon |
1a6537 |
SESSION.commit()
|
|
Pierre-Yves Chibon |
1a6537 |
except SQLAlchemyError as err:
|
|
Pierre-Yves Chibon |
1a6537 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
1a6537 |
APP.logger.exception(err)
|
|
Pierre-Yves Chibon |
1a6537 |
flask.flash(
|
|
Pierre-Yves Chibon |
1a6537 |
'Could not update this pull-request in the database',
|
|
Pierre-Yves Chibon |
1a6537 |
'error')
|
|
Pierre-Yves Chibon |
e31131 |
|
|
Pierre-Yves Chibon |
e31131 |
if diff_commits:
|
|
Pierre-Yves Chibon |
e31131 |
first_commit = repo_obj[diff_commits[-1].oid.hex]
|
|
Pierre-Yves Chibon |
e31131 |
diff = repo_obj.diff(
|
|
Pierre-Yves Chibon |
e31131 |
repo_obj.revparse_single(first_commit.parents[0].oid.hex),
|
|
Pierre-Yves Chibon |
e31131 |
repo_obj.revparse_single(diff_commits[0].oid.hex)
|
|
Pierre-Yves Chibon |
e31131 |
)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
elif orig_repo.is_empty:
|
|
Pierre-Yves Chibon |
47950c |
orig_commit = None
|
|
Pierre-Yves Chibon |
60052e |
repo_commit = repo_obj[request.stop_id]
|
|
Pierre-Yves Chibon |
47950c |
diff = repo_commit.tree.diff_to_tree(swap=True)
|
|
Pierre-Yves Chibon |
47950c |
else:
|
|
Pierre-Yves Chibon |
47950c |
flask.flash(
|
|
Pierre-Yves Chibon |
47950c |
'Fork is empty, there are no commits to request pulling',
|
|
Pierre-Yves Chibon |
47950c |
'error')
|
|
Pierre-Yves Chibon |
47950c |
return flask.redirect(flask.url_for(
|
|
Pierre-Yves Chibon |
ea8735 |
'view_repo', username=username, repo=repo.name))
|
|
Pierre-Yves Chibon |
47950c |
|
|
Johan Cwiklinski |
fd18bd |
form = progit.forms.ConfirmationForm()
|
|
Johan Cwiklinski |
fd18bd |
|
|
Pierre-Yves Chibon |
47950c |
return flask.render_template(
|
|
Pierre-Yves Chibon |
47950c |
'pull_request.html',
|
|
Pierre-Yves Chibon |
47950c |
select='requests',
|
|
Pierre-Yves Chibon |
04ed02 |
requestid=requestid,
|
|
Pierre-Yves Chibon |
47950c |
repo=repo,
|
|
Pierre-Yves Chibon |
41558c |
username=username,
|
|
Pierre-Yves Chibon |
87ba70 |
pull_request=request,
|
|
Pierre-Yves Chibon |
2e8ea2 |
repo_admin=is_repo_admin(request.repo),
|
|
Pierre-Yves Chibon |
47950c |
diff_commits=diff_commits,
|
|
Pierre-Yves Chibon |
e31131 |
diff=diff,
|
|
Johan Cwiklinski |
fd18bd |
mergeform=form,
|
|
Pierre-Yves Chibon |
47950c |
)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
e31131 |
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/<repo>/request-pull/<int:requestid>.patch')</int:requestid></repo>
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/fork/<username>/<repo>/request-pull/<int:requestid>.patch')</int:requestid></repo></username>
|
|
Pierre-Yves Chibon |
e7de33 |
def request_pull_patch(repo, requestid, username=None):
|
|
Pierre-Yves Chibon |
e7de33 |
""" Returns the commits from the specified pull-request as patches.
|
|
Pierre-Yves Chibon |
e7de33 |
"""
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
e7de33 |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
e7de33 |
if not repo:
|
|
Pierre-Yves Chibon |
e7de33 |
flask.abort(404, 'Project not found')
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
ff8c27 |
request = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
e7de33 |
SESSION, project_id=repo.id, requestid=requestid)
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
e7de33 |
if not request:
|
|
Pierre-Yves Chibon |
e7de33 |
flask.abort(404, 'Pull-request not found')
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
8df273 |
repo_from = request.repo_from
|
|
Pierre-Yves Chibon |
8df273 |
repopath = _get_repo_path(repo_from)
|
|
Pierre-Yves Chibon |
e7de33 |
repo_obj = pygit2.Repository(repopath)
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
8df273 |
parentpath = _get_parent_repo_path(repo_from)
|
|
Pierre-Yves Chibon |
8df273 |
orig_repo = pygit2.Repository(parentpath)
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
18e007 |
branch = repo_obj.lookup_branch(request.branch_from)
|
|
Pierre-Yves Chibon |
18e007 |
commitid = branch.get_object().hex
|
|
Pierre-Yves Chibon |
18e007 |
|
|
Pierre-Yves Chibon |
e7de33 |
diff_commits = []
|
|
Pierre-Yves Chibon |
e7de33 |
if not repo_obj.is_empty and not orig_repo.is_empty:
|
|
Pierre-Yves Chibon |
e7de33 |
orig_commit = orig_repo[
|
|
Pierre-Yves Chibon |
18e007 |
orig_repo.lookup_branch(request.branch).get_object().hex]
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
a8ed59 |
# Closed pull-request
|
|
Pierre-Yves Chibon |
a8ed59 |
if request.status is False:
|
|
Pierre-Yves Chibon |
a8ed59 |
commitid = request.commit_stop
|
|
Pierre-Yves Chibon |
a8ed59 |
for commit in repo_obj.walk(commitid, pygit2.GIT_SORT_TIME):
|
|
Pierre-Yves Chibon |
a8ed59 |
diff_commits.append(commit)
|
|
Pierre-Yves Chibon |
a8ed59 |
if commit.oid.hex == request.commit_start:
|
|
Pierre-Yves Chibon |
a8ed59 |
break
|
|
Pierre-Yves Chibon |
a8ed59 |
# Pull-request open
|
|
Pierre-Yves Chibon |
a8ed59 |
else:
|
|
Pierre-Yves Chibon |
a8ed59 |
master_commits = [
|
|
Pierre-Yves Chibon |
a8ed59 |
commit.oid.hex
|
|
Pierre-Yves Chibon |
a8ed59 |
for commit in orig_repo.walk(
|
|
Pierre-Yves Chibon |
a8ed59 |
orig_repo.lookup_branch(request.branch).get_object().hex,
|
|
Pierre-Yves Chibon |
a8ed59 |
pygit2.GIT_SORT_TIME)
|
|
Pierre-Yves Chibon |
a8ed59 |
]
|
|
Pierre-Yves Chibon |
a8ed59 |
for commit in repo_obj.walk(commitid, pygit2.GIT_SORT_TIME):
|
|
Pierre-Yves Chibon |
a8ed59 |
if request.status and commit.oid.hex in master_commits:
|
|
Pierre-Yves Chibon |
a8ed59 |
break
|
|
Pierre-Yves Chibon |
a8ed59 |
diff_commits.append(commit)
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
e7de33 |
elif orig_repo.is_empty:
|
|
Pierre-Yves Chibon |
e7de33 |
orig_commit = None
|
|
Pierre-Yves Chibon |
60052e |
repo_commit = repo_obj[request.stop_id]
|
|
Pierre-Yves Chibon |
e7de33 |
diff = repo_commit.tree.diff_to_tree(swap=True)
|
|
Pierre-Yves Chibon |
e7de33 |
else:
|
|
Pierre-Yves Chibon |
e7de33 |
flask.flash(
|
|
Pierre-Yves Chibon |
e7de33 |
'Fork is empty, there are no commits to request pulling',
|
|
Pierre-Yves Chibon |
e7de33 |
'error')
|
|
Pierre-Yves Chibon |
e7de33 |
return flask.redirect(flask.url_for(
|
|
Pierre-Yves Chibon |
e7de33 |
'view_repo', username=username, repo=repo.name))
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
e7de33 |
diff_commits.reverse()
|
|
Pierre-Yves Chibon |
118968 |
patch = progit.lib.git.commit_to_patch(repo_obj, diff_commits)
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
e7de33 |
return flask.Response(patch, content_type="text/plain;charset=UTF-8")
|
|
Pierre-Yves Chibon |
e7de33 |
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
abdb8f |
@APP.route('/<repo>/request-pull/<int:requestid>/comment/<commit>/'</commit></int:requestid></repo>
|
|
Pierre-Yves Chibon |
abdb8f |
'<filename>/<row>', methods=('GET', 'POST'))</row></filename>
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/fork/<username>/<repo>/request-pull/<int:requestid>/comment/'</int:requestid></repo></username>
|
|
Pierre-Yves Chibon |
abdb8f |
'<commit>/<filename>/<row>', methods=('GET', 'POST'))</row></filename></commit>
|
|
Pierre-Yves Chibon |
abdb8f |
def pull_request_add_comment(repo, requestid, commit, filename, row,
|
|
Pierre-Yves Chibon |
abdb8f |
username=None):
|
|
Pierre-Yves Chibon |
cb10a1 |
""" Add a comment to a commit in a pull-request.
|
|
Pierre-Yves Chibon |
cb10a1 |
"""
|
|
Pierre-Yves Chibon |
cb10a1 |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
cb10a1 |
if not repo:
|
|
Pierre-Yves Chibon |
cb10a1 |
flask.abort(404, 'Project not found')
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
ff8c27 |
request = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
cb10a1 |
SESSION, project_id=repo.id, requestid=requestid)
|
|
Pierre-Yves Chibon |
cb10a1 |
repo = request.repo_from
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
cb10a1 |
if not request:
|
|
Pierre-Yves Chibon |
cb10a1 |
flask.abort(404, 'Pull-request not found')
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
cb10a1 |
form = progit.forms.AddPullRequestCommentForm()
|
|
Pierre-Yves Chibon |
cb10a1 |
form.commit.data = commit
|
|
Pierre-Yves Chibon |
abdb8f |
form.filename.data = filename
|
|
Pierre-Yves Chibon |
cb10a1 |
form.requestid.data = requestid
|
|
Pierre-Yves Chibon |
cb10a1 |
form.row.data = row
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
248f5e |
if form.validate_on_submit():
|
|
Pierre-Yves Chibon |
248f5e |
comment = form.comment.data
|
|
Pierre-Yves Chibon |
248f5e |
|
|
Pierre-Yves Chibon |
248f5e |
try:
|
|
Pierre-Yves Chibon |
248f5e |
message = progit.lib.add_pull_request_comment(
|
|
Pierre-Yves Chibon |
248f5e |
SESSION,
|
|
Pierre-Yves Chibon |
248f5e |
request=request,
|
|
Pierre-Yves Chibon |
248f5e |
commit=commit,
|
|
Pierre-Yves Chibon |
abdb8f |
filename=filename,
|
|
Pierre-Yves Chibon |
248f5e |
row=row,
|
|
Pierre-Yves Chibon |
248f5e |
comment=comment,
|
|
Pierre-Yves Chibon |
248f5e |
user=flask.g.fas_user.username,
|
|
Pierre-Yves Chibon |
248f5e |
)
|
|
Pierre-Yves Chibon |
248f5e |
SESSION.commit()
|
|
Pierre-Yves Chibon |
248f5e |
flask.flash(message)
|
|
Pierre-Yves Chibon |
248f5e |
except SQLAlchemyError, err: # pragma: no cover
|
|
Pierre-Yves Chibon |
248f5e |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
1a6537 |
APP.logger.exception(err)
|
|
Pierre-Yves Chibon |
248f5e |
flask.flash(str(err), 'error')
|
|
Pierre-Yves Chibon |
248f5e |
|
|
Pierre-Yves Chibon |
248f5e |
return flask.redirect(flask.url_for(
|
|
Pierre-Yves Chibon |
248f5e |
'request_pull', username=username,
|
|
Pierre-Yves Chibon |
248f5e |
repo=repo.name, requestid=requestid))
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
cb10a1 |
return flask.render_template(
|
|
Pierre-Yves Chibon |
cb10a1 |
'pull_request_comment.html',
|
|
Pierre-Yves Chibon |
cb10a1 |
select='requests',
|
|
Pierre-Yves Chibon |
cb10a1 |
requestid=requestid,
|
|
Pierre-Yves Chibon |
cb10a1 |
repo=repo,
|
|
Pierre-Yves Chibon |
cb10a1 |
username=username,
|
|
Pierre-Yves Chibon |
cb10a1 |
commit=commit,
|
|
Pierre-Yves Chibon |
abdb8f |
filename=filename,
|
|
Pierre-Yves Chibon |
cb10a1 |
row=row,
|
|
Pierre-Yves Chibon |
cb10a1 |
form=form,
|
|
Pierre-Yves Chibon |
cb10a1 |
)
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
cb10a1 |
|
|
Pierre-Yves Chibon |
57116a |
@APP.route('/<repo>/request-pull/<int:requestid>/merge', methods=['POST'])</int:requestid></repo>
|
|
Pierre-Yves Chibon |
57116a |
@APP.route('/fork/<username>/<repo>/request-pull/<int:requestid>/merge',</int:requestid></repo></username>
|
|
Pierre-Yves Chibon |
0a1731 |
methods=['POST'])
|
|
Pierre-Yves Chibon |
b31474 |
def merge_request_pull(repo, requestid, username=None):
|
|
Pierre-Yves Chibon |
b31474 |
""" Request pulling the changes from the fork into the project.
|
|
Pierre-Yves Chibon |
47950c |
"""
|
|
Johan Cwiklinski |
fd18bd |
|
|
Johan Cwiklinski |
fd18bd |
form = progit.forms.ConfirmationForm()
|
|
Pierre-Yves Chibon |
15e950 |
if not form.validate_on_submit():
|
|
Pierre-Yves Chibon |
15e950 |
flask.flash('Invalid input submitted', 'error')
|
|
Pierre-Yves Chibon |
15e950 |
return flask.redirect(flask.url_for('view_repo', repo=repo.name))
|
|
Johan Cwiklinski |
fd18bd |
|
|
Pierre-Yves Chibon |
47950c |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
if not repo:
|
|
Pierre-Yves Chibon |
47950c |
flask.abort(404, 'Project not found')
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
ff8c27 |
request = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
b31474 |
SESSION, project_id=repo.id, requestid=requestid)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
if not request:
|
|
Pierre-Yves Chibon |
b31474 |
flask.abort(404, 'Pull-request not found')
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
4f2df5 |
if not is_repo_admin(repo):
|
|
Pierre-Yves Chibon |
4f2df5 |
flask.abort(
|
|
Pierre-Yves Chibon |
4f2df5 |
403,
|
|
Pierre-Yves Chibon |
4f2df5 |
'You are not allowed to merge pull-request for this project')
|
|
Pierre-Yves Chibon |
4f2df5 |
|
|
Pierre-Yves Chibon |
b31474 |
error_output = flask.url_for(
|
|
Pierre-Yves Chibon |
b31474 |
'request_pull', repo=repo.name, requestid=requestid)
|
|
Pierre-Yves Chibon |
b31474 |
if username:
|
|
Pierre-Yves Chibon |
b31474 |
error_output = flask.url_for(
|
|
Pierre-Yves Chibon |
b31474 |
'fork_request_pull',
|
|
Pierre-Yves Chibon |
b31474 |
repo=repo.name,
|
|
Pierre-Yves Chibon |
b31474 |
requestid=requestid,
|
|
Pierre-Yves Chibon |
b31474 |
username=username)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
# Get the fork
|
|
Pierre-Yves Chibon |
2c9e9f |
if request.repo_from.is_fork:
|
|
Pierre-Yves Chibon |
2c9e9f |
repopath = os.path.join(
|
|
Pierre-Yves Chibon |
2c9e9f |
APP.config['FORK_FOLDER'], request.repo_from.path)
|
|
Pierre-Yves Chibon |
2c9e9f |
else:
|
|
Pierre-Yves Chibon |
2c9e9f |
repopath = os.path.join(
|
|
Pierre-Yves Chibon |
2c9e9f |
APP.config['GIT_FOLDER'], request.repo_from.path)
|
|
Pierre-Yves Chibon |
b31474 |
fork_obj = pygit2.Repository(repopath)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
# Get the original repo
|
|
Pierre-Yves Chibon |
b31474 |
parentpath = os.path.join(APP.config['GIT_FOLDER'], request.repo.path)
|
|
Pierre-Yves Chibon |
b31474 |
orig_repo = pygit2.Repository(parentpath)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
# Clone the original repo into a temp folder
|
|
Pierre-Yves Chibon |
b31474 |
newpath = tempfile.mkdtemp()
|
|
Pierre-Yves Chibon |
b31474 |
new_repo = pygit2.clone_repository(parentpath, newpath)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
a2dd31 |
repo_commit = fork_obj[
|
|
Pierre-Yves Chibon |
a2dd31 |
fork_obj.lookup_branch(request.branch_from).get_object().hex]
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
ori_remote = new_repo.remotes[0]
|
|
Pierre-Yves Chibon |
b31474 |
# Add the fork as remote repo
|
|
Pierre-Yves Chibon |
e5d251 |
reponame = '%s_%s' % (request.user.user, repo.name)
|
|
Pierre-Yves Chibon |
b31474 |
remote = new_repo.create_remote(reponame, repopath)
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
b31474 |
# Fetch the commits
|
|
Pierre-Yves Chibon |
b31474 |
remote.fetch()
|
|
Pierre-Yves Chibon |
b31474 |
|
|
Pierre-Yves Chibon |
b31474 |
merge = new_repo.merge(repo_commit.oid)
|
|
Pierre-Yves Chibon |
a2dd31 |
if merge is None:
|
|
Pierre-Yves Chibon |
a2dd31 |
mergecode, prefcode = new_repo.merge_analysis(repo_commit.oid)
|
|
Pierre-Yves Chibon |
a2dd31 |
|
|
Pierre-Yves Chibon |
a2dd31 |
try:
|
|
Pierre-Yves Chibon |
a2dd31 |
branch_ref = new_repo.lookup_reference(
|
|
Pierre-Yves Chibon |
a2dd31 |
request.branch).resolve()
|
|
Pierre-Yves Chibon |
a2dd31 |
except ValueError:
|
|
Pierre-Yves Chibon |
a2dd31 |
branch_ref = new_repo.lookup_reference(
|
|
Pierre-Yves Chibon |
a2dd31 |
'refs/heads/%s' % request.branch).resolve()
|
|
Pierre-Yves Chibon |
a2dd31 |
|
|
Pierre-Yves Chibon |
a2dd31 |
refname = '%s:%s' % (branch_ref.name, branch_ref.name)
|
|
Pierre-Yves Chibon |
e04c77 |
if (
|
|
Pierre-Yves Chibon |
e04c77 |
(merge is not None and merge.is_uptodate)
|
|
Pierre-Yves Chibon |
a2dd31 |
or
|
|
Pierre-Yves Chibon |
e04c77 |
(merge is None and
|
|
Pierre-Yves Chibon |
e04c77 |
mergecode & pygit2.GIT_MERGE_ANALYSIS_UP_TO_DATE
|
|
Pierre-Yves Chibon |
e04c77 |
)):
|
|
Pierre-Yves Chibon |
d5c14e |
flask.flash('Nothing to do, changes were already merged', 'error')
|
|
Pierre-Yves Chibon |
d5c14e |
progit.lib.close_pull_request(SESSION, request)
|
|
Pierre-Yves Chibon |
1a6537 |
try:
|
|
Pierre-Yves Chibon |
1a6537 |
SESSION.commit()
|
|
Pierre-Yves Chibon |
1a6537 |
except SQLAlchemyError as err:
|
|
Pierre-Yves Chibon |
1a6537 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
1a6537 |
APP.logger.exception(err)
|
|
Pierre-Yves Chibon |
1a6537 |
flask.flash('Could not close this pull-request', 'error')
|
|
Pierre-Yves Chibon |
d5c14e |
return flask.redirect(error_output)
|
|
Pierre-Yves Chibon |
e04c77 |
elif (
|
|
Pierre-Yves Chibon |
e04c77 |
(merge is not None and merge.is_fastforward)
|
|
Pierre-Yves Chibon |
e04c77 |
or
|
|
Pierre-Yves Chibon |
e04c77 |
(merge is None and
|
|
Pierre-Yves Chibon |
e04c77 |
mergecode & pygit2.GIT_MERGE_ANALYSIS_FASTFORWARD
|
|
Pierre-Yves Chibon |
e04c77 |
)):
|
|
Pierre-Yves Chibon |
a2dd31 |
if merge is not None:
|
|
Pierre-Yves Chibon |
a2dd31 |
branch_ref.target = merge.fastforward_oid
|
|
Pierre-Yves Chibon |
a2dd31 |
sha = merge.fastforward_oid
|
|
Pierre-Yves Chibon |
a2dd31 |
elif merge is None and mergecode is not None:
|
|
Pierre-Yves Chibon |
a2dd31 |
print repo_commit.oid
|
|
Pierre-Yves Chibon |
a2dd31 |
branch_ref.set_target(repo_commit.oid.hex)
|
|
Pierre-Yves Chibon |
a2dd31 |
sha = branch_ref.target
|
|
Pierre-Yves Chibon |
b31474 |
ori_remote.push(refname)
|
|
Pierre-Yves Chibon |
b31474 |
flask.flash('Changes merged!')
|
|
Pierre-Yves Chibon |
a2dd31 |
|
|
Pierre-Yves Chibon |
b31474 |
else:
|
|
Pierre-Yves Chibon |
59b3a3 |
new_repo.index.write()
|
|
Pierre-Yves Chibon |
2c9e9f |
try:
|
|
Pierre-Yves Chibon |
2c9e9f |
tree = new_repo.index.write_tree()
|
|
Pierre-Yves Chibon |
2c9e9f |
except pygit2.GitError:
|
|
Pierre-Yves Chibon |
be7090 |
shutil.rmtree(newpath)
|
|
Pierre-Yves Chibon |
2c9e9f |
flask.flash('Merge conflicts!', 'error')
|
|
Pierre-Yves Chibon |
2c9e9f |
return flask.redirect(flask.url_for(
|
|
Pierre-Yves Chibon |
2c9e9f |
'request_pull',
|
|
Pierre-Yves Chibon |
2c9e9f |
repo=repo.name,
|
|
Pierre-Yves Chibon |
2c9e9f |
username=username,
|
|
Pierre-Yves Chibon |
2c9e9f |
requestid=requestid))
|
|
Pierre-Yves Chibon |
59b3a3 |
head = new_repo.lookup_reference('HEAD').get_object()
|
|
Pierre-Yves Chibon |
59b3a3 |
commit = new_repo[head.oid]
|
|
Pierre-Yves Chibon |
59b3a3 |
sha = new_repo.create_commit(
|
|
Pierre-Yves Chibon |
59b3a3 |
'refs/heads/master',
|
|
Pierre-Yves Chibon |
59b3a3 |
repo_commit.author,
|
|
Pierre-Yves Chibon |
59b3a3 |
repo_commit.committer,
|
|
Pierre-Yves Chibon |
59b3a3 |
'Merge #%s `%s`' % (request.id, request.title),
|
|
Pierre-Yves Chibon |
59b3a3 |
tree,
|
|
Pierre-Yves Chibon |
59b3a3 |
[head.hex, repo_commit.oid.hex])
|
|
Pierre-Yves Chibon |
59b3a3 |
ori_remote.push(refname)
|
|
Pierre-Yves Chibon |
59b3a3 |
flask.flash('Changes merged!')
|
|
Pierre-Yves Chibon |
b31474 |
|
|
Pierre-Yves Chibon |
b31474 |
# Update status
|
|
Pierre-Yves Chibon |
f7606b |
progit.lib.close_pull_request(SESSION, request, flask.g.fas_user)
|
|
Pierre-Yves Chibon |
1a6537 |
try:
|
|
Pierre-Yves Chibon |
1a6537 |
SESSION.commit()
|
|
Pierre-Yves Chibon |
1a6537 |
except SQLAlchemyError as err:
|
|
Pierre-Yves Chibon |
1a6537 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
1a6537 |
APP.logger.exception(err)
|
|
Pierre-Yves Chibon |
1a6537 |
flask.flash(
|
|
Pierre-Yves Chibon |
1a6537 |
'Could not update this pull-request in the database',
|
|
Pierre-Yves Chibon |
1a6537 |
'error')
|
|
Pierre-Yves Chibon |
be7090 |
shutil.rmtree(newpath)
|
|
Pierre-Yves Chibon |
b31474 |
|
|
Pierre-Yves Chibon |
b31474 |
return flask.redirect(flask.url_for('view_repo', repo=repo.name))
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/<repo>/request-pull/cancel/<int:requestid>',</int:requestid></repo>
|
|
Pierre-Yves Chibon |
e04c77 |
methods=['POST'])
|
|
Pierre-Yves Chibon |
c33710 |
@APP.route('/fork/<username>/<repo>/request-pull/cancel/<int:requestid>',</int:requestid></repo></username>
|
|
Pierre-Yves Chibon |
e04c77 |
methods=['POST'])
|
|
Johan Cwiklinski |
86d9c4 |
def cancel_request_pull(repo, requestid, username=None):
|
|
Johan Cwiklinski |
86d9c4 |
""" Cancel request pulling request.
|
|
Johan Cwiklinski |
86d9c4 |
"""
|
|
Johan Cwiklinski |
fd18bd |
|
|
Johan Cwiklinski |
fd18bd |
form = progit.forms.ConfirmationForm()
|
|
Pierre-Yves Chibon |
15e950 |
if form.validate_on_submit():
|
|
Johan Cwiklinski |
fd18bd |
|
|
Pierre-Yves Chibon |
15e950 |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Pierre-Yves Chibon |
15e950 |
if not repo:
|
|
Pierre-Yves Chibon |
15e950 |
flask.abort(404, 'Project not found')
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Pierre-Yves Chibon |
ff8c27 |
request = progit.lib.search_pull_requests(
|
|
Pierre-Yves Chibon |
15e950 |
SESSION, project_id=repo.id, requestid=requestid)
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Pierre-Yves Chibon |
15e950 |
if not request:
|
|
Pierre-Yves Chibon |
15e950 |
flask.abort(404, 'Pull-request not found')
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Pierre-Yves Chibon |
15e950 |
if not is_repo_admin(repo):
|
|
Pierre-Yves Chibon |
15e950 |
flask.abort(
|
|
Pierre-Yves Chibon |
15e950 |
403,
|
|
Pierre-Yves Chibon |
15e950 |
'You are not allowed to cancel pull-request for this project')
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Pierre-Yves Chibon |
f7606b |
progit.lib.close_pull_request(
|
|
Pierre-Yves Chibon |
f7606b |
SESSION, request, flask.g.fas_user, merged=False)
|
|
Pierre-Yves Chibon |
d8fde8 |
try:
|
|
Pierre-Yves Chibon |
d8fde8 |
SESSION.commit()
|
|
Pierre-Yves Chibon |
d8fde8 |
flask.flash('Request pull canceled!')
|
|
Pierre-Yves Chibon |
d8fde8 |
except SQLAlchemyError as err:
|
|
Pierre-Yves Chibon |
d8fde8 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
d8fde8 |
APP.logger.exception(err)
|
|
Pierre-Yves Chibon |
d8fde8 |
flask.flash(
|
|
Pierre-Yves Chibon |
d8fde8 |
'Could not update this pull-request in the database',
|
|
Pierre-Yves Chibon |
d8fde8 |
'error')
|
|
Pierre-Yves Chibon |
15e950 |
else:
|
|
Pierre-Yves Chibon |
15e950 |
flask.flash('Invalid input submitted', 'error')
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Johan Cwiklinski |
86d9c4 |
return flask.redirect(flask.url_for('view_repo', repo=repo.name))
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Johan Cwiklinski |
86d9c4 |
|
|
Pierre-Yves Chibon |
e04c77 |
# Specific actions
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
47950c |
|
|
Pierre-Yves Chibon |
e03e84 |
@APP.route('/do_fork/<repo>')</repo>
|
|
Pierre-Yves Chibon |
e03e84 |
@APP.route('/do_fork/<username>/<repo>')</repo></username>
|
|
Pierre-Yves Chibon |
d5e3d0 |
@cla_required
|
|
Pierre-Yves Chibon |
20ccbb |
def fork_project(repo, username=None):
|
|
Pierre-Yves Chibon |
fac0b1 |
""" Fork the project specified into the user's namespace
|
|
Pierre-Yves Chibon |
fac0b1 |
"""
|
|
Pierre-Yves Chibon |
20ccbb |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
80bba3 |
|
|
Pierre-Yves Chibon |
80bba3 |
if repo is None:
|
|
Pierre-Yves Chibon |
fac0b1 |
flask.abort(404)
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
80bba3 |
try:
|
|
Pierre-Yves Chibon |
80bba3 |
message = progit.lib.fork_project(
|
|
Pierre-Yves Chibon |
80bba3 |
session=SESSION,
|
|
Pierre-Yves Chibon |
80bba3 |
repo=repo,
|
|
Pierre-Yves Chibon |
e54062 |
gitfolder=APP.config['GIT_FOLDER'],
|
|
Pierre-Yves Chibon |
6160b6 |
forkfolder=APP.config['FORK_FOLDER'],
|
|
Pierre-Yves Chibon |
4b7a7d |
docfolder=APP.config['DOCS_FOLDER'],
|
|
Pierre-Yves Chibon |
a33978 |
ticketfolder=APP.config['TICKETS_FOLDER'],
|
|
Pierre-Yves Chibon |
80bba3 |
user=flask.g.fas_user.username)
|
|
Pierre-Yves Chibon |
80bba3 |
|
|
Pierre-Yves Chibon |
80bba3 |
SESSION.commit()
|
|
Pierre-Yves Chibon |
59be40 |
generate_gitolite_acls()
|
|
Pierre-Yves Chibon |
80bba3 |
flask.flash(message)
|
|
Pierre-Yves Chibon |
80bba3 |
return flask.redirect(
|
|
Pierre-Yves Chibon |
792a86 |
flask.url_for(
|
|
Pierre-Yves Chibon |
ea8735 |
'view_repo',
|
|
Pierre-Yves Chibon |
792a86 |
username=flask.g.fas_user.username,
|
|
Pierre-Yves Chibon |
792a86 |
repo=repo.name)
|
|
Pierre-Yves Chibon |
80bba3 |
)
|
|
Pierre-Yves Chibon |
80bba3 |
except progit.exceptions.ProgitException, err:
|
|
Pierre-Yves Chibon |
80bba3 |
flask.flash(str(err), 'error')
|
|
Pierre-Yves Chibon |
80bba3 |
except SQLAlchemyError, err: # pragma: no cover
|
|
Pierre-Yves Chibon |
80bba3 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
80bba3 |
flask.flash(str(err), 'error')
|
|
Pierre-Yves Chibon |
80bba3 |
|
|
Pierre-Yves Chibon |
792a86 |
return flask.redirect(flask.url_for('view_repo', repo=repo.name))
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
fac0b1 |
|
|
Pierre-Yves Chibon |
6d60d2 |
@APP.route('/<repo>/diff/<branch_to>..<branch_from>',</branch_from></branch_to></repo>
|
|
Pierre-Yves Chibon |
2c9e9f |
methods=('GET', 'POST'))
|
|
Pierre-Yves Chibon |
6d60d2 |
@APP.route('/fork/<username>/<repo>/diff/<branch_to>..<branch_from>',</branch_from></branch_to></repo></username>
|
|
Pierre-Yves Chibon |
ac8023 |
methods=('GET', 'POST'))
|
|
Pierre-Yves Chibon |
d5e3d0 |
@cla_required
|
|
Pierre-Yves Chibon |
6d60d2 |
def new_request_pull(repo, branch_to, branch_from, username=None):
|
|
Pierre-Yves Chibon |
82055c |
""" Request pulling the changes from the fork into the project.
|
|
Pierre-Yves Chibon |
82055c |
"""
|
|
Pierre-Yves Chibon |
82055c |
repo = progit.lib.get_project(SESSION, repo, user=username)
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
82055c |
if not repo:
|
|
Pierre-Yves Chibon |
82055c |
flask.abort(404)
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
1e5fe9 |
if not is_repo_admin(repo):
|
|
Pierre-Yves Chibon |
1e5fe9 |
flask.abort(
|
|
Pierre-Yves Chibon |
1e5fe9 |
403,
|
|
Pierre-Yves Chibon |
1e5fe9 |
'You are not allowed to create pull-requests for this project')
|
|
Pierre-Yves Chibon |
1e5fe9 |
|
|
Pierre-Yves Chibon |
8df273 |
repopath = _get_repo_path(repo)
|
|
Pierre-Yves Chibon |
85755c |
repo_obj = pygit2.Repository(repopath)
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
8df273 |
parentpath = _get_parent_repo_path(repo)
|
|
Pierre-Yves Chibon |
8df273 |
orig_repo = pygit2.Repository(parentpath)
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
6d60d2 |
frombranch = repo_obj.lookup_branch(branch_from)
|
|
Pierre-Yves Chibon |
24542f |
if not frombranch:
|
|
Pierre-Yves Chibon |
6d60d2 |
flask.abort(
|
|
Pierre-Yves Chibon |
6d60d2 |
400,
|
|
Pierre-Yves Chibon |
6d60d2 |
'Branch %s does not exist' % branch_from)
|
|
Pierre-Yves Chibon |
24542f |
|
|
Pierre-Yves Chibon |
6d60d2 |
branch = orig_repo.lookup_branch(branch_to)
|
|
Pierre-Yves Chibon |
efda13 |
if not branch:
|
|
Pierre-Yves Chibon |
6d60d2 |
flask.abort(
|
|
Pierre-Yves Chibon |
6d60d2 |
400,
|
|
Pierre-Yves Chibon |
6d60d2 |
'Branch %s could not be found in the target repo' % branch_to)
|
|
Pierre-Yves Chibon |
efda13 |
|
|
Pierre-Yves Chibon |
6d60d2 |
branch = repo_obj.lookup_branch(branch_from)
|
|
Pierre-Yves Chibon |
6d60d2 |
commitid = branch.get_object().hex
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
82055c |
diff_commits = []
|
|
Pierre-Yves Chibon |
82055c |
if not repo_obj.is_empty and not orig_repo.is_empty:
|
|
Pierre-Yves Chibon |
2c9e9f |
orig_commit = orig_repo[
|
|
Pierre-Yves Chibon |
6d60d2 |
orig_repo.lookup_branch(branch_to).get_object().hex]
|
|
Pierre-Yves Chibon |
2c9e9f |
|
|
Pierre-Yves Chibon |
2c9e9f |
master_commits = [
|
|
Pierre-Yves Chibon |
2c9e9f |
commit.oid.hex
|
|
Pierre-Yves Chibon |
2c9e9f |
for commit in orig_repo.walk(
|
|
Pierre-Yves Chibon |
6d60d2 |
orig_commit.oid.hex, pygit2.GIT_SORT_TIME)
|
|
Pierre-Yves Chibon |
2c9e9f |
]
|
|
Pierre-Yves Chibon |
2c9e9f |
|
|
Pierre-Yves Chibon |
82055c |
repo_commit = repo_obj[commitid]
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
2c9e9f |
for commit in repo_obj.walk(
|
|
Pierre-Yves Chibon |
2c9e9f |
repo_commit.oid.hex, pygit2.GIT_SORT_TIME):
|
|
Pierre-Yves Chibon |
2c9e9f |
if commit.oid.hex in master_commits:
|
|
Pierre-Yves Chibon |
82055c |
break
|
|
Pierre-Yves Chibon |
82055c |
diff_commits.append(commit)
|
|
Pierre-Yves Chibon |
6d60d2 |
|
|
Pierre-Yves Chibon |
6d60d2 |
first_commit = repo_obj[diff_commits[-1].oid.hex]
|
|
Pierre-Yves Chibon |
6d60d2 |
diff = repo_obj.diff(
|
|
Pierre-Yves Chibon |
6d60d2 |
repo_obj.revparse_single(first_commit.parents[0].oid.hex),
|
|
Pierre-Yves Chibon |
6d60d2 |
repo_obj.revparse_single(diff_commits[0].oid.hex)
|
|
Pierre-Yves Chibon |
6d60d2 |
)
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
82055c |
elif orig_repo.is_empty:
|
|
Pierre-Yves Chibon |
ac8023 |
orig_commit = None
|
|
Pierre-Yves Chibon |
85755c |
repo_commit = repo_obj[repo_obj.head.target]
|
|
Pierre-Yves Chibon |
85755c |
diff = repo_commit.tree.diff_to_tree(swap=True)
|
|
Pierre-Yves Chibon |
82055c |
else:
|
|
Pierre-Yves Chibon |
82055c |
flask.flash(
|
|
Pierre-Yves Chibon |
82055c |
'Fork is empty, there are no commits to request pulling',
|
|
Pierre-Yves Chibon |
82055c |
'error')
|
|
Pierre-Yves Chibon |
82055c |
return flask.redirect(flask.url_for(
|
|
Pierre-Yves Chibon |
ea8735 |
'view_repo', username=username, repo=repo.name))
|
|
Pierre-Yves Chibon |
82055c |
|
|
Pierre-Yves Chibon |
ac8023 |
form = progit.forms.RequestPullForm()
|
|
Pierre-Yves Chibon |
ac8023 |
if form.validate_on_submit():
|
|
Pierre-Yves Chibon |
ac8023 |
try:
|
|
Pierre-Yves Chibon |
ac8023 |
if orig_commit:
|
|
Pierre-Yves Chibon |
ac8023 |
orig_commit = orig_commit.oid.hex
|
|
Pierre-Yves Chibon |
2c9e9f |
|
|
Pierre-Yves Chibon |
2c9e9f |
parent = repo
|
|
Pierre-Yves Chibon |
2c9e9f |
if repo.parent:
|
|
Pierre-Yves Chibon |
2c9e9f |
parent = repo.parent
|
|
Pierre-Yves Chibon |
2c9e9f |
|
|
Pierre-Yves Chibon |
ac8023 |
message = progit.lib.new_pull_request(
|
|
Pierre-Yves Chibon |
ac8023 |
SESSION,
|
|
Pierre-Yves Chibon |
6d60d2 |
repo_to=parent,
|
|
Pierre-Yves Chibon |
6d60d2 |
branch_to=branch_to,
|
|
Pierre-Yves Chibon |
6d60d2 |
branch_from=branch_from,
|
|
Pierre-Yves Chibon |
ac8023 |
repo_from=repo,
|
|
Pierre-Yves Chibon |
ac8023 |
title=form.title.data,
|
|
Pierre-Yves Chibon |
ac8023 |
user=flask.g.fas_user.username,
|
|
Pierre-Yves Chibon |
ac8023 |
)
|
|
Pierre-Yves Chibon |
d8fde8 |
try:
|
|
Pierre-Yves Chibon |
d8fde8 |
SESSION.commit()
|
|
Pierre-Yves Chibon |
d8fde8 |
flask.flash(message)
|
|
Pierre-Yves Chibon |
d8fde8 |
except SQLAlchemyError as err:
|
|
Pierre-Yves Chibon |
d8fde8 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
d8fde8 |
APP.logger.exception(err)
|
|
Pierre-Yves Chibon |
d8fde8 |
flask.flash(
|
|
Pierre-Yves Chibon |
d8fde8 |
'Could not register this pull-request in the database',
|
|
Pierre-Yves Chibon |
d8fde8 |
'error')
|
|
Pierre-Yves Chibon |
8399d1 |
|
|
Pierre-Yves Chibon |
2c9e9f |
if not parent.is_fork:
|
|
Pierre-Yves Chibon |
8399d1 |
url = flask.url_for(
|
|
Pierre-Yves Chibon |
2c9e9f |
'request_pulls', username=None, repo=parent.name)
|
|
Pierre-Yves Chibon |
8399d1 |
else:
|
|
Pierre-Yves Chibon |
8399d1 |
url = flask.url_for(
|
|
Pierre-Yves Chibon |
2c9e9f |
'request_pulls', username=parent.user, repo=parent.name)
|
|
Pierre-Yves Chibon |
8399d1 |
|
|
Pierre-Yves Chibon |
8399d1 |
return flask.redirect(url)
|
|
Pierre-Yves Chibon |
ac8023 |
except progit.exceptions.ProgitException, err:
|
|
Pierre-Yves Chibon |
ac8023 |
flask.flash(str(err), 'error')
|
|
Pierre-Yves Chibon |
ac8023 |
except SQLAlchemyError, err: # pragma: no cover
|
|
Pierre-Yves Chibon |
ac8023 |
SESSION.rollback()
|
|
Pierre-Yves Chibon |
ac8023 |
flask.flash(str(err), 'error')
|
|
Pierre-Yves Chibon |
ac8023 |
|
|
Pierre-Yves Chibon |
82055c |
return flask.render_template(
|
|
Pierre-Yves Chibon |
82055c |
'pull_request.html',
|
|
Pierre-Yves Chibon |
0c4c0b |
select='requests',
|
|
Pierre-Yves Chibon |
82055c |
repo=repo,
|
|
Pierre-Yves Chibon |
82055c |
username=username,
|
|
Pierre-Yves Chibon |
82055c |
repo_obj=repo_obj,
|
|
Pierre-Yves Chibon |
82055c |
orig_repo=orig_repo,
|
|
Pierre-Yves Chibon |
82055c |
diff_commits=diff_commits,
|
|
Pierre-Yves Chibon |
6d60d2 |
diff=diff,
|
|
Pierre-Yves Chibon |
ac8023 |
form=form,
|
|
Pierre-Yves Chibon |
658f7c |
branches=[
|
|
Pierre-Yves Chibon |
658f7c |
branch.replace('refs/heads/', '')
|
|
Pierre-Yves Chibon |
658f7c |
for branch in sorted(orig_repo.listall_references())
|
|
Pierre-Yves Chibon |
658f7c |
],
|
|
Pierre-Yves Chibon |
6d60d2 |
branch_to=branch_to,
|
|
Pierre-Yves Chibon |
6d60d2 |
branch_from=branch_from,
|
|
Pierre-Yves Chibon |
82055c |
)
|