Add --keep-folders option
This commit is contained in:
parent
eba6fc991b
commit
562ac26345
|
@ -186,6 +186,8 @@ def _import(destination, source, file, album_from_folder, trash,
|
||||||
@click.option('--ignore-tags', '-i', default=set(), multiple=True,
|
@click.option('--ignore-tags', '-i', default=set(), multiple=True,
|
||||||
help='Specific tags or group that will be ignored when\
|
help='Specific tags or group that will be ignored when\
|
||||||
searching for file data. Example \'File:FileModifyDate\' or \'Filename\'' )
|
searching for file data. Example \'File:FileModifyDate\' or \'Filename\'' )
|
||||||
|
@click.option('--keep-folders', '-k', default=None,
|
||||||
|
help='Folder from given level are keep back')
|
||||||
@click.option('--remove-duplicates', '-r', default=False, is_flag=True,
|
@click.option('--remove-duplicates', '-r', default=False, is_flag=True,
|
||||||
help='True to remove files that are exactly the same in name\
|
help='True to remove files that are exactly the same in name\
|
||||||
and a file hash')
|
and a file hash')
|
||||||
|
@ -193,7 +195,7 @@ def _import(destination, source, file, album_from_folder, trash,
|
||||||
help='True if you want to see details of file processing')
|
help='True if you want to see details of file processing')
|
||||||
@click.argument('paths', required=True, nargs=-1, type=click.Path())
|
@click.argument('paths', required=True, nargs=-1, type=click.Path())
|
||||||
def _sort(debug, dry_run, destination, copy, exclude_regex, filter_by_ext, ignore_tags,
|
def _sort(debug, dry_run, destination, copy, exclude_regex, filter_by_ext, ignore_tags,
|
||||||
remove_duplicates, verbose, paths):
|
keep_folders, remove_duplicates, verbose, paths):
|
||||||
"""Sort files or directories by reading their EXIF and organizing them
|
"""Sort files or directories by reading their EXIF and organizing them
|
||||||
according to config.ini preferences.
|
according to config.ini preferences.
|
||||||
"""
|
"""
|
||||||
|
@ -210,6 +212,9 @@ def _sort(debug, dry_run, destination, copy, exclude_regex, filter_by_ext, ignor
|
||||||
else:
|
else:
|
||||||
constants.debug = logging.ERROR
|
constants.debug = logging.ERROR
|
||||||
|
|
||||||
|
if keep_folders is not None:
|
||||||
|
keep_folders = int(keep_folders)
|
||||||
|
|
||||||
logger = logging.getLogger('elodie')
|
logger = logging.getLogger('elodie')
|
||||||
logger.setLevel(constants.debug)
|
logger.setLevel(constants.debug)
|
||||||
|
|
||||||
|
@ -243,7 +248,7 @@ def _sort(debug, dry_run, destination, copy, exclude_regex, filter_by_ext, ignor
|
||||||
else:
|
else:
|
||||||
day_begins = 0
|
day_begins = 0
|
||||||
filesystem = FileSystem(mode, dry_run, exclude_regex_list, logger,
|
filesystem = FileSystem(mode, dry_run, exclude_regex_list, logger,
|
||||||
day_begins, filter_by_ext)
|
day_begins, filter_by_ext, keep_folders)
|
||||||
|
|
||||||
summary, has_errors = filesystem.sort_files(paths, destination, db,
|
summary, has_errors = filesystem.sort_files(paths, destination, db,
|
||||||
remove_duplicates, ignore_tags)
|
remove_duplicates, ignore_tags)
|
||||||
|
|
|
@ -29,7 +29,7 @@ class FileSystem(object):
|
||||||
"""A class for interacting with the file system."""
|
"""A class for interacting with the file system."""
|
||||||
|
|
||||||
def __init__(self, mode='copy', dry_run=False, exclude_regex_list=set(),
|
def __init__(self, mode='copy', dry_run=False, exclude_regex_list=set(),
|
||||||
logger=logging.getLogger(), day_begins=0, filter_by_ext=()):
|
logger=logging.getLogger(), day_begins=0, filter_by_ext=(), keep_folders=None):
|
||||||
# The default folder path is along the lines of 2017-06-17_01-04-14-dsc_1234-some-title.jpg
|
# The default folder path is along the lines of 2017-06-17_01-04-14-dsc_1234-some-title.jpg
|
||||||
self.default_file_name_definition = {
|
self.default_file_name_definition = {
|
||||||
'date': '%Y-%m-%d_%H-%M-%S',
|
'date': '%Y-%m-%d_%H-%M-%S',
|
||||||
|
@ -58,6 +58,7 @@ class FileSystem(object):
|
||||||
self.summary = Summary()
|
self.summary = Summary()
|
||||||
self.day_begins = day_begins
|
self.day_begins = day_begins
|
||||||
self.filter_by_ext = filter_by_ext
|
self.filter_by_ext = filter_by_ext
|
||||||
|
self.keep_folders = keep_folders
|
||||||
|
|
||||||
# Instantiate a plugins object
|
# Instantiate a plugins object
|
||||||
self.plugins = Plugins()
|
self.plugins = Plugins()
|
||||||
|
@ -101,6 +102,22 @@ class FileSystem(object):
|
||||||
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def walklevel(self, src_path, maxlevel=None):
|
||||||
|
"""
|
||||||
|
Walk into input directory recursively until desired maxlevel
|
||||||
|
source: https://stackoverflow.com/questions/229186/os-walk-without-digging-into-directories-below
|
||||||
|
"""
|
||||||
|
src_path = src_path.rstrip(os.path.sep)
|
||||||
|
assert os.path.isdir(src_path)
|
||||||
|
num_sep = src_path.count(os.path.sep)
|
||||||
|
for root, dirs, files in os.walk(src_path):
|
||||||
|
level = root.count(os.path.sep) - num_sep
|
||||||
|
yield root, dirs, files, level
|
||||||
|
if maxlevel is not None and level >= maxlevel:
|
||||||
|
del dirs[:]
|
||||||
|
|
||||||
|
|
||||||
def get_all_files(self, path, extensions=False, exclude_regex_list=set()):
|
def get_all_files(self, path, extensions=False, exclude_regex_list=set()):
|
||||||
"""Recursively get all files which match a path and extension.
|
"""Recursively get all files which match a path and extension.
|
||||||
|
|
||||||
|
@ -759,19 +776,54 @@ class FileSystem(object):
|
||||||
return self.summary, has_errors
|
return self.summary, has_errors
|
||||||
|
|
||||||
|
|
||||||
def get_all_files_in_path(self, path, exclude_regex_list=set()):
|
def get_files_in_path(self, path, extensions=False):
|
||||||
files = set()
|
"""Recursively get files which match a path and extension.
|
||||||
# some error checking
|
|
||||||
if not os.path.exists(path):
|
|
||||||
self.logger.error(f'Directory {path} does not exist')
|
|
||||||
|
|
||||||
path = os.path.expanduser(path)
|
:param str path string: Path to start recursive file listing
|
||||||
if os.path.isdir(path):
|
:param tuple(str) extensions: File extensions to include (whitelist)
|
||||||
files.update(self.get_all_files(path, False, exclude_regex_list))
|
:returns: file_path, subdirs
|
||||||
else:
|
"""
|
||||||
|
if self.filter_by_ext != () and not extensions:
|
||||||
|
# Filtering files by extensions.
|
||||||
|
if '%media' in self.filter_by_ext:
|
||||||
|
extensions = set()
|
||||||
|
subclasses = get_all_subclasses()
|
||||||
|
for cls in subclasses:
|
||||||
|
extensions.update(cls.extensions)
|
||||||
|
else:
|
||||||
|
extensions = self.filter_by_ext
|
||||||
|
|
||||||
|
file_list = set()
|
||||||
|
if os.path.isfile(path):
|
||||||
if not self.should_exclude(path, self.exclude_regex_list, True):
|
if not self.should_exclude(path, self.exclude_regex_list, True):
|
||||||
files.add(path)
|
file_list.add((path, ''))
|
||||||
return files
|
|
||||||
|
# Create a list of compiled regular expressions to match against the file path
|
||||||
|
compiled_regex_list = [re.compile(regex) for regex in self.exclude_regex_list]
|
||||||
|
|
||||||
|
subdirs = ''
|
||||||
|
for dirname, dirnames, filenames, level in self.walklevel(path):
|
||||||
|
if dirname == os.path.join(path, '.elodie'):
|
||||||
|
continue
|
||||||
|
if self.keep_folders is not None:
|
||||||
|
if level < self.keep_folders:
|
||||||
|
subdirs = ''
|
||||||
|
else:
|
||||||
|
subdirs = os.path.join(subdirs, os.path.basename(dirname))
|
||||||
|
|
||||||
|
for filename in filenames:
|
||||||
|
# If file extension is in `extensions`
|
||||||
|
# And if file path is not in exclude regexes
|
||||||
|
# Then append to the list
|
||||||
|
filename_path = os.path.join(dirname, filename)
|
||||||
|
if (
|
||||||
|
extensions == False
|
||||||
|
or os.path.splitext(filename)[1][1:].lower() in extensions
|
||||||
|
and not self.should_exclude(filename_path, compiled_regex_list, False)
|
||||||
|
):
|
||||||
|
file_list.add((filename_path, subdirs))
|
||||||
|
|
||||||
|
return file_list
|
||||||
|
|
||||||
|
|
||||||
def sort_files(self, paths, destination, db, remove_duplicates=False,
|
def sort_files(self, paths, destination, db, remove_duplicates=False,
|
||||||
|
@ -779,11 +831,14 @@ class FileSystem(object):
|
||||||
|
|
||||||
has_errors = False
|
has_errors = False
|
||||||
for path in paths:
|
for path in paths:
|
||||||
files = self.get_all_files_in_path(path, self.exclude_regex_list)
|
# some error checking
|
||||||
num_files = len(files)
|
if not os.path.exists(path):
|
||||||
|
self.logger.error(f'Directory {path} does not exist')
|
||||||
|
|
||||||
|
path = os.path.expanduser(path)
|
||||||
|
|
||||||
conflict_file_list = set()
|
conflict_file_list = set()
|
||||||
for src_path in files:
|
for src_path, subdirs in self.get_files_in_path(path):
|
||||||
# Process files
|
# Process files
|
||||||
media = get_media_class(src_path, ignore_tags)
|
media = get_media_class(src_path, ignore_tags)
|
||||||
if media:
|
if media:
|
||||||
|
@ -798,7 +853,7 @@ class FileSystem(object):
|
||||||
file_name = os.path.basename(src_path)
|
file_name = os.path.basename(src_path)
|
||||||
|
|
||||||
dest_directory = os.path.join(destination, directory_name)
|
dest_directory = os.path.join(destination, directory_name)
|
||||||
dest_path = os.path.join(dest_directory, file_name)
|
dest_path = os.path.join(dest_directory, subdirs, file_name)
|
||||||
self.create_directory(dest_directory)
|
self.create_directory(dest_directory)
|
||||||
result = self.sort_file(src_path, dest_path, remove_duplicates)
|
result = self.sort_file(src_path, dest_path, remove_duplicates)
|
||||||
if result:
|
if result:
|
||||||
|
|
|
@ -113,6 +113,17 @@ def test_delete_directory_if_empty_when_not_empty():
|
||||||
|
|
||||||
shutil.rmtree(parent_folder)
|
shutil.rmtree(parent_folder)
|
||||||
|
|
||||||
|
|
||||||
|
def test_walklevel():
|
||||||
|
filesystem = FileSystem()
|
||||||
|
maxlevel=2
|
||||||
|
for root, dirs, files, level in filesystem.walklevel(helper.get_file_path('dir'), maxlevel):
|
||||||
|
for paths in root, dirs, files:
|
||||||
|
for path in paths:
|
||||||
|
assert isinstance(path, str), path
|
||||||
|
assert level <= maxlevel, level
|
||||||
|
|
||||||
|
|
||||||
def test_get_all_files_success():
|
def test_get_all_files_success():
|
||||||
filesystem = FileSystem()
|
filesystem = FileSystem()
|
||||||
folder = helper.populate_folder(5)
|
folder = helper.populate_folder(5)
|
||||||
|
|
Loading…
Reference in New Issue