mirror of https://github.com/patacrep/patacrep.git
Engine for LaTeX songbooks
http://www.patacrep.com
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
143 lines
4.1 KiB
143 lines
4.1 KiB
"""File system utilities."""
|
|
|
|
from contextlib import contextmanager
|
|
import logging
|
|
import os
|
|
import pkgutil
|
|
import posixpath
|
|
import re
|
|
import sys
|
|
|
|
from patacrep import utils
|
|
from patacrep import __DATADIR__
|
|
|
|
LOGGER = logging.getLogger(__name__)
|
|
|
|
def recursive_find(root_directory, extensions=None):
|
|
"""Recursively find files with the given extensions, from a root_directory.
|
|
|
|
Return a list of files matching those conditions.
|
|
|
|
Arguments:
|
|
- `extensions`: list of accepted extensions (None means every file).
|
|
- `root_directory`: root directory of the search.
|
|
"""
|
|
if not os.path.isdir(root_directory):
|
|
return []
|
|
|
|
matches = []
|
|
if extensions is None:
|
|
pattern = re.compile('.*')
|
|
else:
|
|
pattern = re.compile(r'.*\.({})$'.format('|'.join(extensions)))
|
|
|
|
with chdir(root_directory):
|
|
for root, __ignored, filenames in os.walk(os.curdir):
|
|
for filename in filenames:
|
|
if pattern.match(filename):
|
|
matches.append(os.path.join(root, filename))
|
|
return matches
|
|
|
|
def relpath(path, start=None):
|
|
"""Return relative filepath to path if a subpath of start."""
|
|
if start is None:
|
|
start = os.curdir
|
|
if os.path.abspath(path).startswith(os.path.abspath(start)):
|
|
return os.path.relpath(path, start)
|
|
else:
|
|
return os.path.abspath(path)
|
|
|
|
|
|
def path2posix(string):
|
|
""""Convert path from local format to posix format."""
|
|
if not string or string == "/":
|
|
return string
|
|
if os.path.splitdrive(string)[1] == "\\":
|
|
# Assuming DRIVE:\
|
|
return string[0:-1]
|
|
(head, tail) = os.path.split(string)
|
|
return posixpath.join(
|
|
path2posix(head),
|
|
tail,
|
|
)
|
|
|
|
@contextmanager
|
|
def chdir(*path):
|
|
"""Locally change dir
|
|
|
|
Can be used as:
|
|
|
|
with chdir("some", "directory"):
|
|
do_stuff()
|
|
"""
|
|
olddir = os.getcwd()
|
|
if path:
|
|
os.chdir(os.path.join(*path))
|
|
yield
|
|
os.chdir(olddir)
|
|
else:
|
|
yield
|
|
|
|
def iter_modules(path, prefix):
|
|
"""Iterate over modules located in list of `path`.
|
|
|
|
Prefix is a prefix appended to all module names.
|
|
"""
|
|
for module_finder, name, __is_pkg in pkgutil.walk_packages(path, prefix):
|
|
if name in sys.modules:
|
|
yield sys.modules[name]
|
|
else:
|
|
try:
|
|
yield module_finder.find_spec(name).loader.load_module()
|
|
except ImportError as error:
|
|
LOGGER.debug("[plugins] Could not load module {}: {}".format(name, str(error)))
|
|
continue
|
|
|
|
def load_plugins(datadirs, root_modules, keyword):
|
|
"""Load all plugins, and return a dictionary of those plugins.
|
|
|
|
A plugin is a .py file, submodule of `subdir`, located in one of the
|
|
directories of `datadirs`. It contains a dictionary `keyword`. The return
|
|
value is the union of the dictionaries of the loaded plugins.
|
|
|
|
Arguments:
|
|
- datadirs: List of directories in which plugins are to be searched.
|
|
- root_modules: the submodule in which plugins are to be searched, as a
|
|
list of modules (e.g. ["some", "deep", "module"] for
|
|
"some.deep.module").
|
|
- keyword: attribute containing plugin information.
|
|
|
|
Return value: a dictionary where:
|
|
- keys are the keywords ;
|
|
- values are functions triggered when this keyword is met.
|
|
"""
|
|
plugins = utils.DictOfDict()
|
|
|
|
datadir_path = [
|
|
os.path.join(datadir, "python", *root_modules)
|
|
for datadir
|
|
in datadirs
|
|
]
|
|
sys_path = [
|
|
os.path.join(path, "patacrep", *root_modules)
|
|
for path
|
|
in sys.path
|
|
]
|
|
for module in iter_modules(
|
|
datadir_path + sys_path,
|
|
prefix="patacrep.{}.".format(".".join(root_modules))
|
|
):
|
|
if hasattr(module, keyword):
|
|
plugins.update(getattr(module, keyword))
|
|
return plugins
|
|
|
|
def iter_datadirs(datadirs, *subpath):
|
|
"""Iterate over datadirs.
|
|
|
|
The default datadir is returned last.
|
|
|
|
Subpath are appended after each datadir.
|
|
"""
|
|
for path in datadirs:
|
|
yield os.path.join(path, *subpath)
|
|
yield os.path.join(__DATADIR__, *subpath)
|
|
|