All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.api.generator.gapic.protoparser.MethodSignatureParser Maven / Gradle / Ivy

There is a newer version: 2.50.0
Show newest version
// Copyright 2020 Google LLC
//
// 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.google.api.generator.gapic.protoparser;

import com.google.api.ClientProto;
import com.google.api.generator.engine.ast.TypeNode;
import com.google.api.generator.gapic.model.Field;
import com.google.api.generator.gapic.model.Message;
import com.google.api.generator.gapic.model.MethodArgument;
import com.google.api.generator.gapic.model.ResourceName;
import com.google.api.generator.gapic.model.ResourceReference;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import com.google.common.base.Strings;
import com.google.common.collect.Lists;
import com.google.protobuf.Descriptors.MethodDescriptor;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;

// TODO(miraleung): Add tests for this class. Currently exercised in integration tests.
public class MethodSignatureParser {
  private static final String DOT = ".";
  private static final String METHOD_SIGNATURE_DELIMITER = "\\s*,\\s*";

  /** Parses a list of method signature annotations out of an RPC. */
  public static List> parseMethodSignatures(
      MethodDescriptor methodDescriptor,
      String servicePackage,
      TypeNode methodInputType,
      Map messageTypes,
      Map resourceNames,
      Set outputArgResourceNames) {
    List stringSigs =
        methodDescriptor.getOptions().getExtension(ClientProto.methodSignature);

    List> signatures = new ArrayList<>();
    if (stringSigs.isEmpty()) {
      return signatures;
    }

    Map patternsToResourceNames =
        ResourceParserHelpers.createPatternResourceNameMap(resourceNames);
    Message inputMessage = messageTypes.get(methodInputType.reference().fullName());

    // Example from Expand in echo.proto:
    // stringSigs: ["content,error", "content,error,info"].
    for (String stringSig : stringSigs) {
      if (Strings.isNullOrEmpty(stringSig)) {
        signatures.add(Collections.emptyList());
        continue;
      }

      List argumentNames = new ArrayList<>();
      Map> argumentNameToOverloads = new HashMap<>();

      // stringSig.split: ["content", "error"].
      for (String argumentName : stringSig.split(METHOD_SIGNATURE_DELIMITER)) {
        // For resource names, this will be empty.
        List argumentFieldPathAcc = new ArrayList<>();
        // There should be more than one type returned only when we encounter a resource name.
        Map argumentTypes =
            parseTypeFromArgumentName(
                argumentName,
                servicePackage,
                inputMessage,
                messageTypes,
                resourceNames,
                patternsToResourceNames,
                argumentFieldPathAcc,
                outputArgResourceNames);
        int dotLastIndex = argumentName.lastIndexOf(DOT);
        String actualArgumentName =
            dotLastIndex < 0 ? argumentName : argumentName.substring(dotLastIndex + 1);
        argumentNames.add(actualArgumentName);

        argumentNameToOverloads.put(
            actualArgumentName,
            argumentTypes.entrySet().stream()
                .map(
                    e ->
                        MethodArgument.builder()
                            .setName(actualArgumentName)
                            .setType(e.getKey())
                            .setField(e.getValue())
                            .setIsResourceNameHelper(
                                argumentTypes.size() > 1 && !e.getKey().equals(TypeNode.STRING))
                            .setNestedFields(argumentFieldPathAcc)
                            .build())
                .collect(Collectors.toList()));
      }
      signatures.addAll(flattenMethodSignatureVariants(argumentNames, argumentNameToOverloads));
    }

    // Make the method signature order deterministic, which helps with unit testing and per-version
    // diffs.
    List> sortedMethodSignatures =
        signatures.stream()
            .sorted(
                (s1, s2) -> {
                  // Sort by number of arguments first.
                  if (s1.size() != s2.size()) {
                    return s1.size() - s2.size();
                  }
                  // Then by MethodSignature properties.
                  for (int i = 0; i < s1.size(); i++) {
                    int compareVal = s1.get(i).compareTo(s2.get(i));
                    if (compareVal != 0) {
                      return compareVal;
                    }
                  }
                  return 0;
                })
            .collect(Collectors.toList());

    return sortedMethodSignatures;
  }

  @VisibleForTesting
  static List> flattenMethodSignatureVariants(
      List argumentNames, Map> argumentNameToOverloads) {
    Preconditions.checkState(
        argumentNames.size() == argumentNameToOverloads.size(),
        String.format(
            "Cardinality of argument names %s do not match that of overloaded types %s",
            argumentNames, argumentNameToOverloads));
    for (String name : argumentNames) {
      Preconditions.checkNotNull(
          argumentNameToOverloads.get(name),
          String.format("No corresponding overload types found for argument %s", name));
    }
    return flattenMethodSignatureVariants(argumentNames, argumentNameToOverloads, 0);
  }

  private static List> flattenMethodSignatureVariants(
      List argumentNames,
      Map> argumentNameToOverloads,
      int depth) {
    List> methodArgs = new ArrayList<>();
    if (depth >= argumentNames.size() - 1) {
      for (MethodArgument methodArg : argumentNameToOverloads.get(argumentNames.get(depth))) {
        methodArgs.add(Lists.newArrayList(methodArg));
      }
      return methodArgs;
    }

    List> subsequentArgs =
        flattenMethodSignatureVariants(argumentNames, argumentNameToOverloads, depth + 1);
    for (MethodArgument methodArg : argumentNameToOverloads.get(argumentNames.get(depth))) {
      for (List subsequentArg : subsequentArgs) {
        // Use a new list to avoid appending all subsequent elements (in upcoming loop iterations)
        // to the same list.
        List appendedArgs = new ArrayList<>(subsequentArg);
        appendedArgs.add(0, methodArg);
        methodArgs.add(appendedArgs);
      }
    }
    return methodArgs;
  }

  private static Map parseTypeFromArgumentName(
      String argumentName,
      String servicePackage,
      Message inputMessage,
      Map messageTypes,
      Map resourceNames,
      Map patternsToResourceNames,
      List argumentFieldPathAcc,
      Set outputArgResourceNames) {

    Map typeToField = new HashMap<>();
    int dotIndex = argumentName.indexOf(DOT);
    if (dotIndex < 1) {
      Field field = inputMessage.fieldMap().get(argumentName);
      Preconditions.checkNotNull(
          field,
          String.format(
              "Field %s not found from input message %s values %s",
              argumentName, inputMessage.name(), inputMessage.fieldMap().keySet()));
      if (!field.hasResourceReference()) {
        typeToField.put(field.type(), field);
        return typeToField;
      }

      // Parse the resource name tyeps.
      List resourceNameArgs =
          ResourceReferenceParser.parseResourceNames(
              field.resourceReference(),
              servicePackage,
              field.description(),
              resourceNames,
              patternsToResourceNames);
      outputArgResourceNames.addAll(resourceNameArgs);
      typeToField.put(TypeNode.STRING, field);
      typeToField.putAll(
          resourceNameArgs.stream()
              .collect(
                  Collectors.toMap(
                      r -> r.type(),
                      r ->
                          // Contruct a new field using the parent resource.
                          field
                              .toBuilder()
                              .setResourceReference(
                                  ResourceReference.withType(r.resourceTypeString()))
                              .build())));
      // Only resource name helpers should have more than one entry.
      if (typeToField.size() > 1) {
        typeToField.entrySet().stream()
            .forEach(
                e -> {
                  // Skip string-only variants or ResourceName generics.
                  if (e.getKey().equals(TypeNode.STRING)
                      || e.getKey().reference().name().equals("ResourceName")) {
                    return;
                  }
                  String resourceJavaTypeName = e.getKey().reference().name();
                  String resourceTypeName = e.getValue().resourceReference().resourceTypeString();
                  int indexOfSlash = resourceTypeName.indexOf("/");
                  // We assume that the corresponding Java resource name helper type (i.e. the key)
                  // ends in *Name. Check that it matches the expeced resource name type.
                  Preconditions.checkState(
                      resourceJavaTypeName
                          .substring(0, resourceJavaTypeName.length() - 4)
                          .equals(resourceTypeName.substring(indexOfSlash + 1)),
                      String.format(
                          "Resource Java type %s does not correspond to proto type %s",
                          resourceJavaTypeName, resourceTypeName));
                });
      }
      return typeToField;
    }

    Preconditions.checkState(
        dotIndex < argumentName.length() - 1,
        String.format(
            "Invalid argument name found: dot cannot be at the end of name %s", argumentName));
    String firstFieldName = argumentName.substring(0, dotIndex);
    String remainingArgumentName = argumentName.substring(dotIndex + 1);

    // Must be a sub-message for a type's subfield to be valid.
    Field firstField = inputMessage.fieldMap().get(firstFieldName);

    // Validate the field into which we're descending.
    Preconditions.checkState(
        !firstField.isRepeated(),
        String.format("Cannot descend into repeated field %s", firstField.name()));

    TypeNode firstFieldType = firstField.type();
    Preconditions.checkState(
        TypeNode.isReferenceType(firstFieldType) && !firstFieldType.equals(TypeNode.STRING),
        String.format("Field reference on %s cannot be a primitive type", firstFieldName));

    String firstFieldTypeName = firstFieldType.reference().fullName();
    Message firstFieldMessage = messageTypes.get(firstFieldTypeName);
    Preconditions.checkNotNull(
        firstFieldMessage,
        String.format(
            "Message type %s for field reference %s invalid", firstFieldTypeName, firstFieldName));

    argumentFieldPathAcc.add(firstField);
    return parseTypeFromArgumentName(
        remainingArgumentName,
        servicePackage,
        firstFieldMessage,
        messageTypes,
        resourceNames,
        patternsToResourceNames,
        argumentFieldPathAcc,
        outputArgResourceNames);
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy