2021-02-08 11:32:58 +01:00
|
|
|
import { Component, OnInit } from '@angular/core';
|
2021-02-10 11:23:09 +01:00
|
|
|
import CipherLib from './lib_cipher';
|
2021-02-09 18:26:13 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
2021-02-09 18:35:29 +01:00
|
|
|
|
2021-02-09 18:26:13 +01:00
|
|
|
// documentation:
|
|
|
|
// https://medium.com/@spatocode/symmetric-encryption-in-javascript-bcb5fd14c273
|
|
|
|
// https://nodejs.org/api/crypto.html
|
2021-02-10 10:35:36 +01:00
|
|
|
// https://preview.npmjs.com/package/ecma-nacl
|
2021-02-08 11:32:58 +01:00
|
|
|
@Component({
|
|
|
|
selector: 'app-ciphering',
|
|
|
|
templateUrl: './ciphering.component.html',
|
|
|
|
styleUrls: ['./ciphering.component.scss'],
|
|
|
|
})
|
|
|
|
export class CipheringComponent implements OnInit {
|
2021-02-12 12:06:30 +01:00
|
|
|
public plainText = 'coucou !';
|
2021-02-10 10:35:36 +01:00
|
|
|
// public plainText = '1234';
|
2021-02-09 22:47:57 +01:00
|
|
|
public cipheredText =
|
|
|
|
'121-127-42-110-127-42-122-121-115-128-124-111-125-107-118-127-126-42-118-111-125-42-122-115-122-121-118-111';
|
2021-02-12 12:06:30 +01:00
|
|
|
public algorithm = 'aes-192-cbc';
|
2021-02-10 10:35:36 +01:00
|
|
|
public salt = 'ou du poivre heh';
|
2021-02-09 22:47:57 +01:00
|
|
|
public initial_vector = '';
|
2021-02-09 18:35:29 +01:00
|
|
|
public key;
|
2021-02-09 22:47:57 +01:00
|
|
|
public otherCipheredText = 'le texte à chiffrer, coucou !';
|
2021-02-12 12:06:30 +01:00
|
|
|
public alphab: string;
|
|
|
|
public prem: number;
|
|
|
|
public dico: string[];
|
|
|
|
public separatedCouples: any[];
|
2021-02-12 14:36:10 +01:00
|
|
|
public codingMatrice = '2 3 5 8';
|
2021-02-12 12:06:30 +01:00
|
|
|
private cipheredTextCIL: string;
|
2021-02-08 11:32:58 +01:00
|
|
|
|
2021-02-12 12:06:30 +01:00
|
|
|
constructor() {
|
|
|
|
this.alphab = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789 .+-*/_!?,éèêëàçôî'âù()@&$";
|
|
|
|
this.prem = this.alphab.length;
|
|
|
|
this.dico = this.alphab.split('');
|
|
|
|
}
|
2021-02-08 11:32:58 +01:00
|
|
|
|
|
|
|
ngOnInit(): void {
|
2021-02-12 12:06:30 +01:00
|
|
|
this.encrypt(this.codingMatrice, this.plainText);
|
2021-02-08 11:32:58 +01:00
|
|
|
}
|
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
// ---------------------- make unreadable ----------------------
|
|
|
|
get simpleCipheredText() {
|
|
|
|
return this.convertTextToInt(this.plainText);
|
|
|
|
}
|
2021-02-10 10:35:36 +01:00
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
get simpleCipheredTextWithSalt() {
|
|
|
|
return this.convertTextToInt(this.salt + this.plainText);
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 10:35:36 +01:00
|
|
|
get simpleCipheredTextWithSpreadSalt() {
|
|
|
|
return this.convertTextToIntMixedSalt(this.salt + this.plainText);
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 11:23:09 +01:00
|
|
|
get simpleCipheredTextWithHashedSalt() {
|
|
|
|
return this.convertTextToIntMixedSalt(this.md5(this.salt) + this.plainText);
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 12:15:39 +01:00
|
|
|
get simpleCipheredTextWithHashedSpreadSalt() {
|
|
|
|
return this.convertTextToIntMixedSalt(this.md5(this.salt) + this.plainText);
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 10:35:36 +01:00
|
|
|
get layerOnSpread() {
|
2021-02-10 11:23:09 +01:00
|
|
|
return this.convertIntToText(this.simpleCipheredTextWithHashedSalt, false);
|
2021-02-10 10:35:36 +01:00
|
|
|
}
|
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
// ---------------------- make readable ----------------------
|
|
|
|
get simpleDeCipheredText() {
|
|
|
|
return this.convertIntToText(this.cipheredText);
|
|
|
|
}
|
2021-02-10 10:35:36 +01:00
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
get simpleDeCipheredTextWithSalt() {
|
|
|
|
return this.convertIntToText(this.salt + this.cipheredText);
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 10:35:36 +01:00
|
|
|
get simpleDeCipheredTextWithSpreadSalt() {
|
|
|
|
return this.convertIntToTextMixedSalt(this.salt + this.cipheredText);
|
|
|
|
}
|
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
// ---------------------- conversions ----------------------
|
2021-02-12 12:06:30 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* chiffrer avec la méthode du CIL-gometz
|
|
|
|
* avec une matrice
|
|
|
|
* @param someText
|
|
|
|
*/
|
|
|
|
encrypt(matriceDefinition, messageToCipher) {
|
|
|
|
if (!matriceDefinition || !messageToCipher) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
let cipheredText = '';
|
|
|
|
const lcpl = this.separecpl(this.filterCharacters(messageToCipher));
|
|
|
|
this.separatedCouples = lcpl;
|
|
|
|
console.log('lcpl', lcpl);
|
|
|
|
console.log('this.dico', this.dico);
|
|
|
|
/**
|
|
|
|
* Pour chaque couple de caractères
|
|
|
|
* on crée le couple d'indices de ces caractères
|
|
|
|
* On applique la matrice au couple d'indices
|
|
|
|
* et on en déduit le couple de caractères codés (ou décodés)
|
|
|
|
* qu'on accumule dans la chaine msgout
|
|
|
|
*/
|
|
|
|
lcpl.forEach((couple) => {
|
|
|
|
console.log('couple', couple);
|
|
|
|
const cplnum = [this.dico.indexOf(couple[0]), this.dico.indexOf(couple[1])];
|
|
|
|
console.log('cplnum', cplnum);
|
|
|
|
const cplnumout = this.applique(matriceDefinition, cplnum);
|
|
|
|
console.log('cplnumout', cplnumout);
|
|
|
|
const cplout = [this.alphab[cplnumout[0]], this.alphab[cplnumout[1]]];
|
|
|
|
console.log('cplout', cplout);
|
|
|
|
cipheredText += cplout[0];
|
|
|
|
cipheredText += cplout[1];
|
|
|
|
});
|
|
|
|
console.log('cipheredText', cipheredText);
|
|
|
|
this.cipheredTextCIL = cipheredText;
|
|
|
|
return cipheredText;
|
2021-02-08 11:32:58 +01:00
|
|
|
}
|
2021-02-09 18:26:13 +01:00
|
|
|
|
2021-02-12 12:06:30 +01:00
|
|
|
/**
|
|
|
|
* séparation d'une chaine de caractères en liste de couples de caractères
|
|
|
|
* @param msg
|
|
|
|
*/
|
|
|
|
separecpl(msg) {
|
|
|
|
/**
|
|
|
|
* Produit une liste formée des couples de caractères de la chaine
|
|
|
|
msg reçue en argument.
|
|
|
|
Dans le cas où le nb de caractères de la chaine est impair,
|
|
|
|
un underscore est ajouté en fin de chaine de façon à pouvoir
|
|
|
|
inclure le dernier caractère dans un couple.
|
|
|
|
*/
|
|
|
|
const nbcar = msg.length;
|
|
|
|
if (nbcar % 2) {
|
|
|
|
msg += '_';
|
|
|
|
}
|
|
|
|
const coupleList = [];
|
|
|
|
let i = 0;
|
|
|
|
while (i < nbcar) {
|
|
|
|
coupleList.push([msg[i], msg[i + 1]]);
|
|
|
|
i += 2;
|
|
|
|
}
|
|
|
|
return coupleList;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* calcul de l'inverse de d dans Z/pZ
|
|
|
|
* @param d
|
|
|
|
* @param p
|
|
|
|
*/
|
|
|
|
invdet(d, p) {
|
|
|
|
let inv = 1;
|
|
|
|
while ((d * inv) % p !== 1) {
|
|
|
|
inv++;
|
|
|
|
}
|
|
|
|
return inv;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* inversion de matrice
|
|
|
|
* @param matriceString
|
|
|
|
* @param p
|
|
|
|
*/
|
|
|
|
invmat(matriceString: string, p) {
|
|
|
|
const [a, b, c, d] = this.convertStringToMatrix(matriceString);
|
|
|
|
const mul = this.invdet(a * d - b * c, p);
|
|
|
|
return [(d * mul) % p, (-b * mul) % p, (-c * mul) % p, (a * mul) % p];
|
|
|
|
}
|
|
|
|
|
|
|
|
applique(mat, cplnum) {
|
|
|
|
const [a, b, c, d] = this.convertStringToMatrix(mat);
|
|
|
|
const [x, y] = cplnum;
|
|
|
|
const xout = (a * x + b * y) % this.prem;
|
|
|
|
const yout = (c * x + d * y) % this.prem;
|
|
|
|
return [xout, yout];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pour chaque caractère de la chaine msg, on vérifie qu'il est
|
|
|
|
dans l'alphabet des caractères autorisés : si oui on le garde,
|
|
|
|
sinon on le remplace par un underscore
|
|
|
|
* @param monTexte
|
|
|
|
*/
|
|
|
|
filterCharacters(monTexte) {
|
|
|
|
const convertedText = monTexte.split('').map((letter) => {
|
|
|
|
if (this.alphab.indexOf(letter) === -1) {
|
|
|
|
letter = '_';
|
|
|
|
}
|
|
|
|
return letter;
|
|
|
|
});
|
|
|
|
|
|
|
|
return convertedText.join('');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* on entre une définition de matrice en écrivant 4 nombres dans un input
|
|
|
|
* @param stringMatrix
|
|
|
|
*/
|
2021-02-12 14:36:10 +01:00
|
|
|
convertStringToMatrix(stringMatrix: string): number[] {
|
|
|
|
return stringMatrix
|
|
|
|
.replace(',', ' ')
|
|
|
|
.replace(' ', ' ')
|
|
|
|
.split(' ')
|
|
|
|
.map((elem) => parseInt(elem));
|
2021-02-12 12:06:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* chiffrer avec la méthode du CIL-gometz
|
|
|
|
* @param someText
|
|
|
|
*/
|
2021-02-09 18:26:13 +01:00
|
|
|
decrypt(sometext) {
|
2021-02-09 22:47:57 +01:00
|
|
|
return sometext;
|
2021-02-09 18:26:13 +01:00
|
|
|
}
|
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
convertTextToInt(value) {
|
|
|
|
let result = '';
|
|
|
|
for (let i = 0; i < value.length; i++) {
|
|
|
|
if (i < value.length - 1) {
|
|
|
|
result += value.charCodeAt(i) + 10;
|
|
|
|
result += '-';
|
|
|
|
} else {
|
|
|
|
result += value.charCodeAt(i) + 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2021-02-10 10:35:36 +01:00
|
|
|
|
|
|
|
convertIntToText(value, removeBeginning = true) {
|
2021-02-09 22:47:57 +01:00
|
|
|
let result = '';
|
|
|
|
const array = value.split('-');
|
2021-02-09 18:26:13 +01:00
|
|
|
|
2021-02-09 22:47:57 +01:00
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
|
|
result += String.fromCharCode(array[i] - 10);
|
2021-02-09 18:26:13 +01:00
|
|
|
}
|
2021-02-10 10:35:36 +01:00
|
|
|
// remove salt characters
|
|
|
|
if (removeBeginning) {
|
|
|
|
result = result.slice(this.salt.length, -1);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
convertTextToIntMixedSalt(value) {
|
|
|
|
let result = '';
|
|
|
|
let saltIndex = 0;
|
|
|
|
for (let i = 0; i < value.length; i++) {
|
|
|
|
if (i < value.length - 1) {
|
|
|
|
result += value.charCodeAt(i) + 10 + this.salt.charCodeAt(saltIndex);
|
|
|
|
result += '-';
|
|
|
|
} else {
|
|
|
|
result += value.charCodeAt(i) + 10 + this.salt.charCodeAt(saltIndex);
|
|
|
|
}
|
|
|
|
saltIndex++;
|
|
|
|
if (saltIndex >= this.salt.length) {
|
|
|
|
saltIndex = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
convertIntToTextMixedSalt(value) {
|
|
|
|
let result = '';
|
|
|
|
const array = value.split('-');
|
|
|
|
let saltIndex = -1;
|
|
|
|
|
|
|
|
for (let i = 0; i < array.length; i++) {
|
|
|
|
// console.log(
|
|
|
|
// 'encodage',
|
|
|
|
// i,
|
|
|
|
// value.charCodeAt(i),
|
|
|
|
// saltIndex,
|
|
|
|
// this.salt.charCodeAt(saltIndex),
|
|
|
|
// value.charCodeAt(i) + 10 + this.salt.charCodeAt(saltIndex)
|
|
|
|
// );
|
|
|
|
|
|
|
|
result += String.fromCharCode(array[i] - 10 - this.salt.charCodeAt(saltIndex));
|
|
|
|
if (saltIndex > this.salt.length) {
|
|
|
|
saltIndex = 0;
|
|
|
|
} else {
|
|
|
|
saltIndex++;
|
|
|
|
}
|
|
|
|
}
|
2021-02-09 22:47:57 +01:00
|
|
|
// remove salt characters
|
|
|
|
return result.slice(this.salt.length);
|
2021-02-09 18:26:13 +01:00
|
|
|
}
|
2021-02-10 11:23:09 +01:00
|
|
|
|
|
|
|
md5(someText) {
|
|
|
|
return CipherLib.md5(someText);
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 12:15:39 +01:00
|
|
|
getEntropy(s) {
|
2021-02-12 14:36:10 +01:00
|
|
|
let sum = 0;
|
|
|
|
const len = s.length;
|
2021-02-10 12:15:39 +01:00
|
|
|
this.process(s, (k, f) => {
|
|
|
|
const p = f / len;
|
|
|
|
sum -= (p * Math.log(p)) / Math.log(2);
|
|
|
|
});
|
|
|
|
return sum;
|
|
|
|
}
|
2021-02-12 12:06:30 +01:00
|
|
|
|
2021-02-10 12:15:39 +01:00
|
|
|
process(s, evaluator) {
|
|
|
|
let h = Object.create(null),
|
|
|
|
k;
|
|
|
|
s.split('').forEach((c) => {
|
|
|
|
(h[c] && h[c]++) || (h[c] = 1);
|
|
|
|
});
|
|
|
|
if (evaluator) for (k in h) evaluator(k, h[k]);
|
|
|
|
return h;
|
|
|
|
}
|
2021-02-08 11:32:58 +01:00
|
|
|
}
|