-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathserver.c
More file actions
560 lines (461 loc) · 19.3 KB
/
server.c
File metadata and controls
560 lines (461 loc) · 19.3 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
#define _CRT_SECURE_NO_WARNINGS
#pragma comment(lib, "ws2_32.lib")
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <winsock2.h>
#include <time.h>
#define BUFSIZE 1024
#define NICKNAMEMAXLEN 10
#define NICKNAMEMINLEN 3
#pragma comment(lib, "ws2_32.lib") // ws2_32.lib 링크
// 문자 정보를 저장하는 구조체 정의
typedef struct {
char character; // 문자
int numericValue; // 숫자 값
char binary[9]; // 8비트 이진수 문자열
char xorResult[9]; // XOR 결과 이진수 문자열
char shifted[9]; // Shift Row 결과
char first4Bits[5]; // 첫 4비트
char second4Bits[5];// 둘째 4비트
int first4Decimal; // 첫 4비트의 10진수 값
int second4Decimal; // 둘째 4비트의 10진수 값
int sum1; // 첫 합 (first4Decimal + 닉네임 숫자) mod 26
int sum2; // 둘째 합 (second4Decimal + 닉네임 숫자) mod 26
char letter1; // 알파벳으로 변환한 결과
char letter2; // 알파벳으로 변환한 결과
char* result; // 비트화된 결과
} CharInfo;
// 섞인 테이블
int mixedTable[10][10] = {
{12, 5, 8, 19, 23, 45, 67, 89, 34, 56},
{78, 90, 11, 22, 33, 44, 55, 66, 77, 88},
{99, 0, 1, 2, 3, 4, 6, 7, 9, 10},
{13, 14, 15, 16, 17, 18, 20, 21, 24, 25},
{26, 27, 28, 29, 30, 31, 32, 35, 36, 37},
{38, 39, 40, 41, 42, 43, 46, 47, 48, 49},
{50, 51, 52, 53, 54, 57, 58, 59, 60, 61},
{62, 63, 64, 65, 68, 69, 70, 71, 72, 73},
{74, 75, 76, 79, 80, 81, 82, 83, 84, 85},
{86, 87, 91, 92, 93, 94, 95, 96, 97, 98}
};
// 문자 -> 숫자 변환 함수
int charToNumeric(char c) {
if (c >= 'a' && c <= 'z') {
return c - 'a'; // 'a'는 0부터 시작
}
else if (c >= 'A' && c <= 'Z') {
return c - 'A'; // 'A'는 0부터 시작
}
else if (c >= '0' && c <= '9') { // 숫자
return c - '0';
}
else {
return -1; // 유효하지 않은 문자 처리
}
}
// 정수를 8비트 이진수 문자열로 변환하는 함수
void toBinary8Bit(int num, char* binary) {
for (int i = 7; i >= 0; i--) {
binary[i] = (num % 2) + '0'; // 나머지를 이용하여 이진수 계산
num /= 2;
}
binary[8] = '\0'; // 문자열 끝에 NULL 추가
}
// 정수를 4비트 이진수 문자열로 변환하는 함수
void toBinary4Bit(int num, char* binary) {
for (int i = 3; i >= 0; i--) {
binary[i] = (num % 2) + '0'; // 나머지를 이용하여 이진수 계산
num /= 2;
}
binary[4] = '\0'; // 문자열 끝에 NULL 추가
}
// 두 개의 8비트 이진수 문자열을 XOR 연산하는 함수
void xorBinaryStrings(char* binary1, char* binary2, char* result) {
for (int i = 0; i < 8; i++) {
// 각 비트를 정수로 변환하여 XOR 연산 후, 다시 문자로 변환
result[i] = ((binary1[i] - '0') ^ (binary2[i] - '0')) + '0';
}
result[8] = '\0'; // 문자열 끝에 NULL 추가
}
// 8비트 이진수를 순환 시프트하는 함수
void shiftRow(char* binary, int shiftAmount, int direction) {
char temp[9]; // 임시 저장 배열
int len = 8; // 8비트 길이 고정
if (direction == 0) { // 왼쪽 순환 시프트
for (int i = 0; i < len; i++) {
temp[i] = binary[(i + shiftAmount) % len];
}
}
else if (direction == 1) { // 오른쪽 순환 시프트
for (int i = 0; i < len; i++) {
temp[i] = binary[(i - shiftAmount + len) % len];
}
}
temp[len] = '\0'; // NULL 문자 추가
strcpy(binary, temp); // 문자열 복사
}
// 8비트 이진수를 4비트씩 나누는 함수
void splitTo4Bits(char* binary, char* first4, char* second4) {
strncpy(first4, binary, 4); // 첫 4비트 복사
first4[4] = '\0'; // 문자열 끝에 NULL 추가
strncpy(second4, binary + 4, 4); // 다음 4비트 복사
second4[4] = '\0'; // 문자열 끝에 NULL 추가
}
//
int binaryToDecimal(char* binary) {
int decimal = 0;
int length = strlen(binary);
for (int i = 0; i < length; i++) {
decimal = decimal * 2 + (binary[i] - '0');
}
return decimal;
}
// 현재 시간을 가져와 시와 분의 합을 반환하는 함수
int getCurrentTime() {
time_t now = time(NULL);
struct tm* local = localtime(&now);
return local->tm_hour + local->tm_min;
}
void join4BitsTo8Bit (char* result, char* first4Bits, char* second4Bits) {
strcpy(result, first4Bits);
strcat(result, second4Bits);
return;
}
// 4x4 테이블 함수
int fourByfourTable(int binaryInput) {
// 고정된 랜덤 4x4 테이블 정의
int table[4][4] = {
{3, 7, 11, 2},
{14, 6, 9, 1},
{12, 0, 5, 8},
{10, 13, 4, 15}
};
// 입력된 2진수를 행과 열로 변환
int row = (binaryInput >> 2) & 0x3; // 첫 2비트
int col = binaryInput & 0x3; // 뒤 2비트
return table[row][col];
}
// 역 4x4 테이블 함수
int reverseFourByfourTable(int outputValue) {
// 고정된 랜덤 4x4 테이블 정의
int table[4][4] = {
{3, 7, 11, 2},
{14, 6, 9, 1},
{12, 0, 5, 8},
{10, 13, 4, 15}
};
for (int input = 0; input < 16; input++) {
int row = (input >> 2) & 0x3;
int col = input & 0x3;
if (table[row][col] == outputValue) {
return input;
}
}
// 만약 찾지 못했다면 -1 반환
return -1;
}
void nicknameToNumeric(const char* nickname, int result[]) {
int nicknameLength = strlen(nickname);
for (int i = 0; i < nicknameLength; i++) {
int numeric = charToNumeric(nickname[i]);
if (numeric == -1) {
printf("닉네임의 문자 '%c'는 변환할 수 없습니다.\n", nickname[i]);
return;
}
result[i] = numeric;
}
}
// 인코딩 함수
int encodeMessage(const char* nickname, const char* plaintext, char* encryptedText) {
int nicknameNumeric[100];
int nicknameLength = strlen(nickname);
for (int i = 0; i < nicknameLength; i++) {
int numeric = charToNumeric(nickname[i]);
if (numeric == -1) {
printf("닉네임의 문자 '%c'는 변환할 수 없습니다.\n", nickname[i]);
encryptedText[0] = '\0'; // 에러 처리
return -1;
}
nicknameNumeric[i] = numeric;
}
// 현재 시간의 시와 분 합 계산
int timeSum = getCurrentTime();
char timeBinary[9];
// 두 자리 시간의 앞자리 - 행, 뒷자리 - 열로 테이블 조회 (10x10 테이블)
int crytedTime = mixedTable[(timeSum / 10) % 10][timeSum % 10];
// 테이블을 통과한 숫자를 이진수로 변환
toBinary8Bit(crytedTime, timeBinary);
// 입력받은 문자들을 저장할 구조체 배열
CharInfo charInfos[100];
int charCount = 0;
// 평문의 각 문자 처리
for (int i = 0; i < strlen(plaintext); i++) {
int numeric = charToNumeric(plaintext[i]);
if (numeric == -1) {
printf("문자 '%c'는 변환할 수 없습니다.\n", plaintext[i]);
continue;
}
CharInfo info;
info.character = plaintext[i];
info.numericValue = numeric;
// 평문의 숫자를 이진수로 변환
toBinary8Bit(numeric, info.binary);
// [plaintext와 시간] XOR 결과 계산
char result[9];
xorBinaryStrings(info.binary, timeBinary, result);
// Round 시작
int roundNum = 3;
for(int round=0; round<roundNum; round++) {
int nicknameIndex = 0; // 닉네임 인덱스 초기화
// Shift Row 수행
strcpy(info.shifted, result);
shiftRow(info.shifted, 2, 0); // 왼쪽으로 2비트 시프트
// Shift Row 결과를 4비트씩 분리
splitTo4Bits(info.shifted, info.first4Bits, info.second4Bits);
// 4비트 -> 10진수 변환
int firstDecimal = binaryToDecimal(info.first4Bits);
int secondDecimal = binaryToDecimal(info.second4Bits);
// 4x4 테이블 적용,
info.first4Decimal = fourByfourTable(firstDecimal);
info.second4Decimal = fourByfourTable(secondDecimal);
// 닉네임 숫자와 합산 후 mod 26 (버지니아)
int nicknameLength = strlen(nickname);
info.sum1 = (info.first4Decimal + nicknameNumeric[nicknameIndex % nicknameLength]) % 16;
nicknameIndex++;
info.sum2 = (info.second4Decimal + nicknameNumeric[nicknameIndex % nicknameLength]) % 16;
nicknameIndex++;
// 위 결과를 2진수 4비트화
char first4Bits[5]; char second4Bits[5];
toBinary4Bit(info.sum1, first4Bits);
toBinary4Bit(info.sum2, second4Bits);
join4BitsTo8Bit(result, first4Bits, second4Bits);
}
char firstResult4Bits[5]; char secondResult4Bits[5];
splitTo4Bits(result, firstResult4Bits, secondResult4Bits);
info.letter1 = binaryToDecimal(firstResult4Bits) + 'A'; // 대문자로 변환
info.letter2 = binaryToDecimal(secondResult4Bits) + 'A';
info.result = result;
// 구조체 배열에 저장
charInfos[charCount++] = info;
}
// 암호화된 결과를 저장할 문자열
encryptedText[0] = '\0'; // 초기화
for (int i = 0; i < charCount; i++) {
int len = strlen(encryptedText);
encryptedText[len] = charInfos[i].letter1;
encryptedText[len + 1] = charInfos[i].letter2;
encryptedText[len + 2] = '\0';
}
printf("사용된 시간 합: %d\n", timeSum); // 디버깅을 위해 시간 합 출력
return timeSum;
}
// 디코딩 함수
void decodeMessage(const char* nickname, const char* encryptedText, char* decryptedText, int timeSum) {
// 닉네임의 각 문자를 숫자로 변환
int nicknameNumeric[100];
int nicknameLength = strlen(nickname);
for (int i = 0; i < nicknameLength; i++) {
int numeric = charToNumeric(nickname[i]);
if (numeric == -1) {
printf("닉네임의 문자 '%c'는 변환할 수 없습니다.\n", nickname[i]);
decryptedText[0] = '\0'; // 에러 처리
return;
}
nicknameNumeric[i] = numeric;
}
// 시간 합으로부터 timeBinary 생성
char timeBinary[9];
int crytedTime = mixedTable[(timeSum / 10) % 10][timeSum % 10];
toBinary8Bit(crytedTime, timeBinary);
int decryptedIndex = 0;
// 암호문의 각 문자 쌍 처리
for (int i = 0; i < strlen(encryptedText); i+=2) {
int roundNum = 3;
// 암호문을 해당하는 10진수로 변환
int sum1 = charToNumeric(encryptedText[i]);
int sum2 = charToNumeric(encryptedText[i+1]);
// 해당 10진수를 4비트 2진수로 변환
char firstSum4Bits[5]; char secondSumBits[5];
toBinary4Bit(sum1, firstSum4Bits);
toBinary4Bit(sum2, secondSumBits);
// 해당 4비트 2진수들을 붙여서 8비트로 변환
char roundResult[9];
join4BitsTo8Bit(roundResult, firstSum4Bits, secondSumBits);
// Round start
for (int round = 0; round<roundNum; round++) {
int nicknameIndex = 0; // 닉네임 인덱스 초기화
// 8비트를 4비트로 분할
char roundFirst4Bits[5]; char roundSecond4Bits[5];
splitTo4Bits(roundResult, roundFirst4Bits, roundSecond4Bits);
// 각 4비트 2진수를 10진수로 변환
int sum1 = binaryToDecimal(roundFirst4Bits);
int sum2 = binaryToDecimal(roundSecond4Bits);
// 닉네임 숫자와의 역연산 (버지니아)
int CfirstDecimal = (sum1 - nicknameNumeric[nicknameIndex % nicknameLength] + 32) % 16;
nicknameIndex++;
int CsecondDecimal = (sum2 - nicknameNumeric[nicknameIndex % nicknameLength] + 32) % 16;
nicknameIndex++;
// 4x4 테이블의 역변환
int firstDecimal = reverseFourByfourTable(CfirstDecimal);
int secondDecimal = reverseFourByfourTable(CsecondDecimal);
if (firstDecimal == -1 || secondDecimal == -1) {
printf("4x4 테이블에서 역변환할 수 없습니다.\n");
continue;
}
// 4비트 이진수로 변환
char first4Bits[5], second4Bits[5];
toBinary4Bit(firstDecimal, first4Bits);
toBinary4Bit(secondDecimal, second4Bits);
// 이진수 결합
char shifted[9];
join4BitsTo8Bit(shifted, first4Bits, second4Bits);
// 순환 시프트 복원 (오른쪽으로 2비트)
strcpy(roundResult, shifted);
shiftRow(roundResult, 2, 1); // 오른쪽으로 2비트 시프트
}
// XOR 연산 복원
char binary[9];
xorBinaryStrings(roundResult, timeBinary, binary);
// 이진수를 10진수로 변환
int numericValue = binaryToDecimal(binary);
// 숫자를 문자로 변환
char character = numericValue + 'a'; // 대문자로 변환
decryptedText[decryptedIndex++] = character;
}
decryptedText[decryptedIndex] = '\0'; // 문자열 끝에 NULL 추가
}
int getTimeNow() { // 4자리의 24시간 형식 시간을 가져오는 함수
time_t now = time(NULL);
struct tm* local = localtime(&now);
int serverTime = local->tm_hour * 100 + local->tm_min;
return serverTime;
}
void encryptTransmissionTime(int serverTime, char* encryptedTime) {
// 전송 시간과 서버 시간을 XOR 연산하여 암호화
sprintf(encryptedTime, "%08X", serverTime); // 16진수 문자열로 변환
}
void decryptTransmissionTime(char* timeChar) {
for(int i=0; i<strlen(timeChar); i++) {
timeChar[i] = charToNumeric(timeChar[i]) + '0';
}
}
void splitTransmissionTimeAndMessage(char* decryptedTransmissionTime, char* decryptedText) {
strcpy(decryptedTransmissionTime, &decryptedText[strlen(decryptedText) - 8]);
decryptTransmissionTime(decryptedTransmissionTime);
decryptedText[strlen(decryptedText) - 8] = '\0';
printf("dT: %s\n", decryptedTransmissionTime);
}
void ErrorHandling(char* message) {
WSACleanup();
fputs(message, stderr);
exit(1);
}
int main(int argc, char** argv) {
WSADATA wsaData;
SOCKET servSock, clntSock;
SOCKADDR_IN servAddr, clntAddr;
char message[BUFSIZE];
int strLen = 0, fromLen = 0, nRcv = 0, port = 0;
char nickname[100];
char opponentNickname[100];
int enteringTime;
char encryptedEnteringTime[20];
char encryptedText[BUFSIZE]; // 암호화 메세지
char decryptedText[BUFSIZE]; // 복호화 메세지
if (argc != 2) {
printf("포트 번호를 입력해주세요. : ");
scanf_s("%d", &port);
}
WSAStartup(MAKEWORD(2, 2), &wsaData);
servSock = socket(PF_INET, SOCK_STREAM, 0);
if (servSock == INVALID_SOCKET)
ErrorHandling("소켓 에러\n");
memset(&servAddr, 0, sizeof(SOCKADDR_IN));
servAddr.sin_family = AF_INET;
servAddr.sin_port = htons(argc == 2 ? atoi(argv[1]) : port);
servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(servSock, (struct sockaddr*)&servAddr, sizeof(servAddr)) == SOCKET_ERROR)
ErrorHandling("바인드 에러\n");
if (listen(servSock, 2) == SOCKET_ERROR)
ErrorHandling("리스닝 에러\n");
fromLen = sizeof(clntAddr);
clntSock = accept(servSock, (struct sockaddr*)&clntAddr, &fromLen);
if (clntSock == INVALID_SOCKET)
ErrorHandling("연결 에러\n");
else printf("연결 성공\n");
while (1) {
fflush(stdin);
printf("닉네임을 입력해주세요. (최소 4자 , 최대 10자)\n : ");
fflush(stdin);
fgets(nickname, NICKNAMEMAXLEN, stdin);
fflush(stdin);
nickname[strcspn(nickname, "\n")] = '\0'; // 개행 문자 제거
if ((int)strlen(nickname) < NICKNAMEMINLEN || (int)strlen(nickname) > NICKNAMEMAXLEN) {
printf("닉네임이 올바르지 않습니다. 다시 입력해주세요\n");
}
else {
printf("%s님이 대화방에 입장했습니다.\n", nickname);
send(clntSock, nickname, (int)strlen(nickname), 0);
break;
}
}
// 닉네임 입력 대기
printf("상대방의 대화방 입장 기다리는 중...\n");
nRcv = recv(clntSock, message, sizeof(message) - 1, 0);
if (nRcv == SOCKET_ERROR) {
printf("수신 에러..\n");
return 0;
}
message[nRcv] = '\0';
if (strcmp(message, "exit") == 0) {
printf("클라이언트가 연결을 종료하였습니다.\n");
return 0;
}
printf("%s님이 대화방에 입장했습니다.\n", message);
// 상대방 닉네임 저장
strcpy_s(opponentNickname, 100, message);
// 대화 시작시간 저장 (해시값)
enteringTime = getTimeNow();
encryptTransmissionTime(enteringTime, encryptedEnteringTime);
while (1) {
// 메세지 수신 대기
nRcv = recv(clntSock, message, sizeof(message) - 1, 0);
if (nRcv == SOCKET_ERROR) {
printf("수신 에러..\n");
break;
}
message[nRcv] = '\0';
if (strcmp(message, "exit") == 0) {
printf("클라이언트가 연결을 종료하였습니다.\n");
break;
}
printf("받은 암호 메시지 : %s\n", message);
decodeMessage(opponentNickname, message, decryptedText, getCurrentTime());
// 해시 복호화
char decryptedTransmissionTime[20];
splitTransmissionTimeAndMessage(decryptedTransmissionTime, decryptedText);
printf("해시 유효성: %d (%s and %s)\n", strcmp(decryptedTransmissionTime, encryptedEnteringTime), decryptedTransmissionTime, encryptedEnteringTime);
printf("받은 복호 메시지 : %s\n", decryptedText);
decryptedText[0] = '\0'; // 초기화
printf("보낼 메시지 : ");
fgets(message, BUFSIZE, stdin);
message[strcspn(message, "\n")] = '\0'; // 개행 문자 제거
// 해시 암호화
char encryptedTransmissionTime[20];
encryptTransmissionTime(enteringTime, encryptedTransmissionTime);
// 암호화된 전송 시간 추가
strcat(message, encryptedTransmissionTime);
encodeMessage(nickname, message, encryptedText);
printf("보낼 암호 메시지 : %s\n", encryptedText);
// 전송
send(clntSock, encryptedText, (int)strlen(encryptedText), 0);
encryptedText[0] = '\0';
if (strcmp(message, "exit") == 0) break;
}
closesocket(clntSock);
WSACleanup();
printf("연결 종료..\n");
return 0;
}