org.graylog2.plugin.journal.RawMessage Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of graylog2-plugin Show documentation
Show all versions of graylog2-plugin Show documentation
Graylog interfaces for plugin development
/**
* The MIT License
* Copyright (c) 2012 Graylog, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package org.graylog2.plugin.journal;
import com.eaio.uuid.UUID;
import com.google.common.base.MoreObjects;
import com.google.common.collect.Lists;
import com.google.protobuf.ByteString;
import com.google.protobuf.UninitializedMessageException;
import org.graylog2.plugin.ResolvableInetSocketAddress;
import org.graylog2.plugin.Tools;
import org.graylog2.plugin.configuration.Configuration;
import org.graylog2.plugin.system.NodeId;
import org.joda.time.DateTime;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.Nullable;
import java.io.IOException;
import java.io.Serializable;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.UnknownHostException;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Strings.isNullOrEmpty;
import static org.graylog2.plugin.journal.JournalMessages.JournalMessage;
/**
* A raw message is the unparsed data Graylog was handed by an input.
*
* Typically this is a copy of the exact bytes received over the network, after all de-chunking, removal of transport
* headers, etc has been performed, but before any parsing, decoding, checking of the actual payload has been performed.
*
*
* Each raw message has a unique id, a timestamp it was received at (this might be different from the timestamp contained
* in the payload, if that has any), a tag of what payload type this is supposed to be (e.g. syslog, GELF, RadioMessage etc.),
* as well as an opaque meta data structure.
* The format of the meta data is not stable yet, but will likely be a JSON string.
*
*/
public class RawMessage implements Serializable {
public static final byte CURRENT_VERSION = 1;
private static final Logger log = LoggerFactory.getLogger(RawMessage.class);
private final JournalMessage.Builder msgBuilder;
private final UUID id;
private final long journalOffset;
private Configuration codecConfig;
public RawMessage(byte[] payload) {
this(payload, (ResolvableInetSocketAddress)null);
}
public RawMessage(byte[] payload, InetSocketAddress remoteAddress) {
this(Long.MIN_VALUE, new UUID(), Tools.iso8601(), ResolvableInetSocketAddress.wrap(remoteAddress), payload);
}
public RawMessage(byte[] payload, ResolvableInetSocketAddress remoteAddress) {
this(Long.MIN_VALUE, new UUID(), Tools.iso8601(), remoteAddress, payload);
}
public RawMessage(long journalOffset,
UUID id,
DateTime timestamp,
ResolvableInetSocketAddress remoteAddress,
byte[] payload) {
checkNotNull(id, "The message id must not be null!");
checkNotNull(payload, "The message payload must not be null!");
checkArgument(payload.length > 0, "The message payload must not be empty!");
msgBuilder = JournalMessage.newBuilder();
this.journalOffset = journalOffset;
msgBuilder.setVersion(CURRENT_VERSION);
this.id = id;
msgBuilder.setUuidTime(id.time);
msgBuilder.setUuidClockseq(id.clockSeqAndNode);
msgBuilder.setTimestamp(timestamp.getMillis());
if (null != remoteAddress) {
setRemoteAddress(remoteAddress);
}
msgBuilder.setPayload(ByteString.copyFrom(payload));
}
public void addSourceNode(String sourceInputId, NodeId nodeId, boolean isServer) {
msgBuilder.addSourceNodesBuilder()
.setInputId(sourceInputId)
.setId(nodeId.toString())
.setType(isServer ? JournalMessages.SourceNode.Type.SERVER : JournalMessages.SourceNode.Type.RADIO)
.build();
}
public RawMessage(JournalMessage journalMessage, long journalOffset) {
this.journalOffset = journalOffset;
id = new UUID(journalMessage.getUuidTime(), journalMessage.getUuidClockseq());
msgBuilder = JournalMessage.newBuilder(journalMessage);
codecConfig = Configuration.deserializeFromJson(journalMessage.getCodec().getConfig());
}
@Nullable
public static RawMessage decode(final byte[] buffer, final long journalOffset) {
try {
final JournalMessage journalMessage = JournalMessage.parseFrom(buffer);
// TODO validate message based on field contents and version number
return new RawMessage(journalMessage, journalOffset);
} catch (IOException e) {
log.error("Cannot read raw message from journal, ignoring this message.", e);
return null;
}
}
public byte[] encode() {
try {
final JournalMessages.CodecInfo codec = msgBuilder.getCodec();
final JournalMessages.CodecInfo.Builder builder = JournalMessages.CodecInfo.newBuilder(codec);
final String codecConfigJson = codecConfig.serializeToJson();
if (codecConfigJson != null) {
builder.setConfig(codecConfigJson);
}
msgBuilder.setCodec(builder.build());
final JournalMessage journalMessage = msgBuilder.build();
return journalMessage.toByteArray();
} catch (UninitializedMessageException e) {
log.error(
"Unable to write RawMessage to journal because required fields are missing, " +
"this message will be discarded. This is a bug.", e);
return null;
}
}
public int getVersion() {
return msgBuilder.getVersion();
}
public DateTime getTimestamp() {
return new DateTime(msgBuilder.getTimestamp()); // TODO PERFORMANCE object creation
}
public byte[] getPayload() {
return msgBuilder.getPayload().toByteArray(); // TODO PERFORMANCE array copy
}
public UUID getId() {
return id;
}
public byte[] getIdBytes() {
final long time = id.getTime();
final long clockSeqAndNode = id.getClockSeqAndNode();
return ByteBuffer.allocate(16)
.putLong(time)
.putLong(clockSeqAndNode)
.array(); // TODO PERFORMANCE object creation
}
@Nullable
public ResolvableInetSocketAddress getRemoteAddress() {
if (msgBuilder.hasRemote()) {
final JournalMessages.RemoteAddress address = msgBuilder.getRemote();
final InetAddress inetAddr;
try {
inetAddr = InetAddress.getByAddress(address.getResolved(), address.getAddress().toByteArray());
} catch (UnknownHostException e) {
log.warn("Malformed InetAddress for message {}, expected 4 or 16 bytes, but got {} bytes",
id, address.getAddress().toByteArray());
return null;
}
final int port = address.hasPort() ? address.getPort() : 0;
// TODO PERFORMANCE object creation
return ResolvableInetSocketAddress.wrap(new InetSocketAddress(inetAddr, port));
}
return null;
}
public void setRemoteAddress(ResolvableInetSocketAddress address) {
final JournalMessages.RemoteAddress.Builder builder = msgBuilder.getRemoteBuilder();
builder.setAddress(ByteString.copyFrom(address.getAddressBytes()))
.setPort(address.getPort());
// do not perform any reverse lookup here
if (address.isReverseLookedUp()) {
builder.setResolved(address.getHostName());
}
}
public String getCodecName() {
return msgBuilder.getCodecBuilder().getName();
}
public void setCodecName(String name) {
checkArgument(!isNullOrEmpty(name), "The payload type must not be null or empty!");
msgBuilder.getCodecBuilder().setName(name);
}
public Configuration getCodecConfig() {
return codecConfig;
}
public void setCodecConfig(Configuration codecConfig) {
this.codecConfig = codecConfig;
}
public List getSourceNodes() {
final ArrayList list = Lists.newArrayList();
for (final JournalMessages.SourceNode node : msgBuilder.getSourceNodesList()) {
list.add(new SourceNode(node));
}
return list;
}
@Override
public String toString() {
final MoreObjects.ToStringHelper helper = MoreObjects.toStringHelper(this);
helper.add("id", getId())
.add("journalOffset", getJournalOffset())
.add("codec", getCodecName())
.add("payloadSize", getPayload().length)
.add("timestamp", getTimestamp());
if (getRemoteAddress() != null) {
helper.add("remoteAddress", getRemoteAddress().getInetSocketAddress().toString());
}
return helper.toString();
}
public long getJournalOffset() {
return journalOffset;
}
public static class SourceNode {
public String nodeId;
public String inputId;
public Type type;
public enum Type {
SERVER,
RADIO
}
public SourceNode(JournalMessages.SourceNode node) {
this.nodeId = node.getId();
this.inputId = node.getInputId();
switch (node.getType()) {
case SERVER:
this.type = Type.SERVER;
break;
case RADIO:
this.type = Type.RADIO;
break;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy