com.palantir.atlasdb.table.description.render.TableClassRendererV2 Maven / Gradle / Ivy
/*
* (c) Copyright 2018 Palantir Technologies Inc. All rights reserved.
*
* 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 com.palantir.atlasdb.table.description.render;
import static com.palantir.atlasdb.AtlasDbConstants.SCHEMA_V2_TABLE_NAME;
import static com.palantir.atlasdb.table.description.render.ColumnRenderers.VarName;
import static com.palantir.atlasdb.table.description.render.Renderers.CamelCase;
import static com.palantir.atlasdb.table.description.render.Renderers.addParametersFromRowComponents;
import static com.palantir.atlasdb.table.description.render.Renderers.getArgumentsFromRowComponents;
import static com.palantir.atlasdb.table.description.render.Renderers.getColumnClass;
import static com.palantir.atlasdb.table.description.render.Renderers.getColumnClassForGenericTypeParameter;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMultimap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.palantir.atlasdb.encoding.PtBytes;
import com.palantir.atlasdb.keyvalue.api.Cell;
import com.palantir.atlasdb.keyvalue.api.ColumnSelection;
import com.palantir.atlasdb.keyvalue.api.Namespace;
import com.palantir.atlasdb.keyvalue.api.RangeRequest;
import com.palantir.atlasdb.keyvalue.api.RowResult;
import com.palantir.atlasdb.keyvalue.api.TableReference;
import com.palantir.atlasdb.table.description.NameComponentDescription;
import com.palantir.atlasdb.table.description.NamedColumnDescription;
import com.palantir.atlasdb.table.description.Schemas;
import com.palantir.atlasdb.table.description.TableDefinition;
import com.palantir.atlasdb.table.description.TableMetadata;
import com.palantir.atlasdb.table.generation.ColumnValues;
import com.palantir.atlasdb.transaction.api.Transaction;
import com.palantir.common.base.BatchingVisitableView;
import com.palantir.common.persist.Persistables;
import com.palantir.goethe.Goethe;
import com.squareup.javapoet.AnnotationSpec;
import com.squareup.javapoet.ClassName;
import com.squareup.javapoet.FieldSpec;
import com.squareup.javapoet.JavaFile;
import com.squareup.javapoet.MethodSpec;
import com.squareup.javapoet.ParameterizedTypeName;
import com.squareup.javapoet.TypeName;
import com.squareup.javapoet.TypeSpec;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.NavigableMap;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.SortedSet;
import java.util.function.Function;
import java.util.stream.Collectors;
import javax.annotation.processing.Generated;
import javax.lang.model.element.Modifier;
public class TableClassRendererV2 {
private final String packageName;
private final Namespace namespace;
private final TableMetadata tableMetadata;
private final String rawTableName;
private final String tableName;
private final String tableClassName;
private final ClassName tableType;
private final String rowClassName;
private final ClassName rowType;
private final String rowResultClassName;
private final ClassName rowResultType;
private final String simpleTableName;
private final ClassName simpleTableType;
public TableClassRendererV2(String packageName, Namespace namespace, String rawTableName, TableDefinition table) {
Preconditions.checkArgument(Schemas.isTableNameValid(rawTableName), "Invalid table name %s", rawTableName);
this.packageName = packageName;
this.namespace = namespace;
this.rawTableName = rawTableName;
this.tableName = Renderers.getClassTableName(rawTableName, table);
this.tableMetadata = table.toTableMetadata();
this.simpleTableName = this.tableName + SCHEMA_V2_TABLE_NAME;
this.tableClassName = this.tableName + "Table";
this.rowClassName = this.tableName + "Row";
this.rowResultClassName = this.tableName + "RowResult";
this.tableType = ClassName.get(packageName, this.tableClassName);
this.simpleTableType = ClassName.get(packageName, this.simpleTableName);
this.rowType = this.tableType.nestedClass(this.rowClassName);
this.rowResultType = this.tableType.nestedClass(this.rowResultClassName);
}
public String render() {
JavaFile javaFile = JavaFile.builder(packageName, this.buildTypeSpec())
.skipJavaLangImports(true)
.indent(" ")
.build();
return Goethe.formatAsString(javaFile);
}
private TypeSpec buildTypeSpec() {
TypeSpec.Builder tableBuilder = TypeSpec.classBuilder(this.simpleTableName)
.addAnnotation(AnnotationSpec.builder(Generated.class)
.addMember("value", "$S", TableRendererV2.class.getName())
.build())
.addAnnotation(AnnotationSpec.builder(SuppressWarnings.class)
.addMember("value", "$S", "all")
.addMember("value", "$S", "deprecation")
.build())
.addModifiers(Modifier.PUBLIC);
getFields().forEach(tableBuilder::addField);
getConstructors().forEach(tableBuilder::addMethod);
getStaticFactories().forEach(tableBuilder::addMethod);
getMethods().forEach(tableBuilder::addMethod);
return tableBuilder.build();
}
private List getFields() {
List results = new ArrayList<>();
results.add(FieldSpec.builder(Transaction.class, "t")
.addModifiers(Modifier.PRIVATE, Modifier.FINAL)
.build());
results.add(FieldSpec.builder(String.class, "rawTableName")
.addModifiers(Modifier.PRIVATE, Modifier.FINAL, Modifier.STATIC)
.initializer("$S", rawTableName)
.build());
results.add(FieldSpec.builder(TableReference.class, "tableRef")
.addModifiers(Modifier.PRIVATE, Modifier.FINAL)
.build());
return results;
}
private List getConstructors() {
List results = new ArrayList<>();
results.add(MethodSpec.constructorBuilder()
.addModifiers(Modifier.PRIVATE)
.addParameter(Transaction.class, "t")
.addParameter(Namespace.class, "namespace")
.addStatement("this.tableRef = $T.create(namespace, rawTableName)", TableReference.class)
.addStatement("this.t = t")
.build());
return results;
}
private List getStaticFactories() {
List results = new ArrayList<>();
results.add(MethodSpec.methodBuilder("of")
.addModifiers(Modifier.PUBLIC, Modifier.STATIC)
.addParameter(Transaction.class, "t")
.addParameter(Namespace.class, "namespace")
.returns(simpleTableType)
.addStatement("return new $T($L, $L)", simpleTableType, "t", "namespace")
.build());
return results;
}
private List getMethods() {
List results = new ArrayList<>();
results.add(renderGetRawTableName());
results.add(renderGetTableRef());
results.add(renderGetTableName());
results.add(renderGetNamespace());
if (!tableMetadata.getColumns().hasDynamicColumns()) {
results.addAll(renderNamedGet());
results.addAll(renderNamedDelete());
results.addAll(renderNamedPutAndUpdate());
}
return results;
}
private MethodSpec renderGetRawTableName() {
return MethodSpec.methodBuilder("getRawTableName")
.addModifiers(Modifier.PUBLIC, Modifier.STATIC)
.addStatement("return rawTableName")
.returns(String.class)
.build();
}
private MethodSpec renderGetTableName() {
return MethodSpec.methodBuilder("getTableName")
.addModifiers(Modifier.PUBLIC)
.addStatement("return tableRef.getQualifiedName()")
.returns(String.class)
.build();
}
private MethodSpec renderGetTableRef() {
return MethodSpec.methodBuilder("getTableRef")
.addModifiers(Modifier.PUBLIC)
.addStatement("return tableRef")
.returns(TableReference.class)
.build();
}
private MethodSpec renderGetNamespace() {
return MethodSpec.methodBuilder("getNamespace")
.addModifiers(Modifier.PUBLIC)
.addStatement("return tableRef.getNamespace()")
.returns(Namespace.class)
.build();
}
private List renderNamedGet() {
List getterResults = new ArrayList<>();
for (NamedColumnDescription col : ColumnRenderers.namedColumns(tableMetadata)) {
getterResults.add(renderNamedGetColumn(col));
if (tableMetadata.getRowMetadata().getRowParts().size() == 1) {
getterResults.add(renderNamedGetSeveralRows(col));
if (tableMetadata.isRangeScanAllowed()) {
getterResults.add(renderNamedGetRangeColumn(col));
getterResults.add(renderNamedGetRangeStartEnd(col));
getterResults.add(renderNamedGetRangeColumnLimit(col));
}
} else {
getterResults.add(renderNamedGetSeveralRowObjects(col));
if (tableMetadata.isRangeScanAllowed()) {
getterResults.add(renderNamedGetRangeColumnRowObjects(col));
getterResults.add(renderNamedGetRangeColumnRowObjectsLimit(col));
}
}
}
return getterResults;
}
private MethodSpec renderNamedGetColumn(NamedColumnDescription col) {
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("get" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc("Returns the value for column $L and specified row components.", VarName(col));
getterBuilder = addParametersFromRowComponents(getterBuilder, tableMetadata);
getterBuilder.returns(ParameterizedTypeName.get(
ClassName.get(Optional.class), TypeName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement("$T row = $T.of($L)", rowType, rowType, getArgumentsFromRowComponents(tableMetadata))
.addStatement("byte[] bytes = row.persistToBytes()")
.addStatement(
"$T colSelection = \n" + "$T.create($T.of($T.toCachedBytes($S)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
col.getShortName())
.addCode("\n")
.addStatement(
"$T rowResult = t.getRows(tableRef, $T.of(bytes), colSelection).get(bytes)",
RowResult.class,
ImmutableSet.class)
.addCode(
"if (rowResult == null) {\n"
+ " return $T.empty();\n"
+ "} else {\n"
+ " return $T.of($T.of(rowResult).get$L());\n"
+ "}\n",
Optional.class,
Optional.class,
rowResultType,
VarName(col));
return getterBuilder.build();
}
private MethodSpec renderNamedGetSeveralRows(NamedColumnDescription col) {
com.palantir.logsafe.Preconditions.checkArgument(
tableMetadata.getRowMetadata().getRowParts().size() == 1);
NameComponentDescription rowComponent =
tableMetadata.getRowMetadata().getRowParts().get(0);
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("get" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from the specified row keys to their value at column $L.\n"
+ "As the $L values are all loaded in memory, do not use for large amounts of data.\n"
+ "If the column does not exist for a key, the entry will be omitted from the map.",
VarName(col),
VarName(col))
.addParameter(
ParameterizedTypeName.get(
ClassName.get(Iterable.class),
ClassName.get(rowComponent.getType().getJavaClass())),
"rowKeys");
getterBuilder.returns(ParameterizedTypeName.get(
ClassName.get(Map.class),
ClassName.get(rowComponent.getType().getJavaClass()),
TypeName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T colSelection = \n " + "$T.create($T.of($T.toCachedBytes($S)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
col.getShortName())
.addStatement(
"$T<$T> rows = $T\n"
+ ".newArrayList(rowKeys)\n"
+ ".stream()\n"
+ ".map($T::of)\n"
+ ".collect($T.toList())",
List.class,
rowType,
Lists.class,
rowType,
Collectors.class)
.addCode("\n")
.addStatement(
"$T> results = " + "t.getRows(tableRef, $T.persistAll(rows), colSelection)",
NavigableMap.class,
RowResult.class,
Persistables.class)
.addStatement(
"return results\n"
+ ".values()\n"
+ ".stream()\n"
+ ".map(entry -> $T.of(entry))\n"
+ ".collect($T.toMap(\n"
+ " entry -> entry.getRowName().get$L(), \n"
+ " $T::get$L))",
rowResultType,
Collectors.class,
CamelCase(rowComponent.getComponentName()),
rowResultType,
VarName(col));
return getterBuilder.build();
}
private MethodSpec renderNamedGetSeveralRowObjects(NamedColumnDescription col) {
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("get" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from the specified row objects to their value at column $L.\n"
+ "As the $L values are all loaded in memory, do not use for large amounts of data.\n"
+ "If the column does not exist for a key, the entry will be omitted from the map.",
VarName(col),
VarName(col))
.addParameter(ParameterizedTypeName.get(ClassName.get(Iterable.class), rowType), "rowKeys");
getterBuilder.returns(ParameterizedTypeName.get(
ClassName.get(Map.class), rowType, TypeName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T colSelection = \n " + "$T.create($T.of($T.toCachedBytes($S)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
col.getShortName())
.addCode("\n")
.addStatement(
"$T> results = "
+ "t.getRows(tableRef, $T.persistAll(rowKeys), colSelection)",
NavigableMap.class,
RowResult.class,
Persistables.class)
.addStatement(
"return results\n"
+ ".values()\n"
+ ".stream()\n"
+ ".map(entry -> $T.of(entry))\n"
+ ".collect($T.toMap(\n"
+ " entry -> entry.getRowName(), \n"
+ " $T::get$L))",
rowResultType,
Collectors.class,
rowResultType,
VarName(col));
return getterBuilder.build();
}
private MethodSpec renderNamedGetRangeColumn(NamedColumnDescription col) {
com.palantir.logsafe.Preconditions.checkArgument(
tableMetadata.getRowMetadata().getRowParts().size() == 1);
NameComponentDescription rowComponent =
tableMetadata.getRowMetadata().getRowParts().get(0);
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("getSmallRowRange" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from all the row keys in a rangeRequest to their value at column $L\n"
+ "(if that column exists for the row-key). As the $L values are all loaded in memory,\n"
+ "do not use for large amounts of data. The order of results is preserved in the map.",
VarName(col),
VarName(col))
.addParameter(RangeRequest.class, "rangeRequest")
.returns(ParameterizedTypeName.get(
ClassName.get(LinkedHashMap.class),
ClassName.get(rowComponent.getType().getJavaClass()),
ClassName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T colSelection =\n" + "$T.create($T.of($T.toCachedBytes($L)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
ColumnRenderers.short_name(col))
.addStatement("rangeRequest = rangeRequest.getBuilder().retainColumns(colSelection).build()")
.addStatement(
"$T.checkArgument(rangeRequest.getColumnNames().size() <= 1,\n$S)",
Preconditions.class,
"Must not request columns other than " + VarName(col) + ".")
.addCode("\n")
.addStatement(
"$T<$T, $T> resultsMap = new $T<>()",
LinkedHashMap.class,
rowComponent.getType().getJavaClass(),
getColumnClassForGenericTypeParameter(col),
LinkedHashMap.class)
.addStatement(
"$T.of(t.getRange(tableRef, rangeRequest))\n"
+ ".immutableCopy().forEach(entry -> {\n"
+ " $T resultEntry =\n "
+ " $T.of(entry);\n"
+ " resultsMap.put(resultEntry.getRowName().get$L(), resultEntry.get$L());\n"
+ "})",
BatchingVisitableView.class,
rowResultType,
rowResultType,
CamelCase(rowComponent.getComponentName()),
VarName(col))
.addStatement("return resultsMap");
return getterBuilder.build();
}
private MethodSpec renderNamedGetRangeStartEnd(NamedColumnDescription col) {
com.palantir.logsafe.Preconditions.checkArgument(
tableMetadata.getRowMetadata().getRowParts().size() == 1);
NameComponentDescription rowComponent =
tableMetadata.getRowMetadata().getRowParts().get(0);
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("getSmallRowRange" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from all the row keys in a range to their value at column $L\n"
+ "(if that column exists for the row-key). As the $L values are all loaded in memory,\n"
+ "do not use for large amounts of data. The order of results is preserved in the map.",
VarName(col),
VarName(col))
.addParameter(rowComponent.getType().getJavaClass(), "startInclusive")
.addParameter(rowComponent.getType().getJavaClass(), "endExclusive")
.returns(ParameterizedTypeName.get(
ClassName.get(LinkedHashMap.class),
ClassName.get(rowComponent.getType().getJavaClass()),
ClassName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T rangeRequest = $T.builder()\n"
+ ".startRowInclusive($T.of(startInclusive).persistToBytes())\n"
+ ".endRowExclusive($T.of(endExclusive).persistToBytes())\n"
+ ".build()",
RangeRequest.class,
RangeRequest.class,
rowType,
rowType)
.addStatement("return getSmallRowRange$L(rangeRequest)", VarName(col));
return getterBuilder.build();
}
private MethodSpec renderNamedGetRangeColumnLimit(NamedColumnDescription col) {
com.palantir.logsafe.Preconditions.checkArgument(
tableMetadata.getRowMetadata().getRowParts().size() == 1);
NameComponentDescription rowComponent =
tableMetadata.getRowMetadata().getRowParts().get(0);
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("getSmallRowRange" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from the first sizeLimit row keys in a rangeRequest to their value\n"
+ "at column $L (if that column exists). As the $L entries are all loaded in memory,\n"
+ "do not use for large values of sizeLimit. The order of results is preserved in the"
+ " map.",
VarName(col),
VarName(col))
.addParameter(RangeRequest.class, "rangeRequest")
.addParameter(int.class, "sizeLimit")
.returns(ParameterizedTypeName.get(
ClassName.get(LinkedHashMap.class),
ClassName.get(rowComponent.getType().getJavaClass()),
ClassName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T colSelection =\n" + "$T.create($T.of($T.toCachedBytes($L)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
ColumnRenderers.short_name(col))
.addStatement("rangeRequest = rangeRequest.getBuilder()."
+ "retainColumns(colSelection).batchHint(sizeLimit).build()")
.addStatement(
"$T.checkArgument(rangeRequest.getColumnNames().size() <= 1,\n$S)",
Preconditions.class,
"Must not request columns other than " + VarName(col) + ".")
.addCode("\n")
.addStatement(
"$T<$T, $T> resultsMap = new $T<>()",
LinkedHashMap.class,
rowComponent.getType().getJavaClass(),
getColumnClassForGenericTypeParameter(col),
LinkedHashMap.class)
.addStatement(
"$T.of(t.getRange(tableRef, rangeRequest))\n"
+ ".batchAccept(sizeLimit, batch -> {\n"
+ " batch.forEach(entry -> {\n"
+ " $T resultEntry =\n "
+ " $T.of(entry);\n"
+ " resultsMap.put(resultEntry.getRowName().get$L(), resultEntry.get$L());\n"
+ " });\n"
+ " return false; // stops the traversal after the first batch\n"
+ "})",
BatchingVisitableView.class,
rowResultType,
rowResultType,
CamelCase(rowComponent.getComponentName()),
VarName(col))
.addStatement("return resultsMap");
return getterBuilder.build();
}
private MethodSpec renderNamedGetRangeColumnRowObjects(NamedColumnDescription col) {
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("getSmallRowRange" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from all the rows in a RangeRequest to their value at column $L\n"
+ "(if that column exists for the row). As the $L values are all loaded in memory, \n"
+ "do not use for large amounts of data. The order of results is preserved in the map.",
VarName(col),
VarName(col))
.addParameter(RangeRequest.class, "rangeRequest")
.returns(ParameterizedTypeName.get(
ClassName.get(LinkedHashMap.class),
rowType,
ClassName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T colSelection =\n" + "$T.create($T.of($T.toCachedBytes($L)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
ColumnRenderers.short_name(col))
.addStatement("rangeRequest = rangeRequest.getBuilder().retainColumns(colSelection).build()")
.addStatement(
"$T.checkArgument(rangeRequest.getColumnNames().size() <= 1,\n$S)",
Preconditions.class,
"Must not request additional columns.")
.addCode("\n")
.addStatement(
"$T<$T, $T> resultsMap = new $T<>()",
LinkedHashMap.class,
rowType,
getColumnClassForGenericTypeParameter(col),
LinkedHashMap.class)
.addStatement(
"$T.of(t.getRange(tableRef, rangeRequest))\n"
+ ".immutableCopy().forEach(entry -> {\n"
+ " $T resultEntry =\n "
+ " $T.of(entry);\n"
+ " resultsMap.put(resultEntry.getRowName(), resultEntry.get$L());\n"
+ "})",
BatchingVisitableView.class,
rowResultType,
rowResultType,
VarName(col))
.addStatement("return resultsMap");
return getterBuilder.build();
}
private MethodSpec renderNamedGetRangeColumnRowObjectsLimit(NamedColumnDescription col) {
MethodSpec.Builder getterBuilder = MethodSpec.methodBuilder("getSmallRowRange" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc(
"Returns a mapping from the first sizeLimit rows in a rangeRequest to their value\n"
+ "at column $L (if that column exists). As the $L entries are all loaded in memory,\n"
+ "do not use for large values of sizeLimit. The order of results is preserved in the"
+ " map.",
VarName(col),
VarName(col))
.addParameter(RangeRequest.class, "rangeRequest")
.addParameter(int.class, "sizeLimit")
.returns(ParameterizedTypeName.get(
ClassName.get(LinkedHashMap.class),
rowType,
ClassName.get(getColumnClassForGenericTypeParameter(col))));
getterBuilder
.addStatement(
"$T colSelection =\n" + "$T.create($T.of($T.toCachedBytes($L)))",
ColumnSelection.class,
ColumnSelection.class,
ImmutableList.class,
PtBytes.class,
ColumnRenderers.short_name(col))
.addStatement("rangeRequest = rangeRequest.getBuilder()."
+ "retainColumns(colSelection).batchHint(sizeLimit).build()")
.addStatement(
"$T.checkArgument(rangeRequest.getColumnNames().size() <= 1,\n$S)",
Preconditions.class,
"Must not request columns other than " + VarName(col) + ".")
.addCode("\n")
.addStatement(
"$T<$T, $T> resultsMap = new $T<>()",
LinkedHashMap.class,
rowType,
getColumnClassForGenericTypeParameter(col),
LinkedHashMap.class)
.addStatement(
"$T.of(t.getRange(tableRef, rangeRequest))\n"
+ ".batchAccept(sizeLimit, batch -> {\n"
+ " batch.forEach(entry -> {\n"
+ " $T resultEntry =\n "
+ " $T.of(entry);\n"
+ " resultsMap.put(resultEntry.getRowName(), resultEntry.get$L());\n"
+ " });\n"
+ " return false; // stops the traversal after the first batch\n"
+ "})",
BatchingVisitableView.class,
rowResultType,
rowResultType,
VarName(col))
.addStatement("return resultsMap");
return getterBuilder.build();
}
private List renderNamedDelete() {
List deleteResults = new ArrayList<>();
deleteResults.add(renderNamedDeleteRow());
for (NamedColumnDescription col : ColumnRenderers.namedColumns(tableMetadata)) {
deleteResults.add(renderNamedDeleteColumn(col));
}
return deleteResults;
}
private MethodSpec renderNamedDeleteRow() {
MethodSpec.Builder deleteRowBuilder = MethodSpec.methodBuilder("deleteRow")
.addModifiers(Modifier.PUBLIC)
.addJavadoc("Delete all columns for specified row components.");
deleteRowBuilder = addParametersFromRowComponents(deleteRowBuilder, tableMetadata);
SortedSet namedColumns = ColumnRenderers.namedColumns(tableMetadata);
deleteRowBuilder
.addStatement("$T row = $T.of($L)", rowType, rowType, getArgumentsFromRowComponents(tableMetadata))
.addStatement("byte[] rowBytes = row.persistToBytes()", Persistables.class)
.addStatement(
"$T<$T> cells = $T.newHashSetWithExpectedSize($L)",
Set.class,
Cell.class,
Sets.class,
namedColumns.size());
for (NamedColumnDescription col : namedColumns) {
deleteRowBuilder.addStatement(
"cells.add($T.create(rowBytes, $T.toCachedBytes($L)))",
Cell.class,
PtBytes.class,
ColumnRenderers.short_name(col));
}
deleteRowBuilder.addStatement("t.delete(tableRef, cells)");
return deleteRowBuilder.build();
}
private MethodSpec renderNamedDeleteColumn(NamedColumnDescription col) {
MethodSpec.Builder deleteColumnBuilder = MethodSpec.methodBuilder("delete" + VarName(col))
.addModifiers(Modifier.PUBLIC)
.addJavadoc("Delete the value at column $L (if it exists) for the specified row-key.", VarName(col));
deleteColumnBuilder = addParametersFromRowComponents(deleteColumnBuilder, tableMetadata);
return deleteColumnBuilder
.addStatement("$T row = $T.of($L)", rowType, rowType, getArgumentsFromRowComponents(tableMetadata))
.addStatement("byte[] rowBytes = row.persistToBytes()", Persistables.class)
.addStatement(
"$T<$T> cells = $T.of($T.create(rowBytes, $T.toCachedBytes($L)))",
Set.class,
Cell.class,
ImmutableSet.class,
Cell.class,
PtBytes.class,
ColumnRenderers.short_name(col))
.addStatement("t.delete(tableRef, cells)")
.build();
}
private List renderNamedPutAndUpdate() {
List putAndUpdateResults = new ArrayList<>();
for (NamedColumnDescription col : ColumnRenderers.namedColumns(tableMetadata)) {
putAndUpdateResults.add(renderNamedPutColumn(col));
putAndUpdateResults.add(renderNamedUpdateColumn(col));
}
return putAndUpdateResults;
}
private MethodSpec renderNamedPutColumn(NamedColumnDescription col) {
MethodSpec.Builder putColumnBuilder = MethodSpec.methodBuilder("put" + VarName(col))
.addJavadoc("Takes the row-keys and a value to be inserted at column $L.", VarName(col))
.addModifiers(Modifier.PUBLIC);
putColumnBuilder = addParametersFromRowComponents(putColumnBuilder, tableMetadata);
TypeName columnValueType = tableType.nestedClass(VarName(col));
putColumnBuilder.addParameter(getColumnClass(col), col.getLongName());
putColumnBuilder
.addStatement("$T row = $T.of($L)", rowType, rowType, getArgumentsFromRowComponents(tableMetadata))
.addStatement(
"t.put(tableRef, $T.toCellValues($T.of(row, $T.of($L))))",
ColumnValues.class,
ImmutableMultimap.class,
columnValueType,
col.getLongName());
return putColumnBuilder.build();
}
private MethodSpec renderNamedUpdateColumn(NamedColumnDescription col) {
MethodSpec.Builder updateColumnIfExistsBuilder = MethodSpec.methodBuilder("update" + VarName(col))
.addJavadoc(
"Takes a function that would update the value at column $L, for the specified row\n"
+ "components. No effect if there is no value at that column. Doesn't do an additional\n"
+ "write if the new value is the same as the old one.",
VarName(col))
.addModifiers(Modifier.PUBLIC);
updateColumnIfExistsBuilder = addParametersFromRowComponents(updateColumnIfExistsBuilder, tableMetadata);
updateColumnIfExistsBuilder.addParameter(
ParameterizedTypeName.get(
ClassName.get(Function.class),
TypeName.get(getColumnClassForGenericTypeParameter(col)),
TypeName.get(getColumnClassForGenericTypeParameter(col))),
"processor");
String args = getArgumentsFromRowComponents(tableMetadata);
updateColumnIfExistsBuilder
.addStatement(
"$T<$T> result = get$L($L)",
Optional.class,
getColumnClassForGenericTypeParameter(col),
VarName(col),
args)
.beginControlFlow("if (result.isPresent())")
.addStatement("$T newValue = processor.apply(result.get())", getColumnClassForGenericTypeParameter(col))
.beginControlFlow("if ($T.equals(newValue, result.get()) == false)", Objects.class)
.addStatement("put$L($L, processor.apply(result.get()))", VarName(col), args)
.endControlFlow()
.endControlFlow();
return updateColumnIfExistsBuilder.build();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy