Skip to content

Commit ff2c485

Browse files
zhumin8lqiu96
andauthored
fix: Generator to skip generation for empty services. (#3051)
fixes #2750 Skip parsing a service if no RPCs found for. In the scenario that only one service with no RPCs or all services have no RPCs, falls back to #2460 This pr also reverts a change brought by #985, and removes the relevant tests. For more context, this has been discussed [here](#2750 (comment)). --------- Co-authored-by: Lawrence Qiu <lawrenceqiu@google.com>
1 parent 5af127b commit ff2c485

File tree

8 files changed

+173
-177
lines changed

8 files changed

+173
-177
lines changed

gapic-generator-java/src/main/java/com/google/api/generator/gapic/composer/samplecode/ServiceClientHeaderSampleComposer.java

-4
Original file line numberDiff line numberDiff line change
@@ -52,10 +52,6 @@ public static Sample composeClassHeaderSample(
5252
TypeNode clientType,
5353
Map<String, ResourceName> resourceNames,
5454
Map<String, Message> messageTypes) {
55-
if (service.methods().isEmpty()) {
56-
return ServiceClientMethodSampleComposer.composeEmptyServiceSample(clientType, service);
57-
}
58-
5955
// Use the first pure unary RPC method's sample code as showcase, if no such method exists, use
6056
// the first method in the service's methods list.
6157
Method method =

gapic-generator-java/src/main/java/com/google/api/generator/gapic/protoparser/Parser.java

+11
Original file line numberDiff line numberDiff line change
@@ -435,6 +435,17 @@ public static List<Service> parseService(
435435
Transport transport) {
436436

437437
return fileDescriptor.getServices().stream()
438+
.filter(
439+
serviceDescriptor -> {
440+
List<MethodDescriptor> methodsList = serviceDescriptor.getMethods();
441+
if (methodsList.isEmpty()) {
442+
LOGGER.warning(
443+
String.format(
444+
"Service %s has no RPC methods and will not be generated",
445+
serviceDescriptor.getName()));
446+
}
447+
return !methodsList.isEmpty();
448+
})
438449
.map(
439450
s -> {
440451
// Workaround for a missing default_host and oauth_scopes annotation from a service

gapic-generator-java/src/test/java/com/google/api/generator/gapic/composer/grpcrest/ServiceClientClassComposerTest.java

-14
Original file line numberDiff line numberDiff line change
@@ -40,20 +40,6 @@ void generateServiceClasses() {
4040
Assert.assertCodeEquals(goldenFilePath, visitor.write());
4141
}
4242

43-
@Test
44-
void generateServiceClassesEmpty() {
45-
GapicContext context = GrpcRestTestProtoLoader.instance().parseShowcaseEcho();
46-
Service echoProtoService = context.services().get(1);
47-
GapicClass clazz = ServiceClientClassComposer.instance().generate(context, echoProtoService);
48-
49-
JavaWriterVisitor visitor = new JavaWriterVisitor();
50-
clazz.classDefinition().accept(visitor);
51-
GoldenFileWriter.saveCodegenToFile(this.getClass(), "EchoEmpty.golden", visitor.write());
52-
Path goldenFilePath =
53-
Paths.get(GoldenFileWriter.getGoldenDir(this.getClass()), "EchoEmpty.golden");
54-
Assert.assertCodeEquals(goldenFilePath, visitor.write());
55-
}
56-
5743
@Test
5844
void generateServiceClassesWicked() {
5945
GapicContext context = GrpcRestTestProtoLoader.instance().parseShowcaseWicked();

gapic-generator-java/src/test/java/com/google/api/generator/gapic/composer/grpcrest/goldens/EchoEmpty.golden

-154
This file was deleted.

gapic-generator-java/src/test/java/com/google/api/generator/gapic/protoparser/ParserTest.java

+13
Original file line numberDiff line numberDiff line change
@@ -692,6 +692,19 @@ void testServiceWithoutApiVersionParsed() {
692692
assertNull(parsedBookshopService.apiVersion());
693693
}
694694

695+
@Test
696+
void parseServiceWithNoMethodsTest() {
697+
FileDescriptor fileDescriptor =
698+
com.google.api.service.without.methods.test.ServiceWithNoMethodsOuterClass.getDescriptor();
699+
Map<String, Message> messageTypes = Parser.parseMessages(fileDescriptor);
700+
Map<String, ResourceName> resourceNames = Parser.parseResourceNames(fileDescriptor);
701+
List<com.google.api.generator.gapic.model.Service> services =
702+
Parser.parseService(
703+
fileDescriptor, messageTypes, resourceNames, Optional.empty(), new HashSet<>());
704+
assertEquals(1, services.size());
705+
assertEquals("EchoWithMethods", services.get(0).overriddenName());
706+
}
707+
695708
private void assertMethodArgumentEquals(
696709
String name, TypeNode type, List<TypeNode> nestedFields, MethodArgument argument) {
697710
assertEquals(name, argument.name());

gapic-generator-java/src/test/proto/echo_grpcrest.proto

-5
Original file line numberDiff line numberDiff line change
@@ -163,11 +163,6 @@ service Echo {
163163
}
164164
}
165165

166-
// Generator should not fail when encounter a service without methods
167-
service EchoEmpy {
168-
option (google.api.default_host) = "localhost:7469";
169-
}
170-
171166
// A severity enum used to test enum capabilities in GAPIC surfaces
172167
enum Severity {
173168
UNNECESSARY = 0;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
// Copyright 2018 Google LLC
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
syntax = "proto3";
16+
17+
import "google/api/annotations.proto";
18+
import "google/api/client.proto";
19+
import "google/api/field_behavior.proto";
20+
import "google/api/field_info.proto";
21+
import "google/api/resource.proto";
22+
import "google/longrunning/operations.proto";
23+
import "google/protobuf/duration.proto";
24+
import "google/protobuf/timestamp.proto";
25+
import "google/rpc/status.proto";
26+
27+
package google.api.service.without.methods.test.v1;
28+
29+
option java_package = "com.google.api.service.without.methods.test";
30+
option java_multiple_files = true;
31+
option java_outer_classname = "ServiceWithNoMethodsOuterClass";
32+
33+
option (google.api.resource_definition) = {
34+
type: "showcase.googleapis.com/AnythingGoes"
35+
pattern: "*"
36+
};
37+
// This proto is used to test scenarios where a service does not have any rpc methods
38+
39+
// This service is used as control group when it is not empty.
40+
service EchoWithMethods {
41+
// This service is meant to only run locally on the port 7469 (keypad digits
42+
// for "show").
43+
option (google.api.default_host) = "localhost:7469";
44+
option (google.api.oauth_scopes) =
45+
"https://www.googleapis.com/auth/cloud-platform";
46+
47+
// This method simply echos the request. This method is showcases unary rpcs.
48+
rpc EchoWithMethod(EchoRequest) returns (EchoResponse) {
49+
option (google.api.http) = {
50+
post: "/v1beta1/echo:echo"
51+
body: "*"
52+
};
53+
option (google.api.method_signature) = "content";
54+
option (google.api.method_signature) = "error";
55+
option (google.api.method_signature) = "content,severity";
56+
option (google.api.method_signature) = "name";
57+
option (google.api.method_signature) = "parent";
58+
option (google.api.method_signature) = "";
59+
}
60+
}
61+
62+
// This service is to test when no method specified.
63+
service EchoWithoutMethods {
64+
// This service is meant to only run locally on the port 7469 (keypad digits
65+
// for "show").
66+
option (google.api.default_host) = "localhost:7469";
67+
option (google.api.oauth_scopes) =
68+
"https://www.googleapis.com/auth/cloud-platform";
69+
}
70+
71+
// A severity enum used to test enum capabilities in GAPIC surfaces
72+
enum Severity {
73+
UNNECESSARY = 0;
74+
NECESSARY = 1;
75+
URGENT = 2;
76+
CRITICAL = 3;
77+
}
78+
79+
message Foobar {
80+
option (google.api.resource) = {
81+
type: "showcase.googleapis.com/Foobar"
82+
pattern: "projects/{project}/foobars/{foobar}"
83+
pattern: "projects/{project}/chocolate/variants/{variant}/foobars/{foobar}"
84+
pattern: "foobars/{foobar}"
85+
pattern: "bar_foos/{bar_foo}/foobars/{foobar}"
86+
pattern: "*"
87+
};
88+
89+
string name = 1;
90+
string info = 2;
91+
}
92+
93+
// The request message used for the Echo, Collect and Chat methods.
94+
// If content or opt are set in this message then the request will succeed.
95+
// If status is set in this message
96+
// then the status will be returned as an error.
97+
message EchoRequest {
98+
string name = 5 [
99+
(google.api.resource_reference).type = "showcase.googleapis.com/Foobar",
100+
(google.api.field_behavior) = REQUIRED
101+
];
102+
103+
string parent = 6 [
104+
(google.api.resource_reference).child_type =
105+
"showcase.googleapis.com/AnythingGoes",
106+
(google.api.field_behavior) = REQUIRED
107+
];
108+
109+
oneof response {
110+
// The content to be echoed by the server.
111+
string content = 1;
112+
113+
// The error to be thrown by the server.
114+
google.rpc.Status error = 2;
115+
}
116+
117+
// The severity to be echoed by the server.
118+
Severity severity = 3;
119+
120+
Foobar foobar = 4;
121+
}
122+
123+
// The response message for the Echo methods.
124+
message EchoResponse {
125+
// The content specified in the request.
126+
string content = 1;
127+
128+
// The severity specified in the request.
129+
Severity severity = 2;
130+
}
131+

showcase/gapic-showcase/src/main/resources/META-INF/native-image/com.google.showcase.v1beta1/reflect-config.json

+18
Original file line numberDiff line numberDiff line change
@@ -440,6 +440,24 @@
440440
"allDeclaredClasses": true,
441441
"allPublicClasses": true
442442
},
443+
{
444+
"name": "com.google.api.TypeReference",
445+
"queryAllDeclaredConstructors": true,
446+
"queryAllPublicConstructors": true,
447+
"queryAllDeclaredMethods": true,
448+
"allPublicMethods": true,
449+
"allDeclaredClasses": true,
450+
"allPublicClasses": true
451+
},
452+
{
453+
"name": "com.google.api.TypeReference$Builder",
454+
"queryAllDeclaredConstructors": true,
455+
"queryAllPublicConstructors": true,
456+
"queryAllDeclaredMethods": true,
457+
"allPublicMethods": true,
458+
"allDeclaredClasses": true,
459+
"allPublicClasses": true
460+
},
443461
{
444462
"name": "com.google.cloud.location.GetLocationRequest",
445463
"queryAllDeclaredConstructors": true,

0 commit comments

Comments
 (0)