mirror of
https://github.com/google/nomulus.git
synced 2025-07-20 09:46:03 +02:00
Add commands to set and check the database migration state (#1174)
This commit is contained in:
parent
2fd342fa43
commit
58a7014c1b
7 changed files with 343 additions and 6 deletions
|
@ -179,8 +179,9 @@ public class DatabaseMigrationStateSchedule extends CrossTldSingleton
|
||||||
ImmutableSortedMap.of(START_OF_TIME, MigrationState.DATASTORE_ONLY),
|
ImmutableSortedMap.of(START_OF_TIME, MigrationState.DATASTORE_ONLY),
|
||||||
MigrationStateTransition.class);
|
MigrationStateTransition.class);
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
@Mapify(TimeMapper.class)
|
@Mapify(TimeMapper.class)
|
||||||
TimedTransitionProperty<MigrationState, MigrationStateTransition> migrationTransitions =
|
public TimedTransitionProperty<MigrationState, MigrationStateTransition> migrationTransitions =
|
||||||
TimedTransitionProperty.forMapify(
|
TimedTransitionProperty.forMapify(
|
||||||
MigrationState.DATASTORE_ONLY, MigrationStateTransition.class);
|
MigrationState.DATASTORE_ONLY, MigrationStateTransition.class);
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,34 @@
|
||||||
|
// Copyright 2021 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package google.registry.tools;
|
||||||
|
|
||||||
|
import com.beust.jcommander.Parameters;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule.MigrationState;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule.MigrationStateTransition;
|
||||||
|
import google.registry.model.common.TimedTransitionProperty;
|
||||||
|
|
||||||
|
/** A command to check the current Registry 3.0 migration state of the database. */
|
||||||
|
@Parameters(separators = " =", commandDescription = "Check current Registry 3.0 migration state")
|
||||||
|
public class GetDatabaseMigrationStateCommand implements CommandWithRemoteApi {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() throws Exception {
|
||||||
|
TimedTransitionProperty<MigrationState, MigrationStateTransition> migrationSchedule =
|
||||||
|
DatabaseMigrationStateSchedule.get();
|
||||||
|
System.out.println(
|
||||||
|
String.format("Current migration schedule: %s", migrationSchedule.toValueMap()));
|
||||||
|
}
|
||||||
|
}
|
|
@ -69,6 +69,7 @@ public final class RegistryTool {
|
||||||
.put("get_allocation_token", GetAllocationTokenCommand.class)
|
.put("get_allocation_token", GetAllocationTokenCommand.class)
|
||||||
.put("get_claims_list", GetClaimsListCommand.class)
|
.put("get_claims_list", GetClaimsListCommand.class)
|
||||||
.put("get_contact", GetContactCommand.class)
|
.put("get_contact", GetContactCommand.class)
|
||||||
|
.put("get_database_migration_state", GetDatabaseMigrationStateCommand.class)
|
||||||
.put("get_domain", GetDomainCommand.class)
|
.put("get_domain", GetDomainCommand.class)
|
||||||
.put("get_history_entries", GetHistoryEntriesCommand.class)
|
.put("get_history_entries", GetHistoryEntriesCommand.class)
|
||||||
.put("get_host", GetHostCommand.class)
|
.put("get_host", GetHostCommand.class)
|
||||||
|
@ -110,6 +111,7 @@ public final class RegistryTool {
|
||||||
.put("resave_epp_resource", ResaveEppResourceCommand.class)
|
.put("resave_epp_resource", ResaveEppResourceCommand.class)
|
||||||
.put("save_sql_credential", SaveSqlCredentialCommand.class)
|
.put("save_sql_credential", SaveSqlCredentialCommand.class)
|
||||||
.put("send_escrow_report_to_icann", SendEscrowReportToIcannCommand.class)
|
.put("send_escrow_report_to_icann", SendEscrowReportToIcannCommand.class)
|
||||||
|
.put("set_database_migration_state", SetDatabaseMigrationStateCommand.class)
|
||||||
.put("set_num_instances", SetNumInstancesCommand.class)
|
.put("set_num_instances", SetNumInstancesCommand.class)
|
||||||
.put("set_sql_replay_checkpoint", SetSqlReplayCheckpointCommand.class)
|
.put("set_sql_replay_checkpoint", SetSqlReplayCheckpointCommand.class)
|
||||||
.put("setup_ote", SetupOteCommand.class)
|
.put("setup_ote", SetupOteCommand.class)
|
||||||
|
|
|
@ -0,0 +1,70 @@
|
||||||
|
// Copyright 2021 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package google.registry.tools;
|
||||||
|
|
||||||
|
import static google.registry.persistence.transaction.TransactionManagerFactory.ofyTm;
|
||||||
|
|
||||||
|
import com.beust.jcommander.Parameter;
|
||||||
|
import com.beust.jcommander.Parameters;
|
||||||
|
import com.google.common.collect.ImmutableSortedMap;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule.MigrationState;
|
||||||
|
import google.registry.tools.params.TransitionListParameter.MigrationStateTransitions;
|
||||||
|
import org.joda.time.DateTime;
|
||||||
|
|
||||||
|
/** Command to set the Registry 3.0 database migration state schedule. */
|
||||||
|
@Parameters(
|
||||||
|
separators = " =",
|
||||||
|
commandDescription = "Set the current database migration state schedule.")
|
||||||
|
public class SetDatabaseMigrationStateCommand extends ConfirmingCommand
|
||||||
|
implements CommandWithRemoteApi {
|
||||||
|
|
||||||
|
private static final String WARNING_MESSAGE =
|
||||||
|
"Attempting to change the schedule with an effect that would take place within the next 10 "
|
||||||
|
+ "minutes. The cache expiration duration is 5 minutes so this MAY BE DANGEROUS.\n";
|
||||||
|
|
||||||
|
@Parameter(
|
||||||
|
names = "--migration_schedule",
|
||||||
|
converter = MigrationStateTransitions.class,
|
||||||
|
validateWith = MigrationStateTransitions.class,
|
||||||
|
required = true,
|
||||||
|
description =
|
||||||
|
"Comma-delimited list of database transitions, of the form"
|
||||||
|
+ " <time>=<migration-state>[,<time>=<migration-state>]*")
|
||||||
|
ImmutableSortedMap<DateTime, MigrationState> transitionSchedule;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected String prompt() {
|
||||||
|
return ofyTm()
|
||||||
|
.transact(
|
||||||
|
() -> {
|
||||||
|
StringBuilder result = new StringBuilder();
|
||||||
|
DateTime now = ofyTm().getTransactionTime();
|
||||||
|
DateTime nextTransition = transitionSchedule.ceilingKey(now);
|
||||||
|
if (nextTransition != null && nextTransition.isBefore(now.plusMinutes(10))) {
|
||||||
|
result.append(WARNING_MESSAGE);
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
.append(String.format("Set new migration state schedule %s?", transitionSchedule))
|
||||||
|
.toString();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected String execute() {
|
||||||
|
ofyTm().transact(() -> DatabaseMigrationStateSchedule.set(transitionSchedule));
|
||||||
|
return String.format("Successfully set new migration state schedule %s", transitionSchedule);
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,7 +19,7 @@ import static com.google.common.base.Preconditions.checkArgument;
|
||||||
import com.google.common.collect.ImmutableMap;
|
import com.google.common.collect.ImmutableMap;
|
||||||
import com.google.common.collect.ImmutableSortedMap;
|
import com.google.common.collect.ImmutableSortedMap;
|
||||||
import com.google.common.collect.Ordering;
|
import com.google.common.collect.Ordering;
|
||||||
import google.registry.model.common.DatabaseMigrationStateSchedule.PrimaryDatabase;
|
import google.registry.model.common.DatabaseMigrationStateSchedule.MigrationState;
|
||||||
import google.registry.model.domain.token.AllocationToken.TokenStatus;
|
import google.registry.model.domain.token.AllocationToken.TokenStatus;
|
||||||
import google.registry.model.registry.Registry.TldState;
|
import google.registry.model.registry.Registry.TldState;
|
||||||
import org.joda.money.Money;
|
import org.joda.money.Money;
|
||||||
|
@ -74,11 +74,11 @@ public abstract class TransitionListParameter<V> extends KeyValueMapParameter<Da
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Converter-validator for primary database transitions. */
|
/** Converter-validator for states of the Registry 3.0 database migration. */
|
||||||
public static class PrimaryDatabaseTransitions extends TransitionListParameter<PrimaryDatabase> {
|
public static class MigrationStateTransitions extends TransitionListParameter<MigrationState> {
|
||||||
@Override
|
@Override
|
||||||
protected PrimaryDatabase parseValue(String value) {
|
protected MigrationState parseValue(String value) {
|
||||||
return PrimaryDatabase.valueOf(value);
|
return MigrationState.valueOf(value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,65 @@
|
||||||
|
// Copyright 2021 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package google.registry.tools;
|
||||||
|
|
||||||
|
import static google.registry.model.common.DatabaseMigrationStateSchedule.DEFAULT_TRANSITION_MAP;
|
||||||
|
import static google.registry.persistence.transaction.TransactionManagerFactory.ofyTm;
|
||||||
|
import static google.registry.util.DateTimeUtils.START_OF_TIME;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSortedMap;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule.MigrationState;
|
||||||
|
import google.registry.testing.DualDatabaseTest;
|
||||||
|
import google.registry.testing.TestOfyAndSql;
|
||||||
|
import org.joda.time.DateTime;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
|
||||||
|
/** Tests for {@link GetDatabaseMigrationStateCommand}. */
|
||||||
|
@DualDatabaseTest
|
||||||
|
public class GetDatabaseMigrationStateCommandTest
|
||||||
|
extends CommandTestCase<GetDatabaseMigrationStateCommand> {
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void beforeEach() {
|
||||||
|
ofyTm().transact(() -> DatabaseMigrationStateSchedule.set(DEFAULT_TRANSITION_MAP.toValueMap()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testInitial_returnsDatastoreOnly() throws Exception {
|
||||||
|
runCommand();
|
||||||
|
assertStdoutIs(
|
||||||
|
String.format("Current migration schedule: %s\n", DEFAULT_TRANSITION_MAP.toValueMap()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testFullSchedule() throws Exception {
|
||||||
|
DateTime now = fakeClock.nowUtc();
|
||||||
|
ImmutableSortedMap<DateTime, MigrationState> transitions =
|
||||||
|
ImmutableSortedMap.of(
|
||||||
|
START_OF_TIME,
|
||||||
|
MigrationState.DATASTORE_ONLY,
|
||||||
|
now.plusHours(1),
|
||||||
|
MigrationState.DATASTORE_PRIMARY,
|
||||||
|
now.plusHours(2),
|
||||||
|
MigrationState.DATASTORE_PRIMARY_READ_ONLY,
|
||||||
|
now.plusHours(3),
|
||||||
|
MigrationState.SQL_PRIMARY,
|
||||||
|
now.plusHours(4),
|
||||||
|
MigrationState.SQL_ONLY);
|
||||||
|
ofyTm().transact(() -> DatabaseMigrationStateSchedule.set(transitions));
|
||||||
|
runCommand();
|
||||||
|
assertStdoutIs(String.format("Current migration schedule: %s\n", transitions));
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,165 @@
|
||||||
|
// Copyright 2021 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package google.registry.tools;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
import static com.google.common.truth.Truth8.assertThat;
|
||||||
|
import static google.registry.model.common.DatabaseMigrationStateSchedule.DEFAULT_TRANSITION_MAP;
|
||||||
|
import static google.registry.persistence.transaction.TransactionManagerFactory.ofyTm;
|
||||||
|
import static google.registry.util.DateTimeUtils.START_OF_TIME;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
|
import com.beust.jcommander.ParameterException;
|
||||||
|
import com.google.common.collect.ImmutableSortedMap;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule;
|
||||||
|
import google.registry.model.common.DatabaseMigrationStateSchedule.MigrationState;
|
||||||
|
import google.registry.testing.DualDatabaseTest;
|
||||||
|
import google.registry.testing.TestOfyAndSql;
|
||||||
|
import org.joda.time.DateTime;
|
||||||
|
|
||||||
|
/** Tests for {@link SetDatabaseMigrationStateCommand}. */
|
||||||
|
@DualDatabaseTest
|
||||||
|
public class SetDatabaseMigrationStateCommandTest
|
||||||
|
extends CommandTestCase<SetDatabaseMigrationStateCommand> {
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testSuccess_setsBasicSchedule() throws Exception {
|
||||||
|
assertThat(DatabaseMigrationStateSchedule.get()).isEqualTo(DEFAULT_TRANSITION_MAP);
|
||||||
|
assertThat(ofyTm().transact(() -> ofyTm().loadSingleton(DatabaseMigrationStateSchedule.class)))
|
||||||
|
.isEmpty();
|
||||||
|
runCommandForced("--migration_schedule=1970-01-01T00:00:00.000Z=DATASTORE_ONLY");
|
||||||
|
// use a raw ofy call to check what's in the DB
|
||||||
|
ofyTm()
|
||||||
|
.transact(
|
||||||
|
() ->
|
||||||
|
assertThat(
|
||||||
|
ofyTm()
|
||||||
|
.loadSingleton(DatabaseMigrationStateSchedule.class)
|
||||||
|
.get()
|
||||||
|
.migrationTransitions)
|
||||||
|
.isEqualTo(DEFAULT_TRANSITION_MAP));
|
||||||
|
assertThat(DatabaseMigrationStateSchedule.get()).isEqualTo(DEFAULT_TRANSITION_MAP);
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testSuccess_fullSchedule() throws Exception {
|
||||||
|
DateTime now = fakeClock.nowUtc();
|
||||||
|
DateTime datastorePrimary = now.plusHours(1);
|
||||||
|
DateTime datastorePrimaryReadOnly = now.plusHours(2);
|
||||||
|
DateTime sqlPrimary = now.plusHours(3);
|
||||||
|
DateTime sqlOnly = now.plusHours(4);
|
||||||
|
runCommandForced(
|
||||||
|
String.format(
|
||||||
|
"--migration_schedule=%s=DATASTORE_ONLY,%s=DATASTORE_PRIMARY,"
|
||||||
|
+ "%s=DATASTORE_PRIMARY_READ_ONLY,%s=SQL_PRIMARY,%s=SQL_ONLY",
|
||||||
|
START_OF_TIME, datastorePrimary, datastorePrimaryReadOnly, sqlPrimary, sqlOnly));
|
||||||
|
assertThat(DatabaseMigrationStateSchedule.get().toValueMap())
|
||||||
|
.containsExactlyEntriesIn(
|
||||||
|
ImmutableSortedMap.of(
|
||||||
|
START_OF_TIME,
|
||||||
|
MigrationState.DATASTORE_ONLY,
|
||||||
|
datastorePrimary,
|
||||||
|
MigrationState.DATASTORE_PRIMARY,
|
||||||
|
datastorePrimaryReadOnly,
|
||||||
|
MigrationState.DATASTORE_PRIMARY_READ_ONLY,
|
||||||
|
sqlPrimary,
|
||||||
|
MigrationState.SQL_PRIMARY,
|
||||||
|
sqlOnly,
|
||||||
|
MigrationState.SQL_ONLY));
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testSuccess_warnsOnChangeSoon() throws Exception {
|
||||||
|
DateTime now = fakeClock.nowUtc();
|
||||||
|
runCommandForced(
|
||||||
|
String.format(
|
||||||
|
"--migration_schedule=%s=DATASTORE_ONLY,%s=DATASTORE_PRIMARY",
|
||||||
|
START_OF_TIME, now.plusMinutes(1)));
|
||||||
|
assertThat(DatabaseMigrationStateSchedule.get().toValueMap())
|
||||||
|
.containsExactlyEntriesIn(
|
||||||
|
ImmutableSortedMap.of(
|
||||||
|
START_OF_TIME,
|
||||||
|
MigrationState.DATASTORE_ONLY,
|
||||||
|
now.plusMinutes(1),
|
||||||
|
MigrationState.DATASTORE_PRIMARY));
|
||||||
|
assertInStdout("MAY BE DANGEROUS");
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testSuccess_goesBackward() throws Exception {
|
||||||
|
DateTime now = fakeClock.nowUtc();
|
||||||
|
runCommandForced(
|
||||||
|
String.format(
|
||||||
|
"--migration_schedule=%s=DATASTORE_ONLY,%s=DATASTORE_PRIMARY,"
|
||||||
|
+ "%s=DATASTORE_PRIMARY_READ_ONLY,%s=DATASTORE_PRIMARY",
|
||||||
|
START_OF_TIME, now.plusHours(1), now.plusHours(2), now.plusHours(3)));
|
||||||
|
assertThat(DatabaseMigrationStateSchedule.get().toValueMap())
|
||||||
|
.containsExactlyEntriesIn(
|
||||||
|
ImmutableSortedMap.of(
|
||||||
|
START_OF_TIME,
|
||||||
|
MigrationState.DATASTORE_ONLY,
|
||||||
|
now.plusHours(1),
|
||||||
|
MigrationState.DATASTORE_PRIMARY,
|
||||||
|
now.plusHours(2),
|
||||||
|
MigrationState.DATASTORE_PRIMARY_READ_ONLY,
|
||||||
|
now.plusHours(3),
|
||||||
|
MigrationState.DATASTORE_PRIMARY));
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testFailure_invalidTransition() {
|
||||||
|
assertThat(
|
||||||
|
assertThrows(
|
||||||
|
IllegalArgumentException.class,
|
||||||
|
() ->
|
||||||
|
runCommandForced(
|
||||||
|
String.format(
|
||||||
|
"--migration_schedule=%s=DATASTORE_ONLY,%s=DATASTORE_PRIMARY_READ_ONLY",
|
||||||
|
START_OF_TIME, START_OF_TIME.plusHours(1)))))
|
||||||
|
.hasMessageThat()
|
||||||
|
.isEqualTo(
|
||||||
|
"validStateTransitions map cannot transition from DATASTORE_ONLY "
|
||||||
|
+ "to DATASTORE_PRIMARY_READ_ONLY.");
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testFailure_invalidTransitionFromOldToNew() {
|
||||||
|
// The map we pass in is valid by itself, but we can't go from DATASTORE_ONLY now to
|
||||||
|
// DATASTORE_PRIMARY_READ_ONLY now
|
||||||
|
DateTime now = fakeClock.nowUtc();
|
||||||
|
assertThat(
|
||||||
|
assertThrows(
|
||||||
|
IllegalArgumentException.class,
|
||||||
|
() ->
|
||||||
|
runCommandForced(
|
||||||
|
String.format(
|
||||||
|
"--migration_schedule=%s=DATASTORE_ONLY,"
|
||||||
|
+ "%s=DATASTORE_PRIMARY,%s=DATASTORE_PRIMARY_READ_ONLY",
|
||||||
|
START_OF_TIME, now.minusHours(2), now.minusHours(1)))))
|
||||||
|
.hasMessageThat()
|
||||||
|
.isEqualTo(
|
||||||
|
"Cannot transition from current state-as-of-now DATASTORE_ONLY "
|
||||||
|
+ "to new state-as-of-now DATASTORE_PRIMARY_READ_ONLY");
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestOfyAndSql
|
||||||
|
void testFailure_invalidParams() {
|
||||||
|
assertThrows(ParameterException.class, this::runCommandForced);
|
||||||
|
assertThrows(ParameterException.class, () -> runCommandForced("--migration_schedule=FOOBAR"));
|
||||||
|
assertThrows(
|
||||||
|
ParameterException.class,
|
||||||
|
() -> runCommandForced("--migration_schedule=1970-01-01T00:00:00.000Z=FOOBAR"));
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue