/**
 * agent_mode.c
 * 
 * Emulates the agent side of the communication, which accepts incoming packets
 * from the interaction layer and directly sends it back.
 */

#define _POSIX_C_SOURCE 200809L

#include <errno.h>
#include <inttypes.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/select.h>
#include <sys/socket.h>

#include "common.h"

int run_agent_server(int sock, const struct sockaddr_in *addr);
int run_interaction(int sock);


/**
 * main function parses arguments. See run_agent_server() for functionality.
 */
int main(int argc, char **argv)
{
    struct sockaddr_in addr;

    if (argc != 2) {
        fprintf(stderr, "usage: mock_agent <IP>:<PORT>\n");
        return 1;
    }

    if (atoip(argv[1], &addr)) {
        fprintf(stderr, "Invalid IP:PORT spec\n");
        return 1;
    }

    uint8_t *a = (uint8_t *) &addr.sin_addr.s_addr;
    fprintf(stderr, "IP address: %hhu.%hhu.%hhu.%hhu\n", a[0], a[1], a[2], a[3]);
    fprintf(stderr, "Port:       %hu\n", ntohs(addr.sin_port));

    // Create a TCP socket
    int sock = socket(AF_INET, SOCK_STREAM, 0);
    if (sock < 0) {
        perror("Could not create TCP socket");
        return 1;
    }

    int code = run_agent_server(sock, &addr);
    if (shutdown(sock, SHUT_RDWR)) {
        perror("Error shutting down socket");
        code = 1;
    }

    return code;
}


/**
 * Runs the agent layer server, accepting incoming connections.
 */
int run_agent_server(int sock, const struct sockaddr_in *addr)
{
    char printbuf[INET6_ADDRSTRLEN];
    struct sockaddr layer_addr;
    socklen_t sl;

    if (bind(sock, (const struct sockaddr *) addr, sizeof(struct sockaddr_in))) {
        perror("Could not bind socket to IP address");
        return 1;
    }

    if (listen(sock, 0)) {
        perror("Could not listen to socket");
        return 1;
    }

    fprintf(stderr, "Listening for connections...\n");
    fd_set rfds;
    struct timespec timeout;
    int n_timeouts = 0;

    bool done = false;
    while (!done) {
        timeout.tv_sec = 0;
        timeout.tv_nsec = 1000000;

        FD_ZERO(&rfds);
        FD_SET(sock, &rfds);

        int ret = pselect(sock + 1, &rfds, NULL, NULL, &timeout, NULL);
        if (ret == -1) {
            if (errno == EINTR) {
                done = true;
            } else {
                perror("Error selecting fd");
                return 1;
            }
        } else if (FD_ISSET(sock, &rfds)) {
            fprintf(stderr, "Accepting incoming connection\n");
            int layersock = accept(sock, &layer_addr, &sl);
            if (layersock == -1) {
                perror("Could not accept incoming connection");
                return 1;
            }
            uint16_t connport = 0;
            if (layer_addr.sa_family == AF_INET) {
                inet_ntop(AF_INET, &layer_addr, printbuf, INET6_ADDRSTRLEN);
                connport = ntohs(((struct sockaddr_in *) &layer_addr)->sin_port);
            } else if (layer_addr.sa_family == AF_INET6) {
                inet_ntop(AF_INET6, &layer_addr, printbuf, INET6_ADDRSTRLEN);
                connport = ntohs(((struct sockaddr_in6 *) &layer_addr)->sin6_port);
            } else {
                strcpy(printbuf, "unknown");
            }
            fprintf(stderr, "Got connection from %s:%hu\n", printbuf, connport);
            ret = run_interaction(layersock);
            if (ret)
                return ret;
        } else {
            // timedout
            n_timeouts++;
            if (n_timeouts > 60000) {
                fprintf(stderr, "Waiting for connection...\n");
                n_timeouts = 0;
            }
        }
    }

    fprintf(stderr, "Exiting\n");

    return 0;
}


/**
 * Runs the interaction over an established connection.
 */
int run_interaction(int sock)
{
    fd_set rfds;
    struct timespec timeout;
    static char buf[PACKET_SIZE];
    ssize_t recv_size = 0;

    bool done = false;
    while (!done) {
        FD_ZERO(&rfds);
        FD_SET(sock, &rfds);

        timeout.tv_sec = 0;
        timeout.tv_nsec = 1000000;

        int ret = pselect(sock + 1, &rfds, NULL, NULL, &timeout, NULL);
        if (ret == -1) {
            if (errno == EINTR) {
                done = true;
            } else {
                perror("Error selecting fd in run_interaction");
                return 1;
            }
        } else if (FD_ISSET(sock, &rfds)) {
            ssize_t r = recv(sock, &buf[recv_size], PACKET_SIZE - recv_size, 0);
            //fprintf(stderr, "received %ld bytes of data...\n", r);
            recv_size += r;

            if (r == -1) {
                perror("Error receiving packet");
                return 1;
            } else if (r == 0) {
                fprintf(stderr, "Connection closed\n");
                done = true;
                break;
            } else if (recv_size == PACKET_SIZE) {
                //uint32_t *u32buf = (uint32_t *) &buf[0];
                //fprintf(stderr, "received full packet (%u), sending it straight back...\n", ntohl(u32buf[0]));
                ssize_t w = write(sock, buf, PACKET_SIZE);
                if (w == -1) {
                    perror("Error sending packet");
                    return 1;
                } else if (w != PACKET_SIZE) {
                    fprintf(stderr, "Did not send expected packet size. Sent: %zu bytes\n", w);
                    return 1;
                }
                recv_size = 0;
            }
        }

        int sock_error = 0;
        socklen_t sock_len = sizeof(sock_error);
        if (getsockopt(sock, SOL_SOCKET, SO_ERROR, &sock_error, &sock_len)) {
            perror("Could not get socket status");
            return 1;
        }
        if (sock_error != 0) {
            fprintf(stderr, "Got socket error: %d\n", sock_error);
            return 1;
        }
    }

    return 0;
}
