-
Notifications
You must be signed in to change notification settings - Fork 0
/
otp_dec_d.c
215 lines (181 loc) · 5.8 KB
/
otp_dec_d.c
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
/* this code was adapted from examples found on:
* http://www.linuxhowtos.org/C_C++/socket.htm */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <fcntl.h>
void error(const char *msg)//function to make error handling simpler
{
fprintf(stderr, "%s\n", msg);
exit(1);
}
int sendall(int s, char *buf, int *len);
void decrypt(char keyText[], char message[], char decrypted[]);
int main(int argc, char *argv[])
{
int sockfd, newsockfd, portno;
socklen_t clilen;
char buffer[256];
struct sockaddr_in serv_addr, cli_addr;
int n;
char key[71000], message[70000], decrypted[70000];
//check command line usage
if (argc < 2)
error("USAGE ERROR, no port provided");
//create a new socket
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
//zero out struct and set properties
bzero((char *) &serv_addr, sizeof(serv_addr));
portno = atoi(argv[1]);
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;//will work for localhost
serv_addr.sin_port = htons(portno); //convert port number to big endian byte order
// bind socket to the address set up above
if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0)
error("ERROR on binding");
listen(sockfd,5);//start listening on the socket for connections
while(1)//loop on accept to take incoming connections
{
//accept new connections
clilen = sizeof(cli_addr);
newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);
if (newsockfd < 0)
error("ERROR on accept");
//exchnage verification with client
char verify[256];
bzero(verify, 256);
n = recv(newsockfd, verify, 255, 0);
if(n < 0)
error("recv error");
//send validation word to client to prevent
//otp_dec from connecting here
bzero(buffer, 256);
strcpy(buffer, "decrypt");
n = write(newsockfd, buffer, strlen(buffer));
if(n < 0)
error("ERROR writing to socket");
int status;
int pid = fork();//fork off a new process
if(pid < 0)
perror("fork error\n");
else if(pid == 0)//child
{
close(sockfd);//we no longer need the "listening" socket
//if verification from client is not
//"decrypt" then close connection
if( strcmp(buffer, verify) != 0)
{
//printf("%s %s\n", buffer, verify);
close(newsockfd);
error("ERROR: otp_enc can not connect to this server");
}
char buf[1025];
int total = 0;
bzero(key,sizeof(key));
bzero(buf, sizeof(buf));
//receive the encryption key from client 1024 byte increments
while ((n = recv(newsockfd, buf, 1024, 0)) > 0)
{
total += n;
strcat(key, buf); //add new section to the whole
bzero(buf, sizeof(buf));
if (n < 1024)//the end of the string has been reached
break;
}
//printf("key total: %d\nlength of key: %d\n", total, strlen(key));
//send confirmation to client that bytes were received
//i did this mainly because i had trouble getting
//consecutive sends from the client to work. it works
//better if client and server call and respond
bzero(buffer, 256);
sprintf(buffer, "read %d bytes of key\n", total);
total = strlen(buffer);
sendall(newsockfd, buffer, &total);
//receive the message
total = 0;
bzero(buf, sizeof(buf));
bzero(message, sizeof(message));
while ((n = recv(newsockfd, buf, 1024, 0)) > 0)
{
total += n;
strcat(message, buf);
bzero(buf, sizeof(buf));
if (n < 1024)
break;
}
//printf("message total: %d\nlength of message before function: %d\n", total, strlen(message));
decrypt(key, message, decrypted);//decrypt the message
//send the decryption back to the client
total = strlen(decrypted);
//printf("decrytped length: %d\n", total);
sendall(newsockfd, decrypted, &total);
//printf("decrypted total: %d\n", total);
close(newsockfd);
exit(0);
}
else//parent
{
close(newsockfd);//closed connection being used in child
//waitpid returns 0 when no children have changed state
while(pid > 0)
pid = waitpid(0, &status, WNOHANG);//wait for any children to exit
}
}
//the program shouldn't ever reach here
close(sockfd);
return 0;
}
/*Source of this function is http://beej.us/guide/bgnet/output/html/singlepage/bgnet.html#sendman
*sends data in buf through socket s and keeps track of what was sent to deal
with partial sends. */
int sendall(int s, char *buf, int *len)
{
int total = 0; // how many bytes we've sent
int bytesleft = *len; // how many we have left to send
int n;
while(total < *len) {//loop until all of the data is sent
n = send(s, buf+total, bytesleft, 0);
if (n == -1) { break; }
total += n;
bytesleft -= n;
}
*len = total; // return number actually sent here
return n==-1?-1:0; // return -1 on failure, 0 on success
}
/* function to revers the encryption that took place in otp_enc_d. */
void decrypt(char keyText[], char message[], char decrypted[])
{
int i, msg, key, dec;
memset(decrypted, 0, sizeof(decrypted));
for(i = 0; i < strlen(message); i++)
{
msg = message[i];
key = keyText[i];
//spaces are give decimal
//value 91
if(msg == 32)
msg = 91;
if(key == 32)
key = 91;
//give value between
//0 and 26
msg -= 65;
key -= 65;
dec = msg - key;
if(dec < 0)
dec += 27;
dec = dec % 27;
//convert back to ascii value
dec += 65;
//handle space character
if( dec == 91 )
dec = 32;
decrypted[i] = dec;
}
}