clean and refactor the way fs index gateway work, should work nicer with NFS

This commit is contained in:
kimchy 2010-03-31 17:23:16 +03:00
parent 0586bcd003
commit 38d8fad8d0
12 changed files with 212 additions and 74 deletions

View File

@ -19,6 +19,7 @@
package org.elasticsearch.index.gateway; package org.elasticsearch.index.gateway;
import org.elasticsearch.ElasticSearchIllegalStateException;
import org.elasticsearch.index.deletionpolicy.SnapshotIndexCommit; import org.elasticsearch.index.deletionpolicy.SnapshotIndexCommit;
import org.elasticsearch.index.shard.IndexShardComponent; import org.elasticsearch.index.shard.IndexShardComponent;
import org.elasticsearch.index.translog.Translog; import org.elasticsearch.index.translog.Translog;
@ -36,7 +37,7 @@ public interface IndexShardGateway extends IndexShardComponent {
/** /**
* Snapshots the given shard into the gateway. * Snapshots the given shard into the gateway.
*/ */
void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot); void snapshot(Snapshot snapshot);
/** /**
* Returns <tt>true</tt> if this gateway requires scheduling management for snapshot * Returns <tt>true</tt> if this gateway requires scheduling management for snapshot
@ -45,4 +46,68 @@ public interface IndexShardGateway extends IndexShardComponent {
boolean requiresSnapshotScheduling(); boolean requiresSnapshotScheduling();
void close(); void close();
public static class Snapshot {
private final SnapshotIndexCommit indexCommit;
private final Translog.Snapshot translogSnapshot;
private final long lastIndexVersion;
private final long lastTranslogId;
private final int lastTranslogSize;
public Snapshot(SnapshotIndexCommit indexCommit, Translog.Snapshot translogSnapshot, long lastIndexVersion, long lastTranslogId, int lastTranslogSize) {
this.indexCommit = indexCommit;
this.translogSnapshot = translogSnapshot;
this.lastIndexVersion = lastIndexVersion;
this.lastTranslogId = lastTranslogId;
this.lastTranslogSize = lastTranslogSize;
}
/**
* Indicates that the index has changed from the latest snapshot.
*/
public boolean indexChanged() {
return lastIndexVersion != indexCommit.getVersion();
}
/**
* Indicates that a new transaction log has been created. Note check this <b>before</b> you
* check {@link #sameTranslogNewOperations()}.
*/
public boolean newTranslogCreated() {
return translogSnapshot.translogId() != lastTranslogId;
}
/**
* Indicates that the same translog exists, but new operations have been appended to it. Throws
* {@link ElasticSearchIllegalStateException} if {@link #newTranslogCreated()} is <tt>true</tt>, so
* always check that first.
*/
public boolean sameTranslogNewOperations() {
if (newTranslogCreated()) {
throw new ElasticSearchIllegalStateException("Should not be called when there is a new translog");
}
return translogSnapshot.size() > lastTranslogSize;
}
public SnapshotIndexCommit indexCommit() {
return indexCommit;
}
public Translog.Snapshot translogSnapshot() {
return translogSnapshot;
}
public long lastIndexVersion() {
return lastIndexVersion;
}
public long lastTranslogId() {
return lastTranslogId;
}
public int lastTranslogSize() {
return lastTranslogSize;
}
}
} }

View File

@ -110,14 +110,9 @@ public class IndexShardGatewayService extends AbstractIndexShardComponent implem
StopWatch stopWatch = new StopWatch().start(); StopWatch stopWatch = new StopWatch().start();
RecoveryStatus recoveryStatus = shardGateway.recover(); RecoveryStatus recoveryStatus = shardGateway.recover();
// update the last up to date values lastIndexVersion = recoveryStatus.index().version();
indexShard.snapshot(new Engine.SnapshotHandler() { lastTranslogId = recoveryStatus.translog().translogId();
@Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) throws EngineException { lastTranslogSize = recoveryStatus.translog().numberOfOperations();
lastIndexVersion = snapshotIndexCommit.getVersion();
lastTranslogId = translogSnapshot.translogId();
lastTranslogSize = translogSnapshot.size();
}
});
// start the shard if the gateway has not started it already // start the shard if the gateway has not started it already
if (indexShard.state() != IndexShardState.STARTED) { if (indexShard.state() != IndexShardState.STARTED) {
@ -128,7 +123,7 @@ public class IndexShardGatewayService extends AbstractIndexShardComponent implem
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
sb.append("Recovery completed from ").append(shardGateway).append(", took [").append(stopWatch.totalTime()).append("]\n"); sb.append("Recovery completed from ").append(shardGateway).append(", took [").append(stopWatch.totalTime()).append("]\n");
sb.append(" Index : numberOfFiles [").append(recoveryStatus.index().numberOfFiles()).append("] with totalSize [").append(recoveryStatus.index().totalSize()).append("]\n"); sb.append(" Index : numberOfFiles [").append(recoveryStatus.index().numberOfFiles()).append("] with totalSize [").append(recoveryStatus.index().totalSize()).append("]\n");
sb.append(" Translog : numberOfOperations [").append(recoveryStatus.translog().numberOfOperations()).append("] with totalSize [").append(recoveryStatus.translog().totalSize()).append("]"); sb.append(" Translog : translogId [").append(recoveryStatus.translog().translogId()).append(", numberOfOperations [").append(recoveryStatus.translog().numberOfOperations()).append("] with totalSize [").append(recoveryStatus.translog().totalSize()).append("]");
logger.debug(sb.toString()); logger.debug(sb.toString());
} }
// refresh the shard // refresh the shard
@ -156,7 +151,7 @@ public class IndexShardGatewayService extends AbstractIndexShardComponent implem
@Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) throws EngineException { @Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) throws EngineException {
if (lastIndexVersion != snapshotIndexCommit.getVersion() || lastTranslogId != translogSnapshot.translogId() || lastTranslogSize != translogSnapshot.size()) { if (lastIndexVersion != snapshotIndexCommit.getVersion() || lastTranslogId != translogSnapshot.translogId() || lastTranslogSize != translogSnapshot.size()) {
shardGateway.snapshot(snapshotIndexCommit, translogSnapshot); shardGateway.snapshot(new IndexShardGateway.Snapshot(snapshotIndexCommit, translogSnapshot, lastIndexVersion, lastTranslogId, lastTranslogSize));
lastIndexVersion = snapshotIndexCommit.getVersion(); lastIndexVersion = snapshotIndexCommit.getVersion();
lastTranslogId = translogSnapshot.translogId(); lastTranslogId = translogSnapshot.translogId();

View File

@ -44,14 +44,23 @@ public class RecoveryStatus {
} }
public static class Translog { public static class Translog {
private long translogId;
private int numberOfOperations; private int numberOfOperations;
private SizeValue totalSize; private SizeValue totalSize;
public Translog(int numberOfOperations, SizeValue totalSize) { public Translog(long translogId, int numberOfOperations, SizeValue totalSize) {
this.translogId = translogId;
this.numberOfOperations = numberOfOperations; this.numberOfOperations = numberOfOperations;
this.totalSize = totalSize; this.totalSize = totalSize;
} }
/**
* The translog id recovered, <tt>-1</tt> indicating no translog.
*/
public long translogId() {
return translogId;
}
public int numberOfOperations() { public int numberOfOperations() {
return numberOfOperations; return numberOfOperations;
} }
@ -62,14 +71,19 @@ public class RecoveryStatus {
} }
public static class Index { public static class Index {
private long version;
private int numberOfFiles; private int numberOfFiles;
private SizeValue totalSize; private SizeValue totalSize;
public Index(int numberOfFiles, SizeValue totalSize) { public Index(long version, int numberOfFiles, SizeValue totalSize) {
this.numberOfFiles = numberOfFiles; this.numberOfFiles = numberOfFiles;
this.totalSize = totalSize; this.totalSize = totalSize;
} }
public long version() {
return this.version;
}
public int numberOfFiles() { public int numberOfFiles() {
return numberOfFiles; return numberOfFiles;
} }

View File

@ -20,11 +20,9 @@
package org.elasticsearch.index.gateway.fs; package org.elasticsearch.index.gateway.fs;
import com.google.inject.Inject; import com.google.inject.Inject;
import org.apache.lucene.index.IndexReader;
import org.apache.lucene.store.IndexInput; import org.apache.lucene.store.IndexInput;
import org.elasticsearch.ElasticSearchIllegalStateException;
import org.elasticsearch.index.deletionpolicy.SnapshotIndexCommit; import org.elasticsearch.index.deletionpolicy.SnapshotIndexCommit;
import org.elasticsearch.index.engine.Engine;
import org.elasticsearch.index.engine.EngineException;
import org.elasticsearch.index.gateway.IndexShardGateway; import org.elasticsearch.index.gateway.IndexShardGateway;
import org.elasticsearch.index.gateway.IndexShardGatewayRecoveryException; import org.elasticsearch.index.gateway.IndexShardGatewayRecoveryException;
import org.elasticsearch.index.gateway.IndexShardGatewaySnapshotFailedException; import org.elasticsearch.index.gateway.IndexShardGatewaySnapshotFailedException;
@ -74,12 +72,6 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
private final File locationTranslog; private final File locationTranslog;
private volatile long lastIndexVersion;
private volatile long lastTranslogId = -1;
private volatile int lastTranslogSize;
@Inject public FsIndexShardGateway(ShardId shardId, @IndexSettings Settings indexSettings, ThreadPool threadPool, FsIndexGateway fsIndexGateway, IndexShard indexShard, Store store) { @Inject public FsIndexShardGateway(ShardId shardId, @IndexSettings Settings indexSettings, ThreadPool threadPool, FsIndexGateway fsIndexGateway, IndexShard indexShard, Store store) {
super(shardId, indexSettings); super(shardId, indexSettings);
this.threadPool = threadPool; this.threadPool = threadPool;
@ -107,27 +99,24 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
@Override public RecoveryStatus recover() throws IndexShardGatewayRecoveryException { @Override public RecoveryStatus recover() throws IndexShardGatewayRecoveryException {
RecoveryStatus.Index recoveryStatusIndex = recoverIndex(); RecoveryStatus.Index recoveryStatusIndex = recoverIndex();
RecoveryStatus.Translog recoveryStatusTranslog = recoverTranslog(); RecoveryStatus.Translog recoveryStatusTranslog = recoverTranslog();
// update the last up to date values
indexShard.snapshot(new Engine.SnapshotHandler() {
@Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) throws EngineException {
lastIndexVersion = snapshotIndexCommit.getVersion();
lastTranslogId = translogSnapshot.translogId();
lastTranslogSize = translogSnapshot.size();
}
});
return new RecoveryStatus(recoveryStatusIndex, recoveryStatusTranslog); return new RecoveryStatus(recoveryStatusIndex, recoveryStatusTranslog);
} }
@Override public void snapshot(final SnapshotIndexCommit snapshotIndexCommit, final Translog.Snapshot translogSnapshot) { @Override public void snapshot(Snapshot snapshot) {
boolean indexDirty = false; boolean indexDirty = false;
boolean translogDirty = false; boolean translogDirty = false;
if (lastIndexVersion != snapshotIndexCommit.getVersion()) { final SnapshotIndexCommit snapshotIndexCommit = snapshot.indexCommit();
final Translog.Snapshot translogSnapshot = snapshot.translogSnapshot();
if (snapshot.indexChanged()) {
indexDirty = true; indexDirty = true;
// snapshot into the index // snapshot into the index
final CountDownLatch latch = new CountDownLatch(snapshotIndexCommit.getFiles().length); final CountDownLatch latch = new CountDownLatch(snapshotIndexCommit.getFiles().length);
final AtomicReference<Exception> lastException = new AtomicReference<Exception>(); final AtomicReference<Exception> lastException = new AtomicReference<Exception>();
for (final String fileName : snapshotIndexCommit.getFiles()) { for (final String fileName : snapshotIndexCommit.getFiles()) {
// don't copy over the segments file, it will be copied over later on as part of the
// final snapshot phase
if (fileName.equals(snapshotIndexCommit.getSegmentsFileName())) { if (fileName.equals(snapshotIndexCommit.getSegmentsFileName())) {
latch.countDown(); latch.countDown();
continue; continue;
@ -170,21 +159,14 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
File translogFile = new File(locationTranslog, "translog-" + translogSnapshot.translogId()); File translogFile = new File(locationTranslog, "translog-" + translogSnapshot.translogId());
RandomAccessFile translogRaf = null; RandomAccessFile translogRaf = null;
// if we have a different trnaslogId (or the file does not exists at all), we want to flush the full // if we have a different trnaslogId we want to flush the full translog to a new file (based on the translogId).
// translog to a new file (based on the translogId). If we still work on existing translog, just // If we still work on existing translog, just append the latest translog operations
// append the latest translog operations if (snapshot.newTranslogCreated()) {
if (translogSnapshot.translogId() != lastTranslogId || !translogFile.exists()) {
translogDirty = true; translogDirty = true;
try { try {
translogRaf = new RandomAccessFile(translogFile, "rw"); translogRaf = new RandomAccessFile(translogFile, "rw");
StreamOutput out = new DataOutputStreamOutput(translogRaf); StreamOutput out = new DataOutputStreamOutput(translogRaf);
out.writeInt(-1); // write the number of operations header with -1 currently out.writeInt(-1); // write the number of operations header with -1 currently
// double check that we managed to read/write correctly
translogRaf.seek(0);
if (translogRaf.readInt() != -1) {
throw new ElasticSearchIllegalStateException("Wrote to snapshot file [" + translogFile + "] but did not read...");
}
for (Translog.Operation operation : translogSnapshot) { for (Translog.Operation operation : translogSnapshot) {
writeTranslogOperation(out, operation); writeTranslogOperation(out, operation);
} }
@ -194,25 +176,27 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
} catch (IOException e1) { } catch (IOException e1) {
// ignore // ignore
} }
throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to snapshot translog", e); throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to snapshot translog into [" + translogFile + "]", e);
} }
} else if (translogSnapshot.size() > lastTranslogSize) { } else if (snapshot.sameTranslogNewOperations()) {
translogDirty = true; translogDirty = true;
try { try {
translogRaf = new RandomAccessFile(translogFile, "rw"); translogRaf = new RandomAccessFile(translogFile, "rw");
// seek to the end, since we append // seek to the end, since we append
translogRaf.seek(translogRaf.length()); translogRaf.seek(translogRaf.length());
StreamOutput out = new DataOutputStreamOutput(translogRaf); StreamOutput out = new DataOutputStreamOutput(translogRaf);
for (Translog.Operation operation : translogSnapshot.skipTo(lastTranslogSize)) { for (Translog.Operation operation : translogSnapshot.skipTo(snapshot.lastTranslogSize())) {
writeTranslogOperation(out, operation); writeTranslogOperation(out, operation);
} }
} catch (Exception e) { } catch (Exception e) {
try { try {
if (translogRaf != null) {
translogRaf.close(); translogRaf.close();
} catch (IOException e1) { }
} catch (Exception e1) {
// ignore // ignore
} }
throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to snapshot translog", e); throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to append snapshot translog into [" + translogFile + "]", e);
} }
} }
@ -222,6 +206,18 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
copyFromDirectory(snapshotIndexCommit.getDirectory(), snapshotIndexCommit.getSegmentsFileName(), copyFromDirectory(snapshotIndexCommit.getDirectory(), snapshotIndexCommit.getSegmentsFileName(),
new File(locationIndex, snapshotIndexCommit.getSegmentsFileName())); new File(locationIndex, snapshotIndexCommit.getSegmentsFileName()));
} }
} catch (Exception e) {
try {
if (translogRaf != null) {
translogRaf.close();
}
} catch (Exception e1) {
// ignore
}
throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to finalize index snapshot into [" + new File(locationIndex, snapshotIndexCommit.getSegmentsFileName()) + "]", e);
}
try {
if (translogDirty) { if (translogDirty) {
translogRaf.seek(0); translogRaf.seek(0);
translogRaf.writeInt(translogSnapshot.size()); translogRaf.writeInt(translogSnapshot.size());
@ -232,16 +228,18 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
} }
} catch (Exception e) { } catch (Exception e) {
try { try {
if (translogRaf != null) {
translogRaf.close(); translogRaf.close();
} catch (IOException e1) { }
} catch (Exception e1) {
// ignore // ignore
} }
throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to finalize snapshot", e); throw new IndexShardGatewaySnapshotFailedException(shardId(), "Failed to finalize snapshot into [" + translogFile + "]", e);
} }
// delete the old translog // delete the old translog
if (lastTranslogId != translogSnapshot.translogId()) { if (snapshot.newTranslogCreated()) {
new File(locationTranslog, "translog-" + lastTranslogId).delete(); new File(locationTranslog, "translog-" + snapshot.lastTranslogId()).delete();
} }
// delete files that no longer exists in the index // delete files that no longer exists in the index
@ -260,11 +258,6 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
} }
} }
} }
lastIndexVersion = snapshotIndexCommit.getVersion();
lastTranslogId = translogSnapshot.translogId();
lastTranslogSize = translogSnapshot.size();
} }
private RecoveryStatus.Index recoverIndex() throws IndexShardGatewayRecoveryException { private RecoveryStatus.Index recoverIndex() throws IndexShardGatewayRecoveryException {
@ -297,7 +290,17 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
for (File file : files) { for (File file : files) {
totalSize += file.length(); totalSize += file.length();
} }
return new RecoveryStatus.Index(files.length, new SizeValue(totalSize, SizeUnit.BYTES));
long version = -1;
try {
if (IndexReader.indexExists(store.directory())) {
version = IndexReader.getCurrentVersion(store.directory());
}
} catch (IOException e) {
throw new IndexShardGatewayRecoveryException(shardId(), "Failed to fetch index version after copying it over", e);
}
return new RecoveryStatus.Index(version, files.length, new SizeValue(totalSize, SizeUnit.BYTES));
} }
private RecoveryStatus.Translog recoverTranslog() throws IndexShardGatewayRecoveryException { private RecoveryStatus.Translog recoverTranslog() throws IndexShardGatewayRecoveryException {
@ -307,7 +310,7 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
if (recoveryTranslogId == -1) { if (recoveryTranslogId == -1) {
// no recovery file found, start the shard and bail // no recovery file found, start the shard and bail
indexShard.start(); indexShard.start();
return new RecoveryStatus.Translog(0, new SizeValue(0, SizeUnit.BYTES)); return new RecoveryStatus.Translog(-1, 0, new SizeValue(0, SizeUnit.BYTES));
} }
File recoveryTranslogFile = new File(locationTranslog, "translog-" + recoveryTranslogId); File recoveryTranslogFile = new File(locationTranslog, "translog-" + recoveryTranslogId);
raf = new RandomAccessFile(recoveryTranslogFile, "r"); raf = new RandomAccessFile(recoveryTranslogFile, "r");
@ -317,7 +320,7 @@ public class FsIndexShardGateway extends AbstractIndexShardComponent implements
operations.add(readTranslogOperation(new DataInputStreamInput(raf))); operations.add(readTranslogOperation(new DataInputStreamInput(raf)));
} }
indexShard.performRecovery(operations); indexShard.performRecovery(operations);
return new RecoveryStatus.Translog(operations.size(), new SizeValue(recoveryTranslogFile.length(), SizeUnit.BYTES)); return new RecoveryStatus.Translog(recoveryTranslogId, operations.size(), new SizeValue(recoveryTranslogFile.length(), SizeUnit.BYTES));
} catch (Exception e) { } catch (Exception e) {
throw new IndexShardGatewayRecoveryException(shardId(), "Failed to perform recovery of translog", e); throw new IndexShardGatewayRecoveryException(shardId(), "Failed to perform recovery of translog", e);
} finally { } finally {

View File

@ -20,7 +20,6 @@
package org.elasticsearch.index.gateway.none; package org.elasticsearch.index.gateway.none;
import com.google.inject.Inject; import com.google.inject.Inject;
import org.elasticsearch.index.deletionpolicy.SnapshotIndexCommit;
import org.elasticsearch.index.gateway.IndexShardGateway; import org.elasticsearch.index.gateway.IndexShardGateway;
import org.elasticsearch.index.gateway.IndexShardGatewayRecoveryException; import org.elasticsearch.index.gateway.IndexShardGatewayRecoveryException;
import org.elasticsearch.index.gateway.RecoveryStatus; import org.elasticsearch.index.gateway.RecoveryStatus;
@ -29,7 +28,6 @@ import org.elasticsearch.index.shard.AbstractIndexShardComponent;
import org.elasticsearch.index.shard.ShardId; import org.elasticsearch.index.shard.ShardId;
import org.elasticsearch.index.shard.service.IndexShard; import org.elasticsearch.index.shard.service.IndexShard;
import org.elasticsearch.index.shard.service.InternalIndexShard; import org.elasticsearch.index.shard.service.InternalIndexShard;
import org.elasticsearch.index.translog.Translog;
import org.elasticsearch.util.SizeUnit; import org.elasticsearch.util.SizeUnit;
import org.elasticsearch.util.SizeValue; import org.elasticsearch.util.SizeValue;
import org.elasticsearch.util.settings.Settings; import org.elasticsearch.util.settings.Settings;
@ -49,10 +47,10 @@ public class NoneIndexShardGateway extends AbstractIndexShardComponent implement
@Override public RecoveryStatus recover() throws IndexShardGatewayRecoveryException { @Override public RecoveryStatus recover() throws IndexShardGatewayRecoveryException {
// in the none case, we simply start the shard // in the none case, we simply start the shard
indexShard.start(); indexShard.start();
return new RecoveryStatus(new RecoveryStatus.Index(0, new SizeValue(0, SizeUnit.BYTES)), new RecoveryStatus.Translog(0, new SizeValue(0, SizeUnit.BYTES))); return new RecoveryStatus(new RecoveryStatus.Index(-1, 0, new SizeValue(0, SizeUnit.BYTES)), new RecoveryStatus.Translog(-1, 0, new SizeValue(0, SizeUnit.BYTES)));
} }
@Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) { @Override public void snapshot(Snapshot snapshot) {
// nothing to do here // nothing to do here
} }

View File

@ -28,6 +28,7 @@ import org.elasticsearch.index.LocalNodeId;
import org.elasticsearch.index.settings.IndexSettings; import org.elasticsearch.index.settings.IndexSettings;
import org.elasticsearch.index.shard.IndexShardLifecycle; import org.elasticsearch.index.shard.IndexShardLifecycle;
import org.elasticsearch.index.shard.ShardId; import org.elasticsearch.index.shard.ShardId;
import org.elasticsearch.index.store.support.ForceSyncDirectory;
import org.elasticsearch.util.lucene.store.SwitchDirectory; import org.elasticsearch.util.lucene.store.SwitchDirectory;
import org.elasticsearch.util.settings.Settings; import org.elasticsearch.util.settings.Settings;
@ -80,7 +81,7 @@ public class MmapFsStore extends AbstractFsStore<Directory> {
return suggestUseCompoundFile; return suggestUseCompoundFile;
} }
private static class CustomMMapDirectory extends MMapDirectory { private static class CustomMMapDirectory extends MMapDirectory implements ForceSyncDirectory {
private final boolean syncToDisk; private final boolean syncToDisk;
@ -95,5 +96,9 @@ public class MmapFsStore extends AbstractFsStore<Directory> {
} }
super.sync(name); super.sync(name);
} }
@Override public void forceSync(String name) throws IOException {
super.sync(name);
}
} }
} }

View File

@ -27,6 +27,7 @@ import org.elasticsearch.env.Environment;
import org.elasticsearch.index.LocalNodeId; import org.elasticsearch.index.LocalNodeId;
import org.elasticsearch.index.settings.IndexSettings; import org.elasticsearch.index.settings.IndexSettings;
import org.elasticsearch.index.shard.ShardId; import org.elasticsearch.index.shard.ShardId;
import org.elasticsearch.index.store.support.ForceSyncDirectory;
import org.elasticsearch.util.lucene.store.SwitchDirectory; import org.elasticsearch.util.lucene.store.SwitchDirectory;
import org.elasticsearch.util.settings.Settings; import org.elasticsearch.util.settings.Settings;
@ -78,7 +79,7 @@ public class NioFsStore extends AbstractFsStore<Directory> {
return suggestUseCompoundFile; return suggestUseCompoundFile;
} }
private static class CustomNioFSDirectory extends NIOFSDirectory { private static class CustomNioFSDirectory extends NIOFSDirectory implements ForceSyncDirectory {
private final boolean syncToDisk; private final boolean syncToDisk;
@ -93,5 +94,9 @@ public class NioFsStore extends AbstractFsStore<Directory> {
} }
super.sync(name); super.sync(name);
} }
@Override public void forceSync(String name) throws IOException {
super.sync(name);
}
} }
} }

View File

@ -27,6 +27,7 @@ import org.elasticsearch.env.Environment;
import org.elasticsearch.index.LocalNodeId; import org.elasticsearch.index.LocalNodeId;
import org.elasticsearch.index.settings.IndexSettings; import org.elasticsearch.index.settings.IndexSettings;
import org.elasticsearch.index.shard.ShardId; import org.elasticsearch.index.shard.ShardId;
import org.elasticsearch.index.store.support.ForceSyncDirectory;
import org.elasticsearch.util.lucene.store.SwitchDirectory; import org.elasticsearch.util.lucene.store.SwitchDirectory;
import org.elasticsearch.util.settings.Settings; import org.elasticsearch.util.settings.Settings;
@ -78,7 +79,7 @@ public class SimpleFsStore extends AbstractFsStore<Directory> {
return suggestUseCompoundFile; return suggestUseCompoundFile;
} }
private static class CustomSimpleFSDirectory extends SimpleFSDirectory { private static class CustomSimpleFSDirectory extends SimpleFSDirectory implements ForceSyncDirectory {
private final boolean syncToDisk; private final boolean syncToDisk;
@ -93,5 +94,9 @@ public class SimpleFsStore extends AbstractFsStore<Directory> {
} }
super.sync(name); super.sync(name);
} }
@Override public void forceSync(String name) throws IOException {
super.sync(name);
}
} }
} }

View File

@ -0,0 +1,36 @@
/*
* Licensed to Elastic Search and Shay Banon under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. Elastic Search 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.elasticsearch.index.store.support;
import java.io.IOException;
/**
* A custom directory that allows to forceSync (since the actual directory might disable it)
*
* @author kimchy (shay.banon)
*/
public interface ForceSyncDirectory {
/**
* Similar to {@link org.apache.lucene.store.Directory#sync(String)} but forces it even if its
* disabled.
*/
void forceSync(String name) throws IOException;
}

View File

@ -22,6 +22,7 @@ package org.elasticsearch.util.lucene;
import org.apache.lucene.index.IndexCommit; import org.apache.lucene.index.IndexCommit;
import org.apache.lucene.index.IndexReader; import org.apache.lucene.index.IndexReader;
import org.apache.lucene.store.*; import org.apache.lucene.store.*;
import org.elasticsearch.index.store.support.ForceSyncDirectory;
import org.elasticsearch.util.SizeValue; import org.elasticsearch.util.SizeValue;
import org.elasticsearch.util.io.FileSystemUtils; import org.elasticsearch.util.io.FileSystemUtils;
@ -158,8 +159,12 @@ public class Directories {
} else { } else {
copyToDirectory(new FileInputStream(copyFrom), dir.createOutput(fileName)); copyToDirectory(new FileInputStream(copyFrom), dir.createOutput(fileName));
} }
if (dir instanceof ForceSyncDirectory) {
((ForceSyncDirectory) dir).forceSync(fileName);
} else {
dir.sync(fileName); dir.sync(fileName);
} }
}
public static void copyToDirectory(InputStream is, IndexOutput io) throws IOException { public static void copyToDirectory(InputStream is, IndexOutput io) throws IOException {
byte[] buffer = new byte[16384]; byte[] buffer = new byte[16384];

View File

@ -23,6 +23,7 @@ import com.google.common.collect.ImmutableSet;
import org.apache.lucene.store.Directory; import org.apache.lucene.store.Directory;
import org.apache.lucene.store.IndexInput; import org.apache.lucene.store.IndexInput;
import org.apache.lucene.store.IndexOutput; import org.apache.lucene.store.IndexOutput;
import org.elasticsearch.index.store.support.ForceSyncDirectory;
import java.io.IOException; import java.io.IOException;
import java.util.Set; import java.util.Set;
@ -37,7 +38,7 @@ import java.util.Set;
* *
* @author kimchy (shay.banon) * @author kimchy (shay.banon)
*/ */
public class SwitchDirectory extends Directory { public class SwitchDirectory extends Directory implements ForceSyncDirectory {
private final Directory secondaryDir; private final Directory secondaryDir;
@ -141,6 +142,15 @@ public class SwitchDirectory extends Directory {
getDirectory(name).sync(name); getDirectory(name).sync(name);
} }
@Override public void forceSync(String name) throws IOException {
Directory dir = getDirectory(name);
if (dir instanceof ForceSyncDirectory) {
((ForceSyncDirectory) dir).forceSync(name);
} else {
dir.sync(name);
}
}
@Override public IndexInput openInput(String name) throws IOException { @Override public IndexInput openInput(String name) throws IOException {
return getDirectory(name).openInput(name); return getDirectory(name).openInput(name);
} }

View File

@ -48,7 +48,6 @@ public class FsMetaDataGatewayTests extends AbstractServersTests {
} }
@Test public void testIndexActions() throws Exception { @Test public void testIndexActions() throws Exception {
buildServer("server1"); buildServer("server1");
((InternalServer) server("server1")).injector().getInstance(Gateway.class).reset(); ((InternalServer) server("server1")).injector().getInstance(Gateway.class).reset();
server("server1").start(); server("server1").start();
@ -67,7 +66,5 @@ public class FsMetaDataGatewayTests extends AbstractServersTests {
} catch (IndexAlreadyExistsException e) { } catch (IndexAlreadyExistsException e) {
// all is well // all is well
} }
((InternalServer) server("server1")).injector().getInstance(Gateway.class).reset();
} }
} }