Files
jwinf/j1-Passwoerter/src/main.c
2020-11-23 22:30:17 +01:00

77 lines
3.2 KiB
C

#include <stdio.h>
#include <acl/file.h>
#include <string.h>
#include <stddef.h>
#include <acl/array.h>
#include <stdint.h>
#include <stdlib.h>
#include <inttypes.h>
#define wordCount 3 // Anzahl der Wörter des Passworts
#define randomInt_len (2 * wordCount)
#define wordsForPass_len randomInt_len / 2
void pointerCheck(void *pointer) { // Funktion um auf NULL Pointer zu checken und den Fehler auszugeben.
if(pointer == NULL) {
perror("Error: ");
exit(-1);
}
}
int main(int argc, char *argv[]) {
if(argc != 2) {
printf("file argument required"); // checkt ob der Nutzer die richtige Anzahl von Argumenten verwendet hat.
exit(-1);
}
bool sucess;
/* im binären Modus zu lesen ist deutlich schneller als im Textmodus, da man dort jede Zeile einzeln lesen muss.
* So wurde aus mehreren Sekunden Wartezeit eine nicht bemerkbare Verzögerung. */
FILE *fp = fopen(argv[1], "rb");
char *woerterText = acl_ReadTextFile(fp, &sucess);
if(!sucess) perror("Error: ");
fclose(fp);
unsigned len = strlen(woerterText);
uintptr_t dest = (uintptr_t)woerterText + len;
/* erstellt ein Array für die Pointer zu den Wörtern.
* Man könnte auch erst die Zeilenumbrüche zählen, um den Arbeitsspeicher auf einmal anzufragen.
* Dies würde das Programm vermutlich etwas schneller machen.
* Jedoch ist das Programm so einfacher zu verstehen und im Vergleich zum Auslesen der Textdatei wäre die Geschwindigkeitsverbesserung sehr gering. */
char **woerter = acl_arraylist_create(1, sizeof *woerter);
pointerCheck(woerter);
woerter[0] = woerterText;
for(char *i = woerterText; (uintptr_t)i < dest; ++i) {
if(*i == '\n') {
char *nextWord = i + 1;
woerter = acl_arraylist_append(woerter, &nextWord); // Pointer werden am Anfang jedes Wortes erstellt.
pointerCheck(woerter);
*i = '\0'; // \n wird durch \0 ersetzt um das Ende des Strings markieren.
// diesen Weg habe ich gewählt um viele memory allocations zu vermeiden.
}
}
FILE *randomFile = fopen("/dev/random", "rb"); // /dev/random ist ein kryptografisch sicherer Zufallszahlengenerator.
uint32_t randomInt[randomInt_len];
fread(randomInt, sizeof *randomInt, randomInt_len, randomFile);
size_t passLen = wordsForPass_len + 1; // weil Ziffern im Passwort gleich oft vorkommen wie Wörter. + 1 für ein Satzzeichen am Ende
char *wordsForPass[wordsForPass_len];
// Die Länge des Passwortes wird berechnet.
for(uint32_t i = 0; i < wordsForPass_len; ++i) {
wordsForPass[i] = woerter[randomInt[i] % acl_arraylist_len(woerter)]; // der modulo Operator wird verwendet um die Zufallszahl auf den gewünschten Zahlenbereich zu reduzieren.
passLen += strlen(wordsForPass[i]);
}
char *password = malloc(passLen + 1);
password[0] = '\0';
uint32_t j = 0;
for(uint32_t i = wordsForPass_len; i < randomInt_len; ++i, ++j) {
char buf[2];
buf[1] = '\0';
sprintf(buf, "%" PRIu32, randomInt[i] % 10);
strcat(password, buf);
strcat(password, wordsForPass[j]);
}
free(woerterText);
// ein zufälliges Sonderzeichen wird angehängt.
const char specialCharacter[] = {'?', '.', ',', '!', ';'};
char randomSpecialCharacter[] = {specialCharacter[randomInt[randomInt_len - 1] % (sizeof specialCharacter - 1)], '\0'};
printf("%s\n", strcat(password, randomSpecialCharacter));
}