ClientWebSocketInterceptor.java
/*
* Copyright 2017-2020 original 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
*
* https://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 io.micronaut.websocket.interceptor;
import io.micronaut.aop.InterceptedMethod;
import io.micronaut.aop.MethodInterceptor;
import io.micronaut.aop.MethodInvocationContext;
import io.micronaut.context.annotation.Prototype;
import io.micronaut.core.convert.ConversionService;
import io.micronaut.core.util.ArrayUtils;
import io.micronaut.http.MediaType;
import io.micronaut.http.annotation.Produces;
import io.micronaut.websocket.WebSocketSession;
import io.micronaut.websocket.exceptions.WebSocketClientException;
import java.io.Closeable;
/**
* Intercepts unimplemented {@link io.micronaut.websocket.annotation.ClientWebSocket} methods.
*
* @author graemerocher
* @since 1.0
*/
@Prototype
public class ClientWebSocketInterceptor implements MethodInterceptor<Object, Object> {
private final ConversionService conversionService;
private WebSocketSession webSocketSession;
/**
* @param conversionService The conversion service
*/
public ClientWebSocketInterceptor(ConversionService conversionService) {
this.conversionService = conversionService;
}
@Override
public Object intercept(MethodInvocationContext<Object, Object> context) {
Class<?> declaringType = context.getDeclaringType();
if (declaringType == WebSocketSessionAware.class) {
Object[] values = context.getParameterValues();
if (ArrayUtils.isNotEmpty(values)) {
Object o = values[0];
if (o instanceof WebSocketSession session) {
this.webSocketSession = session;
return null;
}
}
}
if (declaringType == Closeable.class || declaringType == AutoCloseable.class) {
// must be close method
if (webSocketSession != null) {
webSocketSession.close();
}
return null;
} else {
String methodName = context.getMethodName();
if (methodName.startsWith("send") || methodName.startsWith("broadcast")) {
MediaType mediaType = context.stringValue(Produces.class).map(MediaType::of).orElse(MediaType.APPLICATION_JSON_TYPE);
validateSession();
InterceptedMethod interceptedMethod = InterceptedMethod.of(context, conversionService);
Class<?> javaReturnType = context.getReturnType().getType();
if (interceptedMethod.resultType() == InterceptedMethod.ResultType.SYNCHRONOUS && javaReturnType != void.class) {
return context.proceed();
}
try {
Object[] parameterValues = context.getParameterValues();
switch (parameterValues.length) {
case 0 -> throw new IllegalArgumentException("At least 1 parameter is required to a send method");
case 1 -> {
Object value = parameterValues[0];
if (value == null) {
throw new IllegalArgumentException("Parameter cannot be null");
}
return send(interceptedMethod, value, mediaType);
}
default -> {
return send(interceptedMethod, context.getParameterValueMap(), mediaType);
}
}
} catch (Exception e) {
return interceptedMethod.handleException(e);
}
}
}
return context.proceed();
}
private Object send(InterceptedMethod interceptedMethod, Object message, MediaType mediaType) {
switch (interceptedMethod.resultType()) {
case COMPLETION_STAGE -> {
return interceptedMethod.handleResult(webSocketSession.sendAsync(message, mediaType));
}
case PUBLISHER -> {
return interceptedMethod.handleResult(webSocketSession.send(message, mediaType));
}
case SYNCHRONOUS -> {
webSocketSession.sendSync(message, mediaType);
return null;
}
default -> {
return interceptedMethod.unsupported();
}
}
}
private void validateSession() {
if (webSocketSession == null || !webSocketSession.isOpen()) {
throw new WebSocketClientException("No available and open WebSocket session");
}
}
}