HBASE-3929 Add option to HFile tool to produce basic stats

git-svn-id: https://svn.apache.org/repos/asf/hbase/trunk@1187494 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Todd Lipcon 2011-10-21 18:45:31 +00:00
parent a914da38db
commit 53c69f2e9f
2 changed files with 100 additions and 5 deletions

View File

@ -649,6 +649,8 @@ Release 0.92.0 - Unreleased
HBASE-4486 Improve Javadoc for HTableDescriptor (Akash Ashok) HBASE-4486 Improve Javadoc for HTableDescriptor (Akash Ashok)
HBASE-4604 hbase.client.TestHTablePool could start a single HBASE-4604 hbase.client.TestHTablePool could start a single
cluster instead of one per method (nkeywal) cluster instead of one per method (nkeywal)
HBASE-3929 Add option to HFile tool to produce basic stats (Matteo
Bertozzi and todd via todd)
TASKS TASKS
@ -4276,4 +4278,4 @@ Below are the list of changes before 2007-08-18
(Edward Yoon via Stack). (Edward Yoon via Stack).
61. HADOOP-1709 Make HRegionInterface more like that of HTable 61. HADOOP-1709 Make HRegionInterface more like that of HTable
HADOOP-1725 Client find of table regions should not include offlined, split parents HADOOP-1725 Client find of table regions should not include offlined, split parents
= =

View File

@ -64,6 +64,7 @@ public class HFilePrettyPrinter {
private boolean printKey; private boolean printKey;
private boolean shouldPrintMeta; private boolean shouldPrintMeta;
private boolean printBlocks; private boolean printBlocks;
private boolean printStats;
private boolean checkRow; private boolean checkRow;
private boolean checkFamily; private boolean checkFamily;
@ -88,6 +89,7 @@ public class HFilePrettyPrinter {
"File to scan. Pass full-path; e.g. hdfs://a:9000/hbase/.META./12/34"); "File to scan. Pass full-path; e.g. hdfs://a:9000/hbase/.META./12/34");
options.addOption("r", "region", true, options.addOption("r", "region", true,
"Region to scan. Pass region name; e.g. '.META.,,1'"); "Region to scan. Pass region name; e.g. '.META.,,1'");
options.addOption("s", "stats", false, "Print statistics");
} }
public boolean parseOptions(String args[]) throws ParseException, public boolean parseOptions(String args[]) throws ParseException,
@ -105,6 +107,7 @@ public class HFilePrettyPrinter {
printKey = cmd.hasOption("e") || printValue; printKey = cmd.hasOption("e") || printValue;
shouldPrintMeta = cmd.hasOption("m"); shouldPrintMeta = cmd.hasOption("m");
printBlocks = cmd.hasOption("b"); printBlocks = cmd.hasOption("b");
printStats = cmd.hasOption("s");
checkRow = cmd.hasOption("k"); checkRow = cmd.hasOption("k");
checkFamily = cmd.hasOption("a"); checkFamily = cmd.hasOption("a");
@ -189,12 +192,14 @@ public class HFilePrettyPrinter {
Map<byte[], byte[]> fileInfo = reader.loadFileInfo(); Map<byte[], byte[]> fileInfo = reader.loadFileInfo();
if (verbose || printKey || checkRow || checkFamily) { KeyValueStatsCollector fileStats = null;
if (verbose || printKey || checkRow || checkFamily || printStats) {
// scan over file and read key/value's and check if requested // scan over file and read key/value's and check if requested
HFileScanner scanner = reader.getScanner(false, false, false); HFileScanner scanner = reader.getScanner(false, false, false);
scanner.seekTo(); fileStats = new KeyValueStatsCollector();
scanKeysValues(file, scanner); if (scanner.seekTo())
scanKeysValues(file, fileStats, scanner);
} }
// print meta data // print meta data
@ -207,14 +212,23 @@ public class HFilePrettyPrinter {
System.out.println(reader.getDataBlockIndexReader()); System.out.println(reader.getDataBlockIndexReader());
} }
if (printStats) {
fileStats.finish();
System.out.println("Stats:\n" + fileStats);
}
reader.close(); reader.close();
} }
private void scanKeysValues(Path file, HFileScanner scanner) private void scanKeysValues(Path file, KeyValueStatsCollector fileStats, HFileScanner scanner)
throws IOException { throws IOException {
KeyValue pkv = null; KeyValue pkv = null;
do { do {
KeyValue kv = scanner.getKeyValue(); KeyValue kv = scanner.getKeyValue();
// collect stats
if (printStats) {
fileStats.collect(kv);
}
// dump key value // dump key value
if (printKey) { if (printKey) {
System.out.print("K: " + kv); System.out.print("K: " + kv);
@ -305,4 +319,83 @@ public class HFilePrettyPrinter {
} }
} }
private static class LongStats {
private long min = Long.MAX_VALUE;
private long max = Long.MIN_VALUE;
private long sum = 0;
private long count = 0;
void collect(long d) {
if (d < min) min = d;
if (d > max) max = d;
sum += d;
count++;
}
public String toString() {
return "count: " + count +
"\tmin: " + min +
"\tmax: " + max +
"\tmean: " + ((double)sum/count);
}
}
private static class KeyValueStatsCollector {
LongStats keyLen = new LongStats();
LongStats valLen = new LongStats();
LongStats rowSizeBytes = new LongStats();
LongStats rowSizeCols = new LongStats();
long curRowBytes = 0;
long curRowCols = 0;
byte[] biggestRow = null;
private KeyValue prevKV = null;
private long maxRowBytes = 0;
public void collect(KeyValue kv) {
keyLen.collect(kv.getKeyLength());
valLen.collect(kv.getValueLength());
if (prevKV != null &&
KeyValue.COMPARATOR.compareRows(prevKV, kv) != 0) {
// new row
collectRow();
}
curRowBytes += kv.getLength();
curRowCols++;
prevKV = kv;
}
private void collectRow() {
rowSizeBytes.collect(curRowBytes);
rowSizeCols.collect(curRowCols);
if (curRowBytes > maxRowBytes && prevKV != null) {
biggestRow = prevKV.getRow();
}
curRowBytes = 0;
curRowCols = 0;
}
public void finish() {
if (curRowCols > 0) {
collectRow();
}
}
@Override
public String toString() {
if (prevKV == null)
return "no data available for statistics";
return
"Key length: " + keyLen + "\n" +
"Val length: " + valLen + "\n" +
"Row size (bytes): " + rowSizeBytes + "\n" +
"Row size (columns): " + rowSizeCols + "\n" +
"Key of biggest row: " + Bytes.toStringBinary(biggestRow);
}
}
} }