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

com.arcadedb.database.DatabaseComparator Maven / Gradle / Ivy

/*
 * Copyright © 2021-present Arcade Data Ltd ([email protected])
 *
 * 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.
 *
 * SPDX-FileCopyrightText: 2021-present Arcade Data Ltd ([email protected])
 * SPDX-License-Identifier: Apache-2.0
 */
package com.arcadedb.database;

import com.arcadedb.engine.Bucket;
import com.arcadedb.engine.ImmutablePage;
import com.arcadedb.engine.PageId;
import com.arcadedb.exception.ArcadeDBException;
import com.arcadedb.index.Index;
import com.arcadedb.schema.DocumentType;

import java.io.*;
import java.util.*;

/**
 * Compares 2 databases if are identical.
 */
public class DatabaseComparator {

  static public class DatabaseAreNotIdentical extends ArcadeDBException {
    public DatabaseAreNotIdentical(final String text, final Object... args) {
      super(String.format(text, args));
    }
  }

  public void compare(final Database db1, final Database db2) {
    compareTypes(db1, db2);
    compareBuckets((DatabaseInternal) db1, (DatabaseInternal) db2);
    compareIndexes(db1, db2);
  }

  public void compareBuckets(final DatabaseInternal db1, final DatabaseInternal db2) {
    final Collection buckets1 = db1.getSchema().getBuckets();
    final Collection buckets2 = db2.getSchema().getBuckets();
    if (buckets1.size() != buckets2.size())
      throw new DatabaseAreNotIdentical("Buckets: DB1 %d <> DB2 %d", buckets1.size(), buckets2.size());

    final HashMap types1Map = new HashMap<>(buckets1.size());
    final HashMap types2Map = new HashMap<>(buckets2.size());

    for (final Bucket entry : buckets1)
      types1Map.put(entry.getName(), entry);

    for (final Bucket entry : buckets2) {
      if (!types1Map.containsKey(entry.getName()))
        throw new DatabaseAreNotIdentical("Bucket '%s' is not present in DB2", entry.getName());
      types2Map.put(entry.getName(), entry);
    }

    for (final Bucket entry : buckets1)
      if (!types2Map.containsKey(entry.getName()))
        throw new DatabaseAreNotIdentical("Bucket '%s' is not present in DB1", entry.getName());

    for (final Bucket bucket1 : buckets1) {
      final Bucket bucket2 = types2Map.get(bucket1.getName());

      if (bucket1.getPageSize() != bucket2.getPageSize())
        throw new DatabaseAreNotIdentical("Bucket '%s' has different page size in two databases. DB1 %d <> DB2 %d", bucket2.getName(), bucket1.getPageSize(),
            bucket2.getPageSize());

      if (bucket1.getTotalPages() != bucket2.getTotalPages())
        throw new DatabaseAreNotIdentical("Bucket '%s' has different page count in two databases. DB1 %d <> DB2 %d", bucket2.getName(), bucket1.getTotalPages(),
            bucket2.getTotalPages());

      // AT THIS POINT BOTH BUCKETS HAVE THE SAME PAGES
      final int pageSize = bucket1.getPageSize();
      for (int i = 0; i < bucket1.getTotalPages(); ++i) {
        final PageId pageId = new PageId(bucket1.getId(), i);

        final ImmutablePage page1;
        final ImmutablePage page2;

        try {
          page1 = db1.getPageManager().getImmutablePage(pageId, pageSize, false, true);
        } catch (final IOException e) {
          throw new DatabaseAreNotIdentical("Error on reading page %s from bucket '%s' DB1 (cause=%s)", pageId, bucket1.getName(), e.toString());
        }

        try {
          page2 = db2.getPageManager().getImmutablePage(pageId, pageSize, false, true);
        } catch (final IOException e) {
          throw new DatabaseAreNotIdentical("Error on reading page %s from bucket '%s' DB2 (cause=%s)", pageId, bucket2.getName(), e.toString());
        }

        final boolean sameContent = Arrays.equals(page1.getContent().array(), page2.getContent().array());

        if (page1.getVersion() != page2.getVersion())
          throw new DatabaseAreNotIdentical("Page %s has different versions on databases. DB1 %d <> DB2 %d (sameContent=%s)", pageId, page1.getVersion(),
              page2.getVersion(), sameContent);

        if (!sameContent)
          throw new DatabaseAreNotIdentical("Page %s has different content on databases", pageId);

        db2.getPageManager().removePageFromCache(page2.getPageId());
      }
    }
  }

  public void compareTypes(final Database db1, final Database db2) {
    final Collection types1 = db1.getSchema().getTypes();
    final Collection types2 = db2.getSchema().getTypes();
    if (types1.size() != types2.size())
      throw new DatabaseAreNotIdentical("Types: DB1 %d <> DB2 %d", types1.size(), types2.size());

    final HashMap types1Map = new HashMap<>(types1.size());
    final HashMap types2Map = new HashMap<>(types2.size());

    for (final DocumentType entry : types1)
      types1Map.put(entry.getName(), entry);

    for (final DocumentType entry : types2) {
      if (!types1Map.containsKey(entry.getName()))
        throw new DatabaseAreNotIdentical("Types '%s' is not present in DB2", entry.getName());
      types2Map.put(entry.getName(), entry);
    }

    for (final DocumentType entry : types1)
      if (!types2Map.containsKey(entry.getName()))
        throw new DatabaseAreNotIdentical("Types '%s' is not present in DB1", entry.getName());

    // AT THIS POINT BOTH DBS HAVE THE SAME TYPE NAMES, CHECKING TYPE DETAILS
    for (final DocumentType entry1 : types1) {
      final DocumentType entry2 = types2Map.get(entry1.getName());
      if (!entry1.isTheSameAs(entry2))
        throw new DatabaseAreNotIdentical("Types '%s' is configured differently in two databases 1:\n%s\n2:\n%s", entry2.getName(), entry1.toJSON(),
            entry2.toJSON());
    }
  }

  public void compareIndexes(final Database db1, final Database db2) {
    final Index[] indexes1 = db1.getSchema().getIndexes();
    final Index[] indexes2 = db2.getSchema().getIndexes();
    if (indexes1.length != indexes2.length)
      throw new DatabaseAreNotIdentical("Indexes: DB1 %d <> DB2 %d", indexes1.length, indexes2.length);

    final HashMap indexes1Map = new HashMap<>(indexes1.length);
    final HashMap indexes2Map = new HashMap<>(indexes2.length);

    for (final Index entry : indexes1)
      indexes1Map.put(entry.getName(), entry);

    for (final Index entry : indexes2) {
      if (!indexes1Map.containsKey(entry.getName()))
        throw new DatabaseAreNotIdentical("Index '%s' is not present in DB2", entry.getName());
      indexes2Map.put(entry.getName(), entry);
    }

    for (final Index entry : indexes1)
      if (!indexes2Map.containsKey(entry.getName()))
        throw new DatabaseAreNotIdentical("Index '%s' is not present in DB1", entry.getName());

    // AT THIS POINT BOTH DBS HAVE THE SAME INDEX NAMES, CHECKING INDEXED ENTRIES
    for (final Index entry1 : indexes1) {
      final Index entry2 = indexes2Map.get(entry1.getName());

      final long count1 = entry1.countEntries();
      final long count2 = entry2.countEntries();
      if (count1 != count2)
        throw new DatabaseAreNotIdentical("Index '%s' contains %d entries in DB1 <> %d of DB2", entry1.getName(), count1, count2);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy