/* HeliDB -- A simple database for Java, http://www.helidb.org
* Copyright (C) 2008, 2009 Karl Gustafsson
*
* This file is a part of HeliDB.
*
* HeliDB is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* HeliDB 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/>.
*/
package org.helidb.impl.txn.log;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Level;
import org.entityfs.Directory;
import org.entityfs.EFile;
import org.entityfs.support.log.LogAdapterHolder;
import org.entityfs.support.log.StdOutLogAdapter;
import org.entityfs.util.Directories;
import org.helidb.CRSSortedBackendDBTester;
import org.helidb.Database;
import org.helidb.backend.DatabaseBackend;
import org.helidb.backend.heapcrs.ConstantRecordSizeHeapBackend;
import org.helidb.backend.index.bplus.BPlusTreeIndexBackend;
import org.helidb.lang.hasher.CharacterToCharacterHasher;
import org.helidb.lang.hasher.IntegerToIntegerHasher;
import org.helidb.lang.serializer.CharacterNullSerializer;
import org.helidb.lang.serializer.CharacterSerializer;
import org.helidb.lang.serializer.IntegerNullSerializer;
import org.helidb.lang.serializer.LongSerializer;
import org.helidb.txn.TransactionalDatabase;
import org.helidb.util.bplus.BPlusTree;
import org.helidb.util.bplus.FileBackedNodeRepository;
import org.helidb.util.bplus.LruCacheNodeRepository;
import org.helidb.util.bplus.NodeRepository;
import org.helidb.util.bplus.NumberOfRecordsNodeSizeStrategy;
import org.junit.Test;
public class LoggingTxnDbOnConstantRecordSizeHeapBackendWBPlusTreeIndexTest extends AbstractLoggingTxnDbOnConstantRecordSizeBackendTest<Long>
{
private final Map<Database<?, ?>, EFile[]> m_dbFiles = new HashMap<Database<?, ?>, EFile[]>();
private LoggingTransactionalDatabase<Integer, Long, Long> createDbUsingFiles(EFile f, EFile treef, EFile logf)
{
LogAdapterHolder lah = new LogAdapterHolder(new StdOutLogAdapter());
lah.setLevel(Level.SEVERE);
DatabaseBackend<Integer, Long, Long> backend = new ConstantRecordSizeHeapBackend<Integer, Long>(f, false, IntegerNullSerializer.INSTANCE, LongSerializer.INSTANCE, 0, 8192, lah);
NodeRepository<Integer> nr = new LruCacheNodeRepository<Integer, Long>(new FileBackedNodeRepository<Integer, Long>(treef, false, 0, new NumberOfRecordsNodeSizeStrategy(3), true, IntegerNullSerializer.INSTANCE, LongSerializer.INSTANCE, 4, 8192, null, lah), 10);
BPlusTree<Integer, Long> tree = new BPlusTree<Integer, Long>(nr, lah);
backend = new BPlusTreeIndexBackend<Integer, Long, Integer, Long>(backend, false, tree, IntegerToIntegerHasher.INSTANCE, lah);
LoggingTransactionalDatabase<Integer, Long, Long> res = new LoggingTransactionalDatabase<Integer, Long, Long>(backend, logf, IntegerNullSerializer.INSTANCE, LongSerializer.INSTANCE, lah);
m_dbFiles.put(res, new EFile[] { f, treef, logf });
return res;
}
@Override
protected LoggingTransactionalDatabase<Integer, Long, Long> createDatabaseWoTxnInDirectory(Directory dir)
{
EFile f = Directories.newFile(dir, "db");
EFile treef = Directories.newFile(dir, "tree");
EFile logf = Directories.newFile(dir, "log");
return createDbUsingFiles(f, treef, logf);
}
@Override
protected LoggingTransactionalDatabase<Integer, Long, Long> createNewDatabaseUsingSameFiles(TransactionalDatabase<Integer, Long> otherDb)
{
EFile[] files = m_dbFiles.get(otherDb);
return createDbUsingFiles(files[0], files[1], files[2]);
}
@Override
protected LoggingTransactionalDatabase<Integer, Long, Long> createNewDatabaseUsingSameFilesButTruncateRollbackLog(TransactionalDatabase<Integer, Long> otherDb, int howMuchToTruncate)
{
EFile[] files = m_dbFiles.get(otherDb);
return createDbUsingFiles(files[0], files[1], truncate(files[2], howMuchToTruncate));
}
private LoggingTransactionalDatabase<Character, Character, Long> createCharacterDbUsingFiles(EFile f, EFile treef, EFile logf)
{
LogAdapterHolder lah = new LogAdapterHolder(new StdOutLogAdapter());
lah.setLevel(Level.SEVERE);
DatabaseBackend<Character, Character, Long> backend = new ConstantRecordSizeHeapBackend<Character, Character>(f, false, CharacterNullSerializer.INSTANCE, CharacterSerializer.INSTANCE, 0, 8192, lah);
NodeRepository<Character> nr = new LruCacheNodeRepository<Character, Long>(new FileBackedNodeRepository<Character, Long>(treef, false, 0, new NumberOfRecordsNodeSizeStrategy(2), true, CharacterNullSerializer.INSTANCE, LongSerializer.INSTANCE, 4, 8192, null, lah), 10);
BPlusTree<Character, Long> tree = new BPlusTree<Character, Long>(nr, lah);
backend = new BPlusTreeIndexBackend<Character, Character, Character, Long>(backend, false, tree, CharacterToCharacterHasher.INSTANCE, lah);
LoggingTransactionalDatabase<Character, Character, Long> res = new LoggingTransactionalDatabase<Character, Character, Long>(backend, logf, CharacterNullSerializer.INSTANCE, CharacterSerializer.INSTANCE, lah);
m_dbFiles.put(res, new EFile[] { f, treef, logf });
return res;
}
@Override
protected TransactionalDatabase<Character, Character> createCharacterDatabaseWoTxnInDirectory(Directory dir)
{
EFile f = Directories.newFile(dir, "db");
EFile treef = Directories.newFile(dir, "tree");
EFile logf = Directories.newFile(dir, "log");
return createCharacterDbUsingFiles(f, treef, logf);
}
@Test
public void testCursor()
{
Database<Integer, Long> db = createDatabase();
try
{
CRSSortedBackendDBTester.testCursor(db);
}
finally
{
tearDownDatabase(db);
}
}
@Test
public void testCursorIterator()
{
Database<Integer, Long> db = createDatabase();
try
{
CRSSortedBackendDBTester.testCursorIterator(db);
}
finally
{
tearDownDatabase(db);
}
}
@Test
public void testCursorBackwardsIterator()
{
Database<Integer, Long> db = createDatabase();
try
{
CRSSortedBackendDBTester.testCursorBackwardsIterator(db);
}
finally
{
tearDownDatabase(db);
}
}
}