Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
swim.mqtt.MqttPublishPacketEncoder Maven / Gradle / Ivy
Go to download
MQTT packet model and wire protocol codec that incrementally decodes and encodes MQTT streams without intermediate buffering
// Copyright 2015-2024 Nstream, inc.
//
// 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 swim.mqtt;
import swim.codec.Encoder;
import swim.codec.EncoderException;
import swim.codec.OutputBuffer;
final class MqttPublishPacketEncoder extends Encoder> {
final MqttEncoder mqtt;
final MqttPublishPacket packet;
final Encoder, ?> part;
final int length;
final int remaining;
final int step;
MqttPublishPacketEncoder(MqttEncoder mqtt, MqttPublishPacket packet, Encoder, ?> part,
int length, int remaining, int step) {
this.mqtt = mqtt;
this.packet = packet;
this.part = part;
this.length = length;
this.remaining = remaining;
this.step = step;
}
MqttPublishPacketEncoder(MqttEncoder mqtt, MqttPublishPacket packet) {
this(mqtt, packet, null, 0, 0, 1);
}
@Override
public Encoder> pull(OutputBuffer> output) {
return MqttPublishPacketEncoder.encode(output, this.mqtt, this.packet, this.part,
this.length, this.remaining, this.step);
}
static Encoder> encode(OutputBuffer> output, MqttEncoder mqtt,
MqttPublishPacket packet, Encoder, ?> part,
int length, int remaining, int step) {
if (step == 1 && output.isCont()) {
length = packet.variableHeaderSize(mqtt);
remaining = length;
output = output.write(packet.packetType() << 4 | packet.packetFlags & 0x0f);
step = 2;
}
while (step >= 2 && step <= 5 && output.isCont()) {
int b = length & 0x7f;
length = length >>> 7;
if (length > 0) {
b |= 0x80;
}
output = output.write(b);
if (length == 0) {
step = 6;
break;
} else if (step < 5) {
step += 1;
} else {
return Encoder.error(new MqttException("packet length too long: " + remaining));
}
}
if (step == 6) {
final int outputStart = output.index();
final int outputLimit = output.limit();
final int outputRemaining = outputLimit - outputStart;
if (remaining < outputRemaining) {
output = output.limit(outputStart + remaining);
}
final boolean outputPart = output.isPart();
output = output.isPart(remaining > outputRemaining);
if (part == null) {
part = mqtt.encodeString(output, packet.topicName);
} else {
part = part.pull(output);
}
output = output.limit(outputLimit).isPart(outputPart);
remaining -= output.index() - outputStart;
if (part.isDone()) {
part = null;
if (packet.hasPacketId()) {
step = 7;
} else {
step = 9;
}
} else if (part.isError()) {
return part.asError();
}
}
if (step == 7 && remaining > 0 && output.isCont()) {
output = output.write(packet.packetId >>> 8);
remaining -= 1;
step = 8;
}
if (step == 8 && remaining > 0 && output.isCont()) {
output = output.write(packet.packetId);
remaining -= 1;
step = 9;
}
if (step == 9) {
final int outputStart = output.index();
final int outputLimit = output.limit();
final int outputRemaining = outputLimit - outputStart;
if (remaining < outputRemaining) {
output = output.limit(outputStart + remaining);
}
final boolean outputPart = output.isPart();
output = output.isPart(remaining > outputRemaining);
if (part == null) {
part = packet.payloadEncoder.pull(output);
} else {
part = part.pull(output);
}
output = output.limit(outputLimit).isPart(outputPart);
remaining -= output.index() - outputStart;
if (part.isDone()) {
part = null;
step = 10;
} else if (part.isError()) {
return part.asError();
}
}
if (step == 10 && remaining == 0) {
return Encoder.done(packet);
}
if (remaining < 0) {
return Encoder.error(new MqttException("packet length too short"));
} else if (output.isDone()) {
return Encoder.error(new EncoderException("truncated"));
} else if (output.isError()) {
return Encoder.error(output.trap());
}
return new MqttPublishPacketEncoder(mqtt, packet, part, length, remaining, step);
}
static Encoder> encode(OutputBuffer> output, MqttEncoder mqtt,
MqttPublishPacket packet) {
return MqttPublishPacketEncoder.encode(output, mqtt, packet, null, 0, 0, 1);
}
}