From 4f918f195bbf5cd5211e15ed88c2a7a4094f6b6e Mon Sep 17 00:00:00 2001 From: Janus Varmarken Date: Mon, 14 Jan 2019 15:54:36 -0800 Subject: [PATCH] Extract ClusterMatcherObserver interface to separate file and move code that handles observer (de-)registration to AbstractClusterMatcher. --- .../detection/AbstractClusterMatcher.java | 23 +++++++++++++ .../detection/ClusterMatcherObserver.java | 26 +++++++++++++++ .../layer3/Layer3ClusterMatcher.java | 33 ++++--------------- .../detection/layer3/SignatureDetector.java | 6 ++-- 4 files changed, 60 insertions(+), 28 deletions(-) create mode 100644 Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/ClusterMatcherObserver.java diff --git a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/AbstractClusterMatcher.java b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/AbstractClusterMatcher.java index bda6493..45c6a55 100644 --- a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/AbstractClusterMatcher.java +++ b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/AbstractClusterMatcher.java @@ -2,6 +2,7 @@ package edu.uci.iotproject.detection; import org.pcap4j.core.PcapPacket; +import java.util.ArrayList; import java.util.List; import java.util.Objects; @@ -20,6 +21,11 @@ abstract public class AbstractClusterMatcher { */ protected final List> mCluster; + /** + * Observers registered for callbacks from this {@link AbstractClusterMatcher}. + */ + protected final List mObservers; + protected AbstractClusterMatcher(List> cluster) { // ===================== PRECONDITION SECTION ===================== cluster = Objects.requireNonNull(cluster, "cluster cannot be null"); @@ -34,6 +40,23 @@ abstract public class AbstractClusterMatcher { // ================================================================ // Let the subclass prune the provided cluster mCluster = pruneCluster(cluster); + mObservers = new ArrayList<>(); + } + + /** + * Register for callbacks from this cluster matcher. + * @param observer The target of the callbacks. + */ + public final void addObserver(ClusterMatcherObserver observer) { + mObservers.add(observer); + } + + /** + * Deregister for callbacks from this cluster matcher. + * @param observer The callback target that is to be deregistered. + */ + public final void removeObserver(ClusterMatcherObserver observer) { + mObservers.remove(observer); } /** diff --git a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/ClusterMatcherObserver.java b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/ClusterMatcherObserver.java new file mode 100644 index 0000000..d67c520 --- /dev/null +++ b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/ClusterMatcherObserver.java @@ -0,0 +1,26 @@ +package edu.uci.iotproject.detection; + +import org.pcap4j.core.PcapPacket; + +import java.util.List; + +/** + * Interface used by client code to register for receiving a notification whenever an {@link AbstractClusterMatcher} + * detects traffic that matches an element of its associated cluster. + * + * @author Janus Varmarken {@literal } + * @author Rahmadi Trimananda {@literal } + */ +public interface ClusterMatcherObserver { + + /** + * Callback that is invoked by an {@link AbstractClusterMatcher} whenever it detects traffic that matches an element + * of its associated cluster. + * + * @param clusterMatcher The {@link AbstractClusterMatcher} that detected a match (i.e., classified traffic as + * pertaining to its associated cluster). + * @param match The traffic that was deemed to match the cluster associated with {@code clusterMatcher}. + */ + void onMatch(AbstractClusterMatcher clusterMatcher, List match); + +} diff --git a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/Layer3ClusterMatcher.java b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/Layer3ClusterMatcher.java index b5e3077..81002bd 100644 --- a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/Layer3ClusterMatcher.java +++ b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/Layer3ClusterMatcher.java @@ -1,6 +1,7 @@ package edu.uci.iotproject.detection.layer3; import edu.uci.iotproject.detection.AbstractClusterMatcher; +import edu.uci.iotproject.detection.ClusterMatcherObserver; import edu.uci.iotproject.trafficreassembly.layer3.Conversation; import edu.uci.iotproject.trafficreassembly.layer3.TcpReassembler; import edu.uci.iotproject.analysis.TcpConversationUtils; @@ -65,8 +66,6 @@ public class Layer3ClusterMatcher extends AbstractClusterMatcher implements Pack */ private final String mRouterWanIp; - private final ClusterMatchObserver[] mObservers; - /** * Create a {@link Layer3ClusterMatcher}. * @param cluster The cluster that traffic is matched against. @@ -77,12 +76,12 @@ public class Layer3ClusterMatcher extends AbstractClusterMatcher implements Pack * {@code cluster}, i.e., when the examined traffic is classified as pertaining to * {@code cluster}. */ - public Layer3ClusterMatcher(List> cluster, String routerWanIp, ClusterMatchObserver... detectionObservers) { + public Layer3ClusterMatcher(List> cluster, String routerWanIp, + ClusterMatcherObserver... detectionObservers) { super(cluster); - // ===================== PRECONDITION SECTION ===================== - mObservers = Objects.requireNonNull(detectionObservers, "detectionObservers cannot be null"); - if (mObservers.length == 0) { - throw new IllegalArgumentException("no detectionObservers provided"); + Objects.requireNonNull(detectionObservers, "detectionObservers cannot be null"); + for (ClusterMatcherObserver obs : detectionObservers) { + addObserver(obs); } // Build the cluster members' direction sequence. // Note: assumes that the provided cluster was captured within the local network (routerWanIp is set to null). @@ -100,7 +99,6 @@ public class Layer3ClusterMatcher extends AbstractClusterMatcher implements Pack "pattern" ); } - // ================================================================ mRouterWanIp = routerWanIp; } @@ -148,7 +146,7 @@ public class Layer3ClusterMatcher extends AbstractClusterMatcher implements Pack isPresent()) { List matchSeq = match.get(); // Notify observers about the match. - Arrays.stream(mObservers).forEach(o -> o.onMatch(Layer3ClusterMatcher.this, matchSeq)); + mObservers.forEach(o -> o.onMatch(Layer3ClusterMatcher.this, matchSeq)); /* * Get the index in cPkts of the last packet in the sequence of packets that matches the searched * signature sequence. @@ -333,21 +331,4 @@ public class Layer3ClusterMatcher extends AbstractClusterMatcher implements Pack return directions; } - /** - * Interface used by client code to register for receiving a notification whenever the {@link Layer3ClusterMatcher} - * detects traffic that is similar to the traffic that makes up the cluster returned by - * {@link Layer3ClusterMatcher#getCluster()}. - */ - interface ClusterMatchObserver { - /** - * Callback that is invoked whenever a sequence that is similar to a sequence associated with the cluster (i.e., - * a sequence is a member of the cluster) is detected in the traffic that the associated {@link Layer3ClusterMatcher} - * observes. - * @param clusterMatcher The {@link Layer3ClusterMatcher} that detected a match (classified traffic as pertaining to - * its associated cluster). - * @param match The traffic that was deemed to match the cluster associated with {@code clusterMatcher}. - */ - void onMatch(Layer3ClusterMatcher clusterMatcher, List match); - } - } diff --git a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/SignatureDetector.java b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/SignatureDetector.java index ad47106..4095074 100644 --- a/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/SignatureDetector.java +++ b/Code/Projects/SmartPlugDetector/src/main/java/edu/uci/iotproject/detection/layer3/SignatureDetector.java @@ -2,6 +2,8 @@ package edu.uci.iotproject.detection.layer3; import edu.uci.iotproject.analysis.TriggerTrafficExtractor; import edu.uci.iotproject.analysis.UserAction; +import edu.uci.iotproject.detection.AbstractClusterMatcher; +import edu.uci.iotproject.detection.ClusterMatcherObserver; import edu.uci.iotproject.io.PcapHandleReader; import edu.uci.iotproject.util.PrintUtils; import org.jgrapht.GraphPath; @@ -23,7 +25,7 @@ import java.util.function.Consumer; * @author Janus Varmarken {@literal } * @author Rahmadi Trimananda {@literal } */ -public class SignatureDetector implements PacketListener, Layer3ClusterMatcher.ClusterMatchObserver { +public class SignatureDetector implements PacketListener, ClusterMatcherObserver { // Test client public static void main(String[] args) throws PcapNativeException, NotOpenException { @@ -520,7 +522,7 @@ public class SignatureDetector implements PacketListener, Layer3ClusterMatcher.C } @Override - public void onMatch(Layer3ClusterMatcher clusterMatcher, List match) { + public void onMatch(AbstractClusterMatcher clusterMatcher, List match) { // Add the match at the corresponding index pendingMatches[mClusterMatcherIds.get(clusterMatcher)].add(match); checkSignatureMatch(); -- 2.34.1