/*
 * Copyright notice from original mutt:
 * Copyright (C) 1998 Michael R. Elkins <me@mutt.org>
 * Copyright (C) 1999-2005 Brendan Cully <brendan@kublai.com>
 * Copyright (C) 1999-2000 Tommi Komulainen <Tommi.Komulainen@iki.fi>
 *
 * This file is part of mutt-ng, see http://www.muttng.org/.
 * It's licensed under the GNU General Public License,
 * please see the file GPL in the top level source directory.
 */

#if HAVE_CONFIG_H
# include "config.h"
#endif

#include "mutt.h"
#include "globals.h"
#include "mutt_socket.h"
#include "mutt_tunnel.h"
#if defined(USE_SSL) || defined(USE_GNUTLS)
# include "mutt_ssl.h"
#endif

#include "mutt_idna.h"

#include "lib/mem.h"
#include "lib/intl.h"
#include "lib/str.h"
#include "lib/debug.h"

#include <unistd.h>
#include <netinet/in.h>
#include <netdb.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <errno.h>

/* support for multiple socket connections */
static CONNECTION *Connections = NULL;

/* forward declarations */
static int socket_preconnect (void);
static int socket_connect (int fd, struct sockaddr *sa);
static CONNECTION *socket_new_conn (void);

/* Wrappers */
int mutt_socket_open (CONNECTION * conn)
{
  if (socket_preconnect ())
    return -1;

  return conn->conn_open (conn);
}

int mutt_socket_close (CONNECTION * conn)
{
  int rc = -1;

  if (conn->fd < 0)
    debug_print (1, ("Attempt to close closed connection.\n"));
  else
    rc = conn->conn_close (conn);

  conn->fd = -1;
  conn->ssf = 0;

  return rc;
}

int mutt_socket_read (CONNECTION * conn, char *buf, size_t len)
{
  int rc;

  if (conn->fd < 0) {
    debug_print (1, ("attempt to read from closed connection\n"));
    return -1;
  }

  rc = conn->conn_read (conn, buf, len);
  /* EOF */
  if (rc == 0) {
    mutt_error (_("Connection to %s closed"), conn->account.host);
    mutt_sleep (2);
  }
  if (rc <= 0)
    mutt_socket_close (conn);

  return rc;
}

int mutt_socket_write_d (CONNECTION * conn, const char *buf, int dbg)
{
  int rc;
  int len;

  debug_print (dbg, ("> %s", buf));

  if (conn->fd < 0) {
    debug_print (1, ("attempt to write to closed connection\n"));
    return -1;
  }

  len = str_len (buf);
  if ((rc = conn->conn_write (conn, buf, len)) < 0) {
    debug_print (1, ("error writing, closing socket\n"));
    mutt_socket_close (conn);

    return -1;
  }

  if (rc < len) {
    debug_print (1, ("ERROR: wrote %d of %d bytes!\n", rc, len));
  }

  return rc;
}

/* simple read buffering to speed things up. */
int mutt_socket_readchar (CONNECTION * conn, char *c)
{
  if (conn->bufpos >= conn->available) {
    if (conn->fd >= 0)
      conn->available =
        conn->conn_read (conn, conn->inbuf, sizeof (conn->inbuf));
    else {
      debug_print (1, ("attempt to read from closed connection.\n"));
      return -1;
    }
    conn->bufpos = 0;
    if (conn->available == 0) {
      mutt_error (_("Connection to %s closed"), conn->account.host);
      mutt_sleep (2);
    }
    if (conn->available <= 0) {
      mutt_socket_close (conn);
      return -1;
    }
  }
  *c = conn->inbuf[conn->bufpos];
  conn->bufpos++;
  return 1;
}

int mutt_socket_readln_d (char *buf, size_t buflen, CONNECTION * conn,
                          int dbg)
{
  char ch;
  int i;

  for (i = 0; i < buflen - 1; i++) {
    if (mutt_socket_readchar (conn, &ch) != 1) {
      buf[i] = '\0';
      return -1;
    }

    if (ch == '\n')
      break;
    buf[i] = ch;
  }

  /* strip \r from \r\n termination */
  if (i && buf[i - 1] == '\r')
    buf[--i] = '\0';
  else
    buf[i] = '\0';

  debug_print (dbg, ("< %s\n", buf));

  /* number of bytes read, not str_len */
  return i + 1;
}

CONNECTION *mutt_socket_head (void)
{
  return Connections;
}

/* mutt_socket_free: remove connection from connection list and free it */
void mutt_socket_free (CONNECTION * conn)
{
  CONNECTION *iter;
  CONNECTION *tmp;

  iter = Connections;

  /* head is special case, doesn't need prev updated */
  if (iter == conn) {
    Connections = iter->next;
    mem_free (&iter);
    return;
  }

  while (iter->next) {
    if (iter->next == conn) {
      tmp = iter->next;
      iter->next = tmp->next;
      mem_free (&tmp);
      return;
    }
    iter = iter->next;
  }
}

/* mutt_conn_find: find a connection off the list of connections whose
 *   account matches account. If start is not null, only search for
 *   connections after the given connection (allows higher level socket code
 *   to make more fine-grained searches than account info - eg in IMAP we may
 *   wish to find a connection which is not in IMAP_SELECTED state) */
CONNECTION *mutt_conn_find (const CONNECTION * start, const ACCOUNT * account)
{
  CONNECTION *conn;
  ciss_url_t url;
  char hook[LONG_STRING];

  /* account isn't actually modified, since url isn't either */
  mutt_account_tourl ((ACCOUNT *) account, &url);
  url.path = NULL;
  url_ciss_tostring (&url, hook, sizeof (hook), 0);
  mutt_account_hook (hook);

  conn = start ? start->next : Connections;
  while (conn) {
    if (mutt_account_match (account, &(conn->account)))
      return conn;
    conn = conn->next;
  }

  conn = socket_new_conn ();
  memcpy (&conn->account, account, sizeof (ACCOUNT));

  conn->next = Connections;
  Connections = conn;

  if (Tunnel && *Tunnel)
    mutt_tunnel_socket_setup (conn);
  else if (account->flags & M_ACCT_SSL) {
#if defined (USE_SSL) || defined (USE_GNUTLS)
    if (mutt_ssl_socket_setup (conn) < 0) {
      mutt_socket_free (conn);
      return NULL;
    }
#else
    mutt_error _("SSL is unavailable.");

    mutt_sleep (2);
    mutt_socket_free (conn);

    return NULL;
#endif
  }
  else {
    conn->conn_read = raw_socket_read;
    conn->conn_write = raw_socket_write;
    conn->conn_open = raw_socket_open;
    conn->conn_close = raw_socket_close;
  }

  return conn;
}

static int socket_preconnect (void)
{
  int rc;
  int save_errno;

  if (str_len (Preconnect)) {
    debug_print (2, ("Executing preconnect: %s\n", Preconnect));
    rc = mutt_system (Preconnect);
    debug_print (2, ("Preconnect result: %d\n", rc));
    if (rc) {
      save_errno = errno;
      mutt_perror (_("Preconnect command failed."));
      mutt_sleep (1);

      return save_errno;
    }
  }

  return 0;
}

/* socket_connect: set up to connect to a socket fd. */
static int socket_connect (int fd, struct sockaddr *sa)
{
  int sa_size;
  int save_errno;

  if (sa->sa_family == AF_INET)
    sa_size = sizeof (struct sockaddr_in);
#ifdef HAVE_GETADDRINFO
  else if (sa->sa_family == AF_INET6)
    sa_size = sizeof (struct sockaddr_in6);
#endif
  else {
    debug_print (1, ("Unknown address family!\n"));
    return -1;
  }

  if (ConnectTimeout > 0)
    alarm (ConnectTimeout);

  mutt_allow_interrupt (1);

  save_errno = 0;

  if (connect (fd, sa, sa_size) < 0) {
    save_errno = errno;
    debug_print (2, ("Connection failed. errno: %d...\n", errno));
    SigInt = 0;                 /* reset in case we caught SIGINTR while in connect() */
  }

  if (ConnectTimeout > 0)
    alarm (0);
  mutt_allow_interrupt (0);

  return save_errno;
}

/* socket_new_conn: allocate and initialise a new connection. */
static CONNECTION *socket_new_conn (void)
{
  CONNECTION *conn;

  conn = (CONNECTION *) mem_calloc (1, sizeof (CONNECTION));
  conn->fd = -1;

  return conn;
}

int raw_socket_close (CONNECTION * conn)
{
  return close (conn->fd);
}

int raw_socket_read (CONNECTION * conn, char *buf, size_t len)
{
  int rc;

  if ((rc = read (conn->fd, buf, len)) == -1) {
    mutt_error (_("Error talking to %s (%s)"), conn->account.host,
                strerror (errno));
    mutt_sleep (2);
  }

  return rc;
}

int raw_socket_write (CONNECTION * conn, const char *buf, size_t count)
{
  int rc;

  if ((rc = write (conn->fd, buf, count)) == -1) {
    mutt_error (_("Error talking to %s (%s)"), conn->account.host,
                strerror (errno));
    mutt_sleep (2);
  }

  return rc;
}

int raw_socket_open (CONNECTION * conn)
{
  int rc;
  int fd;

  char *host_idna = NULL;

#ifdef HAVE_GETADDRINFO
/* --- IPv4/6 --- */

  /* "65536\0" */
  char port[6];
  struct addrinfo hints;
  struct addrinfo *res;
  struct addrinfo *cur;

  /* we accept v4 or v6 STREAM sockets */
  memset (&hints, 0, sizeof (hints));

  if (option (OPTUSEIPV6))
    hints.ai_family = AF_UNSPEC;
  else
    hints.ai_family = AF_INET;

  hints.ai_socktype = SOCK_STREAM;

  snprintf (port, sizeof (port), "%d", conn->account.port);

# ifdef HAVE_LIBIDN
  if (idna_to_ascii_lz (conn->account.host, &host_idna, 1) != IDNA_SUCCESS) {
    mutt_error (_("Bad IDN \"%s\"."), conn->account.host);
    return -1;
  }
# else
  host_idna = conn->account.host;
# endif

  mutt_message (_("Looking up %s..."), conn->account.host);


  rc = getaddrinfo (host_idna, port, &hints, &res);

# ifdef HAVE_LIBIDN
  mem_free (&host_idna);
# endif

  if (rc) {
    mutt_error (_("Could not find the host \"%s\""), conn->account.host);
    mutt_sleep (2);
    return -1;
  }

  mutt_message (_("Connecting to %s..."), conn->account.host);

  rc = -1;
  for (cur = res; cur != NULL; cur = cur->ai_next) {
    fd = socket (cur->ai_family, cur->ai_socktype, cur->ai_protocol);
    if (fd >= 0) {
      if ((rc = socket_connect (fd, cur->ai_addr)) == 0) {
        fcntl (fd, F_SETFD, FD_CLOEXEC);
        conn->fd = fd;
        break;
      }
      else
        close (fd);
    }
  }

  freeaddrinfo (res);

#else
  /* --- IPv4 only --- */

  struct sockaddr_in sin;
  struct hostent *he;
  int i;

  memset (&sin, 0, sizeof (sin));
  sin.sin_port = htons (conn->account.port);
  sin.sin_family = AF_INET;

# ifdef HAVE_LIBIDN
  if (idna_to_ascii_lz (conn->account.host, &host_idna, 1) != IDNA_SUCCESS) {
    mutt_error (_("Bad IDN \"%s\"."), conn->account.host);
    return -1;
  }
# else
  host_idna = conn->account.host;
# endif

  mutt_message (_("Looking up %s..."), conn->account.host);

  if ((he = gethostbyname (host_idna)) == NULL) {
# ifdef HAVE_LIBIDN
    mem_free (&host_idna);
# endif
    mutt_error (_("Could not find the host \"%s\""), conn->account.host);

    return -1;
  }

# ifdef HAVE_LIBIDN
  mem_free (&host_idna);
# endif

  mutt_message (_("Connecting to %s..."), conn->account.host);

  rc = -1;
  for (i = 0; he->h_addr_list[i] != NULL; i++) {
    memcpy (&sin.sin_addr, he->h_addr_list[i], he->h_length);
    fd = socket (PF_INET, SOCK_STREAM, IPPROTO_IP);

    if (fd >= 0) {
      if ((rc = socket_connect (fd, (struct sockaddr *) &sin)) == 0) {
        fcntl (fd, F_SETFD, FD_CLOEXEC);
        conn->fd = fd;
        break;
      }
      else
        close (fd);
    }
  }

#endif
  if (rc) {
    mutt_error (_("Could not connect to %s (%s)."), conn->account.host,
                (rc > 0) ? strerror (rc) : _("unknown error"));
    mutt_sleep (2);
    return -1;
  }

  return 0;
}


syntax highlighted by Code2HTML, v. 0.9.1