This file is indexed.

/usr/lib/python3/dist-packages/plainbox/vendor/pyglibc/_subreaper.py is in python3-plainbox 0.25-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
163
164
165
166
167
# Copyright (c) 2014 Canonical Ltd.
#
# Author: Zygmunt Krynicki <zygmunt.krynicki@canonical.com>
#
# This program 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 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
"""
:mod:`pyglibc._subreaper` -- python wrapper around PR_SET_CHILD_SUBREAPER
=========================================================================

For more discussion of what a subreaper is, please consult the ``prctl(2)``
manual page and the following Linux kernel mailing thread:
    http://thread.gmane.org/gmane.linux.kernel/1236479
and the following LWN article:
    http://lwn.net/Articles/474787/
"""
from __future__ import absolute_import
from ctypes import addressof
from ctypes import c_int

from plainbox.vendor.glibc import PR_GET_CHILD_SUBREAPER
from plainbox.vendor.glibc import PR_SET_CHILD_SUBREAPER
from plainbox.vendor.glibc import prctl

__all__ = ['subreaper']


def _sr_unsupported():
    raise ValueError("PR_SET_CHILD_SUBREAPER is unsupported")


class _subreaper:
    """
    Pythonic wrapper around ``prctl(PR_{GET,SET}_CHILD_SUBREAPER, ...)``
    """
    SR_UNKNOWN = 0
    SR_UNSUPPORTED = 1
    SR_ENABLED = 2
    SR_DISABLED = 3

    _SR_STATUS_NAME = {
        SR_UNKNOWN: "unknown",
        SR_UNSUPPORTED: "unsupported",
        SR_ENABLED: "enabled",
        SR_DISABLED: "disabled"
    }

    __slots__ = ('_status',)

    def __init__(self):
        """
        Initialize a new subreaper object.

        Typically applications should not need to do this, there is a
        pre-initialized subreaper object available from this module.
        """
        self._status = self.SR_UNKNOWN

    def __repr__(self):
        return "<subreaper status:{}>".format(self.status_name)

    @property
    def status(self):
        """
        status of of the child sub-reaper flag

        Possible values are:

            SR_UNKNOWN:
                The current status of PR_SET_CHILD_SUBREAPER is unknown. Try
                setting or getting the :meth:`enabled` property to determine
                the status.
            SR_UNSUPPORTED:
                The PR_SET_CHILD_SUBREAPER option is not supported on this
                system. This feature requires Linux 3.4 or newer.
            SR_ENABLED:
                The PR_SET_CHILD_SUBREAPER option is supported and this flag
                is currently enabled.
            SR_DISABLED:
                The PR_SET_CHILD_SUBREAPER option is supported and this flag
                is currently disabled.
        """
        return self._status

    @property
    def status_name(self):
        """
        textual form of the current :meth:`status`, this is meant for debugging
        """
        return self._SR_STATUS_NAME[self._status]

    @property
    def enabled(self):
        """
        read or write the child sub-reaper flag of the current process

        This property behaves in the following manner:

        * If a read is attempted and a prior read or write has determined that
          this feature is unavailable (status is equal to ``SR_UNSUPPORTED``)
          then no further attempts are made and the outcome is ``False``.
        * If a read is attempted and the current status is ``SR_UNKNOWN`` then
          a ``prctl(PR_GET_CHILD_SUBREAPER, ...)`` call is made and the outcome
          depends on the returned value. If prctl fails then status is set to
          ``SR_UNSUPPORTED`` and the return value is ``False``. If the prctl
          call succeeds then status is set to either ``SR_ENABLED`` or
          ``SR_DISABLED`` and ``True`` or ``False`` is returned, respectively.
         * If a write is attempted and a prior read or write has determined
           that this feature is unavailable (status is equal to
           ``SR_UNSUPPORTED``) *and* the write would have enabled the flag, a
           ValueError is raised with an appropriate message. Otherwise a write
           is attempted. If the attempt to enable the flag fails a ValueError
           is raised, just as in the previous case.
         * If a write intending to disable the flag fails then this failure is
           silently ignored but status is set to ``SR_UNSUPPORTED``.
         * If a write succeeds then the status is set accordingly to
           ``SR_ENABLED`` or ``SR_DISABLED``, depending on the value written
           ``True`` or ``False`` respectively.

        In other words, this property behaves as if it was really calling
        prctl() but it is not going to repeat operations that will always fail.
        Nor will it ignore failures silently where that matters.
        """
        if self._status == self.SR_UNSUPPORTED:
            return False
        status = c_int()
        try:
            prctl(PR_GET_CHILD_SUBREAPER, addressof(status), 0, 0, 0)
        except OSError:
            self._status = self.SR_UNSUPPORTED
        else:
            self._status = self.SR_ENABLED if status else self.SR_DISABLED
            return self._status == self.SR_ENABLED

    @enabled.setter
    def enabled(self, status):
        if self._status == self.SR_UNSUPPORTED and status:
            _sr_unsupported()
        try:
            prctl(PR_SET_CHILD_SUBREAPER, 1 if status else 0, 0, 0, 0)
        except OSError:
            self._status = self.SR_UNSUPPORTED
        else:
            self._status = self.SR_ENABLED if status else self.SR_DISABLED
        if self._status == self.SR_UNSUPPORTED and status:
            _sr_unsupported()

    def __enter__(self):
        """
        """
        self.enabled = True

    def __exit__(self, *args):
        self.enabled = False


subreaper = _subreaper()