Vous êtes sur la page 1sur 23

Application de

chat cryptée
par l’ AES

24 JANVIER 2021

GSTR 3
Réalisé par : AZGAOUI Hamza DARIF Rania

1
SOMMAIRE

INTRODUCTION

CRYPTAGE AES

DEVELOPPEMENT DE L’APPLICATION

TEST DE L’APPLICATION

2
Introduction:

Dans le cadre du module système embarqué 2, enseigné par Mr. Samir Adib,
nous avons été chargés de créer une application de chat cryptée avec l’AES
(Advanced Encryption Standard), nous avions alors développé deux applications, la
première était une application desktop programmée entièrement à l’aide du
langage python ; cette application est basée sur l’échange des messages texts
cryptés avec l’AES, ainsi que des fichiers et peu importe leurs formats, en outre
cette application fait appel à deux interfaces graphiques une pour le client et une
pour le server. La deuxième application est une application mobile développée en
Android Studio à l’aide du langage JAVA, permettant aussi l’échange des messages
texts cryptés avec l’AES, fonctionnant au niveau des équipements Android.

Le cryptage AES est une norme ou spécification de chiffrement comme RSA, DES, ...

On utilise l’AES, par exemple, pour chiffrer un fichier ou disque dur ou le stockage
de mot de passe comme les gestionnaires de mots de passe, la protection par mot
de passe de fichiers PDF ou de fichiers ZIP.
En effet, les applications telles que WhatsApp, Signal, VeraCrypt ou 7-zip et WinZip
utilisent AES pour chiffrer (crypter) les communications ou le contenu.
En clair donc, AES est très utilise pour protéger ses données.
Nous avons comme objectif principal d’implémenter le cryptage AES au
niveau de notre application de chat afin d’assurer le cryptage et le décryptage
des messages texts envoyés/reçus par deux clients.

3
Cryptage AES:

L'algorithme prend en entrée un bloc de 128 bits (16 octets), la clé fait 128, 192 ou
256 bits. Les 16 octets en entrée sont permutés selon une table définie au
préalable. Ces octets sont ensuite placés dans une matrice de 4x4 éléments et ses
lignes subissent une rotation vers la droite. L'incrément pour la rotation varie
selon le numéro de la ligne. Une transformation linéaire est ensuite appliquée sur
la matrice, elle consiste en la multiplication binaire de chaque élément de la
matrice avec des polynô mes issus d'une matrice auxiliaire, cette multiplication est
soumise à des règles spéciales selon GF(28) (groupe de Galois ou corps fini). La
transformation linéaire garantit une meilleure diffusion (propagation des bits dans
la structure) sur plusieurs tours.

Finalement, un OU exclusif XOR entre la matrice et une autre matrice permet


d'obtenir une matrice intermédiaire. Ces différentes opérations sont répétées
plusieurs fois et définissent un « tour ». Pour une clé de 128, 192 ou 256, AES
nécessite respectivement 10, 12 ou 14 tours.

4
Développement de l’application :

Une application de chat fournit la communication entre deux parties, un expéditeur


et un récepteur. L’expéditeur est celui qui initie et envoie un message à d’autres
sous le nom du récepteur ; Le récepteur à l’autre extrémité reçoit le message. Le
rô le de l’expéditeur et du récepteur n’est pas fixé, ils continuent à l’échanger
pendant la communication, donc en termes simples, à un moment donné,
quelqu’un qui envoie le message est un expéditeur et qui reçoive le message est
appelé récepteur. En d’autres termes, l’expéditeur et le récepteur sont indiqués
comme source et destination respectivement.
La communication peut être de nombreux types selon la méthode de
communication et le nombre de parties impliquées. Voici quelques scénarios :

• La communication simple ou à sens unique : une seule partie est en mesure


d’envoyer le message et les autres parties ne peuvent que recevoir.

• La communication en duplex ou dans les deux sens : Les deux parties


peuvent envoyer et recevoir des messages.

5
 Code du fichier AES.java (SERVEUR)

6
 Code du fichier AES.java
(MESSENGER)

7
 Le code Script Serveur

public class ServerFrame extends javax.swing.JFrame {


class ServerThread extends Thread {

public SocketServer server = null;


public Socket socket = null;
public int ID = -1;
public String username = "";
public ObjectInputStream streamIn = null;
public ObjectOutputStream streamOut = null;
public ServerFrame ui;

public ServerThread(SocketServer _server, Socket _socket){


super();
server = _server;
socket = _socket;
ID = socket.getPort();
ui = _server.ui;
}

public void send(Message msg){


try {
8
streamOut.writeObject(msg);
streamOut.flush();
}
catch (IOException ex) {
System.out.println("Exception [SocketClient : send(...)]");
}
}

public int getID(){


return ID;
}

@SuppressWarnings("deprecation")
public void run(){
SAES rsa=new SAES();
ui.jTextArea1.append("\nServer Thread " + ID + " running.");
while (true){

try{
Message msg = (Message) streamIn.readObject();
server.handle(ID, msg,rsa);
}
catch(Exception ioe){
System.out.println(ID + " ERROR reading: " + ioe.getMessage());
server.remove(ID);
stop();
}
}
}

public void open() throws IOException {


streamOut = new ObjectOutputStream(socket.getOutputStream());
streamOut.flush();
streamIn = new ObjectInputStream(socket.getInputStream());
}

public void close() throws IOException {


if (socket != null) socket.close();
if (streamIn != null) streamIn.close();
if (streamOut != null) streamOut.close();
}
}

public class SocketServer implements Runnable {

public ServerThread clients[];


public ServerSocket server = null;
public Thread thread = null;
public int clientCount = 0, port = 13000;
public ServerFrame ui;

9
public Database db;

public SocketServer(ServerFrame frame){

clients = new ServerThread[50];


ui = frame;
db = new Database(ui.filePath);

try{
server = new ServerSocket(port);
port = server.getLocalPort();
ui.jTextArea1.append("Server startet. IP : " + InetAddress.getLocalHost() + ",
Port : " + server.getLocalPort());
start();
}
catch(IOException ioe){
ui.jTextArea1.append("Can not bind to port : " + port + "\nRetrying");
ui.RetryStart(0);
}
}

public SocketServer(ServerFrame frame, int Port){

clients = new ServerThread[50];


ui = frame;
port = Port;
db = new Database(ui.filePath);

try{
server = new ServerSocket(port);
port = server.getLocalPort();
ui.jTextArea1.append("Server startet. IP : " + InetAddress.getLocalHost() + ",
Port : " + server.getLocalPort());
start();
}
catch(IOException ioe){
ui.jTextArea1.append("\nCan not bind to port " + port + ": " +
ioe.getMessage());
}
}

public void run(){


while (thread != null){
try{
ui.jTextArea1.append("\nWaiting for a client ...");
addThread(server.accept());
}
catch(Exception ioe){
ui.jTextArea1.append("\nServer accept error: \n");
ui.RetryStart(0);
}
}
}

10
public void start(){
if (thread == null){
thread = new Thread(this);
thread.start();
}
}

@SuppressWarnings("deprecation")
public void stop(){
if (thread != null){
thread.stop();
thread = null;
}
}

private int findClient(int ID){


for (int i = 0; i < clientCount; i++){
if (clients[i].getID() == ID){
return i;
}
}
return -1;
}

public synchronized void handle(int ID, Message msg,SAES rsa){


if(msg.content==null){
msg.content="";
}
if (msg.content.equals(".bye")){
Announce("signout", "SERVER", msg.sender);
remove(ID);
}
else{
if(msg.type.equals("RSAdemend")){

clients[findClient(ID)].send(new Message("rsacles", rsa.Gete(),


rsa.GetN(),msg.sender,msg.content, msg.recipient));
}
else if(msg.type.equals("login")){
if(findUserThread(msg.sender) == null){
if(db.checkLogin(msg.sender, msg.content)){
clients[findClient(ID)].username = msg.sender;
clients[findClient(ID)].send(new Message("login", "SERVER",
"TRUE", msg.sender));
Announce("newuser", "SERVER", msg.sender);
SendUserList(msg.sender);
}
else{
clients[findClient(ID)].send(new Message("login", "SERVER",
"FALSE", msg.sender));
}
}
else{
clients[findClient(ID)].send(new Message("login", "SERVER", "FALSE",
msg.sender));

11
}
}
else if(msg.type.equals("message")){
byte[] decrypte=rsa.decrypt(msg.chiffre);

msg.content=new String(new String(decrypte));


System.out.println("mouad "+msg.content);
if(msg.recipient.equals("All")){
Announce("message", msg.sender, msg.content);
}
else{
findUserThread(msg.recipient).send(new Message(msg.type, msg.sender,
msg.content, msg.recipient));
clients[findClient(ID)].send(new Message(msg.type, msg.sender,
msg.content, msg.recipient));
}
}
else if(msg.type.equals("test")){
clients[findClient(ID)].send(new Message("test", "SERVER", "OK",
msg.sender));
}
else if(msg.type.equals("signup")){
if(findUserThread(msg.sender) == null){
if(!db.userExists(msg.sender)){
db.addUser(msg.sender, msg.content);
clients[findClient(ID)].username = msg.sender;
clients[findClient(ID)].send(new Message("signup", "SERVER",
"TRUE", msg.sender));
clients[findClient(ID)].send(new Message("login", "SERVER",
"TRUE", msg.sender));
Announce("newuser", "SERVER", msg.sender);
SendUserList(msg.sender);
}
else{
clients[findClient(ID)].send(new Message("signup", "SERVER",
"FALSE", msg.sender));
}
}
else{
clients[findClient(ID)].send(new Message("signup", "SERVER", "FALSE",
msg.sender));
}
}
else if(msg.type.equals("upload_req")){
if(msg.recipient.equals("All")){
clients[findClient(ID)].send(new Message("message", "SERVER",
"Uploading to 'All' forbidden", msg.sender));
}
else{
findUserThread(msg.recipient).send(new Message("upload_req",
msg.sender, msg.content, msg.recipient));
}
}
else if(msg.type.equals("upload_res")){
if(!msg.content.equals("NO")){

12
String IP =
findUserThread(msg.sender).socket.getInetAddress().getHostAddress();
findUserThread(msg.recipient).send(new Message("upload_res", IP,
msg.content, msg.recipient));
}
else{
findUserThread(msg.recipient).send(new Message("upload_res",
msg.sender, msg.content, msg.recipient));
}
}
}
}
public void Announce(String type, String sender, String content){
Message msg = new Message(type, sender, content, "All");
for(int i = 0; i < clientCount; i++){
clients[i].send(msg);
}
}

public void SendUserList(String toWhom){


for(int i = 0; i < clientCount; i++){
findUserThread(toWhom).send(new Message("newuser", "SERVER",
clients[i].username, toWhom));
}
}

public ServerThread findUserThread(String usr){


for(int i = 0; i < clientCount; i++){
if(clients[i].username.equals(usr)){
return clients[i];
}
}
return null;
}

@SuppressWarnings("deprecation")
public synchronized void remove(int ID){
int pos = findClient(ID);
if (pos >= 0){
ServerThread toTerminate = clients[pos];
ui.jTextArea1.append("\nRemoving client thread " + ID + " at " + pos);
if (pos < clientCount-1){
for (int i = pos+1; i < clientCount; i++){
clients[i-1] = clients[i];
}
}
clientCount--;
try{
toTerminate.close();
}
catch(IOException ioe){
ui.jTextArea1.append("\nError closing thread: " + ioe);
}
toTerminate.stop();
}

13
}

private void addThread(Socket socket){


if (clientCount < clients.length){
ui.jTextArea1.append("\nClient accepted: " + socket);
clients[clientCount] = new ServerThread(this, socket);
try{
clients[clientCount].open();
clients[clientCount].start();
clientCount++;
}
catch(IOException ioe){
ui.jTextArea1.append("\nError opening thread: " + ioe);
}
}
else{
ui.jTextArea1.append("\nClient refused: maximum " + clients.length + "
reached.");
}
}
}

 Le code Script Client

public class SocketClient implements Runnable{


AES rsa;
public int port;
public String serverAddr;
public Socket socket;
public ChatFrame ui;
public ObjectInputStream In;
public ObjectOutputStream Out;
public History hist;

public SocketClient(ChatFrame frame) throws IOException{


ui = frame;
this.serverAddr = "localhost";
this.port = 13000;
socket = new Socket(InetAddress.getByName(serverAddr), port);

Out = new ObjectOutputStream(socket.getOutputStream());


Out.flush();
In = new ObjectInputStream(socket.getInputStream());

hist = ui.hist;
}

14
@Override
public void run()
{
boolean keepRunning = true;
while(keepRunning){
// try {
Message msg = null;
try {
msg = (Message) In.readObject();
System.out.println("brahim ");
} catch (IOException ex) {
Logger.getLogger(SocketClient.class.getName()).log(Level.SEVERE, null,
ex);
} catch (ClassNotFoundException ex) {
Logger.getLogger(SocketClient.class.getName()).log(Level.SEVERE, null,
ex);
}
System.out.println("Incoming : "+msg.toString());

if(msg.type.equals("rsacles")){

rsa=new AES(msg.e,new BigInteger("0"),msg.N);


byte crypt[]=rsa.encrypt(msg.content.getBytes());

send(new Message("message",msg.sender,crypt,msg.recipient));
}
else if(msg.type.equals("message")){
if(msg.recipient.equals(ui.username)){
ui.jTextArea1.append("["+msg.sender +" > Me] : " + msg.content +
"\n");
}
else{
ui.jTextArea1.append("["+ msg.sender +" > "+ msg.recipient +"] : "
+ msg.content + "\n");
}

if(!msg.content.equals(".bye") && !msg.sender.equals(ui.username)){


String msgTime = (new Date()).toString();

try{
hist.addMessage(msg, msgTime);
DefaultTableModel table = (DefaultTableModel)
ui.historyFrame.jTable1.getModel();
table.addRow(new Object[]{msg.sender, msg.content, "Me",
msgTime});
}
catch(Exception ex){}
}
}
else if(msg.type.equals("login")){
if(msg.content.equals("TRUE")){
ui.jButton2.setEnabled(false);
ui.jButton4.setEnabled(true); ui.jButton5.setEnabled(true);
ui.jTextArea1.append("[SERVER > Me] : Login Successful\n");

15
ui.jTextField3.setEnabled(false);
ui.jPasswordField1.setEnabled(false);
}
else{
ui.jTextArea1.append("[SERVER > Me] : Login Failed\n");
}
}
else if(msg.type.equals("test")){
ui.jButton1.setEnabled(false);
ui.jButton2.setEnabled(true);
ui.jTextField3.setEnabled(true); ui.jPasswordField1.setEnabled(true);

}
else if(msg.type.equals("newuser")){
if(!msg.content.equals(ui.username)){
boolean exists = false;
for(int i = 0; i < ui.model.getSize(); i++){
if(ui.model.getElementAt(i).equals(msg.content)){
exists = true; break;
}
}
if(!exists){ ui.model.addElement(msg.content); }
}
}
else if(msg.type.equals("signup")){
if(msg.content.equals("TRUE")){
ui.jButton2.setEnabled(false);
ui.jButton4.setEnabled(true); ui.jButton5.setEnabled(true);
ui.jTextArea1.append("[SERVER > Me] : Singup Successful\n");
}
else{
ui.jTextArea1.append("[SERVER > Me] : Signup Failed\n");
}
}
else if(msg.type.equals("signout")){
if(msg.content.equals(ui.username)){
ui.jTextArea1.append("["+ msg.sender +" > Me] : Bye\n");
ui.jButton1.setEnabled(true); ui.jButton4.setEnabled(false);

for(int i = 1; i < ui.model.size(); i++){


ui.model.removeElementAt(i);
}

ui.clientThread.stop();
}
else{
ui.model.removeElement(msg.content);
ui.jTextArea1.append("["+ msg.sender +" > All] : "+ msg.content +"
has signed out\n");
}
}
else if(msg.type.equals("upload_req")){

16
if(JOptionPane.showConfirmDialog(ui, ("Accept '"+msg.content+"' from
"+msg.sender+" ?")) == 0){

JFileChooser jf = new JFileChooser();


jf.setSelectedFile(new File(msg.content));
int returnVal = jf.showSaveDialog(ui);

String saveTo = jf.getSelectedFile().getPath();


if(saveTo != null && returnVal == JFileChooser.APPROVE_OPTION){
Download dwn = new Download(saveTo, ui);
Thread t = new Thread(dwn);
t.start();
//send(new Message("upload_res",
(""+InetAddress.getLocalHost().getHostAddress()), (""+dwn.port), msg.sender));
send(new Message("upload_res", ui.username, (""+dwn.port),
msg.sender));
}
else{
send(new Message("upload_res", ui.username, "NO",
msg.sender));
}
}
else{
send(new Message("upload_res", ui.username, "NO", msg.sender));
}
}
else if(msg.type.equals("upload_res")){
if(!msg.content.equals("NO")){
int port = Integer.parseInt(msg.content);
String addr = msg.sender;

ui.jButton5.setEnabled(false); ui.jButton6.setEnabled(false);
Upload upl = new Upload(addr, port, ui.file, ui);
Thread t = new Thread(upl);
t.start();
}
else{
ui.jTextArea1.append("[SERVER > Me] : "+msg.sender+" rejected file
request\n");
}
}
else{
ui.jTextArea1.append("[SERVER > Me] : Unknown message type\n");
}
/*}
catch(Exception ex) {
keepRunning = false;
ui.jTextArea1.append("[Application > Me] : Connection Failure\n");

ui.jButton4.setEnabled(false); ui.jButton5.setEnabled(false);
ui.jButton5.setEnabled(false);

for(int i = 1; i < ui.model.size(); i++){

17
ui.model.removeElementAt(i);
}

ui.clientThread.stop();

System.out.println("Exception SocketClient run()");


ex.printStackTrace();
}*/
}
}

public void send(Message msg){


try {
Out.writeObject(msg);
Out.flush();
System.out.println("Outgoing : "+msg.toString());

}
catch (IOException ex) {
System.out.println("Exception SocketClient send()");
}
}

public void closeThread(Thread t){


t = null;
}
}

18
19
Test de l’application :

 Connexion Serveur

 Inscription Utilisateur

20
 Connexion Utilisateur

21
 Messagerie

22
23

Vous aimerez peut-être aussi