2012-12-18 06:42:18 -06:00
|
|
|
# Authors: Karl MacMillan <kmacmillan@mentalrootkit.com>
|
|
|
|
# Petr Viktorin <pviktori@redhat.com>
|
|
|
|
#
|
|
|
|
# Copyright (C) 2008-2012 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/>.
|
|
|
|
#
|
|
|
|
|
2015-05-13 11:49:25 -05:00
|
|
|
from __future__ import absolute_import
|
2015-08-12 06:44:11 -05:00
|
|
|
from __future__ import print_function
|
2015-05-13 11:49:25 -05:00
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import tempfile
|
2014-09-19 08:57:44 -05:00
|
|
|
import time
|
2016-11-23 03:04:43 -06:00
|
|
|
# pylint: disable=deprecated-module
|
2014-09-24 09:41:47 -05:00
|
|
|
from optparse import OptionGroup, SUPPRESS_HELP
|
2016-11-23 03:04:43 -06:00
|
|
|
# pylint: enable=deprecated-module
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2014-09-19 08:57:44 -05:00
|
|
|
import dns.resolver
|
2016-08-24 06:37:30 -05:00
|
|
|
# pylint: disable=import-error
|
2015-09-14 07:03:58 -05:00
|
|
|
from six.moves.configparser import SafeConfigParser
|
2016-08-24 06:37:30 -05:00
|
|
|
# pylint: enable=import-error
|
2014-09-19 08:57:44 -05:00
|
|
|
|
2017-03-17 04:34:08 -05:00
|
|
|
from ipaserver.install import certs, installutils, bindinstance, dsinstance, ca
|
2012-12-18 06:42:18 -06:00
|
|
|
from ipaserver.install.replication import enable_replication_version_checking
|
2016-11-09 08:14:27 -06:00
|
|
|
from ipaserver.install.server.replicainstall import install_ca_cert
|
2012-12-18 06:42:18 -06:00
|
|
|
from ipaserver.install.bindinstance import (
|
|
|
|
add_zone, add_fwd_rr, add_ptr_rr, dns_container_exists)
|
2015-11-09 11:28:47 -06:00
|
|
|
from ipapython import ipautil, admintool
|
2012-12-18 06:42:18 -06:00
|
|
|
from ipapython.dn import DN
|
|
|
|
from ipapython import version
|
|
|
|
from ipalib import api
|
|
|
|
from ipalib import errors
|
2014-05-29 07:47:17 -05:00
|
|
|
from ipaplatform.paths import paths
|
2016-11-22 07:42:33 -06:00
|
|
|
from ipalib.constants import DOMAIN_LEVEL_0
|
2015-10-15 09:07:48 -05:00
|
|
|
|
|
|
|
UNSUPPORTED_DOMAIN_LEVEL_TEMPLATE = """
|
|
|
|
Replica creation using '{command_name}' to generate replica file
|
2015-10-26 11:56:57 -05:00
|
|
|
is supported only in {domain_level}-level IPA domain.
|
2015-10-15 09:07:48 -05:00
|
|
|
|
|
|
|
The current IPA domain level is {curr_domain_level} and thus the replica must
|
|
|
|
be created by promoting an existing IPA client.
|
|
|
|
|
|
|
|
To set up a replica use the following procedure:
|
|
|
|
1.) set up a client on the host using 'ipa-client-install'
|
|
|
|
2.) promote the client to replica running 'ipa-replica-install'
|
|
|
|
*without* replica file specified
|
|
|
|
"""
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
|
|
|
|
class ReplicaPrepare(admintool.AdminTool):
|
|
|
|
command_name = 'ipa-replica-prepare'
|
|
|
|
|
|
|
|
usage = "%prog [options] <replica-fqdn>"
|
|
|
|
|
|
|
|
description = "Prepare a file for replica installation."
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def add_options(cls, parser):
|
|
|
|
super(ReplicaPrepare, cls).add_options(parser, debug_option=True)
|
|
|
|
|
|
|
|
parser.add_option("-p", "--password", dest="password",
|
|
|
|
help="Directory Manager password (for the existing master)")
|
2014-08-27 06:50:21 -05:00
|
|
|
parser.add_option("--ip-address", dest="ip_addresses", type="ip",
|
2014-11-24 07:49:05 -06:00
|
|
|
action="append", default=[], metavar="IP_ADDRESS",
|
|
|
|
help="add A and PTR records of the future replica. This option can be used multiple times")
|
2014-08-27 06:50:21 -05:00
|
|
|
parser.add_option("--reverse-zone", dest="reverse_zones",
|
2014-11-24 07:49:05 -06:00
|
|
|
action="append", default=[], metavar="REVERSE_ZONE",
|
|
|
|
help="the reverse DNS zone to use. This option can be used multiple times")
|
2012-12-18 06:42:18 -06:00
|
|
|
parser.add_option("--no-reverse", dest="no_reverse",
|
|
|
|
action="store_true", default=False,
|
|
|
|
help="do not create reverse DNS zone")
|
2015-12-22 07:53:41 -06:00
|
|
|
parser.add_option("--auto-reverse", dest="auto_reverse", default=False,
|
|
|
|
action="store_true", help="create necessary DNS zones")
|
|
|
|
parser.add_option("--allow-zone-overlap", dest="allow_zone_overlap",
|
|
|
|
action="store_true", default=False, help="create DNS "
|
|
|
|
"zone even if it already exists")
|
2014-05-29 07:47:17 -05:00
|
|
|
parser.add_option("--ca", dest="ca_file", default=paths.CACERT_P12,
|
2012-12-18 06:42:18 -06:00
|
|
|
metavar="FILE",
|
|
|
|
help="location of CA PKCS#12 file, default /root/cacert.p12")
|
2014-09-19 08:57:44 -05:00
|
|
|
parser.add_option('--no-wait-for-dns', dest='wait_for_dns',
|
|
|
|
action='store_false', default=True,
|
|
|
|
help="do not wait until the replica is resolvable in DNS")
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
group = OptionGroup(parser, "SSL certificate options",
|
|
|
|
"Only used if the server was installed using custom SSL certificates")
|
2014-09-24 09:41:47 -05:00
|
|
|
group.add_option("--dirsrv-cert-file", dest="dirsrv_cert_files",
|
|
|
|
action="append", metavar="FILE",
|
|
|
|
help="File containing the Directory Server SSL certificate and private key")
|
|
|
|
group.add_option("--dirsrv_pkcs12", dest="dirsrv_cert_files",
|
|
|
|
action="append",
|
|
|
|
help=SUPPRESS_HELP)
|
|
|
|
group.add_option("--http-cert-file", dest="http_cert_files",
|
|
|
|
action="append", metavar="FILE",
|
|
|
|
help="File containing the Apache Server SSL certificate and private key")
|
|
|
|
group.add_option("--http_pkcs12", dest="http_cert_files",
|
|
|
|
action="append",
|
|
|
|
help=SUPPRESS_HELP)
|
|
|
|
group.add_option("--dirsrv-pin", dest="dirsrv_pin", sensitive=True,
|
|
|
|
metavar="PIN",
|
|
|
|
help="The password to unlock the Directory Server private key")
|
|
|
|
group.add_option("--dirsrv_pin", dest="dirsrv_pin", sensitive=True,
|
|
|
|
help=SUPPRESS_HELP)
|
|
|
|
group.add_option("--http-pin", dest="http_pin", sensitive=True,
|
|
|
|
metavar="PIN",
|
|
|
|
help="The password to unlock the Apache Server private key")
|
|
|
|
group.add_option("--http_pin", dest="http_pin", sensitive=True,
|
|
|
|
help=SUPPRESS_HELP)
|
2014-09-24 09:48:15 -05:00
|
|
|
group.add_option("--dirsrv-cert-name", dest="dirsrv_cert_name",
|
|
|
|
metavar="NAME",
|
|
|
|
help="Name of the Directory Server SSL certificate to install")
|
|
|
|
group.add_option("--http-cert-name", dest="http_cert_name",
|
|
|
|
metavar="NAME",
|
|
|
|
help="Name of the Apache Server SSL certificate to install")
|
2012-12-18 06:42:18 -06:00
|
|
|
parser.add_option_group(group)
|
|
|
|
|
|
|
|
def validate_options(self):
|
|
|
|
options = self.options
|
|
|
|
super(ReplicaPrepare, self).validate_options(needs_root=True)
|
|
|
|
installutils.check_server_configuration()
|
|
|
|
|
2014-08-27 06:50:21 -05:00
|
|
|
if not options.ip_addresses:
|
|
|
|
if options.reverse_zones:
|
2012-12-18 06:42:18 -06:00
|
|
|
self.option_parser.error("You cannot specify a --reverse-zone "
|
|
|
|
"option without the --ip-address option")
|
|
|
|
if options.no_reverse:
|
|
|
|
self.option_parser.error("You cannot specify a --no-reverse "
|
|
|
|
"option without the --ip-address option")
|
2014-08-27 06:50:21 -05:00
|
|
|
elif options.reverse_zones and options.no_reverse:
|
2012-12-18 06:42:18 -06:00
|
|
|
self.option_parser.error("You cannot specify a --reverse-zone "
|
|
|
|
"option together with --no-reverse")
|
|
|
|
|
|
|
|
# If any of the PKCS#12 options are selected, all are required.
|
2014-09-24 09:41:47 -05:00
|
|
|
cert_file_req = (options.dirsrv_cert_files, options.http_cert_files)
|
2017-03-14 08:18:33 -05:00
|
|
|
if any(cert_file_req) and not all(cert_file_req):
|
2012-12-18 06:42:18 -06:00
|
|
|
self.option_parser.error(
|
2017-03-14 08:18:33 -05:00
|
|
|
"--dirsrv-cert-file and --http-cert-file are required if any "
|
|
|
|
"key file options are used."
|
2017-03-01 09:43:20 -06:00
|
|
|
)
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
if len(self.args) < 1:
|
|
|
|
self.option_parser.error(
|
|
|
|
"must provide the fully-qualified name of the replica")
|
|
|
|
elif len(self.args) > 1:
|
|
|
|
self.option_parser.error(
|
|
|
|
"must provide exactly one name for the replica")
|
|
|
|
else:
|
|
|
|
[self.replica_fqdn] = self.args
|
|
|
|
|
2016-12-02 02:10:41 -06:00
|
|
|
api.bootstrap(in_server=True, confdir=paths.ETC_IPA)
|
2012-12-18 06:42:18 -06:00
|
|
|
api.finalize()
|
2016-10-27 03:31:45 -05:00
|
|
|
# Connect to LDAP, connection is closed at the end of run()
|
|
|
|
api.Backend.ldap2.connect()
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2015-10-29 08:53:25 -05:00
|
|
|
self.check_for_supported_domain_level()
|
2015-10-15 09:07:48 -05:00
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
if api.env.host == self.replica_fqdn:
|
|
|
|
raise admintool.ScriptError("You can't create a replica on itself")
|
|
|
|
|
|
|
|
config_dir = dsinstance.config_dirname(
|
2015-04-27 07:42:31 -05:00
|
|
|
installutils.realm_to_serverid(api.env.realm))
|
2012-12-18 06:42:18 -06:00
|
|
|
if not ipautil.dir_exists(config_dir):
|
|
|
|
raise admintool.ScriptError(
|
|
|
|
"could not find directory instance: %s" % config_dir)
|
|
|
|
|
2014-09-24 09:41:47 -05:00
|
|
|
def load_pkcs12(self, cert_files, key_password, key_nickname):
|
|
|
|
return installutils.load_pkcs12(
|
|
|
|
cert_files=cert_files,
|
|
|
|
key_password=key_password,
|
|
|
|
key_nickname=key_nickname,
|
2016-11-22 07:42:33 -06:00
|
|
|
ca_cert_files=[paths.IPA_CA_CRT],
|
2014-09-24 09:41:47 -05:00
|
|
|
host_name=self.replica_fqdn)
|
2013-03-14 07:58:27 -05:00
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
def ask_for_options(self):
|
|
|
|
options = self.options
|
|
|
|
super(ReplicaPrepare, self).ask_for_options()
|
|
|
|
|
|
|
|
# get the directory manager password
|
|
|
|
self.dirman_password = options.password
|
|
|
|
if not options.password:
|
|
|
|
self.dirman_password = installutils.read_password(
|
|
|
|
"Directory Manager (existing master)",
|
|
|
|
confirm=False, validate=False)
|
|
|
|
if self.dirman_password is None:
|
|
|
|
raise admintool.ScriptError(
|
|
|
|
"Directory Manager password required")
|
|
|
|
|
|
|
|
# Try out the password & get the subject base
|
2016-10-27 03:31:45 -05:00
|
|
|
api.Backend.ldap2.disconnect()
|
2012-12-18 06:42:18 -06:00
|
|
|
try:
|
2016-10-27 03:31:45 -05:00
|
|
|
api.Backend.ldap2.connect(bind_pw=self.dirman_password)
|
2015-07-01 02:20:35 -05:00
|
|
|
|
2016-10-27 03:31:45 -05:00
|
|
|
entry_attrs = api.Backend.ldap2.get_ipa_config()
|
2015-07-01 02:20:35 -05:00
|
|
|
self.subject_base = entry_attrs.get(
|
|
|
|
'ipacertificatesubjectbase', [None])[0]
|
|
|
|
|
2014-10-13 07:30:15 -05:00
|
|
|
ca_enabled = api.Command.ca_is_enabled()['result']
|
2012-12-18 06:42:18 -06:00
|
|
|
except errors.ACIError:
|
|
|
|
raise admintool.ScriptError("The password provided is incorrect "
|
2016-10-27 03:31:45 -05:00
|
|
|
"for LDAP server %s" % api.env.host)
|
2012-12-18 06:42:18 -06:00
|
|
|
except errors.LDAPError:
|
|
|
|
raise admintool.ScriptError(
|
|
|
|
"Unable to connect to LDAP server %s" % api.env.host)
|
2015-07-30 09:49:29 -05:00
|
|
|
except errors.DatabaseError as e:
|
2012-12-18 06:42:18 -06:00
|
|
|
raise admintool.ScriptError(e.desc)
|
|
|
|
|
2016-08-01 10:32:04 -05:00
|
|
|
if ca_enabled and not ipautil.file_exists(paths.CA_CS_CFG_PATH):
|
|
|
|
raise admintool.ScriptError(
|
|
|
|
"CA is not installed on this server. "
|
|
|
|
"ipa-replica-prepare must be run on an IPA server with CA.")
|
2014-10-13 07:30:15 -05:00
|
|
|
if not ca_enabled and not options.http_cert_files:
|
|
|
|
raise admintool.ScriptError(
|
|
|
|
"Cannot issue certificates: a CA is not installed. Use the "
|
|
|
|
"--http-cert-file, --dirsrv-cert-file options to provide "
|
|
|
|
"custom certificates.")
|
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
if self.subject_base is not None:
|
|
|
|
self.subject_base = DN(self.subject_base)
|
|
|
|
|
|
|
|
# Validate more options using the password
|
|
|
|
try:
|
|
|
|
installutils.verify_fqdn(self.replica_fqdn, local_hostname=False)
|
2015-07-30 09:49:29 -05:00
|
|
|
except installutils.BadHostError as e:
|
2012-12-18 06:42:18 -06:00
|
|
|
if isinstance(e, installutils.HostLookupError):
|
2014-08-27 06:50:21 -05:00
|
|
|
if not options.ip_addresses:
|
2016-11-11 05:45:11 -06:00
|
|
|
if dns_container_exists(api.env.basedn):
|
2015-06-18 06:07:06 -05:00
|
|
|
self.log.info('You might use the --ip-address option '
|
|
|
|
'to create a DNS entry if the DNS zone '
|
|
|
|
'is managed by IPA.')
|
2012-12-18 06:42:18 -06:00
|
|
|
raise
|
|
|
|
else:
|
|
|
|
# The host doesn't exist in DNS but we're adding it.
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
raise
|
|
|
|
|
2014-08-27 06:50:21 -05:00
|
|
|
if options.ip_addresses:
|
2016-11-11 05:45:11 -06:00
|
|
|
if not dns_container_exists(api.env.basedn):
|
2014-02-21 07:48:05 -06:00
|
|
|
self.log.error(
|
|
|
|
"It is not possible to add a DNS record automatically "
|
|
|
|
"because DNS is not managed by IPA. Please create DNS "
|
|
|
|
"record manually and then omit --ip-address option.")
|
|
|
|
raise admintool.ScriptError("Cannot add DNS record")
|
2014-08-27 06:50:21 -05:00
|
|
|
|
|
|
|
options.reverse_zones = bindinstance.check_reverse_zones(
|
|
|
|
options.ip_addresses, options.reverse_zones, options, False,
|
|
|
|
True)
|
2015-07-02 22:59:55 -05:00
|
|
|
|
2016-10-04 09:54:44 -05:00
|
|
|
_host, zone = self.replica_fqdn.split('.', 1)
|
2015-07-02 22:59:55 -05:00
|
|
|
if not bindinstance.dns_zone_exists(zone, api=api):
|
|
|
|
self.log.error("DNS zone %s does not exist in IPA managed DNS "
|
|
|
|
"server. Either create DNS zone or omit "
|
|
|
|
"--ip-address option." % zone)
|
|
|
|
raise admintool.ScriptError("Cannot add DNS record")
|
|
|
|
|
2017-03-14 08:18:33 -05:00
|
|
|
self.http_pin = self.dirsrv_pin = None
|
2014-09-24 09:41:47 -05:00
|
|
|
|
|
|
|
if options.http_cert_files:
|
2013-09-25 03:40:05 -05:00
|
|
|
if options.http_pin is None:
|
2013-07-09 05:24:14 -05:00
|
|
|
options.http_pin = installutils.read_password(
|
2014-09-24 09:41:47 -05:00
|
|
|
"Enter Apache Server private key unlock",
|
2016-08-16 10:34:06 -05:00
|
|
|
confirm=False, validate=False, retry=False)
|
2013-07-09 05:24:14 -05:00
|
|
|
if options.http_pin is None:
|
|
|
|
raise admintool.ScriptError(
|
2014-09-24 09:41:47 -05:00
|
|
|
"Apache Server private key unlock password required")
|
|
|
|
http_pkcs12_file, http_pin, http_ca_cert = self.load_pkcs12(
|
2014-09-24 09:48:15 -05:00
|
|
|
options.http_cert_files, options.http_pin,
|
|
|
|
options.http_cert_name)
|
2014-09-24 09:41:47 -05:00
|
|
|
self.http_pkcs12_file = http_pkcs12_file
|
|
|
|
self.http_pin = http_pin
|
2013-07-09 05:24:14 -05:00
|
|
|
|
2014-09-24 09:41:47 -05:00
|
|
|
if options.dirsrv_cert_files:
|
2013-09-25 03:40:05 -05:00
|
|
|
if options.dirsrv_pin is None:
|
2013-07-09 05:24:14 -05:00
|
|
|
options.dirsrv_pin = installutils.read_password(
|
2014-09-24 09:41:47 -05:00
|
|
|
"Enter Directory Server private key unlock",
|
2016-08-16 10:34:06 -05:00
|
|
|
confirm=False, validate=False, retry=False)
|
2013-07-09 05:24:14 -05:00
|
|
|
if options.dirsrv_pin is None:
|
|
|
|
raise admintool.ScriptError(
|
2014-09-24 09:41:47 -05:00
|
|
|
"Directory Server private key unlock password required")
|
|
|
|
dirsrv_pkcs12_file, dirsrv_pin, dirsrv_ca_cert = self.load_pkcs12(
|
2014-09-24 09:48:15 -05:00
|
|
|
options.dirsrv_cert_files, options.dirsrv_pin,
|
|
|
|
options.dirsrv_cert_name)
|
2014-09-24 09:41:47 -05:00
|
|
|
self.dirsrv_pkcs12_file = dirsrv_pkcs12_file
|
|
|
|
self.dirsrv_pin = dirsrv_pin
|
2013-07-09 05:24:14 -05:00
|
|
|
|
2014-09-24 09:41:47 -05:00
|
|
|
if (options.http_cert_files and options.dirsrv_cert_files and
|
2014-08-05 02:06:39 -05:00
|
|
|
http_ca_cert != dirsrv_ca_cert):
|
|
|
|
raise admintool.ScriptError(
|
2014-09-24 09:41:47 -05:00
|
|
|
"Apache Server SSL certificate and Directory Server SSL "
|
|
|
|
"certificate are not signed by the same CA certificate")
|
2014-08-05 02:06:39 -05:00
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
def run(self):
|
|
|
|
options = self.options
|
|
|
|
super(ReplicaPrepare, self).run()
|
|
|
|
|
|
|
|
self.log.info("Preparing replica for %s from %s",
|
|
|
|
self.replica_fqdn, api.env.host)
|
2016-11-01 08:52:33 -05:00
|
|
|
enable_replication_version_checking(
|
|
|
|
api.env.realm,
|
2012-12-18 06:42:18 -06:00
|
|
|
self.dirman_password)
|
|
|
|
|
|
|
|
self.top_dir = tempfile.mkdtemp("ipa")
|
|
|
|
self.dir = os.path.join(self.top_dir, "realm_info")
|
2015-12-09 06:40:04 -06:00
|
|
|
os.mkdir(self.dir)
|
|
|
|
os.chmod(self.dir, 0o700)
|
2012-12-18 06:42:18 -06:00
|
|
|
try:
|
|
|
|
self.copy_ds_certificate()
|
|
|
|
self.copy_httpd_certificate()
|
|
|
|
|
2016-11-09 08:14:27 -06:00
|
|
|
self.retrieve_ca_certs()
|
2012-12-18 06:42:18 -06:00
|
|
|
self.copy_misc_files()
|
|
|
|
|
|
|
|
self.save_config()
|
|
|
|
|
|
|
|
self.package_replica_file()
|
|
|
|
finally:
|
|
|
|
shutil.rmtree(self.top_dir)
|
|
|
|
|
2014-08-27 06:50:21 -05:00
|
|
|
if options.ip_addresses:
|
2012-12-18 06:42:18 -06:00
|
|
|
self.add_dns_records()
|
|
|
|
|
2014-09-19 08:57:44 -05:00
|
|
|
if options.wait_for_dns:
|
|
|
|
self.wait_for_dns()
|
|
|
|
|
2016-10-27 03:31:45 -05:00
|
|
|
# Close LDAP connection that was opened in validate_options()
|
|
|
|
api.Backend.ldap2.disconnect()
|
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
def copy_ds_certificate(self):
|
|
|
|
options = self.options
|
|
|
|
|
|
|
|
passwd_fname = os.path.join(self.dir, "dirsrv_pin.txt")
|
|
|
|
with open(passwd_fname, "w") as fd:
|
2014-09-24 09:41:47 -05:00
|
|
|
fd.write("%s\n" % (self.dirsrv_pin or ''))
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2014-09-24 09:41:47 -05:00
|
|
|
if options.dirsrv_cert_files:
|
|
|
|
self.log.info("Copying SSL certificate for the Directory Server")
|
|
|
|
self.copy_info_file(self.dirsrv_pkcs12_file.name, "dscert.p12")
|
2012-12-18 06:42:18 -06:00
|
|
|
else:
|
2013-03-27 08:25:18 -05:00
|
|
|
if ipautil.file_exists(options.ca_file):
|
2013-05-15 04:22:41 -05:00
|
|
|
# Since it is possible that the Directory Manager password
|
|
|
|
# has changed since ipa-server-install, we need to regenerate
|
|
|
|
# the CA PKCS#12 file and update the pki admin user password
|
|
|
|
self.regenerate_ca_file(options.ca_file)
|
|
|
|
self.update_pki_admin_password()
|
2013-03-27 08:25:18 -05:00
|
|
|
self.copy_info_file(options.ca_file, "cacert.p12")
|
|
|
|
else:
|
|
|
|
raise admintool.ScriptError("Root CA PKCS#12 not "
|
|
|
|
"found in %s" % options.ca_file)
|
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
self.log.info(
|
|
|
|
"Creating SSL certificate for the Directory Server")
|
|
|
|
self.export_certdb("dscert", passwd_fname)
|
|
|
|
|
2014-09-24 09:41:47 -05:00
|
|
|
if not options.dirsrv_cert_files:
|
2012-12-18 06:42:18 -06:00
|
|
|
self.log.info(
|
|
|
|
"Creating SSL certificate for the dogtag Directory Server")
|
|
|
|
self.export_certdb("dogtagcert", passwd_fname)
|
|
|
|
self.log.info("Saving dogtag Directory Server port")
|
|
|
|
port_fname = os.path.join(
|
|
|
|
self.dir, "dogtag_directory_port.txt")
|
|
|
|
with open(port_fname, "w") as fd:
|
2015-11-09 11:28:47 -06:00
|
|
|
fd.write("389\n")
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
def copy_httpd_certificate(self):
|
|
|
|
options = self.options
|
|
|
|
|
|
|
|
passwd_fname = os.path.join(self.dir, "http_pin.txt")
|
|
|
|
with open(passwd_fname, "w") as fd:
|
2014-09-24 09:41:47 -05:00
|
|
|
fd.write("%s\n" % (self.http_pin or ''))
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2014-09-24 09:41:47 -05:00
|
|
|
if options.http_cert_files:
|
|
|
|
self.log.info("Copying SSL certificate for the Web Server")
|
|
|
|
self.copy_info_file(self.http_pkcs12_file.name, "httpcert.p12")
|
2012-12-18 06:42:18 -06:00
|
|
|
else:
|
|
|
|
self.log.info("Creating SSL certificate for the Web Server")
|
|
|
|
self.export_certdb("httpcert", passwd_fname)
|
|
|
|
|
|
|
|
self.log.info("Exporting RA certificate")
|
2013-03-27 08:25:18 -05:00
|
|
|
self.export_ra_pkcs12()
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
def copy_misc_files(self):
|
|
|
|
self.log.info("Copying additional files")
|
|
|
|
|
2014-05-29 07:47:17 -05:00
|
|
|
cacert_filename = paths.CACERT_PEM
|
2012-12-18 06:42:18 -06:00
|
|
|
if ipautil.file_exists(cacert_filename):
|
|
|
|
self.copy_info_file(cacert_filename, "cacert.pem")
|
2014-03-18 10:23:30 -05:00
|
|
|
self.copy_info_file(paths.IPA_DEFAULT_CONF, "default.conf")
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2016-11-09 08:14:27 -06:00
|
|
|
def retrieve_ca_certs(self):
|
|
|
|
self.log.info("Retrieving CA certificates")
|
|
|
|
dest = os.path.join(self.dir, "ca.crt")
|
|
|
|
install_ca_cert(api.Backend.ldap2, api.env.basedn,
|
|
|
|
api.env.realm, paths.IPA_CA_CRT, destfile=dest)
|
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
def save_config(self):
|
|
|
|
self.log.info("Finalizing configuration")
|
|
|
|
|
|
|
|
config = SafeConfigParser()
|
|
|
|
config.add_section("realm")
|
|
|
|
config.set("realm", "realm_name", api.env.realm)
|
|
|
|
config.set("realm", "master_host_name", api.env.host)
|
|
|
|
config.set("realm", "domain_name", api.env.domain)
|
|
|
|
config.set("realm", "destination_host", self.replica_fqdn)
|
|
|
|
config.set("realm", "subject_base", str(self.subject_base))
|
|
|
|
config.set("realm", "version", str(version.NUM_VERSION))
|
|
|
|
|
|
|
|
with open(os.path.join(self.dir, "realm_info"), "w") as fd:
|
|
|
|
config.write(fd)
|
|
|
|
|
|
|
|
def package_replica_file(self):
|
2014-05-29 07:47:17 -05:00
|
|
|
replicafile = paths.REPLICA_INFO_TEMPLATE % self.replica_fqdn
|
2012-12-18 06:42:18 -06:00
|
|
|
encfile = "%s.gpg" % replicafile
|
|
|
|
|
|
|
|
self.log.info("Packaging replica information into %s", encfile)
|
|
|
|
ipautil.run(
|
2014-05-29 07:47:17 -05:00
|
|
|
[paths.TAR, "cf", replicafile, "-C", self.top_dir, "realm_info"])
|
2016-11-22 08:38:43 -06:00
|
|
|
installutils.encrypt_file(
|
2012-12-18 06:42:18 -06:00
|
|
|
replicafile, encfile, self.dirman_password, self.top_dir)
|
|
|
|
|
2015-07-15 09:38:06 -05:00
|
|
|
os.chmod(encfile, 0o600)
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
installutils.remove_file(replicafile)
|
|
|
|
|
|
|
|
def add_dns_records(self):
|
|
|
|
options = self.options
|
|
|
|
|
|
|
|
self.log.info("Adding DNS records for %s", self.replica_fqdn)
|
|
|
|
name, domain = self.replica_fqdn.split(".", 1)
|
|
|
|
|
2014-08-27 06:50:21 -05:00
|
|
|
for reverse_zone in options.reverse_zones:
|
|
|
|
self.log.info("Adding reverse zone %s", reverse_zone)
|
|
|
|
add_zone(reverse_zone)
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2014-08-27 06:50:21 -05:00
|
|
|
for ip in options.ip_addresses:
|
|
|
|
ip_address = str(ip)
|
2012-12-18 06:42:18 -06:00
|
|
|
try:
|
2014-08-27 06:50:21 -05:00
|
|
|
add_fwd_rr(domain, name, ip_address)
|
2015-07-30 09:49:29 -05:00
|
|
|
except errors.PublicError as e:
|
2012-12-18 06:42:18 -06:00
|
|
|
raise admintool.ScriptError(
|
2015-06-18 06:13:45 -05:00
|
|
|
"Could not add A/AAAA DNS record for the replica: %s" % e)
|
2014-08-27 06:50:21 -05:00
|
|
|
|
|
|
|
if not options.no_reverse:
|
|
|
|
reverse_zone = bindinstance.find_reverse_zone(ip)
|
2016-04-15 10:05:11 -05:00
|
|
|
if reverse_zone is None:
|
|
|
|
self.log.warning(
|
|
|
|
"Could not find any IPA managed reverse zone. "
|
|
|
|
"Not creating PTR records")
|
|
|
|
return
|
2014-08-27 06:50:21 -05:00
|
|
|
try:
|
|
|
|
add_ptr_rr(reverse_zone, ip_address, self.replica_fqdn)
|
2015-07-30 09:49:29 -05:00
|
|
|
except errors.PublicError as e:
|
2014-08-27 06:50:21 -05:00
|
|
|
raise admintool.ScriptError(
|
2015-06-18 06:13:45 -05:00
|
|
|
"Could not add PTR DNS record for the replica: %s"
|
2014-08-27 06:50:21 -05:00
|
|
|
% e)
|
2012-12-18 06:42:18 -06:00
|
|
|
|
2014-09-19 08:57:44 -05:00
|
|
|
def check_dns(self, replica_fqdn):
|
|
|
|
"""Return true if the replica hostname is resolvable"""
|
|
|
|
resolver = dns.resolver.Resolver()
|
|
|
|
exceptions = (dns.resolver.NXDOMAIN, dns.resolver.NoAnswer,
|
|
|
|
dns.resolver.Timeout, dns.resolver.NoNameservers)
|
|
|
|
|
|
|
|
try:
|
2016-10-04 09:54:44 -05:00
|
|
|
resolver.query(replica_fqdn, 'A', 'IN')
|
2014-09-19 08:57:44 -05:00
|
|
|
except exceptions:
|
|
|
|
try:
|
2016-10-04 09:54:44 -05:00
|
|
|
resolver.query(replica_fqdn, 'AAAA', 'IN')
|
2014-09-19 08:57:44 -05:00
|
|
|
except exceptions:
|
|
|
|
return False
|
|
|
|
except Exception as e:
|
2016-01-15 09:25:33 -06:00
|
|
|
self.log.warning('Exception while waiting for DNS record: %s: %s',
|
2014-09-19 08:57:44 -05:00
|
|
|
type(e).__name__, e)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
def wait_for_dns(self):
|
|
|
|
# Make sure replica_fqdn has a trailing dot, so the
|
|
|
|
# 'search' directive in /etc/resolv.conf doesn't apply
|
|
|
|
replica_fqdn = self.replica_fqdn
|
|
|
|
if not replica_fqdn.endswith('.'):
|
|
|
|
replica_fqdn += '.'
|
|
|
|
|
|
|
|
if self.check_dns(replica_fqdn):
|
|
|
|
self.log.debug('%s A/AAAA record resolvable', replica_fqdn)
|
|
|
|
return
|
|
|
|
|
|
|
|
self.log.info('Waiting for %s A or AAAA record to be resolvable',
|
|
|
|
replica_fqdn)
|
2015-08-12 06:44:11 -05:00
|
|
|
print('This can be safely interrupted (Ctrl+C)')
|
2014-09-19 08:57:44 -05:00
|
|
|
|
|
|
|
try:
|
|
|
|
while not self.check_dns(replica_fqdn):
|
|
|
|
time.sleep(1)
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
self.log.info('Interrupted')
|
|
|
|
else:
|
|
|
|
self.log.debug('%s A/AAAA record resolvable', replica_fqdn)
|
|
|
|
|
2012-12-18 06:42:18 -06:00
|
|
|
def copy_info_file(self, source, dest):
|
|
|
|
"""Copy a file into the info directory
|
|
|
|
|
|
|
|
:param source: The source file (an absolute path)
|
|
|
|
:param dest: The destination file (relative to the info directory)
|
|
|
|
"""
|
|
|
|
dest_path = os.path.join(self.dir, dest)
|
|
|
|
self.log.debug('Copying %s to %s', source, dest_path)
|
|
|
|
try:
|
|
|
|
shutil.copy(source, dest_path)
|
2015-07-30 09:49:29 -05:00
|
|
|
except IOError as e:
|
2012-12-18 06:42:18 -06:00
|
|
|
raise admintool.ScriptError("File copy failed: %s" % e)
|
|
|
|
|
|
|
|
def remove_info_file(self, filename):
|
|
|
|
"""Remove a file from the info directory
|
|
|
|
|
|
|
|
:param filename: The unneeded file (relative to the info directory)
|
|
|
|
"""
|
|
|
|
installutils.remove_file(os.path.join(self.dir, filename))
|
|
|
|
|
2017-03-14 08:18:33 -05:00
|
|
|
def export_certdb(self, fname, passwd_fname):
|
2012-12-18 06:42:18 -06:00
|
|
|
"""Export a cert database
|
|
|
|
|
|
|
|
:param fname: The file to export to (relative to the info directory)
|
|
|
|
:param passwd_fname: File that holds the cert DB password
|
|
|
|
"""
|
|
|
|
hostname = self.replica_fqdn
|
|
|
|
subject_base = self.subject_base
|
2017-03-17 04:34:08 -05:00
|
|
|
ca_subject = ca.lookup_ca_subject(api, subject_base)
|
2017-03-14 08:18:33 -05:00
|
|
|
nickname = "Server-Cert"
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
try:
|
|
|
|
db = certs.CertDB(
|
2017-03-17 04:34:08 -05:00
|
|
|
api.env.realm, nssdir=self.dir, host_name=api.env.host,
|
|
|
|
subject_base=subject_base, ca_subject=ca_subject)
|
2012-12-18 06:42:18 -06:00
|
|
|
db.create_passwd_file()
|
2016-12-13 09:32:32 -06:00
|
|
|
db.create_from_cacert()
|
2016-12-20 03:23:47 -06:00
|
|
|
db.create_server_cert(nickname, hostname)
|
2012-12-18 06:42:18 -06:00
|
|
|
|
|
|
|
pkcs12_fname = os.path.join(self.dir, fname + ".p12")
|
|
|
|
|
|
|
|
try:
|
2017-03-14 08:18:33 -05:00
|
|
|
db.export_pkcs12(pkcs12_fname, passwd_fname, nickname)
|
2015-07-30 09:49:29 -05:00
|
|
|
except ipautil.CalledProcessError as e:
|
2012-12-18 06:42:18 -06:00
|
|
|
self.log.info("error exporting Server certificate: %s", e)
|
|
|
|
installutils.remove_file(pkcs12_fname)
|
|
|
|
installutils.remove_file(passwd_fname)
|
|
|
|
|
|
|
|
self.remove_info_file("cert8.db")
|
|
|
|
self.remove_info_file("key3.db")
|
|
|
|
self.remove_info_file("secmod.db")
|
|
|
|
self.remove_info_file("noise.txt")
|
|
|
|
|
|
|
|
orig_filename = passwd_fname + ".orig"
|
|
|
|
if ipautil.file_exists(orig_filename):
|
|
|
|
installutils.remove_file(orig_filename)
|
2015-07-30 09:49:29 -05:00
|
|
|
except errors.CertificateOperationError as e:
|
2012-12-18 06:42:18 -06:00
|
|
|
raise admintool.ScriptError(str(e))
|
|
|
|
|
|
|
|
def export_ra_pkcs12(self):
|
2017-01-13 02:08:42 -06:00
|
|
|
if (os.path.exists(paths.RA_AGENT_PEM) and
|
|
|
|
os.path.exists(paths.RA_AGENT_KEY)):
|
2017-04-19 04:42:40 -05:00
|
|
|
with ipautil.write_tmp_file(self.dirman_password) as f:
|
|
|
|
ipautil.run([
|
|
|
|
paths.OPENSSL,
|
|
|
|
"pkcs12", "-export",
|
|
|
|
"-inkey", paths.RA_AGENT_KEY,
|
|
|
|
"-in", paths.RA_AGENT_PEM,
|
|
|
|
"-out", os.path.join(self.dir, "ra.p12"),
|
|
|
|
"-passout", "file:{pwfile}".format(pwfile=f.name)
|
|
|
|
])
|
2013-05-15 04:22:41 -05:00
|
|
|
|
|
|
|
def update_pki_admin_password(self):
|
|
|
|
dn = DN('uid=admin', 'ou=people', 'o=ipaca')
|
2016-10-27 03:31:45 -05:00
|
|
|
api.Backend.ldap2.modify_password(dn, self.dirman_password)
|
2013-05-15 04:22:41 -05:00
|
|
|
|
|
|
|
def regenerate_ca_file(self, ca_file):
|
|
|
|
dm_pwd_fd = ipautil.write_tmp_file(self.dirman_password)
|
|
|
|
|
|
|
|
keydb_pwd = ''
|
2014-05-29 07:47:17 -05:00
|
|
|
with open(paths.PKI_TOMCAT_PASSWORD_CONF) as f:
|
2013-05-15 04:22:41 -05:00
|
|
|
for line in f.readlines():
|
|
|
|
key, value = line.strip().split('=')
|
|
|
|
if key == 'internal':
|
|
|
|
keydb_pwd = value
|
|
|
|
break
|
|
|
|
|
|
|
|
keydb_pwd_fd = ipautil.write_tmp_file(keydb_pwd)
|
|
|
|
|
|
|
|
ipautil.run([
|
2014-05-29 07:47:17 -05:00
|
|
|
paths.PKCS12EXPORT,
|
|
|
|
'-d', paths.PKI_TOMCAT_ALIAS_DIR,
|
2013-05-15 04:22:41 -05:00
|
|
|
'-p', keydb_pwd_fd.name,
|
|
|
|
'-w', dm_pwd_fd.name,
|
|
|
|
'-o', ca_file
|
|
|
|
])
|
2015-10-15 09:07:48 -05:00
|
|
|
|
2015-10-29 08:53:25 -05:00
|
|
|
def check_for_supported_domain_level(self):
|
|
|
|
"""
|
|
|
|
check if we are in 0-level topology. If not, raise an error pointing
|
|
|
|
the user to the replica promotion pathway
|
|
|
|
"""
|
|
|
|
|
2015-10-15 09:07:48 -05:00
|
|
|
domain_level = dsinstance.get_domain_level(api)
|
2015-10-26 11:56:57 -05:00
|
|
|
if domain_level > DOMAIN_LEVEL_0:
|
2015-10-29 08:53:25 -05:00
|
|
|
self.log.error(
|
2015-10-15 09:07:48 -05:00
|
|
|
UNSUPPORTED_DOMAIN_LEVEL_TEMPLATE.format(
|
|
|
|
command_name=self.command_name,
|
2015-10-26 11:56:57 -05:00
|
|
|
domain_level=DOMAIN_LEVEL_0,
|
2015-10-29 08:53:25 -05:00
|
|
|
curr_domain_level=domain_level
|
|
|
|
)
|
|
|
|
)
|
|
|
|
raise errors.InvalidDomainLevelError(
|
|
|
|
reason="'{command}' is allowed only in domain level "
|
|
|
|
"{prep_domain_level}".format(
|
|
|
|
command=self.command_name,
|
|
|
|
prep_domain_level=DOMAIN_LEVEL_0
|
|
|
|
)
|
2015-10-15 09:07:48 -05:00
|
|
|
)
|