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

org.springframework.web.socket.WebSocketExtension Maven / Gradle / Ivy

/*
 * Copyright 2002-2014 the original author or authors.
 *
 * 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 org.springframework.web.socket;

import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;

import org.springframework.util.Assert;
import org.springframework.util.CollectionUtils;
import org.springframework.util.LinkedCaseInsensitiveMap;
import org.springframework.util.StringUtils;

/**
 * Represents a WebSocket extension as defined in the RFC 6455.
 * WebSocket extensions add protocol features to the WebSocket protocol. The extensions
 * used within a session are negotiated during the handshake phase as follows:
 * 
    *
  • the client may ask for specific extensions in the HTTP handshake request
  • *
  • the server responds with the final list of extensions to use in the current session
  • *
* *

WebSocket Extension HTTP headers may include parameters and follow * RFC 7230 section 3.2

* *

Note that the order of extensions in HTTP headers defines their order of execution, * e.g. extensions "foo, bar" will be executed as "bar(foo(message))".

* * @author Brian Clozel * @since 4.0 * @see WebSocket Protocol Extensions, RFC 6455 - Section 9 */ public class WebSocketExtension { private final String name; private final Map parameters; /** * Create a WebSocketExtension with the given name. * @param name the name of the extension */ public WebSocketExtension(String name) { this(name, null); } /** * Create a WebSocketExtension with the given name and parameters. * @param name the name of the extension * @param parameters the parameters */ public WebSocketExtension(String name, Map parameters) { Assert.hasLength(name, "extension name must not be empty"); this.name = name; if (!CollectionUtils.isEmpty(parameters)) { Map m = new LinkedCaseInsensitiveMap(parameters.size(), Locale.ENGLISH); m.putAll(parameters); this.parameters = Collections.unmodifiableMap(m); } else { this.parameters = Collections.emptyMap(); } } /** * @return the name of the extension */ public String getName() { return this.name; } /** * @return the parameters of the extension, never {@code null} */ public Map getParameters() { return this.parameters; } /** * Parse the given, comma-separated string into a list of {@code WebSocketExtension} objects. *

This method can be used to parse a "Sec-WebSocket-Extension" extensions. * @param extensions the string to parse * @return the list of extensions * @throws IllegalArgumentException if the string cannot be parsed */ public static List parseExtensions(String extensions) { if (extensions == null || !StringUtils.hasText(extensions)) { return Collections.emptyList(); } else { List result = new ArrayList(); for (String token : extensions.split(",")) { result.add(parseExtension(token)); } return result; } } private static WebSocketExtension parseExtension(String extension) { Assert.doesNotContain(extension, ",", "Expected a single extension value: " + extension); String[] parts = StringUtils.tokenizeToStringArray(extension, ";"); String name = parts[0].trim(); Map parameters = null; if (parts.length > 1) { parameters = new LinkedHashMap(parts.length - 1); for (int i = 1; i < parts.length; i++) { String parameter = parts[i]; int eqIndex = parameter.indexOf('='); if (eqIndex != -1) { String attribute = parameter.substring(0, eqIndex); String value = parameter.substring(eqIndex + 1, parameter.length()); parameters.put(attribute, value); } } } return new WebSocketExtension(name, parameters); } @Override public boolean equals(Object o) { if (this == o) { return true; } if ((o == null) || (getClass() != o.getClass())) { return false; } WebSocketExtension that = (WebSocketExtension) o; if (!name.equals(that.name)) { return false; } if (!parameters.equals(that.parameters)) { return false; } return true; } @Override public int hashCode() { int result = name.hashCode(); result = 31 * result + parameters.hashCode(); return result; } @Override public String toString() { StringBuilder str = new StringBuilder(); str.append(this.name); for (String param : parameters.keySet()) { str.append(';'); str.append(param); str.append('='); str.append(this.parameters.get(param)); } return str.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy