org.apache.eventmesh.runtime.boot.AbstractHTTPServer Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.eventmesh.runtime.boot;
import org.apache.eventmesh.common.Constants;
import org.apache.eventmesh.common.ThreadPoolFactory;
import org.apache.eventmesh.common.protocol.http.HttpCommand;
import org.apache.eventmesh.common.protocol.http.HttpEventWrapper;
import org.apache.eventmesh.common.protocol.http.body.Body;
import org.apache.eventmesh.common.protocol.http.common.EventMeshRetCode;
import org.apache.eventmesh.common.protocol.http.common.ProtocolKey;
import org.apache.eventmesh.common.protocol.http.common.ProtocolVersion;
import org.apache.eventmesh.common.protocol.http.common.RequestCode;
import org.apache.eventmesh.common.protocol.http.header.Header;
import org.apache.eventmesh.common.utils.AssertUtils;
import org.apache.eventmesh.common.utils.JsonUtils;
import org.apache.eventmesh.runtime.common.Pair;
import org.apache.eventmesh.runtime.configuration.EventMeshHTTPConfiguration;
import org.apache.eventmesh.runtime.constants.EventMeshConstants;
import org.apache.eventmesh.runtime.core.protocol.http.async.AsyncContext;
import org.apache.eventmesh.runtime.core.protocol.http.processor.HandlerService;
import org.apache.eventmesh.runtime.core.protocol.http.processor.inf.EventProcessor;
import org.apache.eventmesh.runtime.core.protocol.http.processor.inf.HttpRequestProcessor;
import org.apache.eventmesh.runtime.metrics.http.HTTPMetricsServer;
import org.apache.eventmesh.runtime.trace.TraceUtils;
import org.apache.eventmesh.runtime.util.RemotingHelper;
import org.apache.eventmesh.runtime.util.Utils;
import org.apache.eventmesh.trace.api.common.EventMeshTraceConstants;
import org.apache.commons.collections4.MapUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.http.MethodNotSupportedException;
import org.apache.http.entity.ContentType;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.RejectedExecutionException;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLEngine;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelDuplexHandler;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.http.DefaultFullHttpResponse;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.FullHttpResponse;
import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpHeaderValues;
import io.netty.handler.codec.http.HttpHeaders;
import io.netty.handler.codec.http.HttpMethod;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpRequest;
import io.netty.handler.codec.http.HttpRequestDecoder;
import io.netty.handler.codec.http.HttpResponseEncoder;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.HttpVersion;
import io.netty.handler.codec.http.QueryStringDecoder;
import io.netty.handler.codec.http.multipart.Attribute;
import io.netty.handler.codec.http.multipart.DefaultHttpDataFactory;
import io.netty.handler.codec.http.multipart.DiskAttribute;
import io.netty.handler.codec.http.multipart.HttpPostRequestDecoder;
import io.netty.handler.codec.http.multipart.InterfaceHttpData;
import io.netty.handler.ssl.SslHandler;
import io.netty.handler.timeout.IdleState;
import io.netty.handler.timeout.IdleStateEvent;
import io.netty.util.ReferenceCountUtil;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.semconv.trace.attributes.SemanticAttributes;
import com.fasterxml.jackson.core.type.TypeReference;
import lombok.extern.slf4j.Slf4j;
@Slf4j
public abstract class AbstractHTTPServer extends AbstractRemotingServer {
private HandlerService handlerService;
private HTTPMetricsServer metrics;
private static final DefaultHttpDataFactory DEFAULT_HTTP_DATA_FACTORY = new DefaultHttpDataFactory(false);
private final transient AtomicBoolean started = new AtomicBoolean(false);
private final transient boolean useTLS;
private Boolean useTrace = false; //Determine whether trace is enabled
private final transient EventMeshHTTPConfiguration eventMeshHttpConfiguration;
private final transient ThreadPoolExecutor asyncContextCompleteHandler =
ThreadPoolFactory.createThreadPoolExecutor(10, 10, "EventMesh-http-asyncContext-");
private static final int MAX_CONNECTIONS = 20_000;
static {
DiskAttribute.deleteOnExitTemporaryFile = false;
}
protected final transient Map>
processorTable = new ConcurrentHashMap<>(64);
protected final transient Map>
eventProcessorTable = new ConcurrentHashMap<>(64);
public AbstractHTTPServer(final int port, final boolean useTLS,
final EventMeshHTTPConfiguration eventMeshHttpConfiguration) {
super();
this.setPort(port);
this.useTLS = useTLS;
this.eventMeshHttpConfiguration = eventMeshHttpConfiguration;
}
public void setUseTrace(final Boolean useTrace) {
this.useTrace = useTrace;
}
public Boolean getUseTrace() {
return useTrace;
}
public void setHandlerService(final HandlerService handlerService) {
this.handlerService = handlerService;
}
public HTTPMetricsServer getMetrics() {
return metrics;
}
public void setMetrics(final HTTPMetricsServer metrics) {
this.metrics = metrics;
}
public HandlerService getHandlerService() {
return handlerService;
}
public void sendError(final ChannelHandlerContext ctx, final HttpResponseStatus status) {
final FullHttpResponse response = new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, status);
final HttpHeaders responseHeaders = response.headers();
responseHeaders.add(
HttpHeaderNames.CONTENT_TYPE, String.format("text/plain; charset=%s", EventMeshConstants.DEFAULT_CHARSET)
);
responseHeaders.add(HttpHeaderNames.CONTENT_LENGTH, response.content().readableBytes());
responseHeaders.add(HttpHeaderNames.CONNECTION, HttpHeaderValues.KEEP_ALIVE);
ctx.writeAndFlush(response).addListener(ChannelFutureListener.CLOSE);
}
public void sendResponse(final ChannelHandlerContext ctx, final DefaultFullHttpResponse response) {
ctx.writeAndFlush(response).addListener((ChannelFutureListener) f -> {
if (!f.isSuccess()) {
if (log.isWarnEnabled()) {
log.warn("send response to [{}] fail, will close this channel",
RemotingHelper.parseChannelRemoteAddr(f.channel()));
}
f.channel().close();
}
});
}
@Override
public void start() throws Exception {
final Runnable r = () -> {
final ServerBootstrap b = new ServerBootstrap();
try {
b.group(this.getBossGroup(), this.getWorkerGroup())
.channel(NioServerSocketChannel.class)
.childHandler(new HttpsServerInitializer(
useTLS ? SSLContextFactory.getSslContext(eventMeshHttpConfiguration) : null))
.childOption(ChannelOption.SO_KEEPALIVE, Boolean.TRUE);
if (log.isInfoEnabled()) {
log.info("HTTPServer[port={}] started.", this.getPort());
}
b.bind(this.getPort())
.channel()
.closeFuture()
.sync();
} catch (Exception e) {
log.error("HTTPServer start error!", e);
try {
shutdown();
} catch (Exception ex) {
log.error("HTTPServer shutdown error!", ex);
}
}
};
final Thread t = new Thread(r, "EventMesh-http-server");
t.setDaemon(true);
t.start();
started.compareAndSet(false, true);
}
@Override
public void shutdown() throws Exception {
super.shutdown();
started.compareAndSet(true, false);
}
public void registerProcessor(final Integer requestCode, final HttpRequestProcessor processor,
final ThreadPoolExecutor executor) {
AssertUtils.notNull(requestCode, "requestCode can't be null");
AssertUtils.notNull(processor, "processor can't be null");
AssertUtils.notNull(executor, "executor can't be null");
this.processorTable.put(requestCode.toString(), new Pair<>(processor, executor));
}
public void registerProcessor(final String requestURI, final EventProcessor processor,
final ThreadPoolExecutor executor) {
AssertUtils.notNull(requestURI, "requestURI can't be null");
AssertUtils.notNull(processor, "processor can't be null");
AssertUtils.notNull(executor, "executor can't be null");
this.eventProcessorTable.put(requestURI, new Pair<>(processor, executor));
}
/**
* Validate request, return error status.
*
* @param httpRequest
* @return if request is validated return null else return error status
*/
private HttpResponseStatus validateHttpRequest(final HttpRequest httpRequest) {
if (!started.get()) {
return HttpResponseStatus.SERVICE_UNAVAILABLE;
}
if (!httpRequest.decoderResult().isSuccess()) {
return HttpResponseStatus.BAD_REQUEST;
}
if (!HttpMethod.GET.equals(httpRequest.method()) && !HttpMethod.POST.equals(httpRequest.method())) {
return HttpResponseStatus.METHOD_NOT_ALLOWED;
}
if (!ProtocolVersion.contains(httpRequest.headers().get(ProtocolKey.VERSION))) {
return HttpResponseStatus.BAD_REQUEST;
}
return null;
}
/**
* Inject ip and protocol version, if the protocol version is empty, set default to {@link ProtocolVersion#V1}.
*
* @param ctx
* @param httpRequest
*/
private void preProcessHttpRequestHeader(final ChannelHandlerContext ctx, final HttpRequest httpRequest) {
final long startTime = System.currentTimeMillis();
final HttpHeaders requestHeaders = httpRequest.headers();
requestHeaders.set(EventMeshConstants.REQ_C2EVENTMESH_TIMESTAMP, startTime);
if (StringUtils.isBlank(requestHeaders.get(ProtocolKey.VERSION))) {
requestHeaders.set(ProtocolKey.VERSION, ProtocolVersion.V1.getVersion());
}
requestHeaders.set(ProtocolKey.ClientInstanceKey.IP,
RemotingHelper.parseChannelRemoteAddr(ctx.channel()));
requestHeaders.set(EventMeshConstants.REQ_SEND_EVENTMESH_IP, eventMeshHttpConfiguration.getEventMeshServerIp());
}
/**
* Parse request body to map
*
* @param httpRequest
* @return
*/
private Map parseHttpRequestBody(final HttpRequest httpRequest) throws IOException {
final long bodyDecodeStart = System.currentTimeMillis();
final Map httpRequestBody = new HashMap<>();
if (HttpMethod.GET.equals(httpRequest.method())) {
new QueryStringDecoder(httpRequest.uri())
.parameters()
.forEach((key, value) -> httpRequestBody.put(key, value.get(0)));
} else if (HttpMethod.POST.equals(httpRequest.method())) {
final HttpPostRequestDecoder decoder =
new HttpPostRequestDecoder(DEFAULT_HTTP_DATA_FACTORY, httpRequest);
for (final InterfaceHttpData parm : decoder.getBodyHttpDatas()) {
if (InterfaceHttpData.HttpDataType.Attribute == parm.getHttpDataType()) {
final Attribute data = (Attribute) parm;
httpRequestBody.put(data.getName(), data.getValue());
}
}
decoder.destroy();
}
metrics.getSummaryMetrics().recordDecodeTimeCost(System.currentTimeMillis() - bodyDecodeStart);
return httpRequestBody;
}
private class HTTPHandler extends ChannelInboundHandlerAdapter {
@Override
public void channelRead(final ChannelHandlerContext ctx, final Object message) {
final HttpRequest httpRequest = (HttpRequest) message;
if (httpRequest == null) {
return;
}
if (Objects.nonNull(handlerService) && handlerService.isProcessorWrapper(httpRequest)) {
handlerService.handler(ctx, httpRequest, asyncContextCompleteHandler);
return;
}
try {
Span span = null;
preProcessHttpRequestHeader(ctx, httpRequest);
final Map headerMap = Utils.parseHttpHeader(httpRequest);
final HttpResponseStatus errorStatus = validateHttpRequest(httpRequest);
if (errorStatus != null) {
sendError(ctx, errorStatus);
span = TraceUtils.prepareServerSpan(headerMap,
EventMeshTraceConstants.TRACE_UPSTREAM_EVENTMESH_SERVER_SPAN, false);
TraceUtils.finishSpanWithException(span, headerMap, errorStatus.reasonPhrase(), null);
return;
}
metrics.getSummaryMetrics().recordHTTPRequest();
boolean useRequestURI = false;
for (final String processURI : eventProcessorTable.keySet()) {
if (httpRequest.uri().startsWith(processURI)) {
useRequestURI = true;
break;
}
}
if (useRequestURI) {
if (useTrace) {
span.setAttribute(SemanticAttributes.HTTP_METHOD,
httpRequest.method() == null ? "" : httpRequest.method().name());
span.setAttribute(SemanticAttributes.HTTP_FLAVOR,
httpRequest.protocolVersion() == null ? "" : httpRequest.protocolVersion().protocolName());
span.setAttribute(SemanticAttributes.HTTP_URL,
httpRequest.uri());
}
final HttpEventWrapper httpEventWrapper = parseHttpRequest(httpRequest);
final AsyncContext asyncContext =
new AsyncContext<>(httpEventWrapper, null, asyncContextCompleteHandler);
processHttpRequest(ctx, asyncContext);
} else {
final HttpCommand requestCommand = new HttpCommand();
final Map bodyMap = parseHttpRequestBody(httpRequest);
final String requestCode = HttpMethod.POST.equals(httpRequest.method())
? httpRequest.headers().get(ProtocolKey.REQUEST_CODE)
: MapUtils.getString(bodyMap, StringUtils.lowerCase(ProtocolKey.REQUEST_CODE), "");
requestCommand.setHttpMethod(httpRequest.method().name());
requestCommand.setHttpVersion(httpRequest.protocolVersion() == null ? ""
: httpRequest.protocolVersion().protocolName());
requestCommand.setRequestCode(requestCode);
HttpCommand responseCommand = null;
if (StringUtils.isBlank(requestCode)
|| !processorTable.containsKey(requestCode)
|| !RequestCode.contains(Integer.valueOf(requestCode))) {
responseCommand =
requestCommand.createHttpCommandResponse(EventMeshRetCode.EVENTMESH_REQUESTCODE_INVALID);
sendResponse(ctx, responseCommand.httpResponse());
span = TraceUtils.prepareServerSpan(headerMap,
EventMeshTraceConstants.TRACE_UPSTREAM_EVENTMESH_SERVER_SPAN, false);
TraceUtils.finishSpanWithException(span, headerMap,
EventMeshRetCode.EVENTMESH_REQUESTCODE_INVALID.getErrMsg(), null);
return;
}
try {
requestCommand.setHeader(Header.buildHeader(requestCode, headerMap));
requestCommand.setBody(Body.buildBody(requestCode, bodyMap));
} catch (Exception e) {
responseCommand = requestCommand.createHttpCommandResponse(EventMeshRetCode.EVENTMESH_RUNTIME_ERR);
sendResponse(ctx, responseCommand.httpResponse());
span = TraceUtils.prepareServerSpan(headerMap,
EventMeshTraceConstants.TRACE_UPSTREAM_EVENTMESH_SERVER_SPAN, false);
TraceUtils.finishSpanWithException(span, headerMap,
EventMeshRetCode.EVENTMESH_RUNTIME_ERR.getErrMsg(), e);
return;
}
if (log.isDebugEnabled()) {
log.debug("{}", requestCommand);
}
final AsyncContext asyncContext =
new AsyncContext<>(requestCommand, responseCommand, asyncContextCompleteHandler);
processEventMeshRequest(ctx, asyncContext);
}
} catch (Exception ex) {
log.error("AbrstractHTTPServer.HTTPHandler.channelRead error", ex);
} finally {
ReferenceCountUtil.release(message);
}
}
public void processHttpRequest(final ChannelHandlerContext ctx,
final AsyncContext asyncContext) {
final HttpEventWrapper requestWrapper = asyncContext.getRequest();
final String requestURI = requestWrapper.getRequestURI();
String processorKey = "/";
for (final String eventProcessorKey : eventProcessorTable.keySet()) {
if (requestURI.startsWith(eventProcessorKey)) {
processorKey = eventProcessorKey;
break;
}
}
final Pair choosed = eventProcessorTable.get(processorKey);
try {
choosed.getObject2().submit(() -> {
try {
final EventProcessor processor = choosed.getObject1();
if (processor.rejectRequest()) {
final HttpEventWrapper responseWrapper =
requestWrapper.createHttpResponse(EventMeshRetCode.EVENTMESH_REJECT_BY_PROCESSOR_ERROR);
asyncContext.onComplete(responseWrapper);
if (asyncContext.isComplete()) {
if (log.isDebugEnabled()) {
log.debug("{}", asyncContext.getResponse());
}
sendResponse(ctx, asyncContext.getResponse().httpResponse());
}
return;
}
processor.processRequest(ctx, asyncContext);
if (!asyncContext.isComplete()) {
return;
}
metrics.getSummaryMetrics()
.recordHTTPReqResTimeCost(System.currentTimeMillis() - requestWrapper.getReqTime());
if (log.isDebugEnabled()) {
log.debug("{}", asyncContext.getResponse());
}
sendResponse(ctx, asyncContext.getResponse().httpResponse());
} catch (Exception e) {
log.error("process error", e);
}
});
} catch (RejectedExecutionException re) {
final HttpEventWrapper responseWrapper = requestWrapper.createHttpResponse(EventMeshRetCode.OVERLOAD);
asyncContext.onComplete(responseWrapper);
metrics.getSummaryMetrics().recordHTTPDiscard();
metrics.getSummaryMetrics().recordHTTPReqResTimeCost(
System.currentTimeMillis() - requestWrapper.getReqTime());
try {
sendResponse(ctx, asyncContext.getResponse().httpResponse());
} catch (Exception e) {
log.error("sendResponse error", e);
}
}
}
public void processEventMeshRequest(final ChannelHandlerContext ctx,
final AsyncContext asyncContext) {
final HttpCommand request = asyncContext.getRequest();
final Pair choosed = processorTable.get(request.getRequestCode());
try {
choosed.getObject2().submit(() -> {
try {
final HttpRequestProcessor processor = choosed.getObject1();
if (processor.rejectRequest()) {
final HttpCommand responseCommand =
request.createHttpCommandResponse(EventMeshRetCode.EVENTMESH_REJECT_BY_PROCESSOR_ERROR);
asyncContext.onComplete(responseCommand);
if (asyncContext.isComplete()) {
sendResponse(ctx, responseCommand.httpResponse());
if (log.isDebugEnabled()) {
log.debug("{}", asyncContext.getResponse());
}
final Map traceMap = asyncContext.getRequest().getHeader().toMap();
TraceUtils.finishSpanWithException(TraceUtils.prepareServerSpan(traceMap,
EventMeshTraceConstants.TRACE_UPSTREAM_EVENTMESH_SERVER_SPAN,
false),
traceMap,
EventMeshRetCode.EVENTMESH_REJECT_BY_PROCESSOR_ERROR.getErrMsg(), null);
}
return;
}
processor.processRequest(ctx, asyncContext);
if (!asyncContext.isComplete()) {
return;
}
metrics.getSummaryMetrics()
.recordHTTPReqResTimeCost(System.currentTimeMillis() - request.getReqTime());
if (log.isDebugEnabled()) {
log.debug("{}", asyncContext.getResponse());
}
sendResponse(ctx, asyncContext.getResponse().httpResponse());
} catch (Exception e) {
log.error("process error", e);
}
});
} catch (RejectedExecutionException re) {
asyncContext.onComplete(request.createHttpCommandResponse(EventMeshRetCode.OVERLOAD));
metrics.getSummaryMetrics().recordHTTPDiscard();
metrics.getSummaryMetrics().recordHTTPReqResTimeCost(System.currentTimeMillis() - request.getReqTime());
try {
sendResponse(ctx, asyncContext.getResponse().httpResponse());
final Map traceMap = asyncContext.getRequest().getHeader().toMap();
TraceUtils.finishSpanWithException(
TraceUtils.prepareServerSpan(traceMap,
EventMeshTraceConstants.TRACE_UPSTREAM_EVENTMESH_SERVER_SPAN,
false),
traceMap,
EventMeshRetCode.EVENTMESH_RUNTIME_ERR.getErrMsg(),
re);
} catch (Exception e) {
log.error("processEventMeshRequest fail", re);
}
}
}
@Override
public void channelReadComplete(final ChannelHandlerContext ctx) throws Exception {
super.channelReadComplete(ctx);
ctx.flush();
}
@Override
public void exceptionCaught(final ChannelHandlerContext ctx, final Throwable cause) {
if (null != cause) {
log.error("", cause);
}
if (null != ctx) {
ctx.close();
}
}
Map extractFromRequest(final HttpRequest httpRequest) {
return new HashMap<>();
}
}
private HttpEventWrapper parseHttpRequest(final HttpRequest httpRequest) throws IOException, MethodNotSupportedException {
final HttpEventWrapper httpEventWrapper = new HttpEventWrapper();
httpEventWrapper.setHttpMethod(httpRequest.method().name());
httpEventWrapper.setHttpVersion(httpRequest.protocolVersion().protocolName());
httpEventWrapper.setRequestURI(httpRequest.uri());
//parse http header
for (final String key : httpRequest.headers().names()) {
httpEventWrapper.getHeaderMap().put(key, httpRequest.headers().get(key));
}
final long bodyDecodeStart = System.currentTimeMillis();
final FullHttpRequest fullHttpRequest = (FullHttpRequest) httpRequest;
final Map bodyMap = new HashMap<>();
if (HttpMethod.GET.equals(fullHttpRequest.method())) {
new QueryStringDecoder(fullHttpRequest.uri()).parameters().forEach((key, value) -> bodyMap.put(key, value.get(0)));
} else if (HttpMethod.POST.equals(fullHttpRequest.method())) {
if (StringUtils.contains(httpRequest.headers().get("Content-Type"),
ContentType.APPLICATION_JSON.getMimeType())) {
final int length = fullHttpRequest.content().readableBytes();
if (length > 0) {
final byte[] body = new byte[length];
fullHttpRequest.content().readBytes(body);
bodyMap.putAll(Objects.requireNonNull(JsonUtils.deserialize(new String(body, Constants.DEFAULT_CHARSET),
new TypeReference
© 2015 - 2025 Weber Informatics LLC | Privacy Policy