Merge pull request #47 from PM2-IT21bWIN-ruiz-mach-krea/Java_Doc_and_Formatting
Java doc and formatting
This commit is contained in:
commit
b67b9c84aa
File diff suppressed because one or more lines are too long
Before Width: | Height: | Size: 133 KiB After Width: | Height: | Size: 133 KiB |
|
@ -17,41 +17,71 @@ import java.io.IOException;
|
||||||
|
|
||||||
import static ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State.*;
|
import static ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class Representing the Controller Element of the Window, also Contains the Elements that contacts the View Elements via
|
||||||
|
* Listeners and Observable Objects.
|
||||||
|
* To Contact the Model Elements needed it also holds references to message and the Connectionhandler.
|
||||||
|
*/
|
||||||
public class ChatWindowController {
|
public class ChatWindowController {
|
||||||
private ClientConnectionHandler connectionHandler;
|
private ClientConnectionHandler connectionHandler;
|
||||||
private ClientMessageList messages;
|
private ClientMessageList messages;
|
||||||
|
|
||||||
private final WindowCloseHandler windowCloseHandler = new WindowCloseHandler();
|
private final WindowCloseHandler windowCloseHandler = new WindowCloseHandler();
|
||||||
|
|
||||||
@FXML private Pane rootPane;
|
@FXML
|
||||||
@FXML private TextField serverAddressField;
|
private Pane rootPane;
|
||||||
@FXML private TextField serverPortField;
|
@FXML
|
||||||
@FXML private TextField userNameField;
|
private TextField serverAddressField;
|
||||||
@FXML private TextField messageField;
|
@FXML
|
||||||
@FXML private TextArea messageArea;
|
private TextField serverPortField;
|
||||||
@FXML private Button connectButton;
|
@FXML
|
||||||
@FXML private Button sendButton;
|
private TextField userNameField;
|
||||||
@FXML private TextField filterValue;
|
@FXML
|
||||||
|
private TextField messageField;
|
||||||
|
@FXML
|
||||||
|
private TextArea messageArea;
|
||||||
|
@FXML
|
||||||
|
private Button connectButton;
|
||||||
|
@FXML
|
||||||
|
private TextField filterValue;
|
||||||
|
@FXML
|
||||||
|
private Button sendButton;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes a message object and stores it used as Model and also starts message Listener via messageListener method.
|
||||||
|
*
|
||||||
|
* @param messages Object that will be set for use as Model
|
||||||
|
*/
|
||||||
public void setMessages(ClientMessageList messages) {
|
public void setMessages(ClientMessageList messages) {
|
||||||
this.messages = messages;
|
this.messages = messages;
|
||||||
messageListener();
|
messageListener();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setConnectionHandler(ClientConnectionHandler connectionHandler){
|
/**
|
||||||
|
* Takes a Connection handler object and stores it used as Model and also starts a Listener for it.
|
||||||
|
*
|
||||||
|
* @param connectionHandler that will be set and used as Model.
|
||||||
|
*/
|
||||||
|
public void setConnectionHandler(ClientConnectionHandler connectionHandler) {
|
||||||
this.connectionHandler = connectionHandler;
|
this.connectionHandler = connectionHandler;
|
||||||
startConnectionHandlerListener();
|
startConnectionHandlerListener();
|
||||||
serverAddressField.setText(connectionHandler.getServerAddressProperty().get());
|
serverAddressField.setText(connectionHandler.getServerAddressProperty().get());
|
||||||
serverPortField.setText(String.valueOf(connectionHandler.getServerPortProperty().get()));
|
serverPortField.setText(String.valueOf(connectionHandler.getServerPortProperty().get()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method which closes the Application via use of the disconnect Method.
|
||||||
|
*/
|
||||||
private void applicationClose() {
|
private void applicationClose() {
|
||||||
disconnect();
|
disconnect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method that handles the Connect Button and Initiates connect when connectionHandler is not Connected or
|
||||||
|
* a disconnect when it is connected.
|
||||||
|
*/
|
||||||
@FXML
|
@FXML
|
||||||
private void toggleConnection () {
|
private void toggleConnection() {
|
||||||
if (connectionHandler == null || connectionHandler.getStateProperty().get() != CONNECTED) {
|
if (connectionHandler == null || connectionHandler.getStateProperty().get() != CONNECTED) {
|
||||||
connect();
|
connect();
|
||||||
} else {
|
} else {
|
||||||
|
@ -59,16 +89,22 @@ public class ChatWindowController {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initiates a connection by starting Connection Handler and telling, the Handler to start a connection.
|
||||||
|
*/
|
||||||
private void connect() {
|
private void connect() {
|
||||||
try {
|
try {
|
||||||
messages.clear(); // clear message list
|
messages.clear(); // clear message list
|
||||||
startConnectionHandler();
|
startConnectionHandler();
|
||||||
connectionHandler.connect();
|
connectionHandler.connect();
|
||||||
} catch(ChatProtocolException | IOException e) {
|
} catch (ChatProtocolException | IOException e) {
|
||||||
addError(e.getMessage());
|
addError("Error while starting Connection Handler and connect" + e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initiates disconnecting of the connectionHandler, also checks if connectionHandler is available.
|
||||||
|
*/
|
||||||
private void disconnect() {
|
private void disconnect() {
|
||||||
if (connectionHandler == null) {
|
if (connectionHandler == null) {
|
||||||
addError("No connection handler");
|
addError("No connection handler");
|
||||||
|
@ -81,6 +117,9 @@ public class ChatWindowController {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method which is used when the send button is pressed and handing over a message to the Connection Handler
|
||||||
|
*/
|
||||||
@FXML
|
@FXML
|
||||||
private void message() {
|
private void message() {
|
||||||
String messageString = messageField.getText().strip();
|
String messageString = messageField.getText().strip();
|
||||||
|
@ -97,29 +136,40 @@ public class ChatWindowController {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method which is used when a Filter is applied
|
||||||
|
* Setting the Text in the message area after sending it through the filter.
|
||||||
|
*/
|
||||||
@FXML
|
@FXML
|
||||||
private void applyFilter( ) {
|
private void applyFilter() {
|
||||||
Platform.runLater(() -> this.messageArea.setText(messages.getFilteredMessages(filterValue.getText().strip())));
|
Platform.runLater(() -> this.messageArea.setText(messages.getFilteredMessages(filterValue.getText().strip())));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts the ConnectionHandler setting the username and Checking if the name follows the valid format of no spaces.
|
||||||
|
*
|
||||||
|
* @throws IOException for error that may occur during initialization of connectionHandler.
|
||||||
|
*/
|
||||||
private void startConnectionHandler() throws IOException {
|
private void startConnectionHandler() throws IOException {
|
||||||
String userName = userNameField.getText();
|
String userName = userNameField.getText();
|
||||||
if(!userName.contains(" ")) {
|
if (!userName.contains(" ")) {
|
||||||
String serverAddress = serverAddressField.getText();
|
String serverAddress = serverAddressField.getText();
|
||||||
int serverPort = Integer.parseInt(serverPortField.getText());
|
int serverPort = Integer.parseInt(serverPortField.getText());
|
||||||
connectionHandler.initialize(serverAddress, serverPort, userName);
|
connectionHandler.initialize(serverAddress, serverPort, userName);
|
||||||
new Thread(connectionHandler).start();
|
new Thread(connectionHandler).start();
|
||||||
|
|
||||||
//register Listener
|
|
||||||
//startConnectionHandlerListener();
|
|
||||||
|
|
||||||
// register window close handler
|
|
||||||
rootPane.getScene().getWindow().addEventHandler(WindowEvent.WINDOW_CLOSE_REQUEST, windowCloseHandler);
|
rootPane.getScene().getWindow().addEventHandler(WindowEvent.WINDOW_CLOSE_REQUEST, windowCloseHandler);
|
||||||
} else {
|
} else {
|
||||||
addError("It is not allowed to have spaces in username!");
|
addError("It is not allowed to have spaces in username!");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the state shown according to the state the method receives, if state indicates disconnected it will also inform the
|
||||||
|
* Connection Handler and tell it to stop Receiving more messages.
|
||||||
|
*
|
||||||
|
* @param newState is the state that it should be set to.
|
||||||
|
*/
|
||||||
public void stateChanged(State newState) {
|
public void stateChanged(State newState) {
|
||||||
// update UI (need to be run in UI thread: see Platform.runLater())
|
// update UI (need to be run in UI thread: see Platform.runLater())
|
||||||
Platform.runLater(new Runnable() {
|
Platform.runLater(new Runnable() {
|
||||||
|
@ -128,11 +178,16 @@ public class ChatWindowController {
|
||||||
connectButton.setText((newState == CONNECTED || newState == CONFIRM_DISCONNECT) ? "Disconnect" : "Connect");
|
connectButton.setText((newState == CONNECTED || newState == CONFIRM_DISCONNECT) ? "Disconnect" : "Connect");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
if(newState == DISCONNECTED){
|
if (newState == DISCONNECTED) {
|
||||||
connectionHandler.stopReceiving();
|
connectionHandler.stopReceiving();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets displayed username according to the String provided.
|
||||||
|
*
|
||||||
|
* @param userName provided String that is set as name.
|
||||||
|
*/
|
||||||
public void setUserName(String userName) {
|
public void setUserName(String userName) {
|
||||||
Platform.runLater(new Runnable() {
|
Platform.runLater(new Runnable() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -142,6 +197,11 @@ public class ChatWindowController {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets displayed Server Address.
|
||||||
|
*
|
||||||
|
* @param serverAddress provided String that is set as server address.
|
||||||
|
*/
|
||||||
public void setServerAddress(String serverAddress) {
|
public void setServerAddress(String serverAddress) {
|
||||||
Platform.runLater(new Runnable() {
|
Platform.runLater(new Runnable() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -151,6 +211,11 @@ public class ChatWindowController {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets displayed Server port.
|
||||||
|
*
|
||||||
|
* @param serverPort provided String that is set as server port.
|
||||||
|
*/
|
||||||
public void setServerPort(int serverPort) {
|
public void setServerPort(int serverPort) {
|
||||||
Platform.runLater(new Runnable() {
|
Platform.runLater(new Runnable() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -160,16 +225,32 @@ public class ChatWindowController {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method which adds an incoming String as an Error Message.
|
||||||
|
*
|
||||||
|
* @param message String to be added as Error
|
||||||
|
*/
|
||||||
public void addError(String message) {
|
public void addError(String message) {
|
||||||
messages.addMessage(new Message(Message.MessageType.ERROR, null, null, message));
|
messages.addMessage(new Message(Message.MessageType.ERROR, null, null, message));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Nested Class in charge of Closing the window
|
||||||
|
*/
|
||||||
class WindowCloseHandler implements EventHandler<WindowEvent> {
|
class WindowCloseHandler implements EventHandler<WindowEvent> {
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param event the event which occurred when Windows is closed
|
||||||
|
*/
|
||||||
public void handle(WindowEvent event) {
|
public void handle(WindowEvent event) {
|
||||||
applicationClose();
|
applicationClose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts several new Listener for Connection Handler changes by using several observable properties.
|
||||||
|
*/
|
||||||
public void startConnectionHandlerListener() {
|
public void startConnectionHandlerListener() {
|
||||||
connectionHandler.getStateProperty().addListener(new ChangeListener<State>() {
|
connectionHandler.getStateProperty().addListener(new ChangeListener<State>() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -200,6 +281,9 @@ public class ChatWindowController {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts a new Listener for messages by using the observable Boolean.
|
||||||
|
*/
|
||||||
private void messageListener() {
|
private void messageListener() {
|
||||||
messages.getChangedProperty().addListener(new ChangeListener<Boolean>() {
|
messages.getChangedProperty().addListener(new ChangeListener<Boolean>() {
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -2,8 +2,16 @@ package ch.zhaw.pm2.multichat.client;
|
||||||
|
|
||||||
import javafx.application.Application;
|
import javafx.application.Application;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Client Main Class in charge of starting the UI only contains main method.
|
||||||
|
*/
|
||||||
public class Client {
|
public class Client {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Main method which launches the Client UI
|
||||||
|
*
|
||||||
|
* @param args no arguments needed
|
||||||
|
*/
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
// Start UI
|
// Start UI
|
||||||
System.out.println("Starting Client Application");
|
System.out.println("Starting Client Application");
|
||||||
|
|
|
@ -13,19 +13,30 @@ import java.net.SocketException;
|
||||||
import java.util.Scanner;
|
import java.util.Scanner;
|
||||||
import java.util.regex.Matcher;
|
import java.util.regex.Matcher;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
import static ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State.*;
|
import static ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Client Connection Handler Class is used for the connection of the Client to a server
|
||||||
|
* it is used part as a Model storing properties like userName, state, serverAddress and Port.
|
||||||
|
* Also holds methods to
|
||||||
|
*/
|
||||||
public class ClientConnectionHandler extends ConnectionHandler implements Runnable {
|
public class ClientConnectionHandler extends ConnectionHandler implements Runnable {
|
||||||
|
|
||||||
private final Pattern messagePattern = Pattern.compile( "^(?:@(\\S*))?\\s*(.*)$" );
|
private final Pattern messagePattern = Pattern.compile("^(?:@(\\S*))?\\s*(.*)$");
|
||||||
|
|
||||||
private SimpleStringProperty userName;
|
private final SimpleStringProperty userName;
|
||||||
private SimpleObjectProperty<State> state;
|
private final SimpleObjectProperty<State> state;
|
||||||
private ClientMessageList messages;
|
private final ClientMessageList messages;
|
||||||
private SimpleStringProperty serverAddress;
|
private final SimpleStringProperty serverAddress;
|
||||||
private SimpleIntegerProperty serverPort;
|
private final SimpleIntegerProperty serverPort;
|
||||||
|
|
||||||
public ClientConnectionHandler(ClientMessageList messages) {
|
/**
|
||||||
|
* Constructor initializes ConnectionHandler by Setting default values for the fields and stores the given messages itself.
|
||||||
|
*
|
||||||
|
* @param messages
|
||||||
|
*/
|
||||||
|
public ClientConnectionHandler(ClientMessageList messages) {
|
||||||
super();
|
super();
|
||||||
this.messages = messages;
|
this.messages = messages;
|
||||||
state = new SimpleObjectProperty<>(State.NEW);
|
state = new SimpleObjectProperty<>(State.NEW);
|
||||||
|
@ -34,32 +45,78 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
this.userName = new SimpleStringProperty(null);
|
this.userName = new SimpleStringProperty(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called to initialize the ClientConnectionHandler when trying to start a connection
|
||||||
|
*
|
||||||
|
* @param serverAddress to connect to
|
||||||
|
* @param serverPort to connect to
|
||||||
|
* @param userName to connect as
|
||||||
|
* @throws IOException if connection to Server not possible
|
||||||
|
*/
|
||||||
public void initialize(String serverAddress, int serverPort, String userName) throws IOException {
|
public void initialize(String serverAddress, int serverPort, String userName) throws IOException {
|
||||||
state.set(NEW);
|
state.set(NEW);
|
||||||
this.serverAddress.set(serverAddress);
|
this.serverAddress.set(serverAddress);
|
||||||
this.serverPort.set(serverPort);
|
this.serverPort.set(serverPort);
|
||||||
setConnection(NetworkHandler.openConnection(serverAddress, serverPort));
|
setConnection(NetworkHandler.openConnection(serverAddress, serverPort));
|
||||||
this.userName.set((userName == null || userName.isBlank())? USER_NONE : userName);
|
this.userName.set((userName == null || userName.isBlank()) ? USER_NONE : userName);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SimpleStringProperty getServerAddressProperty() { return serverAddress; }
|
/**
|
||||||
|
* Observable getter Method for the stored serverAddress
|
||||||
|
*
|
||||||
|
* @return the stored serverAddress
|
||||||
|
*/
|
||||||
|
public SimpleStringProperty getServerAddressProperty() {
|
||||||
|
return serverAddress;
|
||||||
|
}
|
||||||
|
|
||||||
public SimpleIntegerProperty getServerPortProperty() { return serverPort; }
|
/**
|
||||||
|
* Observable getter Method for the stored serverPort
|
||||||
|
*
|
||||||
|
* @return the stored serverPort
|
||||||
|
*/
|
||||||
|
public SimpleIntegerProperty getServerPortProperty() {
|
||||||
|
return serverPort;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Observable getter Method for the stored state
|
||||||
|
*
|
||||||
|
* @return the stored state
|
||||||
|
*/
|
||||||
public SimpleObjectProperty<State> getStateProperty() {
|
public SimpleObjectProperty<State> getStateProperty() {
|
||||||
return this.state;
|
return this.state;
|
||||||
}
|
}
|
||||||
|
|
||||||
public SimpleStringProperty getUserNameProperty() { return userName; }
|
/**
|
||||||
|
* Observable getter Method for the stored userName
|
||||||
|
*
|
||||||
|
* @return the stored userName
|
||||||
|
*/
|
||||||
|
public SimpleStringProperty getUserNameProperty() {
|
||||||
|
return userName;
|
||||||
|
}
|
||||||
|
|
||||||
public void setState (State newState) {
|
/**
|
||||||
|
* Method which sets a new State.
|
||||||
|
*
|
||||||
|
* @param newState the state that will be set
|
||||||
|
*/
|
||||||
|
public void setState(State newState) {
|
||||||
state.set(newState);
|
state.set(newState);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void run () {
|
/**
|
||||||
|
* Standard run method which will directly start the startReceiving method.
|
||||||
|
*/
|
||||||
|
public void run() {
|
||||||
startReceiving();
|
startReceiving();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method that is started by the run method, starts a connection handler.
|
||||||
|
* Figures out if connection is available if not determines the error cause and gives an error accordingly.
|
||||||
|
*/
|
||||||
private void startReceiving() {
|
private void startReceiving() {
|
||||||
System.out.println("Starting Connection Handler");
|
System.out.println("Starting Connection Handler");
|
||||||
try {
|
try {
|
||||||
|
@ -68,7 +125,7 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
String data = getConnection().receive();
|
String data = getConnection().receive();
|
||||||
processData(data);
|
processData(data);
|
||||||
}
|
}
|
||||||
System.out.println("Stopped recieving data");
|
System.out.println("Stopped receiving data");
|
||||||
} catch (SocketException e) {
|
} catch (SocketException e) {
|
||||||
System.out.println("Connection terminated locally");
|
System.out.println("Connection terminated locally");
|
||||||
this.setState(DISCONNECTED);
|
this.setState(DISCONNECTED);
|
||||||
|
@ -77,14 +134,17 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
System.out.println("Connection terminated by remote");
|
System.out.println("Connection terminated by remote");
|
||||||
this.setState(DISCONNECTED);
|
this.setState(DISCONNECTED);
|
||||||
System.err.println("Unregistered because connection terminated" + e.getMessage());
|
System.err.println("Unregistered because connection terminated" + e.getMessage());
|
||||||
} catch(IOException e) {
|
} catch (IOException e) {
|
||||||
System.err.println("Communication error" + e);
|
System.err.println("Communication error" + e);
|
||||||
} catch(ClassNotFoundException e) {
|
} catch (ClassNotFoundException e) {
|
||||||
System.err.println("Received object of unknown type" + e.getMessage());
|
System.err.println("Received object of unknown type" + e.getMessage());
|
||||||
}
|
}
|
||||||
System.out.println("Stopped Connection Handler");
|
System.out.println("Stopped Connection Handler");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method which is used to stop receiving data, gets the current connection and closes it.
|
||||||
|
*/
|
||||||
public void stopReceiving() {
|
public void stopReceiving() {
|
||||||
System.out.println("Closing Connection Handler to Server");
|
System.out.println("Closing Connection Handler to Server");
|
||||||
try {
|
try {
|
||||||
|
@ -97,26 +157,31 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
System.out.println("Closed Connection Handler to Server");
|
System.out.println("Closed Connection Handler to Server");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method which processes data and determines its type then uses the corresponding method to process it.
|
||||||
|
*
|
||||||
|
* @param data that is received in a form of a String and then used depending on its determined cause.
|
||||||
|
*/
|
||||||
private void processData(String data) {
|
private void processData(String data) {
|
||||||
try {
|
try {
|
||||||
Scanner scanner = new Scanner(data);
|
Scanner scanner = new Scanner(data);
|
||||||
StringBuilder sender = new StringBuilder();
|
StringBuilder sender = new StringBuilder();
|
||||||
StringBuilder reciever = new StringBuilder();
|
StringBuilder receiver = new StringBuilder();
|
||||||
StringBuilder type = new StringBuilder();
|
StringBuilder type = new StringBuilder();
|
||||||
StringBuilder payload = new StringBuilder();
|
StringBuilder payload = new StringBuilder();
|
||||||
super.processData(scanner,sender,reciever,type,payload);
|
super.processData(scanner, sender, receiver, type, payload);
|
||||||
|
|
||||||
// dispatch operation based on type parameter
|
// dispatch operation based on type parameter
|
||||||
if (type.toString().equals(getDataTypeConnect())) {
|
if (type.toString().equals(getDataTypeConnect())) {
|
||||||
System.err.println("Illegal connect request from server");
|
System.err.println("Illegal connect request from server");
|
||||||
} else if (type.toString().equals(getDataTypeConfirm())) {
|
} else if (type.toString().equals(getDataTypeConfirm())) {
|
||||||
caseConfirm(sender.toString(), reciever.toString(), payload.toString());
|
caseConfirm(sender.toString(), receiver.toString(), payload.toString());
|
||||||
} else if (type.toString().equals(getDataTypeDisconnect())) {
|
} else if (type.toString().equals(getDataTypeDisconnect())) {
|
||||||
caseDisconnect(sender.toString(),reciever.toString(),payload.toString());
|
caseDisconnect(sender.toString(), receiver.toString(), payload.toString());
|
||||||
} else if (type.toString().equals(getDataTypeMessage())) {
|
} else if (type.toString().equals(getDataTypeMessage())) {
|
||||||
caseMessage(sender.toString(),reciever.toString(),payload.toString());
|
caseMessage(sender.toString(), receiver.toString(), payload.toString());
|
||||||
} else if (type.toString().equals(getDataTypeError())) {
|
} else if (type.toString().equals(getDataTypeError())) {
|
||||||
caseError(sender.toString(), reciever.toString(), payload.toString());
|
caseError(sender.toString(), receiver.toString(), payload.toString());
|
||||||
} else {
|
} else {
|
||||||
System.out.println("Unknown data type received: " + type);
|
System.out.println("Unknown data type received: " + type);
|
||||||
}
|
}
|
||||||
|
@ -126,16 +191,16 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void caseConfirm(String sender, String reciever, String payload) {
|
private void caseConfirm(String sender, String receiver, String payload) {
|
||||||
if (state.get() == CONFIRM_CONNECT) {
|
if (state.get() == CONFIRM_CONNECT) {
|
||||||
this.userName.set(reciever);
|
this.userName.set(receiver);
|
||||||
this.serverPort.set(getConnection().getRemotePort());
|
this.serverPort.set(getConnection().getRemotePort());
|
||||||
this.serverAddress.set(getConnection().getRemoteHost());
|
this.serverAddress.set(getConnection().getRemoteHost());
|
||||||
messages.addMessage(new Message(Message.MessageType.INFO,sender,reciever,payload));
|
messages.addMessage(new Message(Message.MessageType.INFO, sender, receiver, payload));
|
||||||
System.out.println("CONFIRM: " + payload);
|
System.out.println("CONFIRM: " + payload);
|
||||||
this.setState(CONNECTED);
|
this.setState(CONNECTED);
|
||||||
} else if (state.get() == CONFIRM_DISCONNECT) {
|
} else if (state.get() == CONFIRM_DISCONNECT) {
|
||||||
messages.addMessage(new Message(Message.MessageType.INFO,sender,reciever,payload));
|
messages.addMessage(new Message(Message.MessageType.INFO, sender, receiver, payload));
|
||||||
System.out.println("CONFIRM: " + payload);
|
System.out.println("CONFIRM: " + payload);
|
||||||
this.setState(DISCONNECTED);
|
this.setState(DISCONNECTED);
|
||||||
} else {
|
} else {
|
||||||
|
@ -143,42 +208,81 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void caseDisconnect(String sender, String reciever, String payload) {
|
/**
|
||||||
|
* Initiates the disconnect sequence and sends the message with all its info.
|
||||||
|
*
|
||||||
|
* @param sender the sender name of the message
|
||||||
|
* @param receiver the receiver name of the message
|
||||||
|
* @param payload the added payload that corresponds to the message
|
||||||
|
*/
|
||||||
|
private void caseDisconnect(String sender, String receiver, String payload) {
|
||||||
if (state.get() == DISCONNECTED) {
|
if (state.get() == DISCONNECTED) {
|
||||||
System.out.println("DISCONNECT: Already in disconnected: " + payload);
|
System.out.println("DISCONNECT: Already in disconnected: " + payload);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
messages.addMessage(new Message(Message.MessageType.INFO,sender,reciever,payload));
|
messages.addMessage(new Message(Message.MessageType.INFO, sender, receiver, payload));
|
||||||
System.out.println("DISCONNECT: " + payload);
|
System.out.println("DISCONNECT: " + payload);
|
||||||
this.setState(DISCONNECTED);
|
this.setState(DISCONNECTED);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void caseMessage(String sender, String reciever, String payload) {
|
/**
|
||||||
|
* Initiates the procedure to send a new message and sends one as such.
|
||||||
|
*
|
||||||
|
* @param sender the sender name of the message
|
||||||
|
* @param receiver the receiver name of the message
|
||||||
|
* @param payload the added payload that corresponds to the message
|
||||||
|
*/
|
||||||
|
private void caseMessage(String sender, String receiver, String payload) {
|
||||||
if (state.get() != CONNECTED) {
|
if (state.get() != CONNECTED) {
|
||||||
System.out.println("MESSAGE: Illegal state " + state + " for message: " + payload);
|
System.out.println("MESSAGE: Illegal state " + state + " for message: " + payload);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
messages.addMessage(new Message(Message.MessageType.MESSAGE,sender,reciever,payload));
|
messages.addMessage(new Message(Message.MessageType.MESSAGE, sender, receiver, payload));
|
||||||
System.out.println("MESSAGE: From " + sender + " to " + reciever + ": "+ payload);
|
System.out.println("MESSAGE: From " + sender + " to " + receiver + ": " + payload);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void caseError(String sender, String reciever, String payload) {
|
/**
|
||||||
messages.addMessage(new Message(Message.MessageType.ERROR,sender,reciever,payload));
|
* Stores the message as an error message and displays it as such as well.
|
||||||
|
*
|
||||||
|
* @param sender the sender name of the message
|
||||||
|
* @param receiver the receiver name of the message
|
||||||
|
* @param payload the added payload that corresponds to the message
|
||||||
|
*/
|
||||||
|
private void caseError(String sender, String receiver, String payload) {
|
||||||
|
messages.addMessage(new Message(Message.MessageType.ERROR, sender, receiver, payload));
|
||||||
System.out.println("ERROR: " + payload);
|
System.out.println("ERROR: " + payload);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send connect attempt to Server
|
||||||
|
*
|
||||||
|
* @throws ChatProtocolException Error that is thrown if the state is not set to NEW
|
||||||
|
*/
|
||||||
public void connect() throws ChatProtocolException {
|
public void connect() throws ChatProtocolException {
|
||||||
if (state.get() != NEW) throw new ChatProtocolException("Illegal state for connect: " + state);
|
if (state.get() != NEW) throw new ChatProtocolException("Illegal state for connect: " + state);
|
||||||
this.sendData(userName.get(), USER_NONE, getDataTypeConnect(),null);
|
this.sendData(userName.get(), USER_NONE, getDataTypeConnect(), null);
|
||||||
this.setState(CONFIRM_CONNECT);
|
this.setState(CONFIRM_CONNECT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send Disconnect attempt to server
|
||||||
|
*
|
||||||
|
* @throws ChatProtocolException Error tha si thrown if state is invalid
|
||||||
|
*/
|
||||||
public void disconnect() throws ChatProtocolException {
|
public void disconnect() throws ChatProtocolException {
|
||||||
if (state.get() != NEW && state.get() != CONNECTED) throw new ChatProtocolException("Illegal state for disconnect: " + state);
|
if (state.get() != NEW && state.get() != CONNECTED)
|
||||||
this.sendData(userName.get(), USER_NONE, getDataTypeDisconnect(),null);
|
throw new ChatProtocolException("Illegal state for disconnect: " + state);
|
||||||
|
this.sendData(userName.get(), USER_NONE, getDataTypeDisconnect(), null);
|
||||||
this.setState(CONFIRM_DISCONNECT);
|
this.setState(CONFIRM_DISCONNECT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends message to Server
|
||||||
|
*
|
||||||
|
* @param messageString The message the user wants to send
|
||||||
|
* @return true if message is valid else false
|
||||||
|
* @throws ChatProtocolException if illegal connection state
|
||||||
|
*/
|
||||||
public boolean message(String messageString) throws ChatProtocolException {
|
public boolean message(String messageString) throws ChatProtocolException {
|
||||||
if (state.get() != CONNECTED) throw new ChatProtocolException("Illegal state for message: " + state);
|
if (state.get() != CONNECTED) throw new ChatProtocolException("Illegal state for message: " + state);
|
||||||
|
|
||||||
|
@ -186,11 +290,11 @@ public class ClientConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
if (matcher.find()) {
|
if (matcher.find()) {
|
||||||
String receiver = matcher.group(1);
|
String receiver = matcher.group(1);
|
||||||
String message = matcher.group(2);
|
String message = matcher.group(2);
|
||||||
if(message.length() < 1){
|
if (message.length() < 1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (receiver == null || receiver.isBlank()) receiver = ClientConnectionHandler.USER_ALL;
|
if (receiver == null || receiver.isBlank()) receiver = ClientConnectionHandler.USER_ALL;
|
||||||
this.sendData(userName.get(), receiver, getDataTypeMessage(),message);
|
this.sendData(userName.get(), receiver, getDataTypeMessage(), message);
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -5,37 +5,63 @@ import javafx.beans.property.SimpleBooleanProperty;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class that is used to store the messages in an ArrayList
|
||||||
|
* And also as a Model that informs the controller via Listeners that can be initialized
|
||||||
|
*/
|
||||||
public class ClientMessageList {
|
public class ClientMessageList {
|
||||||
private List<Message> messages = new ArrayList<>();
|
private List<Message> messages = new ArrayList<>();
|
||||||
private SimpleBooleanProperty changed = new SimpleBooleanProperty(false);
|
private final SimpleBooleanProperty changed = new SimpleBooleanProperty(false);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds a new message to ArrayList and also informs Listener.
|
||||||
|
* @param message that should be added
|
||||||
|
*/
|
||||||
public void addMessage(Message message) {
|
public void addMessage(Message message) {
|
||||||
messages.add(message);
|
messages.add(message);
|
||||||
changed.set(!changed.get());
|
changed.set(!changed.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getFilteredMessages(String filter) {
|
/**
|
||||||
|
* Applies a given filter over all messages and returns the result as a new String.
|
||||||
|
* Method is also in charge of applying the correct prefix to the message according to its type.
|
||||||
|
*
|
||||||
|
* @param filter is the applied filter on all the messages that are stored.
|
||||||
|
* @return String that matches the given filter.
|
||||||
|
*/
|
||||||
|
public String getFilteredMessages(String filter) {
|
||||||
StringBuilder result = new StringBuilder();
|
StringBuilder result = new StringBuilder();
|
||||||
boolean showAll = filter == null || filter.isBlank();
|
boolean showAll = filter == null || filter.isBlank();
|
||||||
for(Message message : messages) {
|
for (Message message : messages) {
|
||||||
if(showAll || message.matchesFilter(filter))
|
if (showAll || message.matchesFilter(filter)) {
|
||||||
{
|
|
||||||
switch (message.getType()) {
|
switch (message.getType()) {
|
||||||
case MESSAGE -> result.append(String.format("[%s -> %s] %s\n", message.getSender(), message.getReceiver(), message.getText()));
|
case MESSAGE ->
|
||||||
|
result.append(String.format("[%s -> %s] %s\n", message.getSender(), message.getReceiver(), message.getText()));
|
||||||
case ERROR -> result.append(String.format("[ERROR] %s\n", message.getText()));
|
case ERROR -> result.append(String.format("[ERROR] %s\n", message.getText()));
|
||||||
case INFO -> result.append(String.format("[INFO] %s\n", message.getText()));
|
case INFO -> result.append(String.format("[INFO] %s\n", message.getText()));
|
||||||
default -> result.append(String.format("[ERROR] %s\n", "Unexpected message type: " + message.getType()));
|
default ->
|
||||||
|
result.append(String.format("[ERROR] %s\n", "Unexpected message type: " + message.getType()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result.toString();
|
return result.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Overwrites the Arraylist of messages, clearing it, also informs all Listeners.
|
||||||
|
*/
|
||||||
public void clear() {
|
public void clear() {
|
||||||
messages = new ArrayList<>();
|
messages = new ArrayList<>();
|
||||||
changed.set(!changed.get());
|
changed.set(!changed.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
public SimpleBooleanProperty getChangedProperty() { return changed; }
|
/**
|
||||||
|
* Getter Method to check the current value of SimpleBooleanProperty changed.
|
||||||
|
*
|
||||||
|
* @return the current value of changed
|
||||||
|
*/
|
||||||
|
public SimpleBooleanProperty getChangedProperty() {
|
||||||
|
return changed;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,15 +6,32 @@ import javafx.scene.Scene;
|
||||||
import javafx.scene.layout.Pane;
|
import javafx.scene.layout.Pane;
|
||||||
import javafx.stage.Stage;
|
import javafx.stage.Stage;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class in charge of initializing and loading elements needed for the GUI.
|
||||||
|
* Also Launching the chat window with pre-given parameters
|
||||||
|
*/
|
||||||
public class ClientUI extends Application {
|
public class ClientUI extends Application {
|
||||||
private ClientMessageList clientMessageList = new ClientMessageList();
|
private final ClientMessageList clientMessageList = new ClientMessageList();
|
||||||
private ClientConnectionHandler connectionHandler = new ClientConnectionHandler(clientMessageList);
|
private final ClientConnectionHandler connectionHandler = new ClientConnectionHandler(clientMessageList);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start method will only run the chatWindow method.
|
||||||
|
*
|
||||||
|
* @param primaryStage the primary stage for this application, onto which
|
||||||
|
* the application scene can be set.
|
||||||
|
* Applications may create other stages, if needed, but they will not be
|
||||||
|
* primary stages.
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void start(Stage primaryStage) {
|
public void start(Stage primaryStage) {
|
||||||
chatWindow(primaryStage);
|
chatWindow(primaryStage);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method is in charge of loading the FXML file and then initializing the controllers and setting some parameters for the stage.
|
||||||
|
*
|
||||||
|
* @param primaryStage of the Application onto which the Scene is set.
|
||||||
|
*/
|
||||||
private void chatWindow(Stage primaryStage) {
|
private void chatWindow(Stage primaryStage) {
|
||||||
try {
|
try {
|
||||||
FXMLLoader loader = new FXMLLoader(getClass().getResource("ChatWindow.fxml"));
|
FXMLLoader loader = new FXMLLoader(getClass().getResource("ChatWindow.fxml"));
|
||||||
|
@ -34,7 +51,7 @@ public class ClientUI extends Application {
|
||||||
primaryStage.setTitle("Multichat Client");
|
primaryStage.setTitle("Multichat Client");
|
||||||
primaryStage.show();
|
primaryStage.show();
|
||||||
primaryStage.setMinWidth(primaryStage.getWidth()); //use automatically computed size as Minimum Size.
|
primaryStage.setMinWidth(primaryStage.getWidth()); //use automatically computed size as Minimum Size.
|
||||||
} catch(Exception e) {
|
} catch (Exception e) {
|
||||||
System.err.println("Error starting up UI" + e.getMessage());
|
System.err.println("Error starting up UI" + e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,20 +1,21 @@
|
||||||
package ch.zhaw.pm2.multichat.client;
|
package ch.zhaw.pm2.multichat.client;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
A Message object represents one Message of a client. Can be stored in ClientMessageList.
|
* A Message object represents one Message of a client. Can be stored in ClientMessageList.
|
||||||
*/
|
*/
|
||||||
public class Message {
|
public class Message {
|
||||||
private MessageType type;
|
private final MessageType type;
|
||||||
private String sender;
|
private final String sender;
|
||||||
private String receiver;
|
private final String receiver;
|
||||||
private String text;
|
private final String text;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor of Message. Needs all Information about a Message to save them.
|
* Constructor of Message. Needs all Information about a Message to save them.
|
||||||
* @param type Message (if it's a message typed by a user), Error or Information (if it is generated automatically, in this case sender and reciever will be null)
|
*
|
||||||
* @param sender The User who has sent the message.
|
* @param type Message (if it's a message typed by a user), Error or Information (if it is generated automatically, in this case sender and receiver will be null)
|
||||||
* @param receiver The User who should recieve the message.
|
* @param sender The User who has sent the message.
|
||||||
* @param text The Text of the message.
|
* @param receiver The User who should receive the message.
|
||||||
|
* @param text The Text of the message.
|
||||||
*/
|
*/
|
||||||
public Message(MessageType type, String sender, String receiver, String text) {
|
public Message(MessageType type, String sender, String receiver, String text) {
|
||||||
this.type = type;
|
this.type = type;
|
||||||
|
@ -24,14 +25,15 @@ public class Message {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the Filter String is contained in one of the datafields sender, receiver or text
|
* Checks if the Filter String is contained in one of the data fields sender, receiver or text
|
||||||
|
*
|
||||||
* @param filter The Filter String
|
* @param filter The Filter String
|
||||||
* @return true if it the Filter String is contained in a datafield.
|
* @return true if it is the Filter String is contained in a data field.
|
||||||
*/
|
*/
|
||||||
public boolean matchesFilter(String filter){
|
public boolean matchesFilter(String filter) {
|
||||||
return (sender != null && sender.contains(filter)) ||
|
return (sender != null && sender.contains(filter)) ||
|
||||||
(receiver != null && receiver.contains(filter)) ||
|
(receiver != null && receiver.contains(filter)) ||
|
||||||
(text != null && text.contains(filter));
|
(text != null && text.contains(filter));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -49,7 +51,7 @@ public class Message {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return The Reciever who recieves the Message.
|
* @return The Receiver who receives the Message.
|
||||||
*/
|
*/
|
||||||
public String getReceiver() {
|
public String getReceiver() {
|
||||||
return receiver;
|
return receiver;
|
||||||
|
@ -63,7 +65,7 @@ public class Message {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enummeration of Message Types.
|
* Enumeration of Message Types.
|
||||||
*/
|
*/
|
||||||
public enum MessageType {
|
public enum MessageType {
|
||||||
INFO, MESSAGE, ERROR;
|
INFO, MESSAGE, ERROR;
|
||||||
|
|
|
@ -12,74 +12,82 @@
|
||||||
<?import javafx.scene.layout.HBox?>
|
<?import javafx.scene.layout.HBox?>
|
||||||
<?import javafx.scene.layout.VBox?>
|
<?import javafx.scene.layout.VBox?>
|
||||||
|
|
||||||
<BorderPane fx:id="rootPane" minWidth="-Infinity" prefHeight="500.0" prefWidth="420.0" xmlns="http://javafx.com/javafx/18" xmlns:fx="http://javafx.com/fxml/1" fx:controller="ch.zhaw.pm2.multichat.client.ChatWindowController">
|
<BorderPane xmlns:fx="http://javafx.com/fxml/1" fx:id="rootPane" minWidth="-Infinity" prefHeight="500.0"
|
||||||
<top>
|
prefWidth="420.0" xmlns="http://javafx.com/javafx/18"
|
||||||
<VBox BorderPane.alignment="CENTER">
|
fx:controller="ch.zhaw.pm2.multichat.client.ChatWindowController">
|
||||||
<children>
|
<top>
|
||||||
<MenuBar>
|
<VBox BorderPane.alignment="CENTER">
|
||||||
<menus>
|
<children>
|
||||||
<Menu mnemonicParsing="false" text="Filter">
|
<MenuBar>
|
||||||
<items>
|
<menus>
|
||||||
<MenuItem mnemonicParsing="false" text="press Enter">
|
<Menu mnemonicParsing="false" text="Filter">
|
||||||
<graphic>
|
<items>
|
||||||
<TextField fx:id="filterValue" onAction="#applyFilter" />
|
<MenuItem mnemonicParsing="false" text="press Enter">
|
||||||
</graphic>
|
<graphic>
|
||||||
</MenuItem>
|
<TextField fx:id="filterValue" onAction="#applyFilter"/>
|
||||||
</items>
|
</graphic>
|
||||||
</Menu>
|
</MenuItem>
|
||||||
</menus>
|
</items>
|
||||||
</MenuBar>
|
</Menu>
|
||||||
<HBox fillHeight="false" spacing="5.0">
|
</menus>
|
||||||
<children>
|
</MenuBar>
|
||||||
<TextField fx:id="userNameField" alignment="CENTER_RIGHT" maxWidth="1.7976931348623157E308" minWidth="110.0" promptText="Username" HBox.hgrow="SOMETIMES" />
|
<HBox fillHeight="false" spacing="5.0">
|
||||||
<Label alignment="CENTER" contentDisplay="CENTER" text="\@" textAlignment="CENTER" textOverrun="CLIP" HBox.hgrow="NEVER">
|
<children>
|
||||||
<HBox.margin>
|
<TextField fx:id="userNameField" alignment="CENTER_RIGHT" maxWidth="1.7976931348623157E308"
|
||||||
<Insets bottom="5.0" top="5.0" />
|
minWidth="110.0" promptText="Username" HBox.hgrow="SOMETIMES"/>
|
||||||
</HBox.margin>
|
<Label alignment="CENTER" contentDisplay="CENTER" text="\@" textAlignment="CENTER"
|
||||||
</Label>
|
textOverrun="CLIP" HBox.hgrow="NEVER">
|
||||||
<TextField fx:id="serverAddressField" alignment="CENTER_RIGHT" minWidth="110.0" promptText="Host" HBox.hgrow="SOMETIMES" />
|
<HBox.margin>
|
||||||
<Label text=":" HBox.hgrow="NEVER">
|
<Insets bottom="5.0" top="5.0"/>
|
||||||
<HBox.margin>
|
</HBox.margin>
|
||||||
<Insets bottom="5.0" top="5.0" />
|
</Label>
|
||||||
</HBox.margin>
|
<TextField fx:id="serverAddressField" alignment="CENTER_RIGHT" minWidth="110.0"
|
||||||
</Label>
|
promptText="Host" HBox.hgrow="SOMETIMES"/>
|
||||||
<TextField fx:id="serverPortField" minWidth="-Infinity" prefWidth="60.0" promptText="Port" HBox.hgrow="NEVER" />
|
<Label text=":" HBox.hgrow="NEVER">
|
||||||
<Button fx:id="connectButton" maxWidth="-Infinity" minWidth="-Infinity" mnemonicParsing="false" onAction="#toggleConnection" prefWidth="80.0" text="Connect" HBox.hgrow="NEVER">
|
<HBox.margin>
|
||||||
<HBox.margin>
|
<Insets bottom="5.0" top="5.0"/>
|
||||||
<Insets left="5.0" />
|
</HBox.margin>
|
||||||
</HBox.margin>
|
</Label>
|
||||||
</Button>
|
<TextField fx:id="serverPortField" minWidth="-Infinity" prefWidth="60.0" promptText="Port"
|
||||||
</children>
|
HBox.hgrow="NEVER"/>
|
||||||
<BorderPane.margin>
|
<Button fx:id="connectButton" maxWidth="-Infinity" minWidth="-Infinity" mnemonicParsing="false"
|
||||||
<Insets />
|
onAction="#toggleConnection" prefWidth="80.0" text="Connect" HBox.hgrow="NEVER">
|
||||||
</BorderPane.margin>
|
<HBox.margin>
|
||||||
<padding>
|
<Insets left="5.0"/>
|
||||||
<Insets bottom="5.0" left="5.0" right="5.0" top="5.0" />
|
</HBox.margin>
|
||||||
</padding>
|
</Button>
|
||||||
</HBox>
|
</children>
|
||||||
</children>
|
<BorderPane.margin>
|
||||||
</VBox>
|
<Insets/>
|
||||||
</top>
|
</BorderPane.margin>
|
||||||
<bottom>
|
<padding>
|
||||||
<HBox spacing="5.0">
|
<Insets bottom="5.0" left="5.0" right="5.0" top="5.0"/>
|
||||||
<children>
|
</padding>
|
||||||
<TextField fx:id="messageField" onAction="#message" HBox.hgrow="ALWAYS" />
|
</HBox>
|
||||||
<Button fx:id="sendButton" alignment="CENTER" maxWidth="-Infinity" minWidth="-Infinity" mnemonicParsing="false" onAction="#message" prefWidth="50.0" text="Send" textAlignment="CENTER">
|
</children>
|
||||||
<HBox.margin>
|
</VBox>
|
||||||
<Insets left="5.0" />
|
</top>
|
||||||
</HBox.margin>
|
<bottom>
|
||||||
</Button>
|
<HBox spacing="5.0">
|
||||||
</children>
|
<children>
|
||||||
<padding>
|
<TextField fx:id="messageField" onAction="#message" HBox.hgrow="ALWAYS"/>
|
||||||
<Insets bottom="5.0" left="5.0" right="5.0" top="5.0" />
|
<Button fx:id="sendButton" alignment="CENTER" maxWidth="-Infinity" minWidth="-Infinity"
|
||||||
</padding>
|
mnemonicParsing="false" onAction="#message" prefWidth="50.0" text="Send" textAlignment="CENTER">
|
||||||
</HBox>
|
<HBox.margin>
|
||||||
</bottom>
|
<Insets left="5.0"/>
|
||||||
<center>
|
</HBox.margin>
|
||||||
<TextArea fx:id="messageArea" editable="false" wrapText="true">
|
</Button>
|
||||||
<BorderPane.margin>
|
</children>
|
||||||
<Insets left="5.0" right="5.0" />
|
<padding>
|
||||||
</BorderPane.margin>
|
<Insets bottom="5.0" left="5.0" right="5.0" top="5.0"/>
|
||||||
</TextArea>
|
</padding>
|
||||||
</center>
|
</HBox>
|
||||||
|
</bottom>
|
||||||
|
<center>
|
||||||
|
<TextArea fx:id="messageArea" editable="false" wrapText="true">
|
||||||
|
<BorderPane.margin>
|
||||||
|
<Insets left="5.0" right="5.0"/>
|
||||||
|
</BorderPane.margin>
|
||||||
|
</TextArea>
|
||||||
|
</center>
|
||||||
</BorderPane>
|
</BorderPane>
|
||||||
|
|
|
@ -8,7 +8,7 @@ import java.util.Scanner;
|
||||||
/**
|
/**
|
||||||
* This abstract class is the superclass for ClientConnectionHandler and ServerConnectionHandler
|
* This abstract class is the superclass for ClientConnectionHandler and ServerConnectionHandler
|
||||||
* It offers the DATA_TYPE Strings and a {@link State} enum for all valid connection states.
|
* It offers the DATA_TYPE Strings and a {@link State} enum for all valid connection states.
|
||||||
* Shared methods are implemented in this class aswell {@link ConnectionHandler#sendData(String, String, String, String)} {@link ConnectionHandler#processData(Scanner, StringBuilder, StringBuilder, StringBuilder, StringBuilder)}
|
* Shared methods are implemented in this class as well {@link ConnectionHandler#sendData(String, String, String, String)} {@link ConnectionHandler#processData(Scanner, StringBuilder, StringBuilder, StringBuilder, StringBuilder)}
|
||||||
*/
|
*/
|
||||||
public abstract class ConnectionHandler {
|
public abstract class ConnectionHandler {
|
||||||
private NetworkHandler.NetworkConnection<String> connection;
|
private NetworkHandler.NetworkConnection<String> connection;
|
||||||
|
@ -29,7 +29,6 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @return {@link ConnectionHandler#DATA_TYPE_CONNECT}
|
* @return {@link ConnectionHandler#DATA_TYPE_CONNECT}
|
||||||
*/
|
*/
|
||||||
public static String getDataTypeConnect() {
|
public static String getDataTypeConnect() {
|
||||||
|
@ -37,7 +36,6 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @return {@link ConnectionHandler#DATA_TYPE_CONFIRM}
|
* @return {@link ConnectionHandler#DATA_TYPE_CONFIRM}
|
||||||
*/
|
*/
|
||||||
public static String getDataTypeConfirm() {
|
public static String getDataTypeConfirm() {
|
||||||
|
@ -45,7 +43,6 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @return {@link ConnectionHandler#DATA_TYPE_DISCONNECT}
|
* @return {@link ConnectionHandler#DATA_TYPE_DISCONNECT}
|
||||||
*/
|
*/
|
||||||
public static String getDataTypeDisconnect() {
|
public static String getDataTypeDisconnect() {
|
||||||
|
@ -53,7 +50,6 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @return {@link ConnectionHandler#DATA_TYPE_MESSAGE
|
* @return {@link ConnectionHandler#DATA_TYPE_MESSAGE
|
||||||
*/
|
*/
|
||||||
public static String getDataTypeMessage() {
|
public static String getDataTypeMessage() {
|
||||||
|
@ -61,7 +57,6 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @return {@link ConnectionHandler#DATA_TYPE_ERROR}
|
* @return {@link ConnectionHandler#DATA_TYPE_ERROR}
|
||||||
*/
|
*/
|
||||||
public static String getDataTypeError() {
|
public static String getDataTypeError() {
|
||||||
|
@ -69,7 +64,6 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @return {@link NetworkHandler.NetworkConnection}
|
* @return {@link NetworkHandler.NetworkConnection}
|
||||||
*/
|
*/
|
||||||
public NetworkHandler.NetworkConnection<String> getConnection() {
|
public NetworkHandler.NetworkConnection<String> getConnection() {
|
||||||
|
@ -78,6 +72,7 @@ public abstract class ConnectionHandler {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method sets the NetworkConnection used for the server <-> client connection
|
* This method sets the NetworkConnection used for the server <-> client connection
|
||||||
|
*
|
||||||
* @param connection NetworkConnection used for the server <-> client connection
|
* @param connection NetworkConnection used for the server <-> client connection
|
||||||
*/
|
*/
|
||||||
protected void setConnection(NetworkHandler.NetworkConnection<String> connection) {
|
protected void setConnection(NetworkHandler.NetworkConnection<String> connection) {
|
||||||
|
@ -85,24 +80,27 @@ public abstract class ConnectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method reads the data when a ConnectionHandler recieves it. It tries to read out the sender, reciever, type and payload.
|
* This method reads the data when a ConnectionHandler receives it. It tries to read out the sender, receiver, type and payload.
|
||||||
* If the data does not contain the expected number of lines, it throws a {@link ChatProtocolException}
|
* If the data does not contain the expected number of lines, it throws a {@link ChatProtocolException}
|
||||||
* @param scanner to read data
|
*
|
||||||
* @param sender of the data
|
* @param scanner to read data
|
||||||
* @param reciever for the data
|
* @param sender of the data
|
||||||
* @param type of data
|
* @param receiver for the data
|
||||||
* @param payload the data sent
|
* @param type of data
|
||||||
|
* @param payload the data sent
|
||||||
* @throws ChatProtocolException if the data does not contain the expected number of lines
|
* @throws ChatProtocolException if the data does not contain the expected number of lines
|
||||||
*/
|
*/
|
||||||
protected void processData(Scanner scanner, StringBuilder sender, StringBuilder reciever, StringBuilder type, StringBuilder payload) throws ChatProtocolException {
|
protected void processData(Scanner scanner, StringBuilder sender, StringBuilder receiver, StringBuilder type, StringBuilder payload) throws ChatProtocolException {
|
||||||
// parse data content
|
// parse data content
|
||||||
if (scanner.hasNextLine()) {
|
if (scanner.hasNextLine()) {
|
||||||
sender.append(scanner.nextLine());
|
sender.append(scanner.nextLine());
|
||||||
|
} else if (scanner.hasNextLine()) {
|
||||||
|
receiver.append(scanner.nextLine());
|
||||||
} else {
|
} else {
|
||||||
throw new ChatProtocolException("No Sender found");
|
throw new ChatProtocolException("No Sender found");
|
||||||
}
|
}
|
||||||
if (scanner.hasNextLine()) {
|
if (scanner.hasNextLine()) {
|
||||||
reciever.append(scanner.nextLine());
|
receiver.append(scanner.nextLine());
|
||||||
} else {
|
} else {
|
||||||
throw new ChatProtocolException("No Reciever found");
|
throw new ChatProtocolException("No Reciever found");
|
||||||
}
|
}
|
||||||
|
@ -118,27 +116,25 @@ public abstract class ConnectionHandler {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method gets called to send data via the socket defined in the {@link NetworkHandler.NetworkConnection}
|
* This method gets called to send data via the socket defined in the {@link NetworkHandler.NetworkConnection}
|
||||||
* @param sender of the data
|
*
|
||||||
|
* @param sender of the data
|
||||||
* @param receiver of the data
|
* @param receiver of the data
|
||||||
* @param type of the data
|
* @param type of the data
|
||||||
* @param payload of the data
|
* @param payload of the data
|
||||||
*/
|
*/
|
||||||
protected void sendData(String sender, String receiver, String type, String payload) {
|
protected void sendData(String sender, String receiver, String type, String payload) {
|
||||||
if (connection.isAvailable()) {
|
if (connection.isAvailable()) {
|
||||||
new StringBuilder();
|
String data = sender + "\n" +
|
||||||
String data = new StringBuilder()
|
receiver + "\n" +
|
||||||
.append(sender+"\n")
|
type + "\n" +
|
||||||
.append(receiver+"\n")
|
payload + "\n";
|
||||||
.append(type+"\n")
|
|
||||||
.append(payload+"\n")
|
|
||||||
.toString();
|
|
||||||
try {
|
try {
|
||||||
connection.send(data);
|
connection.send(data);
|
||||||
} catch (SocketException e) {
|
} catch (SocketException e) {
|
||||||
System.err.println("Connection closed: " + e.getMessage());
|
System.err.println("Connection closed: " + e.getMessage());
|
||||||
} catch (EOFException e) {
|
} catch (EOFException e) {
|
||||||
System.out.println("Connection terminated by remote");
|
System.out.println("Connection terminated by remote");
|
||||||
} catch(IOException e) {
|
} catch (IOException e) {
|
||||||
System.err.println("Communication error: " + e.getMessage());
|
System.err.println("Communication error: " + e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,14 +1,12 @@
|
||||||
package ch.zhaw.pm2.multichat.server;
|
package ch.zhaw.pm2.multichat.server;
|
||||||
|
|
||||||
import ch.zhaw.pm2.multichat.protocol.ConnectionHandler;
|
|
||||||
import ch.zhaw.pm2.multichat.protocol.NetworkHandler;
|
import ch.zhaw.pm2.multichat.protocol.NetworkHandler;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.SocketException;
|
import java.net.SocketException;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.locks.Condition;
|
|
||||||
import java.util.concurrent.locks.ReentrantLock;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This Class represents a Server. The user can start the programm with the port number as a argument.
|
* This Class represents a Server. The user can start the programm with the port number as a argument.
|
||||||
|
@ -16,17 +14,32 @@ import java.util.concurrent.locks.ReentrantLock;
|
||||||
* After initialising the server:
|
* After initialising the server:
|
||||||
* 1. Starts a Socketserver using the logic given in {@link NetworkHandler.NetworkServer#createServer()}
|
* 1. Starts a Socketserver using the logic given in {@link NetworkHandler.NetworkServer#createServer()}
|
||||||
* 2. The server starts to listen for incoming connection using the Logic given in {@link NetworkHandler.NetworkServer#waitForConnection()}
|
* 2. The server starts to listen for incoming connection using the Logic given in {@link NetworkHandler.NetworkServer#waitForConnection()}
|
||||||
* 3. New conections will be atached to a Connectionhandler: {@link ServerConnectionHandler} and placed in a Map containing all active connections {@link Server#connections}
|
* 3. New connections will be attached to a Connection handler: {@link ServerConnectionHandler} and placed in a Map containing all active connections {@link Server#connections}
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
public class Server {
|
public class Server {
|
||||||
|
|
||||||
// Server connection
|
// Server connection
|
||||||
private NetworkHandler.NetworkServer<String> networkServer;
|
private final NetworkHandler.NetworkServer<String> networkServer;
|
||||||
|
|
||||||
// Connection registry
|
// Connection registry
|
||||||
private Map<String,ServerConnectionHandler> connections = new HashMap<>();
|
private final Map<String, ServerConnectionHandler> connections = new HashMap<>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The Constructor to create a new instance.
|
||||||
|
*
|
||||||
|
* @param serverPort to listen for incoming connections.
|
||||||
|
* @throws IOException thrown if an I/O error occurs when opening the socket.
|
||||||
|
*/
|
||||||
|
public Server(int serverPort) throws IOException {
|
||||||
|
// Open server connection
|
||||||
|
System.out.println("Create server connection");
|
||||||
|
networkServer = NetworkHandler.createServer(serverPort);
|
||||||
|
System.out.println("Listening on " + networkServer.getHostAddress() + ":" + networkServer.getHostPort());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param args
|
||||||
|
*/
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
// Parse arguments for server port.
|
// Parse arguments for server port.
|
||||||
try {
|
try {
|
||||||
|
@ -65,33 +78,19 @@ public class Server {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* The Constructor to create a new instance.
|
|
||||||
* @param serverPort to listen for incoming connections.
|
|
||||||
* @throws IOException thrown if an I/O error occurs when opening the socket.
|
|
||||||
*/
|
|
||||||
public Server(int serverPort) throws IOException {
|
|
||||||
// Open server connection
|
|
||||||
System.out.println("Create server connection");
|
|
||||||
networkServer = NetworkHandler.createServer(serverPort);
|
|
||||||
System.out.println("Listening on " + networkServer.getHostAddress() + ":" + networkServer.getHostPort());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* With this methode the instance waits for incoming connections. If a client tries to connect to the server.
|
* With this methode the instance waits for incoming connections. If a client tries to connect to the server.
|
||||||
* The connection will be registered in the connection registry if successful.
|
* The connection will be registered in the connection registry if successful.
|
||||||
*/
|
*/
|
||||||
private void start() {
|
private void start() {
|
||||||
ReentrantLock mutex = new ReentrantLock();
|
|
||||||
Condition nameComplete = mutex.newCondition();
|
|
||||||
System.out.println("Server started.");
|
System.out.println("Server started.");
|
||||||
try {
|
try {
|
||||||
while (true) {
|
while (true) {
|
||||||
NetworkHandler.NetworkConnection<String> connection = networkServer.waitForConnection();
|
NetworkHandler.NetworkConnection<String> connection = networkServer.waitForConnection();
|
||||||
ServerConnectionHandler connectionHandler = new ServerConnectionHandler(connection, connections, mutex, nameComplete);
|
ServerConnectionHandler connectionHandler = new ServerConnectionHandler(connection, connections);
|
||||||
new Thread(connectionHandler).start();
|
new Thread(connectionHandler).start();
|
||||||
}
|
}
|
||||||
} catch(SocketException e) {
|
} catch (SocketException e) {
|
||||||
System.out.println("Server connection terminated");
|
System.out.println("Server connection terminated");
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
System.err.println("Communication error " + e);
|
System.err.println("Communication error " + e);
|
||||||
|
|
|
@ -2,7 +2,9 @@ package ch.zhaw.pm2.multichat.server;
|
||||||
|
|
||||||
import ch.zhaw.pm2.multichat.protocol.ChatProtocolException;
|
import ch.zhaw.pm2.multichat.protocol.ChatProtocolException;
|
||||||
import ch.zhaw.pm2.multichat.protocol.ConnectionHandler;
|
import ch.zhaw.pm2.multichat.protocol.ConnectionHandler;
|
||||||
|
|
||||||
import static ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State.*;
|
import static ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State.*;
|
||||||
|
|
||||||
import ch.zhaw.pm2.multichat.protocol.NetworkHandler;
|
import ch.zhaw.pm2.multichat.protocol.NetworkHandler;
|
||||||
|
|
||||||
import java.io.EOFException;
|
import java.io.EOFException;
|
||||||
|
@ -12,45 +14,40 @@ import java.util.Map;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Scanner;
|
import java.util.Scanner;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import java.util.concurrent.locks.Condition;
|
|
||||||
import java.util.concurrent.locks.ReentrantLock;
|
import java.util.concurrent.locks.ReentrantLock;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class represents the connection between the server and a client and offers the serverside logic.
|
* This class represents the connection between the server and a client and offers the serverside logic.
|
||||||
* The ServerConnectionHandler receives data send from the client aswell as sends data to the client.
|
* The ServerConnectionHandler receives data send from the client as well as sends data to the client.
|
||||||
*
|
* <p>
|
||||||
* The ServeConnectionHandler offers following functionality:
|
* The ServeConnectionHandler offers following functionality:
|
||||||
*
|
* <p>
|
||||||
* Evaluating connection attempts from a client by:
|
* Evaluating connection attempts from a client by:
|
||||||
* 1. Checks if Username is valid (Not used)
|
* 1. Checks if Username is valid (Not used)
|
||||||
* 2. Saves Username in {@link ServerConnectionHandler#userName}
|
* 2. Saves Username in {@link ServerConnectionHandler#userName}
|
||||||
* 3. Saves the connection in the {@link ServerConnectionHandler#connectionRegistry}
|
* 3. Saves the connection in the {@link ServerConnectionHandler#connectionRegistry}
|
||||||
*
|
* <p>
|
||||||
* Processes disconnections from a client by:
|
* Processes disconnections from a client by:
|
||||||
* 1. Removing the connection from the {@link ServerConnectionHandler#connectionRegistry}
|
* 1. Removing the connection from the {@link ServerConnectionHandler#connectionRegistry}
|
||||||
* 2. Terminates the socket by calling {@link NetworkHandler.NetworkConnection#close()}
|
* 2. Terminates the socket by calling {@link NetworkHandler.NetworkConnection#close()}
|
||||||
*
|
* <p>
|
||||||
* Processes Messages send from a client by:
|
* Process Messages send from a client by:
|
||||||
* 1. Evaluating the reciever by differentiating between broadcast or unicast.
|
* 1. Evaluating the receiver by differentiating between broadcast or unicast.
|
||||||
* 2. Sending the message accordingly.
|
* 2. Sending the message accordingly.
|
||||||
*
|
* <p>
|
||||||
* To use this class, start a new instance and start it in a thread.
|
* To use this class, start a new instance and start it in a thread.
|
||||||
* To constructor needs following parameter:
|
* To constructor needs following parameter:
|
||||||
* 1. {@link ch.zhaw.pm2.multichat.protocol.NetworkHandler.NetworkConnection} representing the socket connection between server and client
|
* 1. {@link ch.zhaw.pm2.multichat.protocol.NetworkHandler.NetworkConnection} representing the socket connection between server and client
|
||||||
* 2. {@link Map<String,ServerConnectionHandler>} registry to check for all active connections
|
* 2. {@link Map<String,ServerConnectionHandler>} registry to check for all active connections
|
||||||
* 3. {@link ReentrantLock @link Condition} to lock server thread to evaluate connection.
|
* 3. {@link ReentrantLock @link Condition} to lock server thread to evaluate connection.
|
||||||
* */
|
*/
|
||||||
public class ServerConnectionHandler extends ConnectionHandler implements Runnable{
|
public class ServerConnectionHandler extends ConnectionHandler implements Runnable {
|
||||||
private static final AtomicInteger connectionCounter = new AtomicInteger(0);
|
private static final AtomicInteger connectionCounter = new AtomicInteger(0);
|
||||||
private final int connectionId = connectionCounter.incrementAndGet();
|
private final int connectionId = connectionCounter.incrementAndGet();
|
||||||
private final Map<String,ServerConnectionHandler> connectionRegistry;
|
private final Map<String, ServerConnectionHandler> connectionRegistry;
|
||||||
|
|
||||||
private ReentrantLock mutex;
|
private String userName = "Anonymous-" + connectionId;
|
||||||
|
|
||||||
private Condition nameComplete;
|
|
||||||
|
|
||||||
private String userName = "Anonymous-"+connectionId;
|
|
||||||
private State state = NEW;
|
private State state = NEW;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -62,65 +59,48 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor to intitialize the connection
|
* Constructor to initialize the connection
|
||||||
* @param connection representing the socket connection between server and clinet
|
*
|
||||||
* @param registry map containing all active connections between server and clients
|
* @param connection representing the socket connection between server and client
|
||||||
* @param mutex to lock thread
|
* @param registry map containing all active connections between server and clients
|
||||||
* @param nameComplete condition to call threads
|
|
||||||
|
|
||||||
*/
|
*/
|
||||||
public ServerConnectionHandler(NetworkHandler.NetworkConnection<String> connection,
|
public ServerConnectionHandler(NetworkHandler.NetworkConnection<String> connection,
|
||||||
Map<String,ServerConnectionHandler> registry, ReentrantLock mutex, Condition nameComplete) {
|
Map<String, ServerConnectionHandler> registry) {
|
||||||
super();
|
super();
|
||||||
setConnection(connection);
|
setConnection(connection);
|
||||||
Objects.requireNonNull(connection, "Connection must not be null");
|
Objects.requireNonNull(connection, "Connection must not be null");
|
||||||
Objects.requireNonNull(registry, "Registry must not be null");
|
Objects.requireNonNull(registry, "Registry must not be null");
|
||||||
this.connectionRegistry = registry;
|
this.connectionRegistry = registry;
|
||||||
this.mutex = mutex;
|
|
||||||
this.nameComplete = nameComplete;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
* These methods runs in a while-loop as long as the socket between server and client is available
|
||||||
* @return the username of the connected client
|
|
||||||
*/
|
|
||||||
public String getUserName() {
|
|
||||||
return this.userName;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @return state of the connection. Poosible states are see {@link ch.zhaw.pm2.multichat.protocol.ConnectionHandler.State}
|
|
||||||
*/
|
|
||||||
public State getState() {
|
|
||||||
return state;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This methods runs in a whileloop as long as the socket between server and client is available
|
|
||||||
* and the connection State is not ERROR.
|
* and the connection State is not ERROR.
|
||||||
*/
|
*/
|
||||||
private void startReceiving() {
|
private void startReceiving() {
|
||||||
System.out.println("Starting Connection Handler for new User");
|
System.out.println("Starting Connection Handler for new User");
|
||||||
try {
|
try {
|
||||||
System.out.println("Start receiving data...");
|
System.out.println("Start receiving data...");
|
||||||
while (getConnection().isAvailable() && !(state == ERROR)) {
|
while (getConnection().isAvailable() && !(state == ERROR)) {
|
||||||
String data = getConnection().receive();
|
String data = getConnection().receive();
|
||||||
processData(data);
|
processData(data);
|
||||||
}
|
|
||||||
System.out.println("Stopped recieving data");
|
|
||||||
} catch (SocketException e) {
|
|
||||||
System.out.println("Connection terminated locally");
|
|
||||||
connectionRegistry.remove(userName);
|
|
||||||
System.out.println("Unregistered because client connection terminated: " + userName + " " + e.getMessage());
|
|
||||||
} catch (EOFException e) {
|
|
||||||
System.out.println("Connection terminated by remote");
|
|
||||||
connectionRegistry.remove(userName);
|
|
||||||
System.out.println("Unregistered because client connection terminated: " + userName + " " + e.getMessage());
|
|
||||||
} catch (IOException e) {
|
|
||||||
System.err.println("Communication error: " + e);
|
|
||||||
} catch (ClassNotFoundException e) {
|
|
||||||
System.err.println("Received object of unknown type: " + e.getMessage());
|
|
||||||
}
|
}
|
||||||
|
System.out.println("Stopped receiving data");
|
||||||
|
} catch (SocketException e) {
|
||||||
|
System.out.println("Connection terminated locally");
|
||||||
|
connectionRegistry.remove(userName);
|
||||||
|
System.out.println("Unregistered because client connection terminated: " + userName + " " + e.getMessage());
|
||||||
|
} catch (EOFException e) {
|
||||||
|
System.out.println("Connection terminated by remote");
|
||||||
|
connectionRegistry.remove(userName);
|
||||||
|
System.out.println("Unregistered because client connection terminated: " + userName + " " + e.getMessage());
|
||||||
|
} catch (IOException e) {
|
||||||
|
System.err.println("Communication error: " + e);
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
System.err.println("Received object of unknown type: " + e.getMessage());
|
||||||
|
}
|
||||||
if (state == ERROR) {
|
if (state == ERROR) {
|
||||||
System.out.println("Stopping Connection Handler for Rejected Client");
|
System.out.println("Stopping Connection Handler for Rejected Client");
|
||||||
} else {
|
} else {
|
||||||
|
@ -145,24 +125,25 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method gets called when socket recieves data. The method checks for the data type and reacts accordingly
|
* This method gets called when socket receives data. The method checks for the data type and reacts accordingly
|
||||||
* If data type:
|
* If data type:
|
||||||
* 1. Connect => checks if username is valid. if valid sends response to client with confirmation.
|
* 1. Connect => checks if username is valid. if valid sends response to client with confirmation.
|
||||||
* If username not valid quits connection by changing status to ERROR.
|
* If username not valid quits connection by changing status to ERROR.
|
||||||
* 2. Confirm => Server should not recieve this kind of message. STDOUT informs about it.
|
* 2. Confirm => Server should not receive this kind of message. STDOUT informs about it.
|
||||||
* 3. Disconnect => Disconnects connection by removing connection from registry and calling method to terminate socket.
|
* 3. Disconnect => Disconnects connection by removing connection from registry and calling method to terminate socket.
|
||||||
* 4. Message => Checks if broadcast or unicast. Sends data accordingly
|
* 4. Message => Checks if broadcast or unicast. Sends data accordingly
|
||||||
* 5. Error => STDERR message
|
* 5. Error => STDERR message
|
||||||
* @param data recieved by the server
|
*
|
||||||
|
* @param data received by the server
|
||||||
*/
|
*/
|
||||||
private void processData(String data) {
|
private void processData(String data) {
|
||||||
try {
|
try {
|
||||||
Scanner scanner = new Scanner(data);
|
Scanner scanner = new Scanner(data);
|
||||||
StringBuilder sender = new StringBuilder();
|
StringBuilder sender = new StringBuilder();
|
||||||
StringBuilder reciever = new StringBuilder();
|
StringBuilder receiver = new StringBuilder();
|
||||||
StringBuilder type = new StringBuilder();
|
StringBuilder type = new StringBuilder();
|
||||||
StringBuilder payload = new StringBuilder();
|
StringBuilder payload = new StringBuilder();
|
||||||
super.processData(scanner,sender,reciever,type,payload);
|
super.processData(scanner, sender, receiver, type, payload);
|
||||||
|
|
||||||
// dispatch operation based on type parameter
|
// dispatch operation based on type parameter
|
||||||
if (type.toString().equals(getDataTypeConnect())) {
|
if (type.toString().equals(getDataTypeConnect())) {
|
||||||
|
@ -172,14 +153,14 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
} else if (type.toString().equals(getDataTypeDisconnect())) {
|
} else if (type.toString().equals(getDataTypeDisconnect())) {
|
||||||
caseDisconnect();
|
caseDisconnect();
|
||||||
} else if (type.toString().equals(getDataTypeMessage())) {
|
} else if (type.toString().equals(getDataTypeMessage())) {
|
||||||
caseMessage(sender.toString(), reciever.toString(), type.toString(), payload.toString());
|
caseMessage(sender.toString(), receiver.toString(), type.toString(), payload.toString());
|
||||||
} else if (type.toString().equals(getDataTypeError())) {
|
} else if (type.toString().equals(getDataTypeError())) {
|
||||||
System.err.println("Received error from client (" + sender + "): " + payload);
|
System.err.println("Received error from client (" + sender + "): " + payload);
|
||||||
} else {
|
} else {
|
||||||
System.err.println("Unknown data type received: " + type);
|
System.err.println("Unknown data type received: " + type);
|
||||||
|
|
||||||
}
|
}
|
||||||
} catch(ChatProtocolException e) {
|
} catch (ChatProtocolException e) {
|
||||||
System.out.println("Error while processing data " + e.getMessage());
|
System.out.println("Error while processing data " + e.getMessage());
|
||||||
sendData(USER_NONE, userName, getDataTypeError(), e.getMessage());
|
sendData(USER_NONE, userName, getDataTypeError(), e.getMessage());
|
||||||
}
|
}
|
||||||
|
@ -188,6 +169,7 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
/**
|
/**
|
||||||
* This method is called by method {@link ServerConnectionHandler#processData(String)}
|
* This method is called by method {@link ServerConnectionHandler#processData(String)}
|
||||||
* Checks if username is valid. if valid sends response to client with confirmation.
|
* Checks if username is valid. if valid sends response to client with confirmation.
|
||||||
|
*
|
||||||
* @param sender of the payload
|
* @param sender of the payload
|
||||||
* @throws ChatProtocolException if username not valid
|
* @throws ChatProtocolException if username not valid
|
||||||
*/
|
*/
|
||||||
|
@ -205,7 +187,7 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
//if username valid
|
//if username valid
|
||||||
this.userName = sender;
|
this.userName = sender;
|
||||||
connectionRegistry.put(userName, this);
|
connectionRegistry.put(userName, this);
|
||||||
sendData(USER_NONE, userName, getDataTypeConfirm(), "Registration successfull for " + userName);
|
sendData(USER_NONE, userName, getDataTypeConfirm(), "Registration successful for " + userName);
|
||||||
this.state = CONNECTED;
|
this.state = CONNECTED;
|
||||||
System.out.println(String.format("Connected new Client %s with IP:Port <%s:%d>",
|
System.out.println(String.format("Connected new Client %s with IP:Port <%s:%d>",
|
||||||
userName,
|
userName,
|
||||||
|
@ -216,7 +198,8 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
/**
|
/**
|
||||||
* This method is called by method {@link ServerConnectionHandler#processData(String)}
|
* This method is called by method {@link ServerConnectionHandler#processData(String)}
|
||||||
* Disconnects connection by removing connection from registry and calling method {@link ServerConnectionHandler#stopReceiving()} to terminate socket.
|
* Disconnects connection by removing connection from registry and calling method {@link ServerConnectionHandler#stopReceiving()} to terminate socket.
|
||||||
* @throws ChatProtocolException if state allready DISCONNECTED.
|
*
|
||||||
|
* @throws ChatProtocolException if state already DISCONNECTED.
|
||||||
*/
|
*/
|
||||||
private void caseDisconnect() throws ChatProtocolException {
|
private void caseDisconnect() throws ChatProtocolException {
|
||||||
if (state == DISCONNECTED)
|
if (state == DISCONNECTED)
|
||||||
|
@ -232,27 +215,28 @@ public class ServerConnectionHandler extends ConnectionHandler implements Runnab
|
||||||
/**
|
/**
|
||||||
* This method is called by method {@link ServerConnectionHandler#processData(String)}
|
* This method is called by method {@link ServerConnectionHandler#processData(String)}
|
||||||
* Checks if broadcast or unicast. Sends data accordingly
|
* Checks if broadcast or unicast. Sends data accordingly
|
||||||
* @param sender who sent data
|
*
|
||||||
* @param reciever to recieve data
|
* @param sender who sent data
|
||||||
* @param type of message
|
* @param receiver to receive data
|
||||||
* @param payload data to transmit
|
* @param type of message
|
||||||
|
* @param payload data to transmit
|
||||||
* @throws ChatProtocolException if state not equal to CONNECT
|
* @throws ChatProtocolException if state not equal to CONNECT
|
||||||
*/
|
*/
|
||||||
private void caseMessage(String sender, String reciever, String type, String payload) throws ChatProtocolException{
|
private void caseMessage(String sender, String receiver, String type, String payload) throws ChatProtocolException {
|
||||||
if (state != CONNECTED) throw new ChatProtocolException("Illegal state for message request: " + state);
|
if (state != CONNECTED) throw new ChatProtocolException("Illegal state for message request: " + state);
|
||||||
if (USER_ALL.equals(reciever)) {
|
if (USER_ALL.equals(receiver)) {
|
||||||
for (ServerConnectionHandler handler : connectionRegistry.values()) {
|
for (ServerConnectionHandler handler : connectionRegistry.values()) {
|
||||||
handler.sendData(sender, reciever, type, payload);
|
handler.sendData(sender, receiver, type, payload);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ServerConnectionHandler handler = connectionRegistry.get(reciever);
|
ServerConnectionHandler handler = connectionRegistry.get(receiver);
|
||||||
if (handler != null) {
|
if (handler != null) {
|
||||||
handler.sendData(sender, reciever, type, payload);
|
handler.sendData(sender, receiver, type, payload);
|
||||||
if(!reciever.equals(sender)){
|
if (!receiver.equals(sender)) {
|
||||||
sendData(sender, reciever, type, payload); //send message to sender if it's a direct message and sender is not receiver.
|
sendData(sender, receiver, type, payload); //send message to sender if it's a direct message and sender is not receiver.
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.sendData(USER_NONE, userName, getDataTypeError(), "Unknown User: " + reciever);
|
this.sendData(USER_NONE, userName, getDataTypeError(), "Unknown User: " + receiver);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue