// 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 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 static google.registry.util.PipelineUtils.createJobPath; 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.DomainResource; 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 javax.inject.Inject; import org.joda.time.DateTime; import org.joda.time.DateTimeZone; /** * A mapreduce that enqueues DNS publish tasks on all active domains on the specified TLD(s). * *

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. * *

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. */ @Action( 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 tlds; @Inject RefreshDnsForAllDomainsAction() {} @Override public void run() { assertTldsExist(tlds); response.sendJavaScriptRedirect( createJobPath( mrRunner .setJobName("Refresh DNS for all domains") .setModuleName("tools") .setDefaultMapShards(10) .runMapOnly( new RefreshDnsForAllDomainsActionMapper(tlds), ImmutableList.of(createEntityInput(DomainResource.class))))); } /** Mapper to refresh DNS for all active domain resources. */ public static class RefreshDnsForAllDomainsActionMapper extends Mapper { private static final long serialVersionUID = 1455544013508953083L; @NonFinalForTesting @VisibleForTesting static DnsQueue dnsQueue = DnsQueue.create(); private final ImmutableSet tlds; RefreshDnsForAllDomainsActionMapper(ImmutableSet tlds) { this.tlds = tlds; } @Override public void map(final DomainResource domain) { String domainName = domain.getFullyQualifiedDomainName(); if (tlds.contains(domain.getTld())) { if (isActive(domain, DateTime.now(DateTimeZone.UTC))) { try { dnsQueue.addDomainRefreshTask(domainName); 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"); } } } }