flavien
c5f63795d0
All checks were successful
flavien's git/Projet_JAVA_P2P_STRI2A/pipeline/pr-etape4 This commit looks good
227 lines
8.8 KiB
Java
227 lines
8.8 KiB
Java
package clientP2P;
|
||
|
||
import java.net.UnknownHostException;
|
||
import java.util.Scanner;
|
||
import java.io.IOException;
|
||
import java.nio.file.Files;
|
||
import java.io.File;
|
||
import java.nio.file.Paths;
|
||
import java.nio.file.StandardOpenOption;
|
||
import java.util.Arrays;
|
||
import java.util.List;
|
||
import java.security.MessageDigest;
|
||
import java.security.NoSuchAlgorithmException;
|
||
import localException.InternalError;
|
||
import localException.ProtocolError;
|
||
import localException.SizeError;
|
||
import localException.TransmissionError;
|
||
import localException.VersionError;
|
||
import localException.SocketClosed;
|
||
import remoteException.EmptyFile;
|
||
import remoteException.EmptyDirectory;
|
||
import remoteException.InternalRemoteError;
|
||
import remoteException.NotFound;
|
||
import remoteException.ProtocolRemoteError;
|
||
import remoteException.VersionRemoteError;
|
||
import protocolP2P.ProtocolP2PPacketTCP;
|
||
import protocolP2P.Payload;
|
||
import protocolP2P.RequestResponseCode;
|
||
import protocolP2P.FileList;
|
||
import protocolP2P.FilePart;
|
||
import protocolP2P.LoadRequest;
|
||
import protocolP2P.HashAlgorithm;
|
||
import protocolP2P.HashRequest;
|
||
import protocolP2P.HashResponse;
|
||
import clientP2P.ClientDownloadTCP;
|
||
import tools.HostItem;
|
||
import tools.Logger;
|
||
import tools.LogLevel;
|
||
|
||
/** Implementation of P2P-JAVA-PROJECT CLIENT
|
||
* @author Louis Royer
|
||
* @author Flavien Haas
|
||
* @author JS Auge
|
||
* @version 1.0
|
||
*/
|
||
public class ClientManagementTCP implements Runnable {
|
||
private String baseDirectory;
|
||
private String partsSubdir;
|
||
private List<HostItem> hostList;
|
||
private Logger logger;
|
||
|
||
/** Constructor for TCP implementation, with baseDirectory and TCPPort parameters.
|
||
* @param baseDirectory the root directory where files are stored
|
||
* @param host hostname of the server
|
||
* @param TCPPort the server will listen on this port
|
||
*/
|
||
public ClientManagementTCP(String baseDirectory, List<HostItem> hostList, String partsSubdir, Logger logger) {
|
||
this.baseDirectory = baseDirectory;
|
||
this.hostList = hostList;
|
||
this.partsSubdir = partsSubdir;
|
||
this.logger = logger;
|
||
}
|
||
|
||
/** Implementation of Runnable
|
||
*/
|
||
public void run() {
|
||
try {
|
||
System.out.println("Enter all servers: type \"stop\" when finished");
|
||
Scanner scanner = new Scanner(System.in);
|
||
String[] list = listDirectory();
|
||
System.out.println("Files present on the server:");
|
||
for(String listItem: list) {
|
||
System.out.println(listItem);
|
||
}
|
||
System.out.println("Name of the file to download:");
|
||
String f = scanner.nextLine();
|
||
download(f);
|
||
System.out.println("File " + f + " sucessfully downloaded");
|
||
logger.writeTCP("File " + f + " sucessfully downloaded", LogLevel.Info);
|
||
} catch (EmptyDirectory e) {
|
||
System.err.println("Error: Server has no file in directory");
|
||
logger.writeTCP("Error: Server has no file in directory", LogLevel.Error);
|
||
} catch (InternalError e) {
|
||
System.err.println("Error: Client internal error");
|
||
logger.writeTCP("Error: Client internal error", LogLevel.Error);
|
||
} catch (UnknownHostException e) {
|
||
System.err.println("Error: Server host is unknown");
|
||
logger.writeTCP("Error: Server host is unknown", LogLevel.Error);
|
||
} catch (IOException e) {
|
||
System.err.println("Error: Request cannot be send or response cannot be received");
|
||
logger.writeTCP("Error: Request cannot be send or response cannot be received", LogLevel.Error);
|
||
} catch (TransmissionError e) {
|
||
System.err.println("Error: Message received is too big");
|
||
logger.writeTCP("Error: Message received is too big", LogLevel.Error);
|
||
} catch (ProtocolError e) {
|
||
System.err.println("Error: Cannot decode server’s response");
|
||
logger.writeTCP("Error: Cannot decode server’s response", LogLevel.Error);
|
||
} catch (VersionError e) {
|
||
System.err.println("Error: Server’s response use bad version of the protocol");
|
||
logger.writeTCP("Error: Server’s response use bad version of the protocol", LogLevel.Error);
|
||
} catch (SizeError e) {
|
||
System.err.println("Error: Cannot handle this packets because of internal representation limitations of numbers on the client");
|
||
logger.writeTCP("Error: Cannot handle this packets because of internal representation limitations of numbers on the client", LogLevel.Error);
|
||
} catch (InternalRemoteError e) {
|
||
System.err.println("Error: Server internal error");
|
||
logger.writeTCP("Error: Server internal error", LogLevel.Error);
|
||
} catch (ProtocolRemoteError e) {
|
||
System.err.println("Error: Server cannot decode client’s request");
|
||
logger.writeTCP("Error: Server cannot decode client’s request", LogLevel.Error);
|
||
} catch (VersionRemoteError e) {
|
||
System.err.println("Error: Server cannot decode this version of the protocol");
|
||
logger.writeTCP("Error: Server cannot decode this version of the protocol", LogLevel.Error);
|
||
} catch (NotFound e) {
|
||
System.err.println("Error: Server has not this file in directory");
|
||
logger.writeTCP("Error: Server has not this file in directory", LogLevel.Error);
|
||
} catch (EmptyFile e) {
|
||
System.err.println("Error: File is empty");
|
||
logger.writeTCP("Error: File is empty", LogLevel.Error);
|
||
}
|
||
}
|
||
|
||
/** Try to download a file
|
||
* @param filename name of the file to download
|
||
* @throws NotFound
|
||
* @throws InternalError
|
||
* @throws UnknownHostException
|
||
* @throws IOException
|
||
* @throws TransmissionError
|
||
* @throws ProtocolError
|
||
* @throws VersionError
|
||
* @throws SizeError
|
||
* @throws InternalRemoteError
|
||
* @throws ProtocolRemoteError
|
||
* @throws VersionRemoteError
|
||
* @throws EmptyFile
|
||
*/
|
||
private void download(String filename) throws EmptyFile, NotFound, InternalError, UnknownHostException, IOException, TransmissionError, ProtocolError, VersionError, SizeError, InternalRemoteError, ProtocolRemoteError, VersionRemoteError {
|
||
ClientDownloadTCP downLoader = new ClientDownloadTCP(filename, hostList, partsSubdir, baseDirectory, logger);
|
||
Thread t = new Thread(downLoader);
|
||
t.start();
|
||
try {
|
||
t.join();
|
||
if (downLoader.getSuccess()) {
|
||
byte[] hash512 = downLoader.getHashSum512();
|
||
if (!Arrays.equals(hash512, computeHashsum(filename, HashAlgorithm.SHA512))) {
|
||
System.err.println("Error: Hashsum does not match");
|
||
logger.writeTCP("Error: Hashsum does not match", LogLevel.Error);
|
||
System.err.println("Computed checksum:");
|
||
byte[] c = computeHashsum(filename, HashAlgorithm.SHA512);
|
||
for (byte b: c) {
|
||
System.err.print(String.format("%02X", b));
|
||
logger.writeTCP("Computed checksum:" + String.format("%02X", b), LogLevel.Info);
|
||
}
|
||
System.err.println("");
|
||
System.err.println("Received checksum:");
|
||
for (byte b: hash512) {
|
||
System.err.print(String.format("%02X", b));
|
||
logger.writeTCP("Received checksum:" + String.format("%02X", b), LogLevel.Info);
|
||
}
|
||
System.err.println("");
|
||
throw new InternalError();
|
||
}
|
||
} else {
|
||
throw new InternalError();
|
||
}
|
||
} catch (InterruptedException e) {
|
||
throw new InternalError();
|
||
}
|
||
}
|
||
|
||
/** list server’s directory content
|
||
* @return list of files
|
||
* @throws InternalError
|
||
* @throws UnknowHostException
|
||
* @throws IOException
|
||
* @throws TransmissionError
|
||
* @throws ProtocolError
|
||
* @throws VersionError
|
||
* @throws SizeError
|
||
* @throws EmptyDirectory
|
||
* @throws InternalRemoteError
|
||
* @throws ProtocolRemoteError
|
||
* @throws VersionRemoteError
|
||
*/
|
||
private String[] listDirectory() throws EmptyDirectory, InternalError, UnknownHostException, IOException, TransmissionError, ProtocolError, VersionError, SizeError, InternalRemoteError, ProtocolRemoteError, VersionRemoteError {
|
||
ProtocolP2PPacketTCP d = new ProtocolP2PPacketTCP(new Payload(RequestResponseCode.LIST_REQUEST));
|
||
try {
|
||
d.sendRequest((Object)hostList.get(0).getTCPSocket());
|
||
Payload p = d.receiveResponse().getPayload();
|
||
assert p instanceof FileList : "This payload must be instance of Filelist";
|
||
if (!(p instanceof FileList)) {
|
||
throw new InternalError();
|
||
} else {
|
||
return ((FileList)p).getFileList();
|
||
}
|
||
} catch (NotFound e) {
|
||
logger.writeTCP(e, LogLevel.Error);
|
||
throw new ProtocolError();
|
||
} catch (EmptyFile e) {
|
||
logger.writeTCP(e, LogLevel.Error);
|
||
throw new ProtocolError();
|
||
} catch (SocketClosed e){
|
||
System.err.println("listDirectory : SocketClosed");
|
||
logger.writeTCP("listDirectory : SocketClosed", LogLevel.Error);
|
||
throw new ProtocolError();
|
||
}
|
||
}
|
||
|
||
/** Compute Hashsum of a file.
|
||
* @param filename
|
||
* @return hashsum
|
||
*/
|
||
private byte[] computeHashsum(String filename, HashAlgorithm h) {
|
||
try {
|
||
MessageDigest md = MessageDigest.getInstance(HashAlgorithm.SHA512.getName());
|
||
return md.digest(Files.readAllBytes(Paths.get(baseDirectory + filename)));
|
||
} catch (NoSuchAlgorithmException e) {
|
||
System.out.println("Error: " + h.getName() + " not supported");
|
||
logger.writeTCP("Error: " + h.getName() + " not supported", LogLevel.Error);
|
||
} catch (IOException e) {
|
||
System.out.println("Error: cannot read " + filename);
|
||
logger.writeTCP("Error: cannot read " + filename, LogLevel.Error);
|
||
}
|
||
return new byte[0];
|
||
}
|
||
}
|