com.facebook.swift.codec.metadata.ThriftFieldMetadata Maven / Gradle / Ivy
/*
* Copyright (C) 2012 Facebook, 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 com.facebook.swift.codec.metadata;
import com.google.common.base.Function;
import com.google.common.base.Optional;
import com.google.common.base.Predicate;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import javax.annotation.concurrent.Immutable;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import static com.facebook.swift.codec.ThriftField.RECURSIVE_REFERENCE_ANNOTATION_NAME;
import static com.facebook.swift.codec.ThriftField.Requiredness;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
/**
* ThriftFieldMetadata defines a single thrift field including the value extraction and injection
* points.
*/
@Immutable
public class ThriftFieldMetadata
{
private final short id;
private final ThriftTypeReference thriftTypeReference;
private final String name;
private final FieldKind fieldKind;
private final List injections;
private final Map idlAnnotations;
private final Optional constructorInjection;
private final Optional methodInjection;
private final Optional extraction;
private final Optional coercion;
private final ImmutableList documentation;
private final boolean isRecursiveReference;
private final Requiredness requiredness;
public ThriftFieldMetadata(
short id,
boolean isLegacyId,
boolean isRecursiveReference,
Requiredness requiredness,
Map idlAnnotations,
ThriftTypeReference thriftTypeReference,
String name,
FieldKind fieldKind,
List injections,
Optional constructorInjection,
Optional methodInjection,
Optional extraction,
Optional coercion
)
{
this.isRecursiveReference = isRecursiveReference;
this.requiredness = requiredness;
this.thriftTypeReference = checkNotNull(thriftTypeReference, "thriftType is null");
this.fieldKind = checkNotNull(fieldKind, "type is null");
this.name = checkNotNull(name, "name is null");
this.injections = ImmutableList.copyOf(checkNotNull(injections, "injections is null"));
this.constructorInjection = checkNotNull(constructorInjection, "constructorInjection is null");
this.methodInjection = checkNotNull(methodInjection, "methodInjection is null");
this.extraction = checkNotNull(extraction, "extraction is null");
this.coercion = checkNotNull(coercion, "coercion is null");
switch (fieldKind) {
case THRIFT_FIELD:
if (isLegacyId) {
checkArgument(id < 0, "isLegacyId should only be specified on fields with negative IDs");
} else {
checkArgument(id >= 0, "isLegacyId must be specified on fields with negative IDs");
}
break;
case THRIFT_UNION_ID:
checkArgument(isLegacyId, "isLegacyId should be implicitly set on ThriftUnionId fields");
checkArgument(id == Short.MIN_VALUE, "thrift union id must be Short.MIN_VALUE");
break;
}
checkArgument(!isRecursiveReference
|| requiredness == Requiredness.OPTIONAL,
"Fields marked as recursive references must always be optional");
checkArgument(!injections.isEmpty()
|| extraction.isPresent()
|| constructorInjection.isPresent()
|| methodInjection.isPresent(), "A thrift field must have an injection or extraction point");
this.id = id;
if (extraction.isPresent()) {
if (extraction.get() instanceof ThriftFieldExtractor) {
ThriftFieldExtractor e = (ThriftFieldExtractor)extraction.get();
this.documentation = ThriftCatalog.getThriftDocumentation(e.getField());
} else if (extraction.get() instanceof ThriftMethodExtractor) {
ThriftMethodExtractor e = (ThriftMethodExtractor)extraction.get();
this.documentation = ThriftCatalog.getThriftDocumentation(e.getMethod());
}
else {
this.documentation = ImmutableList.of();
}
} else {
// no extraction = no documentation
this.documentation = ImmutableList.of();
}
this.idlAnnotations = idlAnnotations;
}
public short getId()
{
return id;
}
public ThriftType getThriftType()
{
return thriftTypeReference.get();
}
public Requiredness getRequiredness() { return requiredness; }
public String getName()
{
return name;
}
public FieldKind getType()
{
return fieldKind;
}
public Map getIdlAnnotations()
{
ImmutableMap.Builder annotationsBuilder = ImmutableMap.builder();
annotationsBuilder.putAll(idlAnnotations);
if (isRecursiveReference()) {
annotationsBuilder.put(RECURSIVE_REFERENCE_ANNOTATION_NAME, "true");
}
return annotationsBuilder.build();
}
public boolean isTypeReferenceRecursive()
{
return thriftTypeReference.isRecursive();
}
public boolean isRecursiveReference()
{
return isRecursiveReference;
}
public boolean isInternal()
{
switch (getType()) {
// These are normal thrift fields (i.e. they should be emitted by the swift2thrift generator)
case THRIFT_FIELD:
return false;
// Other fields types are used internally in swift, but do not make up part of the external
// thrift interface
default:
return true;
}
}
public boolean isReadOnly()
{
return injections.isEmpty() && !constructorInjection.isPresent() && !methodInjection.isPresent();
}
public boolean isWriteOnly()
{
return !extraction.isPresent();
}
public List getInjections()
{
return injections;
}
public Optional getConstructorInjection()
{
return constructorInjection;
}
public Optional getMethodInjection()
{
return methodInjection;
}
public Optional getExtraction()
{
return extraction;
}
public Optional getCoercion()
{
return coercion;
}
public ImmutableList getDocumentation()
{
return documentation;
}
@Override
public String toString()
{
final StringBuilder sb = new StringBuilder();
sb.append("ThriftFieldMetadata");
sb.append("{id=").append(id);
sb.append(", thriftType=").append(thriftTypeReference);
sb.append(", name='").append(name).append('\'');
sb.append(", fieldKind=").append(fieldKind);
sb.append(", injections=").append(injections);
sb.append(", constructorInjection=").append(constructorInjection);
sb.append(", methodInjection=").append(methodInjection);
sb.append(", extraction=").append(extraction);
sb.append(", coercion=").append(coercion);
sb.append('}');
return sb.toString();
}
@Override
public int hashCode()
{
return Objects.hash(id, thriftTypeReference, name);
}
@Override
public boolean equals(Object obj)
{
if (this == obj) {
return true;
}
if (obj == null || getClass() != obj.getClass()) {
return false;
}
final ThriftFieldMetadata other = (ThriftFieldMetadata) obj;
return Objects.equals(this.id, other.id) && Objects.equals(this.thriftTypeReference, other.thriftTypeReference) && Objects.equals(this.name, other.name);
}
public static Function getIdGetter()
{
return new Function() {
@Override
public Short apply(ThriftFieldMetadata metadata)
{
return metadata.getId();
}
};
}
public static Predicate isTypePredicate(final FieldKind type)
{
return new Predicate() {
@Override
public boolean apply(ThriftFieldMetadata fieldMetadata)
{
return fieldMetadata.getType() == type;
}
};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy