-
Notifications
You must be signed in to change notification settings - Fork 365
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
Add config for ignoring executable bit changes #4478
base: main
Are you sure you want to change the base?
Conversation
ce8f245
to
76524dd
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not a full review, just a few random comments
file_type: FileType::Normal { | ||
executable: FileExecutableFlag::default(), | ||
}, | ||
file_type: FileType::default(), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Now that the type doesn't vary at compile time between platform, we can use the same value on all platforms here and probably not need to implement FileType::default()
. We should even be able to run this test case on Windows with a value for the the executable bit. Not that I think we need to do that; I'm just saying that there's no longer a need to avoid certain tests on Windows.
lib/src/settings.rs
Outdated
/// This is not a method on UserSettings since it needs to be callable even when | ||
/// only having the config itself (currently only used in `Ui::with_config`). | ||
pub fn report_executable_bit(config: &config::Config) -> Option<bool> { | ||
config.get_bool("core.report-executable-bit").ok() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I feel like core
would be better replaced by working-copy
or local-working-copy
, but I know we already use core
so this is consistent for now. We can decide about renaming after this PR.
lib/src/settings.rs
Outdated
/// | ||
/// This is not a method on UserSettings since it needs to be callable even when | ||
/// only having the config itself (currently only used in `Ui::with_config`). | ||
pub fn report_executable_bit(config: &config::Config) -> Option<bool> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: Since this method is only used in local_working_copy.rs
(right?), you can just define it there.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nope 🙃 It's also used in cli/src/ui.rs
. See the added exec_config
arguments in the merge_tools/{mod,diff_working_copies}.rs
files that originate from the ui.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Though I'm still not sure where to put this. If we choose local_working_copy.rs
does that have any effect on other users of the library? I'm fine moving it there if you recommend
Thanks, btw! It looks good overall |
5ed90ba
to
83a0b0c
Compare
83a0b0c
to
685368d
Compare
docs/config.md
Outdated
executable bit and update the repository. Otherwise we will only update files if | ||
they are manually changed via `jj file chmod`. | ||
executable bits are supported. If they are, we will not ignore them and changing | ||
a files' executable bit will update the repository. If they are not supported we |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nits:
files'
->file's
not ignore
->respect
On unix we
->On Unix, we
On unix setting
->On Unix, setting
Maybe it will be clearer add an introductory paragraph, making the whole thing something like this:
If you update the working copy to a commit, then the executable bit for the files in the file system will normally be updated to match the commit. Similarly, if you update a file's executable bit in the file system, then the subsequent snapshotting of the working copy will result in the executable bit getting updated in the working-copy commit.
This configuration option makes Jujutsu ignore the executable bit in the file system and instead preserve the executable bit in the commit. New files in the file system will be recorded as not executable (you can use
jj file chmod
to set it afterwards).Since there is no executable bit on Windows, this option is effectively always on there. On Unix, Jujutsu will try to detect whether the file system supports executable bits. The configured value will override the detected value.
Just a thought. Take what you like from that text, or ignore it completely.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thoughts on this?
Ignore the executable bit
Whether to ignore changes to the executable bit for files on Unix. This option
is unused on Windows.
core.ignore-executable-bit = true | false
On Unix systems, files have permission bits for whether they are executable as
scripts or binary code. Jujutsu stores this executable bit within each commit
and will update it for files as you operate on a repository. If you update your
working copy to a commit where a file is recorded as executable or not, jj
will adjust the permission of that file. If you change a file's executable bit
through the filesystem, jj
will record that change in the working-copy commit
on the next snapshot.
Setting this to true
will make Jujutsu ignore the executable bit on the
filesystem and only store the executable bit in the commit itself. jj
will
also not attempt to modify file's executable bits and will add new files as "not
executable." This is the default behavior on Windows and can be useful for users
dual-booting Windows from Unix or experimenting with nonstandard filesystems.
On Unix, Jujutsu will try to detect whether the filesystem supports executable
bits to choose a default value, but a configured value will always override the
default. On Windows systems there is no executable bit, and this option is not
used.
You can use always use jj file chmod
to update the recorded executable bit for
a file in the commit manually. If this option is false
, jj
will propagate that
change to the filesystem.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the short description at the top and the explicit line with the available options is really helpful to let a user who is just skimming the docs get what they need and be able to choose whether to read the rest or not.
I made your intro paragraph a bit more explicit about how the working-copy commit reads from the filesystem.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I prefer to use the word "ignore" less, since it has meaning already in a slightly different way.
I also think that an option saying "ignore=true" seems backward. It sounds much better to have "respect=false" since the positive action corresponds to the variable, so when you read the logic (or explain it), you don't have the negative verb and a false value confusing things. An example would be "show=false" versus "hide=true" where the logic is like if show then print
versus if not hide then print
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I feel like ignore works fine, it's far enough from any gitignore files that I don't really care. However I'm fine with either "ignore" or "respect" as the main words.
Should I change it again or not?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just some nits to conform to our code review style here: https://github.com/martinvonz/jj/blob/main/docs/contributing.md#code-reviews
store: Arc<Store>, | ||
working_copy_path: PathBuf, | ||
state_path: PathBuf, | ||
exec_config: Option<bool>, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps, exec_config
can be carried by SnapshotOptions
? When materializing working-copy files, it's probably okay to try chmod()
even if the filesystem doesn't support exec bit.
It's a bit odd that TreeState::empty()
constructor needs it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can see how SnapshotOptions
might work, but I'm not sure that it would preserve the correct behavior when doing a checkout/reset/recover and then calling LockedWorkingCopy::finish()
, since those save to the repo. If we're ok with sending the permission changes regardless of if they might work then I could see it working, but it feels somewhat 'less correct'? Not sure.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(assuming fchmod()
doesn't fail,) checkout is repo->working-copy operation, so it won't have to read exec bit from disk. reset()
and recover()
are similar?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, but then its the call to finish()
afterwards that tries to write to the repo. And finish
doesn't currently take any other configuration. I think it makes sense to keep the config with the tree state.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just to be celar, LockedLocalWorkingCopy::finish()
doesn't write commits to repo, but writes working-copy/tree states to disk. It's more or less a serialization of in-memory data.
That said, it might be okay to keep a copy of filesystem capability options in TreeState
. We already have symlink_support: bool
(and store: Arc<Store>
which isn't state data but a shared resource.) If we make symlink_support
configurable, we'll have to resolve it somewhere in LocalWorkingCopy
, and pass in to TreeState
constructor (or methods.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, that makes more sense forfinish
.
How would you feel about some sort of FileCapability
or Platform
trait/module? It seems like it effectively already exists for symlink and executable bits in file_util.rs
, and it would be nice to add helpers like a from_config()
function.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How would you feel about some sort of
FileCapability
orPlatform
trait/module? It seems like it effectively already exists for symlink and executable bits infile_util.rs
, and it would be nice to add helpers like afrom_config()
function.
I don't think extensibility provided by trait
would be needed. These parameters can be packed into struct FileCapability { some bools }
, but I don't know if that makes sense. Config parameters would have to be Option<bool>
s, and we'll need to explicitly control where/when the actual fs capability is tested. iirc, the error of check_symlink_support()
isn't propagated right now, but it should be.
lib/src/local_working_copy.rs
Outdated
@@ -1644,6 +1777,7 @@ impl LocalWorkingCopy { | |||
state_path: PathBuf, | |||
operation_id: OperationId, | |||
workspace_id: WorkspaceId, | |||
config: &config::Config, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: we typically use &UserSeettings
object.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm only using config because the Ui
object only gets initialized with a config and it felt odd to pass user_settings into everything and then only use it by calling: ignore_exec_bit(user_settings.config())
. I'm ok with changing it but I don't really see a reason to pass the full user settings?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't really see a reason to pass the full user settings?
I don't have a concrete plan, but we'll probably replace config::Config
with our config module that supports layering better. Then, config::Config
APIs might be rehosted on UserSettings
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, then I will move to passing UserSettings
eb5d4cb
to
e7df875
Compare
Pushed some small fixes in the local_working_copy changes, also added a new commit that refactors the check_out functions in I've also started using the update on my machine, and it seems to work correctly on my dual-boot repo with the config, but I haven't really tested all of the behavior. One other note is that I don't think my current changes work correctly if you update the config value. That is, if we alter the config value to stop respecting executable bits, the tree should refresh itself, but I think it still keeps the stored changes instead of refreshing even if I do |
9f2112b
to
ca6e105
Compare
docs/config.md
Outdated
|
||
Setting this to `true` will make Jujutsu ignore the executable bit on the | ||
filesystem and only store the executable bit in the commit itself. `jj` will | ||
also not attempt to modify file's executable bits and will add new files as "not |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: file's -> files' (or make it "a file's executable bit")
pub enum ExecFlag { | ||
Exec(bool), | ||
Ignore, | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is it possible to always use a bool instead of this type and only change how we used that bool based on the IgnoreExec
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could, the main difference is just whether the tree state stores that a file is not executable at a given moment. I assume this would affect users who modify the configuration between snapshots, but I don't really know what that would imply.
I can convert it to just a bool if you'd like.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Alternatively, it might be better to use a bool-equivalent enum, e.g. enum { Exec, NotExec }
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could, the main difference is just whether the tree state stores that a file is not executable at a given moment. I assume this would affect users who modify the configuration between snapshots, but I don't really know what that would imply.
Good point. I think that means it's better to do our best to keep the stored flag up to date even the configuration currently says to ignore the executable bit in the file system. Otherwise we might clear the flag in the file and not notice if the user then sets the config to respect the config and also has cleared the executable bit on some file in the file system.
Alternatively, it might be better to use a bool-equivalent enum, e.g.
enum { Exec, NotExec }
.
I would go with just a bool
until we find that it's hard to read or likely to cause bugs for some reason (e.g. because we confuse that boolean for some other boolean value).
98fe553
to
8c192a9
Compare
…Unix This is just the initial support for the feature. Threading the value into the working copy via user configuration is in the next commit.
…Side enum. I got annoyed by the checkout function above in my previous commit and tried to compress the code a bit, and I think the result is much nicer. I'm not certain that a closure is the right solution here, but I think it becomes easier to reason about the two temp filenames that get generated. I initially changed Option<DiffSide> to a bool, but found an enum more meaningful and more likely to be flexible in the future. I also moved the instruction writing to its own function. This makes the main `DiffEditWorkingCopies::check_out()` function is much nicer. J: This commit contains the following changes:
8c192a9
to
3982b04
Compare
Starting a PR to close #3949 . This is my first time contributing, so I would appreciate feedback on the code. I'm certain that there are some places that can be integrated better. I've left a few TODO markers scattered around that I would appreciate help with.
Also: I have not run these changes locally yet. They build and pass
cargo test
, but I'm still not certain how to run these myself.I also haven't added any tests yet. I'd appreciate feedback on a direction to go for those and where it would be best to test these changes.
Checklist
If applicable:
CHANGELOG.md