2019-02-02 18:17:07 +03:00
import re
import os
import hashlib
2019-04-03 18:16:27 +03:00
from threading import Thread , Timer
2019-02-16 18:42:45 +03:00
import pickle
2019-04-12 18:57:28 +03:00
import json
2019-03-03 23:55:35 +03:00
import urllib
2019-03-10 20:14:50 +03:00
import datetime
2019-04-03 14:20:18 +03:00
import random
2019-04-03 17:03:48 +03:00
import itertools
2019-11-22 01:14:53 +03:00
import base64
2019-03-28 17:06:14 +03:00
from doreah import settings
2019-04-01 17:52:42 +03:00
from doreah import caching
2019-03-29 22:23:32 +03:00
from doreah . logging import log
2019-04-05 18:18:17 +03:00
from doreah . regular import yearly , daily
2019-02-02 18:17:07 +03:00
2019-05-05 12:56:09 +03:00
from external import api_request_track , api_request_artist
2019-02-02 18:17:07 +03:00
2019-04-03 18:16:27 +03:00
2019-04-12 18:57:28 +03:00
#####
## SERIALIZE
#####
def serialize ( obj ) :
try :
return json . dumps ( obj )
except :
if isinstance ( obj , list ) or isinstance ( obj , tuple ) :
return " [ " + " , " . join ( serialize ( o ) for o in obj ) + " ] "
elif isinstance ( obj , dict ) :
return " { " + " , " . join ( serialize ( o ) + " : " + serialize ( obj [ o ] ) for o in obj ) + " } "
return json . dumps ( obj . hashable ( ) )
#if isinstance(obj,list) or if isinstance(obj,tuple):
# return "[" + ",".join(dumps(o) for o in obj) + "]"
#if isinstance(obj,str)
2019-04-03 18:16:27 +03:00
#####
## RULESTATE VALIDATION
#####
2018-12-20 20:46:55 +03:00
def checksumTSV ( folder ) :
2019-03-28 16:18:12 +03:00
2018-12-20 20:46:55 +03:00
sums = " "
2019-03-28 16:18:12 +03:00
2018-12-20 20:46:55 +03:00
for f in os . listdir ( folder + " / " ) :
if ( f . endswith ( " .tsv " ) ) :
f = open ( folder + " / " + f , " rb " )
sums + = hashlib . md5 ( f . read ( ) ) . hexdigest ( ) + " \n "
f . close ( )
2019-03-28 16:18:12 +03:00
2018-12-20 20:46:55 +03:00
return sums
2019-03-28 16:18:12 +03:00
2018-12-21 00:07:22 +03:00
# returns whether checksums match and sets the checksum to invalid if they don't (or sets the new one if no previous one exists)
def combineChecksums ( filename , checksums ) :
import os
2019-03-28 16:18:12 +03:00
2018-12-21 00:07:22 +03:00
if os . path . exists ( filename + " .rulestate " ) :
f = open ( filename + " .rulestate " , " r " )
oldchecksums = f . read ( )
f . close ( )
if oldchecksums == checksums :
# the new checksum given by the calling db server represents the rule state that all current unsaved scrobbles were created under
# if this is the same as the existing one, we're all good
return True
elif ( oldchecksums != " INVALID " ) :
#if not, the file is not consistent to any single rule state (some scrobbles were created with an old ruleset, some not)
f = open ( filename + " .rulestate " , " w " )
f . write ( " INVALID " ) # this will never match any sha256sum
f . close ( )
return False
else :
#if the file already says invalid, no need to open it and rewrite
return False
else :
f = open ( filename + " .rulestate " , " w " )
f . write ( checksums )
f . close ( )
return True
2019-03-28 16:18:12 +03:00
2018-12-21 18:32:21 +03:00
# checks ALL files for their rule state. if they are all the same as the current loaded one, the entire database can be assumed to be consistent with the current ruleset
# in any other case, get out
def consistentRulestate ( folder , checksums ) :
2019-03-28 16:18:12 +03:00
2018-12-21 18:32:21 +03:00
result = [ ]
for scrobblefile in os . listdir ( folder + " / " ) :
2019-03-28 16:18:12 +03:00
2018-12-21 18:32:21 +03:00
if ( scrobblefile . endswith ( " .tsv " ) ) :
2019-03-28 16:18:12 +03:00
2018-12-21 18:32:21 +03:00
try :
2019-08-17 19:21:36 +03:00
with open ( folder + " / " + scrobblefile + " .rulestate " , " r " ) as f :
if f . read ( ) != checksums :
return False
2018-12-21 18:32:21 +03:00
except :
return False
2019-03-28 16:18:12 +03:00
2018-12-21 18:32:21 +03:00
return True
2019-03-28 16:18:12 +03:00
2019-03-12 14:56:53 +03:00
2019-04-03 18:16:27 +03:00
#####
## IMAGES
#####
2018-12-17 17:10:10 +03:00
2019-03-28 19:40:22 +03:00
2019-04-01 17:52:42 +03:00
### Caches
2019-03-28 19:40:22 +03:00
2019-04-01 17:52:42 +03:00
cacheage = settings . get_settings ( " CACHE_EXPIRE_POSITIVE " ) * 24 * 3600
cacheage_neg = settings . get_settings ( " CACHE_EXPIRE_NEGATIVE " ) * 24 * 3600
2019-03-28 19:40:22 +03:00
2019-04-15 13:26:12 +03:00
artist_cache = caching . Cache ( name = " imgcache_artists " , maxage = cacheage , maxage_negative = cacheage_neg , persistent = True )
track_cache = caching . Cache ( name = " imgcache_tracks " , maxage = cacheage , maxage_negative = cacheage_neg , persistent = True )
2019-03-28 19:40:22 +03:00
2019-04-03 17:03:48 +03:00
# removes emojis and weird shit from names
def clean ( name ) :
return " " . join ( c for c in name if c . isalnum ( ) or c in [ ] ) . strip ( )
2019-11-22 01:14:53 +03:00
def get_all_possible_filenames ( artist = None , artists = None , title = None ) :
2019-04-03 17:03:48 +03:00
# check if we're dealing with a track or artist, then clean up names
# (only remove non-alphanumeric, allow korean and stuff)
2019-11-22 01:14:53 +03:00
2019-04-03 17:03:48 +03:00
if title is not None and artists is not None :
track = True
title , artists = clean ( title ) , [ clean ( a ) for a in artists ]
elif artist is not None :
track = False
artist = clean ( artist )
else : return [ ]
superfolder = " images/tracks/ " if track else " images/artists/ "
filenames = [ ]
if track :
#unsafeartists = [artist.translate(None,"-_./\\") for artist in artists]
safeartists = [ re . sub ( " [^a-zA-Z0-9] " , " " , artist ) for artist in artists ]
#unsafetitle = title.translate(None,"-_./\\")
safetitle = re . sub ( " [^a-zA-Z0-9] " , " " , title )
if len ( artists ) < 4 :
unsafeperms = itertools . permutations ( artists )
safeperms = itertools . permutations ( safeartists )
else :
unsafeperms = [ sorted ( artists ) ]
safeperms = [ sorted ( safeartists ) ]
for unsafeartistlist in unsafeperms :
filename = " - " . join ( unsafeartistlist ) + " _ " + title
if filename != " " :
filenames . append ( filename )
filenames . append ( filename . lower ( ) )
for safeartistlist in safeperms :
filename = " - " . join ( safeartistlist ) + " _ " + safetitle
if filename != " " :
filenames . append ( filename )
filenames . append ( filename . lower ( ) )
filenames = list ( set ( filenames ) )
if len ( filenames ) == 0 : filenames . append ( str ( hash ( ( frozenset ( artists ) , title ) ) ) )
else :
#unsafeartist = artist.translate(None,"-_./\\")
safeartist = re . sub ( " [^a-zA-Z0-9] " , " " , artist )
filename = artist
if filename != " " :
filenames . append ( filename )
filenames . append ( filename . lower ( ) )
filename = safeartist
if filename != " " :
filenames . append ( filename )
filenames . append ( filename . lower ( ) )
filenames = list ( set ( filenames ) )
if len ( filenames ) == 0 : filenames . append ( str ( hash ( artist ) ) )
2019-11-22 01:14:53 +03:00
return [ superfolder + name for name in filenames ]
def local_files ( artist = None , artists = None , title = None ) :
filenames = get_all_possible_filenames ( artist , artists , title )
2019-04-03 17:03:48 +03:00
images = [ ]
for purename in filenames :
# direct files
for ext in [ " png " , " jpg " , " jpeg " , " gif " ] :
#for num in [""] + [str(n) for n in range(0,10)]:
2019-11-22 01:14:53 +03:00
if os . path . exists ( purename + " . " + ext ) :
images . append ( " / " + purename + " . " + ext )
2019-04-03 17:03:48 +03:00
# folder
try :
2019-11-22 01:14:53 +03:00
for f in os . listdir ( purename + " / " ) :
2019-04-03 17:03:48 +03:00
if f . split ( " . " ) [ - 1 ] in [ " png " , " jpg " , " jpeg " , " gif " ] :
2019-11-22 01:14:53 +03:00
images . append ( " / " + purename + " / " + f )
2019-04-03 17:03:48 +03:00
except :
pass
return images
2019-03-28 19:40:22 +03:00
2019-05-05 12:56:09 +03:00
# these caches are there so we don't check all files every time, but return the same one
2019-04-03 17:43:09 +03:00
local_cache_age = settings . get_settings ( " LOCAL_IMAGE_ROTATE " )
local_artist_cache = caching . Cache ( maxage = local_cache_age )
local_track_cache = caching . Cache ( maxage = local_cache_age )
2019-02-16 18:42:45 +03:00
2019-03-12 13:39:36 +03:00
def getTrackImage ( artists , title , fast = False ) :
2019-03-06 20:04:12 +03:00
2019-04-03 17:43:09 +03:00
# obj = (frozenset(artists),title)
# filename = "-".join([re.sub("[^a-zA-Z0-9]","",artist) for artist in sorted(artists)]) + "_" + re.sub("[^a-zA-Z0-9]","",title)
# if filename == "": filename = str(hash(obj))
# filepath = "images/tracks/" + filename
2019-03-28 16:18:12 +03:00
2019-04-07 14:22:33 +03:00
if settings . get_settings ( " USE_LOCAL_IMAGES " ) :
try :
return local_track_cache . get ( ( frozenset ( artists ) , title ) )
except :
images = local_files ( artists = artists , title = title )
if len ( images ) != 0 :
#return random.choice(images)
res = random . choice ( images )
local_track_cache . add ( ( frozenset ( artists ) , title ) , res )
return urllib . parse . quote ( res )
2019-04-03 17:03:48 +03:00
2019-04-03 14:20:18 +03:00
2018-12-28 20:06:09 +03:00
# check if custom image exists
2019-04-03 14:20:18 +03:00
# if os.path.exists(filepath + ".png"):
# imgurl = "/" + filepath + ".png"
# return imgurl
# elif os.path.exists(filepath + ".jpg"):
# imgurl = "/" + filepath + ".jpg"
# return imgurl
# elif os.path.exists(filepath + ".jpeg"):
# imgurl = "/" + filepath + ".jpeg"
# return imgurl
# elif os.path.exists(filepath + ".gif"):
# imgurl = "/" + filepath + ".gif"
# return imgurl
2019-03-28 16:18:12 +03:00
2019-03-28 17:06:14 +03:00
2018-12-28 20:06:09 +03:00
try :
2019-03-12 13:39:36 +03:00
# check our cache
# if we have cached the nonexistence of that image, we immediately return the redirect to the artist and let the resolver handle it
# (even if we're not in a fast lookup right now)
2019-03-28 19:40:22 +03:00
#result = cachedTracks[(frozenset(artists),title)]
2019-04-01 17:52:42 +03:00
result = track_cache . get ( ( frozenset ( artists ) , title ) ) #track_from_cache(artists,title)
2019-03-12 13:39:36 +03:00
if result is not None : return result
else :
for a in artists :
res = getArtistImage ( artist = a , fast = True )
if res != " " : return res
return " "
2018-12-28 20:06:09 +03:00
except :
pass
2019-03-28 16:18:12 +03:00
2019-03-28 17:06:14 +03:00
# do we have an api key?
2019-05-05 12:56:09 +03:00
# apikey = settings.get_settings("LASTFM_API_KEY")
# if apikey is None: return "" # DO NOT CACHE THAT
2019-03-28 17:06:14 +03:00
2019-03-28 16:18:12 +03:00
2019-03-03 23:55:35 +03:00
# fast request only retuns cached and local results, generates redirect link for rest
2019-03-12 13:39:36 +03:00
if fast : return " /image?title= " + urllib . parse . quote ( title ) + " & " + " & " . join ( [ " artist= " + urllib . parse . quote ( a ) for a in artists ] )
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
# non-fast lookup (esentially only the resolver lookup)
2019-05-05 12:56:09 +03:00
result = api_request_track ( ( artists , title ) )
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
# cache results (even negative ones)
2019-03-28 19:40:22 +03:00
#cachedTracks[(frozenset(artists),title)] = result
2019-04-01 17:52:42 +03:00
track_cache . add ( ( frozenset ( artists ) , title ) , result ) #cache_track(artists,title,result)
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
# return either result or redirect to artist
if result is not None : return result
2019-01-10 01:29:01 +03:00
else :
2019-03-12 13:39:36 +03:00
for a in artists :
res = getArtistImage ( artist = a , fast = False )
if res != " " : return res
return " "
2019-03-28 16:18:12 +03:00
def getArtistImage ( artist , fast = False ) :
2019-04-03 17:43:09 +03:00
# obj = artist
# filename = re.sub("[^a-zA-Z0-9]","",artist)
# if filename == "": filename = str(hash(obj))
# filepath = "images/artists/" + filename
# #filepath_cache = "info/artists_cache/" + filename
2019-04-07 14:22:33 +03:00
if settings . get_settings ( " USE_LOCAL_IMAGES " ) :
try :
return local_artist_cache . get ( artist )
except :
images = local_files ( artist = artist )
if len ( images ) != 0 :
#return random.choice(images)
res = random . choice ( images )
local_artist_cache . add ( artist , res )
return urllib . parse . quote ( res )
2019-03-28 16:18:12 +03:00
2019-04-03 14:20:18 +03:00
2018-12-17 17:10:10 +03:00
# check if custom image exists
2019-04-03 14:20:18 +03:00
# if os.path.exists(filepath + ".png"):
# imgurl = "/" + filepath + ".png"
# return imgurl
# elif os.path.exists(filepath + ".jpg"):
# imgurl = "/" + filepath + ".jpg"
# return imgurl
# elif os.path.exists(filepath + ".jpeg"):
# imgurl = "/" + filepath + ".jpeg"
# return imgurl
# elif os.path.exists(filepath + ".gif"):
# imgurl = "/" + filepath + ".gif"
# return imgurl
2019-03-28 16:18:12 +03:00
2019-01-10 01:29:01 +03:00
2018-12-17 17:10:10 +03:00
try :
2019-03-28 19:40:22 +03:00
#result = cachedArtists[artist]
2019-04-01 17:52:42 +03:00
result = artist_cache . get ( artist ) #artist_from_cache(artist)
2019-03-12 13:39:36 +03:00
if result is not None : return result
else : return " "
2018-12-28 20:06:09 +03:00
except :
2018-12-17 17:10:10 +03:00
pass
2019-03-28 16:18:12 +03:00
2019-03-28 19:57:56 +03:00
2019-03-28 17:06:14 +03:00
# do we have an api key?
2019-05-05 12:56:09 +03:00
# apikey = settings.get_settings("LASTFM_API_KEY")
# if apikey is None: return "" # DO NOT CACHE THAT
2019-03-28 17:54:39 +03:00
2019-03-28 16:18:12 +03:00
2019-03-28 19:57:56 +03:00
2019-03-03 23:55:35 +03:00
# fast request only retuns cached and local results, generates redirect link for rest
2019-03-12 13:39:36 +03:00
if fast : return " /image?artist= " + urllib . parse . quote ( artist )
2019-03-28 16:18:12 +03:00
# non-fast lookup (esentially only the resolver lookup)
2019-05-05 12:56:09 +03:00
result = api_request_artist ( artist = artist )
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
# cache results (even negative ones)
2019-03-28 19:40:22 +03:00
#cachedArtists[artist] = result
2019-04-01 17:52:42 +03:00
artist_cache . add ( artist , result ) #cache_artist(artist,result)
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
if result is not None : return result
else : return " "
2019-02-02 20:08:30 +03:00
2019-03-12 13:39:36 +03:00
def getTrackImages ( trackobjectlist , fast = False ) :
2019-02-02 20:08:30 +03:00
threads = [ ]
2019-03-28 16:18:12 +03:00
2019-02-02 20:08:30 +03:00
for track in trackobjectlist :
2019-03-12 13:39:36 +03:00
t = Thread ( target = getTrackImage , args = ( track [ " artists " ] , track [ " title " ] , ) , kwargs = { " fast " : fast } )
2019-02-02 20:08:30 +03:00
t . start ( )
threads . append ( t )
2019-03-28 16:18:12 +03:00
2019-02-02 20:08:30 +03:00
for t in threads :
t . join ( )
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
return [ getTrackImage ( t [ " artists " ] , t [ " title " ] ) for t in trackobjectlist ]
2019-03-28 16:18:12 +03:00
2019-03-12 13:39:36 +03:00
def getArtistImages ( artistlist , fast = False ) :
2019-03-28 16:18:12 +03:00
2019-02-02 18:17:07 +03:00
threads = [ ]
2019-03-28 16:18:12 +03:00
2019-02-02 18:17:07 +03:00
for artist in artistlist :
2019-03-12 13:39:36 +03:00
t = Thread ( target = getArtistImage , args = ( artist , ) , kwargs = { " fast " : fast } )
2019-02-02 18:17:07 +03:00
t . start ( )
threads . append ( t )
2019-03-28 16:18:12 +03:00
2019-02-02 18:17:07 +03:00
for t in threads :
t . join ( )
2019-03-28 16:18:12 +03:00
# async calls only cached results, now we need to get them
2019-03-12 13:39:36 +03:00
return [ getArtistImage ( a ) for a in artistlist ]
2019-02-03 18:52:37 +03:00
2019-02-20 23:10:58 +03:00
# new way of serving images
# instead always generate a link locally, but redirect that on the fly
# this way the page can load faster and images will trickle in without having to resort to XHTTP requests
def resolveImage ( artist = None , track = None ) :
if track is not None :
2019-03-12 13:39:36 +03:00
return getTrackImage ( track [ " artists " ] , track [ " title " ] )
2019-02-20 23:10:58 +03:00
elif artist is not None :
2019-03-12 13:39:36 +03:00
return getArtistImage ( artist )
2019-04-03 18:16:27 +03:00
2019-11-22 01:14:53 +03:00
def set_image ( b64 , * * keys ) :
track = " title " in keys
regex = r " data:image/( \ w+);base64,(.+) "
type , b64 = re . fullmatch ( regex , b64 ) . groups ( )
print ( b64 [ : 40 ] )
b64 = base64 . b64decode ( b64 )
filename = " webupload " + str ( int ( datetime . datetime . now ( ) . timestamp ( ) ) ) + " . " + type
for folder in get_all_possible_filenames ( * * keys ) :
if os . path . exists ( folder ) :
with open ( os . path . join ( folder , filename ) , " wb " ) as f :
f . write ( b64 )
# set as current picture in rotation
if track : local_track_cache . add ( ( frozenset ( keys [ " artists " ] ) , keys [ " title " ] ) , os . path . join ( folder , filename ) )
else : local_artist_cache . add ( keys [ " artist " ] , os . path . join ( folder , filename ) )
return
2019-04-03 18:16:27 +03:00
2019-11-22 01:14:53 +03:00
folder = get_all_possible_filenames ( * * keys ) [ 0 ]
os . makedirs ( folder )
with open ( os . path . join ( folder , filename ) , " wb " ) as f :
f . write ( b64 )
2019-04-03 18:16:27 +03:00
2019-11-22 01:14:53 +03:00
# set as current picture in rotation
if track : local_track_cache . add ( ( frozenset ( keys [ " artists " ] ) , keys [ " title " ] ) , os . path . join ( folder , filename ) )
else : local_artist_cache . add ( keys [ " artist " ] , os . path . join ( folder , filename ) )
2019-04-03 18:16:27 +03:00
#####
## PULSE MAINTENANCE
#####
2019-04-05 18:18:17 +03:00
@yearly
def update_medals ( ) :
2019-04-03 18:16:27 +03:00
2019-04-04 22:29:03 +03:00
from database import MEDALS , MEDALS_TRACKS , STAMPS , get_charts_artists , get_charts_tracks
2019-04-03 18:16:27 +03:00
currentyear = datetime . datetime . utcnow ( ) . year
2019-08-17 17:40:23 +03:00
try :
firstyear = datetime . datetime . utcfromtimestamp ( STAMPS [ 0 ] ) . year
except :
firstyear = currentyear
2019-04-04 22:29:03 +03:00
MEDALS . clear ( )
2019-04-03 18:16:27 +03:00
for year in range ( firstyear , currentyear ) :
charts = get_charts_artists ( within = [ year ] )
for a in charts :
artist = a [ " artist " ]
2019-04-04 22:29:03 +03:00
if a [ " rank " ] == 1 : MEDALS . setdefault ( artist , { } ) . setdefault ( " gold " , [ ] ) . append ( year )
elif a [ " rank " ] == 2 : MEDALS . setdefault ( artist , { } ) . setdefault ( " silver " , [ ] ) . append ( year )
elif a [ " rank " ] == 3 : MEDALS . setdefault ( artist , { } ) . setdefault ( " bronze " , [ ] ) . append ( year )
else : break
MEDALS_TRACKS . clear ( )
for year in range ( firstyear , currentyear ) :
charts = get_charts_tracks ( within = [ year ] )
for t in charts :
2019-04-03 18:16:27 +03:00
2019-04-04 22:29:03 +03:00
track = ( frozenset ( t [ " track " ] [ " artists " ] ) , t [ " track " ] [ " title " ] )
if t [ " rank " ] == 1 : MEDALS_TRACKS . setdefault ( track , { } ) . setdefault ( " gold " , [ ] ) . append ( year )
elif t [ " rank " ] == 2 : MEDALS_TRACKS . setdefault ( track , { } ) . setdefault ( " silver " , [ ] ) . append ( year )
elif t [ " rank " ] == 3 : MEDALS_TRACKS . setdefault ( track , { } ) . setdefault ( " bronze " , [ ] ) . append ( year )
else : break
2019-06-27 12:04:45 +03:00
@daily
def update_weekly ( ) :
2019-06-27 12:25:11 +03:00
from database import WEEKLY_TOPTRACKS , WEEKLY_TOPARTISTS , get_charts_artists , get_charts_tracks
from malojatime import ranges , thisweek
2019-06-27 12:04:45 +03:00
2019-06-27 12:25:11 +03:00
WEEKLY_TOPARTISTS . clear ( )
2019-06-27 12:04:45 +03:00
WEEKLY_TOPTRACKS . clear ( )
2019-06-27 12:25:11 +03:00
for week in ranges ( step = " week " ) :
if week == thisweek ( ) : break
for a in get_charts_artists ( timerange = week ) :
artist = a [ " artist " ]
if a [ " rank " ] == 1 : WEEKLY_TOPARTISTS [ artist ] = WEEKLY_TOPARTISTS . setdefault ( artist , 0 ) + 1
2019-06-27 12:04:45 +03:00
2019-06-27 12:25:11 +03:00
for t in get_charts_tracks ( timerange = week ) :
track = ( frozenset ( t [ " track " ] [ " artists " ] ) , t [ " track " ] [ " title " ] )
if t [ " rank " ] == 1 : WEEKLY_TOPTRACKS [ track ] = WEEKLY_TOPTRACKS . setdefault ( track , 0 ) + 1