123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310 |
- # index.py -- File parser/write for the git index file
- # Copyright (C) 2008-2009 Jelmer Vernooij <jelmer@samba.org>
-
- # This program is free software; you can redistribute it and/or
- # modify it under the terms of the GNU General Public License
- # as published by the Free Software Foundation; version 2
- # of the License or (at your opinion) any later version of the license.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program; if not, write to the Free Software
- # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
- # MA 02110-1301, USA.
- """Parser for the git index file format."""
- import os
- import stat
- import struct
- from dulwich.file import GitFile
- from dulwich.objects import (
- S_IFGITLINK,
- S_ISGITLINK,
- Tree,
- hex_to_sha,
- sha_to_hex,
- )
- from dulwich.pack import (
- SHA1Reader,
- SHA1Writer,
- )
- def read_cache_time(f):
- """Read a cache time.
-
- :param f: File-like object to read from
- :return: Tuple with seconds and nanoseconds
- """
- return struct.unpack(">LL", f.read(8))
- def write_cache_time(f, t):
- """Write a cache time.
-
- :param f: File-like object to write to
- :param t: Time to write (as int, float or tuple with secs and nsecs)
- """
- if isinstance(t, int):
- t = (t, 0)
- elif isinstance(t, float):
- (secs, nsecs) = divmod(t, 1.0)
- t = (int(secs), int(nsecs * 1000000000))
- elif not isinstance(t, tuple):
- raise TypeError(t)
- f.write(struct.pack(">LL", *t))
- def read_cache_entry(f):
- """Read an entry from a cache file.
- :param f: File-like object to read from
- :return: tuple with: device, inode, mode, uid, gid, size, sha, flags
- """
- beginoffset = f.tell()
- ctime = read_cache_time(f)
- mtime = read_cache_time(f)
- (dev, ino, mode, uid, gid, size, sha, flags, ) = \
- struct.unpack(">LLLLLL20sH", f.read(20 + 4 * 6 + 2))
- name = f.read((flags & 0x0fff))
- # Padding:
- real_size = ((f.tell() - beginoffset + 8) & ~7)
- data = f.read((beginoffset + real_size) - f.tell())
- return (name, ctime, mtime, dev, ino, mode, uid, gid, size,
- sha_to_hex(sha), flags & ~0x0fff)
- def write_cache_entry(f, entry):
- """Write an index entry to a file.
- :param f: File object
- :param entry: Entry to write, tuple with:
- (name, ctime, mtime, dev, ino, mode, uid, gid, size, sha, flags)
- """
- beginoffset = f.tell()
- (name, ctime, mtime, dev, ino, mode, uid, gid, size, sha, flags) = entry
- write_cache_time(f, ctime)
- write_cache_time(f, mtime)
- flags = len(name) | (flags &~ 0x0fff)
- f.write(struct.pack(">LLLLLL20sH", dev, ino, mode, uid, gid, size, hex_to_sha(sha), flags))
- f.write(name)
- real_size = ((f.tell() - beginoffset + 8) & ~7)
- f.write("\0" * ((beginoffset + real_size) - f.tell()))
- def read_index(f):
- """Read an index file, yielding the individual entries."""
- header = f.read(4)
- if header != "DIRC":
- raise AssertionError("Invalid index file header: %r" % header)
- (version, num_entries) = struct.unpack(">LL", f.read(4 * 2))
- assert version in (1, 2)
- for i in range(num_entries):
- yield read_cache_entry(f)
- def read_index_dict(f):
- """Read an index file and return it as a dictionary.
-
- :param f: File object to read from
- """
- ret = {}
- for x in read_index(f):
- ret[x[0]] = tuple(x[1:])
- return ret
- def write_index(f, entries):
- """Write an index file.
-
- :param f: File-like object to write to
- :param entries: Iterable over the entries to write
- """
- f.write("DIRC")
- f.write(struct.pack(">LL", 2, len(entries)))
- for x in entries:
- write_cache_entry(f, x)
- def write_index_dict(f, entries):
- """Write an index file based on the contents of a dictionary.
- """
- entries_list = []
- for name in sorted(entries):
- entries_list.append((name,) + tuple(entries[name]))
- write_index(f, entries_list)
- def cleanup_mode(mode):
- """Cleanup a mode value.
- This will return a mode that can be stored in a tree object.
-
- :param mode: Mode to clean up.
- """
- if stat.S_ISLNK(mode):
- return stat.S_IFLNK
- elif stat.S_ISDIR(mode):
- return stat.S_IFDIR
- elif S_ISGITLINK(mode):
- return S_IFGITLINK
- ret = stat.S_IFREG | 0644
- ret |= (mode & 0111)
- return ret
- class Index(object):
- """A Git Index file."""
- def __init__(self, filename):
- """Open an index file.
-
- :param filename: Path to the index file
- """
- self._filename = filename
- self.clear()
- self.read()
- def write(self):
- """Write current contents of index to disk."""
- f = GitFile(self._filename, 'wb')
- try:
- f = SHA1Writer(f)
- write_index_dict(f, self._byname)
- finally:
- f.close()
- def read(self):
- """Read current contents of index from disk."""
- f = GitFile(self._filename, 'rb')
- try:
- f = SHA1Reader(f)
- for x in read_index(f):
- self[x[0]] = tuple(x[1:])
- # FIXME: Additional data?
- f.read(os.path.getsize(self._filename)-f.tell()-20)
- f.check_sha()
- finally:
- f.close()
- def __len__(self):
- """Number of entries in this index file."""
- return len(self._byname)
- def __getitem__(self, name):
- """Retrieve entry by relative path.
-
- :return: tuple with (ctime, mtime, dev, ino, mode, uid, gid, size, sha, flags)
- """
- return self._byname[name]
- def __iter__(self):
- """Iterate over the paths in this index."""
- return iter(self._byname)
- def get_sha1(self, path):
- """Return the (git object) SHA1 for the object at a path."""
- return self[path][-2]
- def get_mode(self, path):
- """Return the POSIX file mode for the object at a path."""
- return self[path][-6]
- def iterblobs(self):
- """Iterate over path, sha, mode tuples for use with commit_tree."""
- for path in self:
- entry = self[path]
- yield path, entry[-2], cleanup_mode(entry[-6])
- def clear(self):
- """Remove all contents from this index."""
- self._byname = {}
- def __setitem__(self, name, x):
- assert isinstance(name, str)
- assert len(x) == 10
- # Remove the old entry if any
- self._byname[name] = x
- def iteritems(self):
- return self._byname.iteritems()
- def update(self, entries):
- for name, value in entries.iteritems():
- self[name] = value
- def changes_from_tree(self, object_store, tree, want_unchanged=False):
- """Find the differences between the contents of this index and a tree.
- :param object_store: Object store to use for retrieving tree contents
- :param tree: SHA1 of the root tree
- :param want_unchanged: Whether unchanged files should be reported
- :return: Iterator over tuples with (oldpath, newpath), (oldmode, newmode), (oldsha, newsha)
- """
- mine = set(self._byname.keys())
- for (name, mode, sha) in object_store.iter_tree_contents(tree):
- if name in mine:
- if (want_unchanged or self.get_sha1(name) != sha or
- self.get_mode(name) != mode):
- yield ((name, name), (mode, self.get_mode(name)), (sha, self.get_sha1(name)))
- mine.remove(name)
- else:
- # Was removed
- yield ((name, None), (mode, None), (sha, None))
- # Mention added files
- for name in mine:
- yield ((None, name), (None, self.get_mode(name)), (None, self.get_sha1(name)))
- def commit_tree(object_store, blobs):
- """Commit a new tree.
- :param object_store: Object store to add trees to
- :param blobs: Iterable over blob path, sha, mode entries
- :return: SHA1 of the created tree.
- """
- trees = {"": {}}
- def add_tree(path):
- if path in trees:
- return trees[path]
- dirname, basename = os.path.split(path)
- t = add_tree(dirname)
- assert isinstance(basename, str)
- newtree = {}
- t[basename] = newtree
- trees[path] = newtree
- return newtree
- for path, sha, mode in blobs:
- tree_path, basename = os.path.split(path)
- tree = add_tree(tree_path)
- tree[basename] = (mode, sha)
- def build_tree(path):
- tree = Tree()
- for basename, entry in trees[path].iteritems():
- if type(entry) == dict:
- mode = stat.S_IFDIR
- sha = build_tree(os.path.join(path, basename))
- else:
- (mode, sha) = entry
- tree.add(mode, basename, sha)
- object_store.add_object(tree)
- return tree.id
- return build_tree("")
- def commit_index(object_store, index):
- """Create a new tree from an index.
- :param object_store: Object store to save the tree in
- :param index: Index file
- """
- return commit_tree(object_store, index.iterblobs())
|