2013-07-04 10:45:42 -05:00
|
|
|
#! /usr/bin/python
|
|
|
|
# Authors: Karl MacMillan <kmacmillan@mentalrootkit.com>
|
|
|
|
# Jan Cholasta <jcholast@redhat.com>
|
|
|
|
#
|
|
|
|
# Copyright (C) 2007-2013 Red Hat
|
|
|
|
# see file 'COPYING' for use and warranty information
|
|
|
|
#
|
|
|
|
# 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/>.
|
|
|
|
#
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
import os.path
|
|
|
|
import pwd
|
2013-08-20 01:39:39 -05:00
|
|
|
import optparse
|
2013-07-04 10:45:42 -05:00
|
|
|
|
|
|
|
from ipapython import admintool
|
|
|
|
from ipapython.dn import DN
|
|
|
|
from ipapython.ipautil import user_input, write_tmp_file
|
2013-07-04 11:43:08 -05:00
|
|
|
from ipalib import api, errors
|
2013-07-04 10:45:42 -05:00
|
|
|
from ipaserver.install import certs, dsinstance, httpinstance, installutils
|
|
|
|
from ipaserver.plugins.ldap2 import ldap2
|
|
|
|
|
|
|
|
CACERT = "/etc/ipa/ca.crt"
|
|
|
|
|
|
|
|
class ServerCertInstall(admintool.AdminTool):
|
|
|
|
command_name = 'ipa-server-certinstall'
|
|
|
|
|
2013-08-27 11:17:02 -05:00
|
|
|
usage = "%prog <-d|-w> [options] <PKCS#12 file>"
|
2013-07-04 10:45:42 -05:00
|
|
|
|
|
|
|
description = "Install new SSL server certificates."
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def add_options(cls, parser):
|
|
|
|
super(ServerCertInstall, cls).add_options(parser)
|
|
|
|
|
|
|
|
parser.add_option(
|
|
|
|
"-d", "--dirsrv",
|
|
|
|
dest="dirsrv", action="store_true", default=False,
|
|
|
|
help="install certificate for the directory server")
|
|
|
|
parser.add_option(
|
|
|
|
"-w", "--http",
|
|
|
|
dest="http", action="store_true", default=False,
|
|
|
|
help="install certificate for the http server")
|
|
|
|
parser.add_option(
|
2013-08-20 01:39:39 -05:00
|
|
|
"--pin",
|
|
|
|
dest="pin",
|
|
|
|
help="The password of the PKCS#12 file")
|
2013-07-04 10:45:42 -05:00
|
|
|
parser.add_option(
|
2013-08-20 01:39:39 -05:00
|
|
|
"--dirsrv_pin", "--http_pin",
|
|
|
|
dest="pin",
|
|
|
|
help=optparse.SUPPRESS_HELP)
|
2013-08-27 11:06:24 -05:00
|
|
|
parser.add_option(
|
|
|
|
"-p", "--dirman-password",
|
|
|
|
dest="dirman_password",
|
|
|
|
help="Directory Manager password")
|
2013-07-04 10:45:42 -05:00
|
|
|
|
|
|
|
def validate_options(self):
|
|
|
|
super(ServerCertInstall, self).validate_options(needs_root=True)
|
|
|
|
|
|
|
|
installutils.check_server_configuration()
|
|
|
|
|
|
|
|
if not self.options.dirsrv and not self.options.http:
|
|
|
|
self.option_parser.error("you must specify dirsrv and/or http")
|
|
|
|
|
|
|
|
if len(self.args) != 1:
|
|
|
|
self.option_parser.error("you must provide a pkcs12 filename")
|
|
|
|
|
|
|
|
def ask_for_options(self):
|
|
|
|
super(ServerCertInstall, self).ask_for_options()
|
|
|
|
|
2013-08-27 11:06:24 -05:00
|
|
|
if self.options.dirsrv and not self.options.dirman_password:
|
|
|
|
self.options.dirman_password = installutils.read_password(
|
2013-07-04 10:45:42 -05:00
|
|
|
"Directory Manager", confirm=False, validate=False, retry=False)
|
2013-08-27 11:06:24 -05:00
|
|
|
if self.options.dirman_password is None:
|
2013-07-04 10:45:42 -05:00
|
|
|
raise admintool.ScriptError(
|
|
|
|
"Directory Manager password required")
|
|
|
|
|
2013-08-20 01:44:58 -05:00
|
|
|
if not self.options.pin:
|
|
|
|
self.options.pin = installutils.read_password(
|
|
|
|
"Enter %s unlock" % self.args[0], confirm=False, validate=False)
|
|
|
|
if self.options.pin is None:
|
|
|
|
raise admintool.ScriptError(
|
|
|
|
"%s unlock password required" % self.args[0])
|
|
|
|
|
2013-07-04 10:45:42 -05:00
|
|
|
def run(self):
|
|
|
|
api.bootstrap(in_server=True)
|
|
|
|
api.finalize()
|
|
|
|
|
|
|
|
self.pkcs12_fname = self.args[0]
|
|
|
|
|
|
|
|
if self.options.dirsrv:
|
|
|
|
self.install_dirsrv_cert()
|
|
|
|
|
|
|
|
if self.options.http:
|
|
|
|
self.install_http_cert()
|
|
|
|
|
|
|
|
def install_dirsrv_cert(self):
|
|
|
|
serverid = dsinstance.realm_to_serverid(api.env.realm)
|
|
|
|
dirname = dsinstance.config_dirname(serverid)
|
|
|
|
|
|
|
|
conn = ldap2(shared_instance=False, base_dn='')
|
|
|
|
conn.connect(bind_dn=DN(('cn', 'directory manager')),
|
2013-08-27 11:06:24 -05:00
|
|
|
bind_pw=self.options.dirman_password)
|
2013-07-04 10:45:42 -05:00
|
|
|
|
2013-07-15 03:12:14 -05:00
|
|
|
entry = conn.get_entry(DN(('cn', 'RSA'), ('cn', 'encryption'),
|
|
|
|
('cn', 'config')),
|
|
|
|
['nssslpersonalityssl'])
|
|
|
|
old_cert = entry.single_value('nssslpersonalityssl')
|
|
|
|
|
2013-08-20 01:39:39 -05:00
|
|
|
server_cert = self.import_cert(dirname, self.options.pin,
|
2013-07-15 03:12:50 -05:00
|
|
|
old_cert, 'ldap/%s' % api.env.host,
|
|
|
|
'restart_dirsrv %s' % serverid)
|
2013-07-15 03:12:14 -05:00
|
|
|
|
|
|
|
entry['nssslpersonalityssl'] = [server_cert]
|
2013-07-04 11:43:08 -05:00
|
|
|
try:
|
|
|
|
conn.update_entry(entry)
|
|
|
|
except errors.EmptyModlist:
|
|
|
|
pass
|
2013-07-04 10:45:42 -05:00
|
|
|
|
|
|
|
conn.disconnect()
|
|
|
|
|
|
|
|
def install_http_cert(self):
|
|
|
|
dirname = certs.NSS_DIR
|
|
|
|
|
2013-07-15 03:12:14 -05:00
|
|
|
old_cert = installutils.get_directive(httpinstance.NSS_CONF,
|
|
|
|
'NSSNickname')
|
|
|
|
|
2013-08-20 01:39:39 -05:00
|
|
|
server_cert = self.import_cert(dirname, self.options.pin,
|
2013-07-15 03:12:50 -05:00
|
|
|
old_cert, 'HTTP/%s' % api.env.host,
|
|
|
|
'restart_httpd')
|
2013-07-04 10:45:42 -05:00
|
|
|
|
|
|
|
installutils.set_directive(httpinstance.NSS_CONF,
|
|
|
|
'NSSNickname', server_cert)
|
|
|
|
|
|
|
|
# Fix the database permissions
|
|
|
|
os.chmod(os.path.join(dirname, 'cert8.db'), 0640)
|
|
|
|
os.chmod(os.path.join(dirname, 'key3.db'), 0640)
|
|
|
|
os.chmod(os.path.join(dirname, 'secmod.db'), 0640)
|
|
|
|
|
|
|
|
pent = pwd.getpwnam("apache")
|
|
|
|
os.chown(os.path.join(dirname, 'cert8.db'), 0, pent.pw_gid)
|
|
|
|
os.chown(os.path.join(dirname, 'key3.db'), 0, pent.pw_gid)
|
|
|
|
os.chown(os.path.join(dirname, 'secmod.db'), 0, pent.pw_gid)
|
|
|
|
|
2013-07-15 03:12:50 -05:00
|
|
|
def import_cert(self, dirname, pkcs12_passwd, old_cert, principal, command):
|
2013-07-04 10:45:42 -05:00
|
|
|
pw = write_tmp_file(pkcs12_passwd)
|
|
|
|
server_cert = installutils.check_pkcs12(
|
|
|
|
pkcs12_info=(self.pkcs12_fname, pw.name),
|
|
|
|
ca_file=CACERT,
|
|
|
|
hostname=api.env.host)
|
|
|
|
|
|
|
|
cdb = certs.CertDB(api.env.realm, nssdir=dirname)
|
|
|
|
try:
|
2013-07-15 03:12:50 -05:00
|
|
|
if api.env.enable_ra:
|
|
|
|
cdb.untrack_server_cert(old_cert)
|
|
|
|
|
2013-07-15 03:12:14 -05:00
|
|
|
cdb.delete_cert(old_cert)
|
|
|
|
cdb.import_pkcs12(self.pkcs12_fname, pw.name)
|
2013-07-15 03:12:50 -05:00
|
|
|
|
|
|
|
if api.env.enable_ra:
|
|
|
|
cdb.track_server_cert(server_cert, principal, cdb.passwd_fname,
|
|
|
|
command)
|
2013-07-04 10:45:42 -05:00
|
|
|
except RuntimeError, e:
|
|
|
|
raise admintool.ScriptError(str(e))
|
|
|
|
|
|
|
|
return server_cert
|