mirror of
https://github.com/samba-team/samba.git
synced 2025-01-11 05:18:09 +03:00
e046986d04
Signed-off-by: Andreas Schneider <asn@samba.org> Reviewed-by: Andrew Bartlett <abartlet@samba.org>
1257 lines
55 KiB
Python
1257 lines
55 KiB
Python
# domain_backup
|
|
#
|
|
# Copyright Andrew Bartlett <abartlet@samba.org>
|
|
#
|
|
# 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 datetime
|
|
import os
|
|
import sys
|
|
import logging
|
|
import shutil
|
|
import tempfile
|
|
import samba
|
|
import tdb
|
|
import samba.getopt as options
|
|
from samba.samdb import SamDB, get_default_backend_store
|
|
import ldb
|
|
from ldb import LdbError
|
|
from samba.samba3 import libsmb_samba_internal as libsmb
|
|
from samba.samba3 import param as s3param
|
|
from samba.ntacls import backup_online, backup_restore, backup_offline
|
|
from samba.auth import system_session
|
|
from samba.join import DCJoinContext, join_clone, DCCloneAndRenameContext
|
|
from samba.dcerpc.security import dom_sid
|
|
from samba.netcmd import Option, CommandError
|
|
from samba.dcerpc import misc, security, drsblobs
|
|
from samba import Ldb
|
|
from samba.netcmd.fsmo import cmd_fsmo_seize
|
|
from samba.provision import make_smbconf, DEFAULTSITE
|
|
from samba.upgradehelpers import update_krbtgt_account_password
|
|
from samba.remove_dc import remove_dc
|
|
from samba.provision import secretsdb_self_join
|
|
from samba.dbchecker import dbcheck
|
|
import re
|
|
from samba.provision import guess_names, determine_host_ip, determine_host_ip6
|
|
from samba.provision.sambadns import (fill_dns_data_partitions,
|
|
get_dnsadmins_sid,
|
|
get_domainguid)
|
|
from samba.tdb_util import tdb_copy
|
|
from samba.mdb_util import mdb_copy
|
|
import errno
|
|
from subprocess import CalledProcessError
|
|
from samba import sites
|
|
from samba.dsdb import _dsdb_load_udv_v2
|
|
from samba.ndr import ndr_pack
|
|
from samba.credentials import SMB_SIGNING_REQUIRED
|
|
from samba import safe_tarfile as tarfile
|
|
|
|
|
|
# work out a SID (based on a free RID) to use when the domain gets restored.
|
|
# This ensures that the restored DC's SID won't clash with any other RIDs
|
|
# already in use in the domain
|
|
def get_sid_for_restore(samdb, logger):
|
|
# Allocate a new RID without modifying the database. This should be safe,
|
|
# because we acquire the RID master role after creating an account using
|
|
# this RID during the restore process. Acquiring the RID master role
|
|
# creates a new RID pool which we will fetch RIDs from, so we shouldn't get
|
|
# duplicates.
|
|
try:
|
|
rid = samdb.next_free_rid()
|
|
except LdbError as err:
|
|
logger.info("A SID could not be allocated for restoring the domain. "
|
|
"Either no RID Set was found on this DC, "
|
|
"or the RID Set was not usable.")
|
|
logger.info("To initialise this DC's RID pools, obtain a RID Set from "
|
|
"this domain's RID master, or run samba-tool dbcheck "
|
|
"to fix the existing RID Set.")
|
|
raise CommandError("Cannot create backup", err)
|
|
|
|
# Construct full SID
|
|
sid = dom_sid(samdb.get_domain_sid())
|
|
sid_for_restore = str(sid) + '-' + str(rid)
|
|
|
|
# Confirm the SID is not already in use
|
|
try:
|
|
res = samdb.search(scope=ldb.SCOPE_BASE,
|
|
base='<SID=%s>' % sid_for_restore,
|
|
attrs=[],
|
|
controls=['show_deleted:1',
|
|
'show_recycled:1'])
|
|
if len(res) != 1:
|
|
# This case makes no sense, but neither does a corrupt RID set
|
|
raise CommandError("Cannot create backup - "
|
|
"this DC's RID pool is corrupt, "
|
|
"the next SID (%s) appears to be in use." %
|
|
sid_for_restore)
|
|
raise CommandError("Cannot create backup - "
|
|
"this DC's RID pool is corrupt, "
|
|
"the next SID %s points to existing object %s. "
|
|
"Please run samba-tool dbcheck on the source DC." %
|
|
(sid_for_restore, res[0].dn))
|
|
except ldb.LdbError as e:
|
|
(enum, emsg) = e.args
|
|
if enum != ldb.ERR_NO_SUCH_OBJECT:
|
|
# We want NO_SUCH_OBJECT, anything else is a serious issue
|
|
raise
|
|
|
|
return str(sid) + '-' + str(rid)
|
|
|
|
|
|
def smb_sysvol_conn(server, lp, creds):
|
|
"""Returns an SMB connection to the sysvol share on the DC"""
|
|
# the SMB bindings rely on having a s3 loadparm
|
|
s3_lp = s3param.get_context()
|
|
s3_lp.load(lp.configfile)
|
|
|
|
# Force signing for the connection
|
|
saved_signing_state = creds.get_smb_signing()
|
|
creds.set_smb_signing(SMB_SIGNING_REQUIRED)
|
|
conn = libsmb.Conn(server, "sysvol", lp=s3_lp, creds=creds)
|
|
# Reset signing state
|
|
creds.set_smb_signing(saved_signing_state)
|
|
return conn
|
|
|
|
|
|
def get_timestamp():
|
|
return datetime.datetime.now().isoformat().replace(':', '-')
|
|
|
|
|
|
def backup_filepath(targetdir, name, time_str):
|
|
filename = 'samba-backup-%s-%s.tar.bz2' % (name, time_str)
|
|
return os.path.join(targetdir, filename)
|
|
|
|
|
|
def create_backup_tar(logger, tmpdir, backup_filepath):
|
|
# Adds everything in the tmpdir into a new tar file
|
|
logger.info("Creating backup file %s..." % backup_filepath)
|
|
tf = tarfile.open(backup_filepath, 'w:bz2')
|
|
tf.add(tmpdir, arcname='./')
|
|
tf.close()
|
|
|
|
|
|
def create_log_file(targetdir, lp, backup_type, server, include_secrets,
|
|
extra_info=None):
|
|
# create a summary file about the backup, which will get included in the
|
|
# tar file. This makes it easy for users to see what the backup involved,
|
|
# without having to untar the DB and interrogate it
|
|
f = open(os.path.join(targetdir, "backup.txt"), 'w')
|
|
try:
|
|
time_str = datetime.datetime.now().strftime('%Y-%b-%d %H:%M:%S')
|
|
f.write("Backup created %s\n" % time_str)
|
|
f.write("Using samba-tool version: %s\n" % lp.get('server string'))
|
|
f.write("Domain %s backup, using DC '%s'\n" % (backup_type, server))
|
|
f.write("Backup for domain %s (NetBIOS), %s (DNS realm)\n" %
|
|
(lp.get('workgroup'), lp.get('realm').lower()))
|
|
f.write("Backup contains domain secrets: %s\n" % str(include_secrets))
|
|
if extra_info:
|
|
f.write("%s\n" % extra_info)
|
|
finally:
|
|
f.close()
|
|
|
|
|
|
# Add a backup-specific marker to the DB with info that we'll use during
|
|
# the restore process
|
|
def add_backup_marker(samdb, marker, value):
|
|
m = ldb.Message()
|
|
m.dn = ldb.Dn(samdb, "@SAMBA_DSDB")
|
|
m[marker] = ldb.MessageElement(value, ldb.FLAG_MOD_ADD, marker)
|
|
samdb.modify(m)
|
|
|
|
|
|
def check_targetdir(logger, targetdir):
|
|
if targetdir is None:
|
|
raise CommandError('Target directory required')
|
|
|
|
if not os.path.exists(targetdir):
|
|
logger.info('Creating targetdir %s...' % targetdir)
|
|
os.makedirs(targetdir)
|
|
elif not os.path.isdir(targetdir):
|
|
raise CommandError("%s is not a directory" % targetdir)
|
|
|
|
|
|
# For '--no-secrets' backups, this sets the Administrator user's password to a
|
|
# randomly-generated value. This is similar to the provision behaviour
|
|
def set_admin_password(logger, samdb):
|
|
"""Sets a randomly generated password for the backup DB's admin user"""
|
|
|
|
# match the admin user by RID
|
|
domainsid = samdb.get_domain_sid()
|
|
match_admin = "(objectsid=%s-%s)" % (domainsid,
|
|
security.DOMAIN_RID_ADMINISTRATOR)
|
|
search_expr = "(&(objectClass=user)%s)" % (match_admin,)
|
|
|
|
# retrieve the admin username (just in case it's been renamed)
|
|
res = samdb.search(base=samdb.domain_dn(), scope=ldb.SCOPE_SUBTREE,
|
|
expression=search_expr)
|
|
username = str(res[0]['samaccountname'])
|
|
|
|
adminpass = samba.generate_random_password(12, 32)
|
|
logger.info("Setting %s password in backup to: %s" % (username, adminpass))
|
|
logger.info("Run 'samba-tool user setpassword %s' after restoring DB" %
|
|
username)
|
|
samdb.setpassword(search_expr, adminpass, force_change_at_next_login=False,
|
|
username=username)
|
|
|
|
|
|
class cmd_domain_backup_online(samba.netcmd.Command):
|
|
'''Copy a running DC's current DB into a backup tar file.
|
|
|
|
Takes a backup copy of the current domain from a running DC. If the domain
|
|
were to undergo a catastrophic failure, then the backup file can be used to
|
|
recover the domain. The backup created is similar to the DB that a new DC
|
|
would receive when it joins the domain.
|
|
|
|
Note that:
|
|
- it's recommended to run 'samba-tool dbcheck' before taking a backup-file
|
|
and fix any errors it reports.
|
|
- all the domain's secrets are included in the backup file.
|
|
- although the DB contents can be untarred and examined manually, you need
|
|
to run 'samba-tool domain backup restore' before you can start a Samba DC
|
|
from the backup file.'''
|
|
|
|
synopsis = "%prog --server=<DC-to-backup> --targetdir=<output-dir>"
|
|
takes_optiongroups = {
|
|
"sambaopts": options.SambaOptions,
|
|
"credopts": options.CredentialsOptions,
|
|
}
|
|
|
|
takes_options = [
|
|
Option("--server", help="The DC to backup", type=str),
|
|
Option("--targetdir", type=str,
|
|
help="Directory to write the backup file to"),
|
|
Option("--no-secrets", action="store_true", default=False,
|
|
help="Exclude secret values from the backup created"),
|
|
Option("--backend-store", type="choice", metavar="BACKENDSTORE",
|
|
choices=["tdb", "mdb"],
|
|
help="Specify the database backend to be used "
|
|
"(default is %s)" % get_default_backend_store()),
|
|
]
|
|
|
|
def run(self, sambaopts=None, credopts=None, server=None, targetdir=None,
|
|
no_secrets=False, backend_store=None):
|
|
logger = self.get_logger()
|
|
logger.setLevel(logging.DEBUG)
|
|
|
|
lp = sambaopts.get_loadparm()
|
|
creds = credopts.get_credentials(lp)
|
|
|
|
# Make sure we have all the required args.
|
|
if server is None:
|
|
raise CommandError('Server required')
|
|
|
|
check_targetdir(logger, targetdir)
|
|
|
|
tmpdir = tempfile.mkdtemp(dir=targetdir)
|
|
|
|
# Run a clone join on the remote
|
|
include_secrets = not no_secrets
|
|
try:
|
|
ctx = join_clone(logger=logger, creds=creds, lp=lp,
|
|
include_secrets=include_secrets, server=server,
|
|
dns_backend='SAMBA_INTERNAL', targetdir=tmpdir,
|
|
backend_store=backend_store)
|
|
|
|
# get the paths used for the clone, then drop the old samdb connection
|
|
paths = ctx.paths
|
|
del ctx
|
|
|
|
# Get a free RID to use as the new DC's SID (when it gets restored)
|
|
remote_sam = SamDB(url='ldap://' + server, credentials=creds,
|
|
session_info=system_session(), lp=lp)
|
|
new_sid = get_sid_for_restore(remote_sam, logger)
|
|
realm = remote_sam.domain_dns_name()
|
|
|
|
# Grab the remote DC's sysvol files and bundle them into a tar file
|
|
logger.info("Backing up sysvol files (via SMB)...")
|
|
sysvol_tar = os.path.join(tmpdir, 'sysvol.tar.gz')
|
|
smb_conn = smb_sysvol_conn(server, lp, creds)
|
|
backup_online(smb_conn, sysvol_tar, remote_sam.get_domain_sid())
|
|
|
|
# remove the default sysvol files created by the clone (we want to
|
|
# make sure we restore the sysvol.tar.gz files instead)
|
|
shutil.rmtree(paths.sysvol)
|
|
|
|
# Edit the downloaded sam.ldb to mark it as a backup
|
|
samdb = SamDB(url=paths.samdb, session_info=system_session(), lp=lp,
|
|
flags=ldb.FLG_DONT_CREATE_DB)
|
|
time_str = get_timestamp()
|
|
add_backup_marker(samdb, "backupDate", time_str)
|
|
add_backup_marker(samdb, "sidForRestore", new_sid)
|
|
add_backup_marker(samdb, "backupType", "online")
|
|
|
|
# ensure the admin user always has a password set (same as provision)
|
|
if no_secrets:
|
|
set_admin_password(logger, samdb)
|
|
|
|
# Add everything in the tmpdir to the backup tar file
|
|
backup_file = backup_filepath(targetdir, realm, time_str)
|
|
create_log_file(tmpdir, lp, "online", server, include_secrets)
|
|
create_backup_tar(logger, tmpdir, backup_file)
|
|
finally:
|
|
shutil.rmtree(tmpdir)
|
|
|
|
|
|
class cmd_domain_backup_restore(cmd_fsmo_seize):
|
|
'''Restore the domain's DB from a backup-file.
|
|
|
|
This restores a previously backed up copy of the domain's DB on a new DC.
|
|
|
|
Note that the restored DB will not contain the original DC that the backup
|
|
was taken from (or any other DCs in the original domain). Only the new DC
|
|
(specified by --newservername) will be present in the restored DB.
|
|
|
|
Samba can then be started against the restored DB. Any existing DCs for the
|
|
domain should be shutdown before the new DC is started. Other DCs can then
|
|
be joined to the new DC to recover the network.
|
|
|
|
Note that this command should be run as the root user - it will fail
|
|
otherwise.'''
|
|
|
|
synopsis = ("%prog --backup-file=<tar-file> --targetdir=<output-dir> "
|
|
"--newservername=<DC-name>")
|
|
takes_options = [
|
|
Option("--backup-file", help="Path to backup file", type=str),
|
|
Option("--targetdir", help="Path to write to", type=str),
|
|
Option("--newservername", help="Name for new server", type=str),
|
|
Option("--host-ip", type="string", metavar="IPADDRESS",
|
|
help="set IPv4 ipaddress"),
|
|
Option("--host-ip6", type="string", metavar="IP6ADDRESS",
|
|
help="set IPv6 ipaddress"),
|
|
Option("--site", help="Site to add the new server in", type=str),
|
|
]
|
|
|
|
takes_optiongroups = {
|
|
"sambaopts": options.SambaOptions,
|
|
"credopts": options.CredentialsOptions,
|
|
}
|
|
|
|
def register_dns_zone(self, logger, samdb, lp, ntdsguid, host_ip,
|
|
host_ip6, site):
|
|
'''
|
|
Registers the new realm's DNS objects when a renamed domain backup
|
|
is restored.
|
|
'''
|
|
names = guess_names(lp)
|
|
domaindn = names.domaindn
|
|
forestdn = samdb.get_root_basedn().get_linearized()
|
|
dnsdomain = names.dnsdomain.lower()
|
|
dnsforest = dnsdomain
|
|
hostname = names.netbiosname.lower()
|
|
domainsid = dom_sid(samdb.get_domain_sid())
|
|
dnsadmins_sid = get_dnsadmins_sid(samdb, domaindn)
|
|
domainguid = get_domainguid(samdb, domaindn)
|
|
|
|
# work out the IP address to use for the new DC's DNS records
|
|
host_ip = determine_host_ip(logger, lp, host_ip)
|
|
host_ip6 = determine_host_ip6(logger, lp, host_ip6)
|
|
|
|
if host_ip is None and host_ip6 is None:
|
|
raise CommandError('Please specify a host-ip for the new server')
|
|
|
|
logger.info("DNS realm was renamed to %s" % dnsdomain)
|
|
logger.info("Populating DNS partitions for new realm...")
|
|
|
|
# Add the DNS objects for the new realm (note: the backup clone already
|
|
# has the root server objects, so don't add them again)
|
|
fill_dns_data_partitions(samdb, domainsid, site, domaindn,
|
|
forestdn, dnsdomain, dnsforest, hostname,
|
|
host_ip, host_ip6, domainguid, ntdsguid,
|
|
dnsadmins_sid, add_root=False)
|
|
|
|
def fix_old_dc_references(self, samdb):
|
|
'''Fixes attributes that reference the old/removed DCs'''
|
|
|
|
# we just want to fix up DB problems here that were introduced by us
|
|
# removing the old DCs. We restrict what we fix up so that the restored
|
|
# DB matches the backed-up DB as close as possible. (There may be other
|
|
# DB issues inherited from the backed-up DC, but it's not our place to
|
|
# silently try to fix them here).
|
|
samdb.transaction_start()
|
|
chk = dbcheck(samdb, quiet=True, fix=True, yes=False,
|
|
in_transaction=True)
|
|
|
|
# fix up stale references to the old DC
|
|
setattr(chk, 'fix_all_old_dn_string_component_mismatch', 'ALL')
|
|
attrs = ['lastKnownParent', 'interSiteTopologyGenerator']
|
|
|
|
# fix-up stale one-way links that point to the old DC
|
|
setattr(chk, 'remove_plausible_deleted_DN_links', 'ALL')
|
|
attrs += ['msDS-NC-Replica-Locations']
|
|
|
|
cross_ncs_ctrl = 'search_options:1:2'
|
|
controls = ['show_deleted:1', cross_ncs_ctrl]
|
|
chk.check_database(controls=controls, attrs=attrs)
|
|
samdb.transaction_commit()
|
|
|
|
def create_default_site(self, samdb, logger):
|
|
'''Creates the default site, if it doesn't already exist'''
|
|
|
|
sitename = DEFAULTSITE
|
|
search_expr = "(&(cn={0})(objectclass=site))".format(sitename)
|
|
res = samdb.search(samdb.get_config_basedn(), scope=ldb.SCOPE_SUBTREE,
|
|
expression=search_expr)
|
|
|
|
if len(res) == 0:
|
|
logger.info("Creating default site '{0}'".format(sitename))
|
|
sites.create_site(samdb, samdb.get_config_basedn(), sitename)
|
|
|
|
return sitename
|
|
|
|
def remove_backup_markers(self, samdb):
|
|
"""Remove DB markers added by the backup process"""
|
|
|
|
# check what markers we need to remove (this may vary)
|
|
markers = ['sidForRestore', 'backupRename', 'backupDate', 'backupType']
|
|
res = samdb.search(base=ldb.Dn(samdb, "@SAMBA_DSDB"),
|
|
scope=ldb.SCOPE_BASE,
|
|
attrs=markers)
|
|
|
|
# remove any markers that exist in the DB
|
|
m = ldb.Message()
|
|
m.dn = ldb.Dn(samdb, "@SAMBA_DSDB")
|
|
|
|
for attr in markers:
|
|
if attr in res[0]:
|
|
m[attr] = ldb.MessageElement([], ldb.FLAG_MOD_DELETE, attr)
|
|
|
|
samdb.modify(m)
|
|
|
|
def get_backup_type(self, samdb):
|
|
res = samdb.search(base=ldb.Dn(samdb, "@SAMBA_DSDB"),
|
|
scope=ldb.SCOPE_BASE,
|
|
attrs=['backupRename', 'backupType'])
|
|
|
|
# note that the backupType marker won't exist on backups created on
|
|
# v4.9. However, we can still infer the type, as only rename and
|
|
# online backups are supported on v4.9
|
|
if 'backupType' in res[0]:
|
|
backup_type = str(res[0]['backupType'])
|
|
elif 'backupRename' in res[0]:
|
|
backup_type = "rename"
|
|
else:
|
|
backup_type = "online"
|
|
|
|
return backup_type
|
|
|
|
def save_uptodate_vectors(self, samdb, partitions):
|
|
"""Ensures the UTDV used by DRS is correct after an offline backup"""
|
|
for nc in partitions:
|
|
# load the replUpToDateVector we *should* have
|
|
utdv = _dsdb_load_udv_v2(samdb, nc)
|
|
|
|
# convert it to NDR format and write it into the DB
|
|
utdv_blob = drsblobs.replUpToDateVectorBlob()
|
|
utdv_blob.version = 2
|
|
utdv_blob.ctr.cursors = utdv
|
|
utdv_blob.ctr.count = len(utdv)
|
|
new_value = ndr_pack(utdv_blob)
|
|
|
|
m = ldb.Message()
|
|
m.dn = ldb.Dn(samdb, nc)
|
|
m["replUpToDateVector"] = ldb.MessageElement(new_value,
|
|
ldb.FLAG_MOD_REPLACE,
|
|
"replUpToDateVector")
|
|
samdb.modify(m)
|
|
|
|
def run(self, sambaopts=None, credopts=None, backup_file=None,
|
|
targetdir=None, newservername=None, host_ip=None, host_ip6=None,
|
|
site=None):
|
|
if not (backup_file and os.path.exists(backup_file)):
|
|
raise CommandError('Backup file not found.')
|
|
if targetdir is None:
|
|
raise CommandError('Please specify a target directory')
|
|
# allow restoredc to install into a directory prepopulated by selftest
|
|
if (os.path.exists(targetdir) and os.listdir(targetdir) and
|
|
os.environ.get('SAMBA_SELFTEST') != '1'):
|
|
raise CommandError('Target directory is not empty')
|
|
if not newservername:
|
|
raise CommandError('Server name required')
|
|
|
|
logger = logging.getLogger()
|
|
logger.setLevel(logging.DEBUG)
|
|
logger.addHandler(logging.StreamHandler(sys.stdout))
|
|
|
|
# ldapcmp prefers the server's netBIOS name in upper-case
|
|
newservername = newservername.upper()
|
|
|
|
# extract the backup .tar to a temp directory
|
|
targetdir = os.path.abspath(targetdir)
|
|
tf = tarfile.open(backup_file)
|
|
tf.extractall(targetdir)
|
|
tf.close()
|
|
|
|
# use the smb.conf that got backed up, by default (save what was
|
|
# actually backed up, before we mess with it)
|
|
smbconf = os.path.join(targetdir, 'etc', 'smb.conf')
|
|
shutil.copyfile(smbconf, smbconf + ".orig")
|
|
|
|
# if a smb.conf was specified on the cmd line, then use that instead
|
|
cli_smbconf = sambaopts.get_loadparm_path()
|
|
if cli_smbconf:
|
|
logger.info("Using %s as restored domain's smb.conf" % cli_smbconf)
|
|
shutil.copyfile(cli_smbconf, smbconf)
|
|
|
|
lp = samba.param.LoadParm()
|
|
lp.load(smbconf)
|
|
|
|
# open a DB connection to the restored DB
|
|
private_dir = os.path.join(targetdir, 'private')
|
|
samdb_path = os.path.join(private_dir, 'sam.ldb')
|
|
samdb = SamDB(url=samdb_path, session_info=system_session(), lp=lp,
|
|
flags=ldb.FLG_DONT_CREATE_DB)
|
|
backup_type = self.get_backup_type(samdb)
|
|
|
|
if site is None:
|
|
# There's no great way to work out the correct site to add the
|
|
# restored DC to. By default, add it to Default-First-Site-Name,
|
|
# creating the site if it doesn't already exist
|
|
site = self.create_default_site(samdb, logger)
|
|
logger.info("Adding new DC to site '{0}'".format(site))
|
|
|
|
# read the naming contexts out of the DB
|
|
res = samdb.search(base="", scope=ldb.SCOPE_BASE,
|
|
attrs=['namingContexts'])
|
|
ncs = [str(r) for r in res[0].get('namingContexts')]
|
|
|
|
# for offline backups we need to make sure the upToDateness info
|
|
# contains the invocation-ID and highest-USN of the DC we backed up.
|
|
# Otherwise replication propagation dampening won't correctly filter
|
|
# objects created by that DC
|
|
if backup_type == "offline":
|
|
self.save_uptodate_vectors(samdb, ncs)
|
|
|
|
# Create account using the join_add_objects function in the join object
|
|
# We need namingContexts, account control flags, and the sid saved by
|
|
# the backup process.
|
|
creds = credopts.get_credentials(lp)
|
|
ctx = DCJoinContext(logger, creds=creds, lp=lp, site=site,
|
|
forced_local_samdb=samdb,
|
|
netbios_name=newservername)
|
|
ctx.nc_list = ncs
|
|
ctx.full_nc_list = ncs
|
|
ctx.userAccountControl = (samba.dsdb.UF_SERVER_TRUST_ACCOUNT |
|
|
samba.dsdb.UF_TRUSTED_FOR_DELEGATION)
|
|
|
|
# rewrite the smb.conf to make sure it uses the new targetdir settings.
|
|
# (This doesn't update all filepaths in a customized config, but it
|
|
# corrects the same paths that get set by a new provision)
|
|
logger.info('Updating basic smb.conf settings...')
|
|
make_smbconf(smbconf, newservername, ctx.domain_name,
|
|
ctx.realm, targetdir, lp=lp,
|
|
serverrole="active directory domain controller")
|
|
|
|
# Get the SID saved by the backup process and create account
|
|
res = samdb.search(base=ldb.Dn(samdb, "@SAMBA_DSDB"),
|
|
scope=ldb.SCOPE_BASE,
|
|
attrs=['sidForRestore'])
|
|
sid = res[0].get('sidForRestore')[0]
|
|
logger.info('Creating account with SID: ' + str(sid))
|
|
try:
|
|
ctx.join_add_objects(specified_sid=dom_sid(str(sid)))
|
|
except LdbError as e:
|
|
(enum, emsg) = e.args
|
|
if enum != ldb.ERR_CONSTRAINT_VIOLATION:
|
|
raise
|
|
|
|
dup_res = []
|
|
try:
|
|
dup_res = samdb.search(base=ldb.Dn(samdb, "<SID=%s>" % sid),
|
|
scope=ldb.SCOPE_BASE,
|
|
attrs=['objectGUID'],
|
|
controls=["show_deleted:0",
|
|
"show_recycled:0"])
|
|
except LdbError as dup_e:
|
|
(dup_enum, _) = dup_e.args
|
|
if dup_enum != ldb.ERR_NO_SUCH_OBJECT:
|
|
raise
|
|
|
|
if (len(dup_res) != 1):
|
|
raise
|
|
|
|
objectguid = samdb.schema_format_value("objectGUID",
|
|
dup_res[0]["objectGUID"][0])
|
|
objectguid = objectguid.decode('utf-8')
|
|
logger.error("The RID Pool on the source DC for the backup in %s "
|
|
"may be corrupt "
|
|
"or in conflict with SIDs already allocated "
|
|
"in the domain. " % backup_file)
|
|
logger.error("Running 'samba-tool dbcheck' on the source "
|
|
"DC (and obtaining a new backup) may correct the issue.")
|
|
logger.error("Alternatively please obtain a new backup "
|
|
"against a different DC.")
|
|
logger.error("The SID we wish to use (%s) is recorded in "
|
|
"@SAMBA_DSDB as the sidForRestore attribute."
|
|
% sid)
|
|
|
|
raise CommandError("Domain restore failed because there "
|
|
"is already an existing object (%s) "
|
|
"with SID %s and objectGUID %s. "
|
|
"This conflicts with "
|
|
"the new DC account we want to add "
|
|
"for the restored domain. " % (
|
|
dup_res[0].dn, sid, objectguid))
|
|
|
|
m = ldb.Message()
|
|
m.dn = ldb.Dn(samdb, '@ROOTDSE')
|
|
ntds_guid = str(ctx.ntds_guid)
|
|
m["dsServiceName"] = ldb.MessageElement("<GUID=%s>" % ntds_guid,
|
|
ldb.FLAG_MOD_REPLACE,
|
|
"dsServiceName")
|
|
samdb.modify(m)
|
|
|
|
# if we renamed the backed-up domain, then we need to add the DNS
|
|
# objects for the new realm (we do this in the restore, now that we
|
|
# know the new DC's IP address)
|
|
if backup_type == "rename":
|
|
self.register_dns_zone(logger, samdb, lp, ctx.ntds_guid,
|
|
host_ip, host_ip6, site)
|
|
|
|
secrets_path = os.path.join(private_dir, 'secrets.ldb')
|
|
secrets_ldb = Ldb(secrets_path, session_info=system_session(), lp=lp,
|
|
flags=ldb.FLG_DONT_CREATE_DB)
|
|
secretsdb_self_join(secrets_ldb, domain=ctx.domain_name,
|
|
realm=ctx.realm, dnsdomain=ctx.dnsdomain,
|
|
netbiosname=ctx.myname, domainsid=ctx.domsid,
|
|
machinepass=ctx.acct_pass,
|
|
key_version_number=ctx.key_version_number,
|
|
secure_channel_type=misc.SEC_CHAN_BDC)
|
|
|
|
# Seize DNS roles
|
|
domain_dn = samdb.domain_dn()
|
|
forest_dn = samba.dn_from_dns_name(samdb.forest_dns_name())
|
|
dns_roles = [("domaindns", domain_dn),
|
|
("forestdns", forest_dn)]
|
|
for role, dn in dns_roles:
|
|
if dn in ncs:
|
|
self.seize_dns_role(role, samdb, None, None, None, force=True)
|
|
|
|
# Seize other roles
|
|
for role in ['rid', 'pdc', 'naming', 'infrastructure', 'schema']:
|
|
self.seize_role(role, samdb, force=True)
|
|
|
|
# Get all DCs and remove them (this ensures these DCs cannot
|
|
# replicate because they will not have a password)
|
|
search_expr = "(&(objectClass=Server)(serverReference=*))"
|
|
res = samdb.search(samdb.get_config_basedn(), scope=ldb.SCOPE_SUBTREE,
|
|
expression=search_expr)
|
|
for m in res:
|
|
cn = str(m.get('cn')[0])
|
|
if cn != newservername:
|
|
remove_dc(samdb, logger, cn)
|
|
|
|
# Remove the repsFrom and repsTo from each NC to ensure we do
|
|
# not try (and fail) to talk to the old DCs
|
|
for nc in ncs:
|
|
msg = ldb.Message()
|
|
msg.dn = ldb.Dn(samdb, nc)
|
|
|
|
msg["repsFrom"] = ldb.MessageElement([],
|
|
ldb.FLAG_MOD_REPLACE,
|
|
"repsFrom")
|
|
msg["repsTo"] = ldb.MessageElement([],
|
|
ldb.FLAG_MOD_REPLACE,
|
|
"repsTo")
|
|
samdb.modify(msg)
|
|
|
|
# Update the krbtgt passwords twice, ensuring no tickets from
|
|
# the old domain are valid
|
|
update_krbtgt_account_password(samdb)
|
|
update_krbtgt_account_password(samdb)
|
|
|
|
# restore the sysvol directory from the backup tar file, including the
|
|
# original NTACLs. Note that the backup_restore() will fail if not root
|
|
sysvol_tar = os.path.join(targetdir, 'sysvol.tar.gz')
|
|
dest_sysvol_dir = lp.get('path', 'sysvol')
|
|
if not os.path.exists(dest_sysvol_dir):
|
|
os.makedirs(dest_sysvol_dir)
|
|
backup_restore(sysvol_tar, dest_sysvol_dir, samdb, smbconf)
|
|
os.remove(sysvol_tar)
|
|
|
|
# fix up any stale links to the old DCs we just removed
|
|
logger.info("Fixing up any remaining references to the old DCs...")
|
|
self.fix_old_dc_references(samdb)
|
|
|
|
# Remove DB markers added by the backup process
|
|
self.remove_backup_markers(samdb)
|
|
|
|
logger.info("Backup file successfully restored to %s" % targetdir)
|
|
logger.info("Please check the smb.conf settings are correct before "
|
|
"starting samba.")
|
|
|
|
|
|
class cmd_domain_backup_rename(samba.netcmd.Command):
|
|
'''Copy a running DC's DB to backup file, renaming the domain in the process.
|
|
|
|
Where <new-domain> is the new domain's NetBIOS name, and <new-dnsrealm> is
|
|
the new domain's realm in DNS form.
|
|
|
|
This is similar to 'samba-tool backup online' in that it clones the DB of a
|
|
running DC. However, this option also renames all the domain entries in the
|
|
DB. Renaming the domain makes it possible to restore and start a new Samba
|
|
DC without it interfering with the existing Samba domain. In other words,
|
|
you could use this option to clone your production samba domain and restore
|
|
it to a separate pre-production environment that won't overlap or interfere
|
|
with the existing production Samba domain.
|
|
|
|
Note that:
|
|
- it's recommended to run 'samba-tool dbcheck' before taking a backup-file
|
|
and fix any errors it reports.
|
|
- all the domain's secrets are included in the backup file.
|
|
- although the DB contents can be untarred and examined manually, you need
|
|
to run 'samba-tool domain backup restore' before you can start a Samba DC
|
|
from the backup file.
|
|
- GPO and sysvol information will still refer to the old realm and will
|
|
need to be updated manually.
|
|
- if you specify 'keep-dns-realm', then the DNS records will need updating
|
|
in order to work (they will still refer to the old DC's IP instead of the
|
|
new DC's address).
|
|
- we recommend that you only use this option if you know what you're doing.
|
|
'''
|
|
|
|
synopsis = ("%prog <new-domain> <new-dnsrealm> --server=<DC-to-backup> "
|
|
"--targetdir=<output-dir>")
|
|
takes_optiongroups = {
|
|
"sambaopts": options.SambaOptions,
|
|
"credopts": options.CredentialsOptions,
|
|
}
|
|
|
|
takes_options = [
|
|
Option("--server", help="The DC to backup", type=str),
|
|
Option("--targetdir", help="Directory to write the backup file",
|
|
type=str),
|
|
Option("--keep-dns-realm", action="store_true", default=False,
|
|
help="Retain the DNS entries for the old realm in the backup"),
|
|
Option("--no-secrets", action="store_true", default=False,
|
|
help="Exclude secret values from the backup created"),
|
|
Option("--backend-store", type="choice", metavar="BACKENDSTORE",
|
|
choices=["tdb", "mdb"],
|
|
help="Specify the database backend to be used "
|
|
"(default is %s)" % get_default_backend_store()),
|
|
]
|
|
|
|
takes_args = ["new_domain_name", "new_dns_realm"]
|
|
|
|
def update_dns_root(self, logger, samdb, old_realm, delete_old_dns):
|
|
'''Updates dnsRoot for the partition objects to reflect the rename'''
|
|
|
|
# lookup the crossRef objects that hold the old realm's dnsRoot
|
|
partitions_dn = samdb.get_partitions_dn()
|
|
res = samdb.search(base=partitions_dn, scope=ldb.SCOPE_ONELEVEL,
|
|
attrs=["dnsRoot"],
|
|
expression='(&(objectClass=crossRef)(dnsRoot=*))')
|
|
new_realm = samdb.domain_dns_name()
|
|
|
|
# go through and add the new realm
|
|
for res_msg in res:
|
|
# dnsRoot can be multi-valued, so only look for the old realm
|
|
for dns_root in res_msg["dnsRoot"]:
|
|
dns_root = str(dns_root)
|
|
dn = res_msg.dn
|
|
if old_realm in dns_root:
|
|
new_dns_root = re.sub('%s$' % old_realm, new_realm,
|
|
dns_root)
|
|
logger.info("Adding %s dnsRoot to %s" % (new_dns_root, dn))
|
|
|
|
m = ldb.Message()
|
|
m.dn = dn
|
|
m["dnsRoot"] = ldb.MessageElement(new_dns_root,
|
|
ldb.FLAG_MOD_ADD,
|
|
"dnsRoot")
|
|
samdb.modify(m)
|
|
|
|
# optionally remove the dnsRoot for the old realm
|
|
if delete_old_dns:
|
|
logger.info("Removing %s dnsRoot from %s" % (dns_root,
|
|
dn))
|
|
m["dnsRoot"] = ldb.MessageElement(dns_root,
|
|
ldb.FLAG_MOD_DELETE,
|
|
"dnsRoot")
|
|
samdb.modify(m)
|
|
|
|
# Updates the CN=<domain>,CN=Partitions,CN=Configuration,... object to
|
|
# reflect the domain rename
|
|
def rename_domain_partition(self, logger, samdb, new_netbios_name):
|
|
'''Renames the domain partition object and updates its nETBIOSName'''
|
|
|
|
# lookup the crossRef object that holds the nETBIOSName (nCName has
|
|
# already been updated by this point, but the netBIOS hasn't)
|
|
base_dn = samdb.get_default_basedn()
|
|
nc_name = ldb.binary_encode(str(base_dn))
|
|
partitions_dn = samdb.get_partitions_dn()
|
|
res = samdb.search(base=partitions_dn, scope=ldb.SCOPE_ONELEVEL,
|
|
attrs=["nETBIOSName"],
|
|
expression='ncName=%s' % nc_name)
|
|
|
|
logger.info("Changing backup domain's NetBIOS name to %s" %
|
|
new_netbios_name)
|
|
m = ldb.Message()
|
|
m.dn = res[0].dn
|
|
m["nETBIOSName"] = ldb.MessageElement(new_netbios_name,
|
|
ldb.FLAG_MOD_REPLACE,
|
|
"nETBIOSName")
|
|
samdb.modify(m)
|
|
|
|
# renames the object itself to reflect the change in domain
|
|
new_dn = "CN=%s,%s" % (new_netbios_name, partitions_dn)
|
|
logger.info("Renaming %s --> %s" % (res[0].dn, new_dn))
|
|
samdb.rename(res[0].dn, new_dn, controls=['relax:0'])
|
|
|
|
def delete_old_dns_zones(self, logger, samdb, old_realm):
|
|
# remove the top-level DNS entries for the old realm
|
|
basedn = samdb.get_default_basedn()
|
|
dn = "DC=%s,CN=MicrosoftDNS,DC=DomainDnsZones,%s" % (old_realm, basedn)
|
|
logger.info("Deleting old DNS zone %s" % dn)
|
|
samdb.delete(dn, ["tree_delete:1"])
|
|
|
|
forestdn = samdb.get_root_basedn().get_linearized()
|
|
dn = "DC=_msdcs.%s,CN=MicrosoftDNS,DC=ForestDnsZones,%s" % (old_realm,
|
|
forestdn)
|
|
logger.info("Deleting old DNS zone %s" % dn)
|
|
samdb.delete(dn, ["tree_delete:1"])
|
|
|
|
def fix_old_dn_attributes(self, samdb):
|
|
'''Fixes attributes (i.e. objectCategory) that still use the old DN'''
|
|
|
|
samdb.transaction_start()
|
|
# Just fix any mismatches in DN detected (leave any other errors)
|
|
chk = dbcheck(samdb, quiet=True, fix=True, yes=False,
|
|
in_transaction=True)
|
|
# fix up incorrect objectCategory/etc attributes
|
|
setattr(chk, 'fix_all_old_dn_string_component_mismatch', 'ALL')
|
|
cross_ncs_ctrl = 'search_options:1:2'
|
|
controls = ['show_deleted:1', cross_ncs_ctrl]
|
|
chk.check_database(controls=controls)
|
|
samdb.transaction_commit()
|
|
|
|
def run(self, new_domain_name, new_dns_realm, sambaopts=None,
|
|
credopts=None, server=None, targetdir=None, keep_dns_realm=False,
|
|
no_secrets=False, backend_store=None):
|
|
logger = self.get_logger()
|
|
logger.setLevel(logging.INFO)
|
|
|
|
lp = sambaopts.get_loadparm()
|
|
creds = credopts.get_credentials(lp)
|
|
|
|
# Make sure we have all the required args.
|
|
if server is None:
|
|
raise CommandError('Server required')
|
|
|
|
check_targetdir(logger, targetdir)
|
|
|
|
delete_old_dns = not keep_dns_realm
|
|
|
|
new_dns_realm = new_dns_realm.lower()
|
|
new_domain_name = new_domain_name.upper()
|
|
|
|
new_base_dn = samba.dn_from_dns_name(new_dns_realm)
|
|
logger.info("New realm for backed up domain: %s" % new_dns_realm)
|
|
logger.info("New base DN for backed up domain: %s" % new_base_dn)
|
|
logger.info("New domain NetBIOS name: %s" % new_domain_name)
|
|
|
|
tmpdir = tempfile.mkdtemp(dir=targetdir)
|
|
|
|
# setup a join-context for cloning the remote server
|
|
include_secrets = not no_secrets
|
|
ctx = DCCloneAndRenameContext(new_base_dn, new_domain_name,
|
|
new_dns_realm, logger=logger,
|
|
creds=creds, lp=lp,
|
|
include_secrets=include_secrets,
|
|
dns_backend='SAMBA_INTERNAL',
|
|
server=server, targetdir=tmpdir,
|
|
backend_store=backend_store)
|
|
|
|
# sanity-check we're not "renaming" the domain to the same values
|
|
old_domain = ctx.domain_name
|
|
if old_domain == new_domain_name:
|
|
shutil.rmtree(tmpdir)
|
|
raise CommandError("Cannot use the current domain NetBIOS name.")
|
|
|
|
old_realm = ctx.realm
|
|
if old_realm == new_dns_realm:
|
|
shutil.rmtree(tmpdir)
|
|
raise CommandError("Cannot use the current domain DNS realm.")
|
|
|
|
# do the clone/rename
|
|
ctx.do_join()
|
|
|
|
# get the paths used for the clone, then drop the old samdb connection
|
|
del ctx.local_samdb
|
|
paths = ctx.paths
|
|
|
|
# get a free RID to use as the new DC's SID (when it gets restored)
|
|
remote_sam = SamDB(url='ldap://' + server, credentials=creds,
|
|
session_info=system_session(), lp=lp)
|
|
new_sid = get_sid_for_restore(remote_sam, logger)
|
|
|
|
# Grab the remote DC's sysvol files and bundle them into a tar file.
|
|
# Note we end up with 2 sysvol dirs - the original domain's files (that
|
|
# use the old realm) backed here, as well as default files generated
|
|
# for the new realm as part of the clone/join.
|
|
sysvol_tar = os.path.join(tmpdir, 'sysvol.tar.gz')
|
|
smb_conn = smb_sysvol_conn(server, lp, creds)
|
|
backup_online(smb_conn, sysvol_tar, remote_sam.get_domain_sid())
|
|
|
|
# connect to the local DB (making sure we use the new/renamed config)
|
|
lp.load(paths.smbconf)
|
|
samdb = SamDB(url=paths.samdb, session_info=system_session(), lp=lp,
|
|
flags=ldb.FLG_DONT_CREATE_DB)
|
|
|
|
# Edit the cloned sam.ldb to mark it as a backup
|
|
time_str = get_timestamp()
|
|
add_backup_marker(samdb, "backupDate", time_str)
|
|
add_backup_marker(samdb, "sidForRestore", new_sid)
|
|
add_backup_marker(samdb, "backupRename", old_realm)
|
|
add_backup_marker(samdb, "backupType", "rename")
|
|
|
|
# fix up the DNS objects that are using the old dnsRoot value
|
|
self.update_dns_root(logger, samdb, old_realm, delete_old_dns)
|
|
|
|
# update the netBIOS name and the Partition object for the domain
|
|
self.rename_domain_partition(logger, samdb, new_domain_name)
|
|
|
|
if delete_old_dns:
|
|
self.delete_old_dns_zones(logger, samdb, old_realm)
|
|
|
|
logger.info("Fixing DN attributes after rename...")
|
|
self.fix_old_dn_attributes(samdb)
|
|
|
|
# ensure the admin user always has a password set (same as provision)
|
|
if no_secrets:
|
|
set_admin_password(logger, samdb)
|
|
|
|
# Add everything in the tmpdir to the backup tar file
|
|
backup_file = backup_filepath(targetdir, new_dns_realm, time_str)
|
|
create_log_file(tmpdir, lp, "rename", server, include_secrets,
|
|
"Original domain %s (NetBIOS), %s (DNS realm)" %
|
|
(old_domain, old_realm))
|
|
create_backup_tar(logger, tmpdir, backup_file)
|
|
|
|
shutil.rmtree(tmpdir)
|
|
|
|
|
|
class cmd_domain_backup_offline(samba.netcmd.Command):
|
|
'''Backup the local domain directories safely into a tar file.
|
|
|
|
Takes a backup copy of the current domain from the local files on disk,
|
|
with proper locking of the DB to ensure consistency. If the domain were to
|
|
undergo a catastrophic failure, then the backup file can be used to recover
|
|
the domain.
|
|
|
|
An offline backup differs to an online backup in the following ways:
|
|
- a backup can be created even if the DC isn't currently running.
|
|
- includes non-replicated attributes that an online backup wouldn't store.
|
|
- takes a copy of the raw database files, which has the risk that any
|
|
hidden problems in the DB are preserved in the backup.'''
|
|
|
|
synopsis = "%prog [options]"
|
|
takes_optiongroups = {
|
|
"sambaopts": options.SambaOptions,
|
|
}
|
|
|
|
takes_options = [
|
|
Option("--targetdir",
|
|
help="Output directory (required)",
|
|
type=str),
|
|
]
|
|
|
|
backup_ext = '.bak-offline'
|
|
|
|
def offline_tdb_copy(self, path):
|
|
backup_path = path + self.backup_ext
|
|
try:
|
|
tdb_copy(path, backup_path, readonly=True)
|
|
except CalledProcessError as copy_err:
|
|
# If the copy didn't work, check if it was caused by an EINVAL
|
|
# error on opening the DB. If so, it's a mutex locked database,
|
|
# which we can safely ignore.
|
|
try:
|
|
tdb.open(path)
|
|
except Exception as e:
|
|
if hasattr(e, 'errno') and e.errno == errno.EINVAL:
|
|
return
|
|
raise e
|
|
raise copy_err
|
|
|
|
except FileNotFoundError as e:
|
|
# tdbbackup tool was not found.
|
|
raise CommandError(e.strerror, e)
|
|
|
|
if not os.path.exists(backup_path):
|
|
s = "tdbbackup said backup succeeded but {0} not found"
|
|
raise CommandError(s.format(backup_path))
|
|
|
|
|
|
def offline_mdb_copy(self, path):
|
|
mdb_copy(path, path + self.backup_ext)
|
|
|
|
# Secrets databases are a special case: a transaction must be started
|
|
# on the secrets.ldb file before backing up that file and secrets.tdb
|
|
def backup_secrets(self, private_dir, lp, logger):
|
|
secrets_path = os.path.join(private_dir, 'secrets')
|
|
secrets_obj = Ldb(secrets_path + '.ldb', lp=lp,
|
|
flags=ldb.FLG_DONT_CREATE_DB)
|
|
logger.info('Starting transaction on ' + secrets_path)
|
|
secrets_obj.transaction_start()
|
|
self.offline_tdb_copy(secrets_path + '.ldb')
|
|
self.offline_tdb_copy(secrets_path + '.tdb')
|
|
secrets_obj.transaction_cancel()
|
|
|
|
# sam.ldb must have a transaction started on it before backing up
|
|
# everything in sam.ldb.d with the appropriate backup function.
|
|
#
|
|
# Obtains the sidForRestore (SID for the new DC) and returns it
|
|
# from under the transaction
|
|
def backup_smb_dbs(self, private_dir, samdb, lp, logger):
|
|
sam_ldb_path = os.path.join(private_dir, 'sam.ldb')
|
|
|
|
# First, determine if DB backend is MDB. Assume not unless there is a
|
|
# 'backendStore' attribute on @PARTITION containing the text 'mdb'
|
|
store_label = "backendStore"
|
|
res = samdb.search(base="@PARTITION", scope=ldb.SCOPE_BASE,
|
|
attrs=[store_label])
|
|
mdb_backend = store_label in res[0] and str(res[0][store_label][0]) == 'mdb'
|
|
|
|
# This is needed to keep this variable in scope until the end
|
|
# of the transaction.
|
|
res_iterator = None
|
|
|
|
copy_function = None
|
|
if mdb_backend:
|
|
logger.info('MDB backend detected. Using mdb backup function.')
|
|
copy_function = self.offline_mdb_copy
|
|
|
|
# We can't backup with a write transaction open, so get a
|
|
# read lock with a search_iterator().
|
|
#
|
|
# We have tests in lib/ldb/tests/python/api.py that the
|
|
# search iterator takes a read lock effective against a
|
|
# transaction. This in turn will ensure there are no
|
|
# transactions on either the main or sub-database, even if
|
|
# the read locks were not enforced globally (they are).
|
|
res_iterator = samdb.search_iterator()
|
|
else:
|
|
logger.info('Starting transaction on ' + sam_ldb_path)
|
|
copy_function = self.offline_tdb_copy
|
|
samdb.transaction_start()
|
|
|
|
logger.info(' backing up ' + sam_ldb_path)
|
|
self.offline_tdb_copy(sam_ldb_path)
|
|
sam_ldb_d = sam_ldb_path + '.d'
|
|
for sam_file in os.listdir(sam_ldb_d):
|
|
sam_file = os.path.join(sam_ldb_d, sam_file)
|
|
if sam_file.endswith('.ldb'):
|
|
logger.info(' backing up locked/related file ' + sam_file)
|
|
copy_function(sam_file)
|
|
elif sam_file.endswith('.tdb'):
|
|
logger.info(' tdbbackup of locked/related file ' + sam_file)
|
|
self.offline_tdb_copy(sam_file)
|
|
else:
|
|
logger.info(' copying locked/related file ' + sam_file)
|
|
shutil.copyfile(sam_file, sam_file + self.backup_ext)
|
|
|
|
sid = get_sid_for_restore(samdb, logger)
|
|
|
|
if mdb_backend:
|
|
# Delete the iterator, release the read lock
|
|
del(res_iterator)
|
|
else:
|
|
samdb.transaction_cancel()
|
|
|
|
return sid
|
|
|
|
# Find where a path should go in the fixed backup archive structure.
|
|
def get_arc_path(self, path, conf_paths):
|
|
backup_dirs = {"private": conf_paths.private_dir,
|
|
"state": conf_paths.state_dir,
|
|
"etc": os.path.dirname(conf_paths.smbconf)}
|
|
matching_dirs = [(_, p) for (_, p) in backup_dirs.items() if
|
|
path.startswith(p)]
|
|
arc_path, fs_path = matching_dirs[0]
|
|
|
|
# If more than one directory is a parent of this path, then at least
|
|
# one configured path is a subdir of another. Use closest match.
|
|
if len(matching_dirs) > 1:
|
|
arc_path, fs_path = max(matching_dirs, key=lambda p: len(p[1]))
|
|
arc_path += path[len(fs_path):]
|
|
|
|
return arc_path
|
|
|
|
def run(self, sambaopts=None, targetdir=None):
|
|
|
|
logger = logging.getLogger()
|
|
logger.setLevel(logging.DEBUG)
|
|
logger.addHandler(logging.StreamHandler(sys.stdout))
|
|
|
|
# Get the absolute paths of all the directories we're going to backup
|
|
lp = sambaopts.get_loadparm()
|
|
|
|
paths = samba.provision.provision_paths_from_lp(lp, lp.get('realm'))
|
|
if not (paths.samdb and os.path.exists(paths.samdb)):
|
|
logger.error("No database found at {0}".format(paths.samdb))
|
|
raise CommandError('Please check you are root, and ' +
|
|
'are running this command on an AD DC')
|
|
|
|
check_targetdir(logger, targetdir)
|
|
|
|
# Iterating over the directories in this specific order ensures that
|
|
# when the private directory contains hardlinks that are also contained
|
|
# in other directories to be backed up (such as in paths.binddns_dir),
|
|
# the hardlinks in the private directory take precedence.
|
|
backup_dirs = [paths.private_dir, paths.state_dir,
|
|
os.path.dirname(paths.smbconf)] # etc dir
|
|
logger.info('running backup on dirs: {0}'.format(' '.join(backup_dirs)))
|
|
|
|
# Recursively get all file paths in the backup directories
|
|
all_files = []
|
|
all_stats = set()
|
|
for backup_dir in backup_dirs:
|
|
for (working_dir, _, filenames) in os.walk(backup_dir):
|
|
if working_dir.startswith(paths.sysvol):
|
|
continue
|
|
if working_dir.endswith('.sock') or '.sock/' in working_dir:
|
|
continue
|
|
# The BIND DNS database can be regenerated, so it doesn't need
|
|
# to be backed up.
|
|
if working_dir.startswith(os.path.join(paths.binddns_dir, 'dns')):
|
|
continue
|
|
|
|
for filename in filenames:
|
|
full_path = os.path.join(working_dir, filename)
|
|
|
|
# Ignore files that have already been added. This prevents
|
|
# duplicates if one backup dir is a subdirectory of another,
|
|
# or if backup dirs contain hardlinks.
|
|
try:
|
|
s = os.stat(full_path, follow_symlinks=False)
|
|
except FileNotFoundError:
|
|
logger.warning(f"{full_path} does not exist!")
|
|
continue
|
|
|
|
if (s.st_ino, s.st_dev) in all_stats:
|
|
continue
|
|
|
|
# Assume existing backup files are from a previous backup.
|
|
# Delete and ignore.
|
|
if filename.endswith(self.backup_ext):
|
|
os.remove(full_path)
|
|
continue
|
|
|
|
# Sock files are autogenerated at runtime, ignore.
|
|
if filename.endswith('.sock'):
|
|
continue
|
|
|
|
all_files.append(full_path)
|
|
all_stats.add((s.st_ino, s.st_dev))
|
|
|
|
# We would prefer to open with FLG_RDONLY but then we can't
|
|
# start a transaction which is the strong isolation we want
|
|
# for the backup.
|
|
samdb = SamDB(url=paths.samdb, session_info=system_session(), lp=lp,
|
|
flags=ldb.FLG_DONT_CREATE_DB)
|
|
|
|
# Backup secrets, sam.ldb and their downstream files
|
|
self.backup_secrets(paths.private_dir, lp, logger)
|
|
sid = self.backup_smb_dbs(paths.private_dir, samdb, lp, logger)
|
|
|
|
# Get the domain SID so we can later place it in the backup
|
|
dom_sid_str = samdb.get_domain_sid()
|
|
dom_sid = security.dom_sid(dom_sid_str)
|
|
|
|
# Close the original samdb, to avoid any confusion, we will
|
|
# not use this any more as the data has all been copied under
|
|
# the transaction
|
|
samdb = None
|
|
|
|
# Open the new backed up samdb, flag it as backed up, and write
|
|
# the next SID so the restore tool can add objects. We use
|
|
# options=["modules:"] here to prevent any modules from loading.
|
|
# WARNING: Don't change this code unless you know what you're doing.
|
|
# Writing to a .bak file only works because the DN being
|
|
# written to happens to be top level.
|
|
samdb = Ldb(url=paths.samdb + self.backup_ext,
|
|
session_info=system_session(), lp=lp,
|
|
options=["modules:"], flags=ldb.FLG_DONT_CREATE_DB)
|
|
time_str = get_timestamp()
|
|
add_backup_marker(samdb, "backupDate", time_str)
|
|
add_backup_marker(samdb, "sidForRestore", sid)
|
|
add_backup_marker(samdb, "backupType", "offline")
|
|
|
|
# Close the backed up samdb
|
|
samdb = None
|
|
|
|
# Now handle all the LDB and TDB files that are not linked to
|
|
# anything else. Use transactions for LDBs.
|
|
for path in all_files:
|
|
if not os.path.exists(path + self.backup_ext):
|
|
if path.endswith('.ldb'):
|
|
logger.info('Starting transaction on solo db: ' + path)
|
|
ldb_obj = Ldb(path, lp=lp, flags=ldb.FLG_DONT_CREATE_DB)
|
|
ldb_obj.transaction_start()
|
|
logger.info(' running tdbbackup on the same file')
|
|
self.offline_tdb_copy(path)
|
|
ldb_obj.transaction_cancel()
|
|
elif path.endswith('.tdb'):
|
|
logger.info('running tdbbackup on lone tdb file ' + path)
|
|
self.offline_tdb_copy(path)
|
|
|
|
# Now make the backup tar file and add all
|
|
# backed up files and any other files to it.
|
|
temp_tar_dir = tempfile.mkdtemp(dir=targetdir,
|
|
prefix='INCOMPLETEsambabackupfile')
|
|
temp_tar_name = os.path.join(temp_tar_dir, "samba-backup.tar.bz2")
|
|
tar = tarfile.open(temp_tar_name, 'w:bz2')
|
|
|
|
logger.info('running offline ntacl backup of sysvol')
|
|
sysvol_tar_fn = 'sysvol.tar.gz'
|
|
sysvol_tar = os.path.join(temp_tar_dir, sysvol_tar_fn)
|
|
backup_offline(paths.sysvol, sysvol_tar, paths.smbconf, dom_sid)
|
|
tar.add(sysvol_tar, sysvol_tar_fn)
|
|
os.remove(sysvol_tar)
|
|
|
|
create_log_file(temp_tar_dir, lp, "offline", "localhost", True)
|
|
backup_fn = os.path.join(temp_tar_dir, "backup.txt")
|
|
tar.add(backup_fn, os.path.basename(backup_fn))
|
|
os.remove(backup_fn)
|
|
|
|
logger.info('building backup tar')
|
|
for path in all_files:
|
|
arc_path = self.get_arc_path(path, paths)
|
|
|
|
if os.path.exists(path + self.backup_ext):
|
|
logger.info(' adding backup ' + arc_path + self.backup_ext +
|
|
' to tar and deleting file')
|
|
tar.add(path + self.backup_ext, arcname=arc_path)
|
|
os.remove(path + self.backup_ext)
|
|
elif path.endswith('.ldb') or path.endswith('.tdb'):
|
|
logger.info(' skipping ' + arc_path)
|
|
else:
|
|
logger.info(' adding misc file ' + arc_path)
|
|
tar.add(path, arcname=arc_path)
|
|
|
|
tar.close()
|
|
os.rename(temp_tar_name,
|
|
os.path.join(targetdir,
|
|
'samba-backup-{0}.tar.bz2'.format(time_str)))
|
|
os.rmdir(temp_tar_dir)
|
|
logger.info('Backup succeeded.')
|
|
|
|
|
|
class cmd_domain_backup(samba.netcmd.SuperCommand):
|
|
'''Create or restore a backup of the domain.'''
|
|
subcommands = {'offline': cmd_domain_backup_offline(),
|
|
'online': cmd_domain_backup_online(),
|
|
'rename': cmd_domain_backup_rename(),
|
|
'restore': cmd_domain_backup_restore()}
|