2019-08-24 05:07:16 +00:00
|
|
|
import logging
|
|
|
|
logging.basicConfig(
|
|
|
|
format='%(asctime)s - %(name)s - %(levelname)s - %(message)s',
|
2019-08-28 04:13:02 +00:00
|
|
|
level=logging.DEBUG)
|
2019-08-24 05:07:16 +00:00
|
|
|
|
2019-08-23 08:23:48 +00:00
|
|
|
import requests
|
|
|
|
|
|
|
|
API_TOPSTORIES = lambda x: 'https://hacker-news.firebaseio.com/v0/topstories.json'
|
|
|
|
API_ITEM = lambda x : 'https://hn.algolia.com/api/v1/items/{}'.format(x)
|
|
|
|
|
|
|
|
SITE_LINK = lambda x : 'https://news.ycombinator.com/item?id={}'.format(x)
|
|
|
|
SITE_AUTHOR_LINK = lambda x : 'https://news.ycombinator.com/user?id={}'.format(x)
|
|
|
|
|
|
|
|
def api(route, ref=None):
|
2019-08-24 05:07:16 +00:00
|
|
|
try:
|
|
|
|
r = requests.get(route(ref), timeout=5)
|
|
|
|
if r.status_code != 200:
|
2019-08-28 04:13:02 +00:00
|
|
|
raise Exception('Bad response code ' + str(r.status_code))
|
2019-08-24 05:07:16 +00:00
|
|
|
return r.json()
|
|
|
|
except BaseException as e:
|
|
|
|
logging.error('Problem hitting hackernews API: {}'.format(str(e)))
|
|
|
|
return False
|
2019-08-23 08:23:48 +00:00
|
|
|
|
|
|
|
def feed():
|
2019-08-28 04:13:02 +00:00
|
|
|
return api(API_TOPSTORIES) or []
|
2019-08-23 08:23:48 +00:00
|
|
|
|
|
|
|
def comment(i):
|
2019-08-30 06:23:14 +00:00
|
|
|
if 'author' not in i:
|
|
|
|
return False
|
|
|
|
|
2019-08-23 08:23:48 +00:00
|
|
|
c = {}
|
|
|
|
c['author'] = i.get('author', '')
|
|
|
|
c['score'] = i.get('points', 0)
|
|
|
|
c['date'] = i.get('created_at_i', 0)
|
|
|
|
c['text'] = i.get('text', '')
|
|
|
|
c['comments'] = [comment(j) for j in i['children']]
|
2019-08-30 06:23:14 +00:00
|
|
|
c['comments'] = list(filter(bool, c['comments']))
|
2019-08-23 08:23:48 +00:00
|
|
|
return c
|
|
|
|
|
|
|
|
def comment_count(i):
|
|
|
|
alive = 1 if i['author'] else 0
|
|
|
|
return sum([comment_count(c) for c in i['comments']]) + alive
|
|
|
|
|
|
|
|
def story(ref):
|
|
|
|
r = api(API_ITEM, ref)
|
2019-08-24 05:07:16 +00:00
|
|
|
if not r: return False
|
2019-08-23 08:23:48 +00:00
|
|
|
|
|
|
|
if 'deleted' in r:
|
|
|
|
return False
|
|
|
|
elif r.get('type', '') != 'story':
|
|
|
|
return False
|
|
|
|
|
|
|
|
s = {}
|
|
|
|
s['author'] = r.get('author', '')
|
|
|
|
s['author_link'] = SITE_AUTHOR_LINK(r.get('author', ''))
|
|
|
|
s['score'] = r.get('points', 0)
|
|
|
|
s['date'] = r.get('created_at_i', 0)
|
|
|
|
s['title'] = r.get('title', '')
|
|
|
|
s['link'] = SITE_LINK(ref)
|
|
|
|
s['url'] = r.get('url', '')
|
|
|
|
s['comments'] = [comment(i) for i in r['children']]
|
2019-08-30 06:23:14 +00:00
|
|
|
s['comments'] = list(filter(bool, s['comments']))
|
2019-08-23 08:23:48 +00:00
|
|
|
s['num_comments'] = comment_count(s) - 1
|
|
|
|
|
|
|
|
if 'text' in r and r['text']:
|
|
|
|
s['text'] = r['text']
|
|
|
|
|
|
|
|
return s
|
|
|
|
|
2019-08-28 04:13:02 +00:00
|
|
|
# scratchpad so I can quickly develop the parser
|
2019-08-23 08:23:48 +00:00
|
|
|
if __name__ == '__main__':
|
2019-08-28 04:13:02 +00:00
|
|
|
#print(feed())
|
|
|
|
#print(story(20763961))
|
|
|
|
print(story(20802050))
|