Add support for FortiGate syslog events

This commit is contained in:
Jochen Schalanda 2017-03-09 00:12:40 +01:00
parent 69a6f75ca7
commit 24915485f2
No known key found for this signature in database
GPG Key ID: 2FC1B61A8D1F4BB0
2 changed files with 210 additions and 0 deletions

View File

@ -0,0 +1,176 @@
package org.graylog2.syslog4j.server.impl.event;
import org.graylog2.syslog4j.server.SyslogServerEventIF;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.time.LocalDate;
import java.time.LocalTime;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Collections;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import static java.util.Objects.requireNonNull;
/**
* FortiGateSyslogEvent provides an implementation of the
* SyslogServerEventIF interface that supports receiving of
* FortiGate/FortiOS syslog messages.
*
* @see <a href="http://help.fortinet.com/fos50hlp/54/Content/FortiOS/fortigate-logging-reporting-54/logs.htm#Log_messages">FortiGate logging and reporting overview</a>
*/
public class FortiGateSyslogEvent implements SyslogServerEventIF {
private static final Pattern PRI_PATTERN = Pattern.compile("^<(\\d{1,3})>(.*)$");
private static final Pattern KV_PATTERN = Pattern.compile("(\\w+)=([^\\s\"]*)");
private static final Pattern QUOTED_KV_PATTERN = Pattern.compile("(\\w+)=\"([^\"]*)\"");
private final String rawEvent;
private Date date;
private int facility;
private int level;
private String host;
private String message;
private Charset charSet = StandardCharsets.UTF_8;
private Map<String, String> fields = Collections.emptyMap();
public FortiGateSyslogEvent(final String rawEvent) {
this.rawEvent = requireNonNull(rawEvent, "rawEvent");
parse(rawEvent);
}
private void parse(String event) {
final Matcher matcher = PRI_PATTERN.matcher(event);
if (!matcher.find()) {
throw new IllegalArgumentException("Invalid Fortigate syslog message");
} else {
final String priority = matcher.group(1);
final String message = matcher.group(2);
parsePriority(priority);
setMessage(message);
parseFields(message);
parseDate(fields.get("date"), fields.get("time"));
setHost(fields.get("devname"));
}
}
private void parsePriority(String priorityString) {
try {
final int priority = Integer.parseInt(priorityString);
setFacility(priority / 8);
setLevel(priority % 8);
} catch (NumberFormatException e) {
throw new IllegalArgumentException("Couldn't parse message priority", e);
}
}
private void parseFields(String event) {
final Map<String, String> fields = new HashMap<>();
final Matcher matcher = KV_PATTERN.matcher(event);
while (matcher.find()) {
fields.put(matcher.group(1), matcher.group(2));
}
final Matcher quotedMatcher = QUOTED_KV_PATTERN.matcher(event);
while (quotedMatcher.find()) {
fields.put(quotedMatcher.group(1), quotedMatcher.group(2));
}
setFields(fields);
}
private void parseDate(String date, String time) {
if (date != null && time != null) {
final ZonedDateTime dateTime = ZonedDateTime.of(
LocalDate.parse(date, DateTimeFormatter.ISO_LOCAL_DATE.withZone(ZoneOffset.UTC)),
LocalTime.parse(time, DateTimeFormatter.ISO_LOCAL_TIME.withZone(ZoneOffset.UTC)),
ZoneOffset.UTC);
setDate(Date.from(dateTime.toInstant()));
} else {
setDate(new Date());
}
}
@Override
public byte[] getRaw() {
return rawEvent.getBytes(charSet);
}
@Override
public int getFacility() {
return facility;
}
@Override
public void setFacility(int facility) {
this.facility = requireNonNull(facility, "facility");
}
@Override
public Date getDate() {
return date;
}
@Override
public void setDate(Date date) {
this.date = requireNonNull(date, "date");
}
@Override
public int getLevel() {
return level;
}
@Override
public void setLevel(int level) {
this.level = requireNonNull(level, "level");
}
@Override
public String getHost() {
return host;
}
@Override
public void setHost(String host) {
this.host = host;
}
@Override
public boolean isHostStrippedFromMessage() {
return false;
}
@Override
public String getMessage() {
return message;
}
@Override
public void setMessage(String message) {
this.message = requireNonNull(message, "message");
}
@Override
public String getCharSet() {
return charSet.name();
}
@Override
public void setCharSet(String charSet) {
this.charSet = Charset.forName(charSet);
}
public Map<String, String> getFields() {
return Collections.unmodifiableMap(fields);
}
public void setFields(Map<String, String> fields) {
this.fields = requireNonNull(fields, "fields");
}
}

View File

@ -0,0 +1,34 @@
package org.graylog2.syslog4j.server.impl.event;
import org.junit.Test;
import java.nio.charset.StandardCharsets;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import static org.assertj.core.api.Assertions.assertThat;
public class FortiGateSyslogEventTest {
@Test
public void testFortiGateMessage() {
final String rawMessage = "<45>date=2017-03-06 time=12:53:10 devname=DEVICENAME devid=DEVICEID logid=0000000013 type=traffic subtype=forward level=notice vd=ALIAS srcip=IP srcport=45748 srcintf=\"IF\" dstip=IP dstport=443 dstintf=\"IF\" sessionid=1122686199 status=close policyid=77 dstcountry=\"COUNTRY\" srccountry=\"COUNTRY\" trandisp=dnat tranip=IP tranport=443 service=HTTPS proto=6 appid=41540 app=\"SSL_TLSv1.2\" appcat=\"Network.Service\" applist=\"ACLNAME\" appact=detected duration=1 sentbyte=2313 rcvdbyte=14883 sentpkt=19 rcvdpkt=19 utmaction=passthrough utmevent=app-ctrl attack=\"SSL\" hostname=\"HOSTNAME\" custom=\"white space\"";
final FortiGateSyslogEvent event = new FortiGateSyslogEvent(rawMessage);
assertThat(event).isNotNull();
assertThat(event.getFacility()).isEqualTo(5);
assertThat(event.getLevel()).isEqualTo(5);
assertThat(event.getHost()).isEqualTo("DEVICENAME");
assertThat(ZonedDateTime.ofInstant(event.getDate().toInstant(), ZoneOffset.UTC))
.isEqualTo(ZonedDateTime.of(2017, 3, 6, 12, 53, 10, 0, ZoneOffset.UTC));
assertThat(event.getRaw()).isEqualTo(rawMessage.getBytes(StandardCharsets.UTF_8));
assertThat(event.getMessage()).isEqualTo("date=2017-03-06 time=12:53:10 devname=DEVICENAME devid=DEVICEID logid=0000000013 type=traffic subtype=forward level=notice vd=ALIAS srcip=IP srcport=45748 srcintf=\"IF\" dstip=IP dstport=443 dstintf=\"IF\" sessionid=1122686199 status=close policyid=77 dstcountry=\"COUNTRY\" srccountry=\"COUNTRY\" trandisp=dnat tranip=IP tranport=443 service=HTTPS proto=6 appid=41540 app=\"SSL_TLSv1.2\" appcat=\"Network.Service\" applist=\"ACLNAME\" appact=detected duration=1 sentbyte=2313 rcvdbyte=14883 sentpkt=19 rcvdpkt=19 utmaction=passthrough utmevent=app-ctrl attack=\"SSL\" hostname=\"HOSTNAME\" custom=\"white space\"");
assertThat(event.getFields())
.containsEntry("date", "2017-03-06")
.containsEntry("time", "12:53:10")
.containsEntry("devname", "DEVICENAME")
.containsEntry("devid", "DEVICEID")
.containsEntry("hostname", "HOSTNAME")
.containsEntry("custom", "white space");
}
}