11
11
12
12
import requests
13
13
from flask import Flask , jsonify , request
14
+ import re
14
15
15
16
16
17
# *The* app object
@@ -39,13 +40,27 @@ def gists_for_user(username):
39
40
"""
40
41
gists_url = 'https://api.github.com/users/{username}/gists' .format (
41
42
username = username )
42
- response = requests . get ( gists_url )
43
+
43
44
# 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.
44
47
# 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
45
53
46
54
return response .json ()
47
55
48
56
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
+
49
64
@app .route ("/api/v1/search" , methods = ['POST' ])
50
65
def search ():
51
66
"""Provides matches for a single pattern across a single users gists.
@@ -58,26 +73,70 @@ def search():
58
73
object contains the list of matches along with a 'status' key
59
74
indicating any failure conditions.
60
75
"""
61
- post_data = request .get_json ()
62
- # BONUS: Validate the arguments?
63
-
64
- username = post_data ['username' ]
65
- pattern = post_data ['pattern' ]
66
76
77
+ post_data = request .json
67
78
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'
81
140
82
141
return jsonify (result )
83
142
0 commit comments