/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.
*/
package org.apache.camel.component.hawtdb;
import java.io.File;
import java.io.IOException;
import org.apache.camel.Service;
import org.fusesource.hawtbuf.Buffer;
import org.fusesource.hawtbuf.codec.BufferCodec;
import org.fusesource.hawtbuf.codec.IntegerCodec;
import org.fusesource.hawtbuf.codec.StringCodec;
import org.fusesource.hawtdb.api.BTreeIndexFactory;
import org.fusesource.hawtdb.api.OptimisticUpdateException;
import org.fusesource.hawtdb.api.SortedIndex;
import org.fusesource.hawtdb.api.Transaction;
import org.fusesource.hawtdb.api.TxPageFile;
import org.fusesource.hawtdb.api.TxPageFileFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Manages access to a shared <a href="http://hawtdb.fusesource.org/">HawtDB</a> file.
* <p/>
* Will by default not sync writes which allows it to be faster.
* You can force syncing by setting the sync option to <tt>true</tt>.
*/
public class HawtDBFile extends TxPageFileFactory implements Service {
private static final Logger LOG = LoggerFactory.getLogger(HawtDBFile.class);
// the root which contains an index with name -> page for the real indexes
private static final BTreeIndexFactory<String, Integer> ROOT_INDEXES_FACTORY = new BTreeIndexFactory<String, Integer>();
// the real indexes where we store persisted data in buffers
private static final BTreeIndexFactory<Buffer, Buffer> INDEX_FACTORY = new BTreeIndexFactory<Buffer, Buffer>();
private TxPageFile pageFile;
static {
ROOT_INDEXES_FACTORY.setKeyCodec(StringCodec.INSTANCE);
ROOT_INDEXES_FACTORY.setValueCodec(IntegerCodec.INSTANCE);
//ROOT_INDEXES_FACTORY.setDeferredEncoding(true);
// TODO: use false due CAMEL-3826 until root cause is fixed in hawtdb
ROOT_INDEXES_FACTORY.setDeferredEncoding(false);
INDEX_FACTORY.setKeyCodec(BufferCodec.INSTANCE);
INDEX_FACTORY.setValueCodec(BufferCodec.INSTANCE);
//INDEX_FACTORY.setDeferredEncoding(true);
// TODO: use false due CAMEL-3826 until root cause is fixed in hawtdb
INDEX_FACTORY.setDeferredEncoding(false);
}
public HawtDBFile() {
setSync(false);
}
public void start() {
if (getFile() == null) {
throw new IllegalArgumentException("A file must be configured");
}
if (LOG.isDebugEnabled()) {
LOG.debug("Starting HawtDB using file: {}", getFile());
}
open();
pageFile = getTxPageFile();
execute(new Work<Boolean>() {
public Boolean execute(Transaction tx) {
if (!tx.allocator().isAllocated(0)) {
// if we just created the file, first allocated page should be 0
ROOT_INDEXES_FACTORY.create(tx);
LOG.info("Aggregation repository data store created using file: " + getFile());
} else {
SortedIndex<String, Integer> indexes = ROOT_INDEXES_FACTORY.open(tx);
LOG.info("Aggregation repository data store loaded using file: " + getFile()
+ " containing " + indexes.size() + " repositories.");
}
return true;
}
@Override
public String toString() {
return "Allocation repository file: " + getFile();
}
}, true);
}
public void stop() {
File file = getFile();
LOG.debug("Stopping HawtDB using file: {}", file);
try {
close();
} catch (IOException e) {
LOG.warn("Error closing HawtDB file " + file + ". This exception will be ignored.", e);
}
pageFile = null;
}
public <T> T execute(Work<T> work) {
return execute(work, true);
}
public <T> T execute(Work<T> work, boolean rollbackOnOptimisticUpdateException) {
LOG.trace("Executing work +++ start +++ {}", work);
Transaction tx = pageFile.tx();
T answer = doExecute(work, tx, pageFile, rollbackOnOptimisticUpdateException);
LOG.trace("Executing work +++ done +++ {}", work);
return answer;
}
public SortedIndex<Buffer, Buffer> getRepositoryIndex(Transaction tx, String name, boolean create) {
SortedIndex<Buffer, Buffer> answer = null;
SortedIndex<String, Integer> indexes = ROOT_INDEXES_FACTORY.open(tx);
Integer location = indexes.get(name);
if (create && location == null) {
// create it..
SortedIndex<Buffer, Buffer> created = INDEX_FACTORY.create(tx);
int page = created.getIndexLocation();
// add it to indexes so we can find it the next time
indexes.put(name, page);
LOG.debug("Created new repository index with name {} at location {}", name, page);
answer = created;
} else if (location != null) {
LOG.trace("Repository index with name {} at location {}", name, location);
answer = INDEX_FACTORY.open(tx, location);
}
LOG.trace("Repository index with name {} -> {}", name, answer);
return answer;
}
private static <T> T doExecute(Work<T> work, Transaction tx, TxPageFile page, boolean handleOptimisticLockingException) {
T answer = null;
boolean done = false;
int attempt = 0;
while (!done) {
try {
// only log at DEBUG level if we are retrying
if (attempt > 0) {
LOG.debug("Attempt {} to execute work {}", attempt, work);
}
attempt++;
// execute and get answer
answer = work.execute(tx);
LOG.trace("TX is read only: {} for executed work: {}", tx.isReadOnly(), work);
// commit work
tx.commit();
// and flush so we ensure data is spooled to disk
page.flush();
// and we are done
done = true;
} catch (OptimisticUpdateException e) {
if (handleOptimisticLockingException) {
// retry as we hit an optimistic update error
LOG.warn("OptimisticUpdateException occurred at attempt " + attempt + " executing work " + work + ". Will do rollback and retry.");
// no harm doing rollback before retry and no wait is needed
tx.rollback();
} else {
// we must rollback and rethrow
tx.rollback();
throw e;
}
} catch (RuntimeException e) {
LOG.warn("Error executing work " + work + ". Will do rollback.", e);
tx.rollback();
throw e;
}
}
return answer;
}
}