/* ** client.c -- a stream socket client demo */ #include #include #include #include #include #include #include #include #include #include #include #include #include "protocol.h" #define MAXDATASIZE 100 // max number of bytes we can get at once #define MAX_READ_SIZE 1024 #define LINE_MAX 1024 void *reply_thread_routine(void *args); // get sockaddr, IPv4 or IPv6: void *get_in_addr(struct sockaddr *sa) { if (sa->sa_family == AF_INET) { return &(((struct sockaddr_in*)sa)->sin_addr); } return &(((struct sockaddr_in6*)sa)->sin6_addr); } int main(int argc, char *argv[]) { int sockfd, numbytesrecv, numbytessent, recvbytes; // char buf[MAXDATASIZE]; struct addrinfo hints, *servinfo, *p; int rv; char s[INET6_ADDRSTRLEN]; if (argc != 4) { fprintf(stderr," \n"); exit(1); } memset(&hints, 0, sizeof hints); hints.ai_family = AF_INET; // only IPv4 hints.ai_socktype = SOCK_STREAM; char* tcpPort = argv[2]; // port we use to connect to server's tcp stream char* udpPort = argv[3]; // port we use to connect to server's udp info and command if ((rv = getaddrinfo(argv[1], tcpPort, &hints, &servinfo)) != 0) { fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv)); return 1; } // loop through all the results and connect to the first we can for(p = servinfo; p != NULL; p = p->ai_next) { if ((sockfd = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1) { perror("client: socket"); continue; } if (connect(sockfd, p->ai_addr, p->ai_addrlen) == -1) { close(sockfd); perror("client: connect"); continue; } break; } if (p == NULL) { fprintf(stderr, "client: failed to connect\n"); return 2; } inet_ntop(p->ai_family, get_in_addr((struct sockaddr *)p->ai_addr), s, sizeof s); // printf("client: connecting to %s\n", s); freeaddrinfo(servinfo); // all done with this structure struct Welcome msg; // recv the message, check for errors too if ((recvbytes = recv(sockfd, (char*)&msg, sizeof(struct Welcome), 0)) == -1) { perror("recv"); exit(1); } msg.numStations = ntohs(msg.numStations); printf("Welcome to Snowcast! The server has %d stations.\n", msg.numStations); struct Hello hello; hello.commandType = 0; // convert updPort to an int int udpPortInt = atoi(udpPort); hello.udpPort = htons(udpPortInt); if ((numbytessent = send(sockfd, &hello, sizeof(struct Hello), 0)) == -1) { perror("send"); exit(1); } pthread_t reply_thread; pthread_create(&reply_thread, NULL, reply_thread_routine, (void*)sockfd); char input[LINE_MAX]; printf("Enter a number to change to it's station. Click q to end stream.\n"); while (1) { char *line = fgets(input, LINE_MAX, stdin); if (line == NULL) { continue; } else if (strncmp("q\n", input, LINE_MAX) == 0) { // end code if type in q printf("Exiting.\n"); break; } else { // convert input to an int int inputInt = atoi(input); printf("Changing to station %d.\n", inputInt); // send the command to change the station struct SetStation setStation; setStation.commandType = 1; setStation.stationNumber = htons(inputInt); if ((numbytessent = send(sockfd, &setStation, sizeof(struct SetStation), 0)) == -1) { perror("send"); exit(1); } } } return 0; } void *reply_thread_routine(void* args) { int sockfd = (int)args; int recvbytes; char buf[MAX_READ_SIZE]; while (1) { // recv the message, check for errors too if ((recvbytes = recv(sockfd, &buf, MAX_READ_SIZE, 0)) == -1) { perror("recv"); exit(1); } buf[recvbytes] = '\0'; printf("client: received %d bytes on a reply call \n", recvbytes); // print the two first field of the call printf("client: replyType: %d, stringSize: %d\n", buf[0], buf[1]); // print the while buffer by char for (int i = 0; i < recvbytes; i++) { printf("%c ", buf[i]); } struct Reply reply; memcpy(&reply, buf, 2); // print out the fields of reply on one line printf("\nclient: replyType: %d, stringSize: %d\n", reply.replyType, reply.stringSize); // print the size of reply if (reply.replyType == 3) { printf("client: received an announce message\n"); char *song_name = malloc(reply.stringSize); // printf(sizeof(struct Reply)); memcpy(song_name, buf + 2, reply.stringSize); printf("client: song name: %s\n", song_name); free(song_name); continue; } else if (reply.replyType == 4) { printf("client: received an invalid command message\n"); continue; } printf("client: received an unknown message\n"); memset(buf, 0, MAX_READ_SIZE); } }