Initial commit
This commit is contained in:
commit
1ff10f0cc5
1
.idea/.name
generated
Normal file
1
.idea/.name
generated
Normal file
@ -0,0 +1 @@
|
|||||||
|
Broadcast
|
9
.idea/artifacts/Broadcast_jar.xml
generated
Normal file
9
.idea/artifacts/Broadcast_jar.xml
generated
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<component name="ArtifactManager">
|
||||||
|
<artifact type="jar" name="Broadcast:jar">
|
||||||
|
<output-path>$PROJECT_DIR$/out/artifacts/Broadcast_jar</output-path>
|
||||||
|
<root id="archive" name="Broadcast.jar">
|
||||||
|
<element id="module-output" name="Broadcast" />
|
||||||
|
<element id="dir-copy" path="$PROJECT_DIR$/src" />
|
||||||
|
</root>
|
||||||
|
</artifact>
|
||||||
|
</component>
|
15
.idea/inspectionProfiles/Project_Default.xml
generated
Normal file
15
.idea/inspectionProfiles/Project_Default.xml
generated
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
<component name="InspectionProjectProfileManager">
|
||||||
|
<profile version="1.0">
|
||||||
|
<option name="myName" value="Project Default" />
|
||||||
|
<inspection_tool class="EmptyCatchBlock" enabled="false" level="WARNING" enabled_by_default="false">
|
||||||
|
<option name="m_includeComments" value="true" />
|
||||||
|
<option name="m_ignoreTestCases" value="true" />
|
||||||
|
<option name="m_ignoreIgnoreParameter" value="true" />
|
||||||
|
</inspection_tool>
|
||||||
|
<inspection_tool class="WeakerAccess" enabled="false" level="WARNING" enabled_by_default="false">
|
||||||
|
<option name="SUGGEST_PACKAGE_LOCAL_FOR_MEMBERS" value="true" />
|
||||||
|
<option name="SUGGEST_PACKAGE_LOCAL_FOR_TOP_CLASSES" value="true" />
|
||||||
|
<option name="SUGGEST_PRIVATE_FOR_INNERS" value="false" />
|
||||||
|
</inspection_tool>
|
||||||
|
</profile>
|
||||||
|
</component>
|
6
.idea/misc.xml
generated
Normal file
6
.idea/misc.xml
generated
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_9" default="true" project-jdk-name="9.0" project-jdk-type="JavaSDK">
|
||||||
|
<output url="file://$PROJECT_DIR$/out" />
|
||||||
|
</component>
|
||||||
|
</project>
|
8
.idea/modules.xml
generated
Normal file
8
.idea/modules.xml
generated
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ProjectModuleManager">
|
||||||
|
<modules>
|
||||||
|
<module fileurl="file://$PROJECT_DIR$/Broadcast.iml" filepath="$PROJECT_DIR$/Broadcast.iml" />
|
||||||
|
</modules>
|
||||||
|
</component>
|
||||||
|
</project>
|
11
Broadcast.iml
Normal file
11
Broadcast.iml
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<module type="JAVA_MODULE" version="4">
|
||||||
|
<component name="NewModuleRootManager" inherit-compiler-output="true">
|
||||||
|
<exclude-output />
|
||||||
|
<content url="file://$MODULE_DIR$">
|
||||||
|
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" />
|
||||||
|
</content>
|
||||||
|
<orderEntry type="inheritedJdk" />
|
||||||
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
|
</component>
|
||||||
|
</module>
|
202
src/net/tofvesson/broadcast/client/Accumulator.java
Normal file
202
src/net/tofvesson/broadcast/client/Accumulator.java
Normal file
@ -0,0 +1,202 @@
|
|||||||
|
package net.tofvesson.broadcast.client;
|
||||||
|
|
||||||
|
import net.tofvesson.broadcast.support.ImmutableArray;
|
||||||
|
import net.tofvesson.broadcast.support.ImmutableReferenceMap;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.*;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Accumulates a list of hosts that are broadcasting to a given port.
|
||||||
|
*/
|
||||||
|
public class Accumulator {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Timeout (in milliseconds) until a host is considered to no longer be active.
|
||||||
|
*/
|
||||||
|
public static final long DEFAULT_HOST_TIMEOUT = 60_000;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Minimum timeout until a host is considered inactive.
|
||||||
|
*/
|
||||||
|
public static final long MINIMUM_HOST_TIMEOUT = 50;
|
||||||
|
|
||||||
|
private final Runnable accumulate;
|
||||||
|
private Thread accumulatorThread;
|
||||||
|
private final int port;
|
||||||
|
|
||||||
|
protected final DatagramSocket socket;
|
||||||
|
protected final DatagramPacket packet;
|
||||||
|
|
||||||
|
protected volatile long continueUntil = -1;
|
||||||
|
|
||||||
|
protected final Map<InetAddress, Long> hosts = new HashMap<>();
|
||||||
|
protected final ImmutableReferenceMap<InetAddress, Long> accessHosts = new ImmutableReferenceMap<>(hosts);
|
||||||
|
protected final long hostTimeout;
|
||||||
|
protected final ImmutableArray<Byte> signature;
|
||||||
|
|
||||||
|
public OnHostTimeoutListener timeoutListener;
|
||||||
|
public OnNewHostListener newHostListener;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an accumulator
|
||||||
|
* @param port Port to listen on.
|
||||||
|
* @param hostTimeout Timeout (milliseconds) until a host is lost (unless another broadcast is detected)
|
||||||
|
* @param sig Signature to check if a broadcast is originating from a compatible source.
|
||||||
|
* @throws SocketException Thrown if port is already bound
|
||||||
|
* @throws SecurityException Thrown if program isn't allowed to accept broadcasts
|
||||||
|
*/
|
||||||
|
public Accumulator(int port, long hostTimeout, byte[] sig) throws SocketException, SecurityException {
|
||||||
|
if(System.getSecurityManager()!=null) System.getSecurityManager().checkAccept("255.255.255.255", port); // Do a permission check
|
||||||
|
|
||||||
|
this.socket = new DatagramSocket(this.port = port);
|
||||||
|
socket.setSoTimeout(125);
|
||||||
|
this.packet = new DatagramPacket(new byte[sig.length + 1], sig.length + 1); // One extra byte implicitly serves as metadata during packet comparison
|
||||||
|
this.hostTimeout = hostTimeout;
|
||||||
|
this.signature = ImmutableArray.from(sig);
|
||||||
|
|
||||||
|
accumulate = () -> {
|
||||||
|
while(continueUntil==-1 || System.currentTimeMillis()<continueUntil){
|
||||||
|
try {
|
||||||
|
socket.receive(packet);
|
||||||
|
if(signature.length() == packet.getLength() && signature.compare(ImmutableArray.from(packet.getData()), packet.getOffset(), 0, packet.getLength()))
|
||||||
|
synchronized (hosts){
|
||||||
|
if(hosts.containsKey(packet.getAddress())) hosts.replace(packet.getAddress(), System.currentTimeMillis());
|
||||||
|
else{
|
||||||
|
hosts.put(packet.getAddress(), System.currentTimeMillis());
|
||||||
|
if(newHostListener!=null) newHostListener.onNewHost(packet.getAddress(), port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (SocketTimeoutException e) {
|
||||||
|
// NOP
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an accumulator
|
||||||
|
* @param port Port to listen on.
|
||||||
|
* @param signature Signature to check if a broadcast is originating from a compatible source.
|
||||||
|
* @throws SocketException Thrown if port is already bound
|
||||||
|
* @throws SecurityException Thrown if program isn't allowed to accept broadcasts
|
||||||
|
*/
|
||||||
|
public Accumulator(int port, byte[] signature) throws SocketException, SecurityException { this(port, DEFAULT_HOST_TIMEOUT, signature); }
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start the accumulator and let it run indefinitely
|
||||||
|
*/
|
||||||
|
public void start(){ startFor(-1); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start accumulator
|
||||||
|
* @param timeMillis Milliseconds until accumulator should automatically stop
|
||||||
|
*/
|
||||||
|
public void startFor(long timeMillis){
|
||||||
|
continueUntil = timeMillis==-1?-1:System.currentTimeMillis()+timeMillis;
|
||||||
|
TimeoutManager.theManager.addAccumulator(this);
|
||||||
|
if(accumulatorThread!=null && accumulatorThread.isAlive()) throw new IllegalStateException("Thread is still alive!");
|
||||||
|
accumulatorThread = new Thread(accumulate);
|
||||||
|
accumulatorThread.setName("Accumulator-"+port);
|
||||||
|
accumulatorThread.setPriority(Thread.MAX_PRIORITY);
|
||||||
|
accumulatorThread.setDaemon(true);
|
||||||
|
accumulatorThread.setUncaughtExceptionHandler((t, e) -> {
|
||||||
|
e.printStackTrace();
|
||||||
|
// NOP
|
||||||
|
});
|
||||||
|
accumulatorThread.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pause the accumulator. Resume by calling {@link #start()}
|
||||||
|
*/
|
||||||
|
public void pause(){
|
||||||
|
continueUntil = 0;
|
||||||
|
if(accumulatorThread!=null && accumulatorThread.isAlive())
|
||||||
|
try {
|
||||||
|
accumulatorThread.join();
|
||||||
|
}
|
||||||
|
catch (InterruptedException e) { e.printStackTrace(); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop accumulating hosts and disable host timeout checks. Cannot be resumed from
|
||||||
|
*/
|
||||||
|
public void stop(){
|
||||||
|
pause();
|
||||||
|
TimeoutManager.theManager.removeAccumulator(this);
|
||||||
|
socket.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an immutable map of the currently available hosts.
|
||||||
|
* @return Immutable host map.
|
||||||
|
*/
|
||||||
|
public ImmutableReferenceMap<InetAddress, Long> getHosts() { return accessHosts; }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback for when a new host is found
|
||||||
|
*/
|
||||||
|
public interface OnNewHostListener{
|
||||||
|
/**
|
||||||
|
* Called when a new host is found.
|
||||||
|
* @param host The IP address of the host
|
||||||
|
* @param port Listener port
|
||||||
|
*/
|
||||||
|
void onNewHost(InetAddress host, int port);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback for timed out remote hosts
|
||||||
|
*/
|
||||||
|
public interface OnHostTimeoutListener{
|
||||||
|
/**
|
||||||
|
* Called when a host is timed out.
|
||||||
|
* @param host The IP address of the host
|
||||||
|
* @param port Listener port
|
||||||
|
*/
|
||||||
|
void onTimeout(InetAddress host, int port);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private static class TimeoutManager{
|
||||||
|
static TimeoutManager theManager = new TimeoutManager();
|
||||||
|
|
||||||
|
private final List<Accumulator> checks = new ArrayList<>();
|
||||||
|
|
||||||
|
public TimeoutManager() {
|
||||||
|
Thread t = new Thread(() -> {
|
||||||
|
//noinspection InfiniteLoopStatement
|
||||||
|
while(true){
|
||||||
|
try{ Thread.sleep(Accumulator.MINIMUM_HOST_TIMEOUT); }catch(Exception e){}
|
||||||
|
Accumulator[] a1;
|
||||||
|
synchronized (checks){ a1 = checks.toArray(new Accumulator[checks.size()]); }
|
||||||
|
for(Accumulator a : a1)
|
||||||
|
synchronized (a.hosts){
|
||||||
|
for(InetAddress addr : a.hosts.keySet())
|
||||||
|
if(a.hosts.get(addr) <=System.currentTimeMillis()-a.hostTimeout) {
|
||||||
|
a.hosts.remove(addr);
|
||||||
|
if(a.timeoutListener!=null) a.timeoutListener.onTimeout(addr, a.port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
t.setDaemon(true);
|
||||||
|
t.setPriority(Thread.MIN_PRIORITY);
|
||||||
|
t.setName("Accumulator-TimeoutManager");
|
||||||
|
t.setUncaughtExceptionHandler((r, e)->{});
|
||||||
|
t.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
void addAccumulator(Accumulator a){ checks.removeIf(it -> it==a); synchronized (checks){ checks.add(a); } }
|
||||||
|
void removeAccumulator(Accumulator a){ checks.removeIf(it -> it==a); synchronized (checks){ checks.remove(a); } }
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
98
src/net/tofvesson/broadcast/server/Server.java
Normal file
98
src/net/tofvesson/broadcast/server/Server.java
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
package net.tofvesson.broadcast.server;
|
||||||
|
|
||||||
|
import java.net.*;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Broadcasts a signature on the current subnet
|
||||||
|
*/
|
||||||
|
public class Server {
|
||||||
|
|
||||||
|
private final Runnable serve;
|
||||||
|
private Thread serverThread;
|
||||||
|
private final int port;
|
||||||
|
|
||||||
|
protected final DatagramSocket serverSocket;
|
||||||
|
protected final DatagramPacket packet;
|
||||||
|
|
||||||
|
protected final AtomicBoolean isAlive = new AtomicBoolean(false);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create broadcaster
|
||||||
|
* @param port Port to broadcast to
|
||||||
|
* @param delay Millisecond delay between broadcasts
|
||||||
|
* @param sig Signature to broadcast
|
||||||
|
* @param offset Offset in the signature to broadcast
|
||||||
|
* @param length Length of signature to broadcast
|
||||||
|
* @throws SocketException Thrown if broadcast socket could not be created
|
||||||
|
*/
|
||||||
|
public Server(int port, long delay, byte[] sig, int offset, int length) throws SocketException {
|
||||||
|
try {
|
||||||
|
this.serverSocket = new DatagramSocket();
|
||||||
|
this.packet = new DatagramPacket(sig, offset, length, InetAddress.getByAddress(new byte[]{-1, -1, -1, -1}), this.port = port);
|
||||||
|
serve = () -> {
|
||||||
|
while(getIsAlive()){
|
||||||
|
try {
|
||||||
|
serverSocket.send(packet);
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
try { Thread.sleep(delay); } catch (InterruptedException e) { }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} catch (UnknownHostException e) {
|
||||||
|
throw new RuntimeException(e); // This is an internal Java error and should not be a declared exception
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create broadcaster
|
||||||
|
* @param port Port to broadcast to
|
||||||
|
* @param delay Millisecond delay between broadcasts
|
||||||
|
* @param sig Signature to broadcast
|
||||||
|
* @throws SocketException Thrown if broadcast socket could not be created
|
||||||
|
*/
|
||||||
|
public Server(int port, long delay, byte[] sig) throws SocketException { this(port, delay, sig, 0, sig.length); }
|
||||||
|
|
||||||
|
protected boolean getIsAlive(){
|
||||||
|
boolean b;
|
||||||
|
synchronized (isAlive){ b = isAlive.get(); }
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setIsAlive(boolean b){ synchronized (isAlive){ isAlive.set(b); } }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start broadcasting signature
|
||||||
|
*/
|
||||||
|
public void start(){
|
||||||
|
if(serverSocket.isClosed()) throw new IllegalStateException("Socket is closed");
|
||||||
|
if(serverThread!=null && serverThread.isAlive()) throw new IllegalStateException("Server is still alive");
|
||||||
|
setIsAlive(true);
|
||||||
|
serverThread = new Thread(serve);
|
||||||
|
serverThread.setDaemon(true);
|
||||||
|
serverThread.setPriority(Thread.MAX_PRIORITY);
|
||||||
|
serverThread.setName("Server-"+port);
|
||||||
|
serverThread.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pause broadcasting of signature. Can be resumed by calling {@link #start()}
|
||||||
|
*/
|
||||||
|
public void pause(){
|
||||||
|
if(serverSocket.isClosed() && (serverThread==null || !serverThread.isAlive())) throw new IllegalStateException("Socket is closed");
|
||||||
|
setIsAlive(false);
|
||||||
|
serverThread.interrupt();
|
||||||
|
try { serverThread.join(); }
|
||||||
|
catch (InterruptedException e) { e.printStackTrace(); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop broadcasting and close port. Cannot be resumed from
|
||||||
|
*/
|
||||||
|
public void stop(){
|
||||||
|
pause();
|
||||||
|
serverSocket.close();
|
||||||
|
}
|
||||||
|
}
|
85
src/net/tofvesson/broadcast/support/ImmutableArray.java
Normal file
85
src/net/tofvesson/broadcast/support/ImmutableArray.java
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
package net.tofvesson.broadcast.support;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrapper for arrays to prevent modification
|
||||||
|
* @param <T> Element type
|
||||||
|
*/
|
||||||
|
public class ImmutableArray<T> {
|
||||||
|
|
||||||
|
protected final T[] array;
|
||||||
|
protected ImmutableArray(T[] array){ this.array = array; }
|
||||||
|
|
||||||
|
public int length(){ return array.length; }
|
||||||
|
public T at(int index){ return array[index]; }
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
Object[] compareTo;
|
||||||
|
if(obj instanceof ImmutableArray) compareTo = ((ImmutableArray) obj).array;
|
||||||
|
else if(obj!=null && obj.getClass().isArray()){
|
||||||
|
try{
|
||||||
|
compareTo = (Object[]) obj;
|
||||||
|
}catch(ClassCastException e){
|
||||||
|
return false; // Object was a primitive array
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else return false;
|
||||||
|
return Arrays.equals(array, compareTo);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean compare(ImmutableArray<T> to, int targetOffset, int offset, int length){
|
||||||
|
if(offset+length>array.length || targetOffset+length>to.array.length) return false;
|
||||||
|
for(int i = 0; i<length; ++i)
|
||||||
|
if(!areEqual(to.array[i+targetOffset], array[i+offset]))
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static boolean areEqual(Object o1, Object o2){
|
||||||
|
return o1==o2 || (o1!=null && o1.equals(o2));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> ImmutableArray<T> from(T[] t){ return new ImmutableArray<>(t); }
|
||||||
|
public static ImmutableArray<Boolean> from(boolean[] t){
|
||||||
|
Boolean[] t1 = new Boolean[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Character> from(char[] t){
|
||||||
|
Character[] t1 = new Character[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Byte> from(byte[] t){
|
||||||
|
Byte[] t1 = new Byte[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Short> from(short[] t){
|
||||||
|
Short[] t1 = new Short[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Integer> from(int[] t){
|
||||||
|
Integer[] t1 = new Integer[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Long> from(long[] t){
|
||||||
|
Long[] t1 = new Long[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Float> from(float[] t){
|
||||||
|
Float[] t1 = new Float[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
public static ImmutableArray<Double> from(double[] t){
|
||||||
|
Double[] t1 = new Double[t.length];
|
||||||
|
for(int i = 0; i<t1.length; ++i) t1[i] = t[i];
|
||||||
|
return new ImmutableArray<>(t1);
|
||||||
|
}
|
||||||
|
}
|
137
src/net/tofvesson/broadcast/support/ImmutableReferenceMap.java
Normal file
137
src/net/tofvesson/broadcast/support/ImmutableReferenceMap.java
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
package net.tofvesson.broadcast.support;
|
||||||
|
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.function.BiConsumer;
|
||||||
|
import java.util.function.BiFunction;
|
||||||
|
import java.util.function.Function;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Immutable map that references mutable map. Used to dissuade modification of a map
|
||||||
|
* @param <K> Key type
|
||||||
|
* @param <V> Value type
|
||||||
|
*/
|
||||||
|
public class ImmutableReferenceMap<K, V> implements Map<K, V> {
|
||||||
|
|
||||||
|
protected final Map<K, V> reference;
|
||||||
|
|
||||||
|
public ImmutableReferenceMap(Map<K, V> reference){
|
||||||
|
this.reference = reference;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int size() {
|
||||||
|
return reference.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEmpty() {
|
||||||
|
return reference.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean containsKey(Object key) {
|
||||||
|
return reference.containsKey(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean containsValue(Object value) {
|
||||||
|
return reference.containsValue(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V get(Object key) {
|
||||||
|
return reference.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V put(K key, V value) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V remove(Object key) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void putAll(Map<? extends K, ? extends V> m) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void clear() {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<K> keySet() {
|
||||||
|
return reference.keySet();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Collection<V> values() {
|
||||||
|
return reference.values();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<Entry<K, V>> entrySet() {
|
||||||
|
return reference.entrySet();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V getOrDefault(Object key, V defaultValue) {
|
||||||
|
return reference.getOrDefault(key, defaultValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void forEach(BiConsumer<? super K, ? super V> action) {
|
||||||
|
reference.forEach(action);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void replaceAll(BiFunction<? super K, ? super V, ? extends V> function) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V putIfAbsent(K key, V value) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean remove(Object key, Object value) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean replace(K key, V oldValue, V newValue) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V replace(K key, V value) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V computeIfPresent(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V merge(K key, V value, BiFunction<? super V, ? super V, ? extends V> remappingFunction) {
|
||||||
|
throw new IllegalStateException("Unsupported action");
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user