mirror of
https://github.com/google/nomulus.git
synced 2025-05-01 04:27:51 +02:00
This CL changes the domain and contact transfer flows to check the entire TransferData on the post-transfer resource, rather than just spot-checking certain fields. This approach provides much better code coverage - in particular, it checks that the non-request flows (approve, cancel, reject) don't modify the fields that they shouldn't be modifying, and that they do actually clear out the transfer server-approve entities fields written by the transfer request flow. It's slightly orthogonal, but I also added testing that the server-approve entities fields are actually set in the request flows, which was previously untested. This is pre-work for introducing an exDate-storing field into TransferData, by making it easier to test everywhere that exDate is set *and* unset only in the correct places. As part of this CL, I've introduced a TransferData.copyConstantFieldsToBuilder() method that is like asBuilder() but instead of copying all the fields to the new builder, it only copies the logically constant ones: losing/gaining client IDs, the request time and TRID, and transferPeriod. This is useful both in tests but is also used in the resolvingPendingTransfer() helper that centralizes the core transfer resolution logic (as of [] That method has its own tests, and in the process I removed a bunch of crufty defunct TransferData tests. ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=171053454
1028 lines
45 KiB
Java
1028 lines
45 KiB
Java
// Copyright 2017 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.flows.domain;
|
|
|
|
import static com.google.common.truth.Truth.assertThat;
|
|
import static google.registry.flows.domain.DomainTransferFlowTestCase.persistWithPendingTransfer;
|
|
import static google.registry.model.EppResourceUtils.loadByForeignKey;
|
|
import static google.registry.model.ofy.ObjectifyService.ofy;
|
|
import static google.registry.model.reporting.DomainTransactionRecord.TransactionReportField.DELETED_DOMAINS_GRACE;
|
|
import static google.registry.model.reporting.DomainTransactionRecord.TransactionReportField.DELETED_DOMAINS_NOGRACE;
|
|
import static google.registry.model.reporting.DomainTransactionRecord.TransactionReportField.NET_ADDS_10_YR;
|
|
import static google.registry.model.reporting.DomainTransactionRecord.TransactionReportField.NET_ADDS_1_YR;
|
|
import static google.registry.model.reporting.DomainTransactionRecord.TransactionReportField.NET_RENEWS_3_YR;
|
|
import static google.registry.model.reporting.DomainTransactionRecord.TransactionReportField.RESTORED_DOMAINS;
|
|
import static google.registry.model.reporting.HistoryEntry.Type.DOMAIN_CREATE;
|
|
import static google.registry.model.reporting.HistoryEntry.Type.DOMAIN_DELETE;
|
|
import static google.registry.model.reporting.HistoryEntry.Type.DOMAIN_TRANSFER_REQUEST;
|
|
import static google.registry.testing.DatastoreHelper.assertBillingEvents;
|
|
import static google.registry.testing.DatastoreHelper.createTld;
|
|
import static google.registry.testing.DatastoreHelper.getOnlyHistoryEntryOfType;
|
|
import static google.registry.testing.DatastoreHelper.getOnlyPollMessage;
|
|
import static google.registry.testing.DatastoreHelper.getPollMessages;
|
|
import static google.registry.testing.DatastoreHelper.loadRegistrar;
|
|
import static google.registry.testing.DatastoreHelper.newDomainResource;
|
|
import static google.registry.testing.DatastoreHelper.newHostResource;
|
|
import static google.registry.testing.DatastoreHelper.persistActiveContact;
|
|
import static google.registry.testing.DatastoreHelper.persistActiveDomain;
|
|
import static google.registry.testing.DatastoreHelper.persistDeletedDomain;
|
|
import static google.registry.testing.DatastoreHelper.persistResource;
|
|
import static google.registry.testing.DomainResourceSubject.assertAboutDomains;
|
|
import static google.registry.testing.HistoryEntrySubject.assertAboutHistoryEntries;
|
|
import static google.registry.testing.TaskQueueHelper.assertDnsTasksEnqueued;
|
|
import static google.registry.util.DateTimeUtils.END_OF_TIME;
|
|
import static google.registry.util.DateTimeUtils.START_OF_TIME;
|
|
import static org.joda.money.CurrencyUnit.USD;
|
|
|
|
import com.google.common.collect.FluentIterable;
|
|
import com.google.common.collect.ImmutableMap;
|
|
import com.google.common.collect.ImmutableSet;
|
|
import com.google.common.collect.ImmutableSortedMap;
|
|
import com.google.common.collect.Iterables;
|
|
import com.googlecode.objectify.Key;
|
|
import google.registry.flows.EppRequestSource;
|
|
import google.registry.flows.ResourceFlowTestCase;
|
|
import google.registry.flows.ResourceFlowUtils.ResourceDoesNotExistException;
|
|
import google.registry.flows.ResourceFlowUtils.ResourceNotOwnedException;
|
|
import google.registry.flows.domain.DomainDeleteFlow.DomainToDeleteHasHostsException;
|
|
import google.registry.flows.domain.DomainFlowUtils.BadCommandForRegistryPhaseException;
|
|
import google.registry.flows.domain.DomainFlowUtils.NotAuthorizedForTldException;
|
|
import google.registry.flows.exceptions.OnlyToolCanPassMetadataException;
|
|
import google.registry.flows.exceptions.ResourceStatusProhibitsOperationException;
|
|
import google.registry.model.billing.BillingEvent;
|
|
import google.registry.model.billing.BillingEvent.Flag;
|
|
import google.registry.model.billing.BillingEvent.Reason;
|
|
import google.registry.model.contact.ContactResource;
|
|
import google.registry.model.domain.DomainResource;
|
|
import google.registry.model.domain.GracePeriod;
|
|
import google.registry.model.domain.rgp.GracePeriodStatus;
|
|
import google.registry.model.eppcommon.ProtocolDefinition.ServiceExtension;
|
|
import google.registry.model.eppcommon.StatusValue;
|
|
import google.registry.model.eppcommon.Trid;
|
|
import google.registry.model.host.HostResource;
|
|
import google.registry.model.poll.PendingActionNotificationResponse;
|
|
import google.registry.model.poll.PollMessage;
|
|
import google.registry.model.registry.Registry;
|
|
import google.registry.model.registry.Registry.TldState;
|
|
import google.registry.model.registry.Registry.TldType;
|
|
import google.registry.model.reporting.DomainTransactionRecord;
|
|
import google.registry.model.reporting.HistoryEntry;
|
|
import google.registry.model.transfer.TransferData;
|
|
import google.registry.model.transfer.TransferResponse;
|
|
import google.registry.model.transfer.TransferStatus;
|
|
import java.util.Map;
|
|
import org.joda.money.Money;
|
|
import org.joda.time.DateTime;
|
|
import org.joda.time.Duration;
|
|
import org.junit.Before;
|
|
import org.junit.Test;
|
|
|
|
/** Unit tests for {@link DomainDeleteFlow}. */
|
|
public class DomainDeleteFlowTest extends ResourceFlowTestCase<DomainDeleteFlow, DomainResource> {
|
|
|
|
private DomainResource domain;
|
|
private HistoryEntry earlierHistoryEntry;
|
|
|
|
private static final DateTime TIME_BEFORE_FLOW = DateTime.parse("2000-06-06T22:00:00.0Z");
|
|
private static final DateTime A_MONTH_AGO = TIME_BEFORE_FLOW.minusMonths(1);
|
|
private static final DateTime A_MONTH_FROM_NOW = TIME_BEFORE_FLOW.plusMonths(1);
|
|
|
|
private static final ImmutableMap<String, String> FEE_06_MAP =
|
|
ImmutableMap.of("FEE_VERSION", "0.6", "FEE_NS", "fee");
|
|
private static final ImmutableMap<String, String> FEE_11_MAP =
|
|
ImmutableMap.of("FEE_VERSION", "0.11", "FEE_NS", "fee11");
|
|
private static final ImmutableMap<String, String> FEE_12_MAP =
|
|
ImmutableMap.of("FEE_VERSION", "0.12", "FEE_NS", "fee12");
|
|
|
|
public DomainDeleteFlowTest() {
|
|
setEppInput("domain_delete.xml");
|
|
clock.setTo(TIME_BEFORE_FLOW);
|
|
}
|
|
|
|
@Before
|
|
public void initDomainTest() throws Exception {
|
|
createTld("tld");
|
|
// For flags extension tests.
|
|
}
|
|
|
|
private void setUpSuccessfulTest() throws Exception {
|
|
createReferencedEntities(A_MONTH_FROM_NOW);
|
|
BillingEvent.Recurring autorenewBillingEvent = persistResource(
|
|
createAutorenewBillingEvent("TheRegistrar").build());
|
|
PollMessage.Autorenew autorenewPollMessage = persistResource(
|
|
createAutorenewPollMessage("TheRegistrar").build());
|
|
domain = persistResource(domain.asBuilder()
|
|
.setAutorenewBillingEvent(Key.create(autorenewBillingEvent))
|
|
.setAutorenewPollMessage(Key.create(autorenewPollMessage))
|
|
.build());
|
|
assertTransactionalFlow(true);
|
|
}
|
|
|
|
private void createReferencedEntities(DateTime expirationTime) throws Exception {
|
|
// Persist a linked contact.
|
|
ContactResource contact = persistActiveContact("sh8013");
|
|
domain = newDomainResource(getUniqueIdFromCommand()).asBuilder()
|
|
.setCreationTimeForTest(TIME_BEFORE_FLOW)
|
|
.setRegistrant(Key.create(contact))
|
|
.setRegistrationExpirationTime(expirationTime)
|
|
.build();
|
|
earlierHistoryEntry = persistResource(
|
|
new HistoryEntry.Builder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setParent(domain)
|
|
.build());
|
|
}
|
|
|
|
private void setUpGracePeriods(GracePeriod... gracePeriods) throws Exception {
|
|
domain = persistResource(
|
|
domain.asBuilder().setGracePeriods(ImmutableSet.copyOf(gracePeriods)).build());
|
|
}
|
|
|
|
private void setUpGracePeriodDurations() {
|
|
persistResource(
|
|
Registry.get("tld")
|
|
.asBuilder()
|
|
.setAddGracePeriodLength(Duration.standardDays(3))
|
|
.setRenewGracePeriodLength(Duration.standardDays(2))
|
|
.setAutoRenewGracePeriodLength(Duration.standardDays(1))
|
|
.setRedemptionGracePeriodLength(Duration.standardHours(1))
|
|
.setPendingDeleteLength(Duration.standardHours(2))
|
|
.build());
|
|
}
|
|
|
|
private void setUpAutorenewGracePeriod() throws Exception {
|
|
createReferencedEntities(A_MONTH_AGO.plusYears(1));
|
|
BillingEvent.Recurring autorenewBillingEvent = persistResource(
|
|
createAutorenewBillingEvent("TheRegistrar")
|
|
.setEventTime(A_MONTH_AGO)
|
|
.build());
|
|
PollMessage.Autorenew autorenewPollMessage = persistResource(
|
|
createAutorenewPollMessage("TheRegistrar")
|
|
.setEventTime(A_MONTH_AGO)
|
|
.build());
|
|
domain = persistResource(
|
|
domain.asBuilder()
|
|
.setGracePeriods(ImmutableSet.of(GracePeriod.createForRecurring(
|
|
GracePeriodStatus.AUTO_RENEW,
|
|
A_MONTH_AGO.plusDays(45),
|
|
"TheRegistrar",
|
|
Key.create(autorenewBillingEvent))))
|
|
.setAutorenewBillingEvent(Key.create(autorenewBillingEvent))
|
|
.setAutorenewPollMessage(Key.create(autorenewPollMessage))
|
|
.build());
|
|
assertTransactionalFlow(true);
|
|
}
|
|
|
|
private void assertAutorenewClosedAndCancellationCreatedFor(
|
|
BillingEvent.OneTime graceBillingEvent,
|
|
HistoryEntry historyEntryDomainDelete) throws Exception {
|
|
DateTime eventTime = clock.nowUtc();
|
|
assertBillingEvents(
|
|
createAutorenewBillingEvent("TheRegistrar")
|
|
.setRecurrenceEndTime(eventTime)
|
|
.build(),
|
|
graceBillingEvent,
|
|
new BillingEvent.Cancellation.Builder()
|
|
.setReason(graceBillingEvent.getReason())
|
|
.setTargetId("example.tld")
|
|
.setClientId("TheRegistrar")
|
|
.setEventTime(eventTime)
|
|
.setBillingTime(TIME_BEFORE_FLOW.plusDays(1))
|
|
.setOneTimeEventKey(Key.create(graceBillingEvent))
|
|
.setParent(historyEntryDomainDelete)
|
|
.build());
|
|
}
|
|
|
|
private void assertOnlyBillingEventIsClosedAutorenew(String clientId) throws Exception {
|
|
// There should be no billing events (even timed to when the transfer would have expired) except
|
|
// for the now closed autorenew one.
|
|
assertBillingEvents(
|
|
createAutorenewBillingEvent(clientId).setRecurrenceEndTime(clock.nowUtc()).build());
|
|
}
|
|
|
|
private BillingEvent.OneTime createBillingEvent(Reason reason, Money cost) {
|
|
return new BillingEvent.OneTime.Builder()
|
|
.setReason(reason)
|
|
.setTargetId("example.tld")
|
|
.setClientId("TheRegistrar")
|
|
.setCost(cost)
|
|
.setPeriodYears(2)
|
|
.setEventTime(TIME_BEFORE_FLOW.minusDays(4))
|
|
.setBillingTime(TIME_BEFORE_FLOW.plusDays(1))
|
|
.setParent(earlierHistoryEntry)
|
|
.build();
|
|
}
|
|
|
|
private BillingEvent.Recurring.Builder createAutorenewBillingEvent(String clientId) {
|
|
return new BillingEvent.Recurring.Builder()
|
|
.setReason(Reason.RENEW)
|
|
.setFlags(ImmutableSet.of(Flag.AUTO_RENEW))
|
|
.setTargetId("example.tld")
|
|
.setClientId(clientId)
|
|
.setEventTime(A_MONTH_FROM_NOW)
|
|
.setRecurrenceEndTime(END_OF_TIME)
|
|
.setParent(earlierHistoryEntry);
|
|
}
|
|
|
|
private PollMessage.Autorenew.Builder createAutorenewPollMessage(String clientId) {
|
|
return new PollMessage.Autorenew.Builder()
|
|
.setTargetId("example.tld")
|
|
.setClientId(clientId)
|
|
.setEventTime(A_MONTH_FROM_NOW)
|
|
.setAutorenewEndTime(END_OF_TIME)
|
|
.setParent(earlierHistoryEntry);
|
|
}
|
|
|
|
@Test
|
|
public void testDryRun() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriods(GracePeriod.create(
|
|
GracePeriodStatus.ADD, TIME_BEFORE_FLOW.plusDays(1), "foo", null));
|
|
dryRunFlowAssertResponse(readFile("domain_delete_response.xml"));
|
|
}
|
|
|
|
@Test
|
|
public void testDryRun_noGracePeriods() throws Exception {
|
|
setUpSuccessfulTest();
|
|
dryRunFlowAssertResponse(readFile("domain_delete_response_pending.xml"));
|
|
}
|
|
|
|
private void doImmediateDeleteTest(GracePeriodStatus gracePeriodStatus, String responseFilename)
|
|
throws Exception {
|
|
doImmediateDeleteTest(gracePeriodStatus, responseFilename, ImmutableMap.<String, String>of());
|
|
}
|
|
|
|
private void doImmediateDeleteTest(
|
|
GracePeriodStatus gracePeriodStatus,
|
|
String responseFilename,
|
|
Map<String, String> substitutions) throws Exception {
|
|
// Persist the billing event so it can be retrieved for cancellation generation and checking.
|
|
setUpSuccessfulTest();
|
|
BillingEvent.OneTime graceBillingEvent =
|
|
persistResource(createBillingEvent(Reason.CREATE, Money.of(USD, 123)));
|
|
setUpGracePeriods(GracePeriod.forBillingEvent(gracePeriodStatus, graceBillingEvent));
|
|
// We should see exactly one poll message, which is for the autorenew 1 month in the future.
|
|
assertPollMessages(createAutorenewPollMessage("TheRegistrar").build());
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile(responseFilename, substitutions));
|
|
// Check that the domain is fully deleted.
|
|
assertThat(reloadResourceByForeignKey()).isNull();
|
|
// The add grace period is for a billable action, so it should trigger a cancellation.
|
|
assertAutorenewClosedAndCancellationCreatedFor(
|
|
graceBillingEvent, getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE));
|
|
assertDnsTasksEnqueued("example.tld");
|
|
// There should be no poll messages. The previous autorenew poll message should now be deleted.
|
|
assertThat(getPollMessages("TheRegistrar", A_MONTH_FROM_NOW)).isEmpty();
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_addGracePeriodResultsInImmediateDelete() throws Exception {
|
|
sessionMetadata.setServiceExtensionUris(ImmutableSet.<String>of());
|
|
doImmediateDeleteTest(GracePeriodStatus.ADD, "domain_delete_response.xml");
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_addGracePeriodCredit_v06() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_11.getUri());
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
doImmediateDeleteTest(GracePeriodStatus.ADD, "domain_delete_response_fee.xml", FEE_06_MAP);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_addGracePeriodCredit_v11() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
doImmediateDeleteTest(GracePeriodStatus.ADD, "domain_delete_response_fee.xml", FEE_11_MAP);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_addGracePeriodCredit_v12() throws Exception {
|
|
doImmediateDeleteTest(GracePeriodStatus.ADD, "domain_delete_response_fee.xml", FEE_12_MAP);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_sunrushAddGracePeriodResultsInImmediateDelete() throws Exception {
|
|
sessionMetadata.setServiceExtensionUris(ImmutableSet.<String>of());
|
|
doImmediateDeleteTest(GracePeriodStatus.SUNRUSH_ADD, "domain_delete_response.xml");
|
|
}
|
|
|
|
private void doSuccessfulTest_noAddGracePeriod(String responseFilename) throws Exception {
|
|
doSuccessfulTest_noAddGracePeriod(responseFilename, ImmutableMap.<String, String>of());
|
|
}
|
|
|
|
private void doSuccessfulTest_noAddGracePeriod(
|
|
String responseFilename, Map<String, String> substitutions) throws Exception {
|
|
// Persist the billing event so it can be retrieved for cancellation generation and checking.
|
|
setUpSuccessfulTest();
|
|
BillingEvent.OneTime renewBillingEvent =
|
|
persistResource(createBillingEvent(Reason.RENEW, Money.of(USD, 456)));
|
|
setUpGracePeriods(
|
|
GracePeriod.forBillingEvent(GracePeriodStatus.RENEW, renewBillingEvent),
|
|
// This grace period has no associated billing event, so it won't cause a cancellation.
|
|
GracePeriod.create(GracePeriodStatus.TRANSFER, TIME_BEFORE_FLOW.plusDays(1), "foo", null));
|
|
// We should see exactly one poll message, which is for the autorenew 1 month in the future.
|
|
assertPollMessages(createAutorenewPollMessage("TheRegistrar").build());
|
|
DateTime originalExpirationTime = domain.getRegistrationExpirationTime();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile(responseFilename, substitutions));
|
|
DomainResource resource = reloadResourceByForeignKey();
|
|
// Check that the domain is in the pending delete state.
|
|
assertAboutDomains().that(resource)
|
|
.hasStatusValue(StatusValue.PENDING_DELETE).and()
|
|
.hasDeletionTime(clock.nowUtc().plus(Registry.get("tld").getRedemptionGracePeriodLength())
|
|
.plus(Registry.get("tld").getPendingDeleteLength())).and()
|
|
.hasDeletePollMessage().and()
|
|
.hasExactlyStatusValues(StatusValue.INACTIVE, StatusValue.PENDING_DELETE).and()
|
|
.hasOneHistoryEntryEachOfTypes(DOMAIN_CREATE, DOMAIN_DELETE);
|
|
// We leave the original expiration time unchanged; if the expiration time is before the
|
|
// deletion time, that means once it passes the domain will experience a "phantom autorenew"
|
|
// where the expirationTime advances and the grace period appears, but since the delete flow
|
|
// closed the autorenew recurrences immediately, there are no other autorenew effects.
|
|
assertAboutDomains().that(resource).hasRegistrationExpirationTime(originalExpirationTime);
|
|
// All existing grace periods that were for billable actions should cause cancellations.
|
|
assertAutorenewClosedAndCancellationCreatedFor(
|
|
renewBillingEvent, getOnlyHistoryEntryOfType(resource, DOMAIN_DELETE));
|
|
// All existing grace periods should be gone, and a new REDEMPTION one should be added.
|
|
assertThat(resource.getGracePeriods()).containsExactly(
|
|
GracePeriod.create(
|
|
GracePeriodStatus.REDEMPTION,
|
|
clock.nowUtc().plus(Registry.get("tld").getRedemptionGracePeriodLength()),
|
|
"TheRegistrar",
|
|
null));
|
|
// There should be a future poll message at the deletion time. The previous autorenew poll
|
|
// message should now be deleted.
|
|
DateTime deletionTime = resource.getDeletionTime();
|
|
assertThat(getPollMessages("TheRegistrar", deletionTime.minusMinutes(1)))
|
|
.isEmpty();
|
|
assertThat(getPollMessages("TheRegistrar", deletionTime)).hasSize(1);
|
|
assertThat(resource.getDeletePollMessage())
|
|
.isEqualTo(Key.create(getOnlyPollMessage("TheRegistrar")));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_noAddGracePeriodResultsInPendingDelete() throws Exception {
|
|
sessionMetadata.setServiceExtensionUris(ImmutableSet.<String>of());
|
|
doSuccessfulTest_noAddGracePeriod("domain_delete_response_pending.xml");
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_renewGracePeriodCredit_v06() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_11.getUri());
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
doSuccessfulTest_noAddGracePeriod("domain_delete_response_pending_fee.xml", FEE_06_MAP);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_renewGracePeriodCredit_v11() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
doSuccessfulTest_noAddGracePeriod("domain_delete_response_pending_fee.xml", FEE_11_MAP);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_renewGracePeriodCredit_v12() throws Exception {
|
|
doSuccessfulTest_noAddGracePeriod("domain_delete_response_pending_fee.xml", FEE_12_MAP);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autorenewPollMessageIsNotDeleted() throws Exception {
|
|
setUpSuccessfulTest();
|
|
// Modify the autorenew poll message so that it has unacked messages in the past. This should
|
|
// prevent it from being deleted when the domain is deleted.
|
|
persistResource(
|
|
ofy().load().key(reloadResourceByForeignKey().getAutorenewPollMessage()).now().asBuilder()
|
|
.setEventTime(A_MONTH_FROM_NOW.minusYears(3))
|
|
.build());
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_pending.xml"));
|
|
// There should now be two poll messages; one for the delete of the domain (in the future), and
|
|
// another for the unacked autorenew messages.
|
|
DateTime deletionTime = reloadResourceByForeignKey().getDeletionTime();
|
|
assertThat(getPollMessages("TheRegistrar", deletionTime.minusMinutes(1)))
|
|
.hasSize(1);
|
|
assertThat(getPollMessages("TheRegistrar", deletionTime)).hasSize(2);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_nonDefaultRedemptionGracePeriod() throws Exception {
|
|
sessionMetadata.setServiceExtensionUris(ImmutableSet.<String>of());
|
|
persistResource(
|
|
Registry.get("tld")
|
|
.asBuilder()
|
|
.setRedemptionGracePeriodLength(Duration.standardMinutes(7))
|
|
.build());
|
|
doSuccessfulTest_noAddGracePeriod("domain_delete_response_pending.xml");
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_nonDefaultPendingDeleteLength() throws Exception {
|
|
sessionMetadata.setServiceExtensionUris(ImmutableSet.<String>of());
|
|
persistResource(
|
|
Registry.get("tld")
|
|
.asBuilder()
|
|
.setPendingDeleteLength(Duration.standardMinutes(8))
|
|
.build());
|
|
doSuccessfulTest_noAddGracePeriod("domain_delete_response_pending.xml");
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autoRenewGracePeriod_v06() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_11.getUri());
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
setUpAutorenewGracePeriod();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_autorenew_fee.xml", FEE_06_MAP));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autoRenewGracePeriod_v11() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
setUpAutorenewGracePeriod();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_autorenew_fee.xml", FEE_11_MAP));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autoRenewGracePeriod_v12() throws Exception {
|
|
setUpAutorenewGracePeriod();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_autorenew_fee.xml", FEE_12_MAP));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autoRenewGracePeriod_priceChanges_v06() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_11.getUri());
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
persistResource(
|
|
Registry.get("tld")
|
|
.asBuilder()
|
|
.setRenewBillingCostTransitions(ImmutableSortedMap.of(
|
|
START_OF_TIME, Money.of(USD, 11), TIME_BEFORE_FLOW.minusDays(5), Money.of(USD, 20)))
|
|
.build());
|
|
setUpAutorenewGracePeriod();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_autorenew_fee.xml", FEE_06_MAP));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autoRenewGracePeriod_priceChanges_v11() throws Exception {
|
|
removeServiceExtensionUri(ServiceExtension.FEE_0_12.getUri());
|
|
persistResource(
|
|
Registry.get("tld")
|
|
.asBuilder()
|
|
.setRenewBillingCostTransitions(ImmutableSortedMap.of(
|
|
START_OF_TIME, Money.of(USD, 11), TIME_BEFORE_FLOW.minusDays(5), Money.of(USD, 20)))
|
|
.build());
|
|
setUpAutorenewGracePeriod();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_autorenew_fee.xml", FEE_11_MAP));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_autoRenewGracePeriod_priceChanges_v12() throws Exception {
|
|
persistResource(
|
|
Registry.get("tld")
|
|
.asBuilder()
|
|
.setRenewBillingCostTransitions(ImmutableSortedMap.of(
|
|
START_OF_TIME, Money.of(USD, 11), TIME_BEFORE_FLOW.minusDays(5), Money.of(USD, 20)))
|
|
.build());
|
|
setUpAutorenewGracePeriod();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_autorenew_fee.xml", FEE_12_MAP));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_noPendingTransfer_deletedAndHasNoTransferData() throws Exception {
|
|
setClientIdForFlow("TheRegistrar");
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_pending.xml"));
|
|
DomainResource domain = reloadResourceByForeignKey();
|
|
assertThat(domain.getTransferData()).isEqualTo(TransferData.EMPTY);
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_pendingTransfer() throws Exception {
|
|
setClientIdForFlow("TheRegistrar");
|
|
setUpSuccessfulTest();
|
|
// Modify the domain we are testing to include a pending transfer.
|
|
TransferData oldTransferData =
|
|
persistWithPendingTransfer(reloadResourceByForeignKey()).getTransferData();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_pending.xml"));
|
|
DomainResource domain = reloadResourceByForeignKey();
|
|
// Check that the domain is in the pending delete state.
|
|
// The PENDING_TRANSFER status should be gone.
|
|
assertAboutDomains().that(domain)
|
|
.hasExactlyStatusValues(StatusValue.INACTIVE, StatusValue.PENDING_DELETE).and()
|
|
.hasDeletionTime(clock.nowUtc().plus(Registry.get("tld").getRedemptionGracePeriodLength())
|
|
.plus(Registry.get("tld").getPendingDeleteLength())).and()
|
|
.hasOneHistoryEntryEachOfTypes(DOMAIN_CREATE, DOMAIN_TRANSFER_REQUEST, DOMAIN_DELETE);
|
|
// All existing grace periods should be gone, and a new REDEMPTION one should be added.
|
|
assertThat(domain.getGracePeriods()).containsExactly(
|
|
GracePeriod.create(
|
|
GracePeriodStatus.REDEMPTION,
|
|
clock.nowUtc().plus(Registry.get("tld").getRedemptionGracePeriodLength()),
|
|
"TheRegistrar",
|
|
null));
|
|
// The poll message (in the future) to the losing registrar for implicit ack should be gone.
|
|
assertThat(getPollMessages("TheRegistrar", clock.nowUtc().plusMonths(1)))
|
|
.isEmpty();
|
|
// The poll message in the future to the gaining registrar should be gone too, but there
|
|
// should be one at the current time to the gaining registrar.
|
|
PollMessage gainingPollMessage = getOnlyPollMessage("NewRegistrar");
|
|
assertThat(gainingPollMessage.getEventTime()).isEqualTo(clock.nowUtc());
|
|
assertThat(
|
|
Iterables.getOnlyElement(FluentIterable
|
|
.from(gainingPollMessage.getResponseData())
|
|
.filter(TransferResponse.class))
|
|
.getTransferStatus())
|
|
.isEqualTo(TransferStatus.SERVER_CANCELLED);
|
|
PendingActionNotificationResponse panData = Iterables.getOnlyElement(FluentIterable
|
|
.from(gainingPollMessage.getResponseData())
|
|
.filter(PendingActionNotificationResponse.class));
|
|
assertThat(panData.getTrid())
|
|
.isEqualTo(Trid.create("transferClient-trid", "transferServer-trid"));
|
|
assertThat(panData.getActionResult()).isFalse();
|
|
// There should be a future poll message to the losing registrar at the deletion time.
|
|
DateTime deletionTime = domain.getDeletionTime();
|
|
assertThat(getPollMessages("TheRegistrar", deletionTime.minusMinutes(1)))
|
|
.isEmpty();
|
|
assertThat(getPollMessages("TheRegistrar", deletionTime)).hasSize(1);
|
|
assertOnlyBillingEventIsClosedAutorenew("TheRegistrar");
|
|
// The domain TransferData should reflect the cancelled transfer as we expect, with
|
|
// all the speculative server-approve fields nulled out.
|
|
assertThat(domain.getTransferData())
|
|
.isEqualTo(
|
|
oldTransferData.copyConstantFieldsToBuilder()
|
|
.setTransferStatus(TransferStatus.SERVER_CANCELLED)
|
|
.setPendingTransferExpirationTime(clock.nowUtc())
|
|
.build());
|
|
// The server-approve entities should all be deleted.
|
|
assertThat(ofy().load().key(oldTransferData.getServerApproveBillingEvent()).now()).isNull();
|
|
assertThat(ofy().load().key(oldTransferData.getServerApproveAutorenewEvent()).now()).isNull();
|
|
assertThat(ofy().load().key(oldTransferData.getServerApproveAutorenewPollMessage()).now())
|
|
.isNull();
|
|
assertThat(oldTransferData.getServerApproveEntities()).isNotEmpty(); // Just a sanity check.
|
|
assertThat(ofy().load()
|
|
.keys(oldTransferData.getServerApproveEntities().toArray(new Key<?>[]{})))
|
|
.isEmpty();
|
|
}
|
|
|
|
@Test
|
|
public void testUnlinkingOfResources() throws Exception {
|
|
sessionMetadata.setServiceExtensionUris(ImmutableSet.<String>of());
|
|
setUpSuccessfulTest();
|
|
// Persist the billing event so it can be retrieved for cancellation generation and checking.
|
|
BillingEvent.OneTime graceBillingEvent =
|
|
persistResource(createBillingEvent(Reason.CREATE, Money.of(USD, 123)));
|
|
// Use a grace period so that the delete is immediate, simplifying the assertions below.
|
|
setUpGracePeriods(GracePeriod.forBillingEvent(GracePeriodStatus.ADD, graceBillingEvent));
|
|
// Add a nameserver.
|
|
HostResource host = persistResource(newHostResource("ns1.example.tld"));
|
|
persistResource(loadByForeignKey(
|
|
DomainResource.class, getUniqueIdFromCommand(), clock.nowUtc())
|
|
.asBuilder()
|
|
.setNameservers(ImmutableSet.of(Key.create(host)))
|
|
.build());
|
|
// Persist another domain that's already been deleted and references this contact and host.
|
|
persistResource(
|
|
newDomainResource("example1.tld")
|
|
.asBuilder()
|
|
.setRegistrant(
|
|
Key.create(loadByForeignKey(ContactResource.class, "sh8013", clock.nowUtc())))
|
|
.setNameservers(ImmutableSet.of(Key.create(host)))
|
|
.setDeletionTime(START_OF_TIME)
|
|
.build());
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response.xml"));
|
|
assertDnsTasksEnqueued("example.tld");
|
|
assertAutorenewClosedAndCancellationCreatedFor(
|
|
graceBillingEvent,
|
|
getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_deletedSubordinateDomain() throws Exception {
|
|
setUpSuccessfulTest();
|
|
persistResource(
|
|
newHostResource("ns1." + getUniqueIdFromCommand()).asBuilder()
|
|
.setSuperordinateDomain(Key.create(reloadResourceByForeignKey()))
|
|
.setDeletionTime(clock.nowUtc().minusDays(1))
|
|
.build());
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(readFile("domain_delete_response_pending.xml"));
|
|
assertDnsTasksEnqueued("example.tld");
|
|
assertOnlyBillingEventIsClosedAutorenew("TheRegistrar");
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_predelegation() throws Exception {
|
|
createTld("tld", TldState.PREDELEGATION);
|
|
setUpSuccessfulTest();
|
|
thrown.expect(BadCommandForRegistryPhaseException.class);
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserPredelegation() throws Exception {
|
|
createTld("tld", TldState.PREDELEGATION);
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response_pending.xml"));
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_neverExisted() throws Exception {
|
|
thrown.expect(
|
|
ResourceDoesNotExistException.class,
|
|
String.format("(%s)", getUniqueIdFromCommand()));
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_existedButWasDeleted() throws Exception {
|
|
persistDeletedDomain(getUniqueIdFromCommand(), clock.nowUtc().minusDays(1));
|
|
thrown.expect(
|
|
ResourceDoesNotExistException.class,
|
|
String.format("(%s)", getUniqueIdFromCommand()));
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_hasSubordinateHosts() throws Exception {
|
|
DomainResource domain = persistActiveDomain(getUniqueIdFromCommand());
|
|
HostResource subordinateHost = persistResource(
|
|
newHostResource("ns1." + getUniqueIdFromCommand()).asBuilder()
|
|
.setSuperordinateDomain(Key.create(reloadResourceByForeignKey()))
|
|
.build());
|
|
domain = persistResource(domain.asBuilder()
|
|
.addSubordinateHost(subordinateHost.getFullyQualifiedHostName())
|
|
.build());
|
|
thrown.expect(DomainToDeleteHasHostsException.class);
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_unauthorizedClient() throws Exception {
|
|
sessionMetadata.setClientId("NewRegistrar");
|
|
persistActiveDomain(getUniqueIdFromCommand());
|
|
thrown.expect(ResourceNotOwnedException.class);
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserUnauthorizedClient() throws Exception {
|
|
sessionMetadata.setClientId("NewRegistrar");
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response_pending.xml"));
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_notAuthorizedForTld() throws Exception {
|
|
setUpSuccessfulTest();
|
|
persistResource(
|
|
loadRegistrar("TheRegistrar")
|
|
.asBuilder()
|
|
.setAllowedTlds(ImmutableSet.<String>of())
|
|
.build());
|
|
thrown.expect(NotAuthorizedForTldException.class);
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserNotAuthorizedForTld() throws Exception {
|
|
setUpSuccessfulTest();
|
|
persistResource(
|
|
loadRegistrar("TheRegistrar")
|
|
.asBuilder()
|
|
.setAllowedTlds(ImmutableSet.<String>of())
|
|
.build());
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response_pending.xml"));
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_clientDeleteProhibited() throws Exception {
|
|
persistResource(newDomainResource(getUniqueIdFromCommand()).asBuilder()
|
|
.addStatusValue(StatusValue.CLIENT_DELETE_PROHIBITED)
|
|
.build());
|
|
thrown.expect(ResourceStatusProhibitsOperationException.class, "clientDeleteProhibited");
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_serverDeleteProhibited() throws Exception {
|
|
persistResource(newDomainResource(getUniqueIdFromCommand()).asBuilder()
|
|
.addStatusValue(StatusValue.SERVER_DELETE_PROHIBITED)
|
|
.build());
|
|
thrown.expect(ResourceStatusProhibitsOperationException.class, "serverDeleteProhibited");
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_pendingDelete() throws Exception {
|
|
persistResource(newDomainResource(getUniqueIdFromCommand()).asBuilder()
|
|
.addStatusValue(StatusValue.PENDING_DELETE)
|
|
.build());
|
|
thrown.expect(ResourceStatusProhibitsOperationException.class, "pendingDelete");
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_metadata() throws Exception {
|
|
eppRequestSource = EppRequestSource.TOOL;
|
|
setEppInput("domain_delete_metadata.xml");
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlow();
|
|
assertAboutDomains().that(reloadResourceByForeignKey())
|
|
.hasOneHistoryEntryEachOfTypes(DOMAIN_CREATE, DOMAIN_DELETE);
|
|
assertAboutHistoryEntries()
|
|
.that(getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE))
|
|
.hasType(DOMAIN_DELETE).and()
|
|
.hasMetadataReason("domain-delete-test").and()
|
|
.hasMetadataRequestedByRegistrar(false);
|
|
}
|
|
|
|
@Test
|
|
public void testFailure_metadataNotFromTool() throws Exception {
|
|
setEppInput("domain_delete_metadata.xml");
|
|
persistResource(newDomainResource(getUniqueIdFromCommand()));
|
|
thrown.expect(OnlyToolCanPassMetadataException.class);
|
|
runFlow();
|
|
}
|
|
|
|
@Test
|
|
public void testIcannActivityReportField_getsLogged() throws Exception {
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlow();
|
|
assertIcannReportingActivityFieldLogged("srs-dom-delete");
|
|
assertTldsFieldLogged("tld");
|
|
}
|
|
|
|
@Test
|
|
public void testIcannTransactionRecord_testTld_notStored() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriodDurations();
|
|
persistResource(Registry.get("tld").asBuilder().setTldType(TldType.TEST).build());
|
|
clock.advanceOneMilli();
|
|
earlierHistoryEntry =
|
|
persistResource(
|
|
earlierHistoryEntry
|
|
.asBuilder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(2))
|
|
.setDomainTransactionRecords(
|
|
ImmutableSet.of(
|
|
DomainTransactionRecord.create(
|
|
"tld", TIME_BEFORE_FLOW.plusDays(1), NET_ADDS_1_YR, 1)))
|
|
.build());
|
|
runFlow();
|
|
HistoryEntry persistedEntry = getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE);
|
|
// No transaction records should be recorded for test TLDs
|
|
assertThat(persistedEntry.getDomainTransactionRecords()).isEmpty();
|
|
}
|
|
|
|
@Test
|
|
public void testIcannTransactionRecord_noGrace_entryOutsideMaxGracePeriod() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriodDurations();
|
|
clock.advanceOneMilli();
|
|
earlierHistoryEntry =
|
|
persistResource(
|
|
earlierHistoryEntry
|
|
.asBuilder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(4))
|
|
.setDomainTransactionRecords(
|
|
ImmutableSet.of(
|
|
DomainTransactionRecord.create(
|
|
"tld", TIME_BEFORE_FLOW.plusDays(1), NET_ADDS_1_YR, 1)))
|
|
.build());
|
|
runFlow();
|
|
HistoryEntry persistedEntry = getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE);
|
|
// Transaction record should just be the non-grace period delete
|
|
assertThat(persistedEntry.getDomainTransactionRecords())
|
|
.containsExactly(
|
|
DomainTransactionRecord.create(
|
|
"tld", clock.nowUtc().plusHours(3), DELETED_DOMAINS_NOGRACE, 1));
|
|
}
|
|
|
|
@Test
|
|
public void testIcannTransactionRecord_noGrace_noAddOrRenewRecords() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriodDurations();
|
|
clock.advanceOneMilli();
|
|
earlierHistoryEntry =
|
|
persistResource(
|
|
earlierHistoryEntry
|
|
.asBuilder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(2))
|
|
.setDomainTransactionRecords(
|
|
ImmutableSet.of(
|
|
// Only add or renew records counts should be cancelled
|
|
DomainTransactionRecord.create(
|
|
"tld", TIME_BEFORE_FLOW.plusDays(1), RESTORED_DOMAINS, 1)))
|
|
.build());
|
|
runFlow();
|
|
HistoryEntry persistedEntry = getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE);
|
|
// Transaction record should just be the non-grace period delete
|
|
assertThat(persistedEntry.getDomainTransactionRecords())
|
|
.containsExactly(
|
|
DomainTransactionRecord.create(
|
|
"tld", clock.nowUtc().plusHours(3), DELETED_DOMAINS_NOGRACE, 1));
|
|
}
|
|
|
|
/** Verifies that if there's no add grace period, we still cancel out valid renew records */
|
|
@Test
|
|
public void testIcannTransactionRecord_noGrace_hasRenewRecord() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriodDurations();
|
|
clock.advanceOneMilli();
|
|
DomainTransactionRecord renewRecord =
|
|
DomainTransactionRecord.create("tld", TIME_BEFORE_FLOW.plusDays(1), NET_RENEWS_3_YR, 1);
|
|
// We don't want to cancel non-add or renew records
|
|
DomainTransactionRecord notCancellableRecord =
|
|
DomainTransactionRecord.create("tld", TIME_BEFORE_FLOW.plusDays(1), RESTORED_DOMAINS, 5);
|
|
earlierHistoryEntry =
|
|
persistResource(
|
|
earlierHistoryEntry
|
|
.asBuilder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(2))
|
|
.setDomainTransactionRecords(ImmutableSet.of(renewRecord, notCancellableRecord))
|
|
.build());
|
|
runFlow();
|
|
HistoryEntry persistedEntry = getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE);
|
|
// We should only see the non-grace period delete record and the renew cancellation record
|
|
assertThat(persistedEntry.getDomainTransactionRecords())
|
|
.containsExactly(
|
|
DomainTransactionRecord.create(
|
|
"tld", clock.nowUtc().plusHours(3), DELETED_DOMAINS_NOGRACE, 1),
|
|
renewRecord.asBuilder().setReportAmount(-1).build());
|
|
}
|
|
|
|
@Test
|
|
public void testIcannTransactionRecord_inGrace_noRecords() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriods(
|
|
GracePeriod.create(
|
|
GracePeriodStatus.ADD, TIME_BEFORE_FLOW.plusDays(1), "TheRegistrar", null));
|
|
setUpGracePeriodDurations();
|
|
clock.advanceOneMilli();
|
|
earlierHistoryEntry =
|
|
persistResource(
|
|
earlierHistoryEntry
|
|
.asBuilder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(2))
|
|
.build());
|
|
runFlow();
|
|
HistoryEntry persistedEntry = getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE);
|
|
// Transaction record should just be the grace period delete
|
|
assertThat(persistedEntry.getDomainTransactionRecords())
|
|
.containsExactly(
|
|
DomainTransactionRecord.create("tld", clock.nowUtc(), DELETED_DOMAINS_GRACE, 1));
|
|
}
|
|
|
|
@Test
|
|
public void testIcannTransactionRecord_inGrace_multipleRecords() throws Exception {
|
|
setUpSuccessfulTest();
|
|
setUpGracePeriods(
|
|
GracePeriod.create(
|
|
GracePeriodStatus.ADD, TIME_BEFORE_FLOW.plusDays(1), "TheRegistrar", null));
|
|
setUpGracePeriodDurations();
|
|
clock.advanceOneMilli();
|
|
earlierHistoryEntry =
|
|
persistResource(
|
|
earlierHistoryEntry
|
|
.asBuilder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(2))
|
|
.setDomainTransactionRecords(
|
|
ImmutableSet.of(
|
|
DomainTransactionRecord.create(
|
|
"tld", TIME_BEFORE_FLOW.plusDays(1), NET_ADDS_10_YR, 1)))
|
|
.build());
|
|
DomainTransactionRecord existingRecord =
|
|
DomainTransactionRecord.create("tld", TIME_BEFORE_FLOW.plusDays(2), NET_ADDS_10_YR, 1);
|
|
// Create a HistoryEntry with a later modification time
|
|
persistResource(
|
|
new HistoryEntry.Builder()
|
|
.setType(DOMAIN_CREATE)
|
|
.setParent(domain)
|
|
.setModificationTime(TIME_BEFORE_FLOW.minusDays(1))
|
|
.setDomainTransactionRecords(ImmutableSet.of(existingRecord))
|
|
.build());
|
|
runFlow();
|
|
HistoryEntry persistedEntry = getOnlyHistoryEntryOfType(domain, DOMAIN_DELETE);
|
|
// Transaction record should be the grace period delete, and the more recent cancellation record
|
|
assertThat(persistedEntry.getDomainTransactionRecords())
|
|
.containsExactly(
|
|
DomainTransactionRecord.create("tld", clock.nowUtc(), DELETED_DOMAINS_GRACE, 1),
|
|
// The cancellation record is the same as the original, except with a -1 counter
|
|
existingRecord.asBuilder().setReportAmount(-1).build());
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserExtension_nonZeroDayGrace_nonZeroDayPendingDelete()
|
|
throws Exception {
|
|
eppRequestSource = EppRequestSource.TOOL;
|
|
setEppInput(
|
|
"domain_delete_superuser_extension.xml",
|
|
ImmutableMap.of("REDEMPTION_GRACE_PERIOD_DAYS", "15", "PENDING_DELETE_DAYS", "4"));
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response_pending.xml"));
|
|
DomainResource resource = reloadResourceByForeignKey();
|
|
assertAboutDomains()
|
|
.that(resource)
|
|
.hasExactlyStatusValues(StatusValue.INACTIVE, StatusValue.PENDING_DELETE)
|
|
.and()
|
|
.hasDeletionTime(clock.nowUtc().plus(Duration.standardDays(19)));
|
|
assertThat(resource.getGracePeriods())
|
|
.containsExactly(
|
|
GracePeriod.create(
|
|
GracePeriodStatus.REDEMPTION,
|
|
clock.nowUtc().plus(Duration.standardDays(15)),
|
|
"TheRegistrar",
|
|
null));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserExtension_zeroDayGrace_nonZeroDayPendingDelete()
|
|
throws Exception {
|
|
eppRequestSource = EppRequestSource.TOOL;
|
|
setEppInput(
|
|
"domain_delete_superuser_extension.xml",
|
|
ImmutableMap.of("REDEMPTION_GRACE_PERIOD_DAYS", "0", "PENDING_DELETE_DAYS", "4"));
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response_pending.xml"));
|
|
DomainResource resource = reloadResourceByForeignKey();
|
|
assertAboutDomains()
|
|
.that(resource)
|
|
.hasExactlyStatusValues(StatusValue.INACTIVE, StatusValue.PENDING_DELETE)
|
|
.and()
|
|
.hasDeletionTime(clock.nowUtc().plus(Duration.standardDays(4)));
|
|
assertThat(resource.getGracePeriods()).isEmpty();
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserExtension_nonZeroDayGrace_zeroDayPendingDelete()
|
|
throws Exception {
|
|
eppRequestSource = EppRequestSource.TOOL;
|
|
setEppInput(
|
|
"domain_delete_superuser_extension.xml",
|
|
ImmutableMap.of("REDEMPTION_GRACE_PERIOD_DAYS", "15", "PENDING_DELETE_DAYS", "0"));
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response_pending.xml"));
|
|
DomainResource resource = reloadResourceByForeignKey();
|
|
assertAboutDomains()
|
|
.that(resource)
|
|
.hasExactlyStatusValues(StatusValue.INACTIVE, StatusValue.PENDING_DELETE)
|
|
.and()
|
|
.hasDeletionTime(clock.nowUtc().plus(Duration.standardDays(15)));
|
|
assertThat(resource.getGracePeriods())
|
|
.containsExactly(
|
|
GracePeriod.create(
|
|
GracePeriodStatus.REDEMPTION,
|
|
clock.nowUtc().plus(Duration.standardDays(15)),
|
|
"TheRegistrar",
|
|
null));
|
|
}
|
|
|
|
@Test
|
|
public void testSuccess_superuserExtension_zeroDayGrace_zeroDayPendingDelete() throws Exception {
|
|
eppRequestSource = EppRequestSource.TOOL;
|
|
setEppInput(
|
|
"domain_delete_superuser_extension.xml",
|
|
ImmutableMap.of("REDEMPTION_GRACE_PERIOD_DAYS", "0", "PENDING_DELETE_DAYS", "0"));
|
|
setUpSuccessfulTest();
|
|
clock.advanceOneMilli();
|
|
runFlowAssertResponse(
|
|
CommitMode.LIVE, UserPrivileges.SUPERUSER, readFile("domain_delete_response.xml"));
|
|
assertThat(reloadResourceByForeignKey()).isNull();
|
|
}
|
|
}
|