Skip to content

Commit 2a9df37

Browse files
committed
Added endpoints and functionalty of Gist API
1 parent 749fdfa commit 2a9df37

File tree

2 files changed

+86
-28
lines changed

2 files changed

+86
-28
lines changed

gistapi/gistapi.py

+78-19
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111

1212
import requests
1313
from flask import Flask, jsonify, request
14+
import re
1415

1516

1617
# *The* app object
@@ -39,13 +40,27 @@ def gists_for_user(username):
3940
"""
4041
gists_url = 'https://api.github.com/users/{username}/gists'.format(
4142
username=username)
42-
response = requests.get(gists_url)
43+
4344
# BONUS: What failures could happen?
45+
# Failures like username not found or API limit reached can happen
46+
# These are handled by following if block.
4447
# BONUS: Paging? How does this work for users with tons of gists?
48+
# If a user has tons of gists. The trucated flag will be set to true.
49+
50+
response = requests.get(gists_url)
51+
if response.status_code != 200:
52+
return None
4553

4654
return response.json()
4755

4856

57+
def valid_arguments(arguments):
58+
if set(['username', 'pattern']) == set(arguments) and isinstance(arguments['username'], str) and \
59+
isinstance(arguments['pattern'], str):
60+
return True
61+
return False
62+
63+
4964
@app.route("/api/v1/search", methods=['POST'])
5065
def search():
5166
"""Provides matches for a single pattern across a single users gists.
@@ -58,26 +73,70 @@ def search():
5873
object contains the list of matches along with a 'status' key
5974
indicating any failure conditions.
6075
"""
61-
post_data = request.get_json()
62-
# BONUS: Validate the arguments?
63-
64-
username = post_data['username']
65-
pattern = post_data['pattern']
6676

77+
post_data = request.json
6778
result = {}
68-
gists = gists_for_user(username)
69-
# BONUS: Handle invalid users?
70-
71-
for gist in gists:
72-
# REQUIRED: Fetch each gist and check for the pattern
73-
# BONUS: What about huge gists?
74-
# BONUS: Can we cache results in a datastore/db?
75-
pass
76-
77-
result['status'] = 'success'
78-
result['username'] = username
79-
result['pattern'] = pattern
80-
result['matches'] = []
79+
status = True
80+
warnings = []
81+
gists = None
82+
83+
# BONUS: Validate the arguments?
84+
if not valid_arguments(post_data):
85+
status = False
86+
87+
if status:
88+
username = post_data['username']
89+
pattern = post_data['pattern']
90+
91+
result['matches'] = []
92+
gists = gists_for_user(username)
93+
# BONUS: Handle invalid users?
94+
if gists is not None:
95+
for gist in gists:
96+
# REQUIRED: Fetch each gist and check for the pattern
97+
# Completed
98+
# BONUS: What about huge gists?
99+
# If gists are greater than 300, added warning
100+
# BONUS: Can we cache results in a datastore/db?
101+
# Skipped
102+
103+
response = requests.get(gist['url'])
104+
105+
if gist['truncated']:
106+
warnings.append(f"Gist({gist['id']}): More than 300 files")
107+
108+
# Search if not truncated
109+
if re.search(pattern, response.content.decode('utf-8')):
110+
result['matches'].append(f"https://gist.github.com/{username}/{gist['id']}")
111+
continue
112+
113+
if(response.status_code != 200):
114+
# Skipping if API limit reached
115+
break
116+
117+
gist_json = response.json()
118+
files = gist_json['files']
119+
120+
# Search if truncated
121+
for key in files:
122+
if files[key]['truncated']:
123+
file_response = requests.get(files[key]["raw_url"])
124+
if(file_response.status_code != 200):
125+
# Skipping if API limit reached
126+
break
127+
if re.search(pattern, file_response.content.decode("utf-8")):
128+
result['matches'].append(f"https://gist.github.com/{username}/{gist['id']}")
129+
result['status'] = 'success'
130+
else:
131+
result['status'] = 'failure'
132+
133+
if status:
134+
result['username'] = username
135+
result['pattern'] = pattern
136+
if len(warnings) > 0:
137+
result['warnings'] = warnings
138+
else:
139+
result['status'] = 'arguments not valid'
81140

82141
return jsonify(result)
83142

tests/test_gistapi.py

+8-9
Original file line numberDiff line numberDiff line change
@@ -9,17 +9,17 @@
99

1010
@pytest.fixture
1111
def client(request):
12-
#db_fd, gistapi.app.config['DATABASE'] = tempfile.mkstemp()
12+
# db_fd, gistapi.app.config['DATABASE'] = tempfile.mkstemp()
1313
gistapi.app.config['TESTING'] = True
1414
client = gistapi.app.test_client()
1515

16-
#with gistapi.app.app_context():
16+
# with gistapi.app.app_context():
1717
# gistapi.init_db()
18-
19-
#def teardown():
18+
19+
# def teardown():
2020
# os.close(db_fd)
2121
# os.unlink(flaskr.app.config['DATABASE'])
22-
#request.addfinalizer(teardown)
22+
# request.addfinalizer(teardown)
2323

2424
return client
2525

@@ -33,13 +33,12 @@ def test_ping(client):
3333
def test_search(client):
3434
"""Start with a passing test."""
3535
post_data = {'username': 'justdionysus', 'pattern': 'TerbiumLabsChallenge_[0-9]+'}
36-
rv = client.post('/api/v1/search',
36+
rv = client.post('/api/v1/search',
3737
data=json.dumps(post_data),
38-
headers={'content-type':'application/json'})
38+
headers={'content-type': 'application/json'})
3939
result_dict = json.loads(rv.data.decode('utf-8'))
40-
expected_dict = {'status': 'success',
40+
expected_dict = {'status': 'success',
4141
'username': 'justdionysus',
4242
'pattern': 'TerbiumLabsChallenge_[0-9]+',
4343
'matches': ['https://gist.github.com/justdionysus/6b2972aa971dd605f524']}
4444
assert result_dict == expected_dict
45-

0 commit comments

Comments
 (0)