xref: /aosp_15_r20/external/libvpx/tools/intersect-diffs.py (revision fb1b10ab9aebc7c7068eedab379b749d7e3900be)
1*fb1b10abSAndroid Build Coastguard Worker#!/usr/bin/env python3
2*fb1b10abSAndroid Build Coastguard Worker##  Copyright (c) 2012 The WebM project authors. All Rights Reserved.
3*fb1b10abSAndroid Build Coastguard Worker##
4*fb1b10abSAndroid Build Coastguard Worker##  Use of this source code is governed by a BSD-style license
5*fb1b10abSAndroid Build Coastguard Worker##  that can be found in the LICENSE file in the root of the source
6*fb1b10abSAndroid Build Coastguard Worker##  tree. An additional intellectual property rights grant can be found
7*fb1b10abSAndroid Build Coastguard Worker##  in the file PATENTS.  All contributing project authors may
8*fb1b10abSAndroid Build Coastguard Worker##  be found in the AUTHORS file in the root of the source tree.
9*fb1b10abSAndroid Build Coastguard Worker##
10*fb1b10abSAndroid Build Coastguard Worker"""Calculates the "intersection" of two unified diffs.
11*fb1b10abSAndroid Build Coastguard Worker
12*fb1b10abSAndroid Build Coastguard WorkerGiven two diffs, A and B, it finds all hunks in B that had non-context lines
13*fb1b10abSAndroid Build Coastguard Workerin A and prints them to stdout. This is useful to determine the hunks in B that
14*fb1b10abSAndroid Build Coastguard Workerare relevant to A. The resulting file can be applied with patch(1) on top of A.
15*fb1b10abSAndroid Build Coastguard Worker"""
16*fb1b10abSAndroid Build Coastguard Worker
17*fb1b10abSAndroid Build Coastguard Worker__author__ = "[email protected]"
18*fb1b10abSAndroid Build Coastguard Worker
19*fb1b10abSAndroid Build Coastguard Workerimport sys
20*fb1b10abSAndroid Build Coastguard Worker
21*fb1b10abSAndroid Build Coastguard Workerimport diff
22*fb1b10abSAndroid Build Coastguard Worker
23*fb1b10abSAndroid Build Coastguard Worker
24*fb1b10abSAndroid Build Coastguard Workerdef FormatDiffHunks(hunks):
25*fb1b10abSAndroid Build Coastguard Worker    """Re-serialize a list of DiffHunks."""
26*fb1b10abSAndroid Build Coastguard Worker    r = []
27*fb1b10abSAndroid Build Coastguard Worker    last_header = None
28*fb1b10abSAndroid Build Coastguard Worker    for hunk in hunks:
29*fb1b10abSAndroid Build Coastguard Worker        this_header = hunk.header[0:2]
30*fb1b10abSAndroid Build Coastguard Worker        if last_header != this_header:
31*fb1b10abSAndroid Build Coastguard Worker            r.extend(hunk.header)
32*fb1b10abSAndroid Build Coastguard Worker            last_header = this_header
33*fb1b10abSAndroid Build Coastguard Worker        else:
34*fb1b10abSAndroid Build Coastguard Worker            r.extend(hunk.header[2])
35*fb1b10abSAndroid Build Coastguard Worker        r.extend(hunk.lines)
36*fb1b10abSAndroid Build Coastguard Worker        r.append("\n")
37*fb1b10abSAndroid Build Coastguard Worker    return "".join(r)
38*fb1b10abSAndroid Build Coastguard Worker
39*fb1b10abSAndroid Build Coastguard Worker
40*fb1b10abSAndroid Build Coastguard Workerdef ZipHunks(rhs_hunks, lhs_hunks):
41*fb1b10abSAndroid Build Coastguard Worker    """Join two hunk lists on filename."""
42*fb1b10abSAndroid Build Coastguard Worker    for rhs_hunk in rhs_hunks:
43*fb1b10abSAndroid Build Coastguard Worker        rhs_file = rhs_hunk.right.filename.split("/")[1:]
44*fb1b10abSAndroid Build Coastguard Worker
45*fb1b10abSAndroid Build Coastguard Worker        for lhs_hunk in lhs_hunks:
46*fb1b10abSAndroid Build Coastguard Worker            lhs_file = lhs_hunk.left.filename.split("/")[1:]
47*fb1b10abSAndroid Build Coastguard Worker            if lhs_file != rhs_file:
48*fb1b10abSAndroid Build Coastguard Worker                continue
49*fb1b10abSAndroid Build Coastguard Worker            yield (rhs_hunk, lhs_hunk)
50*fb1b10abSAndroid Build Coastguard Worker
51*fb1b10abSAndroid Build Coastguard Worker
52*fb1b10abSAndroid Build Coastguard Workerdef main():
53*fb1b10abSAndroid Build Coastguard Worker    old_hunks = [x for x in diff.ParseDiffHunks(open(sys.argv[1], "r"))]
54*fb1b10abSAndroid Build Coastguard Worker    new_hunks = [x for x in diff.ParseDiffHunks(open(sys.argv[2], "r"))]
55*fb1b10abSAndroid Build Coastguard Worker    out_hunks = []
56*fb1b10abSAndroid Build Coastguard Worker
57*fb1b10abSAndroid Build Coastguard Worker    # Join the right hand side of the older diff with the left hand side of the
58*fb1b10abSAndroid Build Coastguard Worker    # newer diff.
59*fb1b10abSAndroid Build Coastguard Worker    for old_hunk, new_hunk in ZipHunks(old_hunks, new_hunks):
60*fb1b10abSAndroid Build Coastguard Worker        if new_hunk in out_hunks:
61*fb1b10abSAndroid Build Coastguard Worker            continue
62*fb1b10abSAndroid Build Coastguard Worker        old_lines = old_hunk.right
63*fb1b10abSAndroid Build Coastguard Worker        new_lines = new_hunk.left
64*fb1b10abSAndroid Build Coastguard Worker
65*fb1b10abSAndroid Build Coastguard Worker        # Determine if this hunk overlaps any non-context line from the other
66*fb1b10abSAndroid Build Coastguard Worker        for i in old_lines.delta_line_nums:
67*fb1b10abSAndroid Build Coastguard Worker            if i in new_lines:
68*fb1b10abSAndroid Build Coastguard Worker                out_hunks.append(new_hunk)
69*fb1b10abSAndroid Build Coastguard Worker                break
70*fb1b10abSAndroid Build Coastguard Worker
71*fb1b10abSAndroid Build Coastguard Worker    if out_hunks:
72*fb1b10abSAndroid Build Coastguard Worker        print(FormatDiffHunks(out_hunks))
73*fb1b10abSAndroid Build Coastguard Worker        sys.exit(1)
74*fb1b10abSAndroid Build Coastguard Worker
75*fb1b10abSAndroid Build Coastguard Workerif __name__ == "__main__":
76*fb1b10abSAndroid Build Coastguard Worker    main()
77