nasg/archive.py

1100 lines
33 KiB
Python
Raw Normal View History

2017-12-17 17:37:32 +00:00
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# vim: set fileencoding=utf-8 :
__author__ = "Peter Molnar"
2018-04-30 20:44:04 +01:00
__copyright__ = "Copyright 2018, Peter Molnar"
2017-12-17 17:37:32 +00:00
__license__ = "GPLv3"
2018-04-30 20:44:04 +01:00
__version__ = "2.2.0"
2017-12-17 17:37:32 +00:00
__maintainer__ = "Peter Molnar"
2018-04-30 20:44:04 +01:00
__email__ = "mail@petermolnar.net"
2017-12-17 17:37:32 +00:00
__status__ = "Production"
"""
silo archiver module of NASG
Copyright (C) 2017 Peter Molnar
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software Foundation,
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
"""
import os
import json
import requests
import glob
import logging
import shutil
import subprocess
import imghdr
import arrow
2017-11-01 13:19:39 +00:00
import csv
2017-11-30 16:56:47 +00:00
import re
2017-10-27 15:56:05 +01:00
from requests_oauthlib import OAuth1Session
from requests_oauthlib import oauth1_session
from requests_oauthlib import OAuth2Session
from requests_oauthlib import oauth2_session
from oauthlib.oauth2 import BackendApplicationClient
2017-10-27 15:56:05 +01:00
import shared
2017-11-10 16:04:05 +00:00
2017-11-01 13:19:39 +00:00
class LastFM(object):
url = 'http://ws.audioscrobbler.com/2.0/'
2017-11-10 16:04:05 +00:00
2017-11-01 13:19:39 +00:00
def __init__(self):
self.service = 'lastfm'
self.target = shared.config.get("api_%s" % self.service, 'logfile')
self.params = {
'method': 'user.getrecenttracks',
'user': shared.config.get("api_%s" % self.service, 'username'),
'api_key': shared.config.get("api_%s" % self.service, 'api_key'),
'format': 'json',
'limit': '200'
}
if os.path.isfile(self.target):
mtime = os.path.getmtime(self.target)
self.params.update({'from': mtime})
def hash2flat(self, data):
time = int(data.get('date').get('uts'))
r = {
'date': arrow.get(time).format(shared.ARROWFORMAT['iso']),
'artist': data.get('artist').get('#text'),
'album': data.get('album').get('#text'),
2018-06-01 10:48:32 +01:00
'title': data.get('name')
2018-06-08 10:17:57 +01:00
# 'title_mbid': data.get('mbid'),
# 'artist_mbid': data.get('artist').get('mbid'),
# 'album_mbid': data.get('album').get('mbid'),
2017-11-01 13:19:39 +00:00
}
return (time, r)
def getpaged(self, pagenum):
logging.info('requesting page #%d of paginated results', pagenum)
self.params.update({
'page': pagenum
})
r = requests.get(
self.url,
params=self.params
)
parsed = json.loads(r.text).get('recenttracks', {}).get('track', [])
return parsed
def run(self):
2017-11-10 16:04:05 +00:00
r = requests.get(self.url, params=self.params)
2017-11-01 13:19:39 +00:00
js = json.loads(r.text)
js = js.get('recenttracks', {})
unordered = js.get('track', [])
ordered = {}
total = int(js.get('@attr').get('totalPages'))
current = int(js.get('@attr').get('page'))
cntr = total - current
while cntr > 0:
current = current + 1
paged = self.getpaged(current)
unordered = unordered + paged
cntr = total - current
for track in unordered:
# happens with nowplaying tracks
if 'date' not in track:
continue
time, data = self.hash2flat(track)
ordered[time] = data
# no results
if not len(ordered):
return
ordered = sorted(ordered.items())
with open(self.target, 'a') as f:
fieldnames = ordered[0][1].keys()
writer = csv.DictWriter(f, fieldnames=fieldnames)
# only write csv header once, when the file is first created
if 'from' not in self.params:
writer.writeheader()
for time, track in ordered:
writer.writerow(track)
class Favs(object):
def __init__(self, confgroup):
self.confgroup = confgroup
@property
def lastpulled(self):
mtime = 0
d = os.path.join(
shared.config.get('archive', 'favorite'),
"%s-*" % self.confgroup
)
files = glob.glob(d)
for f in files:
ftime = int(os.path.getmtime(f))
if ftime > mtime:
mtime = ftime
mtime = mtime + 1
2017-11-30 16:56:47 +00:00
logging.debug("last fav timestamp: %s", mtime)
return mtime
class FlickrFavs(Favs):
url = 'https://api.flickr.com/services/rest/'
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('flickr')
self.get_uid()
self.params = {
'method': 'flickr.favorites.getList',
'api_key': shared.config.get('api_flickr', 'api_key'),
'user_id': self.uid,
2017-11-10 16:04:05 +00:00
'extras': ','.join([
2017-10-27 15:56:05 +01:00
'description',
'geo',
'tags',
'owner_name',
'date_upload',
'url_o',
'url_k',
'url_h',
'url_b',
'url_c',
'url_z',
]),
2017-11-10 16:04:05 +00:00
'per_page': 500, # maximim
'format': 'json',
'nojsoncallback': '1',
'min_fave_date': self.lastpulled
}
def get_uid(self):
params = {
'method': 'flickr.people.findByUsername',
'api_key': shared.config.get('api_flickr', 'api_key'),
'format': 'json',
'nojsoncallback': '1',
'username': shared.config.get('api_flickr', 'username'),
}
r = requests.get(
self.url,
params=params
)
parsed = json.loads(r.text)
self.uid = parsed.get('user', {}).get('id')
def getpaged(self, offset):
logging.info('requesting page #%d of paginated results', offset)
self.params.update({
'page': offset
})
r = requests.get(
self.url,
params=self.params
)
parsed = json.loads(r.text)
return parsed.get('photos', {}).get('photo', [])
def run(self):
2017-11-10 16:04:05 +00:00
r = requests.get(self.url, params=self.params)
js = json.loads(r.text)
js = js.get('photos', {})
photos = js.get('photo', [])
total = int(js.get('pages', 1))
current = int(js.get('page', 1))
cntr = total - current
while cntr > 0:
current = current + 1
paged = self.getpaged(current)
photos = photos + paged
cntr = total - current
for photo in photos:
fav = FlickrFav(photo)
if not fav.exists:
fav.run()
2017-11-10 16:04:05 +00:00
# fav.fix_extension()
class FivehpxFavs(Favs):
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('500px')
self.params = {
'consumer_key': shared.config.get('api_500px', 'api_key'),
2017-11-10 16:04:05 +00:00
'rpp': 100, # maximum
'image_size': 4,
'include_tags': 1,
'include_geo': 1,
'sort': 'created_at',
'sort_direction': 'desc'
}
self.oauth = FivehpxOauth()
self.uid = None
self.galid = None
def get_uid(self):
r = self.oauth.request(
'https://api.500px.com/v1/users',
params={}
)
js = json.loads(r.text)
self.uid = js.get('user', {}).get('id')
def get_favgalid(self):
r = self.oauth.request(
'https://api.500px.com/v1/users/%s/galleries' % (self.uid),
params={
2017-11-10 16:04:05 +00:00
'kinds': 5 # see https://github.com/500px/api-documentation/blob/master/basics/formats_and_terms.md#gallery-kinds
}
)
js = json.loads(r.text)
g = js.get('galleries', []).pop()
self.galid = g.get('id')
@property
def url(self):
return 'https://api.500px.com/v1/users/%s/galleries/%s/items' % (
self.uid,
self.galid
)
def getpaged(self, offset):
logging.info('requesting page #%d of paginated results', offset)
self.params.update({
'page': offset
})
r = requests.get(
self.url,
params=self.params
)
parsed = json.loads(r.text)
return parsed.get('photos')
def run(self):
self.get_uid()
self.get_favgalid()
2017-11-10 16:04:05 +00:00
r = requests.get(self.url, params=self.params)
js = json.loads(r.text)
photos = js.get('photos')
total = int(js.get('total_pages', 1))
current = int(js.get('current_page', 1))
cntr = total - current
while cntr > 0:
current = current + 1
paged = self.getpaged(current)
photos = photos + paged
cntr = total - current
for photo in photos:
fav = FivehpxFav(photo)
if not fav.exists:
fav.run()
2017-11-10 16:04:05 +00:00
# fav.fix_extension()
class TumblrFavs(Favs):
url = 'https://api.tumblr.com/v2/user/likes'
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('tumblr')
self.oauth = TumblrOauth()
self.params = {
'after': self.lastpulled
}
self.likes = []
def getpaged(self, offset):
r = self.oauth.request(
self.url,
params={'offset': offset}
)
return json.loads(r.text)
def run(self):
r = self.oauth.request(
self.url,
params=self.params
)
js = json.loads(r.text)
total = int(js.get('response', {}).get('liked_count', 20))
offset = 20
cntr = total - offset
likes = js.get('response', {}).get('liked_posts', [])
while cntr > 0:
paged = self.getpaged(offset)
likes = likes + paged.get('response', {}).get('liked_posts', [])
offset = offset + 20
cntr = total - offset
self.likes = likes
for like in self.likes:
fav = TumblrFav(like)
if not fav.exists:
fav.run()
class DAFavs(Favs):
def __init__(self):
from pprint import pprint
2017-11-30 16:56:47 +00:00
super().__init__('deviantart')
self.username = shared.config.get('api_deviantart', 'username'),
self.oauth = DAOauth()
self.likes = []
self.galid = None
self.params = {
2017-11-10 16:04:05 +00:00
'limit': 24, # this is the max as far as I can tell
'mature_content': 'true',
'username': self.username
}
def get_favgalid(self):
r = self.oauth.request(
'https://www.deviantart.com/api/v1/oauth2/collections/folders',
params={
'username': self.username,
'calculate_size': 'true',
'ext_preload': 'false',
'mature_content': 'true'
}
)
js = json.loads(r.text)
for g in js.get('results', []):
if 'Featured' == g.get('name'):
self.galid = g.get('folderid')
break
@property
def url(self):
2017-11-10 16:04:05 +00:00
return 'https://www.deviantart.com/api/v1/oauth2/collections/%s' % (
self.galid)
def getpaged(self, offset):
self.params.update({'offset': offset})
r = self.oauth.request(
self.url,
self.params
)
js = json.loads(r.text)
return js
def getsinglemeta(self, daid):
r = self.oauth.request(
'https://www.deviantart.com/api/v1/oauth2/deviation/metadata',
params={
'deviationids[]': daid,
'ext_submission': False,
'ext_camera': False,
'ext_stats': False,
'ext_collection': False,
'mature_content': True,
}
)
meta = {}
try:
meta = json.loads(r.text)
return meta.get('metadata', []).pop()
2017-11-10 16:04:05 +00:00
except BaseException:
return meta
def has_more(self, q):
if True == q or 'True' == q or 'true' == q:
return True
return False
def run(self):
self.get_favgalid()
r = self.oauth.request(
self.url,
self.params
)
js = json.loads(r.text)
favs = js.get('results', [])
has_more = self.has_more(js.get('has_more'))
offset = js.get('next_offset')
while True == has_more:
#logging.info('iterating over DA results with offset %d', offset)
paged = self.getpaged(offset)
new = paged.get('results', [])
if not len(new):
#logging.error('empty results from deviantART, breaking loop')
break
favs = [*favs, *new]
has_more = self.has_more(paged.get('has_more'))
if not has_more:
break
n = int(paged.get('next_offset'))
if not n:
break
offset = n
self.favs = favs
for fav in self.favs:
f = DAFav(fav)
if not f.exists:
2017-11-10 16:04:05 +00:00
f.fav.update(
{'meta': self.getsinglemeta(fav.get('deviationid'))})
f.run()
2017-11-10 16:04:05 +00:00
# f.fix_extension()
class ImgFav(object):
def __init__(self):
self.imgurl = ''
self.meta = {
'dt': arrow.utcnow(),
'title': '',
'favorite-of': '',
'tags': [],
'geo': {
'latitude': '',
'longitude': '',
},
'author': {
'name': '',
'url': '',
},
}
self.content = ''
@property
def exists(self):
maybe = glob.glob(self.target.replace('.jpg', '.*'))
if len(maybe):
return True
return False
def fix_extension(self):
# identify file format
what = imghdr.what(self.target)
# rename file
new = self.target.replace('.jpg', '.%s' % what)
if new != self.target:
shutil.move(self.target, new)
self.target = new
def pull_image(self):
logging.info("pulling image %s to %s", self.imgurl, self.target)
r = requests.get(self.imgurl, stream=True)
if r.status_code == 200:
with open(self.target, 'wb') as f:
r.raw.decode_content = True
shutil.copyfileobj(r.raw, f)
def write_exif(self):
what = imghdr.what(self.target)
if 'jpg' != what or 'png' != what:
return
logging.info('populating EXIF data of %s' % self.target)
2017-11-10 16:04:05 +00:00
tags = list(set(self.meta.get('tags', [])))
dt = self.meta.get('dt').to('utc')
geo_lat = False
geo_lon = False
if self.meta.get('geo', None):
geo = self.meta.get('geo', None)
lat = geo.get('latitude', None)
lon = geo.get('longitude', None)
if lat and lon and 'null' != lat and 'null' != lon:
geo_lat = lat
geo_lon = lon
author_name = ''
author_url = ''
if self.meta.get('author', None):
a = self.meta.get('author')
author_name = a.get('name', '')
author_url = a.get('url', '')
author_name = "%s" % author_name
author_url = "%s" % author_url
params = [
'exiftool',
'-overwrite_original',
#'-EXIF:Artist=%s' % author_name[:64],
'-XMP:Copyright=Copyright %s %s (%s)' % (
dt.format('YYYY'),
author_name,
author_url,
),
'-XMP:Source=%s' % self.meta.get('favorite-of'),
'-XMP:ReleaseDate=%s' % dt.format('YYYY:MM:DD HH:mm:ss'),
'-XMP:Headline=%s' % self.meta.get('title'),
'-XMP:Description=%s' % self.content,
2017-11-10 16:04:05 +00:00
]
for t in tags:
params.append('-XMP:HierarchicalSubject+=%s' % t)
params.append('-XMP:Subject+=%s' % t)
if geo_lat and geo_lon:
2017-11-10 16:04:05 +00:00
geo_lat = round(float(geo_lat), 6)
geo_lon = round(float(geo_lon), 6)
if geo_lat < 0:
GPSLatitudeRef = 'S'
else:
GPSLatitudeRef = 'N'
if geo_lon < 0:
GPSLongitudeRef = 'W'
else:
GPSLongitudeRef = 'E'
params.append('-GPSLongitude=%s' % abs(geo_lon))
params.append('-GPSLatitude=%s' % abs(geo_lat))
params.append('-GPSLongitudeRef=%s' % GPSLongitudeRef)
params.append('-GPSLatitudeRef=%s' % GPSLatitudeRef)
2017-11-10 16:04:05 +00:00
params.append(self.target)
p = subprocess.Popen(
params,
stdin=subprocess.PIPE,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
)
stdout, stderr = p.communicate()
_original = '%s_original' % self.target
if os.path.exists(_original):
os.unlink(_original)
class FlickrFav(ImgFav):
url = 'https://api.flickr.com/services/rest/'
def __init__(self, photo):
self.photo = photo
self.ownerid = photo.get('owner')
self.photoid = photo.get('id')
2017-11-10 16:04:05 +00:00
self.url = "https://www.flickr.com/photos/%s/%s" % (
self.ownerid, self.photoid)
self.target = os.path.join(
shared.config.get('archive', 'favorite'),
"flickr-%s-%s.jpg" % (self.ownerid, self.photoid)
)
def run(self):
if self.exists:
logging.warning("%s already exists, skipping", self.target)
return
# the bigger the better, see
# https://www.flickr.com/services/api/misc.urls.html
2017-11-10 16:04:05 +00:00
img = False
for x in ['url_o', 'url_k', 'url_h', 'url_b', 'url_c', 'url_z']:
if x in self.photo:
img = self.photo.get(x)
break
if not img:
logging.error("image url was empty for %s, skipping fav", self.url)
return
self.imgurl = img
self.pull_image()
self.meta = {
'dt': arrow.get(
self.photo.get('date_faved',
2017-11-10 16:04:05 +00:00
arrow.utcnow().timestamp
)
),
'title': '%s' % shared.Pandoc('plain').convert(
self.photo.get('title', '')
).rstrip(),
'favorite-of': self.url,
'tags': self.photo.get('tags', '').split(' '),
'geo': {
'latitude': self.photo.get('latitude', ''),
'longitude': self.photo.get('longitude', ''),
},
'author': {
'name': self.photo.get('ownername'),
'url': 'https://www.flickr.com/people/%s' % (
self.photo.get('owner')
),
},
}
self.content = shared.Pandoc('plain').convert(
self.photo.get('description', {}).get('_content', '')
)
self.fix_extension()
self.write_exif()
2017-11-10 16:04:05 +00:00
class FivehpxFav(ImgFav):
def __init__(self, photo):
self.photo = photo
self.ownerid = photo.get('user_id')
self.photoid = photo.get('id')
self.target = os.path.join(
shared.config.get('archive', 'favorite'),
"500px-%s-%s.jpg" % (self.ownerid, self.photoid)
)
self.url = "https://www.500px.com%s" % (photo.get('url'))
def run(self):
img = self.photo.get('images')[0].get('url')
if not img:
logging.error("image url was empty for %s, skipping fav", self.url)
return
self.imgurl = img
self.pull_image()
self.meta = {
'dt': arrow.get(
self.photo.get('created_at',
2017-11-10 16:04:05 +00:00
arrow.utcnow().timestamp
)
),
'title': '%s' % shared.Pandoc('plain').convert(
self.photo.get('name', '')
).rstrip(),
'favorite-of': self.url,
'tags': self.photo.get('tags', []),
'geo': {
'latitude': self.photo.get('latitude', ''),
'longitude': self.photo.get('longitude', ''),
},
'author': {
'name': self.photo.get('user').get('fullname', self.ownerid),
'url': 'https://www.500px.com/%s' % (
self.photo.get('user').get('username', self.ownerid)
),
},
}
c = "%s" % self.photo.get('description', '')
self.content = shared.Pandoc('plain').convert(c)
self.fix_extension()
self.write_exif()
2017-11-10 16:04:05 +00:00
class DAFav(ImgFav):
def __init__(self, fav):
self.fav = fav
self.deviationid = fav.get('deviationid')
#logging.info('working on %s', self.deviationid)
self.url = fav.get('url')
self.title = fav.get('title', False) or self.deviationid
self.author = self.fav.get('author').get('username')
self.target = os.path.join(
shared.config.get('archive', 'favorite'),
"deviantart-%s-%s.jpg" % (
shared.slugfname(self.title),
shared.slugfname(self.author)
)
)
self.imgurl = None
if 'content' in fav:
if 'src' in fav['content']:
self.imgurl = fav.get('content').get('src')
elif 'preview' in fav:
if 'src' in fav['preview']:
self.imgurl = fav.get('preview').get('src')
self.imgurl = fav.get('content', {}).get('src')
def run(self):
if not self.imgurl:
2017-11-10 16:04:05 +00:00
logging.error(
'imgurl is empty for deviantart %s',
self.deviationid)
return
self.pull_image()
self.meta = {
'dt': arrow.get(
self.fav.get('published_time',
2017-11-10 16:04:05 +00:00
arrow.utcnow().timestamp
)
),
'title': '%s' % shared.Pandoc('plain').convert(self.title).rstrip(),
'favorite-of': self.url,
'tags': [t.get('tag_name') for t in self.fav.get('meta', {}).get('tags', [])],
'author': {
'name': self.author,
'url': 'https://%s.deviantart.com' % (self.author),
},
}
c = "%s" % self.fav.get('meta', {}).get('description', '')
self.content = shared.Pandoc('plain').convert(c)
self.fix_extension()
self.write_exif()
class TumblrFav(object):
def __init__(self, like):
self.like = like
self.blogname = like.get('blog_name')
self.postid = like.get('id')
self.target = os.path.join(
shared.config.get('archive', 'favorite'),
"tumblr-%s-%s.jpg" % (self.blogname, self.postid)
)
self.url = like.get('post_url')
self.images = []
@property
def exists(self):
maybe = glob.glob(self.target.replace('.jpg', '_0.*'))
if len(maybe):
return True
return False
def run(self):
content = "%s" % self.like.get('caption', '')
title = self.like.get('summary', '').strip()
if not len(title):
title = self.like.get('slug', '').strip()
if not len(title):
title = shared.slugfname(self.like.get('post_url'))
meta = {
'dt': arrow.get(
self.like.get('liked_timestamp',
2017-11-10 16:04:05 +00:00
self.like.get('date',
arrow.utcnow().timestamp
)
)
),
'title': title,
'favorite-of': self.url,
'tags': self.like.get('tags'),
'author': {
'name': self.like.get('blog_name'),
'url': 'http://%s.tumblr.com' % self.like.get('blog_name')
},
}
icntr = 0
for p in self.like.get('photos', []):
img = ImgFav()
img.target = self.target.replace('.jpg', '_%d.jpg' % icntr)
img.imgurl = p.get('original_size').get('url')
img.content = content
img.meta = meta
img.pull_image()
img.fix_extension()
img.write_exif()
icntr = icntr + 1
2017-11-30 16:56:47 +00:00
class TwitterFav(object):
def __init__(self, like):
self.like = like
self.postid = like.get('id')
self.target = os.path.join(
shared.config.get('archive', 'favorite'),
"twitter-%s.jpg" % (self.postid)
)
@property
def exists(self):
maybe = glob.glob(self.target.replace('.jpg', '_*.*'))
if len(maybe):
return True
return False
def run(self):
content = "%s" % self.like.get('text', '')
title = self.like.get('id')
user = self.like.get('user')
meta = {
'dt': arrow.get(
self.like.get('created_at'),
shared.ARROWFORMAT.get('twitter')
),
'title': title,
'favorite-of': "https://twitter.com/%s/status/%s" % (
user.get('id'),
self.like.get('id')
),
'tags': self.like.get('hashtags'),
'author': {
'name': user.get('name'),
'username': user.get('screen_name'),
'id': user.get('id'),
'url': 'http://twitter.com/%s' % user.get('screen_name')
},
}
for p in self.like.get('entities', {}).get('media', []):
img = ImgFav()
img.imgurl = p.get('media_url_https')
img.target = self.target.replace(
'.jpg',
'_%s.jpg' % p.get('id')
)
img.content = content
img.meta = meta
img.pull_image()
img.fix_extension()
img.write_exif()
2018-06-08 10:17:57 +01:00
2017-11-30 16:56:47 +00:00
class TwitterFavs(Favs):
url = 'https://api.twitter.com/1.1/favorites/list.json'
def __init__(self):
super().__init__('twitter')
self.oauth = TwitterOauth()
self.params = {
'user_id': shared.config.get('api_twitter', 'userid'),
'count': 200
}
@property
def lastpulled(self):
lastid = 0
d = os.path.join(
shared.config.get('archive', 'favorite'),
"%s-*" % self.confgroup
)
files = glob.glob(d)
for f in files:
tweetid = int(re.sub(
'.*twitter-(?P<tweetid>[0-9]+)_.*',
'\g<tweetid>',
f
))
if tweetid > lastid:
lastid = tweetid
logging.debug("last fav id: %s", lastid)
return lastid
def run(self):
if self.lastpulled > 0:
self.params.update({
'since_id': self.lastpulled
})
r = self.oauth.request(
self.url,
params=self.params
)
for like in json.loads(r.text):
fav = TwitterFav(like)
if not fav.exists:
fav.run()
class Oauth2Flow(object):
token_url = ''
def __init__(self, service):
self.service = service
self.key = shared.config.get("api_%s" % service, 'api_key')
self.secret = shared.config.get("api_%s" % service, 'api_secret')
client = BackendApplicationClient(
client_id=self.key
)
client.prepare_request_body(scope=['browse'])
oauth = OAuth2Session(client=client)
token = oauth.fetch_token(
token_url=self.token_url,
client_id=self.key,
client_secret=self.secret
)
self.client = OAuth2Session(
self.key,
token=token
)
def request(self, url, params={}):
return self.client.get(url, params=params)
class DAOauth(Oauth2Flow):
token_url = 'https://www.deviantart.com/oauth2/token'
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('deviantart')
class Oauth1Flow(object):
request_token_url = ''
access_token_url = ''
authorize_url = ''
def __init__(self, service):
self.service = service
self.key = shared.config.get("api_%s" % service, 'api_key')
self.secret = shared.config.get("api_%s" % service, 'api_secret')
self.tokendb = shared.TokenDB()
self.t = self.tokendb.get_service(self.service)
self.oauth_init()
def oauth_init(self):
if not self.t:
self.request_oauth_token()
t = self.tokendb.get_token(self.t)
2017-11-10 16:04:05 +00:00
if not t.get('access_token', None) or not t.get(
'access_token_secret', None):
self.request_access_token()
def request_oauth_token(self):
client = OAuth1Session(
self.key,
client_secret=self.secret,
callback_uri="%s/oauth1/" % shared.config.get('site', 'url')
)
r = client.fetch_request_token(self.request_token_url)
logging.debug('setting token to %s', r.get('oauth_token'))
self.t = r.get('oauth_token')
logging.debug('updating secret to %s', r.get('oauth_token_secret'))
self.tokendb.update_token(
self.t,
oauth_token_secret=r.get('oauth_token_secret')
)
self.tokendb.set_service(
self.service,
self.t
)
existing = self.tokendb.get_token(self.t)
verified = existing.get('verifier', None)
while not verified:
logging.debug('verifier missing for %s', self.t)
self.auth_url(existing)
self.tokendb.refresh()
existing = self.tokendb.get_token(self.t)
verified = existing.get('verifier', None)
def auth_url(self, existing):
t = self.tokendb.get_token(self.t)
client = OAuth1Session(
self.key,
client_secret=self.secret,
resource_owner_key=self.t,
resource_owner_secret=t.get('oauth_token_secret'),
callback_uri="%s/oauth1/" % shared.config.get('site', 'url')
)
input('Visit: %s and press any key after' % (
client.authorization_url(self.authorize_url)
))
def request_access_token(self):
try:
t = self.tokendb.get_token(self.t)
client = OAuth1Session(
self.key,
client_secret=self.secret,
callback_uri="%s/oauth1/" % shared.config.get('site', 'url'),
resource_owner_key=self.t,
resource_owner_secret=t.get('oauth_token_secret'),
verifier=t.get('verifier')
)
r = client.fetch_access_token(self.access_token_url)
self.tokendb.update_token(
self.t,
access_token=r.get('oauth_token'),
access_token_secret=r.get('oauth_token_secret')
)
except oauth1_session.TokenRequestDenied as e:
2017-11-10 16:04:05 +00:00
logging.error(
'getting access token was denied, clearing former oauth tokens and re-running everyting')
self.tokendb.clear_service(self.service)
self.oauth_init()
def request(self, url, params):
t = self.tokendb.get_token(self.t)
client = OAuth1Session(
self.key,
client_secret=self.secret,
resource_owner_key=t.get('access_token'),
resource_owner_secret=t.get('access_token_secret')
)
return client.get(url, params=params)
class FivehpxOauth(Oauth1Flow):
request_token_url = 'https://api.500px.com/v1/oauth/request_token'
access_token_url = 'https://api.500px.com/v1/oauth/access_token'
authorize_url = 'https://api.500px.com/v1/oauth/authorize'
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('500px')
class FlickrOauth(Oauth1Flow):
request_token_url = 'https://www.flickr.com/services/oauth/request_token'
access_token_url = 'https://www.flickr.com/services/oauth/access_token'
authorize_url = 'https://www.flickr.com/services/oauth/authorize'
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('flickr')
class TumblrOauth(Oauth1Flow):
request_token_url = 'https://www.tumblr.com/oauth/request_token'
access_token_url = 'https://www.tumblr.com/oauth/access_token'
authorize_url = 'https://www.tumblr.com/oauth/authorize'
def __init__(self):
2017-11-30 16:56:47 +00:00
super().__init__('tumblr')
class TwitterOauth(Oauth1Flow):
request_token_url = 'https://api.twitter.com/oauth/request_token'
access_token_url = 'https://api.twitter.com/oauth/access_token'
authorize_url = 'https://api.twitter.com/oauth/authorize'
def __init__(self):
super().__init__('twitter')
if __name__ == '__main__':
logging.basicConfig(level=20)
2017-10-27 15:56:05 +01:00
if shared.config.has_section('api_flickr'):
flickr = FlickrFavs()
flickr.run()
2017-10-27 15:56:05 +01:00
if shared.config.has_section('api_500px'):
fivehpx = FivehpxFavs()
fivehpx.run()
2017-10-27 15:56:05 +01:00
if shared.config.has_section('api_tumblr'):
tumblr = TumblrFavs()
tumblr.run()
2017-10-27 15:56:05 +01:00
if shared.config.has_section('api_deviantart'):
da = DAFavs()
da.run()
2017-11-01 13:19:39 +00:00
if shared.config.has_section('api_lastfm'):
lfm = LastFM()
lfm.run()
2017-11-30 16:56:47 +00:00
if shared.config.has_section('api_twitter'):
tw = TwitterFavs()
tw.run()