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
|
# vim: set fileencoding=utf-8 :
# (C) 2014 Guido Günther <agx@sigxcpu.org>
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
"""Test L{whatmaps.process} config"""
import os
import unittest
import random
from whatmaps.process import Process
from . import context
class TestWhatmapsProcess(unittest.TestCase):
def setUp(self):
self.tmpdir = context.new_tmpdir(__name__)
self.procfs = str(self.tmpdir)
self.pid = random.randint(1, 65535)
self.piddir = os.path.join(self.procfs, str(self.pid))
os.mkdir(self.piddir)
self.exe = os.path.join(self.piddir, 'exe')
self.cmdline = os.path.join(self.piddir, 'cmdline')
self.maps = os.path.join(self.piddir, 'maps')
self._write_cmdline('doesnotmatter') # Write at least an empty cmdline
self._write_exe_symlink('acommand')
self._write_maps([['f32b43221000-7f32b4522000',
'---p',
'00020000',
'fe:02',
'1704011',
'/lib/x86_64-linux-gnu/libselinux.so.1'],
['7f32b4521000-7f32b4623000',
'r--p',
'00020000',
'fe:02',
'1704011',
'/lib/x86_64-linux-gnu/libselinux.so.1'],
])
def _write_exe_symlink(self, name):
exe = os.path.join(str(self.tmpdir), name)
os.symlink(exe, self.exe)
def _write_cmdline(self, text=''):
f = open(self.cmdline, 'w')
f.write(text)
f.close()
def _write_maps(self, data):
f = open(self.maps, 'w')
f.write('\n'.join([' '.join(r) for r in data]))
f.close()
def test_nonexistent(self):
"""No exe link should create an 'empty' object"""
os.unlink(self.exe)
p = Process(self.pid, self.procfs)
self.assertIsNone(p.exe)
self.assertIsNone(p.cmdline)
def test_deleted(self):
"""Handle symlink to deleted binaries"""
exe = '/does/not/matter'
os.unlink(self.exe)
os.symlink(os.path.join(self.piddir, '%s (deleted)' % exe),
self.exe)
p = Process(self.pid, procfs=self.procfs)
self.assertEqual(p.exe, exe)
self.assertTrue(p.deleted, True)
self.assertEqual(p.cmdline, 'doesnotmatter')
def test_existing(self):
p = Process(self.pid, procfs=self.procfs)
exe = os.path.join(str(self.tmpdir), 'acommand')
self.assertEqual(p.exe, exe)
self.assertEqual(p.cmdline, 'doesnotmatter')
self.assertFalse(p.deleted)
self.assertEqual(str(p),
"<Process object pid:%d>" % self.pid)
def test_maps(self):
"""Check whether the process maps a shared object at path"""
p = Process(self.pid, procfs=self.procfs)
self.assertFalse(p.maps('/does/not/exist'))
self.assertTrue(p.maps('/lib/x86_64-linux-gnu/libselinux.so.1'))
def test_no_maps(self):
"""Check if we don't fail if the process went away"""
os.unlink(self.maps)
p = Process(self.pid, procfs=self.procfs)
self.assertFalse(p.maps('/does/not/exist'))
self.assertFalse(p.maps('/lib/x86_64-linux-gnu/libselinux.so.1'))
def test_broken_maps(self):
"""Continue on unparseable map file"""
# Provoke index error by to few items in line
self._write_maps([['do', 'few', 'items']])
p = Process(self.pid, procfs=self.procfs)
self.assertFalse(p.maps('/does/not/exist'))
self.assertFalse(p.maps('/lib/x86_64-linux-gnu/libselinux.so.1'))
@unittest.skipIf(os.getuid() == 0, "Skip if root")
def test_broken_unreadable_map(self):
"""Raise error if map file is unreadable"""
os.chmod(self.maps, 0)
p = Process(self.pid, procfs=self.procfs)
self.assertRaises(IOError, p.maps, '/does/not/exist')
def tearDown(self):
context.teardown()
|