netty5/codec-redis/src/main/java/io/netty/handler/codec/redis/RedisArrayAggregator.java

94 lines
3.3 KiB
Java
Raw Normal View History

/*
* Copyright 2016 The Netty Project
*
* The Netty Project 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 io.netty.handler.codec.redis;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.CodecException;
import io.netty.handler.codec.MessageToMessageDecoder;
import io.netty.util.ReferenceCountUtil;
import io.netty.util.internal.UnstableApi;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Deque;
import java.util.List;
/**
* Aggregates {@link RedisMessage} parts into {@link ArrayRedisMessage}. This decoder
* should be used together with {@link RedisDecoder}.
*/
@UnstableApi
public final class RedisArrayAggregator extends MessageToMessageDecoder<RedisMessage> {
private final Deque<AggregateState> depths = new ArrayDeque<>(4);
@Override
protected void decode(ChannelHandlerContext ctx, RedisMessage msg) throws Exception {
if (msg instanceof ArrayHeaderRedisMessage) {
msg = decodeRedisArrayHeader((ArrayHeaderRedisMessage) msg);
if (msg == null) {
return;
}
} else {
ReferenceCountUtil.retain(msg);
}
while (!depths.isEmpty()) {
AggregateState current = depths.peek();
current.children.add(msg);
// if current aggregation completed, go to parent aggregation.
if (current.children.size() == current.length) {
msg = new ArrayRedisMessage(current.children);
depths.pop();
} else {
// not aggregated yet. try next time.
return;
}
}
ctx.fireChannelRead(msg);
}
private RedisMessage decodeRedisArrayHeader(ArrayHeaderRedisMessage header) {
if (header.isNull()) {
return ArrayRedisMessage.NULL_INSTANCE;
} else if (header.length() == 0L) {
return ArrayRedisMessage.EMPTY_INSTANCE;
} else if (header.length() > 0L) {
// Currently, this codec doesn't support `long` length for arrays because Java's List.size() is int.
if (header.length() > Integer.MAX_VALUE) {
throw new CodecException("this codec doesn't support longer length than " + Integer.MAX_VALUE);
}
// start aggregating array
depths.push(new AggregateState((int) header.length()));
return null;
} else {
throw new CodecException("bad length: " + header.length());
}
}
private static final class AggregateState {
private final int length;
private final List<RedisMessage> children;
AggregateState(int length) {
this.length = length;
this.children = new ArrayList<>(length);
}
}
}