blob: 3ed8b0e55c2ac3f321cd9655f73d088666beb2f0 [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.queue;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import org.apache.qpid.AMQException;
import org.apache.qpid.framing.ContentHeaderBody;
import org.apache.qpid.framing.abstraction.MessagePublishInfo;
import org.apache.qpid.framing.abstraction.ContentChunk;
import org.apache.qpid.server.store.MessageStore;
import org.apache.qpid.server.store.StoreContext;
/**
* @author Robert Greig (robert.j.greig@jpmorgan.com)
*/
public class WeakReferenceMessageHandle implements AMQMessageHandle
{
private WeakReference<ContentHeaderBody> _contentHeaderBody;
private WeakReference<MessagePublishInfo> _messagePublishInfo;
private List<WeakReference<ContentChunk>> _contentBodies;
private boolean _redelivered;
private final MessageStore _messageStore;
private final Long _messageId;
private long _arrivalTime;
public WeakReferenceMessageHandle(final Long messageId, MessageStore messageStore)
{
_messageId = messageId;
_messageStore = messageStore;
}
public ContentHeaderBody getContentHeaderBody(StoreContext context) throws AMQException
{
ContentHeaderBody chb = (_contentHeaderBody != null ? _contentHeaderBody.get() : null);
if (chb == null)
{
MessageMetaData mmd = loadMessageMetaData(context);
chb = mmd.getContentHeaderBody();
}
return chb;
}
public Long getMessageId()
{
return _messageId;
}
private MessageMetaData loadMessageMetaData(StoreContext context)
throws AMQException
{
MessageMetaData mmd = _messageStore.getMessageMetaData(context, _messageId);
populateFromMessageMetaData(mmd);
return mmd;
}
private void populateFromMessageMetaData(MessageMetaData mmd)
{
_arrivalTime = mmd.getArrivalTime();
_contentHeaderBody = new WeakReference<ContentHeaderBody>(mmd.getContentHeaderBody());
_messagePublishInfo = new WeakReference<MessagePublishInfo>(mmd.getMessagePublishInfo());
}
public int getBodyCount(StoreContext context) throws AMQException
{
if (_contentBodies == null)
{
MessageMetaData mmd = _messageStore.getMessageMetaData(context, _messageId);
int chunkCount = mmd.getContentChunkCount();
_contentBodies = new ArrayList<WeakReference<ContentChunk>>(chunkCount);
for (int i = 0; i < chunkCount; i++)
{
_contentBodies.add(new WeakReference<ContentChunk>(null));
}
}
return _contentBodies.size();
}
public long getBodySize(StoreContext context) throws AMQException
{
return getContentHeaderBody(context).bodySize;
}
public ContentChunk getContentChunk(StoreContext context, int index) throws AMQException, IllegalArgumentException
{
if (index > _contentBodies.size() - 1)
{
throw new IllegalArgumentException("Index " + index + " out of valid range 0 to " +
(_contentBodies.size() - 1));
}
WeakReference<ContentChunk> wr = _contentBodies.get(index);
ContentChunk cb = wr.get();
if (cb == null)
{
cb = _messageStore.getContentBodyChunk(context, _messageId, index);
_contentBodies.set(index, new WeakReference<ContentChunk>(cb));
}
return cb;
}
/**
* Content bodies are set <i>before</i> the publish and header frames
*
* @param storeContext
* @param contentChunk
* @param isLastContentBody
* @throws AMQException
*/
public void addContentBodyFrame(StoreContext storeContext, ContentChunk contentChunk, boolean isLastContentBody) throws AMQException
{
if (_contentBodies == null && isLastContentBody)
{
_contentBodies = new ArrayList<WeakReference<ContentChunk>>(1);
}
else
{
if (_contentBodies == null)
{
_contentBodies = new LinkedList<WeakReference<ContentChunk>>();
}
}
_contentBodies.add(new WeakReference<ContentChunk>(contentChunk));
_messageStore.storeContentBodyChunk(storeContext, _messageId, _contentBodies.size() - 1,
contentChunk, isLastContentBody);
}
public MessagePublishInfo getMessagePublishInfo(StoreContext context) throws AMQException
{
MessagePublishInfo bpb = (_messagePublishInfo != null ? _messagePublishInfo.get() : null);
if (bpb == null)
{
MessageMetaData mmd = loadMessageMetaData(context);
bpb = mmd.getMessagePublishInfo();
}
return bpb;
}
public boolean isRedelivered()
{
return _redelivered;
}
public void setRedelivered(boolean redelivered)
{
_redelivered = redelivered;
}
public boolean isPersistent()
{
return true;
}
/**
* This is called when all the content has been received.
*
* @param publishBody
* @param contentHeaderBody
* @throws AMQException
*/
public void setPublishAndContentHeaderBody(StoreContext storeContext, MessagePublishInfo publishBody,
ContentHeaderBody contentHeaderBody)
throws AMQException
{
// if there are no content bodies the list will be null so we must
// create en empty list here
if (contentHeaderBody.bodySize == 0)
{
_contentBodies = new LinkedList<WeakReference<ContentChunk>>();
}
final long arrivalTime = System.currentTimeMillis();
MessageMetaData mmd = new MessageMetaData(publishBody, contentHeaderBody, _contentBodies.size(), arrivalTime);
_messageStore.storeMessageMetaData(storeContext, _messageId, mmd);
populateFromMessageMetaData(mmd);
}
public void removeMessage(StoreContext storeContext) throws AMQException
{
_messageStore.removeMessage(storeContext, _messageId);
}
public long getArrivalTime()
{
return _arrivalTime;
}
}