Skip to content

Commit

Permalink
add onDiskSize to core status API (#223)
Browse files Browse the repository at this point in the history
* add onDiskSize to core status API

* tidy

* add back indexDiskSize

* fix typo in Sizes RamUsageEstimator

* add some needed conditionals to onDiskSize calcs

* tidy

* add backingDirectory to SizeAccountingIndexInput

* tidy
  • Loading branch information
nginthfs authored Oct 1, 2024
1 parent 8770e47 commit b8f0b76
Show file tree
Hide file tree
Showing 7 changed files with 591 additions and 50 deletions.
47 changes: 47 additions & 0 deletions solr/core/src/java/org/apache/solr/core/DirectoryFactory.java
Original file line number Diff line number Diff line change
Expand Up @@ -166,6 +166,18 @@ public long size(Directory directory) throws IOException {
return sizeOfDirectory(directory);
}

/**
* @param directory to calculate size of
* @return size in bytes on disk, regardless of compression, etc
* @throws IOException on low level IO error
*/
public long onDiskSize(Directory directory) throws IOException {
if (directory instanceof OnDiskSizeDirectory) {
return onDiskSizeOfDirectory(directory);
}
return sizeOfDirectory(directory);
}

/**
* @param path to calculate size of
* @return size in bytes
Expand Down Expand Up @@ -267,6 +279,12 @@ public boolean isAbsolute(String path) {

public interface SizeAware {
long size() throws IOException;

long onDiskSize() throws IOException;
}

public interface OnDiskSizeDirectory {
long onDiskFileLength(String name) throws IOException;
}

public static long sizeOfDirectory(Directory directory) throws IOException {
Expand Down Expand Up @@ -295,6 +313,35 @@ public static long sizeOf(Directory directory, String file) throws IOException {
}
}

public static long onDiskSizeOfDirectory(Directory directory) throws IOException {
if (directory instanceof SizeAware) {
return ((SizeAware) directory).onDiskSize();
}
final String[] files = directory.listAll();
long size = 0;

for (final String file : files) {
size += onDiskSizeOf(directory, file);
if (size < 0) {
break;
}
}

return size;
}

public static long onDiskSizeOf(Directory directory, String file) throws IOException {
if (directory instanceof DirectoryFactory.OnDiskSizeDirectory) {
try {
return ((OnDiskSizeDirectory) directory).onDiskFileLength(file);
} catch (IOException e) {
// could be a race, file no longer exists, access denied, is a directory, etc.
return 0;
}
}
return directory.fileLength(file);
}

/** Delete the files in the Directory */
public static boolean empty(Directory dir) {
boolean isSuccess = true;
Expand Down
38 changes: 38 additions & 0 deletions solr/core/src/java/org/apache/solr/core/SolrCore.java
Original file line number Diff line number Diff line change
Expand Up @@ -533,11 +533,49 @@ public long getIndexSize() {
}
}

public long getOnDiskSize() {
SolrRequestInfo requestInfo = SolrRequestInfo.getRequestInfo();
if (requestInfo != null) {
return (Long)
requestInfo
.getReq()
.getContext()
.computeIfAbsent(cachedOnDiskIndexSizeKeyName(), key -> calculateOnDiskSize());
} else {
return calculateOnDiskSize();
}
}

private long calculateOnDiskSize() {
Directory dir;
long size = 0;
try {
if (directoryFactory.exists(getIndexDir())) {
dir =
directoryFactory.get(
getIndexDir(), DirContext.DEFAULT, solrConfig.indexConfig.lockType);
try {
size = directoryFactory.onDiskSize(dir);
} finally {
directoryFactory.release(dir);
}
}
} catch (IOException e) {
log.error("IO error while trying to get the size of the Directory", e);
}
return size;
}

private String cachedIndexSizeKeyName() {
// avoid collision when we put index sizes for multiple cores in the same metrics request
return "indexSize_" + getName();
}

private String cachedOnDiskIndexSizeKeyName() {
// avoid collision when we put index sizes for multiple cores in the same metrics request
return "onDiskIndexSize_" + getName();
}

public int getSegmentCount() {
try {
return withSearcher(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -380,8 +380,14 @@ public static NamedList<Object> getCoreStatus(
SimpleOrderedMap<Object> indexInfo =
LukeRequestHandler.getIndexInfo(searcher.get().getIndexReader());
long size = core.getIndexSize();
long onDiskSize = core.getOnDiskSize();
String readableSize = NumberUtils.readableSize(size);
String readableOnDiskSize =
size == onDiskSize ? readableSize : NumberUtils.readableSize(onDiskSize);
indexInfo.add("sizeInBytes", size);
indexInfo.add("size", NumberUtils.readableSize(size));
indexInfo.add("onDiskSizeInBytes", onDiskSize);
indexInfo.add("onDiskSize", readableOnDiskSize);
info.add("index", indexInfo);
} finally {
searcher.decref();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,8 +38,10 @@
import org.apache.lucene.store.IndexOutput;
import org.apache.lucene.store.OutputStreamDataOutput;
import org.apache.lucene.util.compress.LZ4;
import org.apache.solr.core.DirectoryFactory;

public class CompressingDirectory extends FSDirectory {
public class CompressingDirectory extends FSDirectory
implements DirectoryFactory.OnDiskSizeDirectory {

/**
* Reference to {@code com.sun.nio.file.ExtendedOpenOption.DIRECT} by reflective class and enum
Expand Down Expand Up @@ -148,6 +150,11 @@ public long fileLength(String name) throws IOException {
return readLengthFromHeader(path);
}

@Override
public long onDiskFileLength(String name) throws IOException {
return Files.size(directoryPath.resolve(name));
}

public static long readLengthFromHeader(Path path) throws IOException {
if (Files.size(path) < Long.BYTES) {
return 0;
Expand Down Expand Up @@ -224,7 +231,11 @@ public int transferTo(DataOutput out) throws IOException {
}
}

static final class DirectIOIndexOutput extends IndexOutput {
public interface SizeReportingIndexOutput {
long getBytesWritten();
}

static final class DirectIOIndexOutput extends IndexOutput implements SizeReportingIndexOutput {
private final byte[] compressBuffer = new byte[COMPRESSION_BLOCK_SIZE];
private final LZ4.FastCompressionHashTable ht = new LZ4.FastCompressionHashTable();
private final ByteBuffer preBuffer;
Expand All @@ -245,6 +256,7 @@ static final class DirectIOIndexOutput extends IndexOutput {
static final int HEADER_SIZE = 16; // 16 bytes

private long filePos;
private long bytesWritten;
private boolean isOpen;

private final DirectBufferPool initialBlockBufferPool;
Expand Down Expand Up @@ -277,6 +289,7 @@ public DirectIOIndexOutput(
preBuffer = ByteBuffer.wrap(compressBuffer);
this.initialBlockBufferPool = initialBlockBufferPool;
initialBlock = initialBlockBufferPool.get();
bytesWritten = HEADER_SIZE;

// allocate space for the header
buffer.position(buffer.position() + HEADER_SIZE);
Expand Down Expand Up @@ -321,6 +334,7 @@ private void dump() throws IOException {

LZ4.compressWithDictionary(compressBuffer, 0, 0, COMPRESSION_BLOCK_SIZE, out, ht);
int nextBlockSize = out.resetSize();
bytesWritten += nextBlockSize;
blockDeltas.writeZInt(nextBlockSize - prevBlockSize);
prevBlockSize = nextBlockSize;
filePos += COMPRESSION_BLOCK_SIZE;
Expand All @@ -336,6 +350,7 @@ private void flush() throws IOException {
LZ4.compressWithDictionary(compressBuffer, 0, 0, preBufferRemaining, out, ht);
}
int blockMapFooterSize = blockDeltas.transferTo(out);
bytesWritten += out.resetSize();
if (wroteBlock) {
writeHelper.flush(buffer, true);
initialBlock.putLong(0, filePos);
Expand All @@ -358,6 +373,11 @@ private void flush() throws IOException {
}
}

@Override
public long getBytesWritten() {
return bytesWritten;
}

private final SizeTrackingDataOutput out = new SizeTrackingDataOutput();

private void writeBlock() throws IOException {
Expand Down
Loading

0 comments on commit b8f0b76

Please sign in to comment.