forked from Morningstar/kafka-offset-monitor
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #1 from Morningstar/master
Merge
- Loading branch information
Showing
24 changed files
with
1,245 additions
and
222 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
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -57,7 +57,7 @@ This is a small web app, you can run it locally or on a server, as long as you h | |
``` | ||
java -Djava.security.auth.login.config=conf/server-client-jaas.conf \ | ||
-cp KafkaOffsetMonitor-assembly-0.4.0.jar \ | ||
-cp KafkaOffsetMonitor-assembly-0.4.6.jar \ | ||
com.quantifind.kafka.offsetapp.OffsetGetterWeb \ | ||
--offsetStorage kafka \ | ||
--kafkaBrokers kafkabroker01:6667,kafkabroker02:6667 \ | ||
|
@@ -126,7 +126,7 @@ As long as this is true you will need to use local maven repo and just publish K | |
Assuming you have a custom implementation of OffsetInfoReporter in a jar file, running it is as simple as adding the jar to the classpath when running app: | ||
|
||
``` | ||
java -cp KafkaOffsetMonitor-assembly-0.3.0.jar:kafka-offset-monitor-another-db-reporter.jar \ | ||
java -cp KafkaOffsetMonitor-assembly-0.4.6.jar:kafka-offset-monitor-another-db-reporter.jar \ | ||
com.quantifind.kafka.offsetapp.OffsetGetterWeb \ | ||
--zk zkserver01,zkserver02 \ | ||
--port 8080 \ | ||
|
@@ -141,7 +141,3 @@ Contributing | |
============ | ||
|
||
The KafkaOffsetMonitor is released under the Apache License and we **welcome any contributions** within this license. Any pull request is welcome and will be reviewed and merged as quickly as possible. | ||
|
||
Because this open source tool is released by [Quantifind](http://www.quantifind.com) as a company, if you want to submit a pull request, you will have to sign the following simple contributors agreement: | ||
- If you are an individual, please sign [this contributors agreement](https://docs.google.com/a/quantifind.com/document/d/1RS7qEjq3cCmJ1665UhoCMK8541Ms7KyU3kVFoO4CR_I/) and send it back to [email protected] | ||
- If you are contributing changes that you did as part of your work, please sign [this contributors agreement](https://docs.google.com/a/quantifind.com/document/d/1kNwLT4qG3G0Ct2mEuNdBGmKDYuApN1CpQtZF8TSVTjE/) and send it back to [email protected] |
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
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
50 changes: 50 additions & 0 deletions
50
src/main/java/com/morningstar/kafka/KafkaCommittedOffset.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,50 @@ | ||
package com.morningstar.kafka; | ||
|
||
import com.google.common.base.Preconditions; | ||
import com.google.common.base.Strings; | ||
|
||
public class KafkaCommittedOffset { | ||
|
||
private String groupName; | ||
private boolean groupIsActive; | ||
private String topicName; | ||
private int partitionId; | ||
private KafkaCommittedOffsetMetadata committedOffset; | ||
|
||
|
||
public KafkaCommittedOffset(String groupName, boolean groupIsActive, String topicName, int partitionId, long committedOffset, long committedMillis) { | ||
|
||
Preconditions.checkArgument(!Strings.isNullOrEmpty(groupName), "groupName must not be NULL or empty."); | ||
Preconditions.checkArgument(!Strings.isNullOrEmpty(topicName), "topicName must not be NULL or empty."); | ||
Preconditions.checkArgument(partitionId > -1, "partitionId must be greater than or equal to 0."); | ||
Preconditions.checkArgument(committedOffset > -1, "committedOffset must be greater than or equal to 0."); | ||
Preconditions.checkArgument(committedMillis > -1, "committedMillis must be greater than or equal to 0."); | ||
|
||
this.groupName = groupName; | ||
this.groupIsActive = groupIsActive; | ||
this.topicName = topicName; | ||
this.partitionId = partitionId; | ||
this.committedOffset = new KafkaCommittedOffsetMetadata(committedOffset, committedMillis); | ||
} | ||
|
||
|
||
public String getGroupName() { | ||
return groupName; | ||
} | ||
|
||
public boolean getGroupIsActive() { | ||
return groupIsActive; | ||
} | ||
|
||
public String getTopicName() { | ||
return topicName; | ||
} | ||
|
||
public int getPartitionId() { | ||
return partitionId; | ||
} | ||
|
||
public KafkaCommittedOffsetMetadata getCommittedOffset() { | ||
return committedOffset; | ||
} | ||
} |
44 changes: 44 additions & 0 deletions
44
src/main/java/com/morningstar/kafka/KafkaCommittedOffsetMetadata.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,44 @@ | ||
package com.morningstar.kafka; | ||
|
||
import com.google.common.base.Preconditions; | ||
import com.google.gson.annotations.Expose; | ||
|
||
public class KafkaCommittedOffsetMetadata extends KafkaOffsetMetadata { | ||
|
||
@Expose private long lag = -1; | ||
|
||
|
||
public KafkaCommittedOffsetMetadata(KafkaOffsetMetadata offsetMetadata, long lag) { | ||
super(offsetMetadata.getOffset(), offsetMetadata.getTimestamp()); | ||
verifyParameters(lag); | ||
this.lag = lag; | ||
} | ||
|
||
public KafkaCommittedOffsetMetadata(long committedOffset, long timestamp, long lag) { | ||
super(committedOffset, timestamp); | ||
verifyParameters(lag); | ||
this.lag = lag; | ||
} | ||
|
||
public KafkaCommittedOffsetMetadata(KafkaOffsetMetadata offsetMetadata) { | ||
super(offsetMetadata.getOffset(), offsetMetadata.getTimestamp()); | ||
} | ||
|
||
public KafkaCommittedOffsetMetadata(long committedOffset, long timestamp) { | ||
super(committedOffset, timestamp); | ||
} | ||
|
||
private void verifyParameters(long lag) { | ||
|
||
Preconditions.checkArgument(lag > -2, "lag must not be less than -1."); | ||
} | ||
|
||
|
||
public long getLag() { | ||
return lag; | ||
} | ||
|
||
public void setLag(long lag) { | ||
this.lag = lag; | ||
} | ||
} |
168 changes: 168 additions & 0 deletions
168
src/main/java/com/morningstar/kafka/KafkaConsumerGroup.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,168 @@ | ||
package com.morningstar.kafka; | ||
|
||
import com.google.common.base.Preconditions; | ||
import com.google.common.base.Strings; | ||
import com.google.common.collect.Sets; | ||
import com.google.gson.annotations.Expose; | ||
|
||
import java.util.Optional; | ||
import java.util.Set; | ||
import java.util.stream.Collectors; | ||
|
||
|
||
public class KafkaConsumerGroup { | ||
|
||
private final long COMPLETE_THRESHOLD = 10; | ||
|
||
@Expose private String consumerGroupName; | ||
@Expose private boolean isActive; | ||
@Expose private boolean complete; | ||
@Expose private long mostRecentCommittedMillis; | ||
@Expose private Status status; | ||
@Expose private Set<KafkaTopicPartition> topicPartitions; | ||
|
||
|
||
public KafkaConsumerGroup(String consumerGroupName) { | ||
|
||
Preconditions.checkArgument(!Strings.isNullOrEmpty(consumerGroupName), "consumerGroupName cannot be NULL or empty."); | ||
|
||
this.consumerGroupName = consumerGroupName; | ||
this.isActive = false; | ||
this.complete = false; | ||
this.mostRecentCommittedMillis = -1; | ||
this.status = Status.OK; | ||
this.topicPartitions = Sets.newConcurrentHashSet(); | ||
} | ||
|
||
|
||
public String getConsumerGroupName() { | ||
return consumerGroupName; | ||
} | ||
|
||
public boolean getComplete() { return complete; } | ||
|
||
public long getMaxCommitedMillis() { | ||
return mostRecentCommittedMillis; | ||
} | ||
|
||
public boolean isActive() { | ||
return isActive; | ||
} | ||
|
||
public Set<KafkaTopicPartition> getTopicPartitions() { | ||
|
||
return topicPartitions; | ||
} | ||
|
||
public Set<String> getTopics() { | ||
|
||
return topicPartitions.stream() | ||
.map(KafkaTopicPartition::getTopicName) | ||
.collect(Collectors.toSet()); | ||
} | ||
|
||
public synchronized void updateStatus() { | ||
|
||
if (!isActive) { | ||
this.status = Status.ERR; | ||
return; | ||
} | ||
|
||
Status newStatus = Status.OK; | ||
|
||
for (KafkaTopicPartition topicPartition : topicPartitions) { | ||
|
||
// Set group status to ERROR if any topicPartition's status is STOP | ||
if (Status.STOP == topicPartition.getStatus()) { | ||
newStatus = Status.ERR; | ||
break; | ||
} | ||
|
||
// Set group status to ERROR if any topicPartition's status is REWIND | ||
if (Status.REWIND == topicPartition.getStatus()) { | ||
newStatus = Status.ERR; | ||
break; | ||
} | ||
|
||
// Set group status to ERROR if any topicPartition's status is STALL | ||
if (Status.STALL == topicPartition.getStatus()) { | ||
newStatus = Status.ERR; | ||
break; | ||
} | ||
|
||
// Set group status to WARN if any topicPartition's status is WARN | ||
if (Status.WARN == topicPartition.getStatus()) { | ||
newStatus = Status.WARN; | ||
break; | ||
} | ||
} | ||
|
||
this.status = newStatus; | ||
} | ||
|
||
|
||
private Optional<KafkaTopicPartition> getTopicPartition(String topic, int partitionId) { | ||
|
||
//return committedOffsets.keySet().stream().filter(tp -> (tp.getTopicName().equals(topic) && tp.getPartitionId() == partitionId)).findFirst(); | ||
return topicPartitions.stream() | ||
.filter(tp -> (tp.getTopicName().equals(topic) && tp.getPartitionId() == partitionId)) | ||
.findFirst(); | ||
} | ||
|
||
private void upsertTopicPartition(KafkaCommittedOffset kafkaCommittedOffset) { | ||
|
||
Preconditions.checkArgument(!Strings.isNullOrEmpty(kafkaCommittedOffset.getTopicName()), "topic cannot be NULL or empty."); | ||
Preconditions.checkArgument(kafkaCommittedOffset.getPartitionId() >= 0, "partitionId must be greater-than or equal-to zero."); | ||
|
||
String incomingTopicName = kafkaCommittedOffset.getTopicName(); | ||
int incomingPartitionId = kafkaCommittedOffset.getPartitionId(); | ||
|
||
Optional<KafkaTopicPartition> existingTopicPartition = getTopicPartition(incomingTopicName, incomingPartitionId); | ||
|
||
if (existingTopicPartition.isPresent()) { | ||
// Append committed offset info to existing set item | ||
existingTopicPartition.get().addCommittedOffset(kafkaCommittedOffset.getCommittedOffset()); | ||
} else { | ||
// Add a new entry to the map | ||
KafkaTopicPartition newTopicPartition = new KafkaTopicPartition(incomingTopicName, incomingPartitionId); | ||
newTopicPartition.addCommittedOffset(kafkaCommittedOffset.getCommittedOffset()); | ||
topicPartitions.add(newTopicPartition); | ||
} | ||
} | ||
|
||
private void setMostRecentCommittedMillis(long mostRecentCommittedMillis) { | ||
if (this.mostRecentCommittedMillis < mostRecentCommittedMillis) { | ||
this.mostRecentCommittedMillis = mostRecentCommittedMillis; | ||
} | ||
} | ||
|
||
private void updateCompleteFlag() { | ||
|
||
this.complete = topicPartitions.stream() | ||
.noneMatch(f -> f.getCommittedOffsets().size() < COMPLETE_THRESHOLD); | ||
} | ||
|
||
public void addCommittedOffsetInfo(KafkaCommittedOffset kafkaCommittedOffset) { | ||
|
||
setMostRecentCommittedMillis(kafkaCommittedOffset.getCommittedOffset().getTimestamp()); | ||
this.isActive = kafkaCommittedOffset.getGroupIsActive(); | ||
upsertTopicPartition(kafkaCommittedOffset); | ||
updateCompleteFlag(); | ||
} | ||
|
||
|
||
@Override | ||
public boolean equals(Object o) { | ||
if (this == o) return true; | ||
if (o == null || getClass() != o.getClass()) return false; | ||
|
||
KafkaConsumerGroup that = (KafkaConsumerGroup) o; | ||
|
||
return getConsumerGroupName().equals(that.getConsumerGroupName()); | ||
} | ||
|
||
@Override | ||
public int hashCode() { | ||
return getConsumerGroupName().hashCode(); | ||
} | ||
} |
Oops, something went wrong.