Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add GitLab repo sync and webhook support #1870

Closed
wants to merge 19 commits into from
Closed
Changes from 10 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion docs/features.rst
Original file line number Diff line number Diff line change
@@ -13,7 +13,7 @@ More information can be found in the :doc:`vcs` page.
Auto-updating
-------------

The :doc:`webhooks` page talks about the different ways you can ping RTD to let us know your project has been updated. We have official support for GitHub, and anywhere else we have a generic post-commit hook that allows you to POST to a URL to get your documentation built.
The :doc:`webhooks` page talks about the different ways you can ping RTD to let us know your project has been updated. We have official support for GitHub, Bitbucket and GitLab, and anywhere else we have a generic post-commit hook that allows you to POST to a URL to get your documentation built.

Internationalization
--------------------
15 changes: 13 additions & 2 deletions docs/webhooks.rst
Original file line number Diff line number Diff line change
@@ -9,7 +9,7 @@ worked with push knows, pushing a doc update to your repo and watching it get
updated within seconds is an awesome feeling.

GitHub
---------
------

If your project is hosted on GitHub, you can easily add a hook that will rebuild
your docs whenever you push updates:
@@ -27,7 +27,7 @@ If you ever need to manually set the webhook on GitHub,
you can point it at ``https://readthedocs.org/github``.

Bitbucket
-----------
---------

If your project is hosted on Bitbucket, you can easily add a hook that will rebuild
your docs whenever you push updates:
@@ -40,6 +40,17 @@ your docs whenever you push updates:
If you ever need to manually set the webhook on Bitbucket,
you can point it at ``https://readthedocs.org/bitbucket``.

GitLab
------

If your project is hosted on GitLab, you can manually set the webhook on Gitlab and

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should be formatted as GitLab

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what do you mean here?

Copy link

@ghost ghost Oct 19, 2016

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

GitLab branding is to always call it GitLab.com (though I can't find their public documentation on that at this time).

However it's usually referred to as GitLab.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done.

point it at ``https://readthedocs.org/gitlab``:
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should this webhook be updated to reflect the new webhook end points added in #2433 ?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So, without a webhook management page, these webhooks aren't completely deprecated yet. Once that feature exists, this page will need to be updated to instead reference that page and drop information on manual set up. Likewise, we should add a deprecation notice on GitHub Services for RTD as well. I'll open an issue outlining the work that will go into this one -- fine to point to this webhook endpoint for now.


* Click the settings icon for your project
* Select "Webhooks"
* Enter the above URL, select "Push events" and "Enable SSL verification"
* Click "Add Webhook"

Others
------

6 changes: 6 additions & 0 deletions media/css/core.css
Original file line number Diff line number Diff line change
@@ -672,6 +672,12 @@ a.socialaccount-provider.github:before {
content: "\f09b";
}

div.project-import-remote form.import-connect-gitlab button:before,
a.socialaccount-provider.gitlab:before {
font-family: FontAwesome;
content: "\f1d3";
}

div.project-import-remote form.import-connect-bitbucket button:before,
a.socialaccount-provider.bitbucket:before,
a.socialaccount-provider.bitbucket_oauth2:before {
1 change: 1 addition & 0 deletions media/images/fa-bookmark.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
1 change: 1 addition & 0 deletions media/images/fa-users.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
6 changes: 5 additions & 1 deletion readthedocs/oauth/services/__init__.py
Original file line number Diff line number Diff line change
@@ -10,4 +10,8 @@
getattr(settings, 'OAUTH_BITBUCKET_SERVICE',
'readthedocs.oauth.services.bitbucket.BitbucketService'))

registry = [GitHubService, BitbucketService]
GitLabService = import_by_path(
getattr(settings, 'OAUTH_GITLAB_SERVICE',
'readthedocs.oauth.services.gitlab.GitLabService'))

registry = [GitHubService, BitbucketService, GitLabService]
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure this is required, @agjohnson can you please check.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is required, we use this pattern to override API actions on private resources on readthedocs.com.

235 changes: 235 additions & 0 deletions readthedocs/oauth/services/gitlab.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,235 @@
"""OAuth utility functions"""

import logging
import json
import re

from django.conf import settings
from requests.exceptions import RequestException
from allauth.socialaccount.models import SocialToken
from allauth.socialaccount.providers.gitlab.views import GitLabOAuth2Adapter
from urlparse import urljoin

from readthedocs.restapi.client import api

from ..models import RemoteOrganization, RemoteRepository
from .base import Service, DEFAULT_PRIVACY_LEVEL


log = logging.getLogger(__name__)


class GitLabService(Service):
"""Provider service for GitLab"""

adapter = GitLabOAuth2Adapter
url_pattern = re.compile(re.escape(adapter.provider_base_url))
default_avatar = {
'repo': urljoin(settings.MEDIA_URL, 'images/fa-bookmark.svg'),
'org': urljoin(settings.MEDIA_URL, 'images/fa-users.svg'),
}

def paginate(self, url, **kwargs):
"""Combines return from GitLab pagination. GitLab uses
LinkHeaders, see: http://www.w3.org/wiki/LinkHeader
:param url: start url to get the data from.
:param kwargs: optional parameters passed to .get() method
See http://doc.gitlab.com/ce/api/README.html#pagination

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This URL should be updated to https://docs.gitlab.com/ce/api/README.html#pagination

"""
resp = self.get_session().get(url, data=kwargs)
result = resp.json()
next_url = resp.links.get('next', {}).get('url')
if next_url:
result.extend(self.paginate(next_url, **kwargs))
return result

def sync(self):
"""Sync repositories from GitLab API"""
org = None
repos = self.paginate(
u'{url}/api/v3/projects'.format(url=self.adapter.provider_base_url),
per_page=100,
order_by='path',
sort='asc'
)
for repo in repos:
# Skip archived repositories
if repo.get('archived', False):
continue
if not org or org.slug != repo['namespace']['id']:
org = self.create_organization(repo['namespace'])

self.create_repository(repo, organization=org)

def create_repository(self, fields, privacy=DEFAULT_PRIVACY_LEVEL,
organization=None):
"""Update or create a repository from GitLab API response
:param fields: dictionary of response data from API
:param privacy: privacy level to support
:param organization: remote organization to associate with

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Organizations are called Groups in GitLab, not sure if that matters or if that'd be confusing for users.

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think Groups would be better here.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@connorshea, @destroyerofbuilds of course you're both right, but i had to adapt the given api and this api uses organization. My recommendation would be to get this into master and then discuss how to refactor the oauth-plugin system.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

organization in this case is our internal representation of the various providers names for orgs/groups/what have you. I don't think this usage needs to reflect GitLab's naming.

:type organization: RemoteOrganization
:rtype: RemoteRepository
"""
# See: http://doc.gitlab.com/ce/api/projects.html#projects

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

And this URL as well.

repo_is_public = fields['visibility_level'] == 20

def is_owned_by(owner_id):
return self.account.extra_data['id'] == owner_id

if privacy == 'private' or (repo_is_public and privacy == 'public'):
try:
repo = RemoteRepository.objects.get(
full_name=fields['name_with_namespace'],
users=self.user,
account=self.account,
)
except RemoteRepository.DoesNotExist:
repo = RemoteRepository.objects.create(
full_name=fields['name_with_namespace'],
account=self.account,
)
repo.users.add(self.user)

if repo.organization and repo.organization != organization:
log.debug('Not importing %s because mismatched orgs' %
fields['name'])
return None
else:
repo.organization = organization
repo.name = fields['name']
repo.full_name = fields['name_with_namespace']
repo.description = fields['description']
repo.ssh_url = fields['ssh_url_to_repo']
repo.html_url = fields['web_url']
repo.private = not fields['public']
repo.admin = not repo_is_public
repo.clone_url = (repo.admin and repo.ssh_url or
fields['http_url_to_repo'])
if not repo.admin and 'owner' in fields:
repo.admin = is_owned_by(fields['owner']['id'])
repo.vcs = 'git'
repo.account = self.account
owner = fields.get('owner') or {}
repo.avatar_url = (fields.get('avatar_url') or
owner.get('avatar_url') or
self.default_avatar['repo'])
repo.json = json.dumps(fields)
repo.save()
return repo
else:
log.info(
u'Not importing {0} because mismatched type: public={1}'.format(
fields['name_with_namespace'],
fields['public'],
)
)

def create_organization(self, fields):
"""Update or create remote organization from GitLab API response
:param fields: dictionary response of data from API
:rtype: RemoteOrganization
"""
try:
organization = RemoteOrganization.objects.get(
slug=fields.get('path'),
users=self.user,
account=self.account,
)
except RemoteOrganization.DoesNotExist:
organization = RemoteOrganization.objects.create(
slug=fields.get('path'),
account=self.account,
)
organization.users.add(self.user)

organization.name = fields.get('name')
organization.account = self.account
organization.url = u'{url}/{path}'.format(
url=self.adapter.provider_base_url, path=fields.get('path')
)
avatar = fields.get('avatar') or {}
if avatar.get('url'):
organization.avatar_url = u'{url}/{avatar}'.format(
url=self.adapter.provider_base_url,
avatar=avatar.get('url'),
)
else:
organization.avatar_url = self.default_avatar['org']
organization.json = json.dumps(fields)
organization.save()
return organization

def setup_webhook(self, project):
"""Set up GitLab project webhook for project
:param project: project to set up webhook for
:type project: Project
:returns: boolean based on webhook set up success
:rtype: bool
"""
session = self.get_session()

# See: http://doc.gitlab.com/ce/api/projects.html#add-project-hook

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ditto for this one.

data = json.dumps({
'id': 'readthedocs',
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe we should use PRODUCTION_DOMAIN here, to allow (in theory) more than one readthedocs webhooks.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

indeed, i assume this would fail or at least overwrite the existing webhook if a user set up a webhook with readthedocs.org and readthedocs.com -- or a private instance.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As stated in the documentation of GitLab, the ID of the project (or the NAMESPACE/PROJECT_NAME of a project) should be used.

I've tested it: you can set up multiple webhooks with the same ID.

'push_events': True,
'issues_events': False,
'merge_requests_events': False,
'note_events': False,
'tag_push_events': True,
'url': u'https://{0}/gitlab'.format(settings.PRODUCTION_DOMAIN),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In #2433 these webhook endpoints were deprecated. A new handler for GitLab will be required and this should be updated. We can either address this in the PR, or I'm also +1 on filing a new PR to address this change.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, uh, I already implemented this.

Nothing to see here.

})
resp = None
try:
repositories = RemoteRepository.objects.filter(
clone_url=project.vcs_repo().repo_url
)
assert repositories
repo_id = repositories[0].get_serialized()['id']
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To create the webhook we need id of repository in GitLab.

resp = session.post(
u'{url}/api/v3/projects/{repo_id}/hooks'.format(
url=self.adapter.provider_base_url,
repo_id=repo_id,
),
data=data,
headers={'content-type': 'application/json'}
)
if resp.status_code == 201:
log.info('GitLab webhook creation successful for project: %s', # noqa
project)
return (True, resp)
except (AssertionError, RemoteRepository.DoesNotExist) as ex:
log.error('GitLab remote repository not found', exc_info=ex)
except RequestException as ex:
pass
else:
ex = False

log.error('GitLab webhook creation failed for project: %s', # noqa
project, exc_info=ex)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This try block could shortened, it seems like there is extraneous handling of ex

return (False, resp)

@classmethod
def get_token_for_project(cls, project, force_local=False):
"""Get access token for project by iterating over project users"""
# TODO why does this only target GitHub?
if not getattr(settings, 'ALLOW_PRIVATE_REPOS', False):
return None
token = None
try:
if getattr(settings, 'DONT_HIT_DB', True) and not force_local:
token = api.project(project.pk).token().get()['token']
else:
for user in project.users.all():
tokens = SocialToken.objects.filter(
account__user=user,
app__provider=cls.adapter.provider_id)
if tokens.exists():
token = tokens[0].token
except Exception:
log.error('Failed to get token for user', exc_info=True)
return token
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not entirely sure this is required, or it could be moved to the base class. https://github.com/rtfd/readthedocs.org/blob/e4958838a512b095d0bc8cdf8617c30cc9e489d4/readthedocs/restapi/views/model_views.py#L83-L90 was only targeting Github before refactoring oauth services.

135 changes: 133 additions & 2 deletions readthedocs/rtd_tests/tests/test_oauth.py
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
from django.test import TestCase

from django.contrib.auth.models import User
from allauth.socialaccount.models import SocialToken
from mock import Mock

from readthedocs.projects import constants
from readthedocs.projects.models import Project

from readthedocs.oauth.services import GitHubService, BitbucketService
from readthedocs.oauth.models import RemoteRepository, RemoteOrganization
from readthedocs.oauth.services import GitHubService, BitbucketService, GitLabService


class GitHubOAuthTests(TestCase):
@@ -279,3 +280,133 @@ def test_import_with_no_token(self):
'''User without a Bitbucket SocialToken does not return a service'''
services = BitbucketService.for_user(self.user)
self.assertEqual(services, [])


class GitLabOAuthTests(TestCase):

fixtures = ["eric", "test_data"]

repo_response_data = {
'forks_count': 12,
'container_registry_enabled': None,
'web_url': 'https://gitlab.com/testorga/testrepo',
'wiki_enabled': True,
'public_builds': True,
'id': 2,
'merge_requests_enabled': True,
'archived': False,
'snippets_enabled': False,
'http_url_to_repo': 'https://gitlab.com/testorga/testrepo.git',
'namespace': {
'share_with_group_lock': False,
'name': 'Test Orga',
'created_at': '2014-07-11T13:38:53.510Z',
'description': '',
'updated_at': '2014-07-11T13:38:53.510Z',
'avatar': {
'url': None
},
'path': 'testorga',
'visibility_level': 20,
'id': 5,
'owner_id': None
},
'star_count': 0,
'avatar_url': 'http://placekitten.com/50/50',
'issues_enabled': True,
'path_with_namespace': 'testorga/testrepo',
'public': True,
'description': 'Test Repo',
'default_branch': 'master',
'ssh_url_to_repo': 'git@gitlab.com:testorga/testrepo.git',
'path': 'testrepo',
'visibility_level': 20,
'permissions': {
'group_access': {
'notification_level': 3,
'access_level': 40
},
'project_access': None
},
'open_issues_count': 2,
'last_activity_at': '2016-03-01T09:22:34.344Z',
'name': 'testrepo',
'name_with_namespace': 'testorga / testrepo',
'created_at': '2015-11-02T13:52:42.821Z',
'builds_enabled': True,
'creator_id': 5,
'shared_runners_enabled': True,
'tag_list': []
}

def setUp(self):
self.client.login(username='eric', password='test')
self.user = User.objects.get(pk=1)
self.project = Project.objects.get(slug='pip')
self.org = RemoteOrganization.objects.create(slug='testorga', json='')
self.privacy = self.project.version_privacy_level
self.service = GitLabService(user=self.user, account=None)

def get_private_repo_data(self):
"""Manipulate repo response data to get private repo data."""
data = self.repo_response_data.copy()
data.update({
'visibility_level': 10,
'public': False,
})
return data

def test_make_project_pass(self):
repo = self.service.create_repository(
self.repo_response_data, organization=self.org, privacy=self.privacy)
self.assertIsInstance(repo, RemoteRepository)
self.assertEqual(repo.name, 'testrepo')
self.assertEqual(repo.full_name, 'testorga / testrepo')
self.assertEqual(repo.description, 'Test Repo')
self.assertEqual(repo.avatar_url, 'http://placekitten.com/50/50')
self.assertIn(self.user, repo.users.all())
self.assertEqual(repo.organization, self.org)
self.assertEqual(repo.clone_url, 'https://gitlab.com/testorga/testrepo.git')
self.assertEqual(repo.ssh_url, 'git@gitlab.com:testorga/testrepo.git')
self.assertEqual(repo.html_url, 'https://gitlab.com/testorga/testrepo')

def test_make_private_project_fail(self):
repo = self.service.create_repository(
self.get_private_repo_data(), organization=self.org, privacy=self.privacy)
self.assertIsNone(repo)

def test_make_private_project_success(self):
repo = self.service.create_repository(
self.get_private_repo_data(), organization=self.org, privacy=constants.PRIVATE)
self.assertIsInstance(repo, RemoteRepository)
self.assertTrue(repo.private, True)

def test_make_organization(self):
org = self.service.create_organization(self.repo_response_data['namespace'])
self.assertIsInstance(org, RemoteOrganization)
self.assertEqual(org.slug, 'testorga')
self.assertEqual(org.name, 'Test Orga')
self.assertEqual(org.avatar_url, '/media/images/fa-users.svg')
self.assertEqual(org.url, 'https://gitlab.com/testorga')

def test_sync_skip_archived_repo(self):
data = self.repo_response_data
data['archived'] = True
create_repo_mock = Mock()
create_orga_mock = Mock()
setattr(self.service, 'paginate', Mock(return_value=[data]))
setattr(self.service, 'create_repository', create_repo_mock)
setattr(self.service, 'create_organization', create_orga_mock)
self.service.sync()
self.assertFalse(create_repo_mock.called)
self.assertFalse(create_orga_mock.called)

def test_sync_create_repo_and_orga(self):
create_repo_mock = Mock()
create_orga_mock = Mock(return_value=self.org)
setattr(self.service, 'paginate', Mock(return_value=[self.repo_response_data]))
setattr(self.service, 'create_repository', create_repo_mock)
setattr(self.service, 'create_organization', create_orga_mock)
self.service.sync()
create_repo_mock.assert_called_once_with(self.repo_response_data, organization=self.org)
create_orga_mock.assert_called_once_with(self.repo_response_data['namespace'])
1 change: 1 addition & 0 deletions readthedocs/settings/base.py
Original file line number Diff line number Diff line change
@@ -108,6 +108,7 @@ def INSTALLED_APPS(self): # noqa
'allauth.account',
'allauth.socialaccount',
'allauth.socialaccount.providers.github',
'allauth.socialaccount.providers.gitlab',
'allauth.socialaccount.providers.bitbucket',
'allauth.socialaccount.providers.bitbucket_oauth2',
]