All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.jetty.http.ComplianceViolation Maven / Gradle / Ivy

There is a newer version: 12.1.0.alpha0
Show newest version
//
// ========================================================================
// Copyright (c) 1995 Mort Bay Consulting Pty Ltd and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
//
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
// ========================================================================
//

package org.eclipse.jetty.http;

import java.util.ArrayList;
import java.util.List;
import java.util.Set;

import org.eclipse.jetty.util.Attributes;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * A Compliance Violation represents a requirement of an RFC, specification or Jetty implementation
 * that may be allowed to be violated if it is included in a {@link ComplianceViolation.Mode}.
 * For example, supporting HTTP/0.9 is no longer a requirement of the current HTTP RFC, so by including
 * the {@link HttpCompliance.Violation#HTTP_0_9} in the {@link HttpCompliance} {@link ComplianceViolation.Mode}
 * is interpreted as allowing HTTP/0.9 to be supported.
 */
public interface ComplianceViolation
{
    /**
     * @return The name of the violation.
     */
    String getName();

    /**
     * @return A URL to the specification that provides more information regarding the requirement that may be violated.
     */
    String getURL();

    /**
     * @return A short description of the violation.
     */
    String getDescription();

    /**
     * @param mode A {@link ComplianceViolation.Mode} to test against
     * @return True iff this violations is allowed by the mode.
     */
    default boolean isAllowedBy(Mode mode)
    {
        return mode.allows(this);
    }

    /**
     * A Mode is a set of {@link ComplianceViolation}s that are allowed.
     */
    interface Mode
    {
        /**
         * @return The name of the compliance violation mode.
         */
        String getName();

        /**
         * @param violation The {@link ComplianceViolation} to test
         * @return true iff the violation is allowed by this mode.
         */
        boolean allows(ComplianceViolation violation);

        /**
         * @return The immutable set of all known violations for this mode.
         */
        Set getKnown();

        /**
         * @return The immutable set of violations allowed by this mode.
         */
        Set getAllowed();
    }

    record Event(ComplianceViolation.Mode mode, ComplianceViolation violation, String details)
    {
        @Override
        public String toString()
        {
            return String.format("%s (see %s) in mode %s for %s",
                violation.getDescription(), violation.getURL(), mode, details);
        }
    }

    /**
     * A listener that can be notified of violations.
     */
    interface Listener
    {
        Listener NOOP = new Listener() {};

        /**
         * Initialize the listener in preparation for a new request life cycle.
         * @return The Listener instance to use for the request life cycle.
         */
        default Listener initialize()
        {
            return this;
        }

        /**
         * A new Request has begun.
         *
         * @param request the request attributes, or null if the Request does not exist yet (eg: during parsing of HTTP/1.1 headers, before request is created)
         */
        default void onRequestBegin(Attributes request)
        {
        }

        /**
         * A Request has ended.
         *
         * @param request the request attributes, or null if Request does not exist yet (eg: during handling of a {@link BadMessageException})
         */
        default void onRequestEnd(Attributes request)
        {
        }

        /**
         * The compliance violation event.
         *
         * @param event the compliance violation event
         */
        default void onComplianceViolation(Event event)
        {
            onComplianceViolation(event.mode, event.violation, event.details);
        }

        /**
         * The compliance violation event.
         *
         * @param mode the mode
         * @param violation the violation
         * @param details the details
         * @deprecated use {@link #onComplianceViolation(Event)} instead.  Will be removed in Jetty 12.1.0
         */
        @Deprecated(since = "12.0.6", forRemoval = true)
        default void onComplianceViolation(Mode mode, ComplianceViolation violation, String details)
        {
        }
    }

    class LoggingListener implements Listener
    {
        private static final Logger LOG = LoggerFactory.getLogger(ComplianceViolation.class);

        @Override
        public void onComplianceViolation(Event event)
        {
            if (LOG.isDebugEnabled())
                LOG.debug(event.toString());
        }
    }

    class CapturingListener implements Listener
    {
        public static final String VIOLATIONS_ATTR_KEY = "org.eclipse.jetty.http.compliance.violations";

        private final List events;

        public CapturingListener()
        {
            this(null);
        }

        private CapturingListener(List events)
        {
            this.events = events;
        }

        @Override
        public Listener initialize()
        {
            return new CapturingListener(new ArrayList<>());
        }

        @Override
        public void onRequestBegin(Attributes request)
        {
            if (request != null)
                request.setAttribute(VIOLATIONS_ATTR_KEY, events);
        }

        @Override
        public void onComplianceViolation(Event event)
        {
            events.add(event);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy