Add a "dry run" mode for the "start push notification experiment" command
This commit is contained in:
		
							parent
							
								
									56fdebde75
								
							
						
					
					
						commit
						56ada7f0e9
					
				|  | @ -30,8 +30,19 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
|   @VisibleForTesting |   @VisibleForTesting | ||||||
|   static final String MAX_CONCURRENCY_ARGUMENT = "max-concurrency"; |   static final String MAX_CONCURRENCY_ARGUMENT = "max-concurrency"; | ||||||
| 
 | 
 | ||||||
|   private static final Counter INITIAL_SAMPLE_ALREADY_EXISTS_COUNTER = |   @VisibleForTesting | ||||||
|       Metrics.counter(MetricsUtil.name(StartPushNotificationExperimentCommand.class, "initialSampleAlreadyExists")); |   static final String DRY_RUN_ARGUMENT = "dry-run"; | ||||||
|  | 
 | ||||||
|  |   private static final Counter DEVICE_INSPECTED_COUNTER = | ||||||
|  |       Metrics.counter(MetricsUtil.name(StartPushNotificationExperimentCommand.class, "deviceInspected")); | ||||||
|  | 
 | ||||||
|  |   private static final String RECORD_INITIAL_SAMPLE_COUNTER_NAME = | ||||||
|  |       MetricsUtil.name(StartPushNotificationExperimentCommand.class, "recordInitialSample"); | ||||||
|  | 
 | ||||||
|  |   private static final String APPLY_TREATMENT_COUNTER_NAME = | ||||||
|  |       MetricsUtil.name(StartPushNotificationExperimentCommand.class, "applyTreatment"); | ||||||
|  | 
 | ||||||
|  |   private static final String DRY_RUN_TAG_NAME = "dryRun"; | ||||||
| 
 | 
 | ||||||
|   private static final Logger log = LoggerFactory.getLogger(StartPushNotificationExperimentCommand.class); |   private static final Logger log = LoggerFactory.getLogger(StartPushNotificationExperimentCommand.class); | ||||||
| 
 | 
 | ||||||
|  | @ -52,11 +63,19 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
|         .dest(MAX_CONCURRENCY_ARGUMENT) |         .dest(MAX_CONCURRENCY_ARGUMENT) | ||||||
|         .setDefault(DEFAULT_MAX_CONCURRENCY) |         .setDefault(DEFAULT_MAX_CONCURRENCY) | ||||||
|         .help("Max concurrency for DynamoDB operations"); |         .help("Max concurrency for DynamoDB operations"); | ||||||
|  | 
 | ||||||
|  |     subparser.addArgument("--dry-run") | ||||||
|  |         .type(Boolean.class) | ||||||
|  |         .dest(DRY_RUN_ARGUMENT) | ||||||
|  |         .required(false) | ||||||
|  |         .setDefault(true) | ||||||
|  |         .help("If true, don't actually record samples or apply treatments"); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @Override |   @Override | ||||||
|   protected void crawlAccounts(final Flux<Account> accounts) { |   protected void crawlAccounts(final Flux<Account> accounts) { | ||||||
|     final int maxConcurrency = getNamespace().getInt(MAX_CONCURRENCY_ARGUMENT); |     final int maxConcurrency = getNamespace().getInt(MAX_CONCURRENCY_ARGUMENT); | ||||||
|  |     final boolean dryRun = getNamespace().getBoolean(DRY_RUN_ARGUMENT); | ||||||
| 
 | 
 | ||||||
|     final PushNotificationExperiment<T> experiment = |     final PushNotificationExperiment<T> experiment = | ||||||
|         experimentFactory.buildExperiment(getCommandDependencies(), getConfiguration()); |         experimentFactory.buildExperiment(getCommandDependencies(), getConfiguration()); | ||||||
|  | @ -67,6 +86,7 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
|     accounts |     accounts | ||||||
|         .flatMap(account -> Flux.fromIterable(account.getDevices()) |         .flatMap(account -> Flux.fromIterable(account.getDevices()) | ||||||
|             .map(device -> Tuples.of(account, device))) |             .map(device -> Tuples.of(account, device))) | ||||||
|  |             .doOnNext(ignored -> DEVICE_INSPECTED_COUNTER.increment()) | ||||||
|             .filterWhen(accountAndDevice -> Mono.fromFuture(() -> |             .filterWhen(accountAndDevice -> Mono.fromFuture(() -> | ||||||
|                 experiment.isDeviceEligible(accountAndDevice.getT1(), accountAndDevice.getT2())), |                 experiment.isDeviceEligible(accountAndDevice.getT1(), accountAndDevice.getT2())), | ||||||
|                 maxConcurrency) |                 maxConcurrency) | ||||||
|  | @ -74,7 +94,9 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
|           final UUID accountIdentifier = accountAndDevice.getT1().getIdentifier(IdentityType.ACI); |           final UUID accountIdentifier = accountAndDevice.getT1().getIdentifier(IdentityType.ACI); | ||||||
|           final byte deviceId = accountAndDevice.getT2().getId(); |           final byte deviceId = accountAndDevice.getT2().getId(); | ||||||
| 
 | 
 | ||||||
|           return Mono.fromFuture(() -> { |           final Mono<Boolean> recordInitialSampleMono = dryRun | ||||||
|  |               ? Mono.just(true) | ||||||
|  |               : Mono.fromFuture(() -> { | ||||||
|                     try { |                     try { | ||||||
|                       return pushNotificationExperimentSamples.recordInitialState( |                       return pushNotificationExperimentSamples.recordInitialState( | ||||||
|                           accountIdentifier, |                           accountIdentifier, | ||||||
|  | @ -86,16 +108,17 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
|                       throw new UncheckedIOException(e); |                       throw new UncheckedIOException(e); | ||||||
|                     } |                     } | ||||||
|                   }) |                   }) | ||||||
|               .mapNotNull(stateStored -> { |  | ||||||
|                 if (stateStored) { |  | ||||||
|                   return accountAndDevice; |  | ||||||
|                 } else { |  | ||||||
|                   INITIAL_SAMPLE_ALREADY_EXISTS_COUNTER.increment(); |  | ||||||
|                   return null; |  | ||||||
|                 } |  | ||||||
|               }) |  | ||||||
|                   .retryWhen(Retry.backoff(3, Duration.ofSeconds(1)) |                   .retryWhen(Retry.backoff(3, Duration.ofSeconds(1)) | ||||||
|                   .onRetryExhaustedThrow(((backoffSpec, retrySignal) -> retrySignal.failure()))) |                       .onRetryExhaustedThrow(((backoffSpec, retrySignal) -> retrySignal.failure()))); | ||||||
|  | 
 | ||||||
|  |           return recordInitialSampleMono.mapNotNull(stateStored -> { | ||||||
|  |                 Metrics.counter(RECORD_INITIAL_SAMPLE_COUNTER_NAME, | ||||||
|  |                         DRY_RUN_TAG_NAME, String.valueOf(dryRun), | ||||||
|  |                         "initialSampleAlreadyExists", String.valueOf(!stateStored)) | ||||||
|  |                     .increment(); | ||||||
|  | 
 | ||||||
|  |                 return stateStored ? accountAndDevice : null; | ||||||
|  |               }) | ||||||
|               .onErrorResume(throwable -> { |               .onErrorResume(throwable -> { | ||||||
|                 log.warn("Failed to record initial sample for {}:{} in experiment {}", |                 log.warn("Failed to record initial sample for {}:{} in experiment {}", | ||||||
|                     accountIdentifier, deviceId, experiment.getExperimentName(), throwable); |                     accountIdentifier, deviceId, experiment.getExperimentName(), throwable); | ||||||
|  | @ -109,7 +132,9 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
|           final boolean inExperimentGroup = |           final boolean inExperimentGroup = | ||||||
|               isInExperimentGroup(account.getIdentifier(IdentityType.ACI), device.getId(), experiment.getExperimentName()); |               isInExperimentGroup(account.getIdentifier(IdentityType.ACI), device.getId(), experiment.getExperimentName()); | ||||||
| 
 | 
 | ||||||
|           return Mono.fromFuture(() -> inExperimentGroup |           final Mono<Void> applyTreatmentMono = dryRun | ||||||
|  |               ? Mono.empty() | ||||||
|  |               : Mono.fromFuture(() -> inExperimentGroup | ||||||
|                       ? experiment.applyExperimentTreatment(account, device) |                       ? experiment.applyExperimentTreatment(account, device) | ||||||
|                       : experiment.applyControlTreatment(account, device)) |                       : experiment.applyControlTreatment(account, device)) | ||||||
|                   .onErrorResume(throwable -> { |                   .onErrorResume(throwable -> { | ||||||
|  | @ -122,6 +147,11 @@ public class StartPushNotificationExperimentCommand<T> extends AbstractSinglePas | ||||||
| 
 | 
 | ||||||
|                     return Mono.empty(); |                     return Mono.empty(); | ||||||
|                   }); |                   }); | ||||||
|  | 
 | ||||||
|  |           return applyTreatmentMono | ||||||
|  |                   .doOnSuccess(ignored -> Metrics.counter(APPLY_TREATMENT_COUNTER_NAME, | ||||||
|  |                       DRY_RUN_TAG_NAME, String.valueOf(dryRun), | ||||||
|  |                       "treatment", inExperimentGroup ? "experiment" : "control").increment()); | ||||||
|         }, maxConcurrency) |         }, maxConcurrency) | ||||||
|         .then() |         .then() | ||||||
|         .block(); |         .block(); | ||||||
|  |  | ||||||
|  | @ -18,6 +18,8 @@ import java.util.concurrent.CompletableFuture; | ||||||
| import net.sourceforge.argparse4j.inf.Namespace; | import net.sourceforge.argparse4j.inf.Namespace; | ||||||
| import org.junit.jupiter.api.BeforeEach; | import org.junit.jupiter.api.BeforeEach; | ||||||
| import org.junit.jupiter.api.Test; | import org.junit.jupiter.api.Test; | ||||||
|  | import org.junit.jupiter.params.ParameterizedTest; | ||||||
|  | import org.junit.jupiter.params.provider.ValueSource; | ||||||
| import org.whispersystems.textsecuregcm.experiment.PushNotificationExperiment; | import org.whispersystems.textsecuregcm.experiment.PushNotificationExperiment; | ||||||
| import org.whispersystems.textsecuregcm.experiment.PushNotificationExperimentSamples; | import org.whispersystems.textsecuregcm.experiment.PushNotificationExperimentSamples; | ||||||
| import org.whispersystems.textsecuregcm.identity.IdentityType; | import org.whispersystems.textsecuregcm.identity.IdentityType; | ||||||
|  | @ -30,7 +32,7 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|   private PushNotificationExperimentSamples pushNotificationExperimentSamples; |   private PushNotificationExperimentSamples pushNotificationExperimentSamples; | ||||||
|   private PushNotificationExperiment<String> experiment; |   private PushNotificationExperiment<String> experiment; | ||||||
| 
 | 
 | ||||||
|   private StartPushNotificationExperimentCommand<String> startPushNotificationExperimentCommand; |   private TestStartPushNotificationExperimentCommand startPushNotificationExperimentCommand; | ||||||
| 
 | 
 | ||||||
|   // Taken together, these parameters will produce a device that's enrolled in the experimental group (as opposed to the |   // Taken together, these parameters will produce a device that's enrolled in the experimental group (as opposed to the | ||||||
|   // control group) for an experiment. |   // control group) for an experiment. | ||||||
|  | @ -41,6 +43,7 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|   private static class TestStartPushNotificationExperimentCommand extends StartPushNotificationExperimentCommand<String> { |   private static class TestStartPushNotificationExperimentCommand extends StartPushNotificationExperimentCommand<String> { | ||||||
| 
 | 
 | ||||||
|     private final CommandDependencies commandDependencies; |     private final CommandDependencies commandDependencies; | ||||||
|  |     private boolean dryRun = false; | ||||||
| 
 | 
 | ||||||
|     public TestStartPushNotificationExperimentCommand( |     public TestStartPushNotificationExperimentCommand( | ||||||
|         final PushNotificationExperimentSamples pushNotificationExperimentSamples, |         final PushNotificationExperimentSamples pushNotificationExperimentSamples, | ||||||
|  | @ -67,9 +70,15 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|           null); |           null); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     void setDryRun(final boolean dryRun) { | ||||||
|  |       this.dryRun = dryRun; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     @Override |     @Override | ||||||
|     protected Namespace getNamespace() { |     protected Namespace getNamespace() { | ||||||
|       return new Namespace(Map.of(StartPushNotificationExperimentCommand.MAX_CONCURRENCY_ARGUMENT, 1)); |       return new Namespace(Map.of( | ||||||
|  |           StartPushNotificationExperimentCommand.MAX_CONCURRENCY_ARGUMENT, 1, | ||||||
|  |           StartPushNotificationExperimentCommand.DRY_RUN_ARGUMENT, dryRun)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  | @ -100,8 +109,11 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|         new TestStartPushNotificationExperimentCommand(pushNotificationExperimentSamples, experiment); |         new TestStartPushNotificationExperimentCommand(pushNotificationExperimentSamples, experiment); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @Test |   @ParameterizedTest | ||||||
|   void crawlAccounts() { |   @ValueSource(booleans = {true, false}) | ||||||
|  |   void crawlAccounts(final boolean dryRun) { | ||||||
|  |     startPushNotificationExperimentCommand.setDryRun(dryRun); | ||||||
|  | 
 | ||||||
|     final Device device = mock(Device.class); |     final Device device = mock(Device.class); | ||||||
|     when(device.getId()).thenReturn(DEVICE_ID); |     when(device.getId()).thenReturn(DEVICE_ID); | ||||||
| 
 | 
 | ||||||
|  | @ -110,11 +122,21 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|     when(account.getDevices()).thenReturn(List.of(device)); |     when(account.getDevices()).thenReturn(List.of(device)); | ||||||
| 
 | 
 | ||||||
|     assertDoesNotThrow(() -> startPushNotificationExperimentCommand.crawlAccounts(Flux.just(account))); |     assertDoesNotThrow(() -> startPushNotificationExperimentCommand.crawlAccounts(Flux.just(account))); | ||||||
|  | 
 | ||||||
|  |     if (dryRun) { | ||||||
|  |       verify(experiment, never()).applyExperimentTreatment(any(), any()); | ||||||
|  |     } else { | ||||||
|       verify(experiment).applyExperimentTreatment(account, device); |       verify(experiment).applyExperimentTreatment(account, device); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|   @Test |     verify(experiment, never()).applyControlTreatment(account, device); | ||||||
|   void crawlAccountsExistingSample() throws JsonProcessingException { |   } | ||||||
|  | 
 | ||||||
|  |   @ParameterizedTest | ||||||
|  |   @ValueSource(booleans = {true, false}) | ||||||
|  |   void crawlAccountsExistingSample(final boolean dryRun) throws JsonProcessingException { | ||||||
|  |     startPushNotificationExperimentCommand.setDryRun(dryRun); | ||||||
|  | 
 | ||||||
|     final Device device = mock(Device.class); |     final Device device = mock(Device.class); | ||||||
|     when(device.getId()).thenReturn(DEVICE_ID); |     when(device.getId()).thenReturn(DEVICE_ID); | ||||||
| 
 | 
 | ||||||
|  | @ -126,11 +148,14 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|         .thenReturn(CompletableFuture.completedFuture(false)); |         .thenReturn(CompletableFuture.completedFuture(false)); | ||||||
| 
 | 
 | ||||||
|     assertDoesNotThrow(() -> startPushNotificationExperimentCommand.crawlAccounts(Flux.just(account))); |     assertDoesNotThrow(() -> startPushNotificationExperimentCommand.crawlAccounts(Flux.just(account))); | ||||||
|     verify(experiment, never()).applyExperimentTreatment(account, device); |     verify(experiment, never()).applyExperimentTreatment(any(), any()); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @Test |   @ParameterizedTest | ||||||
|   void crawlAccountsSampleRetry() throws JsonProcessingException { |   @ValueSource(booleans = {true, false}) | ||||||
|  |   void crawlAccountsSampleRetry(final boolean dryRun) throws JsonProcessingException { | ||||||
|  |     startPushNotificationExperimentCommand.setDryRun(dryRun); | ||||||
|  | 
 | ||||||
|     final Device device = mock(Device.class); |     final Device device = mock(Device.class); | ||||||
|     when(device.getId()).thenReturn(DEVICE_ID); |     when(device.getId()).thenReturn(DEVICE_ID); | ||||||
| 
 | 
 | ||||||
|  | @ -144,11 +169,20 @@ class StartPushNotificationExperimentCommandTest { | ||||||
|         .thenReturn(CompletableFuture.completedFuture(true)); |         .thenReturn(CompletableFuture.completedFuture(true)); | ||||||
| 
 | 
 | ||||||
|     assertDoesNotThrow(() -> startPushNotificationExperimentCommand.crawlAccounts(Flux.just(account))); |     assertDoesNotThrow(() -> startPushNotificationExperimentCommand.crawlAccounts(Flux.just(account))); | ||||||
|  | 
 | ||||||
|  |     if (dryRun) { | ||||||
|  |       verify(experiment, never()).applyExperimentTreatment(any(), any()); | ||||||
|  |       verify(pushNotificationExperimentSamples, never()) | ||||||
|  |           .recordInitialState(any(), anyByte(), any(), anyBoolean(), any()); | ||||||
|  |     } else { | ||||||
|       verify(experiment).applyExperimentTreatment(account, device); |       verify(experiment).applyExperimentTreatment(account, device); | ||||||
|       verify(pushNotificationExperimentSamples, times(3)) |       verify(pushNotificationExperimentSamples, times(3)) | ||||||
|           .recordInitialState(ACCOUNT_IDENTIFIER, DEVICE_ID, EXPERIMENT_NAME, true, "test"); |           .recordInitialState(ACCOUNT_IDENTIFIER, DEVICE_ID, EXPERIMENT_NAME, true, "test"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     verify(experiment, never()).applyControlTreatment(account, device); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   @Test |   @Test | ||||||
|   void crawlAccountsExperimentException() { |   void crawlAccountsExperimentException() { | ||||||
|     final Device device = mock(Device.class); |     final Device device = mock(Device.class); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Jon Chambers
						Jon Chambers