HBASE-22743 : ClientUtils for Demo Client classes (#413)
This commit is contained in:
parent
3318b4bc6d
commit
064f5f1394
|
@ -19,12 +19,8 @@
|
|||
package org.apache.hadoop.hbase.rest;
|
||||
|
||||
import java.security.PrivilegedExceptionAction;
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
|
||||
import javax.security.auth.Subject;
|
||||
import javax.security.auth.login.AppConfigurationEntry;
|
||||
import javax.security.auth.login.Configuration;
|
||||
import javax.security.auth.login.LoginContext;
|
||||
|
||||
import org.apache.hadoop.hbase.Cell;
|
||||
|
@ -37,6 +33,7 @@ import org.apache.hadoop.hbase.rest.client.Client;
|
|||
import org.apache.hadoop.hbase.rest.client.Cluster;
|
||||
import org.apache.hadoop.hbase.rest.client.RemoteHTable;
|
||||
import org.apache.hadoop.hbase.util.Bytes;
|
||||
import org.apache.hadoop.hbase.util.ClientUtils;
|
||||
import org.apache.yetus.audience.InterfaceAudience;
|
||||
import org.apache.hbase.thirdparty.com.google.common.base.Preconditions;
|
||||
|
||||
|
@ -112,32 +109,7 @@ public class RESTDemoClient {
|
|||
return new Subject();
|
||||
}
|
||||
|
||||
/*
|
||||
* To authenticate the demo client, kinit should be invoked ahead. Here we try to get the
|
||||
* Kerberos credential from the ticket cache.
|
||||
*/
|
||||
LoginContext context = new LoginContext("", new Subject(), null, new Configuration() {
|
||||
@Override
|
||||
public AppConfigurationEntry[] getAppConfigurationEntry(String name) {
|
||||
Map<String, String> options = new HashMap<>();
|
||||
options.put("useKeyTab", "false");
|
||||
options.put("storeKey", "false");
|
||||
options.put("doNotPrompt", "true");
|
||||
options.put("useTicketCache", "true");
|
||||
options.put("renewTGT", "true");
|
||||
options.put("refreshKrb5Config", "true");
|
||||
options.put("isInitiator", "true");
|
||||
String ticketCache = System.getenv("KRB5CCNAME");
|
||||
if (ticketCache != null) {
|
||||
options.put("ticketCache", ticketCache);
|
||||
}
|
||||
options.put("debug", "true");
|
||||
|
||||
return new AppConfigurationEntry[] {
|
||||
new AppConfigurationEntry("com.sun.security.auth.module.Krb5LoginModule",
|
||||
AppConfigurationEntry.LoginModuleControlFlag.REQUIRED, options) };
|
||||
}
|
||||
});
|
||||
LoginContext context = ClientUtils.getLoginContext();
|
||||
context.login();
|
||||
return context.getSubject();
|
||||
}
|
||||
|
|
|
@ -19,20 +19,13 @@
|
|||
package org.apache.hadoop.hbase.thrift;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.charset.CharacterCodingException;
|
||||
import java.nio.charset.Charset;
|
||||
import java.nio.charset.CharsetDecoder;
|
||||
import java.security.PrivilegedExceptionAction;
|
||||
import java.text.NumberFormat;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.SortedMap;
|
||||
import java.util.TreeMap;
|
||||
import javax.security.auth.Subject;
|
||||
import javax.security.auth.login.AppConfigurationEntry;
|
||||
import javax.security.auth.login.Configuration;
|
||||
import javax.security.auth.login.LoginContext;
|
||||
import javax.security.sasl.Sasl;
|
||||
import org.apache.hadoop.hbase.thrift.generated.AlreadyExists;
|
||||
|
@ -42,6 +35,7 @@ import org.apache.hadoop.hbase.thrift.generated.Mutation;
|
|||
import org.apache.hadoop.hbase.thrift.generated.TCell;
|
||||
import org.apache.hadoop.hbase.thrift.generated.TRowResult;
|
||||
import org.apache.hadoop.hbase.util.Bytes;
|
||||
import org.apache.hadoop.hbase.util.ClientUtils;
|
||||
import org.apache.thrift.protocol.TBinaryProtocol;
|
||||
import org.apache.thrift.protocol.TProtocol;
|
||||
import org.apache.thrift.transport.TSaslClientTransport;
|
||||
|
@ -60,7 +54,6 @@ public class DemoClient {
|
|||
|
||||
static protected int port;
|
||||
static protected String host;
|
||||
CharsetDecoder decoder = null;
|
||||
|
||||
private static boolean secure = false;
|
||||
private static String serverPrincipal = "hbase";
|
||||
|
@ -101,16 +94,6 @@ public class DemoClient {
|
|||
}
|
||||
|
||||
DemoClient() {
|
||||
decoder = Charset.forName("UTF-8").newDecoder();
|
||||
}
|
||||
|
||||
// Helper to translate byte[]'s to UTF8 strings
|
||||
private String utf8(byte[] buf) {
|
||||
try {
|
||||
return decoder.decode(ByteBuffer.wrap(buf)).toString();
|
||||
} catch (CharacterCodingException e) {
|
||||
return "[INVALID UTF-8]";
|
||||
}
|
||||
}
|
||||
|
||||
// Helper to translate strings to UTF8 bytes
|
||||
|
@ -146,15 +129,15 @@ public class DemoClient {
|
|||
System.out.println("scanning tables...");
|
||||
|
||||
for (ByteBuffer name : client.getTableNames()) {
|
||||
System.out.println(" found: " + utf8(name.array()));
|
||||
System.out.println(" found: " + ClientUtils.utf8(name.array()));
|
||||
|
||||
if (utf8(name.array()).equals(utf8(t))) {
|
||||
if (ClientUtils.utf8(name.array()).equals(ClientUtils.utf8(t))) {
|
||||
if (client.isTableEnabled(name)) {
|
||||
System.out.println(" disabling table: " + utf8(name.array()));
|
||||
System.out.println(" disabling table: " + ClientUtils.utf8(name.array()));
|
||||
client.disableTable(name);
|
||||
}
|
||||
|
||||
System.out.println(" deleting table: " + utf8(name.array()));
|
||||
System.out.println(" deleting table: " + ClientUtils.utf8(name.array()));
|
||||
client.deleteTable(name);
|
||||
}
|
||||
}
|
||||
|
@ -172,7 +155,7 @@ public class DemoClient {
|
|||
col.timeToLive = Integer.MAX_VALUE;
|
||||
columns.add(col);
|
||||
|
||||
System.out.println("creating table: " + utf8(t));
|
||||
System.out.println("creating table: " + ClientUtils.utf8(t));
|
||||
|
||||
try {
|
||||
client.createTable(ByteBuffer.wrap(t), columns);
|
||||
|
@ -180,11 +163,12 @@ public class DemoClient {
|
|||
System.out.println("WARN: " + ae.message);
|
||||
}
|
||||
|
||||
System.out.println("column families in " + utf8(t) + ": ");
|
||||
System.out.println("column families in " + ClientUtils.utf8(t) + ": ");
|
||||
Map<ByteBuffer, ColumnDescriptor> columnMap = client.getColumnDescriptors(ByteBuffer.wrap(t));
|
||||
|
||||
for (ColumnDescriptor col2 : columnMap.values()) {
|
||||
System.out.println(" column: " + utf8(col2.name.array()) + ", maxVer: " + col2.maxVersions);
|
||||
System.out.println(" column: " + ClientUtils.utf8(col2.name.array()) + ", maxVer: "
|
||||
+ col2.maxVersions);
|
||||
}
|
||||
|
||||
Map<ByteBuffer, ByteBuffer> dummyAttributes = null;
|
||||
|
@ -358,31 +342,15 @@ public class DemoClient {
|
|||
StringBuilder rowStr = new StringBuilder();
|
||||
|
||||
for (TCell cell : versions) {
|
||||
rowStr.append(utf8(cell.value.array()));
|
||||
rowStr.append(ClientUtils.utf8(cell.value.array()));
|
||||
rowStr.append("; ");
|
||||
}
|
||||
|
||||
System.out.println("row: " + utf8(row.array()) + ", values: " + rowStr);
|
||||
System.out.println("row: " + ClientUtils.utf8(row.array()) + ", values: " + rowStr);
|
||||
}
|
||||
|
||||
private void printRow(TRowResult rowResult) {
|
||||
// copy values into a TreeMap to get them in sorted order
|
||||
TreeMap<String, TCell> sorted = new TreeMap<>();
|
||||
|
||||
for (Map.Entry<ByteBuffer, TCell> column : rowResult.columns.entrySet()) {
|
||||
sorted.put(utf8(column.getKey().array()), column.getValue());
|
||||
}
|
||||
|
||||
StringBuilder rowStr = new StringBuilder();
|
||||
|
||||
for (SortedMap.Entry<String, TCell> entry : sorted.entrySet()) {
|
||||
rowStr.append(entry.getKey());
|
||||
rowStr.append(" => ");
|
||||
rowStr.append(utf8(entry.getValue().value.array()));
|
||||
rowStr.append("; ");
|
||||
}
|
||||
|
||||
System.out.println("row: " + utf8(rowResult.row.array()) + ", cols: " + rowStr);
|
||||
ClientUtils.printRow(rowResult);
|
||||
}
|
||||
|
||||
private void printRow(List<TRowResult> rows) {
|
||||
|
@ -396,37 +364,7 @@ public class DemoClient {
|
|||
return new Subject();
|
||||
}
|
||||
|
||||
/*
|
||||
* To authenticate the DemoClient, kinit should be invoked ahead.
|
||||
* Here we try to get the Kerberos credential from the ticket cache.
|
||||
*/
|
||||
LoginContext context = new LoginContext("", new Subject(), null,
|
||||
new Configuration() {
|
||||
@Override
|
||||
public AppConfigurationEntry[] getAppConfigurationEntry(String name) {
|
||||
Map<String, String> options = new HashMap<>();
|
||||
options.put("useKeyTab", "false");
|
||||
options.put("storeKey", "false");
|
||||
options.put("doNotPrompt", "true");
|
||||
options.put("useTicketCache", "true");
|
||||
options.put("renewTGT", "true");
|
||||
options.put("refreshKrb5Config", "true");
|
||||
options.put("isInitiator", "true");
|
||||
String ticketCache = System.getenv("KRB5CCNAME");
|
||||
|
||||
if (ticketCache != null) {
|
||||
options.put("ticketCache", ticketCache);
|
||||
}
|
||||
|
||||
options.put("debug", "true");
|
||||
|
||||
return new AppConfigurationEntry[]{
|
||||
new AppConfigurationEntry("com.sun.security.auth.module.Krb5LoginModule",
|
||||
AppConfigurationEntry.LoginModuleControlFlag.REQUIRED,
|
||||
options)};
|
||||
}
|
||||
});
|
||||
|
||||
LoginContext context = ClientUtils.getLoginContext();
|
||||
context.login();
|
||||
return context.getSubject();
|
||||
}
|
||||
|
|
|
@ -20,9 +20,6 @@ package org.apache.hadoop.hbase.thrift;
|
|||
|
||||
import java.io.File;
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.charset.CharacterCodingException;
|
||||
import java.nio.charset.Charset;
|
||||
import java.nio.charset.CharsetDecoder;
|
||||
import java.security.Principal;
|
||||
import java.security.PrivilegedExceptionAction;
|
||||
import java.util.ArrayList;
|
||||
|
@ -32,8 +29,6 @@ import java.util.HashSet;
|
|||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
import java.util.SortedMap;
|
||||
import java.util.TreeMap;
|
||||
import javax.security.auth.Subject;
|
||||
import javax.security.auth.kerberos.KerberosPrincipal;
|
||||
import javax.security.auth.login.AppConfigurationEntry;
|
||||
|
@ -45,6 +40,7 @@ import org.apache.hadoop.hbase.thrift.generated.Hbase;
|
|||
import org.apache.hadoop.hbase.thrift.generated.TCell;
|
||||
import org.apache.hadoop.hbase.thrift.generated.TRowResult;
|
||||
import org.apache.hadoop.hbase.util.Bytes;
|
||||
import org.apache.hadoop.hbase.util.ClientUtils;
|
||||
import org.apache.thrift.protocol.TBinaryProtocol;
|
||||
import org.apache.thrift.protocol.TProtocol;
|
||||
import org.apache.thrift.transport.THttpClient;
|
||||
|
@ -69,7 +65,6 @@ public class HttpDoAsClient {
|
|||
|
||||
static protected int port;
|
||||
static protected String host;
|
||||
CharsetDecoder decoder = null;
|
||||
private static boolean secure = false;
|
||||
static protected String doAsUser = null;
|
||||
static protected String principal = null;
|
||||
|
@ -112,16 +107,6 @@ public class HttpDoAsClient {
|
|||
}
|
||||
|
||||
HttpDoAsClient() {
|
||||
decoder = Charset.forName("UTF-8").newDecoder();
|
||||
}
|
||||
|
||||
// Helper to translate byte[]'s to UTF8 strings
|
||||
private String utf8(byte[] buf) {
|
||||
try {
|
||||
return decoder.decode(ByteBuffer.wrap(buf)).toString();
|
||||
} catch (CharacterCodingException e) {
|
||||
return "[INVALID UTF-8]";
|
||||
}
|
||||
}
|
||||
|
||||
// Helper to translate strings to UTF8 bytes
|
||||
|
@ -146,13 +131,13 @@ public class HttpDoAsClient {
|
|||
//
|
||||
System.out.println("scanning tables...");
|
||||
for (ByteBuffer name : refresh(client, httpClient).getTableNames()) {
|
||||
System.out.println(" found: " + utf8(name.array()));
|
||||
if (utf8(name.array()).equals(utf8(t))) {
|
||||
System.out.println(" found: " + ClientUtils.utf8(name.array()));
|
||||
if (ClientUtils.utf8(name.array()).equals(ClientUtils.utf8(t))) {
|
||||
if (refresh(client, httpClient).isTableEnabled(name)) {
|
||||
System.out.println(" disabling table: " + utf8(name.array()));
|
||||
System.out.println(" disabling table: " + ClientUtils.utf8(name.array()));
|
||||
refresh(client, httpClient).disableTable(name);
|
||||
}
|
||||
System.out.println(" deleting table: " + utf8(name.array()));
|
||||
System.out.println(" deleting table: " + ClientUtils.utf8(name.array()));
|
||||
refresh(client, httpClient).deleteTable(name);
|
||||
}
|
||||
}
|
||||
|
@ -172,7 +157,7 @@ public class HttpDoAsClient {
|
|||
col.timeToLive = Integer.MAX_VALUE;
|
||||
columns.add(col);
|
||||
|
||||
System.out.println("creating table: " + utf8(t));
|
||||
System.out.println("creating table: " + ClientUtils.utf8(t));
|
||||
try {
|
||||
|
||||
refresh(client, httpClient).createTable(ByteBuffer.wrap(t), columns);
|
||||
|
@ -180,11 +165,12 @@ public class HttpDoAsClient {
|
|||
System.out.println("WARN: " + ae.message);
|
||||
}
|
||||
|
||||
System.out.println("column families in " + utf8(t) + ": ");
|
||||
System.out.println("column families in " + ClientUtils.utf8(t) + ": ");
|
||||
Map<ByteBuffer, ColumnDescriptor> columnMap = refresh(client, httpClient)
|
||||
.getColumnDescriptors(ByteBuffer.wrap(t));
|
||||
for (ColumnDescriptor col2 : columnMap.values()) {
|
||||
System.out.println(" column: " + utf8(col2.name.array()) + ", maxVer: " + col2.maxVersions);
|
||||
System.out.println(" column: " + ClientUtils.utf8(col2.name.array()) + ", maxVer: "
|
||||
+ col2.maxVersions);
|
||||
}
|
||||
|
||||
transport.close();
|
||||
|
@ -236,27 +222,14 @@ public class HttpDoAsClient {
|
|||
private void printVersions(ByteBuffer row, List<TCell> versions) {
|
||||
StringBuilder rowStr = new StringBuilder();
|
||||
for (TCell cell : versions) {
|
||||
rowStr.append(utf8(cell.value.array()));
|
||||
rowStr.append(ClientUtils.utf8(cell.value.array()));
|
||||
rowStr.append("; ");
|
||||
}
|
||||
System.out.println("row: " + utf8(row.array()) + ", values: " + rowStr);
|
||||
System.out.println("row: " + ClientUtils.utf8(row.array()) + ", values: " + rowStr);
|
||||
}
|
||||
|
||||
private void printRow(TRowResult rowResult) {
|
||||
// copy values into a TreeMap to get them in sorted order
|
||||
TreeMap<String, TCell> sorted = new TreeMap<>();
|
||||
for (Map.Entry<ByteBuffer, TCell> column : rowResult.columns.entrySet()) {
|
||||
sorted.put(utf8(column.getKey().array()), column.getValue());
|
||||
}
|
||||
|
||||
StringBuilder rowStr = new StringBuilder();
|
||||
for (SortedMap.Entry<String, TCell> entry : sorted.entrySet()) {
|
||||
rowStr.append(entry.getKey());
|
||||
rowStr.append(" => ");
|
||||
rowStr.append(utf8(entry.getValue().value.array()));
|
||||
rowStr.append("; ");
|
||||
}
|
||||
System.out.println("row: " + utf8(rowResult.row.array()) + ", cols: " + rowStr);
|
||||
ClientUtils.printRow(rowResult);
|
||||
}
|
||||
|
||||
static Subject getSubject() throws Exception {
|
||||
|
|
|
@ -25,8 +25,6 @@ import java.util.HashMap;
|
|||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import javax.security.auth.Subject;
|
||||
import javax.security.auth.login.AppConfigurationEntry;
|
||||
import javax.security.auth.login.Configuration;
|
||||
import javax.security.auth.login.LoginContext;
|
||||
import javax.security.sasl.Sasl;
|
||||
import org.apache.hadoop.hbase.HBaseConfiguration;
|
||||
|
@ -36,6 +34,7 @@ import org.apache.hadoop.hbase.thrift2.generated.THBaseService;
|
|||
import org.apache.hadoop.hbase.thrift2.generated.TPut;
|
||||
import org.apache.hadoop.hbase.thrift2.generated.TResult;
|
||||
import org.apache.hadoop.hbase.util.Bytes;
|
||||
import org.apache.hadoop.hbase.util.ClientUtils;
|
||||
import org.apache.thrift.protocol.TBinaryProtocol;
|
||||
import org.apache.thrift.protocol.TProtocol;
|
||||
import org.apache.thrift.transport.TFramedTransport;
|
||||
|
@ -153,34 +152,7 @@ public class DemoClient {
|
|||
return new Subject();
|
||||
}
|
||||
|
||||
/*
|
||||
* To authenticate the DemoClient, kinit should be invoked ahead.
|
||||
* Here we try to get the Kerberos credential from the ticket cache.
|
||||
*/
|
||||
LoginContext context = new LoginContext("", new Subject(), null,
|
||||
new Configuration() {
|
||||
@Override
|
||||
public AppConfigurationEntry[] getAppConfigurationEntry(String name) {
|
||||
Map<String, String> options = new HashMap<>();
|
||||
options.put("useKeyTab", "false");
|
||||
options.put("storeKey", "false");
|
||||
options.put("doNotPrompt", "true");
|
||||
options.put("useTicketCache", "true");
|
||||
options.put("renewTGT", "true");
|
||||
options.put("refreshKrb5Config", "true");
|
||||
options.put("isInitiator", "true");
|
||||
String ticketCache = System.getenv("KRB5CCNAME");
|
||||
if (ticketCache != null) {
|
||||
options.put("ticketCache", ticketCache);
|
||||
}
|
||||
options.put("debug", "true");
|
||||
|
||||
return new AppConfigurationEntry[]{
|
||||
new AppConfigurationEntry("com.sun.security.auth.module.Krb5LoginModule",
|
||||
AppConfigurationEntry.LoginModuleControlFlag.REQUIRED,
|
||||
options)};
|
||||
}
|
||||
});
|
||||
LoginContext context = ClientUtils.getLoginContext();
|
||||
context.login();
|
||||
return context.getSubject();
|
||||
}
|
||||
|
|
|
@ -0,0 +1,120 @@
|
|||
/**
|
||||
*
|
||||
* 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.hadoop.hbase.util;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
import java.util.SortedMap;
|
||||
import java.util.TreeMap;
|
||||
import javax.security.auth.Subject;
|
||||
import javax.security.auth.login.AppConfigurationEntry;
|
||||
import javax.security.auth.login.Configuration;
|
||||
import javax.security.auth.login.LoginContext;
|
||||
import javax.security.auth.login.LoginException;
|
||||
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.apache.hadoop.hbase.thrift.generated.TCell;
|
||||
import org.apache.hadoop.hbase.thrift.generated.TRowResult;
|
||||
import org.apache.yetus.audience.InterfaceAudience;
|
||||
|
||||
/**
|
||||
* Common Utility class for clients
|
||||
*/
|
||||
@InterfaceAudience.Private
|
||||
public final class ClientUtils {
|
||||
|
||||
private ClientUtils() {
|
||||
// Empty block
|
||||
}
|
||||
|
||||
/**
|
||||
* To authenticate the demo client, kinit should be invoked ahead. Here we try to get the
|
||||
* Kerberos credential from the ticket cache
|
||||
*
|
||||
* @return LoginContext Object
|
||||
* @throws LoginException Exception thrown if unable to get LoginContext
|
||||
*/
|
||||
public static LoginContext getLoginContext() throws LoginException {
|
||||
|
||||
return new LoginContext(StringUtils.EMPTY, new Subject(), null, new Configuration() {
|
||||
@Override
|
||||
public AppConfigurationEntry[] getAppConfigurationEntry(String name) {
|
||||
Map<String, String> options = new HashMap<>();
|
||||
options.put("useKeyTab", "false");
|
||||
options.put("storeKey", "false");
|
||||
options.put("doNotPrompt", "true");
|
||||
options.put("useTicketCache", "true");
|
||||
options.put("renewTGT", "true");
|
||||
options.put("refreshKrb5Config", "true");
|
||||
options.put("isInitiator", "true");
|
||||
String ticketCache = System.getenv("KRB5CCNAME");
|
||||
if (ticketCache != null) {
|
||||
options.put("ticketCache", ticketCache);
|
||||
}
|
||||
options.put("debug", "true");
|
||||
|
||||
return new AppConfigurationEntry[]{new AppConfigurationEntry(
|
||||
"com.sun.security.auth.module.Krb5LoginModule",
|
||||
AppConfigurationEntry.LoginModuleControlFlag.REQUIRED, options)};
|
||||
}
|
||||
});
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* copy values into a TreeMap to get them in sorted order and print it
|
||||
*
|
||||
* @param rowResult Holds row name and then a map of columns to cells
|
||||
*/
|
||||
public static void printRow(final TRowResult rowResult) {
|
||||
|
||||
TreeMap<String, TCell> sorted = new TreeMap<>();
|
||||
for (Map.Entry<ByteBuffer, TCell> column : rowResult.columns.entrySet()) {
|
||||
sorted.put(utf8(column.getKey().array()), column.getValue());
|
||||
}
|
||||
|
||||
StringBuilder rowStr = new StringBuilder();
|
||||
for (SortedMap.Entry<String, TCell> entry : sorted.entrySet()) {
|
||||
rowStr.append(entry.getKey());
|
||||
rowStr.append(" => ");
|
||||
rowStr.append(utf8(entry.getValue().value.array()));
|
||||
rowStr.append("; ");
|
||||
}
|
||||
System.out.println("row: " + utf8(rowResult.row.array()) + ", cols: " + rowStr);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper to translate byte[]'s to UTF8 strings
|
||||
*
|
||||
* @param buf byte array buffer
|
||||
* @return UTF8 decoded string value
|
||||
*/
|
||||
public static String utf8(final byte[] buf) {
|
||||
try {
|
||||
return Bytes.toString(buf);
|
||||
} catch (IllegalArgumentException e) {
|
||||
return "[INVALID UTF-8]";
|
||||
}
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue