Skip to content

Commit c81cd0f

Browse files
authoredNov 16, 2023
feat: update javadocs for Client classes to include table of methods (#2114)
* feat: update javadocs for Client classes to include table of methods and method variants * fix lint * update showcase goldens * remove unnecessary newlines and encapsulate method description within <p> tags * fix lint * update showcase goldens * test remove cache for golden test * update integration goldens * adding back in cache * update integration goldens * test change to ci * update ci comment * update to use linkedhashmap for consistent ordering * update showcase goldens * fix lint * refactor * refactor * fix lint and showcase goldens * update showcase goldens * include all parameters, not just first one * include primitive types and update indentation
1 parent 741e40c commit c81cd0f

File tree

38 files changed

+6611
-439
lines changed

38 files changed

+6611
-439
lines changed
 

‎.github/workflows/ci.yaml

+2-2
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ jobs:
5454
runs-on: ubuntu-22.04
5555
steps:
5656
- uses: actions/checkout@v3
57-
# Java 8 tests uses JDK 11 to compile and JDK 8 to run tests.
57+
# Java 8 tests uses JDK 17 to compile and JDK 8 to run tests.
5858
- uses: actions/setup-java@v3
5959
with:
6060
java-version: 8
@@ -69,7 +69,7 @@ jobs:
6969
shell: bash
7070
run: |
7171
set -x
72-
export JAVA_HOME=$JAVA11_HOME
72+
export JAVA_HOME=$JAVA_HOME
7373
export PATH=${JAVA_HOME}/bin:$PATH
7474
# Maven surefire plugin lets us to specify the JVM when running tests via
7575
# the "jvm" system property.

‎gapic-generator-java/DEVELOPMENT.md

+9
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,8 @@ than the "test" phase.
5050
To run integration test for gapic-generator-java, run this Bazel command in the
5151
root of the repository (where you have WORKSPACE file for Bazel.)
5252

53+
*Note* Make sure you run `mvn clean install` to gather any changes you have made before updating the integration tests.
54+
5355
```sh
5456
# In the repository root directory
5557
bazelisk test //... # integration tests
@@ -73,6 +75,13 @@ bazelisk test //... # integration tests
7375
bazelisk run //test/integration:update_redis
7476
```
7577

78+
- To update all integration tests you can use this command:
79+
80+
```sh
81+
# In the repository root directory
82+
bazelisk run //test/integration:update_asset && bazelisk run //test/integration:update_credentials && bazelisk run //test/integration:update_iam && bazelisk run //test/integration:update_kms && bazelisk run //test/integration:update_pubsub && bazelisk run //test/integration:update_logging && bazelisk run //test/integration:update_redis && bazelisk run //test/integration:update_storage && bazelisk run //test/integration:update_library && bazelisk run //test/integration:update_compute && bazelisk run //test/integration:update_bigtable && bazelisk run //test/integration:update_apigeeconnect
83+
```
84+
7685
## Running the Plugin under googleapis with local gapic-generator-java
7786

7887
For running the Plugin with showcase protos and local gapic-generator-java, see

‎gapic-generator-java/src/main/java/com/google/api/generator/gapic/composer/comment/ServiceClientCommentComposer.java

+101-17
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@
1414

1515
package com.google.api.generator.gapic.composer.comment;
1616

17+
import static java.util.stream.Collectors.toList;
18+
1719
import com.google.api.generator.engine.ast.CommentStatement;
1820
import com.google.api.generator.engine.ast.JavaDocComment;
1921
import com.google.api.generator.engine.ast.TypeNode;
@@ -27,6 +29,7 @@
2729
import java.util.Arrays;
2830
import java.util.Collections;
2931
import java.util.List;
32+
import java.util.Map;
3033
import java.util.Optional;
3134

3235
public class ServiceClientCommentComposer {
@@ -39,9 +42,6 @@ public class ServiceClientCommentComposer {
3942
private static final String SERVICE_DESCRIPTION_INTRO_STRING =
4043
"This class provides the ability to make remote calls to the backing service through method "
4144
+ "calls that map to API methods. Sample code to get started:";
42-
private static final String SERVICE_DESCRIPTION_SURFACE_SUMMARY_STRING =
43-
"The surface of this class includes several types of Java methods for each of the API's "
44-
+ "methods:";
4545
private static final String SERVICE_DESCRIPTION_SURFACE_CODA_STRING =
4646
"See the individual methods for example code.";
4747
private static final String SERVICE_DESCRIPTION_RESOURCE_NAMES_FORMATTING_STRING =
@@ -61,18 +61,6 @@ public class ServiceClientCommentComposer {
6161

6262
private static final String METHOD_DESCRIPTION_SAMPLE_CODE_SUMMARY_STRING = "Sample code:";
6363

64-
private static final List<String> SERVICE_DESCRIPTION_SURFACE_DESCRIPTION =
65-
Arrays.asList(
66-
"A \"flattened\" method. With this type of method, the fields of the request type have"
67-
+ " been converted into function parameters. It may be the case that not all fields"
68-
+ " are available as parameters, and not every API method will have a flattened"
69-
+ " method entry point.",
70-
"A \"request object\" method. This type of method only takes one parameter, a request"
71-
+ " object, which must be constructed before the call. Not every API method will"
72-
+ " have a request object method.",
73-
"A \"callable\" method. This type of method takes no parameters and returns an immutable "
74-
+ "API callable object, which can be used to initiate calls to the service.");
75-
7664
// Patterns.
7765
private static final String CREATE_METHOD_STUB_ARG_PATTERN =
7866
"Constructs an instance of %s, using the given stub for making calls. This is for"
@@ -109,6 +97,7 @@ public class ServiceClientCommentComposer {
10997
+ " operation returned by another API method call.");
11098

11199
public static List<CommentStatement> createClassHeaderComments(
100+
Map<String, List<String>> methodVariantsForClientHeader,
112101
Service service,
113102
String classMethodSampleCode,
114103
String credentialsSampleCode,
@@ -132,8 +121,14 @@ public static List<CommentStatement> createClassHeaderComments(
132121
classHeaderJavadocBuilder.addParagraph(
133122
String.format(
134123
SERVICE_DESCRIPTION_CLOSE_PATTERN, ClassNames.getServiceClientClassName(service)));
135-
classHeaderJavadocBuilder.addParagraph(SERVICE_DESCRIPTION_SURFACE_SUMMARY_STRING);
136-
classHeaderJavadocBuilder.addOrderedList(SERVICE_DESCRIPTION_SURFACE_DESCRIPTION);
124+
125+
// Build the map of methods and descriptions to create the table in Client Overviews
126+
List<MethodAndVariants> methodAndVariantsList =
127+
service.methods().stream()
128+
.map((Method method) -> createMethodAndVariants(method, methodVariantsForClientHeader))
129+
.collect(toList());
130+
131+
classHeaderJavadocBuilder.addUnescapedComment(createTableOfMethods(methodAndVariantsList));
137132
classHeaderJavadocBuilder.addParagraph(SERVICE_DESCRIPTION_SURFACE_CODA_STRING);
138133

139134
// Formatting resource names.
@@ -215,6 +210,95 @@ public static List<CommentStatement> createRpcMethodHeaderComment(
215210
return comments;
216211
}
217212

213+
private static MethodAndVariants createMethodAndVariants(
214+
Method method, Map<String, List<String>> methodVariantsForClientHeader) {
215+
String name = method.name();
216+
String description = method.description();
217+
if (description == null) description = "";
218+
return new MethodAndVariants(name, description, methodVariantsForClientHeader.get(name));
219+
}
220+
221+
private static String createTableOfMethods(List<MethodAndVariants> methodAndVariantsList) {
222+
String FLATTENED_METHODS =
223+
"<p>\"Flattened\" method variants have converted the fields of the request object into function parameters to enable multiple ways to call the same method.</p>\n";
224+
String REQUEST_OBJECT_METHODS =
225+
"<p>Request object method variants only take one parameter, a request object, which must be constructed before the call.</p>\n";
226+
String CALLABLE_METHODS =
227+
"<p>Callable method variants take no parameters and return an immutable API callable object, which can be used to initiate calls to the service.</p>\n";
228+
String ASYNC_METHODS =
229+
"<p>Methods that return long-running operations have \"Async\" method variants that return `OperationFuture`, which is used to track polling of the service.</p>\n";
230+
231+
StringBuilder tableBuilder = new StringBuilder();
232+
tableBuilder
233+
.append("<table>\n")
234+
.append(" <tr>\n")
235+
.append(" <th>Method</th>\n")
236+
.append(" <th>Description</th>\n")
237+
.append(" <th>Method Variants</th>\n");
238+
for (MethodAndVariants method : methodAndVariantsList) {
239+
tableBuilder
240+
.append(" <tr>\n")
241+
.append(" <td>")
242+
.append(method.method)
243+
.append("</td>\n")
244+
.append(" <td>")
245+
.append("<p>" + method.description + "</p>")
246+
.append("</td>\n")
247+
.append(" <td>\n");
248+
generateUnorderedListMethodVariants(
249+
tableBuilder, REQUEST_OBJECT_METHODS, method.requestObjectVariants);
250+
generateUnorderedListMethodVariants(
251+
tableBuilder, FLATTENED_METHODS, method.flattenedVariants);
252+
generateUnorderedListMethodVariants(tableBuilder, ASYNC_METHODS, method.asyncVariants);
253+
generateUnorderedListMethodVariants(tableBuilder, CALLABLE_METHODS, method.callableVariants);
254+
tableBuilder.append(" </td>\n").append(" </tr>\n");
255+
}
256+
tableBuilder.append(" </tr>\n").append(" </table>\n");
257+
return tableBuilder.toString();
258+
}
259+
260+
private static void generateUnorderedListMethodVariants(
261+
StringBuilder tableBuilder, String methodType, List<String> methodVariants) {
262+
if (!methodVariants.isEmpty()) {
263+
tableBuilder
264+
.append(" " + methodType + " ")
265+
.append("<ul>\n")
266+
.append(" <li>")
267+
.append(String.join("\n <li>", methodVariants))
268+
.append("\n")
269+
.append(" </ul>")
270+
.append("\n");
271+
}
272+
}
273+
274+
private static class MethodAndVariants {
275+
private final String method;
276+
// Description may be empty. It comes from the proto comments above the method. If it is empty,
277+
// then nothing will be displayed.
278+
private final String description;
279+
280+
private final List<String> flattenedVariants;
281+
private final List<String> requestObjectVariants;
282+
private final List<String> callableVariants;
283+
private final List<String> asyncVariants;
284+
285+
private MethodAndVariants(String method, String description, List<String> methodVariants) {
286+
this.method = method;
287+
this.description = description;
288+
requestObjectVariants =
289+
methodVariants.stream().filter(s -> s.contains("request")).collect(toList());
290+
// Flattened method variants do not have a suffix, so the easiest way to identify them is by
291+
// removing all other method variant types.
292+
methodVariants.removeAll(requestObjectVariants);
293+
callableVariants =
294+
methodVariants.stream().filter(s -> s.contains("Callable")).collect(toList());
295+
methodVariants.removeAll(callableVariants);
296+
asyncVariants = methodVariants.stream().filter(s -> s.contains("Async")).collect(toList());
297+
methodVariants.removeAll(asyncVariants);
298+
flattenedVariants = methodVariants;
299+
}
300+
}
301+
218302
public static List<CommentStatement> createRpcMethodHeaderComment(
219303
Method method, Optional<String> sampleCodeOpt) {
220304
return createRpcMethodHeaderComment(method, Collections.emptyList(), sampleCodeOpt);

‎gapic-generator-java/src/main/java/com/google/api/generator/gapic/composer/common/AbstractServiceClientClassComposer.java

+71-7
Original file line numberDiff line numberDiff line change
@@ -139,11 +139,10 @@ public GapicClass generate(GapicContext context, Service service) {
139139

140140
List<Sample> samples = new ArrayList<>();
141141
Map<String, List<String>> grpcRpcsToJavaMethodNames = new HashMap<>();
142+
Map<String, List<String>> methodVariantsForClientHeader = new HashMap<>();
142143

143144
ClassDefinition classDef =
144145
ClassDefinition.builder()
145-
.setHeaderCommentStatements(
146-
createClassHeaderComments(service, typeStore, resourceNames, messageTypes, samples))
147146
.setPackageString(pakkage)
148147
.setAnnotations(createClassAnnotations(service, typeStore))
149148
.setScope(ScopeNode.PUBLIC)
@@ -158,8 +157,17 @@ public GapicClass generate(GapicContext context, Service service) {
158157
resourceNames,
159158
hasLroClient,
160159
grpcRpcsToJavaMethodNames,
160+
methodVariantsForClientHeader,
161161
samples))
162162
.setNestedClasses(createNestedPagingClasses(service, messageTypes, typeStore))
163+
.setHeaderCommentStatements(
164+
createClassHeaderComments(
165+
methodVariantsForClientHeader,
166+
service,
167+
typeStore,
168+
resourceNames,
169+
messageTypes,
170+
samples))
163171
.build();
164172

165173
updateGapicMetadata(context, service, className, grpcRpcsToJavaMethodNames);
@@ -189,6 +197,7 @@ private static List<TypeNode> createClassImplements(TypeStore typeStore) {
189197
}
190198

191199
protected List<CommentStatement> createClassHeaderComments(
200+
Map<String, List<String>> methodVariantsForClientHeader,
192201
Service service,
193202
TypeStore typeStore,
194203
Map<String, ResourceName> resourceNames,
@@ -207,6 +216,7 @@ protected List<CommentStatement> createClassHeaderComments(
207216
clientType, settingsType, service);
208217
samples.addAll(Arrays.asList(classMethodSampleCode, credentialsSampleCode, endpointSampleCode));
209218
return ServiceClientCommentComposer.createClassHeaderComments(
219+
methodVariantsForClientHeader,
210220
service,
211221
SampleCodeWriter.writeInlineSample(classMethodSampleCode.body()),
212222
SampleCodeWriter.writeInlineSample(credentialsSampleCode.body()),
@@ -223,14 +233,21 @@ private List<MethodDefinition> createClassMethods(
223233
Map<String, ResourceName> resourceNames,
224234
boolean hasLroClient,
225235
Map<String, List<String>> grpcRpcToJavaMethodMetadata,
236+
Map<String, List<String>> methodVariantsForClientHeader,
226237
List<Sample> samples) {
227238
List<MethodDefinition> methods = new ArrayList<>();
228239
methods.addAll(createStaticCreatorMethods(service, typeStore));
229240
methods.addAll(createConstructorMethods(service, typeStore, hasLroClient));
230241
methods.addAll(createGetterMethods(service, typeStore, hasLroClient));
231242
methods.addAll(
232243
createServiceMethods(
233-
service, messageTypes, typeStore, resourceNames, grpcRpcToJavaMethodMetadata, samples));
244+
service,
245+
messageTypes,
246+
typeStore,
247+
resourceNames,
248+
grpcRpcToJavaMethodMetadata,
249+
methodVariantsForClientHeader,
250+
samples));
234251
methods.addAll(createBackgroundResourceMethods(service, typeStore));
235252
return methods;
236253
}
@@ -561,18 +578,53 @@ private List<MethodDefinition> createGetterMethods(
561578
.collect(Collectors.toList());
562579
}
563580

581+
private static String getJavaMethod(MethodDefinition m) {
582+
StringBuilder methodSignature = new StringBuilder();
583+
584+
// Method name
585+
methodSignature.append(m.methodIdentifier().name()).append("(");
586+
587+
// Iterate through and add all parameters
588+
List<Variable> parameters =
589+
m.arguments().stream().map(VariableExpr::variable).collect(Collectors.toList());
590+
591+
// If reference is empty, that means the parameter is a non-Object type. Therefore, use the
592+
// typeKind directly.
593+
for (int i = 0; i < parameters.size(); i++) {
594+
Variable param = parameters.get(i);
595+
String paramType =
596+
param.type().reference() != null
597+
? param.type().reference().name() + " "
598+
: param.type().typeKind().name().toLowerCase() + " ";
599+
String paramName = param.identifier().name();
600+
601+
methodSignature.append(paramType).append(paramName);
602+
603+
// Add a comma if there are more parameters
604+
if (i < parameters.size() - 1) {
605+
methodSignature.append(", ");
606+
}
607+
}
608+
609+
methodSignature.append(")");
610+
611+
return methodSignature.toString();
612+
}
613+
564614
private static List<MethodDefinition> createServiceMethods(
565615
Service service,
566616
Map<String, Message> messageTypes,
567617
TypeStore typeStore,
568618
Map<String, ResourceName> resourceNames,
569619
Map<String, List<String>> grpcRpcToJavaMethodMetadata,
620+
Map<String, List<String>> methodVariantsForClientHeader,
570621
List<Sample> samples) {
571622
List<MethodDefinition> javaMethods = new ArrayList<>();
572623
Function<MethodDefinition, String> javaMethodNameFn = m -> m.methodIdentifier().name();
573624
for (Method method : service.methods()) {
574625
if (!grpcRpcToJavaMethodMetadata.containsKey(method.name())) {
575626
grpcRpcToJavaMethodMetadata.put(method.name(), new ArrayList<>());
627+
methodVariantsForClientHeader.put(method.name(), new ArrayList<>());
576628
}
577629
if (method.stream().equals(Stream.NONE)) {
578630
List<MethodDefinition> generatedMethods =
@@ -592,6 +644,14 @@ private static List<MethodDefinition> createServiceMethods(
592644
generatedMethods.stream()
593645
.map(m -> javaMethodNameFn.apply(m))
594646
.collect(Collectors.toList()));
647+
648+
// Collect data for Client header
649+
methodVariantsForClientHeader
650+
.get(method.name())
651+
.addAll(
652+
generatedMethods.stream()
653+
.map(AbstractServiceClientClassComposer::getJavaMethod)
654+
.collect(Collectors.toList()));
595655
javaMethods.addAll(generatedMethods);
596656

597657
MethodDefinition generatedMethod =
@@ -604,33 +664,37 @@ private static List<MethodDefinition> createServiceMethods(
604664
samples,
605665
service);
606666

607-
// Collect data for gapic_metadata.json.
667+
// Collect data for gapic_metadata.json and client header.
608668
grpcRpcToJavaMethodMetadata.get(method.name()).add(javaMethodNameFn.apply(generatedMethod));
669+
methodVariantsForClientHeader.get(method.name()).add(getJavaMethod(generatedMethod));
609670
javaMethods.add(generatedMethod);
610671
}
611672
if (method.hasLro()) {
612673
MethodDefinition generatedMethod =
613674
createLroCallableMethod(
614675
service, method, typeStore, messageTypes, resourceNames, samples);
615676

616-
// Collect data for gapic_metadata.json.
677+
// Collect data for gapic_metadata.json and client header.
617678
grpcRpcToJavaMethodMetadata.get(method.name()).add(javaMethodNameFn.apply(generatedMethod));
679+
methodVariantsForClientHeader.get(method.name()).add(getJavaMethod(generatedMethod));
618680
javaMethods.add(generatedMethod);
619681
}
620682
if (method.isPaged()) {
621683
MethodDefinition generatedMethod =
622684
createPagedCallableMethod(
623685
service, method, typeStore, messageTypes, resourceNames, samples);
624686

625-
// Collect data for gapic_metadata.json.
687+
// Collect data for gapic_metadata.json and client header.
626688
grpcRpcToJavaMethodMetadata.get(method.name()).add(javaMethodNameFn.apply(generatedMethod));
689+
methodVariantsForClientHeader.get(method.name()).add(getJavaMethod(generatedMethod));
627690
javaMethods.add(generatedMethod);
628691
}
629692
MethodDefinition generatedMethod =
630693
createCallableMethod(service, method, typeStore, messageTypes, resourceNames, samples);
631694

632-
// Collect data for the gapic_metadata.json file.
695+
// Collect data for the gapic_metadata.json file and client header.
633696
grpcRpcToJavaMethodMetadata.get(method.name()).add(javaMethodNameFn.apply(generatedMethod));
697+
methodVariantsForClientHeader.get(method.name()).add(getJavaMethod(generatedMethod));
634698
javaMethods.add(generatedMethod);
635699
}
636700
return javaMethods;

0 commit comments

Comments
 (0)