io.undertow.server.ConnectorStatisticsImpl Maven / Gradle / Ivy
/*
* JBoss, Home of Professional Open Source.
* Copyright 2014 Red Hat, Inc., and individual contributors
* as indicated by the @author tags.
*
* 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 io.undertow.server;
import io.undertow.conduits.ByteActivityCallback;
import io.undertow.util.StatusCodes;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
/**
* @author Stuart Douglas
*/
public class ConnectorStatisticsImpl implements ConnectorStatistics {
private static final AtomicLongFieldUpdater requestCountUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "requestCount");
private static final AtomicLongFieldUpdater bytesSentUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "bytesSent");
private static final AtomicLongFieldUpdater bytesReceivedUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "bytesReceived");
private static final AtomicLongFieldUpdater errorCountUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "errorCount");
private static final AtomicLongFieldUpdater processingTimeUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "processingTime");
private static final AtomicLongFieldUpdater maxProcessingTimeUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "maxProcessingTime");
private static final AtomicLongFieldUpdater activeConnectionsUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "activeConnections");
private static final AtomicLongFieldUpdater maxActiveConnectionsUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "maxActiveConnections");
private static final AtomicLongFieldUpdater activeRequestsUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "activeRequests");
private static final AtomicLongFieldUpdater maxActiveRequestsUpdater = AtomicLongFieldUpdater.newUpdater(ConnectorStatisticsImpl.class, "maxActiveRequests");
private volatile long requestCount;
private volatile long bytesSent;
private volatile long bytesReceived;
private volatile long errorCount;
private volatile long processingTime;
private volatile long maxProcessingTime;
private volatile long activeConnections;
private volatile long maxActiveConnections;
private volatile long activeRequests;
private volatile long maxActiveRequests;
private final ExchangeCompletionListener completionListener = new ExchangeCompletionListener() {
@Override
public void exchangeEvent(HttpServerExchange exchange, NextListener nextListener) {
try {
activeRequestsUpdater.decrementAndGet(ConnectorStatisticsImpl.this);
if (exchange.getStatusCode() == StatusCodes.INTERNAL_SERVER_ERROR) {
errorCountUpdater.incrementAndGet(ConnectorStatisticsImpl.this);
}
long start = exchange.getRequestStartTime();
if (start > 0) {
long elapsed = System.nanoTime() - start;
processingTimeUpdater.addAndGet(ConnectorStatisticsImpl.this, elapsed);
long oldMax;
do {
oldMax = maxProcessingTimeUpdater.get(ConnectorStatisticsImpl.this);
if (oldMax >= elapsed) {
break;
}
} while (!maxProcessingTimeUpdater.compareAndSet(ConnectorStatisticsImpl.this, oldMax, elapsed));
}
} finally {
nextListener.proceed();
}
}
};
private final ByteActivityCallback bytesSentAccumulator = new BytesSentAccumulator();
private final ByteActivityCallback bytesReceivedAccumulator = new BytesReceivedAccumulator();
@Override
public long getRequestCount() {
return requestCountUpdater.get(this);
}
@Override
public long getBytesSent() {
return bytesSentUpdater.get(this);
}
@Override
public long getBytesReceived() {
return bytesReceivedUpdater.get(this);
}
@Override
public long getErrorCount() {
return errorCountUpdater.get(this);
}
@Override
public long getProcessingTime() {
return processingTimeUpdater.get(this);
}
@Override
public long getMaxProcessingTime() {
return maxProcessingTimeUpdater.get(this);
}
@Override
public void reset() {
requestCountUpdater.set(this, 0);
bytesSentUpdater.set(this, 0);
bytesReceivedUpdater.set(this, 0);
errorCountUpdater.set(this, 0);
maxProcessingTimeUpdater.set(this, 0);
processingTimeUpdater.set(this, 0);
maxActiveConnectionsUpdater.set(this, 0);
maxActiveRequestsUpdater.set(this, 0);
//we don't update active requests or connections, as these will still be live
}
public void requestFinished(long bytesSent, long bytesReceived, boolean error) {
bytesSentUpdater.addAndGet(this, bytesSent);
bytesReceivedUpdater.addAndGet(this, bytesReceived);
if (error) {
errorCountUpdater.incrementAndGet(this);
}
}
public void updateBytesSent(long bytes) {
bytesSentUpdater.addAndGet(this, bytes);
}
public void updateBytesReceived(long bytes) {
bytesReceivedUpdater.addAndGet(this, bytes);
}
public void setup(HttpServerExchange exchange) {
requestCountUpdater.incrementAndGet(this);
long current = activeRequestsUpdater.incrementAndGet(this);
long maxActiveRequests;
do {
maxActiveRequests = this.maxActiveRequests;
if(current <= maxActiveRequests) {
break;
}
} while (!maxActiveRequestsUpdater.compareAndSet(this, maxActiveRequests, current));
exchange.addExchangeCompleteListener(completionListener);
}
public ByteActivityCallback sentAccumulator() {
return bytesSentAccumulator;
}
public ByteActivityCallback receivedAccumulator() {
return bytesReceivedAccumulator;
}
//todo: we can do a way
private class BytesSentAccumulator implements ByteActivityCallback {
@Override
public void activity(long bytes) {
bytesSentUpdater.addAndGet(ConnectorStatisticsImpl.this, bytes);
}
}
private class BytesReceivedAccumulator implements ByteActivityCallback {
@Override
public void activity(long bytes) {
bytesReceivedUpdater.addAndGet(ConnectorStatisticsImpl.this, bytes);
}
}
@Override
public long getActiveConnections() {
return activeConnections;
}
@Override
public long getMaxActiveConnections() {
return maxActiveConnections;
}
public void incrementConnectionCount() {
long current = activeConnectionsUpdater.incrementAndGet(this);
long maxActiveConnections;
do {
maxActiveConnections = this.maxActiveConnections;
if(current <= maxActiveConnections) {
return;
}
} while (!maxActiveConnectionsUpdater.compareAndSet(this, maxActiveConnections, current));
}
public void decrementConnectionCount() {
activeConnectionsUpdater.decrementAndGet(this);
}
@Override
public long getActiveRequests() {
return activeRequests;
}
@Override
public long getMaxActiveRequests() {
return maxActiveRequests;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy