google.registry.flows.EppController.java Source code

Java tutorial

Introduction

Here is the source code for google.registry.flows.EppController.java

Source

// Copyright 2016 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());
    }
}