1
0
mirror of https://github.com/spl0k/supysonic.git synced 2024-11-15 06:32:16 +00:00
supysonic/scanner.py

214 lines
6.4 KiB
Python
Raw Normal View History

2012-10-13 12:53:09 +00:00
# coding: utf-8
import sys
2012-10-21 14:18:35 +00:00
import os, os.path
import time, mimetypes
import mutagen
import config, db
import math
from web import app
2012-10-13 12:53:09 +00:00
2013-10-15 11:14:20 +00:00
def get_mime(ext):
return mimetypes.guess_type('dummy.' + ext, False)[0] or config.get('mimetypes', ext) or 'application/octet-stream'
2013-10-15 11:14:20 +00:00
2012-10-13 12:53:09 +00:00
class Scanner:
def __init__(self, session):
self.__session = session
self.__tracks = db.Track.query.all()
self.__tracks = {x.path: x for x in self.__tracks}
self.__artists = db.Artist.query.all()
2012-10-21 14:18:35 +00:00
self.__folders = db.Folder.query.all()
2012-10-13 12:53:09 +00:00
self.__added_artists = 0
self.__added_albums = 0
self.__added_tracks = 0
2012-10-13 12:53:09 +00:00
self.__deleted_artists = 0
self.__deleted_albums = 0
self.__deleted_tracks = 0
2012-10-13 12:53:09 +00:00
extensions = config.get('base', 'scanner_extensions')
self.__extensions = map(str.lower, extensions.split()) if extensions else None
2012-10-13 12:53:09 +00:00
def scan(self, folder):
print "scanning", folder.path
valid = [x.lower() for x in config.get('base','filetypes').split(',')]
print "valid filetypes: ",valid
n = 0
for root, subfolders, files in os.walk(folder.path, topdown=False):
2012-10-13 12:53:09 +00:00
for f in files:
suffix = os.path.splitext(f)[1][1:].lower()
n += 1
if n == 1000:
app.logger.debug('commit db')
self.__session.commit()
n = 0
if suffix in valid:
try:
app.logger.debug('Scanning File: ' + os.path.join(root, f))
self.__scan_file(os.path.join(root, f), folder)
self.__session.flush()
except:
app.logger.error('Problem adding file: ' + os.path.join(root,f))
app.logger.error(sys.exc_info())
self.__session.rollback()
self.__session.commit()
folder.last_scan = int(time.time())
2012-10-13 12:53:09 +00:00
def prune(self, folder):
for k, track in self.__tracks.iteritems():
if track.root_folder.id == folder.id and not self.__is_valid_path(k):
self.__remove_track(track)
for album in [ album for artist in self.__artists for album in artist.albums if len(album.tracks) == 0 ]:
album.artist.albums.remove(album)
self.__session.delete(album)
self.__deleted_albums += 1
for artist in [ a for a in self.__artists if len(a.albums) == 0 ]:
self.__session.delete(artist)
self.__deleted_artists += 1
2012-10-13 12:53:09 +00:00
2012-10-21 14:18:35 +00:00
self.__cleanup_folder(folder)
2012-11-11 20:39:26 +00:00
def check_cover_art(self, folder):
folder.has_cover_art = os.path.isfile(os.path.join(folder.path, 'cover.jpg'))
for f in folder.children:
self.check_cover_art(f)
def __is_valid_path(self, path):
if not os.path.exists(path):
return False
if not self.__extensions:
return True
return os.path.splitext(path)[1][1:].lower() in self.__extensions
2012-10-14 11:07:02 +00:00
def __scan_file(self, path, folder):
curmtime = int(math.floor(os.path.getmtime(path)))
if path in self.__tracks:
tr = self.__tracks[path]
if not tr.last_modification:
tr.last_modification = curmtime
if curmtime <= tr.last_modification:
app.logger.debug('\tFile not modified')
return False
app.logger.debug('\tFile modified, updating tag')
app.logger.debug('\tcurmtime %s / last_mod %s', curmtime, tr.last_modification)
app.logger.debug('\t\t%s Seconds Newer\n\t\t', str(curmtime - tr.last_modification))
tag = self.__try_load_tag(path)
if not tag:
app.logger.debug('\tError retrieving tags, removing track from DB')
self.__remove_track(tr)
return False
else:
app.logger.debug('\tReading tag')
tag = self.__try_load_tag(path)
if not tag:
app.logger.debug('\tProblem reading tag')
return False
tr = db.Track(path = path, root_folder = folder, folder = self.__find_folder(path, folder))
self.__tracks[path] = tr
self.__added_tracks += 1
print "Added ", path
tr.last_modification = curmtime
tr.disc = self.__try_read_tag(tag, 'discnumber', 1, lambda x: int(x[0].split('/')[0]))
tr.number = self.__try_read_tag(tag, 'tracknumber', 1, lambda x: int(x[0].split('/')[0]))
2013-10-26 11:13:19 +00:00
tr.title = self.__try_read_tag(tag, 'title', '')
tr.year = self.__try_read_tag(tag, 'date', None, lambda x: int(x[0].split('-')[0]))
tr.genre = self.__try_read_tag(tag, 'genre')
tr.duration = int(tag.info.length)
tr.album = self.__find_album(self.__try_read_tag(tag, 'artist', 'Unknown'), self.__try_read_tag(tag, 'album', 'Unknown'))
2013-10-16 07:53:43 +00:00
tr.bitrate = (tag.info.bitrate if hasattr(tag.info, 'bitrate') else int(os.path.getsize(path) * 8 / tag.info.length)) / 1000
tr.content_type = get_mime(os.path.splitext(path)[1][1:])
2012-10-13 12:53:09 +00:00
return True
2012-10-13 12:53:09 +00:00
def __find_album(self, artist, album):
ar = self.__find_artist(artist)
al = filter(lambda a: a.name == album, ar.albums)
if al:
return al[0]
2012-10-13 12:53:09 +00:00
al = db.Album(name = album, artist = ar)
self.__added_albums += 1
return al
def __find_artist(self, artist):
ar = db.Artist.as_unique(self.__session, name = artist)
2012-10-13 12:53:09 +00:00
self.__artists.append(ar)
2012-10-13 12:53:09 +00:00
self.__added_artists += 1
return ar
2012-10-21 14:18:35 +00:00
def __find_folder(self, path, folder):
path = os.path.dirname(path)
fold = filter(lambda f: f.path == path, self.__folders)
if fold:
return fold[0]
full_path = folder.path
path = path[len(folder.path) + 1:]
for name in path.split(os.sep):
full_path = os.path.join(full_path, name)
fold = filter(lambda f: f.path == full_path, self.__folders)
if fold:
folder = fold[0]
else:
folder = db.Folder(root = False, name = name, path = full_path, parent = folder)
self.__folders.append(folder)
return folder
def __try_load_tag(self, path):
try:
return mutagen.File(path, easy = True)
except:
return None
def __try_read_tag(self, metadata, field, default = None, transform = lambda x: x[0]):
try:
value = metadata[field]
if not value:
return default
if transform:
value = transform(value)
return value if value else default
except:
return default
def __remove_track(self, track):
track.album.tracks.remove(track)
track.folder.tracks.remove(track)
# As we don't have a track -> playlists relationship, SQLAlchemy doesn't know it has to remove tracks
# from playlists as well, so let's help it
for playlist in db.Playlist.query.filter(db.Playlist.tracks.contains(track)):
playlist.tracks.remove(track)
self.__session.delete(track)
self.__deleted_tracks += 1
2012-10-21 14:18:35 +00:00
def __cleanup_folder(self, folder):
for f in folder.children:
self.__cleanup_folder(f)
if len(folder.children) == 0 and len(folder.tracks) == 0 and not folder.root:
folder.parent = None
self.__session.delete(folder)
2012-10-13 12:53:09 +00:00
def stats(self):
return (self.__added_artists, self.__added_albums, self.__added_tracks), (self.__deleted_artists, self.__deleted_albums, self.__deleted_tracks)