Source code for abjad.configuration

import collections
import configparser
import importlib
import os
import pathlib
import subprocess
import tempfile
import time
import traceback
import types
import typing

import uqbar.apis


[docs] class Configuration: """ Configuration. .. container:: example Behavior at instantiation: * Looks for ``$HOME/.abjad/``. * Creates ``$HOME/.abjad/`` if directory does not exist. * Looks for ``$HOME/.abjad/abjad.cfg``. * Creates ``$HOME/.abjad/abjad.cfg`` if file does not exist. * Parses ``$HOME/.abjad/abjad.cfg``. * Provides default key-value pairs for pairs which fail validation. * Writes configuration changes to disk. * Creates Abjad output directory if directory does not exist. Supports mutable mapping dictionary interface. """ ### CLASS VARIABLES ### __documentation_section__ = "System configuration" __slots__ = ("_cached_configuration_directory", "_settings") _configuration_directory_name = ".abjad" _configuration_file_name = "abjad.cfg" # for caching _lilypond_version_string: typing.ClassVar[str | None] = None ### INITIALIZER ### def __init__(self): self._cached_configuration_directory = None if not os.path.exists(str(self.configuration_directory)): try: os.makedirs(str(self.configuration_directory)) except (IOError, OSError): traceback.print_exc() old_contents = "" if self.configuration_file_path.exists(): try: old_contents = self.configuration_file_path.read_text() except AttributeError: with self.configuration_file_path.open(mode="r") as f: old_contents = f.read() configuration = self._configuration_from_string(old_contents) configuration = self._validate_configuration(configuration) new_contents = self._configuration_to_string(configuration) if not self._compare_configurations(old_contents, new_contents): try: with open(str(self.configuration_file_path), "w") as file_pointer: print(new_contents, file=file_pointer) except (IOError, OSError): traceback.print_exc() self._settings = configuration self._make_missing_directories() ### SPECIAL METHODS ###
[docs] def __delitem__(self, i) -> None: """ Deletes item ``i`` from configuration. """ del self._settings[i]
[docs] def __eq__(self, argument): """ Returns true when ``argument`` is configuratioin with same settings. """ if isinstance(argument, type(self)): return self._settings == argument._settings return False
[docs] def __getitem__(self, argument) -> typing.Any: """ Gets item or slice identified by ``argument``. """ return self._settings.__getitem__(argument)
[docs] def __iter__(self) -> typing.Iterator[str]: """ Iterates configuration settings. """ for key in self._settings: yield key
[docs] def __setitem__(self, i, argument) -> None: """ Sets configuration item ``i`` to ``argument``. """ self._settings[i] = argument
### PRIVATE METHODS ### def _compare_configurations(self, old, new): old = "\n".join(old.splitlines()[3:]) new = "\n".join(new.splitlines()[3:]) return old == new def _configuration_from_string(self, string): if "[main]" not in string: string = "[main]\n" + string config_parser = configparser.ConfigParser() try: config_parser.read_string(string) configuration = dict(config_parser["main"].items()) except configparser.ParsingError: configuration = {} return configuration def _configuration_to_string(self, configuration): option_definitions = self._get_option_definitions() known_items, unknown_items = [], [] for key, value in sorted(configuration.items()): if key in option_definitions: known_items.append((key, value)) else: unknown_items.append((key, value)) result = [] for line in self._get_initial_comment(): if line: result.append(f"# {line}") else: result.append("") for key, value in known_items: result.append("") if key in option_definitions: for line in option_definitions[key]["comment"]: if line: result.append(f"# {line}") else: result.append("") if value not in ("", None): result.append(f"{key!s} = {value!s}") else: result.append(f"{key!s} =") if unknown_items: result.append("") result.append("# User-specified keys:") for key, value in unknown_items: result.append("") if value not in ("", None): result.append(f"{key!s} = {value!s}") else: result.append(f"{key!s} =") string = "\n".join(result) return string def _get_config_specification(self): specs = self._get_option_specification() return [f"{key} = {value}" for key, value in sorted(specs.items())] def _get_current_time(self): return time.strftime("%d %B %Y %H:%M:%S") def _get_initial_comment(self): current_time = self._get_current_time() return [ f"Abjad configuration file created on {current_time}.", "This file is interpreted by Python's ConfigParser ", "and follows ini syntax.", ] def _get_option_comments(self): options = self._get_option_definitions() comments = [(key, options[key]["comment"]) for key in options] return dict(comments) def _get_option_definitions(self): options = { "abjad_output_directory": { "comment": [ "Set to the directory where all Abjad-generated files", "(such as PDFs and LilyPond files) should be saved.", "Defaults to $HOME/.abjad/output/", ], "default": os.path.join(str(self.configuration_directory), "output"), "validator": str, }, "lilypond_path": { "comment": [ "Lilypond executable path. Set to override dynamic lookup." ], "default": "lilypond", "validator": str, }, "midi_player": { "comment": [ "MIDI player to open MIDI files.", "When unset your OS should know how to open MIDI files.", ], "default": None, "validator": str, }, "pdf_viewer": { "comment": [ "PDF viewer to open PDF files.", "When unset your OS should know how to open PDFs.", ], "default": None, "validator": str, }, "text_editor": { "comment": [ "Text editor to edit text files.", "When unset your OS should know how to open text files.", ], "default": None, "validator": str, }, } return options def _get_option_specification(self): options = self._get_option_definitions() specs = [(key, options[key]["spec"]) for key in options] return dict(specs) def _make_missing_directories(self): if not os.path.exists(self.abjad_output_directory): try: os.makedirs(self.abjad_output_directory) except (IOError, OSError): traceback.print_exc() def _validate_configuration(self, configuration): option_definitions = self._get_option_definitions() for key in option_definitions: if key not in configuration: configuration[key] = option_definitions[key]["default"] validator = option_definitions[key]["validator"] if isinstance(validator, type): if not isinstance(configuration[key], validator): configuration[key] = option_definitions[key]["default"] else: if not validator(configuration[key]): configuration[key] = option_definitions[key]["default"] for key in configuration: if configuration[key] in ("", "None"): configuration[key] = None return configuration ### PUBLIC PROPERTIES ### @property def abjad_directory(self) -> pathlib.Path: """ Gets Abjad directory. """ return pathlib.Path(__file__).parent.parent @property def abjad_output_directory(self) -> pathlib.Path: """ Gets Abjad output directory. """ if "abjad_output_directory" in self._settings: return pathlib.Path(self._settings["abjad_output_directory"]) return self.configuration_directory / "output" @property def boilerplate_directory(self) -> pathlib.Path: """ Gets Abjad boilerplate directory. """ return self.abjad_directory.parent / "boilerplate" @property def configuration_directory(self) -> pathlib.Path: """ Gets configuration directory. .. container:: example >>> configuration = abjad.Configuration() >>> configuration.configuration_directory PosixPath('...') Defaults to $HOME/{directory_name}. Returns $TEMP/{directory_name} if $HOME is read-only or $HOME/{directory_name} is read-only. Also caches the initial result to reduce filesystem interaction. """ if self._cached_configuration_directory is None: directory_name = self._configuration_directory_name home_directory = str(self.home_directory) flags = os.W_OK | os.X_OK if os.access(home_directory, flags): path = self.home_directory / directory_name if not path.exists() or (path.exists() and os.access(str(path), flags)): self._cached_configuration_directory = path return self._cached_configuration_directory path = pathlib.Path(tempfile.gettempdir()) / directory_name self._cached_configuration_directory = path return self._cached_configuration_directory @property def configuration_file_path(self) -> pathlib.Path: """ Gets configuration file path. .. container:: example >>> configuration = abjad.Configuration() >>> configuration.configuration_file_path PosixPath('...') """ return self.configuration_directory / self._configuration_file_name @property def home_directory(self) -> pathlib.Path: """ Gets home directory. .. container:: example >>> configuration = abjad.Configuration() >>> configuration.home_directory PosixPath('...') """ path = ( os.environ.get("HOME") or os.environ.get("HOMEPATH") or os.environ.get("APPDATA") or tempfile.gettempdir() ) return pathlib.Path(path).absolute() @property def lilypond_log_file_path(self) -> pathlib.Path: """ Gets LilyPond log file path. """ return self.abjad_output_directory / "lily.log" ### PUBLIC METHODS ###
[docs] def get(self, *arguments, **keywords): """ Gets a key. """ return self._settings.get(*arguments, **keywords)
[docs] def get_lilypond_version_string(self) -> str: """ Gets LilyPond version string. .. container:: example >>> configuration = abjad.Configuration() >>> configuration.get_lilypond_version_string() # doctest: +SKIP '2.19.84' """ if self._lilypond_version_string is not None: return self._lilypond_version_string command = ["lilypond", "--version"] proc = subprocess.run(command, stdout=subprocess.PIPE) assert proc.stdout is not None lilypond_version_string = proc.stdout.decode().split()[2] Configuration._lilypond_version_string = lilypond_version_string return lilypond_version_string
[docs] def list_all_classes(modules="abjad", ignored_classes=None): """ Lists all public classes defined in ``path``. .. container:: example >>> for class_ in abjad.list_all_classes(modules="abjad"): class_ <class 'abjad.bind.Wrapper'> <class 'abjad.configuration.Configuration'> <class 'abjad.contextmanagers.ContextManager'> <class 'abjad.contextmanagers.FilesystemState'> <class 'abjad.contextmanagers.ForbidUpdate'> <class 'abjad.contextmanagers.NullContextManager'> <class 'abjad.contextmanagers.ProgressIndicator'> <class 'abjad.contextmanagers.RedirectedStreams'> <class 'abjad.contextmanagers.TemporaryDirectory'> <class 'abjad.contextmanagers.TemporaryDirectoryChange'> <class 'abjad.contextmanagers.Timer'> <class 'abjad.contributions.ContributionsBySite'> <class 'abjad.cyclictuple.CyclicTuple'> <class 'abjad.duration.Duration'> <class 'abjad.duration.Offset'> <class 'abjad.exceptions.AssignabilityError'> <class 'abjad.exceptions.ImpreciseMetronomeMarkError'> <class 'abjad.exceptions.LilyPondParserError'> <class 'abjad.exceptions.MissingContextError'> <class 'abjad.exceptions.MissingMetronomeMarkError'> <class 'abjad.exceptions.ParentageError'> <class 'abjad.exceptions.PersistentIndicatorError'> <class 'abjad.exceptions.SchemeParserFinishedError'> <class 'abjad.exceptions.UnboundedTimeIntervalError'> <class 'abjad.exceptions.WellformednessError'> <class 'abjad.get.Lineage'> <class 'abjad.indicators.Arpeggio'> <class 'abjad.indicators.Articulation'> <class 'abjad.indicators.BarLine'> <class 'abjad.indicators.BeamCount'> <class 'abjad.indicators.BendAfter'> <class 'abjad.indicators.BreathMark'> <class 'abjad.indicators.Clef'> <class 'abjad.indicators.ColorFingering'> <class 'abjad.indicators.Dynamic'> <class 'abjad.indicators.Fermata'> <class 'abjad.indicators.Glissando'> <class 'abjad.indicators.InstrumentName'> <class 'abjad.indicators.KeyCluster'> <class 'abjad.indicators.KeySignature'> <class 'abjad.indicators.LaissezVibrer'> <class 'abjad.indicators.LilyPondLiteral'> <class 'abjad.indicators.Markup'> <class 'abjad.indicators.MetronomeMark'> <class 'abjad.indicators.Mode'> <class 'abjad.indicators.Ottava'> <class 'abjad.indicators.RehearsalMark'> <class 'abjad.indicators.Repeat'> <class 'abjad.indicators.RepeatTie'> <class 'abjad.indicators.ShortInstrumentName'> <class 'abjad.indicators.StaffChange'> <class 'abjad.indicators.StartBeam'> <class 'abjad.indicators.StartGroup'> <class 'abjad.indicators.StartHairpin'> <class 'abjad.indicators.StartPhrasingSlur'> <class 'abjad.indicators.StartPianoPedal'> <class 'abjad.indicators.StartSlur'> <class 'abjad.indicators.StartTextSpan'> <class 'abjad.indicators.StartTrillSpan'> <class 'abjad.indicators.StemTremolo'> <class 'abjad.indicators.StopBeam'> <class 'abjad.indicators.StopGroup'> <class 'abjad.indicators.StopHairpin'> <class 'abjad.indicators.StopPhrasingSlur'> <class 'abjad.indicators.StopPianoPedal'> <class 'abjad.indicators.StopSlur'> <class 'abjad.indicators.StopTextSpan'> <class 'abjad.indicators.StopTrillSpan'> <class 'abjad.indicators.TextMark'> <class 'abjad.indicators.Tie'> <class 'abjad.indicators.TimeSignature'> <class 'abjad.indicators.VoiceNumber'> <class 'abjad.instruments.Accordion'> <class 'abjad.instruments.AltoFlute'> <class 'abjad.instruments.AltoSaxophone'> <class 'abjad.instruments.AltoTrombone'> <class 'abjad.instruments.AltoVoice'> <class 'abjad.instruments.BaritoneSaxophone'> <class 'abjad.instruments.BaritoneVoice'> <class 'abjad.instruments.BassClarinet'> <class 'abjad.instruments.BassFlute'> <class 'abjad.instruments.BassSaxophone'> <class 'abjad.instruments.BassTrombone'> <class 'abjad.instruments.BassVoice'> <class 'abjad.instruments.Bassoon'> <class 'abjad.instruments.Cello'> <class 'abjad.instruments.ClarinetInA'> <class 'abjad.instruments.ClarinetInBFlat'> <class 'abjad.instruments.ClarinetInEFlat'> <class 'abjad.instruments.Contrabass'> <class 'abjad.instruments.ContrabassClarinet'> <class 'abjad.instruments.ContrabassFlute'> <class 'abjad.instruments.ContrabassSaxophone'> <class 'abjad.instruments.Contrabassoon'> <class 'abjad.instruments.EnglishHorn'> <class 'abjad.instruments.Flute'> <class 'abjad.instruments.FrenchHorn'> <class 'abjad.instruments.Glockenspiel'> <class 'abjad.instruments.Guitar'> <class 'abjad.instruments.Harp'> <class 'abjad.instruments.Harpsichord'> <class 'abjad.instruments.Instrument'> <class 'abjad.instruments.Marimba'> <class 'abjad.instruments.MezzoSopranoVoice'> <class 'abjad.instruments.Oboe'> <class 'abjad.instruments.Percussion'> <class 'abjad.instruments.Piano'> <class 'abjad.instruments.Piccolo'> <class 'abjad.instruments.SopraninoSaxophone'> <class 'abjad.instruments.SopranoSaxophone'> <class 'abjad.instruments.SopranoVoice'> <class 'abjad.instruments.StringNumber'> <class 'abjad.instruments.TenorSaxophone'> <class 'abjad.instruments.TenorTrombone'> <class 'abjad.instruments.TenorVoice'> <class 'abjad.instruments.Trumpet'> <class 'abjad.instruments.Tuba'> <class 'abjad.instruments.Tuning'> <class 'abjad.instruments.Vibraphone'> <class 'abjad.instruments.Viola'> <class 'abjad.instruments.Violin'> <class 'abjad.instruments.Xylophone'> <class 'abjad.label.ColorMap'> <class 'abjad.lilypondfile.Block'> <class 'abjad.lilypondfile.LilyPondFile'> <class 'abjad.lyproxy.LilyPondContext'> <class 'abjad.lyproxy.LilyPondEngraver'> <class 'abjad.lyproxy.LilyPondGrob'> <class 'abjad.lyproxy.LilyPondGrobInterface'> <class 'abjad.math.Infinity'> <class 'abjad.math.NegativeInfinity'> <class 'abjad.meter.Meter'> <class 'abjad.meter.MetricAccentKernel'> <class 'abjad.metricmodulation.MetricModulation'> <class 'abjad.obgc.OnBeatGraceContainer'> <class 'abjad.overrides.Interface'> <class 'abjad.overrides.LilyPondOverride'> <class 'abjad.overrides.LilyPondSetting'> <class 'abjad.overrides.OverrideInterface'> <class 'abjad.overrides.SettingInterface'> <class 'abjad.parentage.Parentage'> <class 'abjad.parsers.base.Parser'> <class 'abjad.pattern.Pattern'> <class 'abjad.pattern.PatternTuple'> <class 'abjad.pcollections.PitchClassSegment'> <class 'abjad.pcollections.PitchClassSet'> <class 'abjad.pcollections.PitchRange'> <class 'abjad.pcollections.PitchSegment'> <class 'abjad.pcollections.PitchSet'> <class 'abjad.pcollections.TwelveToneRow'> <class 'abjad.pitch.Accidental'> <class 'abjad.pitch.Interval'> <class 'abjad.pitch.IntervalClass'> <class 'abjad.pitch.NamedInterval'> <class 'abjad.pitch.NamedIntervalClass'> <class 'abjad.pitch.NamedInversionEquivalentIntervalClass'> <class 'abjad.pitch.NamedPitch'> <class 'abjad.pitch.NamedPitchClass'> <class 'abjad.pitch.NumberedInterval'> <class 'abjad.pitch.NumberedIntervalClass'> <class 'abjad.pitch.NumberedInversionEquivalentIntervalClass'> <class 'abjad.pitch.NumberedPitch'> <class 'abjad.pitch.NumberedPitchClass'> <class 'abjad.pitch.Octave'> <class 'abjad.pitch.Pitch'> <class 'abjad.pitch.PitchClass'> <class 'abjad.pitch.StaffPosition'> <class 'abjad.rhythmtrees.RhythmTreeContainer'> <class 'abjad.rhythmtrees.RhythmTreeLeaf'> <class 'abjad.rhythmtrees.RhythmTreeMixin'> <class 'abjad.rhythmtrees.RhythmTreeParser'> <class 'abjad.score.AfterGraceContainer'> <class 'abjad.score.BeforeGraceContainer'> <class 'abjad.score.Chord'> <class 'abjad.score.Cluster'> <class 'abjad.score.Component'> <class 'abjad.score.Container'> <class 'abjad.score.Context'> <class 'abjad.score.DrumNoteHead'> <class 'abjad.score.IndependentAfterGraceContainer'> <class 'abjad.score.Leaf'> <class 'abjad.score.MultimeasureRest'> <class 'abjad.score.Note'> <class 'abjad.score.NoteHead'> <class 'abjad.score.NoteHeadList'> <class 'abjad.score.Rest'> <class 'abjad.score.Score'> <class 'abjad.score.Skip'> <class 'abjad.score.Staff'> <class 'abjad.score.StaffGroup'> <class 'abjad.score.TremoloContainer'> <class 'abjad.score.Tuplet'> <class 'abjad.score.Voice'> <class 'abjad.select.LogicalTie'> <class 'abjad.setclass.SetClass'> <class 'abjad.tag.Tag'> <class 'abjad.timespan.OffsetCounter'> <class 'abjad.timespan.Timespan'> <class 'abjad.timespan.TimespanList'> <class 'abjad.tweaks.Bundle'> <class 'abjad.tweaks.Tweak'> <class 'abjad.verticalmoment.VerticalMoment'> """ all_classes = set() for module in yield_all_modules(modules): if "parser" in module.__name__: continue name = module.__name__.split(".")[-1] for name in dir(module): item = getattr(module, name) if isinstance(item, type): if "sphinx" in repr(item): continue if item.__name__.startswith("_"): continue if "abjad.io" in str(item): continue if "abjad" in repr(item): all_classes.add(item) if ignored_classes: ignored_classes = set(ignored_classes) all_classes.difference_update(ignored_classes) return list(sorted(all_classes, key=lambda _: (_.__module__, _.__name__)))
[docs] def list_all_functions(modules="abjad"): """ Lists all public functions defined in ``modules``. .. container:: example >>> functions = abjad.list_all_functions(modules="abjad") >>> names = [_.__name__ for _ in functions] >>> # for name in sorted(names): name """ all_functions = set() for module in yield_all_modules(modules): name = module.__name__.split(".")[-1] if name.startswith("_"): continue if "sphinx" in repr(module): continue for name in sorted(dir(module)): item = getattr(module, name) if isinstance(item, types.FunctionType): if item.__name__.startswith("_"): continue if "abjad" not in repr(item.__module__): continue all_functions.add(item) return list(sorted(all_functions, key=lambda _: (_.__module__, _.__name__)))
configuration = Configuration()
[docs] def yield_all_modules(paths=None): """ Yields all modules encountered in ``path``. Returns generator. """ _paths = [] if not paths: _paths = configuration.abjad_directory elif isinstance(paths, str): module = importlib.import_module(paths) _paths.extend(module.__path__) elif isinstance(paths, types.ModuleType): _paths.extend(paths.__path__) elif isinstance(paths, collections.abc.Iterable): for path in paths: if isinstance(path, types.ModuleType): _paths.extend(path.__path__) elif isinstance(path, str): module = importlib.import_module(path) _paths.extend(module.__path__) else: raise ValueError(module) for path in _paths: for source_path in uqbar.apis.collect_source_paths([path]): package_path = uqbar.apis.source_path_to_package_path(source_path) yield importlib.import_module(package_path)