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

com.orientechnologies.agent.services.backup.strategy.OBackupStrategyMixBackup Maven / Gradle / Ivy

/*
 * Copyright 2015 OrientDB LTD (info(at)orientdb.com)
 *
 *   Licensed under the Apache License, Version 2.0 (the "License");
 *   you may not use this file except in compliance with the License.
 *   You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 *   Unless required by applicable law or agreed to in writing, software
 *   distributed under the License is distributed on an "AS IS" BASIS,
 *   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *   See the License for the specific language governing permissions and
 *   limitations under the License.
 *
 *   For more information: http://www.orientdb.com
 */

package com.orientechnologies.agent.services.backup.strategy;

import com.orientechnologies.agent.services.backup.OBackupConfig;
import com.orientechnologies.agent.services.backup.OBackupListener;
import com.orientechnologies.agent.services.backup.log.*;
import com.orientechnologies.common.log.OLogManager;
import com.orientechnologies.orient.core.db.ODatabaseInternal;
import com.orientechnologies.orient.core.db.document.ODatabaseDocument;
import com.orientechnologies.orient.core.record.impl.ODocument;
import com.orientechnologies.orient.core.schedule.OCronExpression;
import com.orientechnologies.orient.core.storage.impl.local.paginated.OEnterpriseLocalPaginatedStorage;
import com.orientechnologies.orient.server.handler.OAutomaticBackup;
import java.io.File;
import java.io.IOException;
import java.text.ParseException;
import java.util.Date;

/** Created by Enrico Risa on 25/03/16. */
public class OBackupStrategyMixBackup extends OBackupStrategy {
  // Make sure not to displace / skip a full backup (e.g. in case an incremental backup took
  // longer). Hence (1) store last (and potentially skipped) full backup.
  private Date skippedFull = null;

  protected boolean isIncremental = false;

  public OBackupStrategyMixBackup(ODocument cfg, OBackupLogger logger) {
    super(cfg, logger);
  }

  @Override
  public OAutomaticBackup.MODE getMode() {
    return isIncremental
        ? OAutomaticBackup.MODE.INCREMENTAL_BACKUP
        : OAutomaticBackup.MODE.FULL_BACKUP;
  }

  protected String calculatePath(ODatabaseDocument db) {
    if (!isIncremental) {
      return defaultPath();
    }
    OBackupFinishedLog last = null;
    try {
      last = (OBackupFinishedLog) logger.findLast(OBackupLogType.BACKUP_FINISHED, getUUID());
    } catch (Exception e) {
      OLogManager.instance().error(this, "Error " + e.getMessage(), e);
    }
    try {
      if (last != null
          && ((OEnterpriseLocalPaginatedStorage) ((ODatabaseInternal) db).getStorage())
              .isLastBackupCompatibleWithUUID(new File(last.getPath()))) {
        return last.getPath();
      }
    } catch (IOException e) {
      OLogManager.instance().warn(this, "Error checking backup compatibility", e);
      return last.getPath();
    }

    return defaultPath();
  }

  protected String defaultPath() {
    long begin = System.currentTimeMillis();
    try {
      OBackupLog last = logger.findLast(OBackupLogType.BACKUP_SCHEDULED, getUUID());
      if (last != null) {
        begin = last.getUnitId();
      }
    } catch (IOException e) {
      OLogManager.instance().error(this, "Error " + e.getMessage(), e);
    }
    String basePath = cfg.field(OBackupConfig.DIRECTORY);
    String dbName = cfg.field(OBackupConfig.DBNAME);
    return basePath + File.separator + dbName + "-" + begin;
  }

  @Override
  public Date scheduleNextExecution(final OBackupListener listener) {
    final OBackupScheduledLog lastBackupSchedule = lastUnfiredSchedule();
    if (lastBackupSchedule == null) {
      final ODocument full =
          (ODocument) cfg.eval(OBackupConfig.MODES + "." + OAutomaticBackup.MODE.FULL_BACKUP);
      final String whenFull = full.field(OBackupConfig.WHEN);
      final ODocument incremental =
          (ODocument)
              cfg.eval(OBackupConfig.MODES + "." + OAutomaticBackup.MODE.INCREMENTAL_BACKUP);
      final String whenIncremental = incremental.field(OBackupConfig.WHEN);
      try {
        final OCronExpression eFull = new OCronExpression(whenFull);
        final OCronExpression eIncremental = new OCronExpression(whenIncremental);
        final Date now = new Date();

        final Date nextFull = eFull.getNextValidTimeAfter(now);
        final Date nextIncremental = eIncremental.getNextValidTimeAfter(now);

        long unitId = logger.nextOpId();
        try {
          final OBackupFinishedLog lastCompleted =
              (OBackupFinishedLog) logger.findLast(OBackupLogType.BACKUP_FINISHED, getUUID());
          if (lastCompleted != null
              && nextIncremental.before(nextFull)
              && !Boolean.TRUE.equals(lastCompleted.getPrevChange())) {
            unitId = lastCompleted.getUnitId();

            // (2) when deciding on incremental check for skipped full backup that actually came
            // before the next incremental.
            if (skippedFull != null && skippedFull.before(nextIncremental)) {
              OLogManager.instance()
                  .info(this, "Found skipped full backup (i.e. before incremental)");
              isIncremental = false;
              skippedFull = null;
            } else {
              // (3) else take incremental, but remember skipped full backup.
              OLogManager.instance()
                  .debug(this, "Found incremental backup before full (i.e. remember skipped full)");
              isIncremental = true;
              skippedFull = nextFull;
            }
          } else {
            // (4) if full exercised, forget skipped full backup.
            OLogManager.instance().debug(this, "Found full backup before incremental");
            isIncremental = false;
            skippedFull = null;
          }
        } catch (final IOException e) {
          OLogManager.instance().error(this, "Error " + e.getMessage(), e);
        }
        final Date nextExecution = nextIncremental.before(nextFull) ? nextIncremental : nextFull;
        final OBackupScheduledLog log =
            new OBackupScheduledLog(
                unitId, logger.nextOpId(), getUUID(), getDbName(), getMode().toString());
        log.nextExecution = nextExecution.getTime();
        getLogger().log(log);
        listener.onEvent(cfg, log);
        return nextExecution;
      } catch (final ParseException e) {
        OLogManager.instance().error(this, "Parse exception: " + e.getMessage(), e);
        return null;
      }
    } else {
      isIncremental =
          OAutomaticBackup.MODE
              .INCREMENTAL_BACKUP
              .toString()
              .equalsIgnoreCase(lastBackupSchedule.getMode());
      // (5) if last schedule not null and next backup is incremental, preserve skipped full
      // backup, else forget.
      final Date nextExecution = new Date(lastBackupSchedule.nextExecution);
      if (skippedFull != null && isIncremental == false) {
        skippedFull = null;
      }
      return nextExecution;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy