|
17 | 17 | */ |
18 | 18 | package org.apache.beam.runners.dataflow.worker.streaming.harness; |
19 | 19 |
|
| 20 | +import static com.google.common.truth.Truth.assertThat; |
20 | 21 | import static org.junit.Assert.assertEquals; |
21 | 22 | import static org.junit.Assert.assertFalse; |
22 | 23 | import static org.junit.Assert.assertTrue; |
@@ -348,6 +349,143 @@ public void testOnNewWorkerMetadata_redistributesBudget() throws InterruptedExce |
348 | 349 | TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
349 | 350 | } |
350 | 351 |
|
| 352 | + @Test |
| 353 | + public void testOnNewWorkerMetadata_alternatesConnectivityTypesAndRemovesStaleEndpoints() |
| 354 | + throws InterruptedException { |
| 355 | + String workerToken = "workerToken1"; |
| 356 | + |
| 357 | + WorkerMetadataResponse cloudPathMetadata = |
| 358 | + WorkerMetadataResponse.newBuilder() |
| 359 | + .setMetadataVersion(1) |
| 360 | + .setEndpointType(Windmill.WorkerMetadataResponse.EndpointType.CLOUDPATH) |
| 361 | + .addWorkEndpoints( |
| 362 | + WorkerMetadataResponse.Endpoint.newBuilder() |
| 363 | + .setBackendWorkerToken(workerToken) |
| 364 | + .build()) |
| 365 | + .putAllGlobalDataEndpoints(DEFAULT) |
| 366 | + .build(); |
| 367 | + WorkerMetadataResponse directPathMetadata = |
| 368 | + WorkerMetadataResponse.newBuilder() |
| 369 | + .setMetadataVersion(1) |
| 370 | + .setEndpointType(Windmill.WorkerMetadataResponse.EndpointType.DIRECTPATH) |
| 371 | + .addWorkEndpoints( |
| 372 | + WorkerMetadataResponse.Endpoint.newBuilder() |
| 373 | + .setBackendWorkerToken(workerToken + "1") |
| 374 | + .build()) |
| 375 | + .addWorkEndpoints( |
| 376 | + WorkerMetadataResponse.Endpoint.newBuilder() |
| 377 | + .setBackendWorkerToken(workerToken + "2") |
| 378 | + .build()) |
| 379 | + .putAllGlobalDataEndpoints(DEFAULT) |
| 380 | + .build(); |
| 381 | + WorkerMetadataResponse directPathMetadata2 = |
| 382 | + WorkerMetadataResponse.newBuilder() |
| 383 | + .setMetadataVersion(1) |
| 384 | + .setEndpointType(Windmill.WorkerMetadataResponse.EndpointType.DIRECTPATH) |
| 385 | + .addWorkEndpoints( |
| 386 | + WorkerMetadataResponse.Endpoint.newBuilder() |
| 387 | + .setBackendWorkerToken(workerToken + "3") |
| 388 | + .build()) |
| 389 | + .putAllGlobalDataEndpoints(DEFAULT) |
| 390 | + .build(); |
| 391 | + |
| 392 | + TestGetWorkBudgetDistributor getWorkBudgetDistributor = spy(new TestGetWorkBudgetDistributor()); |
| 393 | + fanOutStreamingEngineWorkProvider = |
| 394 | + newFanOutStreamingEngineWorkerHarness( |
| 395 | + GetWorkBudget.builder().setItems(1).setBytes(1).build(), |
| 396 | + getWorkBudgetDistributor, |
| 397 | + noOpProcessWorkItemFn()); |
| 398 | + |
| 399 | + // Sequence : CLOUDPATH -> DIRECTPATH -> CLOUDPATH -> DIRECTPATH |
| 400 | + // Start with CLOUDPATH (version 1, 1 endpoint) |
| 401 | + // Verifies: version > pendingMetadataVersion condition triggers consumption |
| 402 | + fakeGetWorkerMetadataStub.injectWorkerMetadata(cloudPathMetadata); |
| 403 | + verify(getWorkBudgetDistributor, times(1)).distributeBudget(any(), any()); |
| 404 | + TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
| 405 | + assertThat(fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams()).hasSize(1); |
| 406 | + assertThat( |
| 407 | + fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().keySet().stream() |
| 408 | + .map(endpoint -> endpoint.workerToken().orElse("")) |
| 409 | + .collect(Collectors.toSet())) |
| 410 | + .contains(workerToken); |
| 411 | + |
| 412 | + // Switch to DIRECTPATH (same version 1, 2 endpoints, different type) |
| 413 | + // Verifies: type change at same version triggers consumption (consumeWorkerMetadata lines |
| 414 | + // 284-286) |
| 415 | + fakeGetWorkerMetadataStub.injectWorkerMetadata(directPathMetadata); |
| 416 | + verify(getWorkBudgetDistributor, times(2)).distributeBudget(any(), any()); |
| 417 | + TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
| 418 | + assertThat(fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().values()) |
| 419 | + .hasSize(2); |
| 420 | + // Verifies: stale CLOUDPATH endpoint is not consumed |
| 421 | + Set<String> directPathTokens = |
| 422 | + fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().keySet().stream() |
| 423 | + .map(endpoint -> endpoint.workerToken().orElse("")) |
| 424 | + .collect(Collectors.toSet()); |
| 425 | + assertThat(directPathTokens).contains(workerToken + "1"); |
| 426 | + assertThat(directPathTokens).contains(workerToken + "2"); |
| 427 | + assertThat(directPathTokens).containsNoneIn(java.util.Arrays.asList(workerToken)); |
| 428 | + |
| 429 | + // Switch back to CLOUDPATH (same version 1, 1 endpoint, different type) |
| 430 | + fakeGetWorkerMetadataStub.injectWorkerMetadata(cloudPathMetadata); |
| 431 | + verify(getWorkBudgetDistributor, times(3)).distributeBudget(any(), any()); |
| 432 | + TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
| 433 | + assertThat(fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().values()) |
| 434 | + .hasSize(1); |
| 435 | + // Verifies: stale DIRECTPATH endpoints are not consumed |
| 436 | + Set<String> cloudPathTokens = |
| 437 | + fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().keySet().stream() |
| 438 | + .map(endpoint -> endpoint.workerToken().orElse("")) |
| 439 | + .collect(Collectors.toSet()); |
| 440 | + assertThat(cloudPathTokens).contains(workerToken); |
| 441 | + assertThat(cloudPathTokens) |
| 442 | + .containsNoneIn(java.util.Arrays.asList(workerToken + "1", workerToken + "2")); |
| 443 | + |
| 444 | + // Switch to DIRECTPATH (same version 1, 2 endpoints, different type) |
| 445 | + // Verifies: type change works in both directions |
| 446 | + fakeGetWorkerMetadataStub.injectWorkerMetadata(directPathMetadata); |
| 447 | + verify(getWorkBudgetDistributor, times(4)).distributeBudget(any(), any()); |
| 448 | + TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
| 449 | + assertThat(fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams()).hasSize(2); |
| 450 | + directPathTokens = |
| 451 | + fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().keySet().stream() |
| 452 | + .map(endpoint -> endpoint.workerToken().orElse("")) |
| 453 | + .collect(Collectors.toSet()); |
| 454 | + assertThat(directPathTokens).contains(workerToken + "1"); |
| 455 | + assertThat(directPathTokens).contains(workerToken + "2"); |
| 456 | + assertThat(directPathTokens).containsNoneIn(java.util.Arrays.asList(workerToken)); |
| 457 | + |
| 458 | + // Switch to DIRECTPATH (same version 1, 1 endpoint, same type) |
| 459 | + // Verifies: same version same type does not trigger consumption, endpoints remain the same |
| 460 | + fakeGetWorkerMetadataStub.injectWorkerMetadata(directPathMetadata2); |
| 461 | + verify(getWorkBudgetDistributor, times(4)).distributeBudget(any(), any()); |
| 462 | + TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
| 463 | + assertThat(fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams()).hasSize(2); |
| 464 | + directPathTokens = |
| 465 | + fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().keySet().stream() |
| 466 | + .map(endpoint -> endpoint.workerToken().orElse("")) |
| 467 | + .collect(Collectors.toSet()); |
| 468 | + assertThat(directPathTokens).contains(workerToken + "1"); |
| 469 | + assertThat(directPathTokens).contains(workerToken + "2"); |
| 470 | + assertThat(directPathTokens).containsNoneIn(java.util.Arrays.asList(workerToken + "3")); |
| 471 | + |
| 472 | + directPathMetadata2 = directPathMetadata2.toBuilder().setMetadataVersion(2).build(); |
| 473 | + |
| 474 | + // Final switch back to DIRECTPATH (different version:2, 1 endpoint, same type) |
| 475 | + // Verifies: version change triggers consumption even if type is the same. |
| 476 | + fakeGetWorkerMetadataStub.injectWorkerMetadata(directPathMetadata2); |
| 477 | + verify(getWorkBudgetDistributor, times(5)).distributeBudget(any(), any()); |
| 478 | + TimeUnit.SECONDS.sleep(WAIT_FOR_METADATA_INJECTIONS_SECONDS); |
| 479 | + assertThat(fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams()).hasSize(1); |
| 480 | + directPathTokens = |
| 481 | + fanOutStreamingEngineWorkProvider.currentBackends().windmillStreams().keySet().stream() |
| 482 | + .map(endpoint -> endpoint.workerToken().orElse("")) |
| 483 | + .collect(Collectors.toSet()); |
| 484 | + assertThat(directPathTokens) |
| 485 | + .containsNoneIn(java.util.Arrays.asList(workerToken + "1", workerToken + "2")); |
| 486 | + assertThat(directPathTokens).contains(workerToken + "3"); |
| 487 | + } |
| 488 | + |
351 | 489 | private static class WindmillServiceFakeStub |
352 | 490 | extends CloudWindmillServiceV1Alpha1Grpc.CloudWindmillServiceV1Alpha1ImplBase { |
353 | 491 | @Override |
|
0 commit comments