Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Logz.io support #74

Open
wants to merge 57 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
57 commits
Select commit Hold shift + click to select a range
d986d80
first
May 31, 2018
8a00ac8
queue in workspace
Jun 3, 2018
de18b60
deleting comments
Jun 3, 2018
4ef9a2f
PR fixes
Jun 4, 2018
8c185d6
change dir path
Jun 4, 2018
6c7d6a6
using tmp dir
Jun 4, 2018
9b7621d
change @timestamp field name
Jun 4, 2018
5075433
using stream
Jun 5, 2018
fa317c1
Merge pull request #1 from idohalevi/first
idohalevi Jun 5, 2018
bf87eec
Update Logzio.java
idohalevi Jun 6, 2018
6d53f29
add debug info
Jun 17, 2018
4d9134d
change queue location
Jun 18, 2018
54cd262
change const
Jun 18, 2018
7a9164d
add debug
Jun 19, 2018
959f715
debug prints
Jun 20, 2018
03cb931
debug
Jun 24, 2018
035fb68
gzip logs
Jul 3, 2018
53ffac5
Merge branch 'master' of https://github.com/jenkinsci/logstash-plugin
Jul 3, 2018
84aed45
delete debug print
Jul 3, 2018
193f057
fixing for travis tests (#2)
idohalevi Jul 3, 2018
4dbe73e
Fix travis (#3)
idohalevi Jul 4, 2018
89a41e9
Use a container for more repeatable builds (#4)
tomwillfixit Jul 9, 2018
25c63b4
sync with upstream
Aug 22, 2018
bdab1a3
stream is back
Jul 4, 2018
488037f
logzio in memory https client
Aug 22, 2018
144e922
adding in memory http sender, local logger and updating README
Aug 23, 2018
30ccd6b
remove docker from readme install
Aug 23, 2018
bf200ca
delete comma
Aug 23, 2018
f2dc75e
Merge pull request #5 from idohalevi/PR
idohalevi Aug 23, 2018
ad75e1e
Ranges -> Range
jakub-bochenski Aug 23, 2018
6c27e9b
deleting mail
Aug 23, 2018
84d2051
adding static class/fields - travis findbugs
Aug 23, 2018
bd0ba73
adding charset - travis findbugs
Aug 23, 2018
22d88bc
change reporter class to static class
Aug 23, 2018
63647d6
change LOGGER to static member
Aug 23, 2018
6a70412
delete unused imports
Sep 5, 2018
f57f163
delete unused imports - test
Sep 5, 2018
bd062d0
add flatten data explanation
Sep 6, 2018
594c37d
send logs before size limit reached
Sep 6, 2018
07fcdb4
adding warning about thread safety
Sep 6, 2018
e2fb882
move LogzioHttpsClient to a top-level class
Sep 6, 2018
74fa7f6
fix spelling
Sep 6, 2018
93732e3
change key to token
Sep 12, 2018
203eca1
change messages list to Collections.synchronizedList
Sep 12, 2018
d31850b
style + typo
Sep 12, 2018
b3ffe89
rename help file
Sep 12, 2018
77577d9
change to capital
Sep 12, 2018
e7b056e
delete run-fast script
Sep 13, 2018
1508b3f
Merge branch 'master' of https://github.com/jenkinsci/logstash-plugin…
HananEgbaria Jan 30, 2019
3e37735
Merge pull request #6 from idohalevi/dev
HananEgbaria Feb 5, 2019
605a76c
add close method
HananEgbaria Feb 12, 2019
f5e9581
add "synchronized" to push method to resolve multi-threading (multi-j…
HananEgbaria Mar 5, 2019
1aec70e
update maven dependencies versions
HananEgbaria Mar 11, 2019
84c3602
remove unimportant comments
HananEgbaria Mar 11, 2019
93a5cd2
Merge pull request #7 from idohalevi/fix_comments
idohalevi Mar 18, 2019
b1dc099
remove unused methods
HananEgbaria Mar 19, 2019
261f22c
Merge pull request #8 from idohalevi/fix_comments
idohalevi Mar 19, 2019
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ Currently supported methods of input/output:
* Redis {format => 'json_event'}
* RabbitMQ {mechanism => PLAIN}
* Syslog {format => cee/json ([RFC-5424](https://tools.ietf.org/html/rfc5424),[RFC-3164](https://tools.ietf.org/html/rfc3164)), protocol => UDP}
* Logz.io

Pipeline
========
Expand Down
19 changes: 15 additions & 4 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,10 @@
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<skipIntegrationTests>true</skipIntegrationTests>
<jenkins-test-harness.version>2.23</jenkins-test-harness.version>
<java.level>7</java.level>
<java.level>8</java.level>
<jenkins.version>2.7.4</jenkins.version>
</properties>

<parent>
<groupId>org.jenkins-ci.plugins</groupId>
<artifactId>plugin</artifactId>
Expand Down Expand Up @@ -109,6 +109,17 @@
<scope>compile</scope>
</dependency>

<dependency>
<groupId>io.logz.sender</groupId>
<artifactId>logzio-sender</artifactId>
<version>1.0.10</version>
</dependency>

<dependency>
<groupId>com.github.wnameless</groupId>
<artifactId>json-flattener</artifactId>
<version>0.5.0</version>
</dependency>

<dependency>
<groupId>org.jenkins-ci.plugins</groupId>
Expand Down Expand Up @@ -196,8 +207,8 @@
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>1.7</source>
<target>1.7</target>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
import jenkins.plugins.logstash.configuration.RabbitMq;
import jenkins.plugins.logstash.configuration.Redis;
import jenkins.plugins.logstash.configuration.Syslog;
import jenkins.plugins.logstash.configuration.Logzio;
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
import jenkins.plugins.logstash.persistence.LogstashIndexerDao;
import jenkins.plugins.logstash.persistence.LogstashIndexerDao.IndexerType;
import net.sf.json.JSONObject;
Expand Down Expand Up @@ -211,6 +212,13 @@ public void migrateData()
logstashIndexer = syslog;
enabled = true;
break;
case LOGZIO:
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
LOGGER.log(Level.INFO, "Migrating logstash configuration for Logz.io");
Logzio logz = new Logzio();
logz.setHost(descriptor.getHost());
logz.setKey(descriptor.getKey());
logstashIndexer = logz;
break;
default:
LOGGER.log(Level.INFO, "unknown logstash Indexer type: " + type);
enabled = false;
Expand Down
120 changes: 120 additions & 0 deletions src/main/java/jenkins/plugins/logstash/configuration/Logzio.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,120 @@
package jenkins.plugins.logstash.configuration;

import hudson.Extension;
import hudson.util.FormValidation;
import hudson.util.Secret;

import jenkins.plugins.logstash.persistence.LogzioDao;
import jenkins.plugins.logstash.Messages;

import org.apache.commons.lang.StringUtils;
import org.kohsuke.stapler.DataBoundConstructor;
import org.kohsuke.stapler.DataBoundSetter;
import org.kohsuke.stapler.QueryParameter;

import javax.annotation.Nonnull;

public class Logzio extends LogstashIndexer<LogzioDao>
{
private Secret key;
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
private String host;

@DataBoundConstructor
public Logzio(){}

/*
* We use URL for the setter as stapler can autoconvert a string to a URL but not to a URI
*/
public String getHost(){ return this.host; }

@DataBoundSetter
public void setHost(String host){ this.host = host; }

public String getKey()
{
return Secret.toString(key);
}

@DataBoundSetter
public void setKey(String key)
{
this.key = Secret.fromString(key);
}


@Override
public boolean equals(Object obj)
{
if (obj == null)
return false;
if (this == obj)
return true;
if (getClass() != obj.getClass())
return false;
Logzio other = (Logzio) obj;
if (!Secret.toString(key).equals(other.getKey()))
{
return false;
}
if (host == null)
{
return other.host == null;
}
else return host.equals(other.host);
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
}

@Override
public int hashCode()
{
final int prime = 31;
int result = super.hashCode();
result = prime * result + ((host == null) ? 0 : host.hashCode());
result = prime * result + Secret.toString(key).hashCode();
return result;
}

@Override
public LogzioDao createIndexerInstance() { return new LogzioDao(host, Secret.toString(key)); }

@Extension
public static class LogzioDescriptor extends LogstashIndexerDescriptor
{
private static String EU_HOST = "https://listener-eu.logz.io:8071";
private static String NONEU_HOST = "https://listener.logz.io:8071";

@Nonnull
@Override
public String getDisplayName()
{
return "Logz.io";
}

@Override
public int getDefaultPort()
{
return 0;
}

public FormValidation doCheckKey(@QueryParameter("value") String value)
{
if (StringUtils.isBlank(value))
{
return FormValidation.error(Messages.ValueIsRequired());
}
return FormValidation.ok();
}

public FormValidation doCheckHost(@QueryParameter("value") String value)
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
{
if (StringUtils.isBlank(value))
{
return FormValidation.error(Messages.ValueIsRequired());
}
else if (!(value.equals(EU_HOST) || value.equals(NONEU_HOST))){
return FormValidation.error("Please verify your logz.io host is one of the two possible hosts - "
+ EU_HOST + " or " + NONEU_HOST);
}
return FormValidation.ok();
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,8 @@ static enum IndexerType {
REDIS,
RABBIT_MQ,
ELASTICSEARCH,
SYSLOG
SYSLOG,
LOGZIO
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
}

@Deprecated
Expand Down
127 changes: 127 additions & 0 deletions src/main/java/jenkins/plugins/logstash/persistence/LogzioDao.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,127 @@
package jenkins.plugins.logstash.persistence;

import com.github.wnameless.json.flattener.JsonFlattener;
import com.google.common.io.Files;

import io.logz.sender.LogzioSender;
import io.logz.sender.SenderStatusReporter;
import io.logz.sender.com.google.gson.JsonObject;
import io.logz.sender.exceptions.LogzioParameterErrorException;

import java.io.*;
import java.util.*;
import java.util.concurrent.Executors;
import java.util.logging.Level;
import java.util.logging.Logger;

import jenkins.model.Jenkins;
import jenkins.plugins.logstash.LogstashConfiguration;
import net.sf.json.JSONArray;
import net.sf.json.JSONObject;

/**
* Logz.io Data Access Object.
*
* @author Ido Halevi
*/

public class LogzioDao extends AbstractLogstashIndexerDao {
private static final int CONNECT_TIMEOUT = 10*1000;
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
private static final int CORE_POOL_SIZE = 2;
private static final int DRAIN_TIMEOUT = 2;
private static final int FS_PERCENT_THRESHOLD = 99;
private static final int GC_PERSISTED_QUEUE_FILE_INTERVAL_SECOND = 30;
private static final int SOCKET_TIMEOUT = 10*1000;
private static final String TYPE = "jenkins_plugin";
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
private final LogzioSender logzioSender;

private String key;
private String host;

//primary constructor used by indexer factory
public LogzioDao(String host, String key){
this(null, host, key);
}

// Factored for unit testing
LogzioDao(LogzioSender factory, String host, String key){
this.host = host;
this.key = key;

// create file for sender queue
File fp;
Jenkins jenkins = Jenkins.getInstanceOrNull();
if (jenkins == null){
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
// for testing (mvn package)
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
fp = Files.createTempDir();
}else{
fp = new File(jenkins.getRootDir() + "/logs/logzio_plugin");
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
boolean folderExisted = fp.mkdirs() || fp.setWritable(true);
if (!folderExisted) {
throw new IllegalArgumentException("Unable to create path");
}
}

try{
this.logzioSender = factory == null ? LogzioSender.getOrCreateSenderByType(key, TYPE, DRAIN_TIMEOUT,
FS_PERCENT_THRESHOLD, fp, host, SOCKET_TIMEOUT, CONNECT_TIMEOUT,false, null,
Executors.newScheduledThreadPool(CORE_POOL_SIZE),GC_PERSISTED_QUEUE_FILE_INTERVAL_SECOND, true) : factory;
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
this.logzioSender.start();
}catch (LogzioParameterErrorException e){
throw new IllegalArgumentException(e.getMessage());
}
}

@Override
public void push(String data) {
JSONObject jsonData = JSONObject.fromObject(data);
JSONArray logMessages = jsonData.getJSONArray("message");
for (Object logMsg : logMessages) {
JsonObject logLine = createLogLine(jsonData, logMsg.toString());
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
this.logzioSender.send(logLine);
}
}

protected JsonObject createLogLine(JSONObject jsonData, String logMsg) {
JsonObject logLine = new JsonObject();
logLine.addProperty("message", logMsg);
logLine.addProperty("@timestamp", LogstashConfiguration.getInstance().getDateFormatter().format(Calendar.getInstance().getTime()));
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
for(Object key : jsonData.keySet()){
String keyStr = (String) key;
if (!keyStr.equals("message")){
logLine.addProperty(key.toString(), jsonData.getString(key.toString()));
}
}
return logLine;
}

@Override
public JSONObject buildPayload(BuildData buildData, String jenkinsUrl, List<String> logLines) {
JSONObject payload = new JSONObject();
payload.put("message", logLines);
payload.put("source", "jenkins");
payload.put("source_host", jenkinsUrl);
payload.put("@buildTimestamp", buildData.getTimestamp());
payload.put("@version", 1);
// flatten build data
Map<String, Object> flattenJson = JsonFlattener.flattenAsMap(buildData.toString());
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
for (Map.Entry<String, Object> entry : flattenJson.entrySet()) {
String key = entry.getKey().replace('.','_');
Object value = entry.getValue();
payload.put(key, value);
}

return payload;
}

@Override
public String getDescription(){ return host; }

public String getHost(){ return host; }

public String getKey(){ return key; }

public String getType(){ return TYPE; }

}

2 changes: 1 addition & 1 deletion src/main/resources/index.jelly
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
<div>
Adds the possibility to push builds logs and build data to a Logstash indexer such as Redis, RabbitMQ, Elastic Search or to Syslog.
Adds the possibility to push builds logs and build data to a Logstash indexer such as Redis, RabbitMQ, Elastic Search, Logz.io or to Syslog.
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?jelly escape-by-default='true'?>
<j:jelly xmlns:j="jelly:core" xmlns:st="jelly:stapler" xmlns:d="jelly:define" xmlns:l="/lib/layout" xmlns:t="/lib/hudson" xmlns:f="/lib/form">
<f:entry title="${%Logz.io Host}" field="host">
<f:textbox default="https://listener.logz.io:8071"/>
</f:entry>
<f:entry title="${%Logz.io key}" field="key">
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
<f:password/>
</f:entry>
</j:jelly>
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
<div>
<p>Logz.io listener URL.<br/>
If you are in the EU region insert https://listener-eu.logz.io:8071. Otherwise, use https://listener.logz.io:8071<br/>
You can tell which region you are in by checking your login URL</p>
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
<div>
<p>The Logz.io account token.</p>
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
<?jelly escape-by-default='true'?>
<j:jelly xmlns:j="jelly:core" xmlns:st="jelly:stapler">
Push to Logz.io with HTTPs input.
</j:jelly>
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@
import jenkins.plugins.logstash.configuration.RabbitMq;
import jenkins.plugins.logstash.configuration.Redis;
import jenkins.plugins.logstash.configuration.Syslog;
import jenkins.plugins.logstash.configuration.Logzio;
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
import jenkins.plugins.logstash.persistence.LogstashIndexerDao.IndexerType;
import jenkins.plugins.logstash.persistence.LogstashIndexerDao.SyslogFormat;

Expand Down Expand Up @@ -151,4 +152,18 @@ public void rabbitMqMigration()
assertThat(es.getUsername(), equalTo("user"));
}

@Test
idohalevi marked this conversation as resolved.
Show resolved Hide resolved
public void logzioMigration()
{
when(descriptor.getType()).thenReturn(IndexerType.LOGZIO);
when(descriptor.getHost()).thenReturn("https://listener.logz.io:8071");
when(descriptor.getKey()).thenReturn("key");
configuration.migrateData();
LogstashIndexer<?> indexer = configuration.getLogstashIndexer();
assertThat(indexer, IsInstanceOf.instanceOf(Logzio.class));
assertThat(configuration.isMilliSecondTimestamps(),equalTo(false));
Logzio logz = (Logzio) indexer;
assertThat(logz.getKey(), equalTo("key"));
assertThat(logz.getHost(), equalTo("https://listener.logz.io:8071"));
}
}
Loading