-
Notifications
You must be signed in to change notification settings - Fork 58
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
10 changed files
with
449 additions
and
148 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
51 changes: 51 additions & 0 deletions
51
src/main/java/ch/qos/logback/more/appenders/AwsAppender.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
package ch.qos.logback.more.appenders; | ||
|
||
import com.amazonaws.auth.AWSCredentials; | ||
import com.amazonaws.auth.DefaultAWSCredentialsProviderChain; | ||
import com.amazonaws.auth.PropertiesCredentials; | ||
import ch.qos.logback.core.UnsynchronizedAppenderBase; | ||
|
||
public abstract class AwsAppender<E> extends UnsynchronizedAppenderBase<E> { | ||
|
||
protected AwsConfig config; | ||
protected AWSCredentials credentials; | ||
|
||
@Override | ||
public void start() { | ||
try { | ||
super.start(); | ||
if (config.getCredentialFilePath() != null | ||
&& config.getCredentialFilePath().length() > 0) { | ||
this.credentials = new PropertiesCredentials(getClass().getClassLoader() | ||
.getResourceAsStream(config.getCredentialFilePath())); | ||
} else { | ||
this.credentials = | ||
DefaultAWSCredentialsProviderChain.getInstance().getCredentials(); | ||
} | ||
} catch (Exception e) { | ||
addWarn("Could not initialize " + AwsAppender.class.getCanonicalName() | ||
+ " ( will try to initialize again later ): " + e); | ||
} | ||
} | ||
|
||
public static class AwsConfig { | ||
private String credentialFilePath; | ||
private String region; | ||
|
||
public void setCredentialFilePath(String credentialFilePath) { | ||
this.credentialFilePath = credentialFilePath; | ||
} | ||
|
||
public String getCredentialFilePath() { | ||
return credentialFilePath; | ||
} | ||
|
||
public String getRegion() { | ||
return region; | ||
} | ||
|
||
public void setRegion(String region) { | ||
this.region = region; | ||
} | ||
} | ||
} |
215 changes: 215 additions & 0 deletions
215
src/main/java/ch/qos/logback/more/appenders/CloudWatchLogbackAppender.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,215 @@ | ||
package ch.qos.logback.more.appenders; | ||
|
||
import java.util.List; | ||
import com.amazonaws.auth.AWSStaticCredentialsProvider; | ||
import com.amazonaws.services.logs.AWSLogs; | ||
import com.amazonaws.services.logs.AWSLogsClientBuilder; | ||
import com.amazonaws.services.logs.model.CreateLogGroupRequest; | ||
import com.amazonaws.services.logs.model.CreateLogStreamRequest; | ||
import com.amazonaws.services.logs.model.DescribeLogGroupsRequest; | ||
import com.amazonaws.services.logs.model.DescribeLogGroupsResult; | ||
import com.amazonaws.services.logs.model.DescribeLogStreamsRequest; | ||
import com.amazonaws.services.logs.model.DescribeLogStreamsResult; | ||
import com.amazonaws.services.logs.model.InputLogEvent; | ||
import com.amazonaws.services.logs.model.LogGroup; | ||
import com.amazonaws.services.logs.model.LogStream; | ||
import com.amazonaws.services.logs.model.PutLogEventsRequest; | ||
import com.amazonaws.services.logs.model.PutLogEventsResult; | ||
import ch.qos.logback.classic.spi.ILoggingEvent; | ||
import ch.qos.logback.core.encoder.EchoEncoder; | ||
import ch.qos.logback.core.encoder.Encoder; | ||
import ch.qos.logback.more.appenders.IntervalEmitter.EventMapper; | ||
import ch.qos.logback.more.appenders.IntervalEmitter.IntervalAppender; | ||
|
||
public class CloudWatchLogbackAppender<E> extends AwsAppender<E> { | ||
|
||
private IntervalEmitter<E, InputLogEvent> emitter; | ||
private AWSLogs awsLogs; | ||
private String logGroupName; | ||
private StreamName logStreamName; | ||
private boolean createLogDestination; | ||
private long emitInterval = 10000; | ||
private Encoder<E> encoder = new EchoEncoder<E>(); | ||
|
||
public void setAwsConfig(AwsConfig config) { | ||
this.config = config; | ||
} | ||
|
||
public void setLogGroupName(String logGroupName) { | ||
this.logGroupName = logGroupName; | ||
} | ||
|
||
public void setLogStreamName(String logStreamName) { | ||
this.logStreamName = new StaticStreamName(logStreamName); | ||
} | ||
|
||
public void setLogStreamRolling(StreamName streamName) { | ||
this.logStreamName = streamName; | ||
} | ||
|
||
public void setCreateLogDestination(boolean createLogDestination) { | ||
this.createLogDestination = createLogDestination; | ||
} | ||
|
||
public void setEmitInterval(long emitInterval) { | ||
this.emitInterval = emitInterval; | ||
} | ||
|
||
public void setEncoder(Encoder<E> encoder) { | ||
this.encoder = encoder; | ||
} | ||
|
||
@Override | ||
public void start() { | ||
super.start(); | ||
if (logGroupName == null || logGroupName.length() == 0 || logStreamName == null) { | ||
throw new IllegalArgumentException("logGroupName and logStreamName must be defined."); | ||
} | ||
this.awsLogs = AWSLogsClientBuilder.standard() | ||
.withCredentials(new AWSStaticCredentialsProvider(credentials)) | ||
.withRegion(config.getRegion()).build(); | ||
this.emitter = new IntervalEmitter<E, InputLogEvent>(emitInterval, | ||
new CloudWatchEventMapper(), new CloudWatchIntervalAppender()); | ||
} | ||
|
||
@Override | ||
public void stop() { | ||
try { | ||
emitter.emit(); | ||
} catch (Exception e) { | ||
// Ignore | ||
} | ||
try { | ||
super.stop(); | ||
} finally { | ||
try { | ||
awsLogs.shutdown(); | ||
} catch (Exception e) { | ||
// pass | ||
} | ||
} | ||
} | ||
|
||
@Override | ||
protected void append(E eventObject) { | ||
emitter.append(eventObject); | ||
} | ||
|
||
private void ensureLogGroup() { | ||
DescribeLogGroupsRequest request = | ||
new DescribeLogGroupsRequest().withLogGroupNamePrefix(logGroupName); | ||
DescribeLogGroupsResult result = awsLogs.describeLogGroups(request); | ||
for (LogGroup group : result.getLogGroups()) { | ||
if (logGroupName.equals(group.getLogGroupName())) { | ||
return; | ||
} | ||
} | ||
if (createLogDestination) { | ||
awsLogs.createLogGroup(new CreateLogGroupRequest(logGroupName)); | ||
} else { | ||
throw new IllegalStateException( | ||
"The specified log group does not exist: " + logGroupName); | ||
} | ||
} | ||
|
||
private String ensureLogStream(String name) { | ||
DescribeLogStreamsRequest request = new DescribeLogStreamsRequest() | ||
.withLogGroupName(logGroupName).withLogStreamNamePrefix(name); | ||
DescribeLogStreamsResult result = awsLogs.describeLogStreams(request); | ||
for (LogStream stream : result.getLogStreams()) { | ||
if (name.equals(stream.getLogStreamName())) { | ||
return stream.getUploadSequenceToken(); | ||
} | ||
} | ||
if (createLogDestination) { | ||
awsLogs.createLogStream(new CreateLogStreamRequest(logGroupName, name)); | ||
return null; | ||
} else { | ||
throw new IllegalStateException( | ||
"The specified log stream does not exist: " + logStreamName); | ||
} | ||
} | ||
|
||
private final class CloudWatchEventMapper implements EventMapper<E, InputLogEvent> { | ||
|
||
@Override | ||
public InputLogEvent map(E event) { | ||
InputLogEvent logEvent = new InputLogEvent(); | ||
if (event instanceof ILoggingEvent) { | ||
ILoggingEvent loggingEvent = (ILoggingEvent) event; | ||
logEvent.setTimestamp(loggingEvent.getTimeStamp()); | ||
} else { | ||
logEvent.setTimestamp(System.currentTimeMillis()); | ||
} | ||
logEvent.setMessage(new String(encoder.encode(event))); | ||
return logEvent; | ||
} | ||
} | ||
|
||
private final class CloudWatchIntervalAppender implements IntervalAppender<InputLogEvent> { | ||
private String sequenceToken; | ||
private boolean initialized = false; | ||
private boolean switchingStream = false; | ||
private String currentStreamName; | ||
|
||
@Override | ||
public boolean append(List<InputLogEvent> events) { | ||
if (!initialized) { | ||
synchronized (this) { | ||
if (!initialized) { | ||
ensureLogGroup(); | ||
initialized = true; | ||
} | ||
} | ||
return false; | ||
} | ||
if (switchingStream) { | ||
return false; | ||
} | ||
String streamName = logStreamName.get(events); | ||
if (!streamName.equals(currentStreamName)) { | ||
switchingStream = true; | ||
synchronized (this) { | ||
if (switchingStream) { | ||
sequenceToken = ensureLogStream(streamName); | ||
currentStreamName = streamName; | ||
switchingStream = false; | ||
} else { | ||
return false; | ||
} | ||
} | ||
} | ||
try { | ||
PutLogEventsRequest request = | ||
new PutLogEventsRequest(logGroupName, streamName, events); | ||
if (sequenceToken != null) { | ||
request.withSequenceToken(sequenceToken); | ||
} | ||
PutLogEventsResult result = awsLogs.putLogEvents(request); | ||
sequenceToken = result.getNextSequenceToken(); | ||
return true; | ||
} catch (RuntimeException e) { | ||
sequenceToken = null; | ||
e.printStackTrace(); | ||
throw e; | ||
} | ||
} | ||
} | ||
|
||
protected interface StreamName { | ||
String get(List<InputLogEvent> events); | ||
} | ||
|
||
public static class StaticStreamName implements StreamName { | ||
private String name; | ||
|
||
public StaticStreamName(String name) { | ||
this.name = name; | ||
} | ||
|
||
@Override | ||
public String get(List<InputLogEvent> events) { | ||
return name; | ||
} | ||
} | ||
} |
36 changes: 36 additions & 0 deletions
36
src/main/java/ch/qos/logback/more/appenders/CountBasedStreamName.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
package ch.qos.logback.more.appenders; | ||
|
||
import java.util.List; | ||
import java.util.UUID; | ||
import com.amazonaws.services.logs.model.InputLogEvent; | ||
import ch.qos.logback.more.appenders.CloudWatchLogbackAppender.StreamName; | ||
|
||
public class CountBasedStreamName implements StreamName { | ||
private long count = 0; | ||
private long limit = 1000; | ||
private String baseName = ""; | ||
private String currentName; | ||
|
||
public void setBaseName(String baseName) { | ||
this.baseName = baseName; | ||
} | ||
|
||
public void setLimit(long limit) { | ||
this.limit = limit; | ||
this.count = limit + 1; | ||
} | ||
|
||
@Override | ||
public String get(List<InputLogEvent> events) { | ||
count += events.size(); | ||
if (count > limit) { | ||
synchronized (this) { | ||
if (count > limit) { | ||
currentName = baseName + UUID.randomUUID(); | ||
count = events.size(); | ||
} | ||
} | ||
} | ||
return currentName; | ||
} | ||
} |
Oops, something went wrong.