-
Notifications
You must be signed in to change notification settings - Fork 47
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support XDG base directory specification for user configuration resolution #400
Comments
Hey there, I'm thinking about doing this. I'm apprehensive because I don't have any experience with large codebases (which notably means I've never written a test) or Python. I would be willing to learn those to do this, but I'm not sure if this would end up being more work than think it will be. Would this be a suitable first contribution? From what I understand from a little bit of reading the code, the changes involved would look something like this:
# ...
- from mdformat._conf import DEFAULT_OPTS, InvalidConfError, read_toml_opts
+ from mdformat._conf import DEFAULT_OPTS, InvalidConfError, read_toml_opts, read_user_opts
# ...
for path in file_paths:
try:
+ # Looks for a config in user directories, e.x. $XDG_CONFIG_HOME
+ user_opts = read_user_opts()
+ # Looks for a config in parent directories
- toml_opts = read_toml_opts(path.parent if path else Path.cwd())
+ proj_opts = read_toml_opts(path.parent if path else Path.cwd())
except InvalidConfError as e:
print_error(str(e))
return 1
- opts: Mapping = {**DEFAULT_OPTS, **toml_opts, **cli_opts}
+ opts: Mapping = {**DEFAULT_OPTS, **user_opts, **proj_opts, **cli_opts}
if path:
path_str = str(path)
original_str = sys.stdin.read()
formatted_str = mdformat.text(
original_str,
options=opts,
# ...
Mdformat allows configuration in a [TOML](https://toml.io) file named `.mdformat.toml`.
-The configuration file will be resolved starting from the location of the file being formatted,
-and searching up the file tree until a config file is (or isn't) found.
-When formatting standard input stream, resolution will be started from current working directory.
+The configuration file will be resolved from two sources:
-Command line interface arguments take precedence over the configuration file.
+1. The user's configuration directory.
+ <list OS-specific location directories, e.g. `$XDG_CONFIG_HOME/mdformat` or `~/.config/mdformat` on Linux>
+1. Starting from the location of the file being formatted,
+ and searching up the file tree until a config file is (or isn't) found.
+ When formatting standard input stream, resolution will be started from current working directory.
+
+The configuration file from the file tree take precedence over the user's configuration directory.
+Command line interface arguments take precedence over the configuration files.
## Example configuration
<!-- ... --> I'm not certain if my estimates of this being not terribly difficult are incorrect; and I don't want to be a burden on contributors by trying to merge bad code. |
Skimming over it, your analysis seems reasonable. @hukkin Would you consider a PR that implements the feature in the OP? |
Hey, thanks for the issue! I think I'll reject a global config for the same reasons that Prettier does so. Quoting Prettier docs
Also with how the new exclude configuration works, I don't think a global config can be intuitive. What would the project root for file exclusions be? Has to be current working directory I think for it to make any sense, but then behavior starts to get increasingly unexpected when running mdformat from child directories of the "user expected" project root. |
Prettier feels like the wrong baseline since so many python tools that do things like mdformat allow this. See this comment for more of my rationale on this.
I disagree. For example, |
Python is an implementation detail for mdformat. The CLI is not a Python tool, it's a Commonmark formatter. Commonmark has nothing to do with Python. The API admittedly is a "Python tool" but that does not interact with configuration files.
If you disagree, maybe you can explain how the project root for the purposes of If you really want, you can already create a |
I'm a fan of a global config option. If this were an option, I'd assume that the rules specified in |
Sorry for the slow reply :)
I hope I've understood the difficulty you're raising correctly. Looking at is_excluded() I see that if the root is unspecified, we fall back to the current working directory. As a user, that is what I would expect. |
Context
In #263 we added support for
.mdformat.toml
as an on-disk configuration file.I would like to revisit the possibility of also reading fall-back configuration from
$XDG_CONFIG_HOME
.platformdirs provides tools that can assist with this in a cross-platform way.
Proposal
The best approach I could think of is to start with the lowest priority configuration files first and work down, applying changes as we go. That way, defaults can be over-ridden by
$XDG_CONFIG_HOME/mdformat.toml
(or whatever we call it) but project-specific settings still take priority over any other on-disk configuration.Descending priority would look something like (subject to negotiation):
$HOME/.mdformat.toml
$HOME/.config/mdformat.toml
$XDG_CONFIG_HOME/mdformat.toml
/.mdformat.toml
/path/to/repo/.mdformat.toml
Tasks and updates
The text was updated successfully, but these errors were encountered: