|
Pierre-Yves Chibon |
1e2e25 |
#-*- coding: utf-8 -*-
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
"""
|
|
Pierre-Yves Chibon |
1e2e25 |
(c) 2014 - Copyright Red Hat Inc
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
Authors:
|
|
Pierre-Yves Chibon |
1e2e25 |
Pierre-Yves Chibon <pingou@pingoured.fr></pingou@pingoured.fr>
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
"""
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
9af5a0 |
import datetime
|
|
Pierre-Yves Chibon |
f7d963 |
import json
|
|
Pierre-Yves Chibon |
d8110b |
import os
|
|
Pierre-Yves Chibon |
e2a95c |
import random
|
|
Pierre-Yves Chibon |
d133cd |
import shutil
|
|
Pierre-Yves Chibon |
e2a95c |
import string
|
|
Pierre-Yves Chibon |
d133cd |
import tempfile
|
|
Pierre-Yves Chibon |
9cb884 |
import uuid
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
1e2e25 |
import sqlalchemy
|
|
Pierre-Yves Chibon |
1e2e25 |
from datetime import timedelta
|
|
Pierre-Yves Chibon |
1e2e25 |
from sqlalchemy.orm import sessionmaker
|
|
Pierre-Yves Chibon |
1e2e25 |
from sqlalchemy.orm import scoped_session
|
|
Pierre-Yves Chibon |
1e2e25 |
from sqlalchemy.orm.exc import NoResultFound
|
|
Pierre-Yves Chibon |
1e2e25 |
from sqlalchemy.exc import SQLAlchemyError
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
d8110b |
import pygit2
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
d8110b |
import progit.exceptions
|
|
Pierre-Yves Chibon |
4b55b4 |
import progit.notify
|
|
Pierre-Yves Chibon |
1e2e25 |
from progit import model
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
def create_session(db_url, debug=False, pool_recycle=3600):
|
|
Pierre-Yves Chibon |
695ef6 |
''' Create the Session object to use to query the database.
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
:arg db_url: URL used to connect to the database. The URL contains
|
|
Pierre-Yves Chibon |
1e2e25 |
information with regards to the database engine, the host to connect
|
|
Pierre-Yves Chibon |
1e2e25 |
to, the user and password and the database name.
|
|
Pierre-Yves Chibon |
1e2e25 |
ie: <engine>://<user>:<password>@<host>/<dbname></dbname></host></password></user></engine>
|
|
Pierre-Yves Chibon |
1e2e25 |
:kwarg debug: a boolean specifying wether we should have the verbose
|
|
Pierre-Yves Chibon |
1e2e25 |
output of sqlalchemy or not.
|
|
Pierre-Yves Chibon |
1e2e25 |
:return a Session that can be used to query the database.
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
695ef6 |
'''
|
|
Pierre-Yves Chibon |
1e2e25 |
engine = sqlalchemy.create_engine(
|
|
Pierre-Yves Chibon |
1e2e25 |
db_url, echo=debug, pool_recycle=pool_recycle)
|
|
Pierre-Yves Chibon |
1e2e25 |
scopedsession = scoped_session(sessionmaker(bind=engine))
|
|
Pierre-Yves Chibon |
1e2e25 |
return scopedsession
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
e2a95c |
def id_generator(size=15, chars=string.ascii_uppercase + string.digits):
|
|
Pierre-Yves Chibon |
e2a95c |
""" Generates a random identifier for the given size and using the
|
|
Pierre-Yves Chibon |
e2a95c |
specified characters.
|
|
Pierre-Yves Chibon |
e2a95c |
If no size is specified, it uses 15 as default.
|
|
Pierre-Yves Chibon |
e2a95c |
If no characters are specified, it uses ascii char upper case and
|
|
Pierre-Yves Chibon |
e2a95c |
digits.
|
|
Pierre-Yves Chibon |
e2a95c |
:arg size: the size of the identifier to return.
|
|
Pierre-Yves Chibon |
e2a95c |
:arg chars: the list of characters that can be used in the
|
|
Pierre-Yves Chibon |
e2a95c |
idenfitier.
|
|
Pierre-Yves Chibon |
e2a95c |
"""
|
|
Pierre-Yves Chibon |
e2a95c |
return ''.join(random.choice(chars) for x in range(size))
|
|
Pierre-Yves Chibon |
e2a95c |
|
|
Pierre-Yves Chibon |
e2a95c |
|
|
Pierre-Yves Chibon |
f0256a |
def commit_to_patch(repo_obj, commits):
|
|
Pierre-Yves Chibon |
9af5a0 |
''' For a given commit (PyGit2 commit object) of a specified git repo,
|
|
Pierre-Yves Chibon |
9af5a0 |
returns a string representation of the changes the commit did in a
|
|
Pierre-Yves Chibon |
9af5a0 |
format that allows it to be used as patch.
|
|
Pierre-Yves Chibon |
9af5a0 |
'''
|
|
Pierre-Yves Chibon |
f0256a |
if not isinstance(commits, list):
|
|
Pierre-Yves Chibon |
f0256a |
commits = [commits]
|
|
Pierre-Yves Chibon |
9af5a0 |
|
|
Pierre-Yves Chibon |
f0256a |
patch = ""
|
|
Pierre-Yves Chibon |
f0256a |
for cnt, commit in enumerate(commits):
|
|
Pierre-Yves Chibon |
f0256a |
if commit.parents:
|
|
Pierre-Yves Chibon |
f0256a |
diff = commit.tree.diff_to_tree()
|
|
Pierre-Yves Chibon |
9af5a0 |
|
|
Pierre-Yves Chibon |
f0256a |
parent = repo_obj.revparse_single('%s^' % commit.oid.hex)
|
|
Pierre-Yves Chibon |
f0256a |
diff = repo_obj.diff(parent, commit)
|
|
Pierre-Yves Chibon |
f0256a |
else:
|
|
Pierre-Yves Chibon |
f0256a |
# First commit in the repo
|
|
Pierre-Yves Chibon |
f0256a |
diff = commit.tree.diff_to_tree(swap=True)
|
|
Pierre-Yves Chibon |
f0256a |
|
|
Pierre-Yves Chibon |
b93113 |
subject = message = ''
|
|
Pierre-Yves Chibon |
b93113 |
if '\n' in commit.message:
|
|
Pierre-Yves Chibon |
b93113 |
subject, message = commit.message.split('\n', 1)
|
|
Pierre-Yves Chibon |
b93113 |
else:
|
|
Pierre-Yves Chibon |
b93113 |
subject = commit.message
|
|
Pierre-Yves Chibon |
f0256a |
|
|
Pierre-Yves Chibon |
f0256a |
if len(commits) > 1:
|
|
Pierre-Yves Chibon |
f0256a |
subject = '[PATCH %s/%s] %s' % (cnt + 1, len(commits), subject)
|
|
Pierre-Yves Chibon |
f0256a |
|
|
Pierre-Yves Chibon |
f0256a |
patch += """From %(commit)s Mon Sep 17 00:00:00 2001
|
|
Pierre-Yves Chibon |
9af5a0 |
From: %(author_name)s <%(author_email)s>
|
|
Pierre-Yves Chibon |
9af5a0 |
Date: %(date)s
|
|
Pierre-Yves Chibon |
f0256a |
Subject: %(subject)s
|
|
Pierre-Yves Chibon |
9af5a0 |
|
|
Pierre-Yves Chibon |
9af5a0 |
%(msg)s
|
|
Pierre-Yves Chibon |
f0256a |
---
|
|
Pierre-Yves Chibon |
9af5a0 |
|
|
Pierre-Yves Chibon |
9af5a0 |
%(patch)s
|
|
Pierre-Yves Chibon |
9af5a0 |
""" % (
|
|
Pierre-Yves Chibon |
f0256a |
{
|
|
Pierre-Yves Chibon |
f0256a |
'commit': commit.oid.hex,
|
|
Pierre-Yves Chibon |
f0256a |
'author_name': commit.author.name,
|
|
Pierre-Yves Chibon |
f0256a |
'author_email': commit.author.email,
|
|
Pierre-Yves Chibon |
f0256a |
'date': datetime.datetime.utcfromtimestamp(
|
|
Pierre-Yves Chibon |
f0256a |
commit.commit_time).strftime('%b %d %Y %H:%M:%S +0000'),
|
|
Pierre-Yves Chibon |
f0256a |
'subject': subject,
|
|
Pierre-Yves Chibon |
f0256a |
'msg': message,
|
|
Pierre-Yves Chibon |
f0256a |
'patch': diff.patch,
|
|
Pierre-Yves Chibon |
f0256a |
}
|
|
Pierre-Yves Chibon |
f0256a |
)
|
|
Pierre-Yves Chibon |
9af5a0 |
return patch
|
|
Pierre-Yves Chibon |
9af5a0 |
|
|
Pierre-Yves Chibon |
9af5a0 |
|
|
Pierre-Yves Chibon |
03c11f |
def get_user(session, username):
|
|
Pierre-Yves Chibon |
03c11f |
''' Return the user corresponding to this username, or None. '''
|
|
Pierre-Yves Chibon |
03c11f |
user = session.query(
|
|
Pierre-Yves Chibon |
e1372f |
model.User
|
|
Pierre-Yves Chibon |
e1372f |
).filter(
|
|
Pierre-Yves Chibon |
03c11f |
model.User.user == username
|
|
Pierre-Yves Chibon |
e1372f |
).first()
|
|
Pierre-Yves Chibon |
03c11f |
return user
|
|
Pierre-Yves Chibon |
03c11f |
|
|
Pierre-Yves Chibon |
03c11f |
|
|
Pierre-Yves Chibon |
b81780 |
def get_user_by_email(session, user_mail):
|
|
Pierre-Yves Chibon |
b81780 |
''' Return the user corresponding to this email, or None. '''
|
|
Pierre-Yves Chibon |
b81780 |
mail = session.query(
|
|
Pierre-Yves Chibon |
b81780 |
model.UserEmail
|
|
Pierre-Yves Chibon |
b81780 |
).filter(
|
|
Pierre-Yves Chibon |
b81780 |
model.UserEmail.email == user_mail
|
|
Pierre-Yves Chibon |
b81780 |
).first()
|
|
Pierre-Yves Chibon |
b81780 |
if mail:
|
|
Pierre-Yves Chibon |
b81780 |
return mail.user
|
|
Pierre-Yves Chibon |
b81780 |
|
|
Pierre-Yves Chibon |
b81780 |
|
|
Pierre-Yves Chibon |
34cbca |
def get_user_by_token(session, token):
|
|
Pierre-Yves Chibon |
34cbca |
''' Return a specified User via its token.
|
|
Pierre-Yves Chibon |
34cbca |
|
|
Pierre-Yves Chibon |
34cbca |
:arg session: the session with which to connect to the database.
|
|
Pierre-Yves Chibon |
34cbca |
|
|
Pierre-Yves Chibon |
34cbca |
'''
|
|
Pierre-Yves Chibon |
34cbca |
user = session.query(
|
|
Pierre-Yves Chibon |
34cbca |
model.User
|
|
Pierre-Yves Chibon |
34cbca |
).filter(
|
|
Pierre-Yves Chibon |
34cbca |
model.User.token == token
|
|
Pierre-Yves Chibon |
34cbca |
).first()
|
|
Pierre-Yves Chibon |
34cbca |
return user
|
|
Pierre-Yves Chibon |
34cbca |
|
|
Pierre-Yves Chibon |
34cbca |
|
|
Pierre-Yves Chibon |
556d85 |
def get_all_users(session):
|
|
Pierre-Yves Chibon |
556d85 |
''' Return the user corresponding to this username, or None. '''
|
|
Pierre-Yves Chibon |
556d85 |
users = session.query(
|
|
Pierre-Yves Chibon |
556d85 |
model.User
|
|
Pierre-Yves Chibon |
613fab |
).order_by(
|
|
Pierre-Yves Chibon |
613fab |
model.User.user
|
|
Pierre-Yves Chibon |
556d85 |
).all()
|
|
Pierre-Yves Chibon |
556d85 |
return users
|
|
Pierre-Yves Chibon |
556d85 |
|
|
Pierre-Yves Chibon |
556d85 |
|
|
Pierre-Yves Chibon |
a34835 |
def add_issue_comment(session, issue, comment, user, ticketfolder):
|
|
Pierre-Yves Chibon |
03c11f |
''' Add a comment to an issue. '''
|
|
Pierre-Yves Chibon |
3ddae1 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
496a35 |
if not user_obj:
|
|
Pierre-Yves Chibon |
496a35 |
user_obj = get_user_by_email(session, user)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
e1372f |
if not user_obj:
|
|
Pierre-Yves Chibon |
e1372f |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
e1372f |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
e1372f |
)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
00c3f0 |
issue_comment = model.IssueComment(
|
|
Pierre-Yves Chibon |
359db9 |
issue_id=issue.id,
|
|
Pierre-Yves Chibon |
359db9 |
comment=comment,
|
|
Pierre-Yves Chibon |
e1372f |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
359db9 |
)
|
|
Pierre-Yves Chibon |
00c3f0 |
session.add(issue_comment)
|
|
Pierre-Yves Chibon |
359db9 |
# Make sure we won't have SQLAlchemy error before we create the repo
|
|
Pierre-Yves Chibon |
359db9 |
session.flush()
|
|
Pierre-Yves Chibon |
359db9 |
|
|
Pierre-Yves Chibon |
a34835 |
update_git_ticket(issue, repo=issue.project, ticketfolder=ticketfolder)
|
|
Pierre-Yves Chibon |
a34835 |
|
|
Pierre-Yves Chibon |
caaa01 |
globalid = get_issue_global_id(session, issue.project.id, issue.id)
|
|
Pierre-Yves Chibon |
caaa01 |
|
|
Pierre-Yves Chibon |
caaa01 |
progit.notify.notify_new_comment(issue_comment, globalid)
|
|
Pierre-Yves Chibon |
4b55b4 |
|
|
Pierre-Yves Chibon |
359db9 |
return 'Comment added'
|
|
Pierre-Yves Chibon |
359db9 |
|
|
Pierre-Yves Chibon |
359db9 |
|
|
Pierre-Yves Chibon |
aafc66 |
def add_issue_tag(session, issue, tag, user, ticketfolder):
|
|
Pierre-Yves Chibon |
aafc66 |
''' Add a tag to an issue. '''
|
|
Pierre-Yves Chibon |
aafc66 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
aafc66 |
if not user_obj:
|
|
Pierre-Yves Chibon |
aafc66 |
user_obj = get_user_by_email(session, user)
|
|
Pierre-Yves Chibon |
aafc66 |
|
|
Pierre-Yves Chibon |
aafc66 |
if not user_obj:
|
|
Pierre-Yves Chibon |
aafc66 |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
aafc66 |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
aafc66 |
)
|
|
Pierre-Yves Chibon |
aafc66 |
|
|
Pierre-Yves Chibon |
3df2cd |
issue_tag = model.TagIssue(
|
|
Pierre-Yves Chibon |
aafc66 |
issue_id=issue.id,
|
|
Pierre-Yves Chibon |
aafc66 |
tag=tag,
|
|
Pierre-Yves Chibon |
aafc66 |
)
|
|
Pierre-Yves Chibon |
aafc66 |
session.add(issue_tag)
|
|
Pierre-Yves Chibon |
aafc66 |
# Make sure we won't have SQLAlchemy error before we create the repo
|
|
Pierre-Yves Chibon |
aafc66 |
session.flush()
|
|
Pierre-Yves Chibon |
aafc66 |
|
|
Pierre-Yves Chibon |
aafc66 |
update_git_ticket(issue, repo=issue.project, ticketfolder=ticketfolder)
|
|
Pierre-Yves Chibon |
aafc66 |
|
|
Pierre-Yves Chibon |
aafc66 |
return 'Tag added'
|
|
Pierre-Yves Chibon |
aafc66 |
|
|
Pierre-Yves Chibon |
aafc66 |
|
|
Pierre-Yves Chibon |
b9cb1e |
def add_user_to_project(session, project, user):
|
|
Pierre-Yves Chibon |
b9cb1e |
''' Add a specified user to a specified project. '''
|
|
Pierre-Yves Chibon |
b9cb1e |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
b9cb1e |
if not user_obj:
|
|
Pierre-Yves Chibon |
b9cb1e |
user_obj = get_user_by_email(session, user)
|
|
Pierre-Yves Chibon |
b9cb1e |
|
|
Pierre-Yves Chibon |
b9cb1e |
if not user_obj:
|
|
Pierre-Yves Chibon |
b9cb1e |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
b9cb1e |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
b9cb1e |
)
|
|
Pierre-Yves Chibon |
b9cb1e |
|
|
Pierre-Yves Chibon |
b9cb1e |
project_user = model.ProjectUser(
|
|
Pierre-Yves Chibon |
b9cb1e |
project_id=project.id,
|
|
Pierre-Yves Chibon |
b9cb1e |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
b9cb1e |
)
|
|
Pierre-Yves Chibon |
b9cb1e |
session.add(project_user)
|
|
Pierre-Yves Chibon |
b9cb1e |
# Make sure we won't have SQLAlchemy error before we create the repo
|
|
Pierre-Yves Chibon |
b9cb1e |
session.flush()
|
|
Pierre-Yves Chibon |
b9cb1e |
|
|
Pierre-Yves Chibon |
b9cb1e |
return 'Comment added'
|
|
Pierre-Yves Chibon |
b9cb1e |
|
|
Pierre-Yves Chibon |
a8a719 |
|
|
Pierre-Yves Chibon |
2c11b4 |
def add_pull_request_comment(session, request, commit, row, comment, user):
|
|
Pierre-Yves Chibon |
2c11b4 |
''' Add a comment to a pull-request. '''
|
|
Pierre-Yves Chibon |
2c11b4 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
2c11b4 |
if not user_obj:
|
|
Pierre-Yves Chibon |
2c11b4 |
user_obj = get_user_by_email(session, user)
|
|
Pierre-Yves Chibon |
2c11b4 |
|
|
Pierre-Yves Chibon |
2c11b4 |
if not user_obj:
|
|
Pierre-Yves Chibon |
2c11b4 |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
2c11b4 |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
2c11b4 |
)
|
|
Pierre-Yves Chibon |
2c11b4 |
|
|
Pierre-Yves Chibon |
2c11b4 |
pr_comment = model.PullRequestComment(
|
|
Pierre-Yves Chibon |
2c11b4 |
pull_request_id=request.id,
|
|
Pierre-Yves Chibon |
2c11b4 |
commit_id=commit,
|
|
Pierre-Yves Chibon |
2c11b4 |
line=row,
|
|
Pierre-Yves Chibon |
2c11b4 |
comment=comment,
|
|
Pierre-Yves Chibon |
2c11b4 |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
2c11b4 |
)
|
|
Pierre-Yves Chibon |
2c11b4 |
session.add(pr_comment)
|
|
Pierre-Yves Chibon |
2c11b4 |
# Make sure we won't have SQLAlchemy error before we create the repo
|
|
Pierre-Yves Chibon |
2c11b4 |
session.flush()
|
|
Pierre-Yves Chibon |
2c11b4 |
|
|
Pierre-Yves Chibon |
2c11b4 |
return 'Comment added'
|
|
Pierre-Yves Chibon |
2c11b4 |
|
|
Pierre-Yves Chibon |
2c11b4 |
|
|
Pierre-Yves Chibon |
1e2e25 |
def get_user_project(session, username):
|
|
Pierre-Yves Chibon |
1e2e25 |
''' Retrieve the list of projects managed by a user.
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
'''
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
query = session.query(
|
|
Pierre-Yves Chibon |
1e2e25 |
model.Project
|
|
Pierre-Yves Chibon |
1e2e25 |
).filter(
|
|
Pierre-Yves Chibon |
1e2e25 |
model.Project.user == username
|
|
Pierre-Yves Chibon |
1e2e25 |
)
|
|
Pierre-Yves Chibon |
1e2e25 |
|
|
Pierre-Yves Chibon |
1e2e25 |
return query.all()
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
49b899 |
def new_project(session, user, name, gitfolder, docfolder, ticketfolder,
|
|
Pierre-Yves Chibon |
d8110b |
description=None, parent_id=None):
|
|
Pierre-Yves Chibon |
d8110b |
''' Create a new project based on the information provided.
|
|
Pierre-Yves Chibon |
d8110b |
'''
|
|
Pierre-Yves Chibon |
6160b6 |
gitrepo = os.path.join(gitfolder, '%s.git' % name)
|
|
Pierre-Yves Chibon |
d8110b |
if os.path.exists(gitrepo):
|
|
Pierre-Yves Chibon |
d8110b |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
ca47fa |
'The project repo "%s" already exists' % name
|
|
Pierre-Yves Chibon |
d8110b |
)
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
3ddae1 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
e1372f |
if not user_obj:
|
|
Pierre-Yves Chibon |
e1372f |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
e1372f |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
e1372f |
)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
d8110b |
project = model.Project(
|
|
Pierre-Yves Chibon |
d8110b |
name=name,
|
|
Pierre-Yves Chibon |
d8110b |
description=description,
|
|
Pierre-Yves Chibon |
e1372f |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
d8110b |
parent_id=parent_id
|
|
Pierre-Yves Chibon |
d8110b |
)
|
|
Pierre-Yves Chibon |
d8110b |
session.add(project)
|
|
Pierre-Yves Chibon |
d8110b |
# Make sure we won't have SQLAlchemy error before we create the repo
|
|
Pierre-Yves Chibon |
d8110b |
session.flush()
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
d8110b |
pygit2.init_repository(gitrepo, bare=True)
|
|
Pierre-Yves Chibon |
d8110b |
|
|
Pierre-Yves Chibon |
6160b6 |
gitrepo = os.path.join(docfolder, project.path)
|
|
Pierre-Yves Chibon |
6160b6 |
if os.path.exists(gitrepo):
|
|
Pierre-Yves Chibon |
6160b6 |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
ca47fa |
'The docs repo "%s" already exists' % project.path
|
|
Pierre-Yves Chibon |
6160b6 |
)
|
|
Pierre-Yves Chibon |
6160b6 |
pygit2.init_repository(gitrepo, bare=True)
|
|
Pierre-Yves Chibon |
6160b6 |
|
|
Pierre-Yves Chibon |
49b899 |
gitrepo = os.path.join(ticketfolder, project.path)
|
|
Pierre-Yves Chibon |
49b899 |
if os.path.exists(gitrepo):
|
|
Pierre-Yves Chibon |
49b899 |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
49b899 |
'The tickets repo "%s" already exists' % project.path
|
|
Pierre-Yves Chibon |
49b899 |
)
|
|
Pierre-Yves Chibon |
49b899 |
pygit2.init_repository(gitrepo, bare=True)
|
|
Pierre-Yves Chibon |
49b899 |
|
|
Pierre-Yves Chibon |
d8110b |
return 'Project "%s" created' % name
|
|
Mathieu Bridon |
eb729c |
|
|
Pierre-Yves Chibon |
f74093 |
|
|
Pierre-Yves Chibon |
a34835 |
def new_issue(session, repo, title, content, user, ticketfolder):
|
|
Pierre-Yves Chibon |
33ff2c |
''' Create a new issue for the specified repo. '''
|
|
Pierre-Yves Chibon |
3ddae1 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
e1372f |
if not user_obj:
|
|
Pierre-Yves Chibon |
e1372f |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
e1372f |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
e1372f |
)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
33ff2c |
issue = model.Issue(
|
|
Pierre-Yves Chibon |
33ff2c |
project_id=repo.id,
|
|
Pierre-Yves Chibon |
33ff2c |
title=title,
|
|
Pierre-Yves Chibon |
33ff2c |
content=content,
|
|
Pierre-Yves Chibon |
e1372f |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
9cb884 |
uid=uuid.uuid4().hex,
|
|
Pierre-Yves Chibon |
33ff2c |
)
|
|
Pierre-Yves Chibon |
33ff2c |
session.add(issue)
|
|
Pierre-Yves Chibon |
057ee1 |
# Make sure we won't have SQLAlchemy error before we create the issue
|
|
Pierre-Yves Chibon |
33ff2c |
session.flush()
|
|
Pierre-Yves Chibon |
33ff2c |
|
|
Pierre-Yves Chibon |
e1690f |
global_id = model.GlobalId(
|
|
Pierre-Yves Chibon |
e1690f |
project_id=repo.id,
|
|
Pierre-Yves Chibon |
e1690f |
issue_id=issue.id,
|
|
Pierre-Yves Chibon |
e1690f |
)
|
|
Pierre-Yves Chibon |
e1690f |
|
|
Pierre-Yves Chibon |
e1690f |
session.add(global_id)
|
|
Pierre-Yves Chibon |
e1690f |
session.flush()
|
|
Pierre-Yves Chibon |
e1690f |
|
|
Pierre-Yves Chibon |
a34835 |
update_git_ticket(issue, repo=repo, ticketfolder=ticketfolder)
|
|
Pierre-Yves Chibon |
a34835 |
|
|
Pierre-Yves Chibon |
fe205b |
globalid = get_issue_global_id(session, issue.project.id, issue.id)
|
|
Pierre-Yves Chibon |
fe205b |
|
|
Pierre-Yves Chibon |
fe205b |
progit.notify.notify_new_issue(issue, globalid)
|
|
Pierre-Yves Chibon |
142bae |
|
|
Pierre-Yves Chibon |
33ff2c |
return 'Issue created'
|
|
Pierre-Yves Chibon |
33ff2c |
|
|
Pierre-Yves Chibon |
33ff2c |
|
|
Pierre-Yves Chibon |
62ec72 |
def new_pull_request(
|
|
Pierre-Yves Chibon |
8e8a9f |
session, repo, repo_from, branch, title, user, stop_id,
|
|
Pierre-Yves Chibon |
8e8a9f |
start_id=None):
|
|
Pierre-Yves Chibon |
472a61 |
''' Create a new pull request on the specified repo. '''
|
|
Pierre-Yves Chibon |
3ddae1 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
7d23c0 |
|
|
Pierre-Yves Chibon |
7d23c0 |
if not user_obj:
|
|
Pierre-Yves Chibon |
7d23c0 |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
7d23c0 |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
7d23c0 |
)
|
|
Pierre-Yves Chibon |
7d23c0 |
|
|
Pierre-Yves Chibon |
472a61 |
request = model.PullRequest(
|
|
Pierre-Yves Chibon |
472a61 |
project_id=repo.id,
|
|
Pierre-Yves Chibon |
62ec72 |
project_id_from=repo_from.id,
|
|
Pierre-Yves Chibon |
8e8a9f |
branch=branch,
|
|
Pierre-Yves Chibon |
472a61 |
title=title,
|
|
Pierre-Yves Chibon |
472a61 |
start_id=start_id,
|
|
Pierre-Yves Chibon |
472a61 |
stop_id=stop_id,
|
|
Pierre-Yves Chibon |
7d23c0 |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
472a61 |
)
|
|
Pierre-Yves Chibon |
472a61 |
session.add(request)
|
|
Pierre-Yves Chibon |
472a61 |
# Make sure we won't have SQLAlchemy error before we create the request
|
|
Pierre-Yves Chibon |
472a61 |
session.flush()
|
|
Pierre-Yves Chibon |
472a61 |
|
|
Pierre-Yves Chibon |
3a3f5e |
global_id = model.GlobalId(
|
|
Pierre-Yves Chibon |
3a3f5e |
project_id=repo.id,
|
|
Pierre-Yves Chibon |
3a3f5e |
request_id=request.id,
|
|
Pierre-Yves Chibon |
3a3f5e |
)
|
|
Pierre-Yves Chibon |
3a3f5e |
|
|
Pierre-Yves Chibon |
3a3f5e |
session.add(global_id)
|
|
Pierre-Yves Chibon |
3a3f5e |
session.flush()
|
|
Pierre-Yves Chibon |
3a3f5e |
|
|
Pierre-Yves Chibon |
dabdba |
globalid = get_pull_request_global_id(
|
|
Pierre-Yves Chibon |
dabdba |
session, request.project.id, request.id)
|
|
Pierre-Yves Chibon |
dabdba |
|
|
Pierre-Yves Chibon |
dabdba |
progit.notify.notify_new_pull_request(request, globalid)
|
|
Pierre-Yves Chibon |
e0589f |
|
|
Pierre-Yves Chibon |
472a61 |
return 'Request created'
|
|
Pierre-Yves Chibon |
472a61 |
|
|
Pierre-Yves Chibon |
472a61 |
|
|
Pierre-Yves Chibon |
a34835 |
def edit_issue(session, issue, ticketfolder,
|
|
Pierre-Yves Chibon |
a34835 |
title=None, content=None, status=None):
|
|
Pierre-Yves Chibon |
c71370 |
''' Edit the specified issue.
|
|
Pierre-Yves Chibon |
c71370 |
'''
|
|
Pierre-Yves Chibon |
c71370 |
edit = []
|
|
Pierre-Yves Chibon |
fb8b6c |
if title and title != issue.title:
|
|
Pierre-Yves Chibon |
c71370 |
issue.title = title
|
|
Pierre-Yves Chibon |
c71370 |
edit.append('title')
|
|
Pierre-Yves Chibon |
fb8b6c |
if content and content != issue.content:
|
|
Pierre-Yves Chibon |
c71370 |
issue.content = content
|
|
Pierre-Yves Chibon |
c71370 |
edit.append('content')
|
|
Pierre-Yves Chibon |
fb8b6c |
if status and status != issue.status:
|
|
Pierre-Yves Chibon |
4d1cbe |
issue.status = status
|
|
Pierre-Yves Chibon |
4d1cbe |
edit.append('status')
|
|
Pierre-Yves Chibon |
c71370 |
|
|
Pierre-Yves Chibon |
65c8d7 |
update_git_ticket(issue, repo=issue.project, ticketfolder=ticketfolder)
|
|
Pierre-Yves Chibon |
a34835 |
|
|
Pierre-Yves Chibon |
c71370 |
if not edit:
|
|
Pierre-Yves Chibon |
c71370 |
return 'No changes to edit'
|
|
Pierre-Yves Chibon |
c71370 |
else:
|
|
Pierre-Yves Chibon |
c71370 |
session.add(issue)
|
|
Pierre-Yves Chibon |
c71370 |
session.flush()
|
|
Pierre-Yves Chibon |
c71370 |
return 'Edited successfully issue #%s' % issue.id
|
|
Pierre-Yves Chibon |
c71370 |
|
|
Pierre-Yves Chibon |
c71370 |
|
|
Pierre-Yves Chibon |
b39aa4 |
def update_project_settings(session, repo, issue_tracker, project_docs):
|
|
Pierre-Yves Chibon |
b39aa4 |
''' Update the settings of a project. '''
|
|
Pierre-Yves Chibon |
b39aa4 |
update = []
|
|
Pierre-Yves Chibon |
b39aa4 |
if issue_tracker != repo.issue_tracker:
|
|
Pierre-Yves Chibon |
b39aa4 |
repo.issue_tracker = issue_tracker
|
|
Pierre-Yves Chibon |
b39aa4 |
update.append('issue_tracker')
|
|
Pierre-Yves Chibon |
b39aa4 |
if project_docs != repo.project_docs:
|
|
Pierre-Yves Chibon |
b39aa4 |
repo.project_docs = project_docs
|
|
Pierre-Yves Chibon |
b39aa4 |
update.append('project_docs')
|
|
Pierre-Yves Chibon |
b39aa4 |
|
|
Pierre-Yves Chibon |
b39aa4 |
if not update:
|
|
Pierre-Yves Chibon |
b39aa4 |
return 'No settings to change'
|
|
Pierre-Yves Chibon |
b39aa4 |
else:
|
|
Pierre-Yves Chibon |
b39aa4 |
session.add(repo)
|
|
Pierre-Yves Chibon |
b39aa4 |
session.flush()
|
|
Pierre-Yves Chibon |
b39aa4 |
return 'Edited successfully setting of repo: %s' % repo.fullname
|
|
Pierre-Yves Chibon |
b39aa4 |
|
|
Pierre-Yves Chibon |
b39aa4 |
|
|
Pierre-Yves Chibon |
a33978 |
def fork_project(session, user, repo, gitfolder,
|
|
Pierre-Yves Chibon |
a33978 |
forkfolder, docfolder,ticketfolder):
|
|
Pierre-Yves Chibon |
652c2c |
''' Fork a given project into the user's forks. '''
|
|
Pierre-Yves Chibon |
df2470 |
if repo.is_fork:
|
|
Pierre-Yves Chibon |
df2470 |
reponame = os.path.join(forkfolder, repo.path)
|
|
Pierre-Yves Chibon |
df2470 |
else:
|
|
Pierre-Yves Chibon |
df2470 |
reponame = os.path.join(gitfolder, repo.path)
|
|
Pierre-Yves Chibon |
df2470 |
forkreponame = '%s.git' % os.path.join(forkfolder, user, repo.name)
|
|
Pierre-Yves Chibon |
652c2c |
|
|
Pierre-Yves Chibon |
9e91eb |
if repo.user.user == user:
|
|
Pierre-Yves Chibon |
9e91eb |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
9e91eb |
'You may not fork your own repo')
|
|
Pierre-Yves Chibon |
9e91eb |
|
|
Pierre-Yves Chibon |
652c2c |
if os.path.exists(forkreponame):
|
|
Pierre-Yves Chibon |
652c2c |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
652c2c |
'Repo "%s/%s" already exists' % (user, repo.name))
|
|
Pierre-Yves Chibon |
652c2c |
|
|
Pierre-Yves Chibon |
3ddae1 |
user_obj = get_user(session, user)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
e1372f |
if not user_obj:
|
|
Pierre-Yves Chibon |
e1372f |
raise progit.exceptions.ProgitException(
|
|
Pierre-Yves Chibon |
e1372f |
'No user "%s" found' % user
|
|
Pierre-Yves Chibon |
e1372f |
)
|
|
Pierre-Yves Chibon |
e1372f |
|
|
Pierre-Yves Chibon |
652c2c |
project = model.Project(
|
|
Pierre-Yves Chibon |
652c2c |
name=repo.name,
|
|
Pierre-Yves Chibon |
652c2c |
description=repo.description,
|
|
Pierre-Yves Chibon |
e1372f |
user_id=user_obj.id,
|
|
Pierre-Yves Chibon |
652c2c |
parent_id=repo.id
|
|
Pierre-Yves Chibon |
652c2c |
)
|
|
Pierre-Yves Chibon |
652c2c |
session.add(project)
|
|
Pierre-Yves Chibon |
652c2c |
# Make sure we won't have SQLAlchemy error before we create the repo
|
|
Pierre-Yves Chibon |
652c2c |
session.flush()
|
|
Pierre-Yves Chibon |
652c2c |
|
|
Pierre-Yves Chibon |
d01de7 |
pygit2.clone_repository(reponame, forkreponame, bare=True)
|
|
Pierre-Yves Chibon |
652c2c |
|
|
Pierre-Yves Chibon |
6160b6 |
gitrepo = os.path.join(docfolder, project.path)
|
|
Pierre-Yves Chibon |
6160b6 |
if os.path.exists(gitrepo):
|
|
Pierre-Yves Chibon |
6160b6 |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
4b7a7d |
'The docs "%s" already exists' % project.path
|
|
Pierre-Yves Chibon |
6160b6 |
)
|
|
Pierre-Yves Chibon |
a33978 |
pygit2.init_repository(gitrepo, bare=True)
|
|
Pierre-Yves Chibon |
a33978 |
|
|
Pierre-Yves Chibon |
a33978 |
gitrepo = os.path.join(ticketfolder, project.path)
|
|
Pierre-Yves Chibon |
a33978 |
if os.path.exists(gitrepo):
|
|
Pierre-Yves Chibon |
a33978 |
raise progit.exceptions.RepoExistsException(
|
|
Pierre-Yves Chibon |
a33978 |
'The tickets repo "%s" already exists' % project.path
|
|
Pierre-Yves Chibon |
a33978 |
)
|
|
Pierre-Yves Chibon |
6160b6 |
pygit2.init_repository(gitrepo, bare=True)
|
|
Pierre-Yves Chibon |
6160b6 |
|
|
Pierre-Yves Chibon |
652c2c |
return 'Repo "%s" cloned to "%s/%s"' % (repo.name, user, repo.name)
|
|
Pierre-Yves Chibon |
652c2c |
|
|
Pierre-Yves Chibon |
652c2c |
|
|
Pierre-Yves Chibon |
ea0ee1 |
def list_projects(
|
|
Pierre-Yves Chibon |
7752a8 |
session, username=None, fork=None,
|
|
Pierre-Yves Chibon |
ea0ee1 |
start=None, limit=None, count=False):
|
|
Pierre-Yves Chibon |
695ef6 |
'''List existing projects
|
|
Pierre-Yves Chibon |
695ef6 |
'''
|
|
Pierre-Yves Chibon |
0b9290 |
projects = session.query(
|
|
Pierre-Yves Chibon |
0b9290 |
model.Project
|
|
Pierre-Yves Chibon |
0b9290 |
).order_by(
|
|
Pierre-Yves Chibon |
0b9290 |
model.Project.date_created
|
|
Pierre-Yves Chibon |
0b9290 |
)
|
|
Mathieu Bridon |
eb729c |
|
|
Pierre-Yves Chibon |
ea0ee1 |
if username is not None:
|
|
Pierre-Yves Chibon |
e1372f |
projects = projects.filter(
|
|
Pierre-Yves Chibon |
e1372f |
model.User.user == username
|
|
Pierre-Yves Chibon |
e1372f |
).filter(
|
|
Pierre-Yves Chibon |
e1372f |
model.User.id == model.Project.user_id
|
|
Pierre-Yves Chibon |
ea0ee1 |
)
|
|
Pierre-Yves Chibon |
ea0ee1 |
|
|
Pierre-Yves Chibon |
7752a8 |
if fork is not None:
|
|
Pierre-Yves Chibon |
7752a8 |
if fork is True:
|
|
Pierre-Yves Chibon |
7752a8 |
projects = projects.filter(
|
|
Pierre-Yves Chibon |
7752a8 |
model.Project.parent_id != None
|
|
Pierre-Yves Chibon |
7752a8 |
)
|
|
Pierre-Yves Chibon |
7752a8 |
elif fork is False:
|
|
Pierre-Yves Chibon |
2f8732 |
projects = projects.filter(
|
|
Pierre-Yves Chibon |
2f8732 |
model.Project.parent_id == None
|
|
Pierre-Yves Chibon |
7752a8 |
)
|
|
Pierre-Yves Chibon |
7752a8 |
|
|
Mathieu Bridon |
eb729c |
if start is not None:
|
|
Mathieu Bridon |
eb729c |
projects = projects.offset(start)
|
|
Mathieu Bridon |
eb729c |
|
|
Mathieu Bridon |
eb729c |
if limit is not None:
|
|
Mathieu Bridon |
eb729c |
projects = projects.limit(limit)
|
|
Mathieu Bridon |
eb729c |
|
|
Pierre-Yves Chibon |
f74093 |
if count:
|
|
Pierre-Yves Chibon |
f74093 |
return projects.count()
|
|
Pierre-Yves Chibon |
f74093 |
else:
|
|
Pierre-Yves Chibon |
f74093 |
return projects.all()
|
|
Mathieu Bridon |
eb729c |
|
|
Mathieu Bridon |
998605 |
|
|
Pierre-Yves Chibon |
929595 |
def get_project(session, name, user=None):
|
|
Pierre-Yves Chibon |
695ef6 |
'''Get a project from the database
|
|
Pierre-Yves Chibon |
695ef6 |
'''
|
|
Pierre-Yves Chibon |
929595 |
query = session.query(
|
|
Pierre-Yves Chibon |
f74093 |
model.Project
|
|
Pierre-Yves Chibon |
178d6f |
).filter(
|
|
Pierre-Yves Chibon |
178d6f |
model.Project.name == name
|
|
Pierre-Yves Chibon |
929595 |
)
|
|
Pierre-Yves Chibon |
1ac00b |
|
|
Pierre-Yves Chibon |
929595 |
if user is not None:
|
|
Pierre-Yves Chibon |
929595 |
query = query.filter(
|
|
Pierre-Yves Chibon |
e1372f |
model.User.user == user
|
|
Pierre-Yves Chibon |
e1372f |
).filter(
|
|
Pierre-Yves Chibon |
e1372f |
model.User.id == model.Project.user_id
|
|
Pierre-Yves Chibon |
1ac00b |
).filter(
|
|
Pierre-Yves Chibon |
1ac00b |
model.Project.parent_id != None
|
|
Pierre-Yves Chibon |
929595 |
)
|
|
Pierre-Yves Chibon |
1ac00b |
else:
|
|
Pierre-Yves Chibon |
1ac00b |
query = query.filter(
|
|
Pierre-Yves Chibon |
1ac00b |
model.Project.parent_id == None
|
|
Pierre-Yves Chibon |
1ac00b |
)
|
|
Pierre-Yves Chibon |
1ac00b |
|
|
Pierre-Yves Chibon |
929595 |
return query.first()
|
|
Pierre-Yves Chibon |
562d75 |
|
|
Pierre-Yves Chibon |
562d75 |
|
|
Pierre-Yves Chibon |
bc60cc |
def get_issues(session, repo, status=None, closed=False, tags=None):
|
|
Pierre-Yves Chibon |
562d75 |
''' Retrieve all the issues associated to a project
|
|
Pierre-Yves Chibon |
b6df7d |
|
|
Pierre-Yves Chibon |
b6df7d |
Watch out that the closed argument is incompatible with the status
|
|
Pierre-Yves Chibon |
b6df7d |
argument. The closed argument will return all the issues whose status
|
|
Pierre-Yves Chibon |
b6df7d |
is not 'Open', otherwise it will return the issues having the specified
|
|
Pierre-Yves Chibon |
b6df7d |
status.
|
|
Pierre-Yves Chibon |
bc60cc |
The `tags` argument can be used to filter the issues returned based on
|
|
Pierre-Yves Chibon |
bc60cc |
a certain tag.
|
|
Pierre-Yves Chibon |
bc60cc |
|
|
Pierre-Yves Chibon |
562d75 |
'''
|
|
Pierre-Yves Chibon |
77e16c |
subquery = session.query(
|
|
Pierre-Yves Chibon |
77e16c |
model.GlobalId,
|
|
Pierre-Yves Chibon |
77e16c |
sqlalchemy.over(
|
|
Pierre-Yves Chibon |
77e16c |
sqlalchemy.func.row_number(),
|
|
Pierre-Yves Chibon |
77e16c |
partition_by=model.GlobalId.project_id,
|
|
Pierre-Yves Chibon |
77e16c |
order_by=model.GlobalId.id
|
|
Pierre-Yves Chibon |
77e16c |
).label('global_id')
|
|
Pierre-Yves Chibon |
77e16c |
).subquery()
|
|
Pierre-Yves Chibon |
77e16c |
|
|
Pierre-Yves Chibon |
562d75 |
query = session.query(
|
|
Pierre-Yves Chibon |
77e16c |
model.Issue,
|
|
Pierre-Yves Chibon |
77e16c |
subquery.c.global_id
|
|
Pierre-Yves Chibon |
77e16c |
).filter(
|
|
Pierre-Yves Chibon |
77e16c |
subquery.c.issue_id == model.Issue.id
|
|
Pierre-Yves Chibon |
562d75 |
).filter(
|
|
Pierre-Yves Chibon |
77e16c |
subquery.c.project_id == model.Issue.project_id
|
|
Pierre-Yves Chibon |
95f042 |
).filter(
|
|
Pierre-Yves Chibon |
95f042 |
model.Issue.project_id == repo.id
|
|
Pierre-Yves Chibon |
3a2a1c |
).order_by(
|
|
Pierre-Yves Chibon |
3a2a1c |
model.Issue.id
|
|
Pierre-Yves Chibon |
562d75 |
)
|
|
Pierre-Yves Chibon |
562d75 |
|
|
Pierre-Yves Chibon |
b6df7d |
if status is not None and not closed:
|
|
Pierre-Yves Chibon |
4d1cbe |
query = query.filter(
|
|
Pierre-Yves Chibon |
4d1cbe |
model.Issue.status == status
|
|
Pierre-Yves Chibon |
4d1cbe |
)
|
|
Pierre-Yves Chibon |
b6df7d |
if closed:
|
|
Pierre-Yves Chibon |
b6df7d |
query = query.filter(
|
|
Pierre-Yves Chibon |
b6df7d |
model.Issue.status != 'Open'
|
|
Pierre-Yves Chibon |
b6df7d |
)
|
|
Pierre-Yves Chibon |
d01150 |
if tags is not None and tags != []:
|
|
Pierre-Yves Chibon |
bc60cc |
query = query.filter(
|
|
Pierre-Yves Chibon |
bc60cc |
model.Issue.id == model.TagIssue.issue_id
|
|
Pierre-Yves Chibon |
bc60cc |
).filter(
|
|
Pierre-Yves Chibon |
bc60cc |
model.TagIssue.tag.in_(tags)
|
|
Pierre-Yves Chibon |
bc60cc |
)
|
|
Pierre-Yves Chibon |
4d1cbe |
|
|
Pierre-Yves Chibon |
562d75 |
return query.all()
|
|
Pierre-Yves Chibon |
562d75 |
|
|
Pierre-Yves Chibon |
562d75 |
|
|
Pierre-Yves Chibon |
b922fe |
def get_issue_global_id(session, projectid, issueid):
|
|
Pierre-Yves Chibon |
b922fe |
''' Retrieve the global identifier of a specific issue based on its
|
|
Pierre-Yves Chibon |
b922fe |
identifier.
|
|
Pierre-Yves Chibon |
b922fe |
'''
|
|
Pierre-Yves Chibon |
b922fe |
subquery = session.query(
|
|
Pierre-Yves Chibon |
b922fe |
model.GlobalId,
|
|
Pierre-Yves Chibon |
b922fe |
sqlalchemy.over(
|
|
Pierre-Yves Chibon |
b922fe |
sqlalchemy.func.row_number(),
|
|
Pierre-Yves Chibon |
b922fe |
partition_by=model.GlobalId.project_id,
|
|
Pierre-Yves Chibon |
b922fe |
order_by=model.GlobalId.id
|
|
Pierre-Yves Chibon |
b922fe |
).label('global_id')
|
|
Pierre-Yves Chibon |
b922fe |
).subquery()
|
|
Pierre-Yves Chibon |
b922fe |
|
|
Pierre-Yves Chibon |
b922fe |
query = session.query(
|
|
Pierre-Yves Chibon |
b922fe |
subquery.c.global_id
|
|
Pierre-Yves Chibon |
b922fe |
).filter(
|
|
Pierre-Yves Chibon |
b922fe |
subquery.c.project_id == projectid
|
|
Pierre-Yves Chibon |
b922fe |
).filter(
|
|
Pierre-Yves Chibon |
b922fe |
subquery.c.issue_id == issueid
|
|
Pierre-Yves Chibon |
b922fe |
)
|
|
Pierre-Yves Chibon |
b922fe |
|
|
Pierre-Yves Chibon |
b922fe |
data = query.first()
|
|
Pierre-Yves Chibon |
b922fe |
|
|
Pierre-Yves Chibon |
b922fe |
return data[0]
|
|
Pierre-Yves Chibon |
b922fe |
|
|
Pierre-Yves Chibon |
b922fe |
|
|
Pierre-Yves Chibon |
bcb038 |
def get_issue(session, projectid, issueid):
|
|
Pierre-Yves Chibon |
562d75 |
''' Retrieve the specified issue
|
|
Pierre-Yves Chibon |
562d75 |
'''
|
|
Pierre-Yves Chibon |
251d96 |
subquery = session.query(
|
|
Pierre-Yves Chibon |
251d96 |
model.GlobalId,
|
|
Pierre-Yves Chibon |
251d96 |
sqlalchemy.over(
|
|
Pierre-Yves Chibon |
251d96 |
sqlalchemy.func.row_number(),
|
|
Pierre-Yves Chibon |
251d96 |
partition_by=model.GlobalId.project_id,
|
|
Pierre-Yves Chibon |
251d96 |
order_by=model.GlobalId.id
|
|
Pierre-Yves Chibon |
251d96 |
).label('global_id')
|
|
Pierre-Yves Chibon |
251d96 |
).subquery()
|
|
Pierre-Yves Chibon |
251d96 |
|
|
Pierre-Yves Chibon |
562d75 |
query = session.query(
|
|
Pierre-Yves Chibon |
562d75 |
model.Issue
|
|
Pierre-Yves Chibon |
562d75 |
).filter(
|
|
Pierre-Yves Chibon |
bcb038 |
model.Issue.project_id == projectid
|
|
Pierre-Yves Chibon |
bcb038 |
).filter(
|
|
Pierre-Yves Chibon |
bcb038 |
model.Issue.project_id == projectid
|
|
Pierre-Yves Chibon |
bcb038 |
).filter(
|
|
Pierre-Yves Chibon |
251d96 |
subquery.c.project_id == model.Issue.project_id
|
|
Pierre-Yves Chibon |
251d96 |
).filter(
|
|
Pierre-Yves Chibon |
a93753 |
subquery.c.issue_id == model.Issue.id
|
|
Pierre-Yves Chibon |
a93753 |
).filter(
|
|
Pierre-Yves Chibon |
251d96 |
subquery.c.global_id == issueid
|
|
Pierre-Yves Chibon |
3a2a1c |
).order_by(
|
|
Pierre-Yves Chibon |
3a2a1c |
model.Issue.id
|
|
Pierre-Yves Chibon |
562d75 |
)
|
|
Pierre-Yves Chibon |
562d75 |
|
|
Pierre-Yves Chibon |
562d75 |
return query.first()
|
|
Pierre-Yves Chibon |
35112d |
|
|
Pierre-Yves Chibon |
8efc30 |
def get_tags_of_project(session, project):
|
|
Pierre-Yves Chibon |
8efc30 |
''' Returns the list of tags associated with the issues of a project.
|
|
Pierre-Yves Chibon |
8efc30 |
'''
|
|
Pierre-Yves Chibon |
8efc30 |
query = session.query(
|
|
Pierre-Yves Chibon |
8efc30 |
model.Tag
|
|
Pierre-Yves Chibon |
8efc30 |
).filter(
|
|
Pierre-Yves Chibon |
8efc30 |
model.Tag.tag == model.TagIssue.tag
|
|
Pierre-Yves Chibon |
8efc30 |
).filter(
|
|
Pierre-Yves Chibon |
8efc30 |
model.TagIssue.issue_id == model.Issue.id
|
|
Pierre-Yves Chibon |
8efc30 |
).filter(
|
|
Pierre-Yves Chibon |
8efc30 |
model.Issue.project_id == project.id
|
|
Pierre-Yves Chibon |
8efc30 |
).order_by(
|
|
Pierre-Yves Chibon |
8efc30 |
model.Tag.tag
|
|
Pierre-Yves Chibon |
8efc30 |
)
|
|
Pierre-Yves Chibon |
8efc30 |
|
|
Pierre-Yves Chibon |
8efc30 |
return query.all()
|
|
Pierre-Yves Chibon |
8efc30 |
|
|
Pierre-Yves Chibon |
8efc30 |
|
|
Pierre-Yves Chibon |
2e810e |
def get_tag(session, tag):
|
|
Pierre-Yves Chibon |
2e810e |
''' Returns a Tag object for the given tag text.
|
|
Pierre-Yves Chibon |
2e810e |
'''
|
|
Pierre-Yves Chibon |
2e810e |
query = session.query(
|
|
Pierre-Yves Chibon |
2e810e |
model.Tag
|
|
Pierre-Yves Chibon |
2e810e |
).filter(
|
|
Pierre-Yves Chibon |
2e810e |
model.Tag.tag == tag
|
|
Pierre-Yves Chibon |
2e810e |
)
|
|
Pierre-Yves Chibon |
2e810e |
|
|
Pierre-Yves Chibon |
2e810e |
return query.first()
|
|
Pierre-Yves Chibon |
2e810e |
|
|
Pierre-Yves Chibon |
35112d |
|
|
Pierre-Yves Chibon |
b0265c |
def get_pull_requests(
|
|
Pierre-Yves Chibon |
b0265c |
session, project_id=None, project_id_from=None, status=None):
|
|
Pierre-Yves Chibon |
b0265c |
''' Retrieve the specified issue
|
|
Pierre-Yves Chibon |
b0265c |
'''
|
|
Pierre-Yves Chibon |
33e9fc |
|
|
Pierre-Yves Chibon |
33e9fc |
subquery = session.query(
|
|
Pierre-Yves Chibon |
33e9fc |
model.GlobalId,
|
|
Pierre-Yves Chibon |
33e9fc |
sqlalchemy.over(
|
|
Pierre-Yves Chibon |
33e9fc |
sqlalchemy.func.row_number(),
|
|
Pierre-Yves Chibon |
33e9fc |
partition_by=model.GlobalId.project_id,
|
|
Pierre-Yves Chibon |
33e9fc |
order_by=model.GlobalId.id
|
|
Pierre-Yves Chibon |
33e9fc |
).label('global_id')
|
|
Pierre-Yves Chibon |
33e9fc |
).subquery()
|
|
Pierre-Yves Chibon |
33e9fc |
|
|
Pierre-Yves Chibon |
b0265c |
query = session.query(
|
|
Pierre-Yves Chibon |
33e9fc |
model.PullRequest,
|
|
Pierre-Yves Chibon |
33e9fc |
subquery.c.global_id
|
|
Pierre-Yves Chibon |
33e9fc |
).filter(
|
|
Pierre-Yves Chibon |
33e9fc |
subquery.c.request_id == model.PullRequest.id
|
|
Pierre-Yves Chibon |
33e9fc |
).filter(
|
|
Pierre-Yves Chibon |
33e9fc |
subquery.c.project_id == model.PullRequest.project_id
|
|
Pierre-Yves Chibon |
3a2a1c |
).order_by(
|
|
Pierre-Yves Chibon |
3a2a1c |
model.PullRequest.id
|
|
Pierre-Yves Chibon |
b0265c |
)
|
|
Pierre-Yves Chibon |
b0265c |
|
|
Pierre-Yves Chibon |
b0265c |
if project_id:
|
|
Pierre-Yves Chibon |
b0265c |
query = query.filter(
|
|
Pierre-Yves Chibon |
b0265c |
model.PullRequest.project_id == project_id
|
|
Pierre-Yves Chibon |
b0265c |
)
|
|
Pierre-Yves Chibon |
b0265c |
|
|
Pierre-Yves Chibon |
b0265c |
if project_id_from:
|
|
Pierre-Yves Chibon |
b0265c |
query = query.filter(
|
|
Pierre-Yves Chibon |
b0265c |
model.PullRequest.project_id_from == project_id_from
|
|
Pierre-Yves Chibon |
b0265c |
)
|
|
Pierre-Yves Chibon |
b0265c |
|
|
Pierre-Yves Chibon |
b0265c |
if status is not None:
|
|
Pierre-Yves Chibon |
b0265c |
query = query.filter(
|
|
Pierre-Yves Chibon |
b0265c |
model.PullRequest.status == status
|
|
Pierre-Yves Chibon |
b0265c |
)
|
|
Pierre-Yves Chibon |
b0265c |
|
|
Pierre-Yves Chibon |
b0265c |
return query.all()
|
|
Pierre-Yves Chibon |
b0265c |
|
|
Pierre-Yves Chibon |
b0265c |
|
|
Pierre-Yves Chibon |
35112d |
def get_pull_request(
|
|
Pierre-Yves Chibon |
35112d |
session, requestid, project_id=None, project_id_from=None):
|
|
Pierre-Yves Chibon |
35112d |
''' Retrieve the specified issue
|
|
Pierre-Yves Chibon |
35112d |
'''
|
|
Pierre-Yves Chibon |
3f844b |
|
|
Pierre-Yves Chibon |
3f844b |
subquery = session.query(
|
|
Pierre-Yves Chibon |
3f844b |
model.GlobalId,
|
|
Pierre-Yves Chibon |
3f844b |
sqlalchemy.over(
|
|
Pierre-Yves Chibon |
3f844b |
sqlalchemy.func.row_number(),
|
|
Pierre-Yves Chibon |
3f844b |
partition_by=model.GlobalId.project_id,
|
|
Pierre-Yves Chibon |
3f844b |
order_by=model.GlobalId.id
|
|
Pierre-Yves Chibon |
3f844b |
).label('global_id')
|
|
Pierre-Yves Chibon |
3f844b |
).subquery()
|
|
Pierre-Yves Chibon |
3f844b |
|
|
Pierre-Yves Chibon |
35112d |
query = session.query(
|
|
Pierre-Yves Chibon |
35112d |
model.PullRequest
|
|
Pierre-Yves Chibon |
35112d |
).filter(
|
|
Pierre-Yves Chibon |
3f844b |
subquery.c.project_id == model.PullRequest.project_id
|
|
Pierre-Yves Chibon |
3f844b |
).filter(
|
|
Pierre-Yves Chibon |
9a1f18 |
subquery.c.request_id == model.PullRequest.id
|
|
Pierre-Yves Chibon |
9a1f18 |
).filter(
|
|
Pierre-Yves Chibon |
3f844b |
subquery.c.global_id == requestid
|
|
Pierre-Yves Chibon |
9a1f18 |
).order_by(
|
|
Pierre-Yves Chibon |
9a1f18 |
model.PullRequest.id
|
|
Pierre-Yves Chibon |
35112d |
)
|
|
Pierre-Yves Chibon |
35112d |
|
|
Pierre-Yves Chibon |
35112d |
if project_id:
|
|
Pierre-Yves Chibon |
35112d |
query = query.filter(
|
|
Pierre-Yves Chibon |
35112d |
model.PullRequest.project_id == project_id
|
|
Pierre-Yves Chibon |
35112d |
)
|
|
Pierre-Yves Chibon |
35112d |
|
|
Pierre-Yves Chibon |
35112d |
if project_id_from:
|
|
Pierre-Yves Chibon |
35112d |
query = query.filter(
|
|
Pierre-Yves Chibon |
35112d |
model.PullRequest.project_id_from == project_id_from
|
|
Pierre-Yves Chibon |
35112d |
)
|
|
Pierre-Yves Chibon |
35112d |
|
|
Pierre-Yves Chibon |
35112d |
return query.first()
|
|
Pierre-Yves Chibon |
42a02c |
|
|
Pierre-Yves Chibon |
42a02c |
|
|
Pierre-Yves Chibon |
fc841e |
def get_pull_request_global_id(session, projectid, requestid):
|
|
Pierre-Yves Chibon |
fc841e |
''' Retrieve the global identifier of a specific request based on its
|
|
Pierre-Yves Chibon |
fc841e |
identifier.
|
|
Pierre-Yves Chibon |
fc841e |
'''
|
|
Pierre-Yves Chibon |
fc841e |
subquery = session.query(
|
|
Pierre-Yves Chibon |
fc841e |
model.GlobalId,
|
|
Pierre-Yves Chibon |
fc841e |
sqlalchemy.over(
|
|
Pierre-Yves Chibon |
fc841e |
sqlalchemy.func.row_number(),
|
|
Pierre-Yves Chibon |
fc841e |
partition_by=model.GlobalId.project_id,
|
|
Pierre-Yves Chibon |
fc841e |
order_by=model.GlobalId.id
|
|
Pierre-Yves Chibon |
fc841e |
).label('global_id')
|
|
Pierre-Yves Chibon |
fc841e |
).subquery()
|
|
Pierre-Yves Chibon |
fc841e |
|
|
Pierre-Yves Chibon |
fc841e |
query = session.query(
|
|
Pierre-Yves Chibon |
fc841e |
subquery.c.global_id
|
|
Pierre-Yves Chibon |
fc841e |
).filter(
|
|
Pierre-Yves Chibon |
fc841e |
subquery.c.project_id == projectid
|
|
Pierre-Yves Chibon |
fc841e |
).filter(
|
|
Pierre-Yves Chibon |
fc841e |
subquery.c.request_id == requestid
|
|
Pierre-Yves Chibon |
fc841e |
)
|
|
Pierre-Yves Chibon |
fc841e |
|
|
Pierre-Yves Chibon |
fc841e |
data = query.first()
|
|
Pierre-Yves Chibon |
fc841e |
|
|
Pierre-Yves Chibon |
fc841e |
return data[0]
|
|
Pierre-Yves Chibon |
fc841e |
|
|
Pierre-Yves Chibon |
fc841e |
|
|
Pierre-Yves Chibon |
375fde |
def close_pull_request(session, request, user, merged=True):
|
|
Pierre-Yves Chibon |
42a02c |
''' Close the provided pull-request.
|
|
Pierre-Yves Chibon |
42a02c |
'''
|
|
Pierre-Yves Chibon |
42a02c |
request.status = False
|
|
Pierre-Yves Chibon |
42a02c |
session.add(request)
|
|
Pierre-Yves Chibon |
42a02c |
session.flush()
|
|
Pierre-Yves Chibon |
4d1cbe |
|
|
Pierre-Yves Chibon |
00aee8 |
globalid = get_pull_request_global_id(
|
|
Pierre-Yves Chibon |
00aee8 |
session, request.project.id, request.id)
|
|
Pierre-Yves Chibon |
00aee8 |
|
|
Johan Cwiklinski |
86d9c4 |
if merged == True:
|
|
Pierre-Yves Chibon |
00aee8 |
progit.notify.notify_merge_pull_request(request, user, globalid)
|
|
Johan Cwiklinski |
86d9c4 |
else:
|
|
Pierre-Yves Chibon |
00aee8 |
progit.notify.notify_cancelled_pull_request(request, user, globalid)
|
|
Pierre-Yves Chibon |
5fdf85 |
|
|
Pierre-Yves Chibon |
4d1cbe |
|
|
Pierre-Yves Chibon |
4d1cbe |
def get_issue_statuses(session):
|
|
Pierre-Yves Chibon |
4d1cbe |
''' Return the complete list of status an issue can have.
|
|
Pierre-Yves Chibon |
4d1cbe |
'''
|
|
Pierre-Yves Chibon |
4d1cbe |
output = []
|
|
Pierre-Yves Chibon |
4d1cbe |
statuses = session.query(model.StatusIssue).all()
|
|
Pierre-Yves Chibon |
4d1cbe |
for status in statuses:
|
|
Pierre-Yves Chibon |
4d1cbe |
output.append(status.status)
|
|
Pierre-Yves Chibon |
4d1cbe |
return output
|
|
Pierre-Yves Chibon |
7c47d2 |
|
|
Pierre-Yves Chibon |
7c47d2 |
|
|
Pierre-Yves Chibon |
7c47d2 |
def generate_gitolite_acls(session, configfile):
|
|
Pierre-Yves Chibon |
7c47d2 |
''' Generate the configuration file for gitolite for all projects
|
|
Pierre-Yves Chibon |
7c47d2 |
on the forge.
|
|
Pierre-Yves Chibon |
7c47d2 |
'''
|
|
Pierre-Yves Chibon |
7c47d2 |
config = []
|
|
Pierre-Yves Chibon |
7c47d2 |
for project in session.query(model.Project).all():
|
|
Pierre-Yves Chibon |
7f72e8 |
if project.parent_id:
|
|
Pierre-Yves Chibon |
7f72e8 |
config.append('repo forks/%s' % project.fullname)
|
|
Pierre-Yves Chibon |
7f72e8 |
else:
|
|
Pierre-Yves Chibon |
7f72e8 |
config.append('repo %s' % project.fullname)
|
|
Pierre-Yves Chibon |
dfe989 |
config.append(' RW+ = %s' % project.user.user)
|
|
Pierre-Yves Chibon |
7c47d2 |
for user in project.users:
|
|
Pierre-Yves Chibon |
7c47d2 |
if user != project.user:
|
|
Pierre-Yves Chibon |
3562ab |
config.append(' RW+ = %s' % user.user.user)
|
|
Pierre-Yves Chibon |
7c47d2 |
config.append('')
|
|
Pierre-Yves Chibon |
7c47d2 |
|
|
Pierre-Yves Chibon |
3cec45 |
config.append('repo docs/%s' % project.fullname)
|
|
Pierre-Yves Chibon |
3cec45 |
config.append(' RW+ = %s' % project.user.user)
|
|
Pierre-Yves Chibon |
3cec45 |
for user in project.users:
|
|
Pierre-Yves Chibon |
3cec45 |
if user != project.user:
|
|
Pierre-Yves Chibon |
3562ab |
config.append(' RW+ = %s' % user.user.user)
|
|
Pierre-Yves Chibon |
3cec45 |
config.append('')
|
|
Pierre-Yves Chibon |
3cec45 |
|
|
Pierre-Yves Chibon |
1a37c6 |
config.append('repo tickets/%s' % project.fullname)
|
|
Pierre-Yves Chibon |
1a37c6 |
config.append(' RW+ = %s' % project.user.user)
|
|
Pierre-Yves Chibon |
1a37c6 |
for user in project.users:
|
|
Pierre-Yves Chibon |
1a37c6 |
if user != project.user:
|
|
Pierre-Yves Chibon |
3562ab |
config.append(' RW+ = %s' % user.user.user)
|
|
Pierre-Yves Chibon |
1a37c6 |
config.append('')
|
|
Pierre-Yves Chibon |
1a37c6 |
|
|
Pierre-Yves Chibon |
edd115 |
with open(configfile, 'w') as stream:
|
|
Pierre-Yves Chibon |
9df3ed |
for row in config:
|
|
Pierre-Yves Chibon |
30ec31 |
stream.write(row + '\n')
|
|
Pierre-Yves Chibon |
c2aa0d |
|
|
Pierre-Yves Chibon |
c2aa0d |
|
|
Pierre-Yves Chibon |
c2aa0d |
def set_up_user(session, username, fullname, user_email):
|
|
Pierre-Yves Chibon |
c2aa0d |
''' Set up a new user into the database or update its information. '''
|
|
Pierre-Yves Chibon |
03c11f |
user = get_user(session, username)
|
|
Pierre-Yves Chibon |
c2aa0d |
if not user:
|
|
Pierre-Yves Chibon |
c2aa0d |
user = model.User(
|
|
Pierre-Yves Chibon |
c2aa0d |
user=username,
|
|
Pierre-Yves Chibon |
c2aa0d |
fullname=fullname)
|
|
Pierre-Yves Chibon |
c2aa0d |
session.add(user)
|
|
Pierre-Yves Chibon |
c2aa0d |
session.flush()
|
|
Pierre-Yves Chibon |
c2aa0d |
|
|
Pierre-Yves Chibon |
c2aa0d |
if user.fullname != fullname:
|
|
Pierre-Yves Chibon |
c2aa0d |
user.fullname = fullname
|
|
Pierre-Yves Chibon |
c2aa0d |
session.add(user)
|
|
Pierre-Yves Chibon |
c2aa0d |
session.flush()
|
|
Pierre-Yves Chibon |
c2aa0d |
|
|
Pierre-Yves Chibon |
8b2894 |
emails = [email.email for email in user.emails]
|
|
Pierre-Yves Chibon |
8b2894 |
if user_email not in emails:
|
|
Pierre-Yves Chibon |
c2aa0d |
useremail = model.UserEmail(
|
|
Pierre-Yves Chibon |
c2aa0d |
user_id=user.id,
|
|
Pierre-Yves Chibon |
c2aa0d |
email=user_email)
|
|
Pierre-Yves Chibon |
c2aa0d |
session.add(useremail)
|
|
Pierre-Yves Chibon |
c2aa0d |
session.flush()
|
|
Pierre-Yves Chibon |
8f25f2 |
|
|
Pierre-Yves Chibon |
8f25f2 |
|
|
Pierre-Yves Chibon |
8f25f2 |
def update_user_ssh(session, user, ssh_key):
|
|
Pierre-Yves Chibon |
8f25f2 |
''' Set up a new user into the database or update its information. '''
|
|
Pierre-Yves Chibon |
8f25f2 |
if isinstance(user, basestring):
|
|
Pierre-Yves Chibon |
8f25f2 |
user = get_user(session, user)
|
|
Pierre-Yves Chibon |
8f25f2 |
|
|
Pierre-Yves Chibon |
8f25f2 |
message = 'Nothing to update'
|
|
Pierre-Yves Chibon |
8f25f2 |
|
|
Pierre-Yves Chibon |
8f25f2 |
if ssh_key != user.public_ssh_key:
|
|
Pierre-Yves Chibon |
8f25f2 |
user.public_ssh_key = ssh_key
|
|
Pierre-Yves Chibon |
8f25f2 |
session.add(user)
|
|
Pierre-Yves Chibon |
8f25f2 |
session.flush()
|
|
Pierre-Yves Chibon |
8f25f2 |
message = 'Public ssh key updated'
|
|
Pierre-Yves Chibon |
8f25f2 |
|
|
Pierre-Yves Chibon |
8f25f2 |
return message
|
|
Pierre-Yves Chibon |
f7d963 |
|
|
Pierre-Yves Chibon |
f7d963 |
|
|
Pierre-Yves Chibon |
f7d963 |
def issue_to_json(issue):
|
|
Pierre-Yves Chibon |
f7d963 |
""" Convert all the data related to an issue (a ticket) as json object.
|
|
Pierre-Yves Chibon |
f7d963 |
|
|
Pierre-Yves Chibon |
f7d963 |
"""
|
|
Pierre-Yves Chibon |
f7d963 |
output = {
|
|
Pierre-Yves Chibon |
f7d963 |
'title': issue.title,
|
|
Pierre-Yves Chibon |
f7d963 |
'content': issue.content,
|
|
Pierre-Yves Chibon |
f7d963 |
'status': issue.status,
|
|
Pierre-Yves Chibon |
f7d963 |
'date_created': issue.date_created.strftime('%s'),
|
|
Pierre-Yves Chibon |
f7d963 |
'user': {
|
|
Pierre-Yves Chibon |
f7d963 |
'name': issue.user.user,
|
|
Pierre-Yves Chibon |
f7d963 |
'emails': [email.email for email in issue.user.emails],
|
|
Pierre-Yves Chibon |
f7d963 |
}
|
|
Pierre-Yves Chibon |
f7d963 |
}
|
|
Pierre-Yves Chibon |
f7d963 |
|
|
Pierre-Yves Chibon |
f7d963 |
comments = []
|
|
Pierre-Yves Chibon |
f7d963 |
for comment in issue.comments:
|
|
Pierre-Yves Chibon |
f7d963 |
cmt = {
|
|
Pierre-Yves Chibon |
f7d963 |
'id': comment.id,
|
|
Pierre-Yves Chibon |
f7d963 |
'comment': comment.comment,
|
|
Pierre-Yves Chibon |
f7d963 |
'parent': comment.parent_id,
|
|
Pierre-Yves Chibon |
f7d963 |
'date_created': comment.date_created.strftime('%s'),
|
|
Pierre-Yves Chibon |
f7d963 |
'user': {
|
|
Pierre-Yves Chibon |
f7d963 |
'name': comment.user.user,
|
|
Pierre-Yves Chibon |
f7d963 |
'emails': [email.email for email in comment.user.emails],
|
|
Pierre-Yves Chibon |
f7d963 |
}
|
|
Pierre-Yves Chibon |
f7d963 |
}
|
|
Pierre-Yves Chibon |
f7d963 |
comments.append(cmt)
|
|
Pierre-Yves Chibon |
f7d963 |
|
|
Pierre-Yves Chibon |
f7d963 |
output['comments'] = comments
|
|
Pierre-Yves Chibon |
f7d963 |
|
|
Pierre-Yves Chibon |
f7d963 |
return json.dumps(output)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
def update_git_ticket(issue, repo, ticketfolder):
|
|
Pierre-Yves Chibon |
d133cd |
""" Update the given issue in its git.
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
This method forks the provided repo, add/edit the issue whose file name
|
|
Pierre-Yves Chibon |
d133cd |
is defined by the uid field of the issue and if there are additions/
|
|
Pierre-Yves Chibon |
d133cd |
changes commit them and push them back to the original repo.
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
"""
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Get the fork
|
|
Pierre-Yves Chibon |
d133cd |
repopath = os.path.join(ticketfolder, repo.path)
|
|
Pierre-Yves Chibon |
d133cd |
ticket_repo = pygit2.Repository(repopath)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Clone the repo into a temp folder
|
|
Pierre-Yves Chibon |
d133cd |
newpath = tempfile.mkdtemp()
|
|
Pierre-Yves Chibon |
d133cd |
new_repo = pygit2.clone_repository(repopath, newpath)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
file_path = os.path.join(newpath, issue.uid)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Get the current index
|
|
Pierre-Yves Chibon |
d133cd |
index = new_repo.index
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Are we adding files
|
|
Pierre-Yves Chibon |
d133cd |
added = False
|
|
Pierre-Yves Chibon |
d133cd |
if not os.path.exists(file_path):
|
|
Pierre-Yves Chibon |
d133cd |
added = True
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Write down what changed
|
|
Pierre-Yves Chibon |
d133cd |
with open(file_path, 'w') as stream:
|
|
Pierre-Yves Chibon |
d133cd |
stream.write(issue_to_json(issue))
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Retrieve the list of files that changed
|
|
Pierre-Yves Chibon |
d133cd |
diff = new_repo.diff()
|
|
Pierre-Yves Chibon |
d133cd |
files = [patch.new_file_path for patch in diff]
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Add the changes to the index
|
|
Pierre-Yves Chibon |
d133cd |
if added:
|
|
Pierre-Yves Chibon |
d133cd |
index.add(issue.uid)
|
|
Pierre-Yves Chibon |
d133cd |
for filename in files:
|
|
Pierre-Yves Chibon |
d133cd |
index.add(filename)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# If not change, return
|
|
Pierre-Yves Chibon |
63324c |
if not files and not added:
|
|
Pierre-Yves Chibon |
d133cd |
shutil.rmtree(newpath)
|
|
Pierre-Yves Chibon |
d133cd |
return
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# See if there is a parent to this commit
|
|
Pierre-Yves Chibon |
d133cd |
parent = None
|
|
Pierre-Yves Chibon |
d133cd |
try:
|
|
Pierre-Yves Chibon |
d133cd |
parent = new_repo.head.get_object().oid
|
|
Pierre-Yves Chibon |
d133cd |
except pygit2.GitError:
|
|
Pierre-Yves Chibon |
d133cd |
pass
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
parents = []
|
|
Pierre-Yves Chibon |
d133cd |
if parent:
|
|
Pierre-Yves Chibon |
d133cd |
parents.append(parent)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Author/commiter will always be this one
|
|
Pierre-Yves Chibon |
d133cd |
author = pygit2.Signature(name='progit', email='progit')
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Actually commit
|
|
Pierre-Yves Chibon |
d133cd |
sha = new_repo.create_commit(
|
|
Pierre-Yves Chibon |
d133cd |
'refs/heads/master',
|
|
Pierre-Yves Chibon |
d133cd |
author,
|
|
Pierre-Yves Chibon |
d133cd |
author,
|
|
Pierre-Yves Chibon |
d133cd |
'Updated ticket %s: %s' % (issue.uid, issue.title),
|
|
Pierre-Yves Chibon |
d133cd |
new_repo.index.write_tree(),
|
|
Pierre-Yves Chibon |
d133cd |
parents)
|
|
Pierre-Yves Chibon |
d133cd |
index.write()
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Push to origin
|
|
Pierre-Yves Chibon |
d133cd |
ori_remote = new_repo.remotes[0]
|
|
Pierre-Yves Chibon |
d133cd |
master_ref = new_repo.lookup_reference('HEAD').resolve()
|
|
Pierre-Yves Chibon |
d133cd |
refname = '%s:%s' % (master_ref.name, master_ref.name)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
ori_remote.push(refname)
|
|
Pierre-Yves Chibon |
d133cd |
|
|
Pierre-Yves Chibon |
d133cd |
# Remove the clone
|
|
Pierre-Yves Chibon |
d133cd |
shutil.rmtree(newpath)
|
|
Pierre-Yves Chibon |
95245b |
|
|
Pierre-Yves Chibon |
95245b |
|
|
Pierre-Yves Chibon |
95245b |
def avatar_url(username, size=64, default='retro'):
|
|
Pierre-Yves Chibon |
95245b |
openid = "http://%s.id.fedoraproject.org/" % username
|
|
Pierre-Yves Chibon |
95245b |
return avatar_url_from_openid(openid, size, default)
|
|
Pierre-Yves Chibon |
95245b |
|
|
Pierre-Yves Chibon |
95245b |
|
|
Pierre-Yves Chibon |
95245b |
def avatar_url_from_openid(openid, size=64, default='retro', dns=False):
|
|
Pierre-Yves Chibon |
95245b |
"""
|
|
Pierre-Yves Chibon |
95245b |
Our own implementation since fas doesn't support this nicely yet.
|
|
Pierre-Yves Chibon |
95245b |
"""
|
|
Pierre-Yves Chibon |
95245b |
|
|
Pierre-Yves Chibon |
95245b |
if dns: # pragma: no cover
|
|
Pierre-Yves Chibon |
95245b |
# This makes an extra DNS SRV query, which can slow down our webapps.
|
|
Pierre-Yves Chibon |
95245b |
# It is necessary for libravatar federation, though.
|
|
Pierre-Yves Chibon |
95245b |
import libravatar
|
|
Pierre-Yves Chibon |
95245b |
return libravatar.libravatar_url(
|
|
Pierre-Yves Chibon |
95245b |
openid=openid,
|
|
Pierre-Yves Chibon |
95245b |
size=size,
|
|
Pierre-Yves Chibon |
95245b |
default=default,
|
|
Pierre-Yves Chibon |
95245b |
)
|
|
Pierre-Yves Chibon |
95245b |
else:
|
|
Pierre-Yves Chibon |
95245b |
import urllib, hashlib
|
|
Pierre-Yves Chibon |
95245b |
query = urllib.urlencode({'s': size, 'd': default})
|
|
Pierre-Yves Chibon |
95245b |
hash = hashlib.sha256(openid).hexdigest()
|
|
Pierre-Yves Chibon |
95245b |
return "https://seccdn.libravatar.org/avatar/%s?%s" % (hash, query)
|
|
Pierre-Yves Chibon |
b0d578 |
|
|
Pierre-Yves Chibon |
b0d578 |
|
|
Pierre-Yves Chibon |
b0d578 |
def get_session_by_visitkey(session, sessionid):
|
|
Pierre-Yves Chibon |
b0d578 |
''' Return a specified VisitUser via its session identifier (visit_key).
|
|
Pierre-Yves Chibon |
b0d578 |
|
|
Pierre-Yves Chibon |
b0d578 |
:arg session: the session with which to connect to the database.
|
|
Pierre-Yves Chibon |
b0d578 |
|
|
Pierre-Yves Chibon |
b0d578 |
'''
|
|
Pierre-Yves Chibon |
b0d578 |
query = session.query(
|
|
Pierre-Yves Chibon |
4313f4 |
model.ProgitUserVisit
|
|
Pierre-Yves Chibon |
b0d578 |
).filter(
|
|
Pierre-Yves Chibon |
4313f4 |
model.ProgitUserVisit.visit_key == sessionid
|
|
Pierre-Yves Chibon |
b0d578 |
)
|
|
Pierre-Yves Chibon |
b0d578 |
|
|
Pierre-Yves Chibon |
b0d578 |
return query.first()
|
|
Pierre-Yves Chibon |
a8a719 |
|
|
Pierre-Yves Chibon |
a8a719 |
|
|
Pierre-Yves Chibon |
a8a719 |
def get_groups(session):
|
|
Pierre-Yves Chibon |
a8a719 |
''' Return the list of groups present in the database.
|
|
Pierre-Yves Chibon |
a8a719 |
|
|
Pierre-Yves Chibon |
a8a719 |
:arg session: the session with which to connect to the database.
|
|
Pierre-Yves Chibon |
a8a719 |
|
|
Pierre-Yves Chibon |
a8a719 |
'''
|
|
Pierre-Yves Chibon |
a8a719 |
query = session.query(
|
|
Pierre-Yves Chibon |
a8a719 |
model.ProgitGroup
|
|
Pierre-Yves Chibon |
a8a719 |
).order_by(
|
|
Pierre-Yves Chibon |
a8a719 |
model.ProgitGroup.group_name
|
|
Pierre-Yves Chibon |
a8a719 |
)
|
|
Pierre-Yves Chibon |
a8a719 |
|
|
Pierre-Yves Chibon |
a8a719 |
return query.all()
|
|
Pierre-Yves Chibon |
b97cda |
|
|
Pierre-Yves Chibon |
b97cda |
|
|
Pierre-Yves Chibon |
b97cda |
def get_group(session, group):
|
|
Pierre-Yves Chibon |
b97cda |
''' Return a specific group for the specified group name.
|
|
Pierre-Yves Chibon |
b97cda |
|
|
Pierre-Yves Chibon |
b97cda |
:arg session: the session with which to connect to the database.
|
|
Pierre-Yves Chibon |
b97cda |
|
|
Pierre-Yves Chibon |
b97cda |
'''
|
|
Pierre-Yves Chibon |
b97cda |
query = session.query(
|
|
Pierre-Yves Chibon |
b97cda |
model.ProgitGroup
|
|
Pierre-Yves Chibon |
b97cda |
).filter(
|
|
Pierre-Yves Chibon |
b97cda |
model.ProgitGroup.group_name == group
|
|
Pierre-Yves Chibon |
b97cda |
).order_by(
|
|
Pierre-Yves Chibon |
b97cda |
model.ProgitGroup.group_name
|
|
Pierre-Yves Chibon |
b97cda |
)
|
|
Pierre-Yves Chibon |
b97cda |
|
|
Pierre-Yves Chibon |
b97cda |
return query.first()
|
|
Pierre-Yves Chibon |
025568 |
|
|
Pierre-Yves Chibon |
025568 |
|
|
Pierre-Yves Chibon |
025568 |
def get_users_by_group(session, group):
|
|
Pierre-Yves Chibon |
025568 |
''' Return the list of users for a specified group.
|
|
Pierre-Yves Chibon |
025568 |
|
|
Pierre-Yves Chibon |
025568 |
:arg session: the session with which to connect to the database.
|
|
Pierre-Yves Chibon |
025568 |
|
|
Pierre-Yves Chibon |
025568 |
'''
|
|
Pierre-Yves Chibon |
025568 |
query = session.query(
|
|
Pierre-Yves Chibon |
025568 |
model.User
|
|
Pierre-Yves Chibon |
025568 |
).filter(
|
|
Pierre-Yves Chibon |
025568 |
model.User.id == model.ProgitUserGroup.user_id
|
|
Pierre-Yves Chibon |
025568 |
).filter(
|
|
Pierre-Yves Chibon |
025568 |
model.ProgitUserGroup.group_id == model.ProgitGroup.id
|
|
Pierre-Yves Chibon |
025568 |
).filter(
|
|
Pierre-Yves Chibon |
025568 |
model.ProgitGroup.group_name == group
|
|
Pierre-Yves Chibon |
025568 |
).order_by(
|
|
Pierre-Yves Chibon |
025568 |
model.User.user
|
|
Pierre-Yves Chibon |
025568 |
)
|
|
Pierre-Yves Chibon |
025568 |
|
|
Pierre-Yves Chibon |
025568 |
return query.all()
|
|
Pierre-Yves Chibon |
f77dac |
|
|
Pierre-Yves Chibon |
f77dac |
|
|
Pierre-Yves Chibon |
f77dac |
def get_user_group(session, userid, groupid):
|
|
Pierre-Yves Chibon |
f77dac |
''' Return a specific user_group for the specified group and user
|
|
Pierre-Yves Chibon |
f77dac |
identifiers.
|
|
Pierre-Yves Chibon |
f77dac |
|
|
Pierre-Yves Chibon |
f77dac |
:arg session: the session with which to connect to the database.
|
|
Pierre-Yves Chibon |
f77dac |
|
|
Pierre-Yves Chibon |
f77dac |
'''
|
|
Pierre-Yves Chibon |
f77dac |
query = session.query(
|
|
Pierre-Yves Chibon |
f77dac |
model.ProgitUserGroup
|
|
Pierre-Yves Chibon |
f77dac |
).filter(
|
|
Pierre-Yves Chibon |
f77dac |
model.ProgitUserGroup.user_id == userid
|
|
Pierre-Yves Chibon |
f77dac |
).filter(
|
|
Pierre-Yves Chibon |
f77dac |
model.ProgitUserGroup.group_id == groupid
|
|
Pierre-Yves Chibon |
f77dac |
)
|
|
Pierre-Yves Chibon |
f77dac |
|
|
Pierre-Yves Chibon |
f77dac |
return query.first()
|