// 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.host; import static com.google.common.truth.Truth.assertThat; import static google.registry.testing.DatastoreHelper.assertNoBillingEvents; import static google.registry.testing.DatastoreHelper.createTld; import static google.registry.testing.DatastoreHelper.newDomainApplication; import static google.registry.testing.DatastoreHelper.newDomainResource; import static google.registry.testing.DatastoreHelper.newHostResource; import static google.registry.testing.DatastoreHelper.persistActiveHost; import static google.registry.testing.DatastoreHelper.persistDeletedHost; import static google.registry.testing.DatastoreHelper.persistResource; import static google.registry.testing.EppExceptionSubject.assertAboutEppExceptions; import static google.registry.testing.HostResourceSubject.assertAboutHosts; import static google.registry.testing.JUnitBackports.expectThrows; import static google.registry.testing.TaskQueueHelper.assertNoDnsTasksEnqueued; import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableSet; import com.googlecode.objectify.Key; import google.registry.flows.EppException; import google.registry.flows.ResourceFlowTestCase; import google.registry.flows.ResourceFlowUtils.ResourceDoesNotExistException; import google.registry.flows.ResourceFlowUtils.ResourceNotOwnedException; import google.registry.flows.exceptions.ResourceStatusProhibitsOperationException; import google.registry.flows.exceptions.ResourceToDeleteIsReferencedException; import google.registry.flows.host.HostFlowUtils.HostNameNotLowerCaseException; import google.registry.flows.host.HostFlowUtils.HostNameNotNormalizedException; import google.registry.flows.host.HostFlowUtils.HostNameNotPunyCodedException; import google.registry.model.domain.DomainResource; import google.registry.model.eppcommon.StatusValue; import google.registry.model.eppcommon.Trid; import google.registry.model.host.HostResource; import google.registry.model.registry.Registry; import google.registry.model.reporting.HistoryEntry; import google.registry.model.transfer.TransferData; import google.registry.model.transfer.TransferStatus; import org.joda.time.DateTime; import org.junit.Before; import org.junit.Test; /** Unit tests for {@link HostDeleteFlow}. */ public class HostDeleteFlowTest extends ResourceFlowTestCase { @Before public void initFlowTest() { setEppInput("host_delete.xml", ImmutableMap.of("HOSTNAME", "ns1.example.tld")); } @Test public void testDryRun() throws Exception { persistActiveHost("ns1.example.tld"); dryRunFlowAssertResponse(loadFile("host_delete_response.xml")); } @Test public void testSuccess() throws Exception { persistActiveHost("ns1.example.tld"); clock.advanceOneMilli(); assertTransactionalFlow(true); runFlowAssertResponse(loadFile("host_delete_response.xml")); HostResource deletedHost = reloadResourceByForeignKey(); assertAboutHosts().that(deletedHost).hasStatusValue(StatusValue.PENDING_DELETE); assertAsyncDeletionTaskEnqueued( deletedHost, "TheRegistrar", Trid.create("ABC-12345", "server-trid"), false); assertAboutHosts() .that(deletedHost) .hasOnlyOneHistoryEntryWhich() .hasType(HistoryEntry.Type.HOST_PENDING_DELETE); assertNoBillingEvents(); assertNoDnsTasksEnqueued(); } @Test public void testFailure_neverExisted() throws Exception { ResourceDoesNotExistException thrown = expectThrows(ResourceDoesNotExistException.class, this::runFlow); assertThat(thrown).hasMessageThat().contains("(ns1.example.tld)"); } @Test public void testFailure_existedButWasDeleted() throws Exception { persistDeletedHost("ns1.example.tld", clock.nowUtc().minusDays(1)); ResourceDoesNotExistException thrown = expectThrows(ResourceDoesNotExistException.class, this::runFlow); assertThat(thrown).hasMessageThat().contains("(ns1.example.tld)"); } private void doFailingStatusTest(StatusValue statusValue, Class exception) throws Exception { persistResource( newHostResource("ns1.example.tld") .asBuilder() .setStatusValues(ImmutableSet.of(statusValue)) .build()); Exception e = expectThrows(exception, this::runFlow); assertThat(e).hasMessageThat().contains(statusValue.getXmlName()); } @Test public void testFailure_existedButWasClientDeleteProhibited() throws Exception { doFailingStatusTest( StatusValue.CLIENT_DELETE_PROHIBITED, ResourceStatusProhibitsOperationException.class); } @Test public void testFailure_existedButWasServerDeleteProhibited() throws Exception { doFailingStatusTest( StatusValue.SERVER_DELETE_PROHIBITED, ResourceStatusProhibitsOperationException.class); } @Test public void testFailure_existedButWasPendingDelete() throws Exception { doFailingStatusTest( StatusValue.PENDING_DELETE, ResourceStatusProhibitsOperationException.class); } @Test public void testFailure_unauthorizedClient() throws Exception { sessionMetadata.setClientId("NewRegistrar"); persistActiveHost("ns1.example.tld"); EppException thrown = expectThrows(ResourceNotOwnedException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testSuccess_superuserUnauthorizedClient() throws Exception { sessionMetadata.setClientId("NewRegistrar"); persistActiveHost("ns1.example.tld"); clock.advanceOneMilli(); runFlowAssertResponse( CommitMode.LIVE, UserPrivileges.SUPERUSER, loadFile("host_delete_response.xml")); HostResource deletedHost = reloadResourceByForeignKey(); assertAboutHosts().that(deletedHost).hasStatusValue(StatusValue.PENDING_DELETE); assertAsyncDeletionTaskEnqueued( deletedHost, "NewRegistrar", Trid.create("ABC-12345", "server-trid"), true); assertAboutHosts() .that(deletedHost) .hasOnlyOneHistoryEntryWhich() .hasType(HistoryEntry.Type.HOST_PENDING_DELETE); assertNoBillingEvents(); assertNoDnsTasksEnqueued(); } @Test public void testSuccess_authorizedClientReadFromSuperordinate() throws Exception { sessionMetadata.setClientId("TheRegistrar"); createTld("tld"); DomainResource domain = persistResource( newDomainResource("example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("TheRegistrar") .build()); persistResource( newHostResource("ns1.example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("NewRegistrar") // Shouldn't hurt. .setSuperordinateDomain(Key.create(domain)) .build()); clock.advanceOneMilli(); runFlowAssertResponse(loadFile("host_delete_response.xml")); } @Test public void testFailure_unauthorizedClientReadFromSuperordinate() throws Exception { sessionMetadata.setClientId("TheRegistrar"); createTld("tld"); DomainResource domain = persistResource( newDomainResource("example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("NewRegistrar") .build()); persistResource( newHostResource("ns1.example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("TheRegistrar") // Shouldn't help. .setSuperordinateDomain(Key.create(domain)) .build()); EppException thrown = expectThrows(ResourceNotOwnedException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testSuccess_authorizedClientReadFromTransferredSuperordinate() throws Exception { sessionMetadata.setClientId("NewRegistrar"); createTld("tld"); // Setup a transfer that should have been server approved a day ago. DateTime now = clock.nowUtc(); DateTime requestTime = now.minusDays(1).minus(Registry.DEFAULT_AUTOMATIC_TRANSFER_LENGTH); DateTime transferExpirationTime = now.minusDays(1); DomainResource domain = persistResource( newDomainResource("example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("NewRegistrar") // Shouldn't hurt. .addStatusValue(StatusValue.PENDING_TRANSFER) .setTransferData( new TransferData.Builder() .setTransferStatus(TransferStatus.PENDING) .setGainingClientId("NewRegistrar") .setTransferRequestTime(requestTime) .setLosingClientId("TheRegistrar") .setPendingTransferExpirationTime(transferExpirationTime) .build()) .build()); persistResource( newHostResource("ns1.example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("TheRegistrar") // Shouldn't hurt. .setSuperordinateDomain(Key.create(domain)) .build()); clock.advanceOneMilli(); runFlowAssertResponse(loadFile("host_delete_response.xml")); } @Test public void testFailure_unauthorizedClientReadFromTransferredSuperordinate() throws Exception { sessionMetadata.setClientId("NewRegistrar"); createTld("tld"); // Setup a transfer that should have been server approved a day ago. DateTime now = clock.nowUtc(); DateTime requestTime = now.minusDays(1).minus(Registry.DEFAULT_AUTOMATIC_TRANSFER_LENGTH); DateTime transferExpirationTime = now.minusDays(1); DomainResource domain = persistResource( newDomainResource("example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("NewRegistrar") // Shouldn't help. .addStatusValue(StatusValue.PENDING_TRANSFER) .setTransferData( new TransferData.Builder() .setTransferStatus(TransferStatus.PENDING) .setGainingClientId("TheRegistrar") .setTransferRequestTime(requestTime) .setLosingClientId("NewRegistrar") .setPendingTransferExpirationTime(transferExpirationTime) .build()) .build()); persistResource( newHostResource("ns1.example.tld") .asBuilder() .setPersistedCurrentSponsorClientId("NewRegistrar") // Shouldn't help. .setSuperordinateDomain(Key.create(domain)) .build()); EppException thrown = expectThrows(ResourceNotOwnedException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testFailure_failfastWhenLinkedToDomain() throws Exception { createTld("tld"); persistResource( newDomainResource("example.tld") .asBuilder() .setNameservers(ImmutableSet.of(Key.create(persistActiveHost("ns1.example.tld")))) .build()); EppException thrown = expectThrows(ResourceToDeleteIsReferencedException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testFailure_failfastWhenLinkedToApplication() throws Exception { createTld("tld"); persistResource( newDomainApplication("example.tld") .asBuilder() .setNameservers(ImmutableSet.of(Key.create(persistActiveHost("ns1.example.tld")))) .build()); EppException thrown = expectThrows(ResourceToDeleteIsReferencedException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testFailure_nonLowerCaseHostname() throws Exception { setEppInput("host_delete.xml", ImmutableMap.of("HOSTNAME", "NS1.EXAMPLE.NET")); EppException thrown = expectThrows(HostNameNotLowerCaseException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testFailure_nonPunyCodedHostname() throws Exception { setEppInput("host_delete.xml", ImmutableMap.of("HOSTNAME", "ns1.çauçalito.tld")); HostNameNotPunyCodedException thrown = expectThrows(HostNameNotPunyCodedException.class, this::runFlow); assertThat(thrown).hasMessageThat().contains("expected ns1.xn--aualito-txac.tld"); } @Test public void testFailure_nonCanonicalHostname() throws Exception { setEppInput("host_delete.xml", ImmutableMap.of("HOSTNAME", "ns1.example.tld.")); EppException thrown = expectThrows(HostNameNotNormalizedException.class, this::runFlow); assertAboutEppExceptions().that(thrown).marshalsToXml(); } @Test public void testIcannActivityReportField_getsLogged() throws Exception { persistActiveHost("ns1.example.tld"); clock.advanceOneMilli(); runFlow(); assertIcannReportingActivityFieldLogged("srs-host-delete"); } }