mirror of
https://github.com/google/nomulus.git
synced 2025-04-30 20:17:51 +02:00
138 lines
5.8 KiB
Java
138 lines
5.8 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;
|
|
|
|
import static com.google.common.base.Strings.nullToEmpty;
|
|
import static com.google.common.io.BaseEncoding.base64;
|
|
import static google.registry.flows.EppXmlTransformer.unmarshal;
|
|
import static java.nio.charset.StandardCharsets.UTF_8;
|
|
|
|
import com.google.common.annotations.VisibleForTesting;
|
|
import com.google.common.base.Joiner;
|
|
import com.google.common.base.Optional;
|
|
import com.google.common.base.Strings;
|
|
import com.google.common.collect.ImmutableMap;
|
|
import google.registry.flows.FlowModule.EppExceptionInProviderException;
|
|
import google.registry.model.eppcommon.Trid;
|
|
import google.registry.model.eppinput.EppInput;
|
|
import google.registry.model.eppoutput.EppOutput;
|
|
import google.registry.model.eppoutput.EppResponse;
|
|
import google.registry.model.eppoutput.Result;
|
|
import google.registry.model.eppoutput.Result.Code;
|
|
import google.registry.monitoring.whitebox.BigQueryMetricsEnqueuer;
|
|
import google.registry.monitoring.whitebox.EppMetric;
|
|
import google.registry.util.FormattingLogger;
|
|
import javax.inject.Inject;
|
|
import org.json.simple.JSONValue;
|
|
|
|
/**
|
|
* An implementation of the EPP command/response protocol.
|
|
*
|
|
* @see <a href="http://tools.ietf.org/html/rfc5730">RFC 5730 - Extensible Provisioning Protocol</a>
|
|
*/
|
|
public final class EppController {
|
|
|
|
private static final FormattingLogger logger = FormattingLogger.getLoggerForCallerClass();
|
|
|
|
@Inject FlowComponent.Builder flowComponentBuilder;
|
|
@Inject EppMetric.Builder metricBuilder;
|
|
@Inject EppMetrics eppMetrics;
|
|
@Inject BigQueryMetricsEnqueuer bigQueryMetricsEnqueuer;
|
|
@Inject EppController() {}
|
|
|
|
/** Reads EPP XML, executes the matching flow, and returns an {@link EppOutput}. */
|
|
public EppOutput handleEppCommand(
|
|
SessionMetadata sessionMetadata,
|
|
TransportCredentials credentials,
|
|
EppRequestSource eppRequestSource,
|
|
boolean isDryRun,
|
|
boolean isSuperuser,
|
|
byte[] inputXmlBytes) {
|
|
metricBuilder.setClientId(Optional.fromNullable(sessionMetadata.getClientId()));
|
|
metricBuilder.setPrivilegeLevel(isSuperuser ? "SUPERUSER" : "NORMAL");
|
|
try {
|
|
EppInput eppInput;
|
|
try {
|
|
eppInput = unmarshal(EppInput.class, inputXmlBytes);
|
|
} catch (EppException e) {
|
|
// Log the unmarshalling error, with the raw bytes (in base64) to help with debugging.
|
|
logger.infofmt(
|
|
e,
|
|
"EPP request XML unmarshalling failed - \"%s\":\n%s\n%s\n%s\n%s",
|
|
e.getMessage(),
|
|
JSONValue.toJSONString(
|
|
ImmutableMap.<String, Object>of(
|
|
"clientId", nullToEmpty(sessionMetadata.getClientId()),
|
|
"resultCode", e.getResult().getCode().code,
|
|
"resultMessage", e.getResult().getCode().msg,
|
|
"xmlBytes", base64().encode(inputXmlBytes))),
|
|
Strings.repeat("=", 40),
|
|
new String(inputXmlBytes, UTF_8).trim(), // Charset decoding failures are swallowed.
|
|
Strings.repeat("=", 40));
|
|
// Return early by sending an error message, with no clTRID since we couldn't unmarshal it.
|
|
metricBuilder.setStatus(e.getResult().getCode());
|
|
return getErrorResponse(e.getResult(), Trid.create(null));
|
|
}
|
|
metricBuilder.setCommandName(eppInput.getCommandName());
|
|
if (!eppInput.getTargetIds().isEmpty()) {
|
|
metricBuilder.setEppTarget(Joiner.on(',').join(eppInput.getTargetIds()));
|
|
}
|
|
EppOutput output = runFlowConvertEppErrors(flowComponentBuilder
|
|
.flowModule(new FlowModule.Builder()
|
|
.setSessionMetadata(sessionMetadata)
|
|
.setCredentials(credentials)
|
|
.setEppRequestSource(eppRequestSource)
|
|
.setIsDryRun(isDryRun)
|
|
.setIsSuperuser(isSuperuser)
|
|
.setInputXmlBytes(inputXmlBytes)
|
|
.setEppInput(eppInput)
|
|
.build())
|
|
.build());
|
|
if (output.isResponse()) {
|
|
metricBuilder.setStatus(output.getResponse().getResult().getCode());
|
|
}
|
|
return output;
|
|
} finally {
|
|
EppMetric metric = metricBuilder.build();
|
|
bigQueryMetricsEnqueuer.export(metric);
|
|
eppMetrics.incrementEppRequests(metric);
|
|
eppMetrics.recordProcessingTime(metric);
|
|
}
|
|
}
|
|
|
|
/** Runs an EPP flow and converts known exceptions into EPP error responses. */
|
|
private EppOutput runFlowConvertEppErrors(FlowComponent flowComponent) {
|
|
try {
|
|
return flowComponent.flowRunner().run();
|
|
} catch (EppException | EppExceptionInProviderException e) {
|
|
// The command failed. Send the client an error message.
|
|
EppException eppEx = (EppException) (e instanceof EppException ? e : e.getCause());
|
|
return getErrorResponse(eppEx.getResult(), flowComponent.trid());
|
|
} catch (Throwable e) {
|
|
// Something bad and unexpected happened. Send the client a generic error, and log it.
|
|
logger.severe(e, "Unexpected failure");
|
|
return getErrorResponse(Result.create(Code.COMMAND_FAILED), flowComponent.trid());
|
|
}
|
|
}
|
|
|
|
/** Creates a response indicating an EPP failure. */
|
|
@VisibleForTesting
|
|
static EppOutput getErrorResponse(Result result, Trid trid) {
|
|
return EppOutput.create(new EppResponse.Builder()
|
|
.setResult(result)
|
|
.setTrid(trid)
|
|
.build());
|
|
}
|
|
}
|