Can Mercurial be made to preserve file permissions?

14,841

Solution 1

It looks like it can be done using hooks and an auxiliary tool (and a little chewing gum and baling wire):

  1. Get David Hardeman's Metastore, which saves and restores file metadata.

  2. Alter the sources so it will ignore directory .hg as well as .git.

  3. Use the following Mercurial hooks:

     precommit.meta = metastore -s
    
     changegroup.update = hg update
     update.meta   = /usr/unsup/nr/bin/metastore -a
    

You have to add the .metadata file to the repo.

This lashup will work most of the time, but if you change only permissions and want to propagate it, you'll have to run metastore -s in order to push those changes into the .metadata file where hg will see the change; otherwise the commit thinks nothing is new.

Solution 2

What about using this solution from the Mercurial FAQ:

If you're using Mercurial for config file management, you might want to track file properties (ownership and permissions) too. Mercurial only tracks the executable bit of each file.

Here is an example of how to save the properties along with the files (works on Linux if you've the acl package installed):

# cd /etc && getfacl -R . >/tmp/acl.$$ && mv /tmp/acl.$$ .acl
# hg commit

This is far from perfect, but you get the idea. For a more sophisticated solution, check out etckeeper.

Solution 3

For the specific case of the /etc directory, etckeeper looks interesting.

Share:
14,841

Related videos on Youtube

Norman Ramsey
Author by

Norman Ramsey

Every time I see a question about "strong" or "weak" typing, I kill a kitten.

Updated on October 22, 2020

Comments

  • Norman Ramsey
    Norman Ramsey over 3 years

    I've seen a number of blog posts, and have experienced for myself, that Mercurial does not preserve the permissions on files pushed from one repo to another. Does anyone know of a Mercurial extension that would preserve the permissions? I'm assuming it can't be done with a hook, because what does a hook know about permissions at the originating repo?

    Requested elaboration:

    • If the only change to a file is a change in permissions (e.g., chmod o+r filename), attempts to commit the file fail with a message saying that the file has not changed.

    • If I commit a file with permissions 600 (rw-------), then clone the repo, the same file in the clone has permissions 664 (rw-rw-r--):

      : nr@yorkie 6522 ; hg clone one two
      updating working directory
      1 files updated, 0 files merged, 0 files removed, 0 files unresolved
      : nr@yorkie 6523 ; ls -l one two
      one:
      total 4
      -rw------- 1 nr nr 8 Aug 18 21:50 foo
      
      two:
      total 4
      -rw-rw-r-- 1 nr nr 8 Aug 18 21:51 foo
      

    This examples shows that hg clone does not preserve permissions, but hg push does not preserve them either.

    In my application, one repo is on a publically accessible path, and it's of major importance that

    • Multiple users have the right to change the repo

    • Files in the public repo become readable only when explicitly made readable.

    • quark
      quark almost 15 years
      Can you elaborate a little on this? I can change permissions on my local files, commit the change, push them to a clone, and the clone will, when I update it, change the permissions. The key there is that I have to both commit and update. Do you want the pushed to clone to change permissions just from the push? To notice local changes without the commit? To infer all existing permissions? What's the exact issue / what am I missing?
    • Norman Ramsey
      Norman Ramsey almost 15 years
      @quark: How is it done? I'm using hg 1.2.1 on Debian Linux and I can't even get it to acknowledge that a change in permissinos is a change worth committing. (N.B. I have elaborated as per your request.)
    • quark
      quark almost 15 years
      Norman: turns out what I'm missing is the fact that you want to hold on to change to "r" and "w", and what I'm talking about is changes to "x" (which Mercurial has tracked since 0.6). I should've realized that you meant read/write not executable.
  • Norman Ramsey
    Norman Ramsey almost 15 years
    It may not be a good idea in general, but in my application, permissions are critical metadata, and they have to be right. DVCS is not the ideal tool but seems the best available.
  • Maxim Kholyavkin
    Maxim Kholyavkin almost 10 years
    link to Devid Hardeman's is obsolette. New link: github.com/przemoc/metastore.git
  • Maxim Kholyavkin
    Maxim Kholyavkin almost 10 years
    BTW. Here: stackoverflow.com/a/17583212/751932 was mentioned one more tool git-cache-meta.sh gist.github.com/Cojad/9205547