2009-01-21 14:39:29 +01:00
|
|
|
# -*- python -*-
|
|
|
|
# -*- coding: utf-8 -*-
|
2009-01-21 16:11:39 +01:00
|
|
|
#
|
|
|
|
# DDPortfolio service application key ring analyzer tool
|
2012-01-07 01:46:57 +01:00
|
|
|
# Copyright © 2009, 2010, 2011, 2012 Jan Dittberner <jan@dittberner.info>
|
2009-01-21 16:11:39 +01:00
|
|
|
#
|
|
|
|
# This file is part of DDPortfolio service.
|
|
|
|
#
|
|
|
|
# DDPortfolio service is free software: you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU Affero General Public License
|
|
|
|
# as published by the Free Software Foundation, either version 3 of
|
|
|
|
# the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# DDPortfolio service 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
|
|
|
|
# Affero General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Affero General Public
|
|
|
|
# License along with this program. If not, see
|
|
|
|
# <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
2009-01-21 14:39:29 +01:00
|
|
|
"""
|
|
|
|
This is a tool that analyzes GPG and PGP keyrings and stores the
|
|
|
|
retrieved data in a file database. The tool was inspired by Debian
|
|
|
|
qa's carnivore.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import anydbm
|
|
|
|
import pkg_resources
|
|
|
|
import glob
|
|
|
|
import ConfigParser
|
|
|
|
import os
|
2010-06-06 00:31:41 +02:00
|
|
|
import os.path
|
2010-10-26 15:40:04 +02:00
|
|
|
import logging
|
|
|
|
import subprocess
|
|
|
|
import sys
|
2009-01-21 14:39:29 +01:00
|
|
|
|
|
|
|
|
|
|
|
def _get_keyrings():
|
2012-01-07 01:46:57 +01:00
|
|
|
"""
|
|
|
|
Gets the available keyring files from the keyring directory
|
|
|
|
configured in ddportfolio.ini.
|
|
|
|
"""
|
2009-01-21 14:39:29 +01:00
|
|
|
my_config = ConfigParser.ConfigParser()
|
2012-01-07 01:46:57 +01:00
|
|
|
my_config.readfp(pkg_resources.resource_stream(
|
|
|
|
__name__, 'ddportfolio.ini'))
|
2010-06-06 00:31:41 +02:00
|
|
|
keyringdir = os.path.expanduser(my_config.get('DEFAULT', 'keyring.dir'))
|
2010-10-26 15:40:04 +02:00
|
|
|
logging.debug("keyring dir is %s", keyringdir)
|
2010-06-06 00:31:41 +02:00
|
|
|
keyrings = glob.glob(os.path.join(keyringdir, '*.gpg'))
|
|
|
|
keyrings.extend(glob.glob(os.path.join(keyringdir, '*.pgp')))
|
2009-01-21 14:39:29 +01:00
|
|
|
keyrings.sort()
|
|
|
|
return keyrings
|
|
|
|
|
|
|
|
|
|
|
|
def _parse_uid(uid):
|
2012-01-07 01:46:57 +01:00
|
|
|
"""
|
|
|
|
Parse a uid of the form 'Real Name <email@example.com>' into email
|
|
|
|
and realname parts.
|
|
|
|
"""
|
2009-01-21 14:39:29 +01:00
|
|
|
uid = uid.strip()
|
2012-01-07 01:46:57 +01:00
|
|
|
# First, strip comment
|
2009-01-21 14:39:29 +01:00
|
|
|
s = uid.find('(')
|
|
|
|
e = uid.find(')')
|
|
|
|
if s >= 0 and e >= 0:
|
2012-01-07 01:46:57 +01:00
|
|
|
uid = uid[:s] + uid[e + 1:]
|
2009-01-21 14:39:29 +01:00
|
|
|
s = uid.find('<')
|
|
|
|
e = uid.find('>')
|
|
|
|
email = None
|
|
|
|
if s >= 0 and e >= 0:
|
2012-01-07 01:46:57 +01:00
|
|
|
email = uid[s + 1:e]
|
|
|
|
uid = uid[:s] + uid[e + 1:]
|
2009-01-21 14:39:29 +01:00
|
|
|
uid = uid.strip()
|
|
|
|
if not email and uid.find('@') >= 0:
|
|
|
|
email, uid = uid, email
|
|
|
|
return (uid, email)
|
|
|
|
|
|
|
|
resultdict = {}
|
|
|
|
|
2012-01-07 01:46:57 +01:00
|
|
|
|
2009-01-21 14:39:29 +01:00
|
|
|
def _get_canonical(key):
|
|
|
|
if not key in resultdict:
|
|
|
|
resultdict[key] = []
|
|
|
|
return key
|
|
|
|
|
|
|
|
|
|
|
|
def _add_to_result(key, newvalue):
|
|
|
|
thekey = _get_canonical(key)
|
|
|
|
if newvalue not in resultdict[thekey]:
|
|
|
|
resultdict[thekey].append(newvalue)
|
|
|
|
|
|
|
|
|
|
|
|
def process_keyrings():
|
|
|
|
"""Process the keyrings and store the extracted data in an anydbm
|
|
|
|
file."""
|
|
|
|
for keyring in _get_keyrings():
|
2010-10-26 15:40:04 +02:00
|
|
|
logging.debug("get data from %s", keyring)
|
|
|
|
proc = subprocess.Popen(["gpg", "--no-default-keyring",
|
|
|
|
"--no-expensive-trust-checks",
|
|
|
|
"--keyring", keyring, "--list-keys",
|
|
|
|
"--with-colons", "--fingerprint"],
|
|
|
|
stdout=subprocess.PIPE)
|
2009-01-21 14:39:29 +01:00
|
|
|
fpr = None
|
|
|
|
entry = None
|
2012-01-07 01:46:57 +01:00
|
|
|
lastpub = None
|
2010-10-26 15:40:04 +02:00
|
|
|
for line in proc.stdout.readlines():
|
2009-01-21 14:39:29 +01:00
|
|
|
items = line.split(':')
|
|
|
|
uid = None
|
|
|
|
if items[0] == 'pub':
|
|
|
|
fpr = entry = None
|
|
|
|
lastpub = items[9].strip()
|
|
|
|
continue
|
|
|
|
elif items[0] == 'fpr':
|
|
|
|
fpr = items[9].strip()
|
|
|
|
uid = lastpub
|
|
|
|
elif items[0] == 'uid':
|
|
|
|
uid = items[9].strip()
|
|
|
|
else:
|
|
|
|
continue
|
|
|
|
# Do stuff with 'uid'
|
|
|
|
uid, email = _parse_uid(uid)
|
|
|
|
if email:
|
|
|
|
if email.endswith('@debian.org'):
|
|
|
|
login = email[0:-len('@debian.org')]
|
|
|
|
_add_to_result('login:email:%s' % email, login)
|
|
|
|
_add_to_result('login:fpr:%s' % fpr, login)
|
2009-01-21 17:18:02 +01:00
|
|
|
_add_to_result('fpr:login:%s' % login, fpr)
|
|
|
|
_add_to_result('fpr:email:%s' % email, fpr)
|
2009-01-21 14:39:29 +01:00
|
|
|
_add_to_result('email:fpr:%s' % fpr, email)
|
|
|
|
if uid:
|
|
|
|
_add_to_result('name:fpr:%s' % fpr, uid)
|
|
|
|
if email:
|
|
|
|
_add_to_result('name:email:%s' % email, uid)
|
2010-10-26 15:40:04 +02:00
|
|
|
retcode = proc.wait()
|
|
|
|
if retcode != 0:
|
|
|
|
logging.error("subprocess ended with return code %d", retcode)
|
2009-01-21 14:39:29 +01:00
|
|
|
db = anydbm.open(pkg_resources.resource_filename(__name__,
|
|
|
|
'keyringcache'), 'c')
|
|
|
|
for key in resultdict:
|
|
|
|
db[key] = ":".join(resultdict[key])
|
|
|
|
db.close()
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2010-10-26 15:40:04 +02:00
|
|
|
logging.basicConfig(stream=sys.stderr, level=logging.WARNING)
|
2009-01-21 14:39:29 +01:00
|
|
|
process_keyrings()
|