Fix #105
This commit is contained in:
parent
aae25fbd99
commit
ed63375865
@ -21,7 +21,8 @@ import protocolP2P.UpdateRatio;
|
|||||||
import localException.InternalError;
|
import localException.InternalError;
|
||||||
import remoteException.EmptyDirectory;
|
import remoteException.EmptyDirectory;
|
||||||
import exception.LocalException;
|
import exception.LocalException;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
/** Tracker management implementation
|
/** Tracker management implementation
|
||||||
* @author Louis Royer
|
* @author Louis Royer
|
||||||
@ -37,6 +38,8 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
protected Map<HostItem, Long> ratioDown = new HashMap<>();
|
protected Map<HostItem, Long> ratioDown = new HashMap<>();
|
||||||
protected Map<String, List<HostItem>> fileList = new HashMap<>();
|
protected Map<String, List<HostItem>> fileList = new HashMap<>();
|
||||||
protected volatile boolean stop;
|
protected volatile boolean stop;
|
||||||
|
protected AtomicBoolean writeLock;
|
||||||
|
protected AtomicInteger readLock;
|
||||||
|
|
||||||
/** Constructor
|
/** Constructor
|
||||||
* @param tracker Tracker HostItem
|
* @param tracker Tracker HostItem
|
||||||
@ -46,6 +49,8 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
stop = false;
|
stop = false;
|
||||||
this.tracker = tracker;
|
this.tracker = tracker;
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
|
writeLock = new AtomicBoolean();
|
||||||
|
readLock = new AtomicInteger();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Handle Discover request
|
/** Handle Discover request
|
||||||
@ -60,7 +65,17 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
} else {
|
} else {
|
||||||
String filename = ((DiscoverRequest)p).getFilename();
|
String filename = ((DiscoverRequest)p).getFilename();
|
||||||
try {
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
while(writeLock.get()) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
readLock.incrementAndGet();
|
||||||
pd.sendResponse(createProtocolP2PPacket(new DiscoverResponse(filename, fileList.getOrDefault(filename, hostList))));
|
pd.sendResponse(createProtocolP2PPacket(new DiscoverResponse(filename, fileList.getOrDefault(filename, hostList))));
|
||||||
|
readLock.decrementAndGet();
|
||||||
|
this.notify();
|
||||||
|
}
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
throw new InternalError();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
writeLog(e, LogLevel.Error);
|
writeLog(e, LogLevel.Error);
|
||||||
}
|
}
|
||||||
@ -81,6 +96,11 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
HostItem host = ((RatioRequest)p).getHostItem();
|
HostItem host = ((RatioRequest)p).getHostItem();
|
||||||
writeLog("Ratio request for host " + host, LogLevel.Debug);
|
writeLog("Ratio request for host " + host, LogLevel.Debug);
|
||||||
try {
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
while(writeLock.get()) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
readLock.incrementAndGet();
|
||||||
if (!hostList.contains(host)) {
|
if (!hostList.contains(host)) {
|
||||||
String l = "";
|
String l = "";
|
||||||
for (HostItem h: hostList) {
|
for (HostItem h: hostList) {
|
||||||
@ -89,10 +109,16 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
writeLog(host + " is not in hostlist [ " + l + "]", LogLevel.Debug);
|
writeLog(host + " is not in hostlist [ " + l + "]", LogLevel.Debug);
|
||||||
pd.sendResponse(createProtocolP2PPacket(new Payload(RequestResponseCode.UNKNOWN_HOST)));
|
pd.sendResponse(createProtocolP2PPacket(new Payload(RequestResponseCode.UNKNOWN_HOST)));
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
pd.sendResponse(createProtocolP2PPacket(new RatioResponse(host,
|
pd.sendResponse(createProtocolP2PPacket(new RatioResponse(host,
|
||||||
ratioUp.get(host).longValue(),
|
ratioUp.get(host).longValue(),
|
||||||
ratioDown.get(host).longValue())));
|
ratioDown.get(host).longValue())));
|
||||||
}
|
}
|
||||||
|
readLock.decrementAndGet();
|
||||||
|
this.notify();
|
||||||
|
}
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
throw new InternalError();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
writeLog(e, LogLevel.Error);
|
writeLog(e, LogLevel.Error);
|
||||||
}
|
}
|
||||||
@ -110,6 +136,14 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
throw new InternalError();
|
throw new InternalError();
|
||||||
} else {
|
} else {
|
||||||
String[] f = ((FileList)p).getFileList();
|
String[] f = ((FileList)p).getFileList();
|
||||||
|
try {
|
||||||
|
synchronized(this) {
|
||||||
|
while(writeLock.getAndSet(true)) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
while(readLock.get() > 0) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
for (String file: f) {
|
for (String file: f) {
|
||||||
List<HostItem> h = fileList.get(file);
|
List<HostItem> h = fileList.get(file);
|
||||||
if (h != null) {
|
if (h != null) {
|
||||||
@ -122,6 +156,12 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
fileList.put(file, emptyH);
|
fileList.put(file, emptyH);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
writeLock.getAndSet(false);
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
} catch(InterruptedException e) {
|
||||||
|
throw new InternalError();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,14 +178,28 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
}
|
}
|
||||||
HostItem host = ((Unregister)p).getHostItem();
|
HostItem host = ((Unregister)p).getHostItem();
|
||||||
writeLog("Received UNREGISTER from host " + pd.getHostItem() + ". Removing host " + host, LogLevel.Action);
|
writeLog("Received UNREGISTER from host " + pd.getHostItem() + ". Removing host " + host, LogLevel.Action);
|
||||||
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
while(writeLock.getAndSet(true)) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
while(readLock.get() > 0) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
hostList.remove(host);
|
hostList.remove(host);
|
||||||
for(String f: fileList.keySet()) {
|
Map<String, List<HostItem>> fileListCpy = new HashMap<>(fileList);
|
||||||
|
for(String f: fileListCpy.keySet()) {
|
||||||
fileList.get(f).remove(host);
|
fileList.get(f).remove(host);
|
||||||
if(fileList.get(f).isEmpty()) {
|
if(fileList.get(f).isEmpty()) {
|
||||||
fileList.remove(f);
|
fileList.remove(f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
writeLock.getAndSet(false);
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
throw new InternalError();
|
||||||
|
}
|
||||||
/* Note: we do not remove host from ratioUp and ratioDown since Unregistering is only here to indicate
|
/* Note: we do not remove host from ratioUp and ratioDown since Unregistering is only here to indicate
|
||||||
* the host is temporary not serving any files (ie. is down).
|
* the host is temporary not serving any files (ie. is down).
|
||||||
*/
|
*/
|
||||||
@ -177,12 +231,26 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
HostItem updateRatioClient = updateRatio.getClient();
|
HostItem updateRatioClient = updateRatio.getClient();
|
||||||
long ratioSize = updateRatio.getDataSize();
|
long ratioSize = updateRatio.getDataSize();
|
||||||
writeLog("Ratio += " + ratioSize + ", client: " + updateRatioClient + " / server: " + updateRatioServer, LogLevel.Debug);
|
writeLog("Ratio += " + ratioSize + ", client: " + updateRatioClient + " / server: " + updateRatioServer, LogLevel.Debug);
|
||||||
|
try {
|
||||||
|
synchronized(this) {
|
||||||
|
while(writeLock.getAndSet(true)) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
while(readLock.get() > 0) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
if (!ratioDown.containsKey(updateRatioClient) || ! ratioUp.containsKey(updateRatioServer)) {
|
if (!ratioDown.containsKey(updateRatioClient) || ! ratioUp.containsKey(updateRatioServer)) {
|
||||||
sendUnknownHost(pd);
|
sendUnknownHost(pd);
|
||||||
} else {
|
} else {
|
||||||
ratioDown.put(updateRatioClient, Long.valueOf(ratioDown.get(updateRatioClient).longValue() + ratioSize));
|
ratioDown.put(updateRatioClient, Long.valueOf(ratioDown.get(updateRatioClient).longValue() + ratioSize));
|
||||||
ratioUp.put(updateRatioServer, Long.valueOf(ratioUp.get(updateRatioServer).longValue() + ratioSize));
|
ratioUp.put(updateRatioServer, Long.valueOf(ratioUp.get(updateRatioServer).longValue() + ratioSize));
|
||||||
}
|
}
|
||||||
|
writeLock.getAndSet(false);
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
throw new InternalError();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Handle Registering
|
/** Handle Registering
|
||||||
@ -197,13 +265,26 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
}
|
}
|
||||||
// add host to known host list
|
// add host to known host list
|
||||||
HostItem host = ((Register)p).getHostItem();
|
HostItem host = ((Register)p).getHostItem();
|
||||||
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
while(writeLock.getAndSet(true)) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
while(readLock.get() > 0) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
if (!hostList.contains(host)) {
|
if (!hostList.contains(host)) {
|
||||||
hostList.add(host);
|
hostList.add(host);
|
||||||
}
|
}
|
||||||
|
|
||||||
// initialize ratios if this is a new host
|
// initialize ratios if this is a new host
|
||||||
ratioUp.putIfAbsent(host, Long.valueOf(0));
|
ratioUp.putIfAbsent(host, Long.valueOf(0));
|
||||||
ratioDown.putIfAbsent(host, Long.valueOf(0));
|
ratioDown.putIfAbsent(host, Long.valueOf(0));
|
||||||
|
writeLock.getAndSet(false);
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
} catch(InterruptedException e) {
|
||||||
|
throw new InternalError();
|
||||||
|
}
|
||||||
|
|
||||||
// send a list request
|
// send a list request
|
||||||
try {
|
try {
|
||||||
@ -215,11 +296,45 @@ public abstract class TrackerManagement extends ServeErrors implements Runnable
|
|||||||
closeHostItemSocket(host);
|
closeHostItemSocket(host);
|
||||||
} catch (EmptyDirectory e) {
|
} catch (EmptyDirectory e) {
|
||||||
writeLog("Empty Directory", LogLevel.Debug);
|
writeLog("Empty Directory", LogLevel.Debug);
|
||||||
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
while(writeLock.getAndSet(true)) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
while(readLock.get() > 0) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
if (!hostList.contains(host)) {
|
||||||
|
hostList.add(host);
|
||||||
|
}
|
||||||
hostList.remove(host);
|
hostList.remove(host);
|
||||||
|
writeLock.getAndSet(false);
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
} catch(InterruptedException e2) {
|
||||||
|
throw new InternalError();
|
||||||
|
}
|
||||||
writeLog("Received EMPTY DIRECTORY from host " + pd.getHostItem() + ". Aborting.", LogLevel.Action);
|
writeLog("Received EMPTY DIRECTORY from host " + pd.getHostItem() + ". Aborting.", LogLevel.Action);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
// remove from list because list request could not be send
|
// remove from list because list request could not be send
|
||||||
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
while(writeLock.getAndSet(true)) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
while(readLock.get() > 0) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
if (!hostList.contains(host)) {
|
||||||
|
hostList.add(host);
|
||||||
|
}
|
||||||
hostList.remove(host);
|
hostList.remove(host);
|
||||||
|
writeLock.getAndSet(false);
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
} catch(InterruptedException e2) {
|
||||||
|
throw new InternalError();
|
||||||
|
}
|
||||||
writeLog("Aborting the add of host " + host, LogLevel.Action);
|
writeLog("Aborting the add of host " + host, LogLevel.Action);
|
||||||
writeLog(e, LogLevel.Error);
|
writeLog(e, LogLevel.Error);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user