Skip to content

Commit

Permalink
Merge pull request #2667 from murgatroid99/grpc-js_round_robin_idle_fix
Browse files Browse the repository at this point in the history
grpc-js: round_robin: always have children reconnect immediately
  • Loading branch information
murgatroid99 authored Feb 15, 2024
2 parents 1b753af + 429a66d commit 513a61a
Show file tree
Hide file tree
Showing 5 changed files with 45 additions and 5 deletions.
7 changes: 3 additions & 4 deletions packages/grpc-js-xds/test/backend.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@
*
*/

import { loadPackageDefinition, sendUnaryData, Server, ServerCredentials, ServerUnaryCall, UntypedServiceImplementation } from "@grpc/grpc-js";
import { loadPackageDefinition, sendUnaryData, Server, ServerCredentials, ServerOptions, ServerUnaryCall, UntypedServiceImplementation } from "@grpc/grpc-js";
import { loadSync } from "@grpc/proto-loader";
import { ProtoGrpcType } from "./generated/echo";
import { EchoRequest__Output } from "./generated/grpc/testing/EchoRequest";
Expand Down Expand Up @@ -43,7 +43,7 @@ export class Backend {
private receivedCallCount = 0;
private callListeners: (() => void)[] = [];
private port: number | null = null;
constructor() {
constructor(private serverOptions?: ServerOptions) {
}
Echo(call: ServerUnaryCall<EchoRequest__Output, EchoResponse>, callback: sendUnaryData<EchoResponse>) {
// call.request.params is currently ignored
Expand Down Expand Up @@ -76,13 +76,12 @@ export class Backend {
if (this.server) {
throw new Error("Backend already running");
}
this.server = new Server();
this.server = new Server(this.serverOptions);
this.server.addService(loadedProtos.grpc.testing.EchoTestService.service, this as unknown as UntypedServiceImplementation);
const boundPort = this.port ?? 0;
this.server.bindAsync(`localhost:${boundPort}`, ServerCredentials.createInsecure(), (error, port) => {
if (!error) {
this.port = port;
this.server!.start();
}
callback(error, port);
})
Expand Down
28 changes: 28 additions & 0 deletions packages/grpc-js-xds/test/test-core.ts
Original file line number Diff line number Diff line change
Expand Up @@ -91,4 +91,32 @@ describe('core xDS functionality', () => {
});
}, reason => done(reason));
});
it('should handle connections aging out', function(done) {
this.timeout(5000);
const cluster = new FakeEdsCluster('cluster1', 'endpoint1', [{backends: [new Backend({'grpc.max_connection_age_ms': 1000})], locality:{region: 'region1'}}]);
const routeGroup = new FakeRouteGroup('listener1', 'route1', [{cluster: cluster}]);
routeGroup.startAllBackends().then(() => {
xdsServer.setEdsResource(cluster.getEndpointConfig());
xdsServer.setCdsResource(cluster.getClusterConfig());
xdsServer.setRdsResource(routeGroup.getRouteConfiguration());
xdsServer.setLdsResource(routeGroup.getListener());
xdsServer.addResponseListener((typeUrl, responseState) => {
if (responseState.state === 'NACKED') {
client.stopCalls();
assert.fail(`Client NACKED ${typeUrl} resource with message ${responseState.errorMessage}`);
}
})
client = XdsTestClient.createFromServer('listener1', xdsServer);
client.sendOneCall(error => {
assert.ifError(error);
// Make another call after the max_connection_age_ms expires
setTimeout(() => {
client.sendOneCall(error => {
done(error);
})
}, 1100);
});
}, reason => done(reason));

})
});
2 changes: 1 addition & 1 deletion packages/grpc-js/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@grpc/grpc-js",
"version": "1.10.0",
"version": "1.10.1",
"description": "gRPC Library for Node - pure JS implementation",
"homepage": "https://grpc.io/",
"repository": "https://github.com/grpc/grpc-node/tree/master/packages/grpc-js",
Expand Down
4 changes: 4 additions & 0 deletions packages/grpc-js/src/load-balancer-pick-first.ts
Original file line number Diff line number Diff line change
Expand Up @@ -605,6 +605,10 @@ export class LeafLoadBalancer {
return this.endpoint;
}

exitIdle() {
this.pickFirstBalancer.exitIdle();
}

destroy() {
this.pickFirstBalancer.destroy();
}
Expand Down
9 changes: 9 additions & 0 deletions packages/grpc-js/src/load-balancer-round-robin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -161,6 +161,15 @@ export class RoundRobinLoadBalancer implements LoadBalancer {
} else {
this.updateState(ConnectivityState.IDLE, new QueuePicker(this));
}
/* round_robin should keep all children connected, this is how we do that.
* We can't do this more efficiently in the individual child's updateState
* callback because that doesn't have a reference to which child the state
* change is associated with. */
for (const child of this.children) {
if (child.getConnectivityState() === ConnectivityState.IDLE) {
child.exitIdle();
}
}
}

private updateState(newState: ConnectivityState, picker: Picker) {
Expand Down

0 comments on commit 513a61a

Please sign in to comment.