This file is indexed.

/usr/lib/python2.7/dist-packages/cluster/cluster.py is in python-cluster 1.3.3-1.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
#
# This is part of "python-cluster". A library to group similar items together.
# Copyright (C) 2006    Michel Albert
#
# This library is free software; you can redistribute it and/or modify it
# under the terms of the GNU Lesser General Public License as published by the
# Free Software Foundation; either version 2.1 of the License, or (at your
# option) any later version.
# This library is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
# for more details.
# You should have received a copy of the GNU Lesser General Public License
# along with this library; if not, write to the Free Software Foundation,
# Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#

from __future__ import print_function

from .util import fullyflatten


class Cluster(object):
    """
    A collection of items. This is internally used to detect clustered items
    in the data so we could distinguish other collection types (lists, dicts,
    ...) from the actual clusters. This means that you could also create
    clusters of lists with this class.
    """

    def __repr__(self):
        return "<Cluster@%s(%s)>" % (self.level, self.items)

    def __str__(self):
        return self.__str__()

    def __init__(self, level, *args):
        """
        Constructor

        :param level: The level of this cluster. This is used in hierarchical
            clustering to retrieve a specific set of clusters. The higher the
            level, the smaller the count of clusters returned. The level depends
            on the difference function used.
        :param *args: every additional argument passed following the level value
            will get added as item to the cluster. You could also pass a list as
            second parameter to initialise the cluster with that list as content
        """
        self.level = level
        if len(args) == 0:
            self.items = []
        else:
            self.items = args

    def __iter__(self):
        for item in self.items:
            if isinstance(item, Cluster):
                for recursed_item in item:
                    yield recursed_item
            else:
                yield item

    def display(self, depth=0):
        """
        Pretty-prints this cluster. Useful for debuging.
        """
        print(depth * "    " + "[level %s]" % self.level)
        for item in self.items:
            if isinstance(item, Cluster):
                item.display(depth + 1)
            else:
                print(depth * "    " + "%s" % item)

    def topology(self):
        """
        Returns the structure (topology) of the cluster as tuples.

        Output from cl.data::

                [<Cluster@0.833333333333(['CVS',
                 <Cluster@0.818181818182(['34.xls',
                 <Cluster@0.789473684211([<Cluster@0.555555555556(['0.txt',
                 <Cluster@0.181818181818(['ChangeLog', 'ChangeLog.txt'])>])>,
                 <Cluster@0.684210526316(['20060730.py',
                 <Cluster@0.684210526316(['.cvsignore',
                 <Cluster@0.647058823529(['About.py', <Cluster@0.625(['.idlerc',
                 '.pylint.d'])>])>])>])>])>])>])>]

        Corresponding output from cl.topo()::

                ('CVS', ('34.xls', (('0.txt', ('ChangeLog', 'ChangeLog.txt')),
                ('20060730.py', ('.cvsignore', ('About.py',
                ('.idlerc', '.pylint.d')))))))
        """

        left = self.items[0]
        right = self.items[1]

        if isinstance(left, Cluster):
            first = left.topology()
        else:
            first = left

        if isinstance(right, Cluster):
            second = right.topology()
        else:
            second = right

        return first, second

    def getlevel(self, threshold):
        """
        Retrieve all clusters up to a specific level threshold. This
        level-threshold represents the maximum distance between two clusters.
        So the lower you set this threshold, the more clusters you will
        receive and the higher you set it, you will receive less but bigger
        clusters.

        :param threshold: The level threshold:

        .. note::
            It is debatable whether the value passed into this method should
            really be as strongly linked to the real cluster-levels as it is
            right now. The end-user will not know the range of this value
            unless s/he first inspects the top-level cluster. So instead you
            might argue that a value ranging from 0 to 1 might be a more
            useful approach.
        """

        left = self.items[0]
        right = self.items[1]

        # if this object itself is below the threshold value we only need to
        # return it's contents as a list
        if self.level <= threshold:
            return [fullyflatten(self.items)]

        # if this cluster's level is higher than the threshold we will
        # investgate it's left and right part. Their level could be below the
        # threshold
        if isinstance(left, Cluster) and left.level <= threshold:
            if isinstance(right, Cluster):
                return [fullyflatten(left.items)] + right.getlevel(threshold)
            else:
                return [fullyflatten(left.items)] + [[right]]
        elif isinstance(right, Cluster) and right.level <= threshold:
            if isinstance(left, Cluster):
                return left.getlevel(threshold) + [fullyflatten(right.items)]
            else:
                return [[left]] + [fullyflatten(right.items)]

        # Alright. We covered the cases where one of the clusters was below
        # the threshold value. Now we'll deal with the clusters that are above
        # by recursively applying the previous cases.
        if isinstance(left, Cluster) and isinstance(right, Cluster):
            return left.getlevel(threshold) + right.getlevel(threshold)
        elif isinstance(left, Cluster):
            return left.getlevel(threshold) + [[right]]
        elif isinstance(right, Cluster):
            return [[left]] + right.getlevel(threshold)
        else:
            return [[left], [right]]