Skip to content

Commit

Permalink
devel: add reposec management command
Browse files Browse the repository at this point in the history
Add a new command that scans pkg.tar.xz files with elf binaries in
/usr/bin/ and checks for security hardening issues. This adds a new
dashboard view which shows packages with these issues.
  • Loading branch information
jelly committed Oct 9, 2019
1 parent 767feb1 commit 8f8b31d
Show file tree
Hide file tree
Showing 5 changed files with 311 additions and 2 deletions.
236 changes: 236 additions & 0 deletions devel/management/commands/reposec.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
"""
reposec command
Parses all packages in a given repo and creates PackageSecurity
objects which check for PIE, RELRO, Stack Canary's and Fortify.
Usage: ./manage.py reposec ARCH PATH
ARCH: architecture to check
PATH: full path to the repository directory.
Example:
./manage.py reposec x86_64 /srv/ftp/core
"""

import io
import os
import re
import sys
import logging

from functools import partial
from glob import glob
from multiprocessing import Pool, cpu_count

from elftools.elf.constants import P_FLAGS
from elftools.elf.dynamic import DynamicSection
from elftools.elf.elffile import ELFFile
from elftools.elf.sections import SymbolTableSection
from libarchive import file_reader

from django.core.management.base import BaseCommand, CommandError
from django.db import transaction

from main.models import Arch, Package, PackageSecurity, Repo


PKG_EXT = '.tar.xz'
STACK_CHK = set(["__stack_chk_fail", "__stack_smash_handler"])


logging.basicConfig(
level=logging.WARNING,
format='%(asctime)s -> %(levelname)s: %(message)s',
datefmt='%Y-%m-%d %H:%M:%S',
stream=sys.stderr)
TRACE = 5
logging.addLevelName(TRACE, 'TRACE')
logger = logging.getLogger()

class Command(BaseCommand):
help = ""
missing_args_message = 'missing arch and file.'

def add_arguments(self, parser):
parser.add_argument('args', nargs='*', help='<arch> <filename>')
parser.add_argument('--processes',
action='store_true',
dest='processes',
default=cpu_count(),
help=f'number of parallel processes (default: {cpu_count()})')


def handle(self, arch=None, directory=None, processes=cpu_count(), **options):
if not arch:
raise CommandError('Architecture is required.')
if not directory:
raise CommandError('Repo location is required.')
directory = os.path.normpath(directory)
if not os.path.exists(directory):
raise CommandError('Specified repository location does not exists.')

v = int(options.get('verbosity', 0))
if v == 0:
logger.level = logging.ERROR
elif v == 1:
logger.level = logging.INFO
elif v >= 2:
logger.level = logging.DEBUG

return read_repo(arch, directory, processes, options)


def read_file(arch, repo, filename):
pkgsec = None
basename = os.path.basename(filename)
pkgname = basename.rsplit('-', 3)[0]

with file_reader(filename) as pkg:
for entry in pkg:
if not entry.isfile:
continue

# Retrieve pkgname
if entry.name == '.PKGINFO':
continue

if not entry.name.startswith('usr/bin/'):
continue

fp = io.BytesIO(b''.join(entry.get_blocks()))
elf = Elf(fp)

if not elf.is_elf():
continue

pkg = Package.objects.get(pkgname=pkgname, arch=arch, repo=repo)
pkgsec = PackageSecurity(pkg=pkg, pie=elf.pie, relro=elf.relro, canary=elf.canary)

return pkgsec



def read_repo(arch, source_dir, processes, options):
tasks = []

directory = os.path.join(source_dir, 'os', arch)
for filename in glob(os.path.join(directory, f'*{PKG_EXT}')):
tasks.append((filename))

arch = Arch.objects.get(name=arch)

reponame = os.path.basename(source_dir).title()
repo = Repo.objects.get(name=reponame)


with Pool(processes=processes) as pool:
results = pool.map(partial(read_file, arch, repo), tasks)

results = [r for r in results if r and (not r.pie or not r.relro or not r.canary)]

with transaction.atomic():
PackageSecurity.objects.all().delete()
PackageSecurity.objects.bulk_create(results)


class Elf:
def __init__(self, fileobj):
self.fileobj = fileobj
self._elffile = None

@property
def elffile(self):
if not self._elffile:
self._elffile = ELFFile(self.fileobj)
return self._elffile

def _file_has_magic(self, fileobj, magic_bytes):
length = len(magic_bytes)
magic = fileobj.read(length)
fileobj.seek(0)
return magic == magic_bytes

def is_elf(self):
"Take file object, peek at the magic bytes to check if ELF file."
return self._file_has_magic(self.fileobj, b"\x7fELF")

def dynamic_tags(self, key):
for section in self.elffile.iter_sections():
if not isinstance(section, DynamicSection):
continue
for tag in section.iter_tags():
if tag.entry.d_tag == key:
return tag
return None

def rpath(self, key="DT_RPATH", verbose=False):
tag = self.dynamic_tags(key)
if tag and verbose:
return tag.rpath
if tag:
return 'RPATH'
return ''

def runpath(self, key="DT_RUNPATH", verbose=False):
tag = self.dynamic_tags(key)
if tag and verbose:
return tag.runpath
if tag:
return 'RUNPATH'

return ''

@property
def relro(self):
if self.elffile.num_segments() == 0:
return "Disabled"

have_relro = False
for segment in self.elffile.iter_segments():
if re.search("GNU_RELRO", str(segment['p_type'])):
have_relro = True
break

if self.dynamic_tags("DT_BIND_NOW") and have_relro:
return True
if have_relro: # partial
return False
return False

@property
def pie(self):
header = self.elffile.header
if self.dynamic_tags("EXEC"):
return "Disabled"
if "ET_DYN" in header['e_type']:
if self.dynamic_tags("DT_DEBUG"):
return True
return True # DSO is PIE
return False

@property
def canary(self):
for section in self.elffile.iter_sections():
if not isinstance(section, SymbolTableSection):
continue
if section['sh_entsize'] == 0:
continue
for _, symbol in enumerate(section.iter_symbols()):
if symbol.name in STACK_CHK:
return True
return False

def program_headers(self):
pflags = P_FLAGS()
if self.elffile.num_segments() == 0:
return ""

found = False
for segment in self.elffile.iter_segments():
if search("GNU_STACK", str(segment['p_type'])):
found = True
if segment['p_flags'] & pflags.PF_X:
return "Disabled"
if found:
return "Enabled"
return "Disabled"
27 changes: 25 additions & 2 deletions devel/reports.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
from django.template.defaultfilters import filesizeformat
from django.db import connection
from django.utils.timezone import now
from main.models import Package, PackageFile
from main.models import Package, PackageFile, PackageSecurity
from packages.models import Depend, PackageRelation

from .models import DeveloperKey
Expand Down Expand Up @@ -167,6 +167,20 @@ def non_existing_dependencies(packages):
return packages


def security_packages_overview(packages):
filtered = []
packages_ids = packages.values_list('id',
flat=True).order_by().distinct()
packages = PackageSecurity.objects.filter(id__in=set(packages_ids))
for package in packages:
package.pkg.pie = 'PIE enabled' if package.pie else 'No PIE'
package.pkg.relro = 'Full RELRO' if package.relro else 'None'
package.pkg.canary = 'Canary found' if package.canary else 'No canary found'
package.pkg.fortify = 'Yes' if package.canary else 'No'
filtered.append(package.pkg)

return filtered


REPORT_OLD = DeveloperReport(
'old', 'Old', 'Packages last built more than two years ago', old)
Expand Down Expand Up @@ -223,6 +237,14 @@ def non_existing_dependencies(packages):
['nonexistingdep'],
personal=False)

REPORT_SECURITY = DeveloperReport(
'security-issue-packages',
'Security of Packages',
'Packages that have security issues',
security_packages_overview,
['PIE', 'RELRO', 'CANARY', 'FORTIFY'], ['pie', 'relro', 'canary', 'fortify'])


def available_reports():
return (REPORT_OLD,
REPORT_OUTOFDATE,
Expand All @@ -233,4 +255,5 @@ def available_reports():
REPORT_ORPHANS,
REPORT_SIGNATURE,
REPORT_SIG_TIME,
NON_EXISTING_DEPENDENCIES, )
NON_EXISTING_DEPENDENCIES,
REPORT_SECURITY, )
28 changes: 28 additions & 0 deletions main/migrations/0003_packagesecurity.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
# Generated by Django 2.2.5 on 2019-10-09 19:24

from django.db import migrations, models
import django.db.models.deletion


class Migration(migrations.Migration):

dependencies = [
('main', '0002_repo_public_testing'),
]

operations = [
migrations.CreateModel(
name='PackageSecurity',
fields=[
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('pie', models.BooleanField(default=False)),
('relro', models.PositiveIntegerField(choices=[(1, 'No RELRO'), (2, 'Partial RELRO'), (2, 'Full RELRO')], default=1)),
('canary', models.BooleanField(default=False)),
('fortify', models.BooleanField(default=False)),
('pkg', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='main.Package')),
],
options={
'db_table': 'package_security',
},
),
]
20 changes: 20 additions & 0 deletions main/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -444,6 +444,26 @@ class Meta:
db_table = 'package_files'


class PackageSecurity(models.Model):
NO_RELRO = 1
PARTIAL_RELRO = 2
FULL_RELRO = 2
RELRO_CHOICES = (
(NO_RELRO, 'No RELRO'),
(PARTIAL_RELRO, 'Partial RELRO'),
(FULL_RELRO, 'Full RELRO'),
)

pkg = models.ForeignKey(Package, on_delete=models.CASCADE)
pie = models.BooleanField(default=False)
relro = models.PositiveIntegerField(choices=RELRO_CHOICES, default=NO_RELRO)
canary = models.BooleanField(default=False)
fortify = models.BooleanField(default=False)

class Meta:
db_table = 'package_security'


from django.db.models.signals import pre_save

# note: reporead sets the 'created' field on Package objects, so no signal
Expand Down
2 changes: 2 additions & 0 deletions requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,5 @@ django-jinja==2.4.1
sqlparse==0.3.0
django-csp==3.5
ptpython==2.0.4
pyelftools==0.25
libarchive-c==2.8

0 comments on commit 8f8b31d

Please sign in to comment.