/* * Copyright 2002-2018 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.lang.Nullable; 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: * * *

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 * @author Juergen Hoeller * @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, @Nullable Map parameters) { Assert.hasLength(name, "Extension name must not be empty"); this.name = name; if (!CollectionUtils.isEmpty(parameters)) { Map map = new LinkedCaseInsensitiveMap<>(parameters.size(), Locale.ENGLISH); map.putAll(parameters); this.parameters = Collections.unmodifiableMap(map); } else { this.parameters = Collections.emptyMap(); } } /** * Return the name of the extension (never {@code null) or empty}. */ public String getName() { return this.name; } /** * Return the parameters of the extension (never {@code null}). */ public Map getParameters() { return this.parameters; } @Override public boolean equals(@Nullable Object other) { if (this == other) { return true; } if (other == null || getClass() != other.getClass()) { return false; } WebSocketExtension otherExt = (WebSocketExtension) other; return (this.name.equals(otherExt.name) && this.parameters.equals(otherExt.parameters)); } @Override public int hashCode() { return this.name.hashCode() * 31 + this.parameters.hashCode(); } @Override public String toString() { StringBuilder str = new StringBuilder(); str.append(this.name); this.parameters.forEach((key, value) -> str.append(';').append(key).append('=').append(value)); return str.toString(); } /** * Parse the given, comma-separated string into a list of {@code WebSocketExtension} objects. *

This method can be used to parse a "Sec-WebSocket-Extension" header. * @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 (StringUtils.hasText(extensions)) { String[] tokens = StringUtils.tokenizeToStringArray(extensions, ","); List result = new ArrayList<>(tokens.length); for (String token : tokens) { result.add(parseExtension(token)); } return result; } else { return Collections.emptyList(); } } private static WebSocketExtension parseExtension(String extension) { if (extension.contains(",")) { throw new IllegalArgumentException("Expected 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); } }