-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.c
159 lines (134 loc) · 3.69 KB
/
server.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
/* Project10: Server
* Shivani Kohli and James Anders */
/* Based on code by DWB */
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <pthread.h>
#define BUF_SIZE 500
#define MAX_CLIENTS 5
struct chat_client {
int ID;
char name[BUF_SIZE];
struct sockaddr_in address;
};
FILE *fp;
struct chat_client connected_clients[MAX_CLIENTS];
int main (int argc, char *argv[])
{
int sockfd, myport, nread, their_addr_len;
struct sockaddr_in my_addr, their_addr;
char buf[BUF_SIZE];
if (argc != 2)
{
fprintf (stderr, "Usage: %s portnum\n", argv[0]);
exit (EXIT_FAILURE);
}
myport = atoi (argv[1]);
sockfd = socket (AF_INET, SOCK_DGRAM, 0);
my_addr.sin_family = AF_INET;
my_addr.sin_port = htons (myport);
my_addr.sin_addr.s_addr = INADDR_ANY;
memset (&(my_addr.sin_zero), '\0', 8);
if (bind (sockfd, (struct sockaddr *) &my_addr, sizeof (struct sockaddr)))
{
close (sockfd);
fprintf (stderr, "Failed to bind socket!\n");
exit (EXIT_FAILURE);
}
else
{
printf ("Server listening on port %d\n", myport);
}
fp = fopen("./chat_history.txt", "w+");
char host[NI_MAXHOST], service[NI_MAXSERV];
int result;
int active_clients;
active_clients = 0;
while (1) {
their_addr_len = sizeof (struct sockaddr_in);
nread = recvfrom (sockfd, buf, BUF_SIZE, 0,
(struct sockaddr *) &their_addr, &their_addr_len);
printf("buf: %s\n", buf);
char bexit[BUF_SIZE];
sprintf(bexit,"exit\n");
if (!strcmp(buf, bexit)) {
fclose(fp);
close(sockfd);
exit(1);
}
result = getnameinfo ((struct sockaddr *) &their_addr, their_addr_len,
host, NI_MAXHOST, service, NI_MAXSERV,
NI_NUMERICSERV);
printf("addr: %lu\n", their_addr.sin_addr.s_addr);
int i;
char sendername[BUF_SIZE];
int senderID;
struct chat_client client;
int new_client;
new_client = 1;
for (i = 0; i < MAX_CLIENTS; i++) {
client = connected_clients[i];
if (client.address.sin_addr.s_addr == their_addr.sin_addr.s_addr) {
// client exists in array
new_client = 0;
sprintf(sendername,"%s",client.name);
senderID = client.ID;
break;
}
}
struct chat_client newguy;
if (new_client) {
// Add to array
if (active_clients > 5) {
fprintf (stderr, "Too many clients\n");
}
else {
newguy.address = their_addr;
newguy.ID = active_clients;
sprintf(newguy.name, "%s", buf);
connected_clients[active_clients] = newguy;
active_clients++;
sprintf(sendername,"%s",newguy.name);
}
}
// Broadcast message
for (i = 0; i < active_clients; i++) {
client = connected_clients[i];
char msg[BUF_SIZE];
if (new_client) {
sprintf(msg, "%s has entered ze chat room\n", buf);
fprintf(fp, "%s has entered ze chat room\n", buf);
if (sendto (sockfd, msg, BUF_SIZE, 0, (struct sockaddr *) &client.address,
sizeof (struct sockaddr_in)) != nread)
{
fprintf (stderr, "Error sending response\n");
}
}
else {
if (client.ID != senderID) {
sprintf(msg, "%s:%s", sendername, buf);
fprintf(fp, "%s:%s", sendername, buf);
if (sendto (sockfd, msg, BUF_SIZE, 0, (struct sockaddr *) &client.address,
sizeof (struct sockaddr_in)) != nread)
{
fprintf (stderr, "Error sending response\n");
}
}
}
}
if (result == 0)
printf ("Received %d bytes from %s:%s\n", (long) nread, host, service);
else
fprintf (stderr, "getnameinfo: %s\n", gai_strerror (result));
}
close(fp);
close (sockfd);
return 0;
}