All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.firebirdsql.management.FBBackupManagerBase Maven / Gradle / Ivy

The newest version!
/*
 * Firebird Open Source JDBC Driver
 *
 * Distributable under LGPL license.
 * You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
 *
 * 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
 * LGPL License for more details.
 *
 * This file was created by members of the firebird development team.
 * All individual contributions remain the Copyright (C) of those
 * individuals.  Contributors to this file are either listed here or
 * can be obtained from a source control history command.
 *
 * All rights reserved.
 */
package org.firebirdsql.management;

import org.firebirdsql.gds.ServiceRequestBuffer;
import org.firebirdsql.gds.impl.GDSType;
import org.firebirdsql.gds.ng.FbService;
import org.firebirdsql.jaybird.fb.constants.SpbItems;

import java.sql.SQLException;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import static org.firebirdsql.gds.ISCConstants.*;
import static org.firebirdsql.util.FirebirdSupportInfo.supportInfoFor;

/**
 * Implements the common functionality between regular and streaming backup/restore
 *
 * @author Roman Rokytskyy
 * @author Mark Rotteveel
 */
public abstract class FBBackupManagerBase extends FBServiceManager implements BackupManager {

    /**
     * Structure that holds path to the database and corresponding size of the file (in case of backup - that is
     * size of the file in megabytes, in case of restore - size of the database file in pages).
     */
    protected static class PathSizeStruct {
        private final int size;
        private final String path;

        protected PathSizeStruct(String path, int size) {
            this.path = path;
            this.size = size;
        }

        public String getPath() {
            return path;
        }

        public int getSize() {
            return size;
        }

        public boolean equals(Object obj) {
            if (obj == this) return true;
            if (!(obj instanceof PathSizeStruct that)) return false;

            return this.path.equals(that.path);
        }

        public int hashCode() {
            return path.hashCode();
        }

        public String toString() {
            return path + " " + size;
        }
    }

    protected boolean noLimitRestore;
    protected final List restorePaths = new ArrayList<>();

    protected boolean verbose;

    private int restoreBufferCount = -1;
    private int restorePageSize = -1;
    private boolean restoreReadOnly;
    private boolean restoreReplace;

    private static final int RESTORE_REPLACE = isc_spb_res_replace;
    private static final int RESTORE_CREATE = isc_spb_res_create;

    /**
     * Create a new instance of FBBackupManagerBase based on the default GDSType.
     */
    protected FBBackupManagerBase() {
    }

    /**
     * Create a new instance of FBBackupManagerBase based on a given GDSType.
     *
     * @param gdsType
     *         type must be PURE_JAVA, EMBEDDED, or NATIVE
     */
    protected FBBackupManagerBase(String gdsType) {
        super(gdsType);
    }

    /**
     * Create a new instance of FBBackupManagerBase based on a given GDSType.
     *
     * @param gdsType
     *         type must be PURE_JAVA, EMBEDDED, or NATIVE
     */
    protected FBBackupManagerBase(GDSType gdsType) {
        super(gdsType);
    }

    @Override
    public void addBackupPath(String path) {
        addBackupPath(path, -1);
    }

    @Override
    public void setDatabase(String database) {
        super.setDatabase(database);
        addRestorePath(database, -1);
        noLimitRestore = true;
    }

    @Override
    public void addRestorePath(String path, int size) {
        if (noLimitRestore) {
            throw new IllegalArgumentException(
                    "You cannot use setDatabase(String) and addRestorePath(String, int) methods simultaneously.");
        }
        restorePaths.add(new PathSizeStruct(path, size));
    }

    @Override
    public void clearRestorePaths() {
        restorePaths.clear();
        noLimitRestore = false;
    }

    @Override
    public void backupDatabase() throws SQLException {
        backupDatabase(0);
    }

    @Override
    public void backupMetadata() throws SQLException {
        backupDatabase(BACKUP_METADATA_ONLY);
    }

    /**
     * Creates and returns the "backup" service request buffer for the Service Manager.
     *
     * @param service
     *         Service handle
     * @param options
     *         The isc_spb_bkp_* parameters options to be used
     * @return the "backup" service request buffer for the Service Manager.
     */
    protected ServiceRequestBuffer getBackupSRB(FbService service, int options) throws SQLException {
        ServiceRequestBuffer backupSPB = service.createServiceRequestBuffer();
        backupSPB.addArgument(isc_action_svc_backup);
        backupSPB.addArgument(SpbItems.isc_spb_dbname, getDatabase());
        addBackupsToBackupRequestBuffer(service, backupSPB);

        if (verboseBackup()) {
            backupSPB.addArgument(SpbItems.isc_spb_verbose);
        }

        if (getParallelWorkers() > 0 && supportInfoFor(service).supportsParallelWorkers())  {
            backupSPB.addArgument(isc_spb_bkp_parallel_workers, getParallelWorkers());
        }

        backupSPB.addArgument(SpbItems.isc_spb_options, options);

        return backupSPB;
    }

    @Override
    public void restoreDatabase() throws SQLException {
        restoreDatabase(0);
    }

    /**
     * Set whether the operations of this {@code BackupManager} will result in verbose logging to the configured logger.
     *
     * @param verbose
     *         If true, operations will be logged verbosely, otherwise they will not be logged verbosely
     */
    @Override
    public void setVerbose(boolean verbose) {
        this.verbose = verbose;
    }

    /**
     * Set the default number of pages to be buffered (cached) by default in a restored database.
     *
     * @param bufferCount
     *         The page-buffer size to be used, a positive value
     */
    @Override
    public void setRestorePageBufferCount(int bufferCount) {
        if (bufferCount < 0) {
            throw new IllegalArgumentException("Buffer count must be positive");
        }
        this.restoreBufferCount = bufferCount;
    }

    /**
     * Set the page size that will be used for a restored database. The value for {@code pageSize} must be
     * one of {@link PageSizeConstants}. The default value depends on the Firebird version.
     * 

* Be aware that not all page sizes are supported by all Firebird versions. *

* * @param pageSize * The page size to be used in a restored database, see {@link PageSizeConstants} * @see PageSizeConstants */ @Override public void setRestorePageSize(int pageSize) { this.restorePageSize = PageSizeConstants.requireValidPageSize(pageSize); } /** * Set the restore operation to create a new database, as opposed to overwriting an existing database. This is true * by default. * * @param replace * If true, the restore operation will attempt to create a new database, otherwise the restore * operation will overwrite an existing database */ @Override public void setRestoreReplace(boolean replace) { this.restoreReplace = replace; } /** * Set the read-only attribute on a restored database. * * @param readOnly * If true, a restored database will be read-only, otherwise it will be read-write. */ @Override public void setRestoreReadOnly(boolean readOnly) { this.restoreReadOnly = readOnly; } /** * Creates and returns the "backup" service request buffer for the Service Manager. * * @param service * Service handle * @param options * The options to be used for the backup operation * @return the "backup" service request buffer for the Service Manager. */ protected ServiceRequestBuffer getRestoreSRB(FbService service, int options) { ServiceRequestBuffer restoreSPB = service.createServiceRequestBuffer(); restoreSPB.addArgument(isc_action_svc_restore); // restore files with sizes except the last one for (Iterator iter = restorePaths.iterator(); iter.hasNext(); ) { PathSizeStruct pathSize = iter.next(); restoreSPB.addArgument(SpbItems.isc_spb_dbname, pathSize.getPath()); if (iter.hasNext() && pathSize.getSize() != -1) { restoreSPB.addArgument(isc_spb_res_length, pathSize.getSize()); } } addBackupsToRestoreRequestBuffer(service, restoreSPB); if (restoreBufferCount != -1) { restoreSPB.addArgument(isc_spb_res_buffers, restoreBufferCount); } if (restorePageSize != -1) { restoreSPB.addArgument(isc_spb_res_page_size, restorePageSize); } restoreSPB.addArgument(isc_spb_res_access_mode, (byte) (restoreReadOnly ? isc_spb_res_am_readonly : isc_spb_res_am_readwrite)); if (verbose) { restoreSPB.addArgument(SpbItems.isc_spb_verbose); } if (getParallelWorkers() > 0 && supportInfoFor(service).supportsParallelWorkers()) { restoreSPB.addArgument(isc_spb_res_parallel_workers, getParallelWorkers()); } if ((options & RESTORE_CREATE) != RESTORE_CREATE && (options & RESTORE_REPLACE) != RESTORE_REPLACE) { options |= restoreReplace ? RESTORE_REPLACE : RESTORE_CREATE; } restoreSPB.addArgument(SpbItems.isc_spb_options, options); return restoreSPB; } /** * Adds the backup source for the backup opration, depending on the manager used * * @param backupSPB * The buffer to be used during the backup operation */ protected abstract void addBackupsToBackupRequestBuffer(FbService service, ServiceRequestBuffer backupSPB) throws SQLException; /** * Adds the backup files to be used during restore */ protected abstract void addBackupsToRestoreRequestBuffer(FbService service, ServiceRequestBuffer restoreSPB); /** * Whether the backup will produce verbose output */ protected abstract boolean verboseBackup(); }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy