##// END OF EJS Templates
try to fallback to pysqlite2.dbapi2 as sqlite3...
try to fallback to pysqlite2.dbapi2 as sqlite3 for situations where Python is built without sqlite3 and can not easily be changed, it would be nice to still be able to get history with pysqlite2.

File last commit:

r7866:26fdbf6d
r8934:09c5f96b
Show More
gh_api.py
201 lines | 6.4 KiB | text/x-python | PythonLexer
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694 """Functions for Github authorisation."""
from __future__ import print_function
try:
input = raw_input
except NameError:
pass
MinRK
enable posting GitHub downloads
r7761 import os
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694 import requests
import getpass
import json
# Keyring stores passwords by a 'username', but we're not storing a username and
# password
fake_username = 'ipython_tools'
Thomas Kluyver
Refactor test_pr.py to a more coherent design.
r7863 class Obj(dict):
"""Dictionary with attribute access to names."""
def __getattr__(self, name):
try:
return self[name]
except KeyError:
raise AttributeError(name)
def __setattr__(self, name, val):
self[name] = val
Thomas Kluyver
Add option to test without posting results.
r6711 token = None
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694 def get_auth_token():
Thomas Kluyver
Add option to test without posting results.
r6711 global token
if token is not None:
return token
import keyring
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694 token = keyring.get_password('github', fake_username)
if token is not None:
return token
print("Please enter your github username and password. These are not "
"stored, only used to get an oAuth token. You can revoke this at "
"any time on Github.")
user = input("Username: ")
pw = getpass.getpass("Password: ")
auth_request = {
"scopes": [
Thomas Kluyver
Add option to test without posting results.
r6711 "public_repo",
"gist"
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694 ],
Thomas Kluyver
Add option to test without posting results.
r6711 "note": "IPython tools",
"note_url": "https://github.com/ipython/ipython/tree/master/tools",
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694 }
response = requests.post('https://api.github.com/authorizations',
auth=(user, pw), data=json.dumps(auth_request))
response.raise_for_status()
token = json.loads(response.text)['token']
keyring.set_password('github', fake_username, token)
return token
Thomas Kluyver
Add option to test without posting results.
r6711 def make_auth_header():
return {'Authorization': 'token ' + get_auth_token()}
def post_issue_comment(project, num, body):
url = 'https://api.github.com/repos/{project}/issues/{num}/comments'.format(project=project, num=num)
payload = json.dumps({'body': body})
r = requests.post(url, data=payload, headers=make_auth_header())
def post_gist(content, description='', filename='file', auth=False):
"""Post some text to a Gist, and return the URL."""
post_data = json.dumps({
"description": description,
"public": True,
"files": {
filename: {
"content": content
}
}
}).encode('utf-8')
headers = make_auth_header() if auth else {}
response = requests.post("https://api.github.com/gists", data=post_data, headers=headers)
response.raise_for_status()
response_data = json.loads(response.text)
return response_data['html_url']
Thomas Kluyver
PR tester can now post results to Github as a comment.
r6694
Thomas Kluyver
Drop support for deceased Github API v2
r7866 def get_pull_request(project, num):
Matthias BUSSONNIER
some fixs
r7448 """get pull request info by number
"""
Thomas Kluyver
Drop support for deceased Github API v2
r7866 url = "https://api.github.com/repos/{project}/pulls/{num}".format(project=project, num=num)
Thomas Kluyver
Add option to test without posting results.
r6711 response = requests.get(url)
response.raise_for_status()
Thomas Kluyver
Refactor test_pr.py to a more coherent design.
r7863 return json.loads(response.text, object_hook=Obj)
Matthias BUSSONNIER
merge pull request by number...
r7254
Thomas Kluyver
Drop support for deceased Github API v2
r7866 def get_pulls_list(project):
Matthias BUSSONNIER
some fixs
r7448 """get pull request list
"""
Thomas Kluyver
Drop support for deceased Github API v2
r7866 url = "https://api.github.com/repos/{project}/pulls".format(project=project)
Matthias BUSSONNIER
merge pull request by number...
r7254 response = requests.get(url)
response.raise_for_status()
Thomas Kluyver
Add option to test without posting results.
r6711 return json.loads(response.text)
MinRK
enable posting GitHub downloads
r7761
# encode_multipart_formdata is from urllib3.filepost
# The only change is to iter_fields, to enforce S3's required key ordering
def iter_fields(fields):
fields = fields.copy()
for key in ('key', 'acl', 'Filename', 'success_action_status', 'AWSAccessKeyId',
'Policy', 'Signature', 'Content-Type', 'file'):
yield (key, fields.pop(key))
for (k,v) in fields.items():
yield k,v
def encode_multipart_formdata(fields, boundary=None):
"""
Encode a dictionary of ``fields`` using the multipart/form-data mime format.
:param fields:
Dictionary of fields or list of (key, value) field tuples. The key is
treated as the field name, and the value as the body of the form-data
bytes. If the value is a tuple of two elements, then the first element
is treated as the filename of the form-data section.
Field names and filenames must be unicode.
:param boundary:
If not specified, then a random boundary will be generated using
:func:`mimetools.choose_boundary`.
"""
# copy requests imports in here:
from io import BytesIO
from requests.packages.urllib3.filepost import (
choose_boundary, six, writer, b, get_content_type
)
body = BytesIO()
if boundary is None:
boundary = choose_boundary()
for fieldname, value in iter_fields(fields):
body.write(b('--%s\r\n' % (boundary)))
if isinstance(value, tuple):
filename, data = value
writer(body).write('Content-Disposition: form-data; name="%s"; '
'filename="%s"\r\n' % (fieldname, filename))
body.write(b('Content-Type: %s\r\n\r\n' %
(get_content_type(filename))))
else:
data = value
writer(body).write('Content-Disposition: form-data; name="%s"\r\n'
% (fieldname))
body.write(b'Content-Type: text/plain\r\n\r\n')
if isinstance(data, int):
data = str(data) # Backwards compatibility
if isinstance(data, six.text_type):
writer(body).write(data)
else:
body.write(data)
body.write(b'\r\n')
body.write(b('--%s--\r\n' % (boundary)))
content_type = b('multipart/form-data; boundary=%s' % boundary)
return body.getvalue(), content_type
def post_download(project, filename, name=None, description=""):
"""Upload a file to the GitHub downloads area"""
if name is None:
name = os.path.basename(filename)
with open(filename, 'rb') as f:
filedata = f.read()
url = "https://api.github.com/repos/{project}/downloads".format(project=project)
payload = json.dumps(dict(name=name, size=len(filedata),
description=description))
response = requests.post(url, data=payload, headers=make_auth_header())
response.raise_for_status()
reply = json.loads(response.content)
s3_url = reply['s3_url']
fields = dict(
key=reply['path'],
acl=reply['acl'],
success_action_status=201,
Filename=reply['name'],
AWSAccessKeyId=reply['accesskeyid'],
Policy=reply['policy'],
Signature=reply['signature'],
file=(reply['name'], filedata),
)
fields['Content-Type'] = reply['mime_type']
data, content_type = encode_multipart_formdata(fields)
s3r = requests.post(s3_url, data=data, headers={'Content-Type': content_type})
return s3r