##// END OF EJS Templates
setup: change url to github
setup: change url to github

File last commit:

r153:32f4b641
r196:472d1df0 master
Show More
bitbucket.py
188 lines | 6.0 KiB | text/x-python | PythonLexer
# -*- coding: utf-8 -*-
# Copyright 2010 - 2017 RhodeCode GmbH and the AppEnlight project authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import requests
from requests_oauthlib import OAuth1
from appenlight.models.integrations import IntegrationBase, IntegrationException
_ = str
class NotFoundException(Exception):
pass
class BitbucketIntegration(IntegrationBase):
__mapper_args__ = {"polymorphic_identity": "bitbucket"}
front_visible = True
as_alert_channel = False
supports_report_alerting = False
action_notification = True
integration_action = "Add issue to Bitbucket"
@classmethod
def create_client(cls, request, user_name=None, repo_name=None):
"""
Creates REST client that can authenticate to specific repo
uses auth tokens for current request user
"""
config = request.registry.settings
token = None
secret = None
for identity in request.user.external_identities:
if identity.provider_name == "bitbucket":
token = identity.access_token
secret = identity.token_secret
break
if not token:
raise IntegrationException("No valid auth token present for this service")
client = BitbucketClient(
token,
secret,
user_name,
repo_name,
config["authomatic.pr.bitbucket.key"],
config["authomatic.pr.bitbucket.secret"],
)
return client
class BitbucketClient(object):
api_url = "https://bitbucket.org/api/1.0"
repo_type = "bitbucket"
def __init__(
self,
token,
secret,
owner,
repo_name,
bitbucket_consumer_key,
bitbucket_consumer_secret,
):
self.access_token = token
self.token_secret = secret
self.owner = owner
self.repo_name = repo_name
self.bitbucket_consumer_key = bitbucket_consumer_key
self.bitbucket_consumer_secret = bitbucket_consumer_secret
possible_keys = {
"status": [
"new",
"open",
"resolved",
"on hold",
"invalid",
"duplicate",
"wontfix",
],
"priority": ["trivial", "minor", "major", "critical", "blocker"],
"kind": ["bug", "enhancement", "proposal", "task"],
}
def get_statuses(self):
"""Gets list of possible item statuses"""
return self.possible_keys["status"]
def get_priorities(self):
"""Gets list of possible item statuses"""
return self.possible_keys["priority"]
def make_request(self, url, method="get", data=None, headers=None):
"""
Performs HTTP request to bitbucket
"""
auth = OAuth1(
self.bitbucket_consumer_key,
self.bitbucket_consumer_secret,
self.access_token,
self.token_secret,
)
try:
resp = getattr(requests, method)(url, data=data, auth=auth, timeout=10)
except Exception as e:
raise IntegrationException(
_("Error communicating with Bitbucket: %s") % (e,)
)
if resp.status_code == 401:
raise IntegrationException(_("You are not authorized to access this repo"))
elif resp.status_code == 404:
raise IntegrationException(_("User or repo name are incorrect"))
elif resp.status_code not in [200, 201]:
raise IntegrationException(
_("Bitbucket response_code: %s") % resp.status_code
)
try:
return resp.json()
except Exception as e:
raise IntegrationException(
_("Error decoding response from Bitbucket: %s") % (e,)
)
def get_assignees(self):
"""Gets list of possible assignees"""
url = "%(api_url)s/privileges/%(owner)s/%(repo_name)s" % {
"api_url": self.api_url,
"owner": self.owner,
"repo_name": self.repo_name,
}
data = self.make_request(url)
results = [{"user": self.owner, "name": "(Repo owner)"}]
if data:
for entry in data:
results.append(
{
"user": entry["user"]["username"],
"name": entry["user"].get("display_name"),
}
)
return results
def create_issue(self, form_data):
"""
Sends creates a new issue in tracker using REST call
"""
url = "%(api_url)s/repositories/%(owner)s/%(repo_name)s/issues/" % {
"api_url": self.api_url,
"owner": self.owner,
"repo_name": self.repo_name,
}
payload = {
"title": form_data["title"],
"content": form_data["content"],
"kind": form_data["kind"],
"priority": form_data["priority"],
"responsible": form_data["responsible"],
}
data = self.make_request(url, "post", payload)
f_args = {
"owner": self.owner,
"repo_name": self.repo_name,
"issue_id": data["local_id"],
}
web_url = (
"https://bitbucket.org/%(owner)s/%(repo_name)s"
"/issue/%(issue_id)s/issue-title" % f_args
)
to_return = {
"id": data["local_id"],
"resource_url": data["resource_uri"],
"web_url": web_url,
}
return to_return