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

com.github.mthizo247.cloud.netflix.zuul.web.socket.ZuulWebSocketConfiguration Maven / Gradle / Ivy

/*
 * Copyright 2002-2017 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 com.github.mthizo247.cloud.netflix.zuul.web.socket;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

import javax.annotation.PostConstruct;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.cloud.netflix.zuul.filters.ZuulProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.messaging.converter.MessageConverter;
import org.springframework.messaging.simp.SimpMessagingTemplate;
import org.springframework.messaging.simp.config.MessageBrokerRegistry;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
import org.springframework.util.ErrorHandler;
import org.springframework.web.socket.WebSocketHandler;
import org.springframework.web.socket.WebSocketHttpHeaders;
import org.springframework.web.socket.client.standard.StandardWebSocketClient;
import org.springframework.web.socket.config.annotation.AbstractWebSocketMessageBrokerConfigurer;
import org.springframework.web.socket.config.annotation.StompEndpointRegistry;
import org.springframework.web.socket.config.annotation.WebSocketTransportRegistration;
import org.springframework.web.socket.handler.WebSocketHandlerDecoratorFactory;
import org.springframework.web.socket.messaging.WebSocketStompClient;
import org.springframework.web.socket.sockjs.client.SockJsClient;
import org.springframework.web.socket.sockjs.client.Transport;
import org.springframework.web.socket.sockjs.client.WebSocketTransport;

/**
 * Zuul reverse proxy web socket configuration
 *
 * @author Ronald Mthombeni
 * @author Salman Noor
 */
@Configuration
@ConditionalOnWebApplication
@ConditionalOnClass(WebSocketHandler.class)
@ConditionalOnProperty(prefix = "zuul.ws", name = "enabled", havingValue = "true", matchIfMissing = true)
@EnableConfigurationProperties(ZuulWebSocketProperties.class)
public class ZuulWebSocketConfiguration extends AbstractWebSocketMessageBrokerConfigurer {
	@Autowired
	ZuulWebSocketProperties zuulWebSocketProperties;
	@Autowired
	SimpMessagingTemplate messagingTemplate;
	@Autowired
	ZuulProperties zuulProperties;
	@Autowired
	ZuulPropertiesResolver zuulPropertiesResolver;
	@Autowired
	ProxyWebSocketErrorHandler proxyWebSocketErrorHandler;
	@Autowired
	WebSocketStompClient stompClient;
	@Autowired
	WebSocketHttpHeadersCallback webSocketHttpHeadersCallback;

	@Override
	public void registerStompEndpoints(StompEndpointRegistry registry) {
		for (Map.Entry entry : zuulWebSocketProperties
				.getBrokerages().entrySet()) {
			ZuulWebSocketProperties.WsBrokerage wsBrokerage = entry.getValue();
			if (wsBrokerage.isEnabled()) {
				registry.addEndpoint(wsBrokerage.getEndPoints())
						// bypasses spring web security
						.setAllowedOrigins("*").withSockJS();
			}
		}
	}

	@Override
	public void configureMessageBroker(MessageBrokerRegistry config) {
		// prefix for subscribe
		for (Map.Entry entry : zuulWebSocketProperties
				.getBrokerages().entrySet()) {
			ZuulWebSocketProperties.WsBrokerage wsBrokerage = entry.getValue();
			if (wsBrokerage.isEnabled()) {
				config.enableSimpleBroker(
						mergeBrokersWithApplicationDestinationPrefixes(wsBrokerage));
				// prefix for send
				config.setApplicationDestinationPrefixes(
						wsBrokerage.getDestinationPrefixes());
			}
		}
	}

	private String[] mergeBrokersWithApplicationDestinationPrefixes(
			ZuulWebSocketProperties.WsBrokerage wsBrokerage) {
		List brokers = new ArrayList<>(Arrays.asList(wsBrokerage.getBrokers()));

		for (String adp : wsBrokerage.getDestinationPrefixes()) {
			if (!brokers.contains(adp)) {
				brokers.add(adp);
			}
		}

		return brokers.toArray(new String[brokers.size()]);
	}

	@Override
	public void configureWebSocketTransport(WebSocketTransportRegistration registration) {
		registration.addDecoratorFactory(new WebSocketHandlerDecoratorFactory() {
			@Override
			public WebSocketHandler decorate(WebSocketHandler handler) {
				ProxyWebSocketHandler proxyWebSocketHandler = new ProxyWebSocketHandler(
						handler, stompClient, webSocketHttpHeadersCallback,
						messagingTemplate, zuulPropertiesResolver,
						zuulWebSocketProperties);
				proxyWebSocketHandler.errorHandler(proxyWebSocketErrorHandler);
				return proxyWebSocketHandler;
			}
		});
	}

	@Bean
	@ConditionalOnMissingBean(WebSocketHttpHeadersCallback.class)
	public WebSocketHttpHeadersCallback webSocketHttpHeadersCallback() {
		return new WebSocketHttpHeadersCallback() {
			@Override
			public WebSocketHttpHeaders getWebSocketHttpHeaders() {
				return new WebSocketHttpHeaders();
			}
		};
	}

	@Bean
	@ConditionalOnMissingBean
	public ZuulPropertiesResolver zuulPropertiesResolver(
			final ZuulProperties zuulProperties) {
		return new ZuulPropertiesResolver() {
			@Override
			public String getRouteHost(ZuulWebSocketProperties.WsBrokerage wsBrokerage) {
				return zuulProperties.getRoutes().get(wsBrokerage.getId()).getUrl();
			}
		};
	}

	@Bean
	@ConditionalOnMissingBean(WebSocketStompClient.class)
	public WebSocketStompClient stompClient(MessageConverter messageConverter,
			ThreadPoolTaskScheduler taskScheduler) {
		int bufferSizeLimit = 1024 * 1024 * 8;

		StandardWebSocketClient webSocketClient = new StandardWebSocketClient();
		List transports = new ArrayList<>();
		transports.add(new WebSocketTransport(webSocketClient));
		SockJsClient sockJsClient = new SockJsClient(transports);
		WebSocketStompClient client = new WebSocketStompClient(sockJsClient);
		client.setInboundMessageSizeLimit(bufferSizeLimit);
		client.setMessageConverter(messageConverter);
		client.setTaskScheduler(taskScheduler);
		client.setDefaultHeartbeat(new long[] { 0, 0 });
		return client;
	}

	@Bean
	@ConditionalOnMissingBean(TaskScheduler.class)
	public TaskScheduler stompClientTaskScheduler() {
		return new ThreadPoolTaskScheduler();
	}

	@Bean
	@ConditionalOnMissingBean(ProxyWebSocketErrorHandler.class)
	public ProxyWebSocketErrorHandler proxyWebSocketErrorHandler() {
		return new DefaultProxyWebSocketErrorHandler();
	}

	@PostConstruct
	public void init() {
		ignorePattern("**/websocket");
		ignorePattern("**/info");
	}

	private void ignorePattern(String ignoredPattern) {
		for (String pattern : zuulProperties.getIgnoredPatterns()) {
			if (pattern.toLowerCase().contains(ignoredPattern))
				return;
		}

		zuulProperties.getIgnoredPatterns().add(ignoredPattern);
	}

	/**
	 * An {@link ErrorHandler} implementation that logs the Throwable at error level. It
	 * does not perform any additional error handling. This can be useful when suppression
	 * of errors is the intended behavior.
	 */
	private static class DefaultProxyWebSocketErrorHandler
			implements ProxyWebSocketErrorHandler {

		private final Log logger = LogFactory
				.getLog(DefaultProxyWebSocketErrorHandler.class);

		@Override
		public void handleError(Throwable t) {
			if (logger.isErrorEnabled()) {
				logger.error("Proxy web socket error occurred.", t);
			}
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy