0000-12-31 18:09:24 -05:50
|
|
|
#! /usr/bin/python -E
|
|
|
|
# Authors: Karl MacMillan <kmacmillan@mentalrootkit.com>
|
|
|
|
#
|
|
|
|
# Copyright (C) 2007 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
|
2008-02-04 14:15:52 -06:00
|
|
|
# published by the Free Software Foundation; version 2 only
|
0000-12-31 18:09:24 -05:50
|
|
|
#
|
|
|
|
# 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, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
#
|
|
|
|
|
|
|
|
import sys
|
|
|
|
|
|
|
|
import logging, tempfile, shutil, os, pwd
|
2008-02-05 11:23:53 -06:00
|
|
|
import traceback
|
0000-12-31 18:09:24 -05:50
|
|
|
from ConfigParser import SafeConfigParser
|
|
|
|
import krbV
|
2008-02-05 11:23:53 -06:00
|
|
|
from optparse import OptionParser
|
0000-12-31 18:09:24 -05:50
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
import ipa.config
|
0000-12-31 18:09:24 -05:50
|
|
|
from ipa import ipautil
|
2008-02-15 19:47:29 -06:00
|
|
|
from ipaserver import dsinstance, installutils, certs, ipaldap
|
|
|
|
import ldap
|
0000-12-31 18:09:24 -05:50
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
def usage():
|
|
|
|
print "ipa-replica-prepate FQDN (e.g. replica.example.com)"
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
def parse_options():
|
|
|
|
parser = OptionParser()
|
|
|
|
|
|
|
|
args = ipa.config.init_config(sys.argv)
|
|
|
|
options, args = parser.parse_args(args)
|
|
|
|
|
|
|
|
if len(args) != 2:
|
|
|
|
parser.error("must provide the fully-qualified name of the replica")
|
|
|
|
|
|
|
|
return options, args
|
|
|
|
|
0000-12-31 18:09:24 -05:50
|
|
|
def get_host_name():
|
|
|
|
hostname = installutils.get_fqdn()
|
|
|
|
try:
|
|
|
|
installutils.verify_fqdn(hostname)
|
|
|
|
except RuntimeError, e:
|
|
|
|
logging.error(str(e))
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
return hostname
|
|
|
|
|
|
|
|
def get_realm_name():
|
2008-02-15 19:47:29 -06:00
|
|
|
try:
|
|
|
|
c = krbV.default_context()
|
|
|
|
return c.default_realm
|
|
|
|
except Exception, e:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_domain_name():
|
|
|
|
try:
|
|
|
|
conn = ipaldap.IPAdmin("127.0.0.1")
|
|
|
|
conn.simple_bind_s("", "")
|
|
|
|
|
|
|
|
context = conn.getEntry("", ldap.SCOPE_BASE, '(objectclass=*)', [ 'namingContexts' ])
|
|
|
|
conn.unbind()
|
|
|
|
except Exception, e:
|
|
|
|
return None
|
|
|
|
|
|
|
|
domain_name = context.getValue('namingContexts')
|
|
|
|
domain_name = domain_name.replace('dc=','')
|
|
|
|
domain_name = domain_name.replace(',','.')
|
|
|
|
|
|
|
|
return domain_name
|
0000-12-31 18:09:24 -05:50
|
|
|
|
|
|
|
def check_ipa_configuration(realm_name):
|
2008-02-05 11:23:53 -06:00
|
|
|
config_dir = dsinstance.config_dirname(dsinstance.realm_to_serverid(realm_name))
|
0000-12-31 18:09:24 -05:50
|
|
|
if not ipautil.dir_exists(config_dir):
|
|
|
|
logging.error("could not find directory instance: %s" % config_dir)
|
|
|
|
sys.exit(1)
|
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
def export_certdb(realm_name, ds_dir, dir, fname, subject):
|
|
|
|
"""realm is the kerberos realm for the IPA server.
|
|
|
|
ds_dir is the location of the master DS we are creating a replica for.
|
|
|
|
dir is the location of the files for the replica we are creating.
|
|
|
|
fname is the filename of the PKCS#12 file for this cert (minus the .p12).
|
|
|
|
subject is the subject of the certificate we are creating
|
|
|
|
"""
|
2008-02-20 15:31:32 -06:00
|
|
|
try:
|
|
|
|
ds_ca = certs.CertDB(dsinstance.config_dirname(dsinstance.realm_to_serverid(realm_name)))
|
|
|
|
ca = certs.CertDB(dir)
|
|
|
|
ca.create_from_cacert(ds_ca.cacert_fname)
|
|
|
|
ca.create_server_cert("Server-Cert", subject, ds_ca)
|
|
|
|
except Exception, e:
|
|
|
|
raise e
|
2008-02-05 11:23:53 -06:00
|
|
|
|
|
|
|
pkcs12_fname = dir + "/" + fname + ".p12"
|
0000-12-31 18:09:24 -05:50
|
|
|
passwd_fname = dir + "/pwdfile.txt"
|
|
|
|
fd = open(passwd_fname, "w")
|
|
|
|
fd.write("\n")
|
|
|
|
fd.close()
|
0000-12-31 18:09:24 -05:50
|
|
|
|
0000-12-31 18:09:24 -05:50
|
|
|
try:
|
2008-02-05 11:23:53 -06:00
|
|
|
ca.export_pkcs12(pkcs12_fname, passwd_fname, "Server-Cert")
|
0000-12-31 18:09:24 -05:50
|
|
|
except ipautil.CalledProcessError, e:
|
|
|
|
print "error exporting CA certificate: " + str(e)
|
|
|
|
try:
|
|
|
|
os.unlink(pkcs12_fname)
|
|
|
|
os.unlink(passwd_fname)
|
|
|
|
except:
|
|
|
|
pass
|
0000-12-31 18:09:24 -05:50
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
os.unlink(dir + "/cert8.db")
|
|
|
|
os.unlink(dir + "/key3.db")
|
|
|
|
os.unlink(dir + "/secmod.db")
|
2008-02-15 19:47:29 -06:00
|
|
|
os.unlink(dir + "/noise.txt")
|
|
|
|
if ipautil.file_exists(passwd_fname + ".orig"):
|
|
|
|
os.unlink(passwd_fname + ".orig")
|
0000-12-31 18:09:24 -05:50
|
|
|
|
|
|
|
def get_ds_user(ds_dir):
|
|
|
|
uid = os.stat(ds_dir).st_uid
|
|
|
|
user = pwd.getpwuid(uid)[0]
|
|
|
|
|
|
|
|
return user
|
|
|
|
|
2008-02-15 19:47:29 -06:00
|
|
|
def save_config(dir, realm_name, host_name, ds_user, domain_name):
|
0000-12-31 18:09:24 -05:50
|
|
|
config = SafeConfigParser()
|
|
|
|
config.add_section("realm")
|
|
|
|
config.set("realm", "realm_name", realm_name)
|
|
|
|
config.set("realm", "master_host_name", host_name)
|
|
|
|
config.set("realm", "ds_user", ds_user)
|
2008-02-15 19:47:29 -06:00
|
|
|
config.set("realm", "domain_name", domain_name)
|
0000-12-31 18:09:24 -05:50
|
|
|
fd = open(dir + "/realm_info", "w")
|
|
|
|
config.write(fd)
|
0000-12-31 18:09:24 -05:50
|
|
|
|
|
|
|
def copy_files(realm_name, dir):
|
2008-02-05 11:23:53 -06:00
|
|
|
config_dir = dsinstance.config_dirname(dsinstance.realm_to_serverid(realm_name))
|
|
|
|
try:
|
|
|
|
shutil.copy("/var/kerberos/krb5kdc/ldappwd", dir + "/ldappwd")
|
|
|
|
shutil.copy("/usr/share/ipa/html/preferences.html", dir + "/preferences.html")
|
|
|
|
shutil.copy("/usr/share/ipa/html/configure.jar", dir + "/configure.jar")
|
|
|
|
shutil.copy(config_dir + "/cacert.asc", dir + "/ca.crt")
|
|
|
|
except Exception, e:
|
|
|
|
print "error copying files: " + str(e)
|
|
|
|
sys.exit(1)
|
0000-12-31 18:09:24 -05:50
|
|
|
|
|
|
|
def main():
|
2008-02-05 11:23:53 -06:00
|
|
|
options, args = parse_options()
|
|
|
|
|
|
|
|
replica_fqdn = args[1]
|
|
|
|
|
2008-02-15 19:47:29 -06:00
|
|
|
print "Determining current realm name"
|
0000-12-31 18:09:24 -05:50
|
|
|
realm_name = get_realm_name()
|
2008-02-15 19:47:29 -06:00
|
|
|
if realm_name is None:
|
|
|
|
print "Unable to determine default realm"
|
|
|
|
sys.exit(1)
|
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
check_ipa_configuration(realm_name)
|
|
|
|
|
2008-02-15 19:47:29 -06:00
|
|
|
print "Getting domain name from LDAP"
|
|
|
|
domain_name = get_domain_name()
|
|
|
|
if domain_name is None:
|
|
|
|
print "Unable to determine LDAP default domain"
|
|
|
|
sys.exit(1)
|
|
|
|
|
0000-12-31 18:09:24 -05:50
|
|
|
host_name = get_host_name()
|
2008-02-08 14:10:26 -06:00
|
|
|
ds_dir = dsinstance.config_dirname(dsinstance.realm_to_serverid(realm_name))
|
0000-12-31 18:09:24 -05:50
|
|
|
ds_user = get_ds_user(ds_dir)
|
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
print "Preparing replica for %s from %s" % (replica_fqdn, host_name)
|
0000-12-31 18:09:24 -05:50
|
|
|
|
|
|
|
top_dir = tempfile.mkdtemp("ipa")
|
|
|
|
dir = top_dir + "/realm_info"
|
|
|
|
os.mkdir(dir, 0700)
|
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
print "Creating SSL certificate for the Directory Server"
|
|
|
|
export_certdb(realm_name, ds_dir, dir, "dscert", "cn=%s,ou=Fedora Directory Server" % replica_fqdn)
|
|
|
|
print "Creating SSL certificate for the Web Server"
|
|
|
|
export_certdb(realm_name, ds_dir, dir, "httpcert", "cn=%s,ou=Apache Web Server" % replica_fqdn)
|
|
|
|
print "Copying additional files"
|
0000-12-31 18:09:24 -05:50
|
|
|
copy_files(realm_name, dir)
|
2008-02-05 11:23:53 -06:00
|
|
|
print "Finalizing configuration"
|
2008-02-15 19:47:29 -06:00
|
|
|
save_config(dir, realm_name, host_name, ds_user, domain_name)
|
0000-12-31 18:09:24 -05:50
|
|
|
|
2008-02-14 19:39:06 -06:00
|
|
|
print "Packaging the replica into %s" % "replica-info-" + replica_fqdn
|
|
|
|
ipautil.run(["/bin/tar", "cfz", "replica-info-" + replica_fqdn, "-C", top_dir, "realm_info"])
|
0000-12-31 18:09:24 -05:50
|
|
|
|
|
|
|
shutil.rmtree(dir)
|
|
|
|
|
2008-02-05 11:23:53 -06:00
|
|
|
try:
|
|
|
|
if not os.geteuid()==0:
|
|
|
|
sys.exit("\nYou must be root to run this script.\n")
|
|
|
|
if not ipautil.file_exists("/usr/share/ipa/serial"):
|
2008-02-15 19:47:29 -06:00
|
|
|
sys.exit("The replica must be created on the primary IPA server.")
|
2008-02-05 11:23:53 -06:00
|
|
|
|
|
|
|
main()
|
2008-02-08 14:10:26 -06:00
|
|
|
except SystemExit, e:
|
|
|
|
sys.exit(e)
|
2008-02-05 11:23:53 -06:00
|
|
|
except Exception, e:
|
|
|
|
print "preparation of replica failed: %s" % str(e)
|
|
|
|
message = str(e)
|
|
|
|
for str in traceback.format_tb(sys.exc_info()[2]):
|
|
|
|
message = message + "\n" + str
|
|
|
|
logging.debug(message)
|
|
|
|
print message
|
|
|
|
sys.exit(1)
|