mirror of
https://github.com/google/nomulus.git
synced 2025-04-30 20:17:51 +02:00
Although the delta implies that this is actually adding code, it's better than it looks, because some of the stuff in ContactFlowUtils is duplicating more generic methods in ResourceFlowUtils, which can be deleted when the domain and host flows are cut over. ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=133149104
261 lines
12 KiB
Java
261 lines
12 KiB
Java
// Copyright 2016 The Domain Registry 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;
|
|
|
|
import static com.google.common.base.Preconditions.checkState;
|
|
import static google.registry.model.EppResourceUtils.queryDomainsUsingResource;
|
|
import static google.registry.model.domain.DomainResource.extendRegistrationWithCap;
|
|
import static google.registry.model.ofy.ObjectifyService.ofy;
|
|
|
|
import com.google.common.base.Function;
|
|
import com.google.common.base.Optional;
|
|
import com.google.common.base.Predicate;
|
|
import com.google.common.collect.ImmutableList;
|
|
import com.google.common.collect.ImmutableSet;
|
|
import com.google.common.collect.Iterables;
|
|
import com.google.common.collect.Sets;
|
|
import com.googlecode.objectify.Key;
|
|
import com.googlecode.objectify.Work;
|
|
import google.registry.flows.EppException.AuthorizationErrorException;
|
|
import google.registry.flows.EppException.InvalidAuthorizationInformationErrorException;
|
|
import google.registry.flows.exceptions.ResourceStatusProhibitsOperationException;
|
|
import google.registry.flows.exceptions.ResourceToDeleteIsReferencedException;
|
|
import google.registry.flows.exceptions.ResourceToMutateDoesNotExistException;
|
|
import google.registry.model.EppResource;
|
|
import google.registry.model.EppResource.Builder;
|
|
import google.registry.model.EppResource.ForeignKeyedEppResource;
|
|
import google.registry.model.contact.ContactResource;
|
|
import google.registry.model.domain.DomainBase;
|
|
import google.registry.model.domain.DomainResource;
|
|
import google.registry.model.eppcommon.AuthInfo;
|
|
import google.registry.model.eppcommon.AuthInfo.BadAuthInfoException;
|
|
import google.registry.model.eppcommon.StatusValue;
|
|
import google.registry.model.eppcommon.Trid;
|
|
import google.registry.model.index.ForeignKeyIndex;
|
|
import google.registry.model.poll.PendingActionNotificationResponse;
|
|
import google.registry.model.poll.PendingActionNotificationResponse.ContactPendingActionNotificationResponse;
|
|
import google.registry.model.poll.PendingActionNotificationResponse.DomainPendingActionNotificationResponse;
|
|
import google.registry.model.poll.PollMessage;
|
|
import google.registry.model.reporting.HistoryEntry;
|
|
import google.registry.model.transfer.TransferData;
|
|
import google.registry.model.transfer.TransferResponse;
|
|
import google.registry.model.transfer.TransferResponse.ContactTransferResponse;
|
|
import google.registry.model.transfer.TransferResponse.DomainTransferResponse;
|
|
import google.registry.model.transfer.TransferStatus;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
import org.joda.time.DateTime;
|
|
|
|
/** Static utility functions for resource transfer flows. */
|
|
public class ResourceFlowUtils {
|
|
|
|
/** Statuses for which an exDate should be added to transfer responses. */
|
|
private static final ImmutableSet<TransferStatus> ADD_EXDATE_STATUSES = Sets.immutableEnumSet(
|
|
TransferStatus.PENDING, TransferStatus.CLIENT_APPROVED, TransferStatus.SERVER_APPROVED);
|
|
|
|
/** In {@link #failfastForAsyncDelete}, check this (arbitrary) number of query results. */
|
|
private static final int FAILFAST_CHECK_COUNT = 5;
|
|
|
|
/**
|
|
* Create a transfer response using the id and type of this resource and the specified
|
|
* {@link TransferData}.
|
|
*/
|
|
public static TransferResponse createTransferResponse(
|
|
EppResource eppResource, TransferData transferData, DateTime now) {
|
|
assertIsContactOrDomain(eppResource);
|
|
TransferResponse.Builder<? extends TransferResponse, ?> builder;
|
|
if (eppResource instanceof ContactResource) {
|
|
builder = new ContactTransferResponse.Builder().setContactId(eppResource.getForeignKey());
|
|
} else {
|
|
DomainResource domain = (DomainResource) eppResource;
|
|
builder = new DomainTransferResponse.Builder()
|
|
.setFullyQualifiedDomainNameName(eppResource.getForeignKey())
|
|
.setExtendedRegistrationExpirationTime(
|
|
ADD_EXDATE_STATUSES.contains(transferData.getTransferStatus())
|
|
? extendRegistrationWithCap(
|
|
now,
|
|
domain.getRegistrationExpirationTime(),
|
|
transferData.getExtendedRegistrationYears())
|
|
: null);
|
|
}
|
|
builder.setGainingClientId(transferData.getGainingClientId())
|
|
.setLosingClientId(transferData.getLosingClientId())
|
|
.setPendingTransferExpirationTime(transferData.getPendingTransferExpirationTime())
|
|
.setTransferRequestTime(transferData.getTransferRequestTime())
|
|
.setTransferStatus(transferData.getTransferStatus());
|
|
return builder.build();
|
|
}
|
|
|
|
/**
|
|
* Create a pending action notification response indicating the resolution of a transfer.
|
|
*
|
|
* <p>The returned object will use the id and type of this resource, the trid of the resource's
|
|
* last transfer request, and the specified status and date.
|
|
*/
|
|
public static PendingActionNotificationResponse createPendingTransferNotificationResponse(
|
|
EppResource eppResource,
|
|
Trid transferRequestTrid,
|
|
boolean actionResult,
|
|
DateTime processedDate) {
|
|
assertIsContactOrDomain(eppResource);
|
|
return eppResource instanceof ContactResource
|
|
? ContactPendingActionNotificationResponse.create(
|
|
eppResource.getForeignKey(), actionResult, transferRequestTrid, processedDate)
|
|
: DomainPendingActionNotificationResponse.create(
|
|
eppResource.getForeignKey(), actionResult, transferRequestTrid, processedDate);
|
|
}
|
|
|
|
private static void assertIsContactOrDomain(EppResource eppResource) {
|
|
checkState(eppResource instanceof ContactResource || eppResource instanceof DomainResource);
|
|
}
|
|
|
|
/** Check that the given clientId corresponds to the owner of given resource. */
|
|
public static void verifyResourceOwnership(String myClientId, EppResource resource)
|
|
throws EppException {
|
|
if (!myClientId.equals(resource.getCurrentSponsorClientId())) {
|
|
throw new ResourceNotOwnedException();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Performs common deletion operations on an EPP resource and returns a builder for further
|
|
* modifications. This is broken out into ResourceFlowUtils in order to expose the functionality
|
|
* to async flows (i.e. mapreduces).
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public static <R extends EppResource> Builder<R, ? extends Builder<R, ?>>
|
|
prepareDeletedResourceAsBuilder(R existingResource, DateTime now) {
|
|
Builder<R, ? extends Builder<R, ?>> builder =
|
|
(Builder<R, ? extends Builder<R, ?>>) existingResource.asBuilder()
|
|
.setDeletionTime(now)
|
|
.setStatusValues(null)
|
|
.setTransferData(
|
|
existingResource.getStatusValues().contains(StatusValue.PENDING_TRANSFER)
|
|
? existingResource.getTransferData().asBuilder()
|
|
.setTransferStatus(TransferStatus.SERVER_CANCELLED)
|
|
.setServerApproveEntities(null)
|
|
.setServerApproveBillingEvent(null)
|
|
.setServerApproveAutorenewEvent(null)
|
|
.setServerApproveAutorenewPollMessage(null)
|
|
.setPendingTransferExpirationTime(null)
|
|
.build()
|
|
: existingResource.getTransferData())
|
|
.wipeOut();
|
|
return builder;
|
|
}
|
|
|
|
/** Update the relevant {@link ForeignKeyIndex} to cache the new deletion time. */
|
|
public static <R extends EppResource> void updateForeignKeyIndexDeletionTime(R resource) {
|
|
if (resource instanceof ForeignKeyedEppResource) {
|
|
ofy().save().entity(ForeignKeyIndex.create(resource, resource.getDeletionTime()));
|
|
}
|
|
}
|
|
|
|
/** If there is a transfer out, delete the server-approve entities and enqueue a poll message. */
|
|
public static <R extends EppResource> void handlePendingTransferOnDelete(
|
|
R existingResource, R newResource, DateTime now, HistoryEntry historyEntry) {
|
|
if (existingResource.getStatusValues().contains(StatusValue.PENDING_TRANSFER)) {
|
|
TransferData oldTransferData = existingResource.getTransferData();
|
|
ofy().delete().keys(oldTransferData.getServerApproveEntities());
|
|
ofy().save().entity(new PollMessage.OneTime.Builder()
|
|
.setClientId(oldTransferData.getGainingClientId())
|
|
.setEventTime(now)
|
|
.setMsg(TransferStatus.SERVER_CANCELLED.getMessage())
|
|
.setResponseData(ImmutableList.of(
|
|
createTransferResponse(newResource, newResource.getTransferData(), now),
|
|
createPendingTransferNotificationResponse(
|
|
existingResource, oldTransferData.getTransferRequestTrid(), false, now)))
|
|
.setParent(historyEntry)
|
|
.build());
|
|
}
|
|
}
|
|
|
|
/** Check whether an asynchronous delete would obviously fail, and throw an exception if so. */
|
|
public static <R extends EppResource> void failfastForAsyncDelete(
|
|
final String targetId,
|
|
final DateTime now,
|
|
final Class<R> resourceClass,
|
|
final Function<DomainBase, ImmutableSet<?>> getPotentialReferences) throws EppException {
|
|
// Enter a transactionless context briefly.
|
|
EppException failfastException = ofy().doTransactionless(new Work<EppException>() {
|
|
@Override
|
|
public EppException run() {
|
|
final ForeignKeyIndex<R> fki = ForeignKeyIndex.load(resourceClass, targetId, now);
|
|
if (fki == null) {
|
|
return new ResourceToMutateDoesNotExistException(resourceClass, targetId);
|
|
}
|
|
// Query for the first few linked domains, and if found, actually load them. The query is
|
|
// eventually consistent and so might be very stale, but the direct load will not be stale,
|
|
// just non-transactional. If we find at least one actual reference then we can reliably
|
|
// fail. If we don't find any, we can't trust the query and need to do the full mapreduce.
|
|
List<Key<DomainBase>> keys = queryDomainsUsingResource(
|
|
resourceClass, fki.getResourceKey(), now, FAILFAST_CHECK_COUNT);
|
|
Predicate<DomainBase> predicate = new Predicate<DomainBase>() {
|
|
@Override
|
|
public boolean apply(DomainBase domain) {
|
|
return getPotentialReferences.apply(domain).contains(fki.getResourceKey());
|
|
}};
|
|
return Iterables.any(ofy().load().keys(keys).values(), predicate)
|
|
? new ResourceToDeleteIsReferencedException()
|
|
: null;
|
|
}
|
|
});
|
|
if (failfastException != null) {
|
|
throw failfastException;
|
|
}
|
|
}
|
|
|
|
/** The specified resource belongs to another client. */
|
|
public static class ResourceNotOwnedException extends AuthorizationErrorException {
|
|
public ResourceNotOwnedException() {
|
|
super("The specified resource belongs to another client");
|
|
}
|
|
}
|
|
|
|
/** Check that the given AuthInfo is either missing or else is valid for the given resource. */
|
|
public static void verifyOptionalAuthInfoForResource(
|
|
Optional<AuthInfo> authInfo, EppResource resource) throws EppException {
|
|
if (authInfo.isPresent()) {
|
|
verifyAuthInfoForResource(authInfo.get(), resource);
|
|
}
|
|
}
|
|
|
|
/** Check that the given AuthInfo is valid for the given resource. */
|
|
public static void verifyAuthInfoForResource(AuthInfo authInfo, EppResource resource)
|
|
throws EppException {
|
|
try {
|
|
authInfo.verifyAuthorizedFor(resource);
|
|
} catch (BadAuthInfoException e) {
|
|
throw new BadAuthInfoForResourceException();
|
|
}
|
|
}
|
|
|
|
/** Check that the resource does not have any disallowed status values. */
|
|
public static void verifyNoDisallowedStatuses(
|
|
EppResource resource, ImmutableSet<StatusValue> disallowedStatuses) throws EppException {
|
|
Set<StatusValue> problems = Sets.intersection(resource.getStatusValues(), disallowedStatuses);
|
|
if (!problems.isEmpty()) {
|
|
throw new ResourceStatusProhibitsOperationException(problems);
|
|
}
|
|
}
|
|
|
|
/** Authorization information for accessing resource is invalid. */
|
|
public static class BadAuthInfoForResourceException
|
|
extends InvalidAuthorizationInformationErrorException {
|
|
public BadAuthInfoForResourceException() {
|
|
super("Authorization information for accessing resource is invalid");
|
|
}
|
|
}
|
|
}
|