/*
* Copyright © 2014 Cask Data, Inc.
*
* Licensed 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 co.cask.cdap.data2.dataset2.lib.table;
import co.cask.cdap.api.common.Bytes;
import co.cask.cdap.api.dataset.lib.AbstractDataset;
import co.cask.cdap.api.dataset.table.Delete;
import co.cask.cdap.api.dataset.table.Put;
import co.cask.cdap.api.dataset.table.Row;
import co.cask.cdap.api.dataset.table.Scanner;
import co.cask.cdap.api.dataset.table.Table;
import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.base.Throwables;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.gson.Gson;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Map;
import javax.annotation.Nullable;
/**
* Handy dataset to be used for managing metadata
*/
public class MetadataStoreDataset extends AbstractDataset {
private static final Gson GSON = new Gson();
/**
* All rows we store use single column of this name.
*/
private static final byte[] COLUMN = Bytes.toBytes("c");
private final Table table;
public MetadataStoreDataset(Table table) {
super("ignored", table);
this.table = table;
}
protected <T> byte[] serialize(T value) {
return Bytes.toBytes(GSON.toJson(value));
}
protected <T> T deserialize(byte[] serialized, Type typeOfT) {
return GSON.fromJson(Bytes.toString(serialized), typeOfT);
}
public boolean exists(MDSKey id) {
Row row = table.get(id.getKey());
if (row.isEmpty()) {
return false;
}
byte[] value = row.get(COLUMN);
if (value == null) {
return false;
}
return true;
}
@Nullable
public <T> T get(MDSKey id, Type typeOfT) {
Row row = table.get(id.getKey());
if (row.isEmpty()) {
return null;
}
byte[] value = row.get(COLUMN);
if (value == null) {
return null;
}
return deserialize(value, typeOfT);
}
// returns first that matches
@Nullable
public <T> T getFirst(MDSKey id, Type typeOfT) {
try {
Scanner scan = table.scan(id.getKey(), Bytes.stopKeyForPrefix(id.getKey()));
try {
Row row = scan.next();
if (row == null || row.isEmpty()) {
return null;
}
byte[] value = row.get(COLUMN);
if (value == null) {
return null;
}
return deserialize(value, typeOfT);
} finally {
scan.close();
}
} catch (Exception e) {
throw Throwables.propagate(e);
}
}
// lists all that has same first id parts
public <T> List<T> list(MDSKey id, Type typeOfT) {
return list(id, typeOfT, Integer.MAX_VALUE);
}
// lists all that has same first id parts, with a limit
public <T> List<T> list(MDSKey id, Type typeOfT, int limit) {
return list(id, null, typeOfT, limit, Predicates.<T>alwaysTrue());
}
// lists all that has first id parts in range of startId and stopId
public <T> List<T> list(MDSKey startId, @Nullable MDSKey stopId, Type typeOfT, int limit,
Predicate<T> filter) {
return Lists.newArrayList(listKV(startId, stopId, typeOfT, limit, filter).values());
}
// returns mapping of all that has same first id parts
public <T> Map<MDSKey, T> listKV(MDSKey id, Type typeOfT) {
return listKV(id, typeOfT, Integer.MAX_VALUE);
}
// returns mapping of all that has same first id parts, with a limit
public <T> Map<MDSKey, T> listKV(MDSKey id, Type typeOfT, int limit) {
return listKV(id, null, typeOfT, limit, Predicates.<T>alwaysTrue());
}
// returns mapping of all that has first id parts in range of startId and stopId
public <T> Map<MDSKey, T> listKV(MDSKey startId, @Nullable MDSKey stopId, Type typeOfT, int limit,
Predicate<T> filter) {
byte[] startKey = startId.getKey();
byte[] stopKey = stopId == null ? Bytes.stopKeyForPrefix(startKey) : stopId.getKey();
try {
Map<MDSKey, T> map = Maps.newLinkedHashMap();
Scanner scan = table.scan(startKey, stopKey);
try {
Row next;
while ((limit > 0) && (next = scan.next()) != null) {
byte[] columnValue = next.get(COLUMN);
if (columnValue == null) {
continue;
}
T value = deserialize(columnValue, typeOfT);
if (filter.apply(value)) {
MDSKey key = new MDSKey(next.getRow());
map.put(key, value);
--limit;
}
}
return map;
} finally {
scan.close();
}
} catch (Exception e) {
throw Throwables.propagate(e);
}
}
/**
* Run a scan on MDS.
*
* @param startId scan start key
* @param stopId scan stop key
* @param typeOfT type of value
* @param function function to process each element returned from scan.
* If function.apply returns false then the scan is stopped.
* Also, function.apply should not return null.
* @param <T> type of value
*/
public <T> void scan(MDSKey startId, @Nullable MDSKey stopId, Type typeOfT, Function<KeyValue<T>, Boolean> function) {
byte[] startKey = startId.getKey();
byte[] stopKey = stopId == null ? Bytes.stopKeyForPrefix(startKey) : stopId.getKey();
Scanner scan = table.scan(startKey, stopKey);
try {
Row next;
while ((next = scan.next()) != null) {
byte[] columnValue = next.get(COLUMN);
if (columnValue == null) {
continue;
}
T value = deserialize(columnValue, typeOfT);
MDSKey key = new MDSKey(next.getRow());
//noinspection ConstantConditions
if (!function.apply(new KeyValue<>(key, value))) {
break;
}
}
} finally {
scan.close();
}
}
/**
* Output value of a scan.
*
* @param <T> Type of scan result object
*/
public class KeyValue<T> {
private final MDSKey key;
private final T value;
public KeyValue(MDSKey key, T value) {
this.key = key;
this.value = value;
}
public MDSKey getKey() {
return key;
}
public T getValue() {
return value;
}
}
public void deleteAll(MDSKey id) {
byte[] prefix = id.getKey();
byte[] stopKey = Bytes.stopKeyForPrefix(prefix);
try {
Scanner scan = table.scan(prefix, stopKey);
try {
Row next;
while ((next = scan.next()) != null) {
String columnValue = next.getString(COLUMN);
if (columnValue == null) {
continue;
}
table.delete(new Delete(next.getRow()).add(COLUMN));
}
} finally {
scan.close();
}
} catch (Exception e) {
throw Throwables.propagate(e);
}
}
public <T> void write(MDSKey id, T value) {
try {
table.put(new Put(id.getKey()).add(COLUMN, serialize(value)));
} catch (Exception e) {
throw Throwables.propagate(e);
}
}
}