~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/changeset.py

  • Committer: Martin Pool
  • Date: 2005-08-29 10:57:01 UTC
  • mfrom: (1092.1.41)
  • Revision ID: mbp@sourcefrog.net-20050829105701-7aaa81ecf1bfee05
- merge in merge improvements and additional tests 
  from aaron and lifeless

robertc@robertcollins.net-20050825131100-85772edabc817481

Show diffs side-by-side

added added

removed removed

Lines of Context:
13
13
#    You should have received a copy of the GNU General Public License
14
14
#    along with this program; if not, write to the Free Software
15
15
#    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
 
 
17
 
"""Represent and apply a changeset.
18
 
 
19
 
Conflicts in applying a changeset are represented as exceptions.
20
 
 
21
 
This only handles the in-memory objects representing changesets, which are
22
 
primarily used by the merge code. 
23
 
"""
24
 
 
25
16
import os.path
26
17
import errno
 
18
import patch
27
19
import stat
28
 
from shutil import rmtree
29
 
from itertools import izip
30
 
 
31
 
from bzrlib.trace import mutter, warning
32
 
from bzrlib.osutils import rename, sha_file, pathjoin, mkdtemp
33
 
import bzrlib
34
 
from bzrlib.errors import BzrCheckError
 
20
from bzrlib.trace import mutter
 
21
 
 
22
# XXX: mbp: I'm not totally convinced that we should handle conflicts
 
23
# as part of changeset application, rather than only in the merge
 
24
# operation.
 
25
 
 
26
"""Represent and apply a changeset
 
27
 
 
28
Conflicts in applying a changeset are represented as exceptions.
 
29
"""
35
30
 
36
31
__docformat__ = "restructuredtext"
37
32
 
38
 
 
39
33
NULL_ID = "!NULL"
40
34
 
41
 
 
42
35
class OldFailedTreeOp(Exception):
43
36
    def __init__(self):
44
37
        Exception.__init__(self, "bzr-tree-change contains files from a"
45
38
                           " previous failed merge operation.")
46
 
 
47
 
 
48
39
def invert_dict(dict):
49
40
    newdict = {}
50
41
    for (key,value) in dict.iteritems():
51
42
        newdict[value] = key
52
43
    return newdict
53
44
 
 
45
 
54
46
       
55
 
class ChangeExecFlag(object):
 
47
class ChangeUnixPermissions(object):
56
48
    """This is two-way change, suitable for file modification, creation,
57
49
    deletion"""
58
 
    def __init__(self, old_exec_flag, new_exec_flag):
59
 
        self.old_exec_flag = old_exec_flag
60
 
        self.new_exec_flag = new_exec_flag
 
50
    def __init__(self, old_mode, new_mode):
 
51
        self.old_mode = old_mode
 
52
        self.new_mode = new_mode
61
53
 
62
 
    def apply(self, filename, conflict_handler):
63
 
        from_exec_flag = self.old_exec_flag
64
 
        to_exec_flag = self.new_exec_flag
 
54
    def apply(self, filename, conflict_handler, reverse=False):
 
55
        if not reverse:
 
56
            from_mode = self.old_mode
 
57
            to_mode = self.new_mode
 
58
        else:
 
59
            from_mode = self.new_mode
 
60
            to_mode = self.old_mode
65
61
        try:
66
 
            current_exec_flag = bool(os.stat(filename).st_mode & 0111)
 
62
            current_mode = os.stat(filename).st_mode &0777
67
63
        except OSError, e:
68
64
            if e.errno == errno.ENOENT:
69
 
                if conflict_handler.missing_for_exec_flag(filename) == "skip":
 
65
                if conflict_handler.missing_for_chmod(filename) == "skip":
70
66
                    return
71
67
                else:
72
 
                    current_exec_flag = from_exec_flag
 
68
                    current_mode = from_mode
73
69
 
74
 
        if from_exec_flag is not None and current_exec_flag != from_exec_flag:
75
 
            if conflict_handler.wrong_old_exec_flag(filename,
76
 
                        from_exec_flag, current_exec_flag) != "continue":
 
70
        if from_mode is not None and current_mode != from_mode:
 
71
            if conflict_handler.wrong_old_perms(filename, from_mode, 
 
72
                                                current_mode) != "continue":
77
73
                return
78
74
 
79
 
        if to_exec_flag is not None:
80
 
            current_mode = os.stat(filename).st_mode
81
 
            if to_exec_flag:
82
 
                umask = os.umask(0)
83
 
                os.umask(umask)
84
 
                to_mode = current_mode | (0100 & ~umask)
85
 
                # Enable x-bit for others only if they can read it.
86
 
                if current_mode & 0004:
87
 
                    to_mode |= 0001 & ~umask
88
 
                if current_mode & 0040:
89
 
                    to_mode |= 0010 & ~umask
90
 
            else:
91
 
                to_mode = current_mode & ~0111
 
75
        if to_mode is not None:
92
76
            try:
93
77
                os.chmod(filename, to_mode)
94
78
            except IOError, e:
95
79
                if e.errno == errno.ENOENT:
96
 
                    conflict_handler.missing_for_exec_flag(filename)
 
80
                    conflict_handler.missing_for_chmod(filename)
97
81
 
98
82
    def __eq__(self, other):
99
 
        return (isinstance(other, ChangeExecFlag) and
100
 
                self.old_exec_flag == other.old_exec_flag and
101
 
                self.new_exec_flag == other.new_exec_flag)
 
83
        if not isinstance(other, ChangeUnixPermissions):
 
84
            return False
 
85
        elif self.old_mode != other.old_mode:
 
86
            return False
 
87
        elif self.new_mode != other.new_mode:
 
88
            return False
 
89
        else:
 
90
            return True
102
91
 
103
92
    def __ne__(self, other):
104
93
        return not (self == other)
105
94
 
106
 
 
107
 
def dir_create(filename, conflict_handler, reverse=False):
 
95
def dir_create(filename, conflict_handler, reverse):
108
96
    """Creates the directory, or deletes it if reverse is true.  Intended to be
109
97
    used with ReplaceContents.
110
98
 
129
117
        try:
130
118
            os.rmdir(filename)
131
119
        except OSError, e:
132
 
            if e.errno != errno.ENOTEMPTY:
 
120
            if e.errno != 39:
133
121
                raise
134
122
            if conflict_handler.rmdir_non_empty(filename) == "skip":
135
123
                return
136
124
            os.rmdir(filename)
137
125
 
 
126
                
 
127
            
138
128
 
139
129
class SymlinkCreate(object):
140
130
    """Creates or deletes a symlink (for use with ReplaceContents)"""
146
136
        """
147
137
        self.target = contents
148
138
 
149
 
    def __repr__(self):
150
 
        return "SymlinkCreate(%s)" % self.target
151
 
 
152
 
    def __call__(self, filename, conflict_handler, reverse=False):
 
139
    def __call__(self, filename, conflict_handler, reverse):
153
140
        """Creates or destroys the symlink.
154
141
 
155
142
        :param filename: The name of the symlink to create
178
165
    def __ne__(self, other):
179
166
        return not (self == other)
180
167
 
181
 
 
182
168
class FileCreate(object):
183
169
    """Create or delete a file (for use with ReplaceContents)"""
184
170
    def __init__(self, contents):
203
189
    def __ne__(self, other):
204
190
        return not (self == other)
205
191
 
206
 
    def __call__(self, filename, conflict_handler, reverse=False):
 
192
    def __call__(self, filename, conflict_handler, reverse):
207
193
        """Create or delete a file
208
194
 
209
195
        :param filename: The name of the file to create
235
221
                if conflict_handler.missing_for_rm(filename, undo) == "skip":
236
222
                    return
237
223
 
238
 
 
239
 
class TreeFileCreate(object):
240
 
    """Create or delete a file (for use with ReplaceContents)"""
241
 
    def __init__(self, tree, file_id):
242
 
        """Constructor
243
 
 
244
 
        :param contents: The contents of the file to write
245
 
        :type contents: str
246
 
        """
247
 
        self.tree = tree
248
 
        self.file_id = file_id
249
 
 
250
 
    def __repr__(self):
251
 
        return "TreeFileCreate(%s)" % self.file_id
252
 
 
253
 
    def __eq__(self, other):
254
 
        if not isinstance(other, TreeFileCreate):
255
 
            return False
256
 
        return self.tree.get_file_sha1(self.file_id) == \
257
 
            other.tree.get_file_sha1(other.file_id)
258
 
 
259
 
    def __ne__(self, other):
260
 
        return not (self == other)
261
 
 
262
 
    def write_file(self, filename):
263
 
        outfile = file(filename, "wb")
264
 
        for line in self.tree.get_file(self.file_id):
265
 
            outfile.write(line)
266
 
 
267
 
    def same_text(self, filename):
268
 
        in_file = file(filename, "rb")
269
 
        return sha_file(in_file) == self.tree.get_file_sha1(self.file_id)
270
 
 
271
 
    def __call__(self, filename, conflict_handler, reverse=False):
272
 
        """Create or delete a file
273
 
 
274
 
        :param filename: The name of the file to create
275
 
        :type filename: str
276
 
        :param reverse: Delete the file instead of creating it
277
 
        :type reverse: bool
278
 
        """
279
 
        if not reverse:
280
 
            try:
281
 
                self.write_file(filename)
282
 
            except IOError, e:
283
 
                if e.errno == errno.ENOENT:
284
 
                    if conflict_handler.missing_parent(filename)=="continue":
285
 
                        self.write_file(filename)
286
 
                else:
287
 
                    raise
288
 
 
289
 
        else:
290
 
            try:
291
 
                if not self.same_text(filename):
292
 
                    direction = conflict_handler.wrong_old_contents(filename,
293
 
                        self.tree.get_file(self.file_id).read())
294
 
                    if  direction != "continue":
295
 
                        return
296
 
                os.unlink(filename)
297
 
            except IOError, e:
298
 
                if e.errno != errno.ENOENT:
299
 
                    raise
300
 
                if conflict_handler.missing_for_rm(filename, undo) == "skip":
301
 
                    return
302
 
 
 
224
                    
 
225
 
 
226
def reversed(sequence):
 
227
    max = len(sequence) - 1
 
228
    for i in range(len(sequence)):
 
229
        yield sequence[max - i]
303
230
 
304
231
class ReplaceContents(object):
305
232
    """A contents-replacement framework.  It allows a file/directory/symlink to
337
264
    def __ne__(self, other):
338
265
        return not (self == other)
339
266
 
340
 
    def apply(self, filename, conflict_handler):
 
267
    def apply(self, filename, conflict_handler, reverse=False):
341
268
        """Applies the FileReplacement to the specified filename
342
269
 
343
270
        :param filename: The name of the file to apply changes to
344
271
        :type filename: str
 
272
        :param reverse: If true, apply the change in reverse
 
273
        :type reverse: bool
345
274
        """
346
 
        undo = self.old_contents
347
 
        perform = self.new_contents
 
275
        if not reverse:
 
276
            undo = self.old_contents
 
277
            perform = self.new_contents
 
278
        else:
 
279
            undo = self.new_contents
 
280
            perform = self.old_contents
348
281
        mode = None
349
282
        if undo is not None:
350
283
            try:
358
291
                    return
359
292
            undo(filename, conflict_handler, reverse=True)
360
293
        if perform is not None:
361
 
            perform(filename, conflict_handler)
 
294
            perform(filename, conflict_handler, reverse=False)
362
295
            if mode is not None:
363
296
                os.chmod(filename, mode)
364
297
 
365
 
    def is_creation(self):
366
 
        return self.new_contents is not None and self.old_contents is None
367
 
 
368
 
    def is_deletion(self):
369
 
        return self.old_contents is not None and self.new_contents is None
 
298
class ApplySequence(object):
 
299
    def __init__(self, changes=None):
 
300
        self.changes = []
 
301
        if changes is not None:
 
302
            self.changes.extend(changes)
 
303
 
 
304
    def __eq__(self, other):
 
305
        if not isinstance(other, ApplySequence):
 
306
            return False
 
307
        elif len(other.changes) != len(self.changes):
 
308
            return False
 
309
        else:
 
310
            for i in range(len(self.changes)):
 
311
                if self.changes[i] != other.changes[i]:
 
312
                    return False
 
313
            return True
 
314
 
 
315
    def __ne__(self, other):
 
316
        return not (self == other)
 
317
 
 
318
    
 
319
    def apply(self, filename, conflict_handler, reverse=False):
 
320
        if not reverse:
 
321
            iter = self.changes
 
322
        else:
 
323
            iter = reversed(self.changes)
 
324
        for change in iter:
 
325
            change.apply(filename, conflict_handler, reverse)
370
326
 
371
327
 
372
328
class Diff3Merge(object):
373
 
    history_based = False
374
329
    def __init__(self, file_id, base, other):
375
330
        self.file_id = file_id
376
331
        self.base = base
377
332
        self.other = other
378
333
 
379
 
    def is_creation(self):
380
 
        return False
381
 
 
382
 
    def is_deletion(self):
383
 
        return False
384
 
 
385
334
    def __eq__(self, other):
386
335
        if not isinstance(other, Diff3Merge):
387
336
            return False
391
340
    def __ne__(self, other):
392
341
        return not (self == other)
393
342
 
394
 
    def dump_file(self, temp_dir, name, tree):
395
 
        out_path = pathjoin(temp_dir, name)
396
 
        out_file = file(out_path, "wb")
397
 
        in_file = tree.get_file(self.file_id)
398
 
        for line in in_file:
399
 
            out_file.write(line)
400
 
        return out_path
401
 
 
402
 
    def apply(self, filename, conflict_handler):
403
 
        import bzrlib.patch
404
 
        temp_dir = mkdtemp(prefix="bzr-")
405
 
        try:
406
 
            new_file = filename+".new"
407
 
            base_file = self.dump_file(temp_dir, "base", self.base)
408
 
            other_file = self.dump_file(temp_dir, "other", self.other)
 
343
    def apply(self, filename, conflict_handler, reverse=False):
 
344
        new_file = filename+".new"
 
345
        base_file = self.base.readonly_path(self.file_id)
 
346
        other_file = self.other.readonly_path(self.file_id)
 
347
        if not reverse:
409
348
            base = base_file
410
349
            other = other_file
411
 
            status = bzrlib.patch.diff3(new_file, filename, base, other)
412
 
            if status == 0:
413
 
                os.chmod(new_file, os.stat(filename).st_mode)
414
 
                rename(new_file, filename)
415
 
                return
416
 
            else:
417
 
                assert(status == 1)
418
 
                def get_lines(filename):
419
 
                    my_file = file(filename, "rb")
420
 
                    lines = my_file.readlines()
421
 
                    my_file.close()
422
 
                    return lines
423
 
                base_lines = get_lines(base)
424
 
                other_lines = get_lines(other)
425
 
                conflict_handler.merge_conflict(new_file, filename, base_lines, 
426
 
                                                other_lines)
427
 
        finally:
428
 
            rmtree(temp_dir)
 
350
        else:
 
351
            base = other_file
 
352
            other = base_file
 
353
        status = patch.diff3(new_file, filename, base, other)
 
354
        if status == 0:
 
355
            os.chmod(new_file, os.stat(filename).st_mode)
 
356
            os.rename(new_file, filename)
 
357
            return
 
358
        else:
 
359
            assert(status == 1)
 
360
            def get_lines(filename):
 
361
                my_file = file(base, "rb")
 
362
                lines = my_file.readlines()
 
363
                my_file.close()
 
364
            base_lines = get_lines(base)
 
365
            other_lines = get_lines(other)
 
366
            conflict_handler.merge_conflict(new_file, filename, base_lines, 
 
367
                                            other_lines)
429
368
 
430
369
 
431
370
def CreateDir():
436
375
    """
437
376
    return ReplaceContents(None, dir_create)
438
377
 
439
 
 
440
378
def DeleteDir():
441
379
    """Convenience function to delete a directory.
442
380
 
445
383
    """
446
384
    return ReplaceContents(dir_create, None)
447
385
 
448
 
 
449
386
def CreateFile(contents):
450
387
    """Convenience fucntion to create a file.
451
388
    
456
393
    """
457
394
    return ReplaceContents(None, FileCreate(contents))
458
395
 
459
 
 
460
396
def DeleteFile(contents):
461
397
    """Convenience fucntion to delete a file.
462
398
    
467
403
    """
468
404
    return ReplaceContents(FileCreate(contents), None)
469
405
 
470
 
 
471
 
def ReplaceFileContents(old_tree, new_tree, file_id):
 
406
def ReplaceFileContents(old_contents, new_contents):
472
407
    """Convenience fucntion to replace the contents of a file.
473
408
    
474
409
    :param old_contents: The contents of the file to replace 
478
413
    :return: A ReplaceContents that will replace the contents of a file a file 
479
414
    :rtype: `ReplaceContents`
480
415
    """
481
 
    return ReplaceContents(TreeFileCreate(old_tree, file_id), 
482
 
                           TreeFileCreate(new_tree, file_id))
483
 
 
 
416
    return ReplaceContents(FileCreate(old_contents), FileCreate(new_contents))
484
417
 
485
418
def CreateSymlink(target):
486
419
    """Convenience fucntion to create a symlink.
492
425
    """
493
426
    return ReplaceContents(None, SymlinkCreate(target))
494
427
 
495
 
 
496
428
def DeleteSymlink(target):
497
429
    """Convenience fucntion to delete a symlink.
498
430
    
503
435
    """
504
436
    return ReplaceContents(SymlinkCreate(target), None)
505
437
 
506
 
 
507
438
def ChangeTarget(old_target, new_target):
508
439
    """Convenience fucntion to change the target of a symlink.
509
440
    
547
478
        msg = 'Child of !NULL is named "%s", not "./.".' % name
548
479
        InvalidEntry.__init__(self, entry, msg)
549
480
 
550
 
 
551
481
class NullIDAssigned(InvalidEntry):
552
482
    """The id !NULL was assigned to a real entry"""
553
483
    def __init__(self, entry):
559
489
        msg = '"!NULL" id assigned to a file "%s".' % entry.path
560
490
        InvalidEntry.__init__(self, entry, msg)
561
491
 
562
 
 
563
492
class ParentIDIsSelf(InvalidEntry):
564
493
    """An entry is marked as its own parent"""
565
494
    def __init__(self, entry):
572
501
            (entry.path, entry.id)
573
502
        InvalidEntry.__init__(self, entry, msg)
574
503
 
575
 
 
576
504
class ChangesetEntry(object):
577
505
    """An entry the changeset"""
578
506
    def __init__(self, id, parent, path):
597
525
        if self.id  == self.parent:
598
526
            raise ParentIDIsSelf(self)
599
527
 
600
 
    def __repr__(self):
 
528
    def __str__(self):
601
529
        return "ChangesetEntry(%s)" % self.id
602
530
 
603
 
    __str__ = __repr__
604
 
 
605
531
    def __get_dir(self):
606
532
        if self.path is None:
607
533
            return None
608
534
        return os.path.dirname(self.path)
609
535
 
610
536
    def __set_dir(self, dir):
611
 
        self.path = pathjoin(dir, os.path.basename(self.path))
 
537
        self.path = os.path.join(dir, os.path.basename(self.path))
612
538
 
613
539
    dir = property(__get_dir, __set_dir)
614
540
    
618
544
        return os.path.basename(self.path)
619
545
 
620
546
    def __set_name(self, name):
621
 
        self.path = pathjoin(os.path.dirname(self.path), name)
 
547
        self.path = os.path.join(os.path.dirname(self.path), name)
622
548
 
623
549
    name = property(__get_name, __set_name)
624
550
 
628
554
        return os.path.dirname(self.new_path)
629
555
 
630
556
    def __set_new_dir(self, dir):
631
 
        self.new_path = pathjoin(dir, os.path.basename(self.new_path))
 
557
        self.new_path = os.path.join(dir, os.path.basename(self.new_path))
632
558
 
633
559
    new_dir = property(__get_new_dir, __set_new_dir)
634
560
 
638
564
        return os.path.basename(self.new_path)
639
565
 
640
566
    def __set_new_name(self, name):
641
 
        self.new_path = pathjoin(os.path.dirname(self.new_path), name)
 
567
        self.new_path = os.path.join(os.path.dirname(self.new_path), name)
642
568
 
643
569
    new_name = property(__get_new_name, __set_new_name)
644
570
 
650
576
 
651
577
        return (self.parent != self.new_parent or self.name != self.new_name)
652
578
 
653
 
    def is_deletion(self, reverse=False):
 
579
    def is_deletion(self, reverse):
654
580
        """Return true if applying the entry would delete a file/directory.
655
581
 
656
582
        :param reverse: if true, the changeset is being applied in reverse
657
583
        :rtype: bool
658
584
        """
659
 
        return self.is_creation(not reverse)
 
585
        return ((self.new_parent is None and not reverse) or 
 
586
                (self.parent is None and reverse))
660
587
 
661
 
    def is_creation(self, reverse=False):
 
588
    def is_creation(self, reverse):
662
589
        """Return true if applying the entry would create a file/directory.
663
590
 
664
591
        :param reverse: if true, the changeset is being applied in reverse
665
592
        :rtype: bool
666
593
        """
667
 
        if self.contents_change is None:
668
 
            return False
669
 
        if reverse:
670
 
            return self.contents_change.is_deletion()
671
 
        else:
672
 
            return self.contents_change.is_creation()
 
594
        return ((self.parent is None and not reverse) or 
 
595
                (self.new_parent is None and reverse))
673
596
 
674
597
    def is_creation_or_deletion(self):
675
598
        """Return true if applying the entry would create or delete a 
677
600
 
678
601
        :rtype: bool
679
602
        """
680
 
        return self.is_creation() or self.is_deletion()
 
603
        return self.parent is None or self.new_parent is None
681
604
 
682
605
    def get_cset_path(self, mod=False):
683
606
        """Determine the path of the entry according to the changeset.
703
626
                return None
704
627
            return self.path
705
628
 
706
 
    def summarize_name(self):
 
629
    def summarize_name(self, reverse=False):
707
630
        """Produce a one-line summary of the filename.  Indicates renames as
708
631
        old => new, indicates creation as None => new, indicates deletion as
709
632
        old => None.
710
633
 
 
634
        :param changeset: The changeset to get paths from
 
635
        :type changeset: `Changeset`
 
636
        :param reverse: If true, reverse the names in the output
 
637
        :type reverse: bool
711
638
        :rtype: str
712
639
        """
713
640
        orig_path = self.get_cset_path(False)
714
641
        mod_path = self.get_cset_path(True)
715
 
        if orig_path and orig_path.startswith('./'):
 
642
        if orig_path is not None:
716
643
            orig_path = orig_path[2:]
717
 
        if mod_path and mod_path.startswith('./'):
 
644
        if mod_path is not None:
718
645
            mod_path = mod_path[2:]
719
646
        if orig_path == mod_path:
720
647
            return orig_path
721
648
        else:
722
 
            return "%s => %s" % (orig_path, mod_path)
723
 
 
724
 
    def get_new_path(self, id_map, changeset):
 
649
            if not reverse:
 
650
                return "%s => %s" % (orig_path, mod_path)
 
651
            else:
 
652
                return "%s => %s" % (mod_path, orig_path)
 
653
 
 
654
 
 
655
    def get_new_path(self, id_map, changeset, reverse=False):
725
656
        """Determine the full pathname to rename to
726
657
 
727
658
        :param id_map: The map of ids to filenames for the tree
728
659
        :type id_map: Dictionary
729
660
        :param changeset: The changeset to get data from
730
661
        :type changeset: `Changeset`
 
662
        :param reverse: If true, we're applying the changeset in reverse
 
663
        :type reverse: bool
731
664
        :rtype: str
732
665
        """
733
 
        mutter("Finding new path for %s", self.summarize_name())
734
 
        parent = self.new_parent
735
 
        to_dir = self.new_dir
736
 
        from_dir = self.dir
737
 
        to_name = self.new_name
738
 
        from_name = self.name
 
666
        mutter("Finding new path for %s" % self.summarize_name())
 
667
        if reverse:
 
668
            parent = self.parent
 
669
            to_dir = self.dir
 
670
            from_dir = self.new_dir
 
671
            to_name = self.name
 
672
            from_name = self.new_name
 
673
        else:
 
674
            parent = self.new_parent
 
675
            to_dir = self.new_dir
 
676
            from_dir = self.dir
 
677
            to_name = self.new_name
 
678
            from_name = self.name
739
679
 
740
680
        if to_name is None:
741
681
            return None
742
682
 
743
683
        if parent == NULL_ID or parent is None:
744
 
            if to_name != u'.':
 
684
            if to_name != '.':
745
685
                raise SourceRootHasName(self, to_name)
746
686
            else:
747
 
                return u'.'
748
 
        parent_entry = changeset.entries.get(parent)
749
 
        if parent_entry is None:
 
687
                return '.'
 
688
        if from_dir == to_dir:
750
689
            dir = os.path.dirname(id_map[self.id])
751
690
        else:
752
 
            mutter("path, new_path: %r %r", self.path, self.new_path)
753
 
            dir = parent_entry.get_new_path(id_map, changeset)
 
691
            mutter("path, new_path: %r %r" % (self.path, self.new_path))
 
692
            parent_entry = changeset.entries[parent]
 
693
            dir = parent_entry.get_new_path(id_map, changeset, reverse)
754
694
        if from_name == to_name:
755
695
            name = os.path.basename(id_map[self.id])
756
696
        else:
757
697
            name = to_name
758
698
            assert(from_name is None or from_name == os.path.basename(id_map[self.id]))
759
 
        return pathjoin(dir, name)
 
699
        return os.path.join(dir, name)
760
700
 
761
701
    def is_boring(self):
762
702
        """Determines whether the entry does nothing
775
715
        else:
776
716
            return True
777
717
 
778
 
    def apply(self, filename, conflict_handler):
 
718
    def apply(self, filename, conflict_handler, reverse=False):
779
719
        """Applies the file content and/or metadata changes.
780
720
 
781
721
        :param filename: the filename of the entry
782
722
        :type filename: str
 
723
        :param reverse: If true, apply the changes in reverse
 
724
        :type reverse: bool
783
725
        """
784
 
        if self.is_deletion() and self.metadata_change is not None:
785
 
            self.metadata_change.apply(filename, conflict_handler)
 
726
        if self.is_deletion(reverse) and self.metadata_change is not None:
 
727
            self.metadata_change.apply(filename, conflict_handler, reverse)
786
728
        if self.contents_change is not None:
787
 
            self.contents_change.apply(filename, conflict_handler)
788
 
        if not self.is_deletion() and self.metadata_change is not None:
789
 
            self.metadata_change.apply(filename, conflict_handler)
790
 
 
 
729
            self.contents_change.apply(filename, conflict_handler, reverse)
 
730
        if not self.is_deletion(reverse) and self.metadata_change is not None:
 
731
            self.metadata_change.apply(filename, conflict_handler, reverse)
791
732
 
792
733
class IDPresent(Exception):
793
734
    def __init__(self, id):
796
737
        Exception.__init__(self, msg)
797
738
        self.id = id
798
739
 
799
 
 
800
740
class Changeset(object):
801
741
    """A set of changes to apply"""
802
742
    def __init__(self):
808
748
            raise IDPresent(entry.id)
809
749
        self.entries[entry.id] = entry
810
750
 
 
751
def my_sort(sequence, key, reverse=False):
 
752
    """A sort function that supports supplying a key for comparison
 
753
    
 
754
    :param sequence: The sequence to sort
 
755
    :param key: A callable object that returns the values to be compared
 
756
    :param reverse: If true, sort in reverse order
 
757
    :type reverse: bool
 
758
    """
 
759
    def cmp_by_key(entry_a, entry_b):
 
760
        if reverse:
 
761
            tmp=entry_a
 
762
            entry_a = entry_b
 
763
            entry_b = tmp
 
764
        return cmp(key(entry_a), key(entry_b))
 
765
    sequence.sort(cmp_by_key)
811
766
 
812
 
def get_rename_entries(changeset, inventory):
 
767
def get_rename_entries(changeset, inventory, reverse):
813
768
    """Return a list of entries that will be renamed.  Entries are sorted from
814
769
    longest to shortest source path and from shortest to longest target path.
815
770
 
817
772
    :type changeset: `Changeset`
818
773
    :param inventory: The source of current tree paths for the given ids
819
774
    :type inventory: Dictionary
 
775
    :param reverse: If true, the changeset is being applied in reverse
 
776
    :type reverse: bool
820
777
    :return: source entries and target entries as a tuple
821
778
    :rtype: (List, List)
822
779
    """
823
780
    source_entries = [x for x in changeset.entries.itervalues() 
824
 
                      if x.needs_rename() or x.is_creation_or_deletion()]
 
781
                      if x.needs_rename()]
825
782
    # these are done from longest path to shortest, to avoid deleting a
826
783
    # parent before its children are deleted/renamed 
827
784
    def longest_to_shortest(entry):
830
787
            return 0
831
788
        else:
832
789
            return len(path)
833
 
    source_entries.sort(None, longest_to_shortest, True)
 
790
    my_sort(source_entries, longest_to_shortest, reverse=True)
834
791
 
835
792
    target_entries = source_entries[:]
836
793
    # These are done from shortest to longest path, to avoid creating a
837
794
    # child before its parent has been created/renamed
838
795
    def shortest_to_longest(entry):
839
 
        path = entry.get_new_path(inventory, changeset)
 
796
        path = entry.get_new_path(inventory, changeset, reverse)
840
797
        if path is None:
841
798
            return 0
842
799
        else:
843
800
            return len(path)
844
 
    target_entries.sort(None, shortest_to_longest)
 
801
    my_sort(target_entries, shortest_to_longest)
845
802
    return (source_entries, target_entries)
846
803
 
847
 
 
848
804
def rename_to_temp_delete(source_entries, inventory, dir, temp_dir, 
849
 
                          conflict_handler):
 
805
                          conflict_handler, reverse):
850
806
    """Delete and rename entries as appropriate.  Entries are renamed to temp
851
807
    names.  A map of id -> temp name (or None, for deletions) is returned.
852
808
 
856
812
    :type inventory: Dictionary
857
813
    :param dir: The directory to apply changes to
858
814
    :type dir: str
 
815
    :param reverse: Apply changes in reverse
 
816
    :type reverse: bool
859
817
    :return: a mapping of id to temporary name
860
818
    :rtype: Dictionary
861
819
    """
862
820
    temp_name = {}
863
821
    for i in range(len(source_entries)):
864
822
        entry = source_entries[i]
865
 
        if entry.is_deletion():
866
 
            path = pathjoin(dir, inventory[entry.id])
867
 
            entry.apply(path, conflict_handler)
 
823
        if entry.is_deletion(reverse):
 
824
            path = os.path.join(dir, inventory[entry.id])
 
825
            entry.apply(path, conflict_handler, reverse)
868
826
            temp_name[entry.id] = None
869
827
 
870
 
        elif entry.needs_rename():
871
 
            if entry.is_creation():
872
 
                continue
873
 
            to_name = pathjoin(temp_dir, str(i))
 
828
        else:
 
829
            to_name = os.path.join(temp_dir, str(i))
874
830
            src_path = inventory.get(entry.id)
875
831
            if src_path is not None:
876
 
                src_path = pathjoin(dir, src_path)
 
832
                src_path = os.path.join(dir, src_path)
877
833
                try:
878
 
                    rename(src_path, to_name)
 
834
                    os.rename(src_path, to_name)
879
835
                    temp_name[entry.id] = to_name
880
836
                except OSError, e:
881
837
                    if e.errno != errno.ENOENT:
882
838
                        raise
883
 
                    if conflict_handler.missing_for_rename(src_path, to_name) \
884
 
                        == "skip":
 
839
                    if conflict_handler.missing_for_rename(src_path) == "skip":
885
840
                        continue
886
841
 
887
842
    return temp_name
888
843
 
889
844
 
890
845
def rename_to_new_create(changed_inventory, target_entries, inventory, 
891
 
                         changeset, dir, conflict_handler):
 
846
                         changeset, dir, conflict_handler, reverse):
892
847
    """Rename entries with temp names to their final names, create new files.
893
848
 
894
849
    :param changed_inventory: A mapping of id to temporary name
899
854
    :type changeset: `Changeset`
900
855
    :param dir: The directory to apply changes to
901
856
    :type dir: str
 
857
    :param reverse: If true, apply changes in reverse
 
858
    :type reverse: bool
902
859
    """
903
860
    for entry in target_entries:
904
 
        new_tree_path = entry.get_new_path(inventory, changeset)
 
861
        new_tree_path = entry.get_new_path(inventory, changeset, reverse)
905
862
        if new_tree_path is None:
906
863
            continue
907
 
        new_path = pathjoin(dir, new_tree_path)
 
864
        new_path = os.path.join(dir, new_tree_path)
908
865
        old_path = changed_inventory.get(entry.id)
909
 
        if bzrlib.osutils.lexists(new_path):
 
866
        if os.path.exists(new_path):
910
867
            if conflict_handler.target_exists(entry, new_path, old_path) == \
911
868
                "skip":
912
869
                continue
913
 
        if entry.is_creation():
914
 
            entry.apply(new_path, conflict_handler)
 
870
        if entry.is_creation(reverse):
 
871
            entry.apply(new_path, conflict_handler, reverse)
915
872
            changed_inventory[entry.id] = new_tree_path
916
 
        elif entry.needs_rename():
917
 
            if entry.is_deletion():
918
 
                continue
 
873
        else:
919
874
            if old_path is None:
920
875
                continue
921
876
            try:
922
 
                mutter('rename %s to final name %s', old_path, new_path)
923
 
                rename(old_path, new_path)
 
877
                os.rename(old_path, new_path)
924
878
                changed_inventory[entry.id] = new_tree_path
925
879
            except OSError, e:
926
 
                raise BzrCheckError('failed to rename %s to %s for changeset entry %s: %s'
927
 
                        % (old_path, new_path, entry, e))
928
 
 
 
880
                raise Exception ("%s is missing" % new_path)
929
881
 
930
882
class TargetExists(Exception):
931
883
    def __init__(self, entry, target):
934
886
        self.entry = entry
935
887
        self.target = target
936
888
 
937
 
 
938
889
class RenameConflict(Exception):
939
890
    def __init__(self, id, this_name, base_name, other_name):
940
891
        msg = """Trees all have different names for a file
947
898
        self.base_name = base_name
948
899
        self_other_name = other_name
949
900
 
950
 
 
951
901
class MoveConflict(Exception):
952
902
    def __init__(self, id, this_parent, base_parent, other_parent):
953
903
        msg = """The file is in different directories in every tree
960
910
        self.base_parent = base_parent
961
911
        self_other_parent = other_parent
962
912
 
963
 
 
964
913
class MergeConflict(Exception):
965
914
    def __init__(self, this_path):
966
915
        Exception.__init__(self, "Conflict applying changes to %s" % this_path)
967
916
        self.this_path = this_path
968
917
 
 
918
class MergePermissionConflict(Exception):
 
919
    def __init__(self, this_path, base_path, other_path):
 
920
        this_perms = os.stat(this_path).st_mode & 0755
 
921
        base_perms = os.stat(base_path).st_mode & 0755
 
922
        other_perms = os.stat(other_path).st_mode & 0755
 
923
        msg = """Conflicting permission for %s
 
924
this: %o
 
925
base: %o
 
926
other: %o
 
927
        """ % (this_path, this_perms, base_perms, other_perms)
 
928
        self.this_path = this_path
 
929
        self.base_path = base_path
 
930
        self.other_path = other_path
 
931
        Exception.__init__(self, msg)
969
932
 
970
933
class WrongOldContents(Exception):
971
934
    def __init__(self, filename):
973
936
        self.filename = filename
974
937
        Exception.__init__(self, msg)
975
938
 
976
 
 
977
 
class WrongOldExecFlag(Exception):
978
 
    def __init__(self, filename, old_exec_flag, new_exec_flag):
979
 
        msg = "Executable flag missmatch on %s:\n" \
980
 
        "Expected %s, got %s." % (filename, old_exec_flag, new_exec_flag)
 
939
class WrongOldPermissions(Exception):
 
940
    def __init__(self, filename, old_perms, new_perms):
 
941
        msg = "Permission missmatch on %s:\n" \
 
942
        "Expected 0%o, got 0%o." % (filename, old_perms, new_perms)
981
943
        self.filename = filename
982
944
        Exception.__init__(self, msg)
983
945
 
984
 
 
985
946
class RemoveContentsConflict(Exception):
986
947
    def __init__(self, filename):
987
948
        msg = "Conflict deleting %s, which has different contents in BASE"\
989
950
        self.filename = filename
990
951
        Exception.__init__(self, msg)
991
952
 
992
 
 
993
953
class DeletingNonEmptyDirectory(Exception):
994
954
    def __init__(self, filename):
995
955
        msg = "Trying to remove dir %s while it still had files" % filename
1003
963
        Exception.__init__(self, msg)
1004
964
        self.filename = filename
1005
965
 
1006
 
 
1007
 
class MissingForSetExec(Exception):
 
966
class MissingPermsFile(Exception):
1008
967
    def __init__(self, filename):
1009
968
        msg = "Attempt to change permissions on  %s, which does not exist" %\
1010
969
            filename
1011
970
        Exception.__init__(self, msg)
1012
971
        self.filename = filename
1013
972
 
1014
 
 
1015
973
class MissingForRm(Exception):
1016
974
    def __init__(self, filename):
1017
975
        msg = "Attempt to remove missing path %s" % filename
1020
978
 
1021
979
 
1022
980
class MissingForRename(Exception):
1023
 
    def __init__(self, filename, to_path):
1024
 
        msg = "Attempt to move missing path %s to %s" % (filename, to_path)
 
981
    def __init__(self, filename):
 
982
        msg = "Attempt to move missing path %s" % (filename)
1025
983
        Exception.__init__(self, msg)
1026
984
        self.filename = filename
1027
985
 
1028
 
 
1029
986
class NewContentsConflict(Exception):
1030
987
    def __init__(self, filename):
1031
988
        msg = "Conflicting contents for new file %s" % (filename)
1032
989
        Exception.__init__(self, msg)
1033
990
 
1034
991
 
1035
 
class WeaveMergeConflict(Exception):
1036
 
    def __init__(self, filename):
1037
 
        msg = "Conflicting contents for file %s" % (filename)
1038
 
        Exception.__init__(self, msg)
1039
 
 
1040
 
 
1041
 
class ThreewayContentsConflict(Exception):
1042
 
    def __init__(self, filename):
1043
 
        msg = "Conflicting contents for file %s" % (filename)
1044
 
        Exception.__init__(self, msg)
1045
 
 
1046
 
 
1047
992
class MissingForMerge(Exception):
1048
993
    def __init__(self, filename):
1049
994
        msg = "The file %s was modified, but does not exist in this tree"\
1058
1003
    descend from this class if they have a better way to handle some or
1059
1004
    all types of conflict.
1060
1005
    """
 
1006
    def __init__(self, dir):
 
1007
        self.dir = dir
 
1008
    
1061
1009
    def missing_parent(self, pathname):
1062
1010
        parent = os.path.dirname(pathname)
1063
1011
        raise Exception("Parent directory missing for %s" % pathname)
1081
1029
        os.unlink(new_file)
1082
1030
        raise MergeConflict(this_path)
1083
1031
 
 
1032
    def permission_conflict(self, this_path, base_path, other_path):
 
1033
        raise MergePermissionConflict(this_path, base_path, other_path)
 
1034
 
1084
1035
    def wrong_old_contents(self, filename, expected_contents):
1085
1036
        raise WrongOldContents(filename)
1086
1037
 
1087
1038
    def rem_contents_conflict(self, filename, this_contents, base_contents):
1088
1039
        raise RemoveContentsConflict(filename)
1089
1040
 
1090
 
    def wrong_old_exec_flag(self, filename, old_exec_flag, new_exec_flag):
1091
 
        raise WrongOldExecFlag(filename, old_exec_flag, new_exec_flag)
 
1041
    def wrong_old_perms(self, filename, old_perms, new_perms):
 
1042
        raise WrongOldPermissions(filename, old_perms, new_perms)
1092
1043
 
1093
1044
    def rmdir_non_empty(self, filename):
1094
1045
        raise DeletingNonEmptyDirectory(filename)
1099
1050
    def patch_target_missing(self, filename, contents):
1100
1051
        raise PatchTargetMissing(filename)
1101
1052
 
1102
 
    def missing_for_exec_flag(self, filename):
1103
 
        raise MissingForExecFlag(filename)
 
1053
    def missing_for_chmod(self, filename):
 
1054
        raise MissingPermsFile(filename)
1104
1055
 
1105
1056
    def missing_for_rm(self, filename, change):
1106
1057
        raise MissingForRm(filename)
1107
1058
 
1108
 
    def missing_for_rename(self, filename, to_path):
1109
 
        raise MissingForRename(filename, to_path)
 
1059
    def missing_for_rename(self, filename):
 
1060
        raise MissingForRename(filename)
1110
1061
 
1111
1062
    def missing_for_merge(self, file_id, other_path):
1112
1063
        raise MissingForMerge(other_path)
1114
1065
    def new_contents_conflict(self, filename, other_contents):
1115
1066
        raise NewContentsConflict(filename)
1116
1067
 
1117
 
    def weave_merge_conflict(self, filename, weave, other_i, out_file):
1118
 
        raise WeaveMergeConflict(filename)
1119
 
 
1120
 
    def threeway_contents_conflict(self, filename, this_contents,
1121
 
                                   base_contents, other_contents):
1122
 
        raise ThreewayContentsConflict(filename)
1123
 
 
1124
1068
    def finalize(self):
1125
1069
        pass
1126
1070
 
1127
 
 
1128
 
def apply_changeset(changeset, inventory, dir, conflict_handler=None):
 
1071
def apply_changeset(changeset, inventory, dir, conflict_handler=None, 
 
1072
                    reverse=False):
1129
1073
    """Apply a changeset to a directory.
1130
1074
 
1131
1075
    :param changeset: The changes to perform
1134
1078
    :type inventory: Dictionary
1135
1079
    :param dir: The path of the directory to apply the changes to
1136
1080
    :type dir: str
 
1081
    :param reverse: If true, apply the changes in reverse
 
1082
    :type reverse: bool
1137
1083
    :return: The mapping of the changed entries
1138
1084
    :rtype: Dictionary
1139
1085
    """
1140
1086
    if conflict_handler is None:
1141
 
        conflict_handler = ExceptionConflictHandler()
1142
 
    temp_dir = pathjoin(dir, "bzr-tree-change")
 
1087
        conflict_handler = ExceptionConflictHandler(dir)
 
1088
    temp_dir = os.path.join(dir, "bzr-tree-change")
1143
1089
    try:
1144
1090
        os.mkdir(temp_dir)
1145
1091
    except OSError, e:
1155
1101
    
1156
1102
    #apply changes that don't affect filenames
1157
1103
    for entry in changeset.entries.itervalues():
1158
 
        if not entry.is_creation_or_deletion() and not entry.is_boring():
1159
 
            if entry.id not in inventory:
1160
 
                warning("entry {%s} no longer present, can't be updated",
1161
 
                        entry.id)
1162
 
                continue
1163
 
            path = pathjoin(dir, inventory[entry.id])
1164
 
            entry.apply(path, conflict_handler)
 
1104
        if not entry.is_creation_or_deletion():
 
1105
            path = os.path.join(dir, inventory[entry.id])
 
1106
            entry.apply(path, conflict_handler, reverse)
1165
1107
 
1166
1108
    # Apply renames in stages, to minimize conflicts:
1167
1109
    # Only files whose name or parent change are interesting, because their
1168
1110
    # target name may exist in the source tree.  If a directory's name changes,
1169
1111
    # that doesn't make its children interesting.
1170
 
    (source_entries, target_entries) = get_rename_entries(changeset, inventory)
 
1112
    (source_entries, target_entries) = get_rename_entries(changeset, inventory,
 
1113
                                                          reverse)
1171
1114
 
1172
1115
    changed_inventory = rename_to_temp_delete(source_entries, inventory, dir,
1173
 
                                              temp_dir, conflict_handler)
 
1116
                                              temp_dir, conflict_handler,
 
1117
                                              reverse)
1174
1118
 
1175
1119
    rename_to_new_create(changed_inventory, target_entries, inventory,
1176
 
                         changeset, dir, conflict_handler)
 
1120
                         changeset, dir, conflict_handler, reverse)
1177
1121
    os.rmdir(temp_dir)
1178
1122
    return changed_inventory
1179
1123
 
1180
1124
 
1181
 
def apply_changeset_tree(cset, tree):
 
1125
def apply_changeset_tree(cset, tree, reverse=False):
1182
1126
    r_inventory = {}
1183
1127
    for entry in tree.source_inventory().itervalues():
1184
1128
        inventory[entry.id] = entry.path
1185
 
    new_inventory = apply_changeset(cset, r_inventory, tree.basedir)
 
1129
    new_inventory = apply_changeset(cset, r_inventory, tree.root,
 
1130
                                    reverse=reverse)
1186
1131
    new_entries, remove_entries = \
1187
 
        get_inventory_change(inventory, new_inventory, cset)
 
1132
        get_inventory_change(inventory, new_inventory, cset, reverse)
1188
1133
    tree.update_source_inventory(new_entries, remove_entries)
1189
1134
 
1190
1135
 
1191
 
def get_inventory_change(inventory, new_inventory, cset):
 
1136
def get_inventory_change(inventory, new_inventory, cset, reverse=False):
1192
1137
    new_entries = {}
1193
1138
    remove_entries = []
1194
1139
    for entry in cset.entries.itervalues():
1213
1158
        print entry.id
1214
1159
        print entry.summarize_name(cset)
1215
1160
 
1216
 
 
1217
 
class UnsupportedFiletype(Exception):
1218
 
    def __init__(self, kind, full_path):
1219
 
        msg = "The file \"%s\" is a %s, which is not a supported filetype." \
1220
 
            % (full_path, kind)
 
1161
class CompositionFailure(Exception):
 
1162
    def __init__(self, old_entry, new_entry, problem):
 
1163
        msg = "Unable to conpose entries.\n %s" % problem
 
1164
        Exception.__init__(self, msg)
 
1165
 
 
1166
class IDMismatch(CompositionFailure):
 
1167
    def __init__(self, old_entry, new_entry):
 
1168
        problem = "Attempt to compose entries with different ids: %s and %s" %\
 
1169
            (old_entry.id, new_entry.id)
 
1170
        CompositionFailure.__init__(self, old_entry, new_entry, problem)
 
1171
 
 
1172
def compose_changesets(old_cset, new_cset):
 
1173
    """Combine two changesets into one.  This works well for exact patching.
 
1174
    Otherwise, not so well.
 
1175
 
 
1176
    :param old_cset: The first changeset that would be applied
 
1177
    :type old_cset: `Changeset`
 
1178
    :param new_cset: The second changeset that would be applied
 
1179
    :type new_cset: `Changeset`
 
1180
    :return: A changeset that combines the changes in both changesets
 
1181
    :rtype: `Changeset`
 
1182
    """
 
1183
    composed = Changeset()
 
1184
    for old_entry in old_cset.entries.itervalues():
 
1185
        new_entry = new_cset.entries.get(old_entry.id)
 
1186
        if new_entry is None:
 
1187
            composed.add_entry(old_entry)
 
1188
        else:
 
1189
            composed_entry = compose_entries(old_entry, new_entry)
 
1190
            if composed_entry.parent is not None or\
 
1191
                composed_entry.new_parent is not None:
 
1192
                composed.add_entry(composed_entry)
 
1193
    for new_entry in new_cset.entries.itervalues():
 
1194
        if not old_cset.entries.has_key(new_entry.id):
 
1195
            composed.add_entry(new_entry)
 
1196
    return composed
 
1197
 
 
1198
def compose_entries(old_entry, new_entry):
 
1199
    """Combine two entries into one.
 
1200
 
 
1201
    :param old_entry: The first entry that would be applied
 
1202
    :type old_entry: ChangesetEntry
 
1203
    :param old_entry: The second entry that would be applied
 
1204
    :type old_entry: ChangesetEntry
 
1205
    :return: A changeset entry combining both entries
 
1206
    :rtype: `ChangesetEntry`
 
1207
    """
 
1208
    if old_entry.id != new_entry.id:
 
1209
        raise IDMismatch(old_entry, new_entry)
 
1210
    output = ChangesetEntry(old_entry.id, old_entry.parent, old_entry.path)
 
1211
 
 
1212
    if (old_entry.parent != old_entry.new_parent or 
 
1213
        new_entry.parent != new_entry.new_parent):
 
1214
        output.new_parent = new_entry.new_parent
 
1215
 
 
1216
    if (old_entry.path != old_entry.new_path or 
 
1217
        new_entry.path != new_entry.new_path):
 
1218
        output.new_path = new_entry.new_path
 
1219
 
 
1220
    output.contents_change = compose_contents(old_entry, new_entry)
 
1221
    output.metadata_change = compose_metadata(old_entry, new_entry)
 
1222
    return output
 
1223
 
 
1224
def compose_contents(old_entry, new_entry):
 
1225
    """Combine the contents of two changeset entries.  Entries are combined
 
1226
    intelligently where possible, but the fallback behavior returns an 
 
1227
    ApplySequence.
 
1228
 
 
1229
    :param old_entry: The first entry that would be applied
 
1230
    :type old_entry: `ChangesetEntry`
 
1231
    :param new_entry: The second entry that would be applied
 
1232
    :type new_entry: `ChangesetEntry`
 
1233
    :return: A combined contents change
 
1234
    :rtype: anything supporting the apply(reverse=False) method
 
1235
    """
 
1236
    old_contents = old_entry.contents_change
 
1237
    new_contents = new_entry.contents_change
 
1238
    if old_entry.contents_change is None:
 
1239
        return new_entry.contents_change
 
1240
    elif new_entry.contents_change is None:
 
1241
        return old_entry.contents_change
 
1242
    elif isinstance(old_contents, ReplaceContents) and \
 
1243
        isinstance(new_contents, ReplaceContents):
 
1244
        if old_contents.old_contents == new_contents.new_contents:
 
1245
            return None
 
1246
        else:
 
1247
            return ReplaceContents(old_contents.old_contents,
 
1248
                                   new_contents.new_contents)
 
1249
    elif isinstance(old_contents, ApplySequence):
 
1250
        output = ApplySequence(old_contents.changes)
 
1251
        if isinstance(new_contents, ApplySequence):
 
1252
            output.changes.extend(new_contents.changes)
 
1253
        else:
 
1254
            output.changes.append(new_contents)
 
1255
        return output
 
1256
    elif isinstance(new_contents, ApplySequence):
 
1257
        output = ApplySequence((old_contents.changes,))
 
1258
        output.extend(new_contents.changes)
 
1259
        return output
 
1260
    else:
 
1261
        return ApplySequence((old_contents, new_contents))
 
1262
 
 
1263
def compose_metadata(old_entry, new_entry):
 
1264
    old_meta = old_entry.metadata_change
 
1265
    new_meta = new_entry.metadata_change
 
1266
    if old_meta is None:
 
1267
        return new_meta
 
1268
    elif new_meta is None:
 
1269
        return old_meta
 
1270
    elif isinstance(old_meta, ChangeUnixPermissions) and \
 
1271
        isinstance(new_meta, ChangeUnixPermissions):
 
1272
        return ChangeUnixPermissions(old_meta.old_mode, new_meta.new_mode)
 
1273
    else:
 
1274
        return ApplySequence(old_meta, new_meta)
 
1275
 
 
1276
 
 
1277
def changeset_is_null(changeset):
 
1278
    for entry in changeset.entries.itervalues():
 
1279
        if not entry.is_boring():
 
1280
            return False
 
1281
    return True
 
1282
 
 
1283
class UnsuppportedFiletype(Exception):
 
1284
    def __init__(self, full_path, stat_result):
 
1285
        msg = "The file \"%s\" is not a supported filetype." % full_path
1221
1286
        Exception.__init__(self, msg)
1222
1287
        self.full_path = full_path
1223
 
        self.kind = kind
1224
 
 
 
1288
        self.stat_result = stat_result
1225
1289
 
1226
1290
def generate_changeset(tree_a, tree_b, interesting_ids=None):
1227
1291
    return ChangesetGenerator(tree_a, tree_b, interesting_ids)()
1228
1292
 
1229
 
 
1230
1293
class ChangesetGenerator(object):
1231
1294
    def __init__(self, tree_a, tree_b, interesting_ids=None):
1232
1295
        object.__init__(self)
1268
1331
    def get_entry(self, file_id, tree):
1269
1332
        if not tree.has_or_had_id(file_id):
1270
1333
            return None
1271
 
        return tree.inventory[file_id]
 
1334
        return tree.tree.inventory[file_id]
1272
1335
 
1273
1336
    def get_entry_parent(self, entry):
1274
1337
        if entry is None:
1317
1380
            return self.make_entry(id, only_interesting=False)
1318
1381
        else:
1319
1382
            return cs_entry
 
1383
        
1320
1384
 
1321
1385
    def make_entry(self, id, only_interesting=True):
1322
1386
        cs_entry = self.make_basic_entry(id, only_interesting)
1323
1387
 
1324
1388
        if cs_entry is None:
1325
1389
            return None
1326
 
 
1327
 
        cs_entry.metadata_change = self.make_exec_flag_change(id)
1328
 
 
1329
1390
        if id in self.tree_a and id in self.tree_b:
1330
1391
            a_sha1 = self.tree_a.get_file_sha1(id)
1331
1392
            b_sha1 = self.tree_b.get_file_sha1(id)
1332
1393
            if None not in (a_sha1, b_sha1) and a_sha1 == b_sha1:
1333
1394
                return cs_entry
1334
1395
 
1335
 
        cs_entry.contents_change = self.make_contents_change(id)
 
1396
        full_path_a = self.tree_a.readonly_path(id)
 
1397
        full_path_b = self.tree_b.readonly_path(id)
 
1398
        stat_a = self.lstat(full_path_a)
 
1399
        stat_b = self.lstat(full_path_b)
 
1400
        
 
1401
        cs_entry.metadata_change = self.make_mode_change(stat_a, stat_b)
 
1402
        cs_entry.contents_change = self.make_contents_change(full_path_a,
 
1403
                                                             stat_a, 
 
1404
                                                             full_path_b, 
 
1405
                                                             stat_b)
1336
1406
        return cs_entry
1337
1407
 
1338
 
    def make_exec_flag_change(self, file_id):
1339
 
        exec_flag_a = exec_flag_b = None
1340
 
        if file_id in self.tree_a and self.tree_a.kind(file_id) == "file":
1341
 
            exec_flag_a = self.tree_a.is_executable(file_id)
1342
 
 
1343
 
        if file_id in self.tree_b and self.tree_b.kind(file_id) == "file":
1344
 
            exec_flag_b = self.tree_b.is_executable(file_id)
1345
 
 
1346
 
        if exec_flag_a == exec_flag_b:
1347
 
            return None
1348
 
        return ChangeExecFlag(exec_flag_a, exec_flag_b)
1349
 
 
1350
 
    def make_contents_change(self, file_id):
1351
 
        a_contents = get_contents(self.tree_a, file_id)
1352
 
        b_contents = get_contents(self.tree_b, file_id)
 
1408
    def make_mode_change(self, stat_a, stat_b):
 
1409
        mode_a = None
 
1410
        if stat_a is not None and not stat.S_ISLNK(stat_a.st_mode):
 
1411
            mode_a = stat_a.st_mode & 0777
 
1412
        mode_b = None
 
1413
        if stat_b is not None and not stat.S_ISLNK(stat_b.st_mode):
 
1414
            mode_b = stat_b.st_mode & 0777
 
1415
        if mode_a == mode_b:
 
1416
            return None
 
1417
        return ChangeUnixPermissions(mode_a, mode_b)
 
1418
 
 
1419
    def make_contents_change(self, full_path_a, stat_a, full_path_b, stat_b):
 
1420
        if stat_a is None and stat_b is None:
 
1421
            return None
 
1422
        if None not in (stat_a, stat_b) and stat.S_ISDIR(stat_a.st_mode) and\
 
1423
            stat.S_ISDIR(stat_b.st_mode):
 
1424
            return None
 
1425
        if None not in (stat_a, stat_b) and stat.S_ISREG(stat_a.st_mode) and\
 
1426
            stat.S_ISREG(stat_b.st_mode):
 
1427
            if stat_a.st_ino == stat_b.st_ino and \
 
1428
                stat_a.st_dev == stat_b.st_dev:
 
1429
                return None
 
1430
 
 
1431
        a_contents = self.get_contents(stat_a, full_path_a)
 
1432
        b_contents = self.get_contents(stat_b, full_path_b)
1353
1433
        if a_contents == b_contents:
1354
1434
            return None
1355
1435
        return ReplaceContents(a_contents, b_contents)
1356
1436
 
 
1437
    def get_contents(self, stat_result, full_path):
 
1438
        if stat_result is None:
 
1439
            return None
 
1440
        elif stat.S_ISREG(stat_result.st_mode):
 
1441
            return FileCreate(file(full_path, "rb").read())
 
1442
        elif stat.S_ISDIR(stat_result.st_mode):
 
1443
            return dir_create
 
1444
        elif stat.S_ISLNK(stat_result.st_mode):
 
1445
            return SymlinkCreate(os.readlink(full_path))
 
1446
        else:
 
1447
            raise UnsupportedFiletype(full_path, stat_result)
1357
1448
 
1358
 
def get_contents(tree, file_id):
1359
 
    """Return the appropriate contents to create a copy of file_id from tree"""
1360
 
    if file_id not in tree:
1361
 
        return None
1362
 
    kind = tree.kind(file_id)
1363
 
    if kind == "file":
1364
 
        return TreeFileCreate(tree, file_id)
1365
 
    elif kind in ("directory", "root_directory"):
1366
 
        return dir_create
1367
 
    elif kind == "symlink":
1368
 
        return SymlinkCreate(tree.get_symlink_target(file_id))
1369
 
    else:
1370
 
        raise UnsupportedFiletype(kind, tree.id2path(file_id))
 
1449
    def lstat(self, full_path):
 
1450
        stat_result = None
 
1451
        if full_path is not None:
 
1452
            try:
 
1453
                stat_result = os.lstat(full_path)
 
1454
            except OSError, e:
 
1455
                if e.errno != errno.ENOENT:
 
1456
                    raise
 
1457
        return stat_result
1371
1458
 
1372
1459
 
1373
1460
def full_path(entry, tree):
1374
 
    return pathjoin(tree.basedir, entry.path)
1375
 
 
 
1461
    return os.path.join(tree.root, entry.path)
1376
1462
 
1377
1463
def new_delete_entry(entry, tree, inventory, delete):
1378
1464
    if entry.path == "":
1390
1476
    status = os.lstat(full_path)
1391
1477
    if stat.S_ISDIR(file_stat.st_mode):
1392
1478
        action = dir_create
1393
 
 
1394
 
 
 
1479
    
 
1480
 
 
1481
 
 
1482
        
1395
1483
# XXX: Can't we unify this with the regular inventory object
1396
1484
class Inventory(object):
1397
1485
    def __init__(self, inventory):
1426
1514
            return None
1427
1515
        directory = self.get_dir(id)
1428
1516
        if directory == '.':
1429
 
            directory = u'./.'
 
1517
            directory = './.'
1430
1518
        if directory is None:
1431
1519
            return NULL_ID
1432
1520
        return self.get_rinventory().get(directory)
1433
 
 
1434