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

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

The newest version!
/*
 * Public Firebird Java API.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *    1. Redistributions of source code must retain the above copyright notice,
 *       this list of conditions and the following disclaimer.
 *    2. Redistributions in binary form must reproduce the above copyright
 *       notice, this list of conditions and the following disclaimer in the
 *       documentation and/or other materials provided with the distribution.
 *    3. The name of the author may not be used to endorse or promote products
 *       derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package org.firebirdsql.management;

import org.firebirdsql.gds.ServiceRequestBuffer;
import org.firebirdsql.gds.impl.GDSType;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.FbService;
import org.firebirdsql.util.NumericHelper;

import java.io.ByteArrayOutputStream;
import java.io.OutputStream;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;

import static org.firebirdsql.gds.ISCConstants.*;
import static org.firebirdsql.gds.VaxEncoding.iscVaxInteger2;
import static org.firebirdsql.gds.VaxEncoding.iscVaxLong;

/**
 * The {@code FBMaintenanceManager} class is responsible for replicating the functionality provided by
 * the {@code gfix} command-line tool.
 * 

* Among the responsibilities of this class are: *

    *
  • Database shutdown
  • *
  • Extended database shutdown/online modes new with Firebird 2.5
  • *
  • Changing database mode to read-only or read-write
  • *
  • Enabling or disabling forced writes in the database
  • *
  • Changing the dialect of the database
  • *
  • Setting the cache size at database-level
  • *
  • Mending databases and making minor repairs
  • *
  • Sweeping databases
  • *
  • Activating and killing shadow files
  • *
  • Displaying, committing, or recovering limbo transactions
  • *
*

* * @author Gabriel Reid * @author Thomas Steinmaurer * @author Mark Rotteveel */ public class FBMaintenanceManager extends FBServiceManager implements MaintenanceManager { /** * Create a new instance of {@code FBMaintenanceManager} based on the default GDSType. */ @SuppressWarnings("unused") public FBMaintenanceManager() { super(); } /** * Create a new instance of {@code FBMaintenanceManager} based on a given GDSType. * * @param gdsType * type must be PURE_JAVA, EMBEDDED, or NATIVE */ @SuppressWarnings("unused") public FBMaintenanceManager(String gdsType) { super(gdsType); } /** * Create a new instance of {@code FBMaintenanceManager} based on a given GDSType. * * @param gdsType * The GDS implementation type to use */ public FBMaintenanceManager(GDSType gdsType) { super(gdsType); } @Override public void setDatabaseAccessMode(int mode) throws SQLException { if (mode != ACCESS_MODE_READ_WRITE && mode != ACCESS_MODE_READ_ONLY) { throw new IllegalArgumentException("mode must be one of ACCESS_MODE_READ_WRITE or ACCESS_MODE_READ_ONLY"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_access_mode, (byte) mode); executeServicesOperation(service, srb); } } @Override public void setDatabaseDialect(int dialect) throws SQLException { if (dialect != 1 && dialect != 3) { throw new IllegalArgumentException("dialect must be either 1 or 3"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_set_sql_dialect, dialect); executeServicesOperation(service, srb); } } @Override public void setDefaultCacheBuffer(int pageCount) throws SQLException { if (pageCount != 0 && pageCount < 50) { throw new IllegalArgumentException("page count must be 0 or >= 50, value was: " + pageCount); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_page_buffers, pageCount); executeServicesOperation(service, srb); } } @Override public void setForcedWrites(boolean forced) throws SQLException { try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_write_mode, (byte) (forced ? isc_spb_prp_wm_sync : isc_spb_prp_wm_async)); executeServicesOperation(service, srb); } } @Override public void setPageFill(int pageFill) throws SQLException { if (pageFill != PAGE_FILL_FULL && pageFill != PAGE_FILL_RESERVE) { throw new IllegalArgumentException("Page fill must be either PAGE_FILL_FULL or PAGE_FILL_RESERVE"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_reserve_space, (byte) pageFill); executeServicesOperation(service, srb); } } // ----------- Database Shutdown ------------------- @Override public void shutdownDatabase(int shutdownMode, int timeout) throws SQLException { if (shutdownMode != SHUTDOWN_ATTACH && shutdownMode != SHUTDOWN_TRANSACTIONAL && shutdownMode != SHUTDOWN_FORCE) { throw new IllegalArgumentException("Shutdown mode must be " + "one of: SHUTDOWN_ATTACH, SHUTDOWN_TRANSACTIONAL, SHUTDOWN_FORCE"); } if (timeout < 0) { throw new IllegalArgumentException("Timeout must be >= 0"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(shutdownMode, timeout); executeServicesOperation(service, srb); } } @Override public void shutdownDatabase(byte operationMode, int shutdownModeEx, int timeout) throws SQLException { if (operationMode != OPERATION_MODE_MULTI && operationMode != OPERATION_MODE_SINGLE && operationMode != OPERATION_MODE_FULL_SHUTDOWN) { throw new IllegalArgumentException("Operation mode must be one of: OPERATION_MODE_MULTI, " + "OPERATION_MODE_SINGLE, OPERATION_MODE_FULL_SHUTDOWN"); } if (shutdownModeEx != SHUTDOWNEX_FORCE && shutdownModeEx != SHUTDOWNEX_ATTACHMENTS && shutdownModeEx != SHUTDOWNEX_TRANSACTIONS) { throw new IllegalArgumentException("Extended shutdown mode must be " + "one of: SHUTDOWNEX_FORCE, SHUTDOWNEX_ATTACHMENTS, SHUTDOWNEX_TRANSACTIONS"); } if (timeout < 0) { throw new IllegalArgumentException("Timeout must be >= 0"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_shutdown_mode, operationMode); srb.addArgument(shutdownModeEx, timeout); executeServicesOperation(service, srb); } } @Override public void bringDatabaseOnline() throws SQLException { executePropertiesOperation(isc_spb_prp_db_online); } @Override public void bringDatabaseOnline(byte operationMode) throws SQLException { if (operationMode != OPERATION_MODE_NORMAL && operationMode != OPERATION_MODE_MULTI && operationMode != OPERATION_MODE_SINGLE) { throw new IllegalArgumentException("Operation mode must be " + "one of: OPERATION_MODE_NORMAL, OPERATION_MODE_MULTI, OPERATION_MODE_SINGLE"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_online_mode, operationMode); executeServicesOperation(service, srb); } } // -------------- Database Repair ---------------------- @Override public void markCorruptRecords() throws SQLException { executeRepairOperation(isc_spb_rpr_mend_db); } @Override public void validateDatabase() throws SQLException { executeRepairOperation(isc_spb_rpr_validate_db); } @Override public void validateDatabase(int options) throws SQLException { if (options < 0 || options != 0 && options != VALIDATE_IGNORE_CHECKSUM && (options & ~VALIDATE_IGNORE_CHECKSUM) != VALIDATE_READ_ONLY && (options & ~VALIDATE_IGNORE_CHECKSUM) != VALIDATE_FULL && (options | (VALIDATE_READ_ONLY | VALIDATE_IGNORE_CHECKSUM)) != options && (options | (VALIDATE_FULL | VALIDATE_IGNORE_CHECKSUM)) != options) { throw new IllegalArgumentException("options must be either 0, " + "VALIDATE_READ_ONLY, or VALIDATE_FULL, optionally combined with VALIDATE_IGNORE_CHECKSUM"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createRepairSRB(service, options | isc_spb_rpr_validate_db); executeServicesOperation(service, srb); } } // ----------- Sweeping ------------------------- @Override public void setSweepThreshold(int transactions) throws SQLException { if (transactions < 0) { throw new IllegalArgumentException("transactions must be >= 0"); } try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultPropertiesSRB(service); srb.addArgument(isc_spb_prp_sweep_interval, transactions); executeServicesOperation(service, srb); } } @Override public void sweepDatabase() throws SQLException { executeRepairOperation(isc_spb_rpr_sweep_db); } // ----------- Shadow Files ------------------------------------ @Override public void activateShadowFile() throws SQLException { executePropertiesOperation(isc_spb_prp_activate); } @Override public void killUnavailableShadows() throws SQLException { executeRepairOperation(isc_spb_rpr_kill_shadows); } // ----------- Transaction Management ---------------------------- @Override @SuppressWarnings("java:S2093") public List limboTransactionsAsList() throws SQLException { // See also fbscvmgr.cpp method printInfo final OutputStream saveOut = getLogger(); final List result = new ArrayList<>(); final ByteArrayOutputStream out = new ByteArrayOutputStream(); final byte[] output; try { setLogger(out); executeRepairOperation(isc_spb_rpr_list_limbo_trans); output = out.toByteArray(); } finally { setLogger(saveOut); } int idx = 0; while (idx < output.length) { switch (output[idx++]) { case isc_spb_single_tra_id, isc_spb_multi_tra_id -> { long trId = iscVaxLong(output, idx, 4); idx += 4; result.add(trId); } case isc_spb_single_tra_id_64, isc_spb_multi_tra_id_64 -> { long trId = iscVaxLong(output, idx, 8); idx += 8; result.add(trId); } // Information items we will ignore for now case isc_spb_tra_id -> idx += 4; case isc_spb_tra_id_64 -> idx += 8; case isc_spb_tra_state, isc_spb_tra_advise -> idx++; case isc_spb_tra_host_site, isc_spb_tra_remote_site, isc_spb_tra_db_path -> { int length = iscVaxInteger2(output, idx); idx += 2; idx += length; } default -> throw FbExceptionBuilder.forException(isc_fbsvcmgr_info_err) .messageParameter(output[idx - 1] & 0xFF) .toSQLException(); } } return result; } @Override public long[] getLimboTransactions() throws SQLException { final List limboTransactions = limboTransactionsAsList(); final long[] trans = new long[limboTransactions.size()]; int idx = 0; for (long trId : limboTransactions) { trans[idx++] = trId; } return trans; } @Override public void commitTransaction(final long transactionId) throws SQLException { handleTransaction(transactionId, isc_spb_rpr_commit_trans, isc_spb_rpr_commit_trans_64); } @Override public void rollbackTransaction(final long transactionId) throws SQLException { handleTransaction(transactionId, isc_spb_rpr_rollback_trans, isc_spb_rpr_rollback_trans_64); } private void handleTransaction(final long transactionId, final int action32bit, final int action64bit) throws SQLException { if (transactionId < 0) { throw new SQLException("Only positive transactionIds are supported"); } final boolean is32Bit = NumericHelper.fitsUnsigned32BitInteger(transactionId); try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createDefaultRepairSRB(service); srb.addArgument(is32Bit ? action32bit : action64bit, transactionId); executeServicesOperation(service, srb); } } @Override public void upgradeOds() throws SQLException { executeRepairOperation(isc_spb_rpr_upgrade_db); } @Override public void fixIcu() throws SQLException { executeRepairOperation(isc_spb_rpr_icu); } // ----------- Private implementation methods -------------------- /** * Execute an isc_spb_rpr_* (repair) services operation. * * @param operation * The identifier for the operation to be executed * @throws SQLException * if a database access error occurs */ private void executeRepairOperation(int operation) throws SQLException { try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createRepairSRB(service, operation); executeServicesOperation(service, srb); } } /** * Execute a isc_spb_prp_* (properties) services operation. * * @param operation * The identifier for the operation to be executed * @throws SQLException * if a database access error occurs */ private void executePropertiesOperation(int operation) throws SQLException { try (FbService service = attachServiceManager()) { ServiceRequestBuffer srb = createPropertiesSRB(service, operation); executeServicesOperation(service, srb); } } /** * Get a mostly empty properties-operation buffer that can be filled in as * needed. The buffer created by this method cannot have the options * bitmask set on it. */ private ServiceRequestBuffer createDefaultPropertiesSRB(FbService service) { return createPropertiesSRB(service, 0); } /** * Get a mostly empty repair-operation buffer that can be filled in as * needed. The buffer created by this method cannot have the options * bitmask set on it. */ private ServiceRequestBuffer createDefaultRepairSRB(FbService service) { return createRepairSRB(service, 0); } /** * Get a mostly-empty properties-operation request buffer that can be * filled as needed. * * @param service * Service handle * @param options * The options bitmask for the request buffer */ private ServiceRequestBuffer createPropertiesSRB(FbService service, int options) { return createRequestBuffer(service, isc_action_svc_properties, options); } /** * Get a mostly-empty repair-operation request buffer that can be filled as needed. * * @param service * Service handle * @param options * The options bitmask for the request buffer */ private ServiceRequestBuffer createRepairSRB(FbService service, int options) { ServiceRequestBuffer srb = createRequestBuffer(service, isc_action_svc_repair, options); if (getParallelWorkers() > 0 && (options & getParallelRepairOptions(service)) != 0) { srb.addArgument(isc_spb_rpr_par_workers, getParallelWorkers()); } return srb; } /** * Bitmask of repair options which support parallel workers. * * @param service service attachment, to determine supported options * @return bitmask of repair options */ private int getParallelRepairOptions(FbService service) { if (service.getServerVersion().isEqualOrAbove(5, 0)) { return isc_spb_rpr_sweep_db | isc_spb_rpr_icu; } else { return 0; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy