blob: 6448e0d6e702169a193bc89201d8f708ae229efb [file] [log] [blame]
/*
*
* 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.qpid.server.protocol.v0_8;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Set;
import org.apache.qpid.bytebuffer.QpidByteBuffer;
import org.apache.qpid.codec.MarkableDataInput;
import org.apache.qpid.framing.AMQFrameDecodingException;
import org.apache.qpid.framing.AMQProtocolVersionException;
import org.apache.qpid.framing.AMQShortString;
import org.apache.qpid.framing.BasicContentHeaderProperties;
import org.apache.qpid.framing.ContentHeaderBody;
import org.apache.qpid.framing.EncodingUtils;
import org.apache.qpid.framing.FieldTable;
import org.apache.qpid.framing.MessagePublishInfo;
import org.apache.qpid.server.message.AMQMessageHeader;
import org.apache.qpid.server.plugin.MessageMetaDataType;
import org.apache.qpid.server.store.StorableMessageMetaData;
import org.apache.qpid.server.util.ConnectionScopedRuntimeException;
import org.apache.qpid.transport.ByteBufferSender;
/**
* Encapsulates a publish body and a content header. In the context of the message store these are treated as a
* single unit.
*/
public class MessageMetaData implements StorableMessageMetaData
{
private final MessagePublishInfo _messagePublishInfo;
private final ContentHeaderBody _contentHeaderBody;
private long _arrivalTime;
private static final byte MANDATORY_FLAG = 1;
private static final byte IMMEDIATE_FLAG = 2;
public static final MessageMetaDataType.Factory<MessageMetaData> FACTORY = new MetaDataFactory();
private static final MessageMetaDataType_0_8 TYPE = new MessageMetaDataType_0_8();
public MessageMetaData(MessagePublishInfo publishBody, ContentHeaderBody contentHeaderBody)
{
this(publishBody,contentHeaderBody, System.currentTimeMillis());
}
public MessageMetaData(MessagePublishInfo publishBody,
ContentHeaderBody contentHeaderBody,
long arrivalTime)
{
_contentHeaderBody = contentHeaderBody;
_messagePublishInfo = publishBody;
_arrivalTime = arrivalTime;
}
public synchronized ContentHeaderBody getContentHeaderBody()
{
return _contentHeaderBody;
}
public MessagePublishInfo getMessagePublishInfo()
{
return _messagePublishInfo;
}
public long getArrivalTime()
{
return _arrivalTime;
}
public MessageMetaDataType getType()
{
return TYPE;
}
public int getStorableSize()
{
int size = _contentHeaderBody.getSize();
size += 4;
size += EncodingUtils.encodedShortStringLength(_messagePublishInfo.getExchange());
size += EncodingUtils.encodedShortStringLength(_messagePublishInfo.getRoutingKey());
size += 1; // flags for immediate/mandatory
size += EncodingUtils.encodedLongLength();
return size;
}
public int writeToBuffer(final QpidByteBuffer dest)
{
int oldPosition = dest.position();
try
{
dest.putInt(_contentHeaderBody.getSize());
_contentHeaderBody.writePayload(dest);
EncodingUtils.writeShortStringBytes(dest, _messagePublishInfo.getExchange());
EncodingUtils.writeShortStringBytes(dest, _messagePublishInfo.getRoutingKey());
byte flags = 0;
if(_messagePublishInfo.isMandatory())
{
flags |= MANDATORY_FLAG;
}
if(_messagePublishInfo.isImmediate())
{
flags |= IMMEDIATE_FLAG;
}
dest.put(flags);
dest.putLong(_arrivalTime);
}
catch (IOException e)
{
// This shouldn't happen as we are not actually using anything that can throw an IO Exception
throw new ConnectionScopedRuntimeException(e);
}
return dest.position()-oldPosition;
}
@Override
public Collection<QpidByteBuffer> asByteBuffers()
{
try
{
final List<QpidByteBuffer> buffers = new ArrayList<>();
QpidByteBuffer buf = QpidByteBuffer.allocateDirect(4);
buffers.add(buf);
buf.putInt(0, _contentHeaderBody.getSize());
_contentHeaderBody.writePayload(new ByteBufferSender()
{
@Override
public void send(final QpidByteBuffer msg)
{
buffers.add(msg.duplicate());
}
@Override
public void flush()
{
}
@Override
public void close()
{
}
});
buf = QpidByteBuffer.allocateDirect(9+EncodingUtils.encodedShortStringLength(_messagePublishInfo.getExchange())+EncodingUtils.encodedShortStringLength(_messagePublishInfo.getRoutingKey()));
EncodingUtils.writeShortStringBytes(buf, _messagePublishInfo.getExchange());
EncodingUtils.writeShortStringBytes(buf, _messagePublishInfo.getRoutingKey());
byte flags = 0;
if(_messagePublishInfo.isMandatory())
{
flags |= MANDATORY_FLAG;
}
if(_messagePublishInfo.isImmediate())
{
flags |= IMMEDIATE_FLAG;
}
buf.put(flags);
buf.putLong(_arrivalTime);
buf.flip();
buffers.add(buf);
return buffers;
}
catch (IOException e)
{
// This shouldn't happen as we are not actually using anything that can throw an IO Exception
throw new ConnectionScopedRuntimeException(e);
}
}
public int getContentSize()
{
return (int) _contentHeaderBody.getBodySize();
}
public boolean isPersistent()
{
return _contentHeaderBody.getProperties().getDeliveryMode() == BasicContentHeaderProperties.PERSISTENT;
}
@Override
public synchronized void dispose()
{
_contentHeaderBody.dispose();
}
public synchronized void clearEncodedForm()
{
_contentHeaderBody.clearEncodedForm();
}
private static class MetaDataFactory implements MessageMetaDataType.Factory<MessageMetaData>
{
public MessageMetaData createMetaData(QpidByteBuffer buf)
{
try
{
MarkableDataInput dataInput = buf.asDataInput();
int size = EncodingUtils.readInteger(dataInput);
ContentHeaderBody chb = ContentHeaderBody.createFromBuffer(dataInput, size);
final AMQShortString exchange = EncodingUtils.readAMQShortString(dataInput);
final AMQShortString routingKey = EncodingUtils.readAMQShortString(dataInput);
final byte flags = EncodingUtils.readByte(dataInput);
long arrivalTime = EncodingUtils.readLong(dataInput);
MessagePublishInfo publishBody =
new MessagePublishInfo(exchange,
(flags & IMMEDIATE_FLAG) != 0,
(flags & MANDATORY_FLAG) != 0,
routingKey);
return new MessageMetaData(publishBody, chb, arrivalTime);
}
catch (IOException | AMQFrameDecodingException | AMQProtocolVersionException e)
{
throw new ConnectionScopedRuntimeException(e);
}
}
}
public AMQMessageHeader getMessageHeader()
{
return new MessageHeaderAdapter();
}
private final class MessageHeaderAdapter implements AMQMessageHeader
{
private BasicContentHeaderProperties getProperties()
{
return getContentHeaderBody().getProperties();
}
public String getUserId()
{
return getProperties().getUserIdAsString();
}
public String getAppId()
{
return getProperties().getAppIdAsString();
}
public String getCorrelationId()
{
return getProperties().getCorrelationIdAsString();
}
public long getExpiration()
{
return getProperties().getExpiration();
}
public String getMessageId()
{
return getProperties().getMessageIdAsString();
}
public String getMimeType()
{
return getProperties().getContentTypeAsString();
}
public String getEncoding()
{
return getProperties().getEncodingAsString();
}
public byte getPriority()
{
return getProperties().getPriority();
}
public long getTimestamp()
{
return getProperties().getTimestamp();
}
public String getType()
{
return getProperties().getTypeAsString();
}
public String getReplyTo()
{
return getProperties().getReplyToAsString();
}
@Override
public long getNotValidBefore()
{
Object header = getHeader("x-qpid-not-valid-before");
return header instanceof Number ? ((Number)header).longValue() : 0L;
}
public Object getHeader(String name)
{
FieldTable ft = getProperties().getHeaders();
return ft.get(name);
}
public boolean containsHeaders(Set<String> names)
{
FieldTable ft = getProperties().getHeaders();
for(String name : names)
{
if(!ft.containsKey(name))
{
return false;
}
}
return true;
}
@Override
public Collection<String> getHeaderNames()
{
return getProperties().getHeaders().keys();
}
public boolean containsHeader(String name)
{
FieldTable ft = getProperties().getHeaders();
return ft.containsKey(name);
}
}
}