~bzr-pqm/bzr/bzr.dev

493 by Martin Pool
- Merge aaron's merge command
1
# Copyright (C) 2004 Aaron Bentley <aaron.bentley@utoronto.ca>
2
#
3
#    This program is free software; you can redistribute it and/or modify
4
#    it under the terms of the GNU General Public License as published by
5
#    the Free Software Foundation; either version 2 of the License, or
6
#    (at your option) any later version.
7
#
8
#    This program is distributed in the hope that it will be useful,
9
#    but WITHOUT ANY WARRANTY; without even the implied warranty of
10
#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
#    GNU General Public License for more details.
12
#
13
#    You should have received a copy of the GNU General Public License
14
#    along with this program; if not, write to the Free Software
15
#    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
import os.path
17
import errno
18
import patch
19
import stat
1185.12.35 by abentley
Avoided readonly_path in ApplyDiff3
20
from tempfile import mkdtemp
21
from shutil import rmtree
909.1.2 by aaron.bentley at utoronto
Fixed rename handling in merge
22
from bzrlib.trace import mutter
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
23
from bzrlib.osutils import rename, sha_file
1092.2.24 by Robert Collins
merge from martins newformat branch - brings in transport abstraction
24
import bzrlib
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
25
from itertools import izip
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
26
27
# XXX: mbp: I'm not totally convinced that we should handle conflicts
28
# as part of changeset application, rather than only in the merge
29
# operation.
30
31
"""Represent and apply a changeset
32
33
Conflicts in applying a changeset are represented as exceptions.
34
"""
35
493 by Martin Pool
- Merge aaron's merge command
36
__docformat__ = "restructuredtext"
37
38
NULL_ID = "!NULL"
39
850 by Martin Pool
- Merge merge updates from aaron
40
class OldFailedTreeOp(Exception):
41
    def __init__(self):
42
        Exception.__init__(self, "bzr-tree-change contains files from a"
43
                           " previous failed merge operation.")
493 by Martin Pool
- Merge aaron's merge command
44
def invert_dict(dict):
45
    newdict = {}
46
    for (key,value) in dict.iteritems():
47
        newdict[value] = key
48
    return newdict
49
974.1.15 by aaron.bentley at utoronto
Removed use of patch and diff in merge, removed patch.diff
50
       
1434 by Robert Collins
merge Gustavos executable2 patch
51
class ChangeExecFlag(object):
493 by Martin Pool
- Merge aaron's merge command
52
    """This is two-way change, suitable for file modification, creation,
53
    deletion"""
1434 by Robert Collins
merge Gustavos executable2 patch
54
    def __init__(self, old_exec_flag, new_exec_flag):
55
        self.old_exec_flag = old_exec_flag
56
        self.new_exec_flag = new_exec_flag
493 by Martin Pool
- Merge aaron's merge command
57
58
    def apply(self, filename, conflict_handler, reverse=False):
59
        if not reverse:
1434 by Robert Collins
merge Gustavos executable2 patch
60
            from_exec_flag = self.old_exec_flag
61
            to_exec_flag = self.new_exec_flag
493 by Martin Pool
- Merge aaron's merge command
62
        else:
1434 by Robert Collins
merge Gustavos executable2 patch
63
            from_exec_flag = self.new_exec_flag
64
            to_exec_flag = self.old_exec_flag
493 by Martin Pool
- Merge aaron's merge command
65
        try:
1434 by Robert Collins
merge Gustavos executable2 patch
66
            current_exec_flag = bool(os.stat(filename).st_mode & 0111)
493 by Martin Pool
- Merge aaron's merge command
67
        except OSError, e:
68
            if e.errno == errno.ENOENT:
1434 by Robert Collins
merge Gustavos executable2 patch
69
                if conflict_handler.missing_for_exec_flag(filename) == "skip":
493 by Martin Pool
- Merge aaron's merge command
70
                    return
71
                else:
1434 by Robert Collins
merge Gustavos executable2 patch
72
                    current_exec_flag = from_exec_flag
493 by Martin Pool
- Merge aaron's merge command
73
1434 by Robert Collins
merge Gustavos executable2 patch
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":
493 by Martin Pool
- Merge aaron's merge command
77
                return
78
1434 by Robert Collins
merge Gustavos executable2 patch
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
493 by Martin Pool
- Merge aaron's merge command
92
            try:
93
                os.chmod(filename, to_mode)
94
            except IOError, e:
95
                if e.errno == errno.ENOENT:
1434 by Robert Collins
merge Gustavos executable2 patch
96
                    conflict_handler.missing_for_exec_flag(filename)
493 by Martin Pool
- Merge aaron's merge command
97
98
    def __eq__(self, other):
1434 by Robert Collins
merge Gustavos executable2 patch
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)
493 by Martin Pool
- Merge aaron's merge command
102
103
    def __ne__(self, other):
104
        return not (self == other)
105
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
106
493 by Martin Pool
- Merge aaron's merge command
107
def dir_create(filename, conflict_handler, reverse):
108
    """Creates the directory, or deletes it if reverse is true.  Intended to be
109
    used with ReplaceContents.
110
111
    :param filename: The name of the directory to create
112
    :type filename: str
113
    :param reverse: If true, delete the directory, instead
114
    :type reverse: bool
115
    """
116
    if not reverse:
117
        try:
118
            os.mkdir(filename)
119
        except OSError, e:
120
            if e.errno != errno.EEXIST:
121
                raise
122
            if conflict_handler.dir_exists(filename) == "continue":
123
                os.mkdir(filename)
124
        except IOError, e:
125
            if e.errno == errno.ENOENT:
126
                if conflict_handler.missing_parent(filename)=="continue":
127
                    file(filename, "wb").write(self.contents)
128
    else:
129
        try:
130
            os.rmdir(filename)
131
        except OSError, e:
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
132
            if e.errno != errno.ENOTEMPTY:
493 by Martin Pool
- Merge aaron's merge command
133
                raise
134
            if conflict_handler.rmdir_non_empty(filename) == "skip":
135
                return
136
            os.rmdir(filename)
137
138
558 by Martin Pool
- All top-level classes inherit from object
139
class SymlinkCreate(object):
493 by Martin Pool
- Merge aaron's merge command
140
    """Creates or deletes a symlink (for use with ReplaceContents)"""
141
    def __init__(self, contents):
142
        """Constructor.
143
144
        :param contents: The filename of the target the symlink should point to
145
        :type contents: str
146
        """
147
        self.target = contents
148
1185.12.34 by Aaron Bentley
Added symlink three-way tests
149
    def __repr__(self):
150
        return "SymlinkCreate(%s)" % self.target
151
493 by Martin Pool
- Merge aaron's merge command
152
    def __call__(self, filename, conflict_handler, reverse):
153
        """Creates or destroys the symlink.
154
155
        :param filename: The name of the symlink to create
156
        :type filename: str
157
        """
158
        if reverse:
159
            assert(os.readlink(filename) == self.target)
160
            os.unlink(filename)
161
        else:
162
            try:
163
                os.symlink(self.target, filename)
164
            except OSError, e:
165
                if e.errno != errno.EEXIST:
166
                    raise
167
                if conflict_handler.link_name_exists(filename) == "continue":
168
                    os.symlink(self.target, filename)
169
170
    def __eq__(self, other):
171
        if not isinstance(other, SymlinkCreate):
172
            return False
173
        elif self.target != other.target:
174
            return False
175
        else:
176
            return True
177
178
    def __ne__(self, other):
179
        return not (self == other)
180
558 by Martin Pool
- All top-level classes inherit from object
181
class FileCreate(object):
493 by Martin Pool
- Merge aaron's merge command
182
    """Create or delete a file (for use with ReplaceContents)"""
183
    def __init__(self, contents):
184
        """Constructor
185
186
        :param contents: The contents of the file to write
187
        :type contents: str
188
        """
189
        self.contents = contents
190
191
    def __repr__(self):
192
        return "FileCreate(%i b)" % len(self.contents)
193
194
    def __eq__(self, other):
195
        if not isinstance(other, FileCreate):
196
            return False
197
        elif self.contents != other.contents:
198
            return False
199
        else:
200
            return True
201
202
    def __ne__(self, other):
203
        return not (self == other)
204
205
    def __call__(self, filename, conflict_handler, reverse):
206
        """Create or delete a file
207
208
        :param filename: The name of the file to create
209
        :type filename: str
210
        :param reverse: Delete the file instead of creating it
211
        :type reverse: bool
212
        """
213
        if not reverse:
214
            try:
215
                file(filename, "wb").write(self.contents)
216
            except IOError, e:
217
                if e.errno == errno.ENOENT:
218
                    if conflict_handler.missing_parent(filename)=="continue":
219
                        file(filename, "wb").write(self.contents)
220
                else:
221
                    raise
222
223
        else:
224
            try:
225
                if (file(filename, "rb").read() != self.contents):
226
                    direction = conflict_handler.wrong_old_contents(filename,
227
                                                                    self.contents)
228
                    if  direction != "continue":
229
                        return
230
                os.unlink(filename)
231
            except IOError, e:
232
                if e.errno != errno.ENOENT:
233
                    raise
234
                if conflict_handler.missing_for_rm(filename, undo) == "skip":
235
                    return
236
237
                    
238
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
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):
1185.12.34 by Aaron Bentley
Added symlink three-way tests
251
        return "TreeFileCreate(%s)" % self.file_id
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
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):
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
303
                    
304
493 by Martin Pool
- Merge aaron's merge command
305
def reversed(sequence):
306
    max = len(sequence) - 1
307
    for i in range(len(sequence)):
308
        yield sequence[max - i]
309
558 by Martin Pool
- All top-level classes inherit from object
310
class ReplaceContents(object):
493 by Martin Pool
- Merge aaron's merge command
311
    """A contents-replacement framework.  It allows a file/directory/symlink to
312
    be created, deleted, or replaced with another file/directory/symlink.
313
    Arguments must be callable with (filename, reverse).
314
    """
315
    def __init__(self, old_contents, new_contents):
316
        """Constructor.
317
318
        :param old_contents: The change to reverse apply (e.g. a deletion), \
319
        when going forwards.
320
        :type old_contents: `dir_create`, `SymlinkCreate`, `FileCreate`, \
321
        NoneType, etc.
322
        :param new_contents: The second change to apply (e.g. a creation), \
323
        when going forwards.
324
        :type new_contents: `dir_create`, `SymlinkCreate`, `FileCreate`, \
325
        NoneType, etc.
326
        """
327
        self.old_contents=old_contents
328
        self.new_contents=new_contents
329
330
    def __repr__(self):
331
        return "ReplaceContents(%r -> %r)" % (self.old_contents,
332
                                              self.new_contents)
333
334
    def __eq__(self, other):
335
        if not isinstance(other, ReplaceContents):
336
            return False
337
        elif self.old_contents != other.old_contents:
338
            return False
339
        elif self.new_contents != other.new_contents:
340
            return False
341
        else:
342
            return True
343
    def __ne__(self, other):
344
        return not (self == other)
345
346
    def apply(self, filename, conflict_handler, reverse=False):
347
        """Applies the FileReplacement to the specified filename
348
349
        :param filename: The name of the file to apply changes to
350
        :type filename: str
351
        :param reverse: If true, apply the change in reverse
352
        :type reverse: bool
353
        """
354
        if not reverse:
355
            undo = self.old_contents
356
            perform = self.new_contents
357
        else:
358
            undo = self.new_contents
359
            perform = self.old_contents
360
        mode = None
361
        if undo is not None:
362
            try:
363
                mode = os.lstat(filename).st_mode
364
                if stat.S_ISLNK(mode):
365
                    mode = None
366
            except OSError, e:
367
                if e.errno != errno.ENOENT:
368
                    raise
369
                if conflict_handler.missing_for_rm(filename, undo) == "skip":
370
                    return
371
            undo(filename, conflict_handler, reverse=True)
372
        if perform is not None:
373
            perform(filename, conflict_handler, reverse=False)
374
            if mode is not None:
375
                os.chmod(filename, mode)
376
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
377
    def is_creation(self):
378
        return self.new_contents is not None and self.old_contents is None
379
380
    def is_deletion(self):
381
        return self.old_contents is not None and self.new_contents is None
382
558 by Martin Pool
- All top-level classes inherit from object
383
class ApplySequence(object):
493 by Martin Pool
- Merge aaron's merge command
384
    def __init__(self, changes=None):
385
        self.changes = []
386
        if changes is not None:
387
            self.changes.extend(changes)
388
389
    def __eq__(self, other):
390
        if not isinstance(other, ApplySequence):
391
            return False
392
        elif len(other.changes) != len(self.changes):
393
            return False
394
        else:
395
            for i in range(len(self.changes)):
396
                if self.changes[i] != other.changes[i]:
397
                    return False
398
            return True
399
400
    def __ne__(self, other):
401
        return not (self == other)
402
403
    
404
    def apply(self, filename, conflict_handler, reverse=False):
405
        if not reverse:
406
            iter = self.changes
407
        else:
408
            iter = reversed(self.changes)
409
        for change in iter:
410
            change.apply(filename, conflict_handler, reverse)
411
412
558 by Martin Pool
- All top-level classes inherit from object
413
class Diff3Merge(object):
974.1.23 by Aaron Bentley
Avoided unnecessary temp files
414
    def __init__(self, file_id, base, other):
415
        self.file_id = file_id
416
        self.base = base
417
        self.other = other
493 by Martin Pool
- Merge aaron's merge command
418
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
419
    def is_creation(self):
420
        return False
421
422
    def is_deletion(self):
423
        return False
424
493 by Martin Pool
- Merge aaron's merge command
425
    def __eq__(self, other):
426
        if not isinstance(other, Diff3Merge):
427
            return False
974.1.23 by Aaron Bentley
Avoided unnecessary temp files
428
        return (self.base == other.base and 
429
                self.other == other.other and self.file_id == other.file_id)
493 by Martin Pool
- Merge aaron's merge command
430
431
    def __ne__(self, other):
432
        return not (self == other)
433
1185.12.35 by abentley
Avoided readonly_path in ApplyDiff3
434
    def dump_file(self, temp_dir, name, tree):
435
        out_path = os.path.join(temp_dir, name)
436
        out_file = file(out_path, "wb")
437
        in_file = tree.get_file(self.file_id)
438
        for line in in_file:
439
            out_file.write(line)
440
        return out_path
441
493 by Martin Pool
- Merge aaron's merge command
442
    def apply(self, filename, conflict_handler, reverse=False):
1185.12.35 by abentley
Avoided readonly_path in ApplyDiff3
443
        temp_dir = mkdtemp(prefix="bzr-")
444
        try:
445
            new_file = filename+".new"
446
            base_file = self.dump_file(temp_dir, "base", self.base)
447
            other_file = self.dump_file(temp_dir, "other", self.other)
448
            if not reverse:
449
                base = base_file
450
                other = other_file
451
            else:
452
                base = other_file
453
                other = base_file
454
            status = patch.diff3(new_file, filename, base, other)
455
            if status == 0:
456
                os.chmod(new_file, os.stat(filename).st_mode)
457
                rename(new_file, filename)
458
                return
459
            else:
460
                assert(status == 1)
461
                def get_lines(filename):
462
                    my_file = file(filename, "rb")
463
                    lines = my_file.readlines()
464
                    my_file.close()
465
                    return lines
466
                base_lines = get_lines(base)
467
                other_lines = get_lines(other)
468
                conflict_handler.merge_conflict(new_file, filename, base_lines, 
469
                                                other_lines)
470
        finally:
471
            rmtree(temp_dir)
493 by Martin Pool
- Merge aaron's merge command
472
473
474
def CreateDir():
475
    """Convenience function to create a directory.
476
477
    :return: A ReplaceContents that will create a directory
478
    :rtype: `ReplaceContents`
479
    """
480
    return ReplaceContents(None, dir_create)
481
482
def DeleteDir():
483
    """Convenience function to delete a directory.
484
485
    :return: A ReplaceContents that will delete a directory
486
    :rtype: `ReplaceContents`
487
    """
488
    return ReplaceContents(dir_create, None)
489
490
def CreateFile(contents):
491
    """Convenience fucntion to create a file.
492
    
493
    :param contents: The contents of the file to create 
494
    :type contents: str
495
    :return: A ReplaceContents that will create a file 
496
    :rtype: `ReplaceContents`
497
    """
498
    return ReplaceContents(None, FileCreate(contents))
499
500
def DeleteFile(contents):
501
    """Convenience fucntion to delete a file.
502
    
503
    :param contents: The contents of the file to delete
504
    :type contents: str
505
    :return: A ReplaceContents that will delete a file 
506
    :rtype: `ReplaceContents`
507
    """
508
    return ReplaceContents(FileCreate(contents), None)
509
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
510
def ReplaceFileContents(old_tree, new_tree, file_id):
493 by Martin Pool
- Merge aaron's merge command
511
    """Convenience fucntion to replace the contents of a file.
512
    
513
    :param old_contents: The contents of the file to replace 
514
    :type old_contents: str
515
    :param new_contents: The contents to replace the file with
516
    :type new_contents: str
517
    :return: A ReplaceContents that will replace the contents of a file a file 
518
    :rtype: `ReplaceContents`
519
    """
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
520
    return ReplaceContents(TreeFileCreate(old_tree, file_id), 
521
                           TreeFileCreate(new_tree, file_id))
493 by Martin Pool
- Merge aaron's merge command
522
523
def CreateSymlink(target):
524
    """Convenience fucntion to create a symlink.
525
    
526
    :param target: The path the link should point to
527
    :type target: str
528
    :return: A ReplaceContents that will delete a file 
529
    :rtype: `ReplaceContents`
530
    """
531
    return ReplaceContents(None, SymlinkCreate(target))
532
533
def DeleteSymlink(target):
534
    """Convenience fucntion to delete a symlink.
535
    
536
    :param target: The path the link should point to
537
    :type target: str
538
    :return: A ReplaceContents that will delete a file 
539
    :rtype: `ReplaceContents`
540
    """
541
    return ReplaceContents(SymlinkCreate(target), None)
542
543
def ChangeTarget(old_target, new_target):
544
    """Convenience fucntion to change the target of a symlink.
545
    
546
    :param old_target: The current link target
547
    :type old_target: str
548
    :param new_target: The new link target to use
549
    :type new_target: str
550
    :return: A ReplaceContents that will delete a file 
551
    :rtype: `ReplaceContents`
552
    """
553
    return ReplaceContents(SymlinkCreate(old_target), SymlinkCreate(new_target))
554
555
556
class InvalidEntry(Exception):
557
    """Raise when a ChangesetEntry is invalid in some way"""
558
    def __init__(self, entry, problem):
559
        """Constructor.
560
561
        :param entry: The invalid ChangesetEntry
562
        :type entry: `ChangesetEntry`
563
        :param problem: The problem with the entry
564
        :type problem: str
565
        """
566
        msg = "Changeset entry for %s (%s) is invalid.\n%s" % (entry.id, 
567
                                                               entry.path, 
568
                                                               problem)
569
        Exception.__init__(self, msg)
570
        self.entry = entry
571
572
573
class SourceRootHasName(InvalidEntry):
574
    """This changeset entry has a name other than "", but its parent is !NULL"""
575
    def __init__(self, entry, name):
576
        """Constructor.
577
578
        :param entry: The invalid ChangesetEntry
579
        :type entry: `ChangesetEntry`
580
        :param name: The name of the entry
581
        :type name: str
582
        """
583
        msg = 'Child of !NULL is named "%s", not "./.".' % name
584
        InvalidEntry.__init__(self, entry, msg)
585
586
class NullIDAssigned(InvalidEntry):
587
    """The id !NULL was assigned to a real entry"""
588
    def __init__(self, entry):
589
        """Constructor.
590
591
        :param entry: The invalid ChangesetEntry
592
        :type entry: `ChangesetEntry`
593
        """
594
        msg = '"!NULL" id assigned to a file "%s".' % entry.path
595
        InvalidEntry.__init__(self, entry, msg)
596
597
class ParentIDIsSelf(InvalidEntry):
598
    """An entry is marked as its own parent"""
599
    def __init__(self, entry):
600
        """Constructor.
601
602
        :param entry: The invalid ChangesetEntry
603
        :type entry: `ChangesetEntry`
604
        """
605
        msg = 'file %s has "%s" id for both self id and parent id.' % \
606
            (entry.path, entry.id)
607
        InvalidEntry.__init__(self, entry, msg)
608
609
class ChangesetEntry(object):
610
    """An entry the changeset"""
611
    def __init__(self, id, parent, path):
612
        """Constructor. Sets parent and name assuming it was not
613
        renamed/created/deleted.
614
        :param id: The id associated with the entry
615
        :param parent: The id of the parent of this entry (or !NULL if no
616
        parent)
617
        :param path: The file path relative to the tree root of this entry
618
        """
619
        self.id = id
620
        self.path = path 
621
        self.new_path = path
622
        self.parent = parent
623
        self.new_parent = parent
624
        self.contents_change = None
625
        self.metadata_change = None
626
        if parent == NULL_ID and path !='./.':
627
            raise SourceRootHasName(self, path)
628
        if self.id == NULL_ID:
629
            raise NullIDAssigned(self)
630
        if self.id  == self.parent:
631
            raise ParentIDIsSelf(self)
632
633
    def __str__(self):
634
        return "ChangesetEntry(%s)" % self.id
635
636
    def __get_dir(self):
637
        if self.path is None:
638
            return None
639
        return os.path.dirname(self.path)
640
641
    def __set_dir(self, dir):
642
        self.path = os.path.join(dir, os.path.basename(self.path))
643
644
    dir = property(__get_dir, __set_dir)
645
    
646
    def __get_name(self):
647
        if self.path is None:
648
            return None
649
        return os.path.basename(self.path)
650
651
    def __set_name(self, name):
652
        self.path = os.path.join(os.path.dirname(self.path), name)
653
654
    name = property(__get_name, __set_name)
655
656
    def __get_new_dir(self):
657
        if self.new_path is None:
658
            return None
659
        return os.path.dirname(self.new_path)
660
661
    def __set_new_dir(self, dir):
662
        self.new_path = os.path.join(dir, os.path.basename(self.new_path))
663
664
    new_dir = property(__get_new_dir, __set_new_dir)
665
666
    def __get_new_name(self):
667
        if self.new_path is None:
668
            return None
669
        return os.path.basename(self.new_path)
670
671
    def __set_new_name(self, name):
672
        self.new_path = os.path.join(os.path.dirname(self.new_path), name)
673
674
    new_name = property(__get_new_name, __set_new_name)
675
676
    def needs_rename(self):
677
        """Determines whether the entry requires renaming.
678
679
        :rtype: bool
680
        """
681
682
        return (self.parent != self.new_parent or self.name != self.new_name)
683
684
    def is_deletion(self, reverse):
685
        """Return true if applying the entry would delete a file/directory.
686
687
        :param reverse: if true, the changeset is being applied in reverse
688
        :rtype: bool
689
        """
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
690
        return self.is_creation(not reverse)
493 by Martin Pool
- Merge aaron's merge command
691
692
    def is_creation(self, reverse):
693
        """Return true if applying the entry would create a file/directory.
694
695
        :param reverse: if true, the changeset is being applied in reverse
696
        :rtype: bool
697
        """
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
698
        if self.contents_change is None:
699
            return False
700
        if reverse:
701
            return self.contents_change.is_deletion()
702
        else:
703
            return self.contents_change.is_creation()
493 by Martin Pool
- Merge aaron's merge command
704
705
    def is_creation_or_deletion(self):
706
        """Return true if applying the entry would create or delete a 
707
        file/directory.
708
709
        :rtype: bool
710
        """
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
711
        return self.is_creation(False) or self.is_deletion(False)
493 by Martin Pool
- Merge aaron's merge command
712
713
    def get_cset_path(self, mod=False):
714
        """Determine the path of the entry according to the changeset.
715
716
        :param changeset: The changeset to derive the path from
717
        :type changeset: `Changeset`
718
        :param mod: If true, generate the MOD path.  Otherwise, generate the \
719
        ORIG path.
720
        :return: the path of the entry, or None if it did not exist in the \
721
        requested tree.
722
        :rtype: str or NoneType
723
        """
724
        if mod:
725
            if self.new_parent == NULL_ID:
726
                return "./."
727
            elif self.new_parent is None:
728
                return None
729
            return self.new_path
730
        else:
731
            if self.parent == NULL_ID:
732
                return "./."
733
            elif self.parent is None:
734
                return None
735
            return self.path
736
974.1.47 by Aaron Bentley
Merged changes from the merge4 branch
737
    def summarize_name(self, reverse=False):
493 by Martin Pool
- Merge aaron's merge command
738
        """Produce a one-line summary of the filename.  Indicates renames as
739
        old => new, indicates creation as None => new, indicates deletion as
740
        old => None.
741
742
        :param changeset: The changeset to get paths from
743
        :type changeset: `Changeset`
744
        :param reverse: If true, reverse the names in the output
745
        :type reverse: bool
746
        :rtype: str
747
        """
748
        orig_path = self.get_cset_path(False)
749
        mod_path = self.get_cset_path(True)
750
        if orig_path is not None:
751
            orig_path = orig_path[2:]
752
        if mod_path is not None:
753
            mod_path = mod_path[2:]
754
        if orig_path == mod_path:
755
            return orig_path
756
        else:
757
            if not reverse:
758
                return "%s => %s" % (orig_path, mod_path)
759
            else:
760
                return "%s => %s" % (mod_path, orig_path)
761
762
763
    def get_new_path(self, id_map, changeset, reverse=False):
764
        """Determine the full pathname to rename to
765
766
        :param id_map: The map of ids to filenames for the tree
767
        :type id_map: Dictionary
768
        :param changeset: The changeset to get data from
769
        :type changeset: `Changeset`
770
        :param reverse: If true, we're applying the changeset in reverse
771
        :type reverse: bool
772
        :rtype: str
773
        """
974.1.47 by Aaron Bentley
Merged changes from the merge4 branch
774
        mutter("Finding new path for %s" % self.summarize_name())
493 by Martin Pool
- Merge aaron's merge command
775
        if reverse:
776
            parent = self.parent
777
            to_dir = self.dir
778
            from_dir = self.new_dir
779
            to_name = self.name
780
            from_name = self.new_name
781
        else:
782
            parent = self.new_parent
783
            to_dir = self.new_dir
784
            from_dir = self.dir
785
            to_name = self.new_name
786
            from_name = self.name
787
788
        if to_name is None:
789
            return None
790
791
        if parent == NULL_ID or parent is None:
792
            if to_name != '.':
793
                raise SourceRootHasName(self, to_name)
794
            else:
795
                return '.'
796
        if from_dir == to_dir:
797
            dir = os.path.dirname(id_map[self.id])
798
        else:
909.1.2 by aaron.bentley at utoronto
Fixed rename handling in merge
799
            mutter("path, new_path: %r %r" % (self.path, self.new_path))
493 by Martin Pool
- Merge aaron's merge command
800
            parent_entry = changeset.entries[parent]
801
            dir = parent_entry.get_new_path(id_map, changeset, reverse)
802
        if from_name == to_name:
803
            name = os.path.basename(id_map[self.id])
804
        else:
805
            name = to_name
806
            assert(from_name is None or from_name == os.path.basename(id_map[self.id]))
807
        return os.path.join(dir, name)
808
809
    def is_boring(self):
810
        """Determines whether the entry does nothing
811
        
812
        :return: True if the entry does no renames or content changes
813
        :rtype: bool
814
        """
815
        if self.contents_change is not None:
816
            return False
817
        elif self.metadata_change is not None:
818
            return False
819
        elif self.parent != self.new_parent:
820
            return False
821
        elif self.name != self.new_name:
822
            return False
823
        else:
824
            return True
825
826
    def apply(self, filename, conflict_handler, reverse=False):
827
        """Applies the file content and/or metadata changes.
828
829
        :param filename: the filename of the entry
830
        :type filename: str
831
        :param reverse: If true, apply the changes in reverse
832
        :type reverse: bool
833
        """
834
        if self.is_deletion(reverse) and self.metadata_change is not None:
835
            self.metadata_change.apply(filename, conflict_handler, reverse)
836
        if self.contents_change is not None:
837
            self.contents_change.apply(filename, conflict_handler, reverse)
838
        if not self.is_deletion(reverse) and self.metadata_change is not None:
839
            self.metadata_change.apply(filename, conflict_handler, reverse)
840
841
class IDPresent(Exception):
842
    def __init__(self, id):
843
        msg = "Cannot add entry because that id has already been used:\n%s" %\
844
            id
845
        Exception.__init__(self, msg)
846
        self.id = id
847
558 by Martin Pool
- All top-level classes inherit from object
848
class Changeset(object):
493 by Martin Pool
- Merge aaron's merge command
849
    """A set of changes to apply"""
850
    def __init__(self):
851
        self.entries = {}
852
853
    def add_entry(self, entry):
854
        """Add an entry to the list of entries"""
855
        if self.entries.has_key(entry.id):
856
            raise IDPresent(entry.id)
857
        self.entries[entry.id] = entry
858
859
def my_sort(sequence, key, reverse=False):
860
    """A sort function that supports supplying a key for comparison
861
    
862
    :param sequence: The sequence to sort
863
    :param key: A callable object that returns the values to be compared
864
    :param reverse: If true, sort in reverse order
865
    :type reverse: bool
866
    """
867
    def cmp_by_key(entry_a, entry_b):
868
        if reverse:
869
            tmp=entry_a
870
            entry_a = entry_b
871
            entry_b = tmp
872
        return cmp(key(entry_a), key(entry_b))
873
    sequence.sort(cmp_by_key)
874
875
def get_rename_entries(changeset, inventory, reverse):
876
    """Return a list of entries that will be renamed.  Entries are sorted from
877
    longest to shortest source path and from shortest to longest target path.
878
879
    :param changeset: The changeset to look in
880
    :type changeset: `Changeset`
881
    :param inventory: The source of current tree paths for the given ids
882
    :type inventory: Dictionary
883
    :param reverse: If true, the changeset is being applied in reverse
884
    :type reverse: bool
885
    :return: source entries and target entries as a tuple
886
    :rtype: (List, List)
887
    """
888
    source_entries = [x for x in changeset.entries.itervalues() 
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
889
                      if x.needs_rename() or x.is_creation_or_deletion()]
493 by Martin Pool
- Merge aaron's merge command
890
    # these are done from longest path to shortest, to avoid deleting a
891
    # parent before its children are deleted/renamed 
892
    def longest_to_shortest(entry):
893
        path = inventory.get(entry.id)
894
        if path is None:
895
            return 0
896
        else:
897
            return len(path)
898
    my_sort(source_entries, longest_to_shortest, reverse=True)
899
900
    target_entries = source_entries[:]
901
    # These are done from shortest to longest path, to avoid creating a
902
    # child before its parent has been created/renamed
903
    def shortest_to_longest(entry):
904
        path = entry.get_new_path(inventory, changeset, reverse)
905
        if path is None:
906
            return 0
907
        else:
908
            return len(path)
909
    my_sort(target_entries, shortest_to_longest)
910
    return (source_entries, target_entries)
911
850 by Martin Pool
- Merge merge updates from aaron
912
def rename_to_temp_delete(source_entries, inventory, dir, temp_dir, 
913
                          conflict_handler, reverse):
493 by Martin Pool
- Merge aaron's merge command
914
    """Delete and rename entries as appropriate.  Entries are renamed to temp
850 by Martin Pool
- Merge merge updates from aaron
915
    names.  A map of id -> temp name (or None, for deletions) is returned.
493 by Martin Pool
- Merge aaron's merge command
916
917
    :param source_entries: The entries to rename and delete
918
    :type source_entries: List of `ChangesetEntry`
919
    :param inventory: The map of id -> filename in the current tree
920
    :type inventory: Dictionary
921
    :param dir: The directory to apply changes to
922
    :type dir: str
923
    :param reverse: Apply changes in reverse
924
    :type reverse: bool
925
    :return: a mapping of id to temporary name
926
    :rtype: Dictionary
927
    """
928
    temp_name = {}
929
    for i in range(len(source_entries)):
930
        entry = source_entries[i]
931
        if entry.is_deletion(reverse):
932
            path = os.path.join(dir, inventory[entry.id])
933
            entry.apply(path, conflict_handler, reverse)
850 by Martin Pool
- Merge merge updates from aaron
934
            temp_name[entry.id] = None
493 by Martin Pool
- Merge aaron's merge command
935
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
936
        elif entry.needs_rename():
850 by Martin Pool
- Merge merge updates from aaron
937
            to_name = os.path.join(temp_dir, str(i))
493 by Martin Pool
- Merge aaron's merge command
938
            src_path = inventory.get(entry.id)
939
            if src_path is not None:
940
                src_path = os.path.join(dir, src_path)
941
                try:
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
942
                    rename(src_path, to_name)
493 by Martin Pool
- Merge aaron's merge command
943
                    temp_name[entry.id] = to_name
944
                except OSError, e:
945
                    if e.errno != errno.ENOENT:
946
                        raise
1185.12.38 by abentley
semi-broke merge
947
                    if conflict_handler.missing_for_rename(src_path, to_name) \
948
                        == "skip":
493 by Martin Pool
- Merge aaron's merge command
949
                        continue
950
951
    return temp_name
952
953
850 by Martin Pool
- Merge merge updates from aaron
954
def rename_to_new_create(changed_inventory, target_entries, inventory, 
955
                         changeset, dir, conflict_handler, reverse):
493 by Martin Pool
- Merge aaron's merge command
956
    """Rename entries with temp names to their final names, create new files.
957
850 by Martin Pool
- Merge merge updates from aaron
958
    :param changed_inventory: A mapping of id to temporary name
959
    :type changed_inventory: Dictionary
493 by Martin Pool
- Merge aaron's merge command
960
    :param target_entries: The entries to apply changes to
961
    :type target_entries: List of `ChangesetEntry`
962
    :param changeset: The changeset to apply
963
    :type changeset: `Changeset`
964
    :param dir: The directory to apply changes to
965
    :type dir: str
966
    :param reverse: If true, apply changes in reverse
967
    :type reverse: bool
968
    """
969
    for entry in target_entries:
1185.12.38 by abentley
semi-broke merge
970
        print entry.contents_change
850 by Martin Pool
- Merge merge updates from aaron
971
        new_tree_path = entry.get_new_path(inventory, changeset, reverse)
972
        if new_tree_path is None:
493 by Martin Pool
- Merge aaron's merge command
973
            continue
850 by Martin Pool
- Merge merge updates from aaron
974
        new_path = os.path.join(dir, new_tree_path)
975
        old_path = changed_inventory.get(entry.id)
1092.2.24 by Robert Collins
merge from martins newformat branch - brings in transport abstraction
976
        if bzrlib.osutils.lexists(new_path):
493 by Martin Pool
- Merge aaron's merge command
977
            if conflict_handler.target_exists(entry, new_path, old_path) == \
978
                "skip":
979
                continue
980
        if entry.is_creation(reverse):
981
            entry.apply(new_path, conflict_handler, reverse)
850 by Martin Pool
- Merge merge updates from aaron
982
            changed_inventory[entry.id] = new_tree_path
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
983
        elif entry.needs_rename():
493 by Martin Pool
- Merge aaron's merge command
984
            if old_path is None:
985
                continue
986
            try:
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
987
                rename(old_path, new_path)
850 by Martin Pool
- Merge merge updates from aaron
988
                changed_inventory[entry.id] = new_tree_path
493 by Martin Pool
- Merge aaron's merge command
989
            except OSError, e:
990
                raise Exception ("%s is missing" % new_path)
991
992
class TargetExists(Exception):
993
    def __init__(self, entry, target):
994
        msg = "The path %s already exists" % target
995
        Exception.__init__(self, msg)
996
        self.entry = entry
997
        self.target = target
998
999
class RenameConflict(Exception):
1000
    def __init__(self, id, this_name, base_name, other_name):
1001
        msg = """Trees all have different names for a file
1002
 this: %s
1003
 base: %s
1004
other: %s
1005
   id: %s""" % (this_name, base_name, other_name, id)
1006
        Exception.__init__(self, msg)
1007
        self.this_name = this_name
1008
        self.base_name = base_name
1009
        self_other_name = other_name
1010
1011
class MoveConflict(Exception):
1012
    def __init__(self, id, this_parent, base_parent, other_parent):
1013
        msg = """The file is in different directories in every tree
1014
 this: %s
1015
 base: %s
1016
other: %s
1017
   id: %s""" % (this_parent, base_parent, other_parent, id)
1018
        Exception.__init__(self, msg)
1019
        self.this_parent = this_parent
1020
        self.base_parent = base_parent
1021
        self_other_parent = other_parent
1022
1023
class MergeConflict(Exception):
1024
    def __init__(self, this_path):
1025
        Exception.__init__(self, "Conflict applying changes to %s" % this_path)
1026
        self.this_path = this_path
1027
1028
class WrongOldContents(Exception):
1029
    def __init__(self, filename):
1030
        msg = "Contents mismatch deleting %s" % filename
1031
        self.filename = filename
1032
        Exception.__init__(self, msg)
1033
1434 by Robert Collins
merge Gustavos executable2 patch
1034
class WrongOldExecFlag(Exception):
1035
    def __init__(self, filename, old_exec_flag, new_exec_flag):
1036
        msg = "Executable flag missmatch on %s:\n" \
1037
        "Expected %s, got %s." % (filename, old_exec_flag, new_exec_flag)
493 by Martin Pool
- Merge aaron's merge command
1038
        self.filename = filename
1039
        Exception.__init__(self, msg)
1040
1041
class RemoveContentsConflict(Exception):
1042
    def __init__(self, filename):
1043
        msg = "Conflict deleting %s, which has different contents in BASE"\
1044
            " and THIS" % filename
1045
        self.filename = filename
1046
        Exception.__init__(self, msg)
1047
1048
class DeletingNonEmptyDirectory(Exception):
1049
    def __init__(self, filename):
1050
        msg = "Trying to remove dir %s while it still had files" % filename
1051
        self.filename = filename
1052
        Exception.__init__(self, msg)
1053
1054
1055
class PatchTargetMissing(Exception):
1056
    def __init__(self, filename):
1057
        msg = "Attempt to patch %s, which does not exist" % filename
1058
        Exception.__init__(self, msg)
1059
        self.filename = filename
1060
1434 by Robert Collins
merge Gustavos executable2 patch
1061
class MissingForSetExec(Exception):
493 by Martin Pool
- Merge aaron's merge command
1062
    def __init__(self, filename):
1063
        msg = "Attempt to change permissions on  %s, which does not exist" %\
1064
            filename
1065
        Exception.__init__(self, msg)
1066
        self.filename = filename
1067
1068
class MissingForRm(Exception):
1069
    def __init__(self, filename):
1070
        msg = "Attempt to remove missing path %s" % filename
1071
        Exception.__init__(self, msg)
1072
        self.filename = filename
1073
1074
1075
class MissingForRename(Exception):
1185.12.38 by abentley
semi-broke merge
1076
    def __init__(self, filename, to_path):
1077
        msg = "Attempt to move missing path %s to %s" % (filename, to_path)
493 by Martin Pool
- Merge aaron's merge command
1078
        Exception.__init__(self, msg)
1079
        self.filename = filename
1080
850 by Martin Pool
- Merge merge updates from aaron
1081
class NewContentsConflict(Exception):
1082
    def __init__(self, filename):
1083
        msg = "Conflicting contents for new file %s" % (filename)
1084
        Exception.__init__(self, msg)
1085
1185.12.33 by Aaron Bentley
Fixed symlink reverting
1086
class ThreewayContentsConflict(Exception):
1087
    def __init__(self, filename):
1088
        msg = "Conflicting contents for file %s" % (filename)
1089
        Exception.__init__(self, msg)
1090
850 by Martin Pool
- Merge merge updates from aaron
1091
1092
class MissingForMerge(Exception):
1093
    def __init__(self, filename):
1094
        msg = "The file %s was modified, but does not exist in this tree"\
1095
            % (filename)
1096
        Exception.__init__(self, msg)
1097
1098
558 by Martin Pool
- All top-level classes inherit from object
1099
class ExceptionConflictHandler(object):
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
1100
    """Default handler for merge exceptions.
1101
1102
    This throws an error on any kind of conflict.  Conflict handlers can
1103
    descend from this class if they have a better way to handle some or
1104
    all types of conflict.
1105
    """
493 by Martin Pool
- Merge aaron's merge command
1106
    def missing_parent(self, pathname):
1107
        parent = os.path.dirname(pathname)
1108
        raise Exception("Parent directory missing for %s" % pathname)
1109
1110
    def dir_exists(self, pathname):
1111
        raise Exception("Directory already exists for %s" % pathname)
1112
1113
    def failed_hunks(self, pathname):
1114
        raise Exception("Failed to apply some hunks for %s" % pathname)
1115
1116
    def target_exists(self, entry, target, old_path):
1117
        raise TargetExists(entry, target)
1118
1119
    def rename_conflict(self, id, this_name, base_name, other_name):
1120
        raise RenameConflict(id, this_name, base_name, other_name)
1121
974.1.20 by Aaron Bentley
Eliminated ThreeWayInventory
1122
    def move_conflict(self, id, this_dir, base_dir, other_dir):
493 by Martin Pool
- Merge aaron's merge command
1123
        raise MoveConflict(id, this_dir, base_dir, other_dir)
1124
974.1.23 by Aaron Bentley
Avoided unnecessary temp files
1125
    def merge_conflict(self, new_file, this_path, base_lines, other_lines):
493 by Martin Pool
- Merge aaron's merge command
1126
        os.unlink(new_file)
1127
        raise MergeConflict(this_path)
1128
1129
    def wrong_old_contents(self, filename, expected_contents):
1130
        raise WrongOldContents(filename)
1131
1132
    def rem_contents_conflict(self, filename, this_contents, base_contents):
1133
        raise RemoveContentsConflict(filename)
1134
1434 by Robert Collins
merge Gustavos executable2 patch
1135
    def wrong_old_exec_flag(self, filename, old_exec_flag, new_exec_flag):
1136
        raise WrongOldExecFlag(filename, old_exec_flag, new_exec_flag)
493 by Martin Pool
- Merge aaron's merge command
1137
1138
    def rmdir_non_empty(self, filename):
1139
        raise DeletingNonEmptyDirectory(filename)
1140
1141
    def link_name_exists(self, filename):
1142
        raise TargetExists(filename)
1143
1144
    def patch_target_missing(self, filename, contents):
1145
        raise PatchTargetMissing(filename)
1146
1434 by Robert Collins
merge Gustavos executable2 patch
1147
    def missing_for_exec_flag(self, filename):
1148
        raise MissingForExecFlag(filename)
493 by Martin Pool
- Merge aaron's merge command
1149
1150
    def missing_for_rm(self, filename, change):
1151
        raise MissingForRm(filename)
1152
1185.12.38 by abentley
semi-broke merge
1153
    def missing_for_rename(self, filename, to_path):
1154
        raise MissingForRename(filename, to_path)
493 by Martin Pool
- Merge aaron's merge command
1155
974.1.20 by Aaron Bentley
Eliminated ThreeWayInventory
1156
    def missing_for_merge(self, file_id, other_path):
1157
        raise MissingForMerge(other_path)
850 by Martin Pool
- Merge merge updates from aaron
1158
1159
    def new_contents_conflict(self, filename, other_contents):
1160
        raise NewContentsConflict(filename)
1161
1185.12.33 by Aaron Bentley
Fixed symlink reverting
1162
    def threeway_contents_conflict(self, filename, this_contents,
1163
                                   base_contents, other_contents):
1164
        raise ThreewayContentsConflict(filename)
1165
1114 by Martin Pool
- fix bad method declaration
1166
    def finalize(self):
622 by Martin Pool
Updated merge patch from Aaron
1167
        pass
1168
493 by Martin Pool
- Merge aaron's merge command
1169
def apply_changeset(changeset, inventory, dir, conflict_handler=None, 
1170
                    reverse=False):
1171
    """Apply a changeset to a directory.
1172
1173
    :param changeset: The changes to perform
1174
    :type changeset: `Changeset`
1175
    :param inventory: The mapping of id to filename for the directory
1176
    :type inventory: Dictionary
1177
    :param dir: The path of the directory to apply the changes to
1178
    :type dir: str
1179
    :param reverse: If true, apply the changes in reverse
1180
    :type reverse: bool
1181
    :return: The mapping of the changed entries
1182
    :rtype: Dictionary
1183
    """
1184
    if conflict_handler is None:
974.1.83 by Aaron Bentley
Removed unused dir parameter from ExceptionConflictHandler
1185
        conflict_handler = ExceptionConflictHandler()
850 by Martin Pool
- Merge merge updates from aaron
1186
    temp_dir = os.path.join(dir, "bzr-tree-change")
1187
    try:
1188
        os.mkdir(temp_dir)
1189
    except OSError, e:
1190
        if e.errno == errno.EEXIST:
1191
            try:
1192
                os.rmdir(temp_dir)
1193
            except OSError, e:
1194
                if e.errno == errno.ENOTEMPTY:
1195
                    raise OldFailedTreeOp()
1196
            os.mkdir(temp_dir)
1197
        else:
1198
            raise
493 by Martin Pool
- Merge aaron's merge command
1199
    
1200
    #apply changes that don't affect filenames
1201
    for entry in changeset.entries.itervalues():
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
1202
        if not entry.is_creation_or_deletion() and not entry.is_boring():
493 by Martin Pool
- Merge aaron's merge command
1203
            path = os.path.join(dir, inventory[entry.id])
1204
            entry.apply(path, conflict_handler, reverse)
1205
1206
    # Apply renames in stages, to minimize conflicts:
1207
    # Only files whose name or parent change are interesting, because their
1208
    # target name may exist in the source tree.  If a directory's name changes,
1209
    # that doesn't make its children interesting.
1210
    (source_entries, target_entries) = get_rename_entries(changeset, inventory,
1211
                                                          reverse)
1212
850 by Martin Pool
- Merge merge updates from aaron
1213
    changed_inventory = rename_to_temp_delete(source_entries, inventory, dir,
1214
                                              temp_dir, conflict_handler,
1215
                                              reverse)
493 by Martin Pool
- Merge aaron's merge command
1216
850 by Martin Pool
- Merge merge updates from aaron
1217
    rename_to_new_create(changed_inventory, target_entries, inventory,
1218
                         changeset, dir, conflict_handler, reverse)
493 by Martin Pool
- Merge aaron's merge command
1219
    os.rmdir(temp_dir)
850 by Martin Pool
- Merge merge updates from aaron
1220
    return changed_inventory
493 by Martin Pool
- Merge aaron's merge command
1221
1222
1223
def apply_changeset_tree(cset, tree, reverse=False):
1224
    r_inventory = {}
1225
    for entry in tree.source_inventory().itervalues():
1226
        inventory[entry.id] = entry.path
1227
    new_inventory = apply_changeset(cset, r_inventory, tree.root,
1228
                                    reverse=reverse)
1229
    new_entries, remove_entries = \
1230
        get_inventory_change(inventory, new_inventory, cset, reverse)
1231
    tree.update_source_inventory(new_entries, remove_entries)
1232
1233
1234
def get_inventory_change(inventory, new_inventory, cset, reverse=False):
1235
    new_entries = {}
1236
    remove_entries = []
1237
    for entry in cset.entries.itervalues():
1238
        if entry.needs_rename():
850 by Martin Pool
- Merge merge updates from aaron
1239
            new_path = entry.get_new_path(inventory, cset)
1240
            if new_path is None:
1241
                remove_entries.append(entry.id)
493 by Martin Pool
- Merge aaron's merge command
1242
            else:
850 by Martin Pool
- Merge merge updates from aaron
1243
                new_entries[new_path] = entry.id
493 by Martin Pool
- Merge aaron's merge command
1244
    return new_entries, remove_entries
1245
1246
1247
def print_changeset(cset):
1248
    """Print all non-boring changeset entries
1249
    
1250
    :param cset: The changeset to print
1251
    :type cset: `Changeset`
1252
    """
1253
    for entry in cset.entries.itervalues():
1254
        if entry.is_boring():
1255
            continue
1256
        print entry.id
1257
        print entry.summarize_name(cset)
1258
1259
class CompositionFailure(Exception):
1260
    def __init__(self, old_entry, new_entry, problem):
1261
        msg = "Unable to conpose entries.\n %s" % problem
1262
        Exception.__init__(self, msg)
1263
1264
class IDMismatch(CompositionFailure):
1265
    def __init__(self, old_entry, new_entry):
1266
        problem = "Attempt to compose entries with different ids: %s and %s" %\
1267
            (old_entry.id, new_entry.id)
1268
        CompositionFailure.__init__(self, old_entry, new_entry, problem)
1269
1270
def compose_changesets(old_cset, new_cset):
1271
    """Combine two changesets into one.  This works well for exact patching.
1272
    Otherwise, not so well.
1273
1274
    :param old_cset: The first changeset that would be applied
1275
    :type old_cset: `Changeset`
1276
    :param new_cset: The second changeset that would be applied
1277
    :type new_cset: `Changeset`
1278
    :return: A changeset that combines the changes in both changesets
1279
    :rtype: `Changeset`
1280
    """
1281
    composed = Changeset()
1282
    for old_entry in old_cset.entries.itervalues():
1283
        new_entry = new_cset.entries.get(old_entry.id)
1284
        if new_entry is None:
1285
            composed.add_entry(old_entry)
1286
        else:
1287
            composed_entry = compose_entries(old_entry, new_entry)
1288
            if composed_entry.parent is not None or\
1289
                composed_entry.new_parent is not None:
1290
                composed.add_entry(composed_entry)
1291
    for new_entry in new_cset.entries.itervalues():
1292
        if not old_cset.entries.has_key(new_entry.id):
1293
            composed.add_entry(new_entry)
1294
    return composed
1295
1296
def compose_entries(old_entry, new_entry):
1297
    """Combine two entries into one.
1298
1299
    :param old_entry: The first entry that would be applied
1300
    :type old_entry: ChangesetEntry
1301
    :param old_entry: The second entry that would be applied
1302
    :type old_entry: ChangesetEntry
1303
    :return: A changeset entry combining both entries
1304
    :rtype: `ChangesetEntry`
1305
    """
1306
    if old_entry.id != new_entry.id:
1307
        raise IDMismatch(old_entry, new_entry)
1308
    output = ChangesetEntry(old_entry.id, old_entry.parent, old_entry.path)
1309
1310
    if (old_entry.parent != old_entry.new_parent or 
1311
        new_entry.parent != new_entry.new_parent):
1312
        output.new_parent = new_entry.new_parent
1313
1314
    if (old_entry.path != old_entry.new_path or 
1315
        new_entry.path != new_entry.new_path):
1316
        output.new_path = new_entry.new_path
1317
1318
    output.contents_change = compose_contents(old_entry, new_entry)
1319
    output.metadata_change = compose_metadata(old_entry, new_entry)
1320
    return output
1321
1322
def compose_contents(old_entry, new_entry):
1323
    """Combine the contents of two changeset entries.  Entries are combined
1324
    intelligently where possible, but the fallback behavior returns an 
1325
    ApplySequence.
1326
1327
    :param old_entry: The first entry that would be applied
1328
    :type old_entry: `ChangesetEntry`
1329
    :param new_entry: The second entry that would be applied
1330
    :type new_entry: `ChangesetEntry`
1331
    :return: A combined contents change
1332
    :rtype: anything supporting the apply(reverse=False) method
1333
    """
1334
    old_contents = old_entry.contents_change
1335
    new_contents = new_entry.contents_change
1336
    if old_entry.contents_change is None:
1337
        return new_entry.contents_change
1338
    elif new_entry.contents_change is None:
1339
        return old_entry.contents_change
1340
    elif isinstance(old_contents, ReplaceContents) and \
1341
        isinstance(new_contents, ReplaceContents):
1342
        if old_contents.old_contents == new_contents.new_contents:
1343
            return None
1344
        else:
1345
            return ReplaceContents(old_contents.old_contents,
1346
                                   new_contents.new_contents)
1347
    elif isinstance(old_contents, ApplySequence):
1348
        output = ApplySequence(old_contents.changes)
1349
        if isinstance(new_contents, ApplySequence):
1350
            output.changes.extend(new_contents.changes)
1351
        else:
1352
            output.changes.append(new_contents)
1353
        return output
1354
    elif isinstance(new_contents, ApplySequence):
1355
        output = ApplySequence((old_contents.changes,))
1356
        output.extend(new_contents.changes)
1357
        return output
1358
    else:
1359
        return ApplySequence((old_contents, new_contents))
1360
1361
def compose_metadata(old_entry, new_entry):
1362
    old_meta = old_entry.metadata_change
1363
    new_meta = new_entry.metadata_change
1364
    if old_meta is None:
1365
        return new_meta
1366
    elif new_meta is None:
1367
        return old_meta
1434 by Robert Collins
merge Gustavos executable2 patch
1368
    elif (isinstance(old_meta, ChangeExecFlag) and
1369
          isinstance(new_meta, ChangeExecFlag)):
1370
        return ChangeExecFlag(old_meta.old_exec_flag, new_meta.new_exec_flag)
493 by Martin Pool
- Merge aaron's merge command
1371
    else:
1372
        return ApplySequence(old_meta, new_meta)
1373
1374
1375
def changeset_is_null(changeset):
1376
    for entry in changeset.entries.itervalues():
1377
        if not entry.is_boring():
1378
            return False
1379
    return True
1380
1185.12.30 by Aaron Bentley
Moved all fooCreate into get_contents
1381
class UnsupportedFiletype(Exception):
1382
    def __init__(self, kind, full_path):
1383
        msg = "The file \"%s\" is a %s, which is not a supported filetype." \
1384
            % (full_path, kind)
493 by Martin Pool
- Merge aaron's merge command
1385
        Exception.__init__(self, msg)
1386
        self.full_path = full_path
1185.12.30 by Aaron Bentley
Moved all fooCreate into get_contents
1387
        self.kind = kind
493 by Martin Pool
- Merge aaron's merge command
1388
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1389
def generate_changeset(tree_a, tree_b, interesting_ids=None):
1390
    return ChangesetGenerator(tree_a, tree_b, interesting_ids)()
493 by Martin Pool
- Merge aaron's merge command
1391
1448 by Robert Collins
revert symlinks correctly
1392
493 by Martin Pool
- Merge aaron's merge command
1393
class ChangesetGenerator(object):
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1394
    def __init__(self, tree_a, tree_b, interesting_ids=None):
493 by Martin Pool
- Merge aaron's merge command
1395
        object.__init__(self)
1396
        self.tree_a = tree_a
1397
        self.tree_b = tree_b
974.1.17 by Aaron Bentley
Switched to using a set of interesting file_ids instead of SourceFile attribute
1398
        self._interesting_ids = interesting_ids
493 by Martin Pool
- Merge aaron's merge command
1399
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1400
    def iter_both_tree_ids(self):
1401
        for file_id in self.tree_a:
1402
            yield file_id
1403
        for file_id in self.tree_b:
1404
            if file_id not in self.tree_a:
1405
                yield file_id
493 by Martin Pool
- Merge aaron's merge command
1406
1407
    def __call__(self):
1408
        cset = Changeset()
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1409
        for file_id in self.iter_both_tree_ids():
1410
            cs_entry = self.make_entry(file_id)
493 by Martin Pool
- Merge aaron's merge command
1411
            if cs_entry is not None and not cs_entry.is_boring():
1412
                cset.add_entry(cs_entry)
1413
1414
        for entry in list(cset.entries.itervalues()):
1415
            if entry.parent != entry.new_parent:
1416
                if not cset.entries.has_key(entry.parent) and\
1417
                    entry.parent != NULL_ID and entry.parent is not None:
1418
                    parent_entry = self.make_boring_entry(entry.parent)
1419
                    cset.add_entry(parent_entry)
1420
                if not cset.entries.has_key(entry.new_parent) and\
1421
                    entry.new_parent != NULL_ID and \
1422
                    entry.new_parent is not None:
1423
                    parent_entry = self.make_boring_entry(entry.new_parent)
1424
                    cset.add_entry(parent_entry)
1425
        return cset
1426
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1427
    def iter_inventory(self, tree):
1428
        for file_id in tree:
1429
            yield self.get_entry(file_id, tree)
1430
1431
    def get_entry(self, file_id, tree):
1185.12.38 by abentley
semi-broke merge
1432
        if not tree.has_id(file_id, allow_root=True):
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1433
            return None
1185.12.38 by abentley
semi-broke merge
1434
        return tree.inventory[file_id]
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1435
1436
    def get_entry_parent(self, entry):
1069 by Martin Pool
- merge merge improvements from aaron
1437
        if entry is None:
1438
            return None
974.1.18 by Aaron Bentley
Stopped using SourceFile in inventory
1439
        return entry.parent_id
1440
1441
    def get_path(self, file_id, tree):
1185.12.38 by abentley
semi-broke merge
1442
        if not tree.has_id(file_id, allow_root=True):
974.1.18 by Aaron Bentley
Stopped using SourceFile in inventory
1443
            return None
1444
        path = tree.id2path(file_id)
1445
        if path == '':
1446
            return './.'
1447
        else:
1448
            return path
1449
1450
    def make_basic_entry(self, file_id, only_interesting):
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1451
        entry_a = self.get_entry(file_id, self.tree_a)
1452
        entry_b = self.get_entry(file_id, self.tree_b)
493 by Martin Pool
- Merge aaron's merge command
1453
        if only_interesting and not self.is_interesting(entry_a, entry_b):
974.1.12 by aaron.bentley at utoronto
Switched from text-id to hashcache for merge optimization
1454
            return None
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1455
        parent = self.get_entry_parent(entry_a)
974.1.18 by Aaron Bentley
Stopped using SourceFile in inventory
1456
        path = self.get_path(file_id, self.tree_a)
1457
        cs_entry = ChangesetEntry(file_id, parent, path)
974.1.19 by Aaron Bentley
Removed MergeTree.inventory
1458
        new_parent = self.get_entry_parent(entry_b)
1069 by Martin Pool
- merge merge improvements from aaron
1459
974.1.18 by Aaron Bentley
Stopped using SourceFile in inventory
1460
        new_path = self.get_path(file_id, self.tree_b)
493 by Martin Pool
- Merge aaron's merge command
1461
1462
        cs_entry.new_path = new_path
1463
        cs_entry.new_parent = new_parent
974.1.12 by aaron.bentley at utoronto
Switched from text-id to hashcache for merge optimization
1464
        return cs_entry
493 by Martin Pool
- Merge aaron's merge command
1465
1466
    def is_interesting(self, entry_a, entry_b):
974.1.17 by Aaron Bentley
Switched to using a set of interesting file_ids instead of SourceFile attribute
1467
        if self._interesting_ids is None:
1468
            return True
493 by Martin Pool
- Merge aaron's merge command
1469
        if entry_a is not None:
974.1.18 by Aaron Bentley
Stopped using SourceFile in inventory
1470
            file_id = entry_a.file_id
974.1.17 by Aaron Bentley
Switched to using a set of interesting file_ids instead of SourceFile attribute
1471
        elif entry_b is not None:
974.1.18 by Aaron Bentley
Stopped using SourceFile in inventory
1472
            file_id = entry_b.file_id
974.1.17 by Aaron Bentley
Switched to using a set of interesting file_ids instead of SourceFile attribute
1473
        else:
1474
            return False
1475
        return file_id in self._interesting_ids
493 by Martin Pool
- Merge aaron's merge command
1476
1477
    def make_boring_entry(self, id):
974.1.12 by aaron.bentley at utoronto
Switched from text-id to hashcache for merge optimization
1478
        cs_entry = self.make_basic_entry(id, only_interesting=False)
493 by Martin Pool
- Merge aaron's merge command
1479
        if cs_entry.is_creation_or_deletion():
1480
            return self.make_entry(id, only_interesting=False)
1481
        else:
1482
            return cs_entry
1483
        
1484
1485
    def make_entry(self, id, only_interesting=True):
974.1.12 by aaron.bentley at utoronto
Switched from text-id to hashcache for merge optimization
1486
        cs_entry = self.make_basic_entry(id, only_interesting)
493 by Martin Pool
- Merge aaron's merge command
1487
1488
        if cs_entry is None:
1489
            return None
1398 by Robert Collins
integrate in Gustavos x-bit patch
1490
1185.12.28 by Aaron Bentley
Removed use of readonly path for executability test
1491
        cs_entry.metadata_change = self.make_exec_flag_change(id)
1398 by Robert Collins
integrate in Gustavos x-bit patch
1492
974.1.12 by aaron.bentley at utoronto
Switched from text-id to hashcache for merge optimization
1493
        if id in self.tree_a and id in self.tree_b:
1494
            a_sha1 = self.tree_a.get_file_sha1(id)
1495
            b_sha1 = self.tree_b.get_file_sha1(id)
1496
            if None not in (a_sha1, b_sha1) and a_sha1 == b_sha1:
1497
                return cs_entry
1498
1185.12.29 by Aaron Bentley
Removed readonly_path from changeset generation
1499
        cs_entry.contents_change = self.make_contents_change(id)
493 by Martin Pool
- Merge aaron's merge command
1500
        return cs_entry
1501
1185.12.28 by Aaron Bentley
Removed use of readonly path for executability test
1502
    def make_exec_flag_change(self, file_id):
1434 by Robert Collins
merge Gustavos executable2 patch
1503
        exec_flag_a = exec_flag_b = None
1185.12.28 by Aaron Bentley
Removed use of readonly path for executability test
1504
        if file_id in self.tree_a and self.tree_a.kind(file_id) == "file":
1505
            exec_flag_a = self.tree_a.is_executable(file_id)
1506
1507
        if file_id in self.tree_b and self.tree_b.kind(file_id) == "file":
1508
            exec_flag_b = self.tree_b.is_executable(file_id)
1509
1434 by Robert Collins
merge Gustavos executable2 patch
1510
        if exec_flag_a == exec_flag_b:
493 by Martin Pool
- Merge aaron's merge command
1511
            return None
1434 by Robert Collins
merge Gustavos executable2 patch
1512
        return ChangeExecFlag(exec_flag_a, exec_flag_b)
493 by Martin Pool
- Merge aaron's merge command
1513
1185.12.29 by Aaron Bentley
Removed readonly_path from changeset generation
1514
    def make_contents_change(self, file_id):
1185.12.30 by Aaron Bentley
Moved all fooCreate into get_contents
1515
        a_contents = get_contents(self.tree_a, file_id)
1516
        b_contents = get_contents(self.tree_b, file_id)
493 by Martin Pool
- Merge aaron's merge command
1517
        if a_contents == b_contents:
1518
            return None
1519
        return ReplaceContents(a_contents, b_contents)
1520
1185.12.30 by Aaron Bentley
Moved all fooCreate into get_contents
1521
1522
def get_contents(tree, file_id):
1523
    """Return the appropriate contents to create a copy of file_id from tree"""
1524
    if file_id not in tree:
1525
        return None
1526
    kind = tree.kind(file_id)
1527
    if kind == "file":
1185.12.31 by Aaron Bentley
Replaced FileCreate with TreeFileCreate, fixed revert with empty parents
1528
        return TreeFileCreate(tree, file_id)
1185.12.30 by Aaron Bentley
Moved all fooCreate into get_contents
1529
    elif kind in ("directory", "root_directory"):
1530
        return dir_create
1531
    elif kind == "symlink":
1532
        return SymlinkCreate(tree.get_symlink_target(file_id))
1533
    else:
1534
        raise UnsupportedFiletype(kind, tree.id2path(file_id))
493 by Martin Pool
- Merge aaron's merge command
1535
1536
1537
def full_path(entry, tree):
1538
    return os.path.join(tree.root, entry.path)
1539
1540
def new_delete_entry(entry, tree, inventory, delete):
1541
    if entry.path == "":
1542
        parent = NULL_ID
1543
    else:
1544
        parent = inventory[dirname(entry.path)].id
1545
    cs_entry = ChangesetEntry(parent, entry.path)
1546
    if delete:
1547
        cs_entry.new_path = None
1548
        cs_entry.new_parent = None
1549
    else:
1550
        cs_entry.path = None
1551
        cs_entry.parent = None
1552
    full_path = full_path(entry, tree)
1553
    status = os.lstat(full_path)
1554
    if stat.S_ISDIR(file_stat.st_mode):
1555
        action = dir_create
1556
    
1557
1558
1559
        
959 by Martin Pool
doc
1560
# XXX: Can't we unify this with the regular inventory object
558 by Martin Pool
- All top-level classes inherit from object
1561
class Inventory(object):
493 by Martin Pool
- Merge aaron's merge command
1562
    def __init__(self, inventory):
1563
        self.inventory = inventory
1564
        self.rinventory = None
1565
1566
    def get_rinventory(self):
1567
        if self.rinventory is None:
1568
            self.rinventory  = invert_dict(self.inventory)
1569
        return self.rinventory
1570
1571
    def get_path(self, id):
1572
        return self.inventory.get(id)
1573
1574
    def get_name(self, id):
850 by Martin Pool
- Merge merge updates from aaron
1575
        path = self.get_path(id)
1576
        if path is None:
1577
            return None
1578
        else:
1579
            return os.path.basename(path)
493 by Martin Pool
- Merge aaron's merge command
1580
1581
    def get_dir(self, id):
1582
        path = self.get_path(id)
1583
        if path == "":
1584
            return None
850 by Martin Pool
- Merge merge updates from aaron
1585
        if path is None:
1586
            return None
493 by Martin Pool
- Merge aaron's merge command
1587
        return os.path.dirname(path)
1588
1589
    def get_parent(self, id):
850 by Martin Pool
- Merge merge updates from aaron
1590
        if self.get_path(id) is None:
1591
            return None
493 by Martin Pool
- Merge aaron's merge command
1592
        directory = self.get_dir(id)
1593
        if directory == '.':
1594
            directory = './.'
1595
        if directory is None:
1596
            return NULL_ID
1597
        return self.get_rinventory().get(directory)