mirror of
https://github.com/google/nomulus.git
synced 2025-05-09 08:18:21 +02:00
These updates can generally ran slowly in the background (i.e. they're low priority), and shouldn't clog up the system to the point where they slow down live incoming customer-requested updates. ------------- Created by MOE: https://github.com/google/moe MOE_MIGRATED_REVID=250732171
145 lines
5.5 KiB
Java
145 lines
5.5 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.tools.server;
|
|
|
|
import static com.google.common.base.Preconditions.checkArgument;
|
|
import static google.registry.mapreduce.inputs.EppResourceInputs.createEntityInput;
|
|
import static google.registry.model.EppResourceUtils.isActive;
|
|
import static google.registry.model.registry.Registries.assertTldsExist;
|
|
import static google.registry.request.RequestParameters.PARAM_TLDS;
|
|
|
|
import com.google.appengine.tools.mapreduce.Mapper;
|
|
import com.google.common.annotations.VisibleForTesting;
|
|
import com.google.common.collect.ImmutableList;
|
|
import com.google.common.collect.ImmutableSet;
|
|
import com.google.common.flogger.FluentLogger;
|
|
import google.registry.dns.DnsQueue;
|
|
import google.registry.mapreduce.MapreduceRunner;
|
|
import google.registry.model.domain.DomainBase;
|
|
import google.registry.request.Action;
|
|
import google.registry.request.Parameter;
|
|
import google.registry.request.Response;
|
|
import google.registry.request.auth.Auth;
|
|
import google.registry.util.NonFinalForTesting;
|
|
import java.util.Random;
|
|
import javax.inject.Inject;
|
|
import org.joda.time.DateTime;
|
|
import org.joda.time.DateTimeZone;
|
|
import org.joda.time.Duration;
|
|
|
|
/**
|
|
* A mapreduce that enqueues DNS publish tasks on all active domains on the specified TLD(s).
|
|
*
|
|
* <p>This refreshes DNS both for all domain names and all in-bailiwick hostnames, as DNS writers
|
|
* are responsible for enqueuing refresh tasks for subordinate hosts. So this action thus refreshes
|
|
* DNS for everything applicable under all TLDs under management.
|
|
*
|
|
* <p>Because there are no auth settings in the {@link Action} annotation, this command can only be
|
|
* run internally, or by pretending to be internal by setting the X-AppEngine-QueueName header,
|
|
* which only admin users can do.
|
|
*
|
|
* <p>You must pass in a number of <code>smearMinutes</code> as a URL parameter so that the DNS
|
|
* queue doesn't get overloaded. A rough rule of thumb for Cloud DNS is 1 minute per every 1,000
|
|
* domains. This smears the updates out over the next N minutes. For small TLDs consisting of fewer
|
|
* than 1,000 domains, passing in 1 is fine (which will execute all the updates immediately).
|
|
*/
|
|
@Action(
|
|
service = Action.Service.TOOLS,
|
|
path = "/_dr/task/refreshDnsForAllDomains",
|
|
auth = Auth.AUTH_INTERNAL_OR_ADMIN)
|
|
public class RefreshDnsForAllDomainsAction implements Runnable {
|
|
|
|
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
|
|
|
|
@Inject MapreduceRunner mrRunner;
|
|
@Inject Response response;
|
|
|
|
@Inject
|
|
@Parameter(PARAM_TLDS)
|
|
ImmutableSet<String> tlds;
|
|
|
|
@Inject
|
|
@Parameter("smearMinutes")
|
|
int smearMinutes;
|
|
|
|
@Inject Random random;
|
|
|
|
@Inject
|
|
RefreshDnsForAllDomainsAction() {}
|
|
|
|
@Override
|
|
public void run() {
|
|
assertTldsExist(tlds);
|
|
checkArgument(smearMinutes > 0, "Must specify a positive number of smear minutes");
|
|
mrRunner
|
|
.setJobName("Refresh DNS for all domains")
|
|
.setModuleName("tools")
|
|
.setDefaultMapShards(10)
|
|
.runMapOnly(
|
|
new RefreshDnsForAllDomainsActionMapper(tlds, smearMinutes, random),
|
|
ImmutableList.of(createEntityInput(DomainBase.class)))
|
|
.sendLinkToMapreduceConsole(response);
|
|
}
|
|
|
|
/** Mapper to refresh DNS for all active domain resources. */
|
|
public static class RefreshDnsForAllDomainsActionMapper
|
|
extends Mapper<DomainBase, Void, Void> {
|
|
|
|
private static final long serialVersionUID = -5103865047156795489L;
|
|
|
|
@NonFinalForTesting private static DnsQueue dnsQueue = DnsQueue.create();
|
|
|
|
private final ImmutableSet<String> tlds;
|
|
private final int smearMinutes;
|
|
private final Random random;
|
|
|
|
RefreshDnsForAllDomainsActionMapper(
|
|
ImmutableSet<String> tlds, int smearMinutes, Random random) {
|
|
this.tlds = tlds;
|
|
this.smearMinutes = smearMinutes;
|
|
this.random = random;
|
|
}
|
|
|
|
@Override
|
|
public void map(final DomainBase domain) {
|
|
String domainName = domain.getFullyQualifiedDomainName();
|
|
if (tlds.contains(domain.getTld())) {
|
|
if (isActive(domain, DateTime.now(DateTimeZone.UTC))) {
|
|
try {
|
|
// Smear the task execution time over the next N minutes.
|
|
dnsQueue.addDomainRefreshTask(
|
|
domainName, Duration.standardMinutes(random.nextInt(smearMinutes)));
|
|
getContext().incrementCounter("active domains refreshed");
|
|
} catch (Throwable t) {
|
|
logger.atSevere().withCause(t).log(
|
|
"Error while refreshing DNS for domain %s", domainName);
|
|
getContext().incrementCounter("active domains errored");
|
|
}
|
|
} else {
|
|
getContext().incrementCounter("inactive domains skipped");
|
|
}
|
|
} else {
|
|
getContext().incrementCounter("domains on non-targeted TLDs skipped");
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
public static DnsQueue setDnsQueueForTest(DnsQueue testQueue) {
|
|
DnsQueue currentQueue = dnsQueue;
|
|
dnsQueue = testQueue;
|
|
return currentQueue;
|
|
}
|
|
}
|
|
}
|