/* * ftp.c - ftp code * * Copyright (C) 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006 * Red Hat, Inc. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * Author(s): Erik Troan * Matt Wilson * Jeremy Katz * David Cantrell */ #define HAVE_ALLOCA_H 1 #define HAVE_NETINET_IN_SYSTM_H 1 #define HAVE_SYS_SOCKET_H 1 #if HAVE_ALLOCA_H # include #endif #if HAVE_SYS_SOCKET_H # include #endif #if HAVE_NETINET_IN_SYSTM_H # include # include #endif #if ! HAVE_HERRNO extern int h_errno; #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define TIMEOUT_SECS 60 #define BUFFER_SIZE 4096 #ifndef IPPORT_FTP # define IPPORT_FTP 21 #endif #include "ftp.h" #include "log.h" #include "net.h" static int ftpCheckResponse(int sock, char ** str); static int ftpCommand(int sock, char * command, ...); static int getHostAddress(const char * host, void * address, int family); static int ftpCheckResponse(int sock, char ** str) { static char buf[BUFFER_SIZE + 1]; int bufLength = 0; fd_set emptySet, readSet; char * chptr, * start; struct timeval timeout; int bytesRead, rc = 0; int doesContinue = 1; char errorCode[4]; errorCode[0] = '\0'; do { FD_ZERO(&emptySet); FD_ZERO(&readSet); FD_SET(sock, &readSet); timeout.tv_sec = TIMEOUT_SECS; timeout.tv_usec = 0; rc = select(sock + 1, &readSet, &emptySet, &emptySet, &timeout); if (rc < 1) { if (rc==0) return FTPERR_BAD_SERVER_RESPONSE; else rc = FTPERR_UNKNOWN; } else { rc = 0; } bytesRead = read(sock, buf + bufLength, sizeof(buf) - bufLength - 1); bufLength += bytesRead; buf[bufLength] = '\0'; /* divide the response into lines, checking each one to see if we are finished or need to continue */ start = chptr = buf; do { while (*chptr != '\n' && *chptr) chptr++; if (*chptr == '\n') { *chptr = '\0'; if (*(chptr - 1) == '\r') *(chptr - 1) = '\0'; if (str) *str = start; if (errorCode[0]) { if (!strncmp(start, errorCode, 3) && start[3] == ' ') doesContinue = 0; } else { strncpy(errorCode, start, 3); errorCode[3] = '\0'; if (start[3] != '-') { doesContinue = 0; } } start = chptr + 1; chptr++; } else { chptr++; } } while (*chptr); if (doesContinue && chptr > start) { memcpy(buf, start, chptr - start - 1); bufLength = chptr - start - 1; } else { bufLength = 0; } } while (doesContinue && !rc); if (*errorCode == '4' || *errorCode == '5') { if (!strncmp(errorCode, "421", 3)) { return FTPERR_TOO_MANY_CONNECTIONS; } else if (!strncmp(errorCode, "550", 3)) { return FTPERR_FILE_NOT_FOUND; } return FTPERR_BAD_SERVER_RESPONSE; } if (rc) return rc; return 0; } int ftpCommand(int sock, char * command, ...) { va_list ap; int len; char * s; char * buf; int rc; va_start(ap, command); len = strlen(command) + 2; s = va_arg(ap, char *); while (s) { len += strlen(s) + 1; s = va_arg(ap, char *); } va_end(ap); buf = alloca(len + 1); va_start(ap, command); strcpy(buf, command); strcat(buf, " "); s = va_arg(ap, char *); while (s) { strcat(buf, s); strcat(buf, " "); s = va_arg(ap, char *); } va_end(ap); buf[len - 2] = '\r'; buf[len - 1] = '\n'; buf[len] = '\0'; if (write(sock, buf, len) != len) { return FTPERR_SERVER_IO_ERROR; } if ((rc = ftpCheckResponse(sock, NULL))) return rc; return 0; } static int getHostAddress(const char * host, void * address, int family) { char *hostname, *port; struct hostent *hostent; splitHostname((char *) host, &hostname, &port); if (family == AF_INET) { if (isdigit(host[0])) { if (inet_pton(AF_INET, hostname, (struct in_addr *)address) >= 1) { return 0; } else { return FTPERR_BAD_HOST_ADDR; } } else { if ((hostent = gethostbyname(hostname)) != NULL) { memcpy((struct in_addr *) address, hostent->h_addr_list[0], hostent->h_length); return 0; } else { errno = h_errno; return FTPERR_BAD_HOSTNAME; } } } else if (family == AF_INET6) { if (strchr(hostname, ':')) { if (inet_pton(AF_INET6, hostname, (struct in_addr6 *)address) >= 1) { return 0; } else return FTPERR_BAD_HOST_ADDR; } else { if ((hostent = gethostbyname(hostname)) != NULL) { memcpy((struct in_addr6 *) address, hostent->h_addr_list[0], hostent->h_length); return 0; } else { errno = h_errno; return FTPERR_BAD_HOSTNAME; } } } else { return FTPERR_UNSUPPORTED_FAMILY; } } int ftpOpen(char *host, int family, char *name, char *password, int port) { static int sock; struct in_addr addr; struct in6_addr addr6; struct sockaddr_in destPort; struct sockaddr_in6 destPort6; struct passwd * pw; int rc = 0; if (port < 0) port = IPPORT_FTP; if (!name) name = "anonymous"; if (!password) { password = "root@"; if (getuid()) { pw = getpwuid(getuid()); if (pw) { password = alloca(strlen(pw->pw_name) + 2); strcpy(password, pw->pw_name); strcat(password, "@"); } } } if (family == AF_INET) rc = getHostAddress(host, &addr, AF_INET); else if (family == AF_INET6) rc = getHostAddress(host, &addr6, AF_INET6); if (rc) return rc; sock = socket(family, SOCK_STREAM, IPPROTO_IP); if (sock < 0) { return FTPERR_FAILED_CONNECT; } if (family == AF_INET) { destPort.sin_family = family; destPort.sin_port = htons(port); destPort.sin_addr = addr; if (connect(sock, (struct sockaddr *) &destPort, sizeof(destPort))) { close(sock); return FTPERR_FAILED_CONNECT; } } else if (family == AF_INET6) { destPort6.sin6_family = family; destPort6.sin6_port = htons(port); destPort6.sin6_addr = addr6; if (connect(sock, (struct sockaddr *) &destPort6, sizeof(destPort6))) { close(sock); return FTPERR_FAILED_CONNECT; } } /* ftpCheckResponse() assumes the socket is nonblocking */ if (fcntl(sock, F_SETFL, O_NONBLOCK)) { close(sock); return FTPERR_FAILED_CONNECT; } if ((rc = ftpCheckResponse(sock, NULL))) { return rc; } if ((rc = ftpCommand(sock, "USER", name, NULL))) { close(sock); return rc; } if ((rc = ftpCommand(sock, "PASS", password, NULL))) { close(sock); return rc; } if ((rc = ftpCommand(sock, "TYPE", "I", NULL))) { close(sock); return rc; } return sock; } /* * FTP specification: * RFC 959 FILE TRANSFER PROTOCOL (FTP) * RFC 2428 FTP Extensions for IPv6 and NATs */ int ftpGetFileDesc(int sock, struct in6_addr host, int family, char * remotename, long long *size) { int dataSocket; struct sockaddr_in dataAddress; struct sockaddr_in6 dataAddress6; int i, j; char * passReply; char * chptr; char * sizeCommand; char * sizeReply; char * retrCommand; int rc; if (family == AF_INET) { if (write(sock, "PASV\r\n", 6) != 6) { return FTPERR_SERVER_IO_ERROR; } } else if (family == AF_INET6) { if (write(sock, "EPSV\r\n", 6) != 6) { return FTPERR_SERVER_IO_ERROR; } } if ((rc = ftpCheckResponse(sock, &passReply))) { return FTPERR_PASSIVE_ERROR; } /* get IP address and port number from server response */ if (family == AF_INET) { /* we have a PASV response of the form: * 227 Entering Passive Mode (209,132,176,30,57,229) * where 209.132.176.30 is the IP, and 57 & 229 are the ports */ chptr = passReply; while (*chptr && *chptr != '(') chptr++; if (*chptr != '(') { return FTPERR_PASSIVE_ERROR; } chptr++; passReply = chptr; while (*chptr && *chptr != ')') chptr++; if (*chptr != ')') { return FTPERR_PASSIVE_ERROR; } *chptr-- = '\0'; while (*chptr && *chptr != ',') chptr--; if (*chptr != ',') { return FTPERR_PASSIVE_ERROR; } chptr--; while (*chptr && *chptr != ',') chptr--; if (*chptr != ',') { return FTPERR_PASSIVE_ERROR; } *chptr++ = '\0'; /* now passReply points to the IP portion * and chptr points to the port number portion */ if (sscanf(chptr, "%d,%d", &i, &j) != 2) { return FTPERR_PASSIVE_ERROR; } } else if (family == AF_INET6) { /* we have an EPSV response of the form: * 229 Entering Extended Passive Mode (|||51626|) * where 51626 is the port */ chptr = passReply; while (*chptr && *chptr != '(') chptr++; if (*chptr != '(') { return FTPERR_PASSIVE_ERROR; } chptr++; while (*chptr && *chptr == '|') chptr++; passReply = chptr; while (*chptr && *chptr != '|') chptr++; *chptr = '\0'; chptr = passReply; /* now chptr contains our port number */ if (sscanf(chptr, "%d", &i) != 1) { return FTPERR_PASSIVE_ERROR; } } /* build our sockaddr */ if (family == AF_INET) { dataAddress.sin_family = family; dataAddress.sin_port = htons((i << 8) + j); /* passReply contains the IP address, but with commands insteaad of * periods, so change those */ chptr = passReply; while (*chptr++) { if (*chptr == ',') *chptr = '.'; } if (!inet_pton(family, passReply, &dataAddress.sin_addr)) { return FTPERR_PASSIVE_ERROR; } } else if (family == AF_INET6) { dataAddress6.sin6_family = family; dataAddress6.sin6_port = htons(i); /* we don't get this in an EPSV reply, but we got it as a param */ memset(&dataAddress6.sin6_addr, 0, sizeof(struct in6_addr)); memcpy(&dataAddress6.sin6_addr, &host, sizeof(host)); } dataSocket = socket(family, SOCK_STREAM, IPPROTO_IP); if (dataSocket < 0) { return FTPERR_FAILED_CONNECT; } sizeCommand = alloca(strlen(remotename) + 20); sprintf(sizeCommand, "SIZE %s\r\n", remotename); i = strlen(sizeCommand); if (write(sock, sizeCommand, i) != i) { return FTPERR_SERVER_IO_ERROR; } if (ftpCheckResponse(sock, &sizeReply)) { /* No worries, the SIZE command isn't in RFC 959 anyway. */ *size = 0; } else { /* We have a SIZE response of the form: * 213 95600640 * where 95600640 is the size in bytes. */ /* Skip to first non-space character */ while (isspace(*sizeReply) && *sizeReply) sizeReply++; /* Skip reply code */ while (!isspace(*sizeReply) && *sizeReply) sizeReply++; /* Skip any remaining whitespace */ while (isspace(*sizeReply) && *sizeReply) sizeReply++; /* sizeReply now points to the beginning of the size */ if (sscanf(sizeReply, "%lld", size) != 1) *size = 0; if (*size < 0) *size = 0; } retrCommand = alloca(strlen(remotename) + 20); sprintf(retrCommand, "RETR %s\r\n", remotename); i = strlen(retrCommand); if (write(sock, retrCommand, i) != i) { return FTPERR_SERVER_IO_ERROR; } if (family == AF_INET) { if (connect(dataSocket, (struct sockaddr *) &dataAddress, sizeof(dataAddress))) { close(dataSocket); return FTPERR_FAILED_DATA_CONNECT; } } else if (family == AF_INET6) { if (connect(dataSocket, (struct sockaddr *) &dataAddress6, sizeof(dataAddress6))) { close(dataSocket); return FTPERR_FAILED_DATA_CONNECT; } } if ((rc = ftpCheckResponse(sock, NULL))) { close(dataSocket); return rc; } return dataSocket; } int ftpGetFileDone(int sock) { if (ftpCheckResponse(sock, NULL)) { return FTPERR_BAD_SERVER_RESPONSE; } return 0; } const char *ftpStrerror(int errorNumber, urlprotocol protocol) { switch (errorNumber) { case FTPERR_PERMISSION_DENIED: return(protocol == URL_METHOD_FTP ? "FTP permission denied" : "HTTP permission denied"); case FTPERR_BAD_SERVER_RESPONSE: return(protocol == URL_METHOD_FTP ? "Bad FTP server response" : "Bad HTTP server response"); case FTPERR_SERVER_IO_ERROR: return(protocol == URL_METHOD_FTP ? "FTP IO error" : "HTTP IO error"); case FTPERR_SERVER_TIMEOUT: return(protocol == URL_METHOD_FTP ? "FTP server timeout" : "HTTP server timeout"); case FTPERR_BAD_HOST_ADDR: return(protocol == URL_METHOD_FTP ? "Unable to lookup FTP server host address" : "Unable to lookup HTTP server host address"); case FTPERR_BAD_HOSTNAME: return(protocol == URL_METHOD_FTP ? "Unable to lookup FTP server host name" : "Unable to lookup HTTP server host name"); case FTPERR_FAILED_CONNECT: return(protocol == URL_METHOD_FTP ? "Failed to connect to FTP server" : "Failed to connect to HTTP server"); case FTPERR_FAILED_DATA_CONNECT: return(protocol == URL_METHOD_FTP ? "Failed to establish data connection to FTP server" : "Failed to establish data connection to HTTP server"); case FTPERR_FILE_IO_ERROR: return("IO error to local file"); case FTPERR_PASSIVE_ERROR: return("Error setting remote server to passive mode"); case FTPERR_FILE_NOT_FOUND: return("File not found on server"); case FTPERR_TOO_MANY_CONNECTIONS: return(protocol == URL_METHOD_FTP ? "Too many connections to FTP server" : "Too many connections to HTTP server"); case FTPERR_UNSUPPORTED_FAMILY: return(protocol == URL_METHOD_FTP ? "Unsupported address family on FTP server" : "Unsupported address family on HTTP server"); case FTPERR_UNKNOWN: default: return("Unknown or unexpected error"); } } static int read_headers (char **headers, fd_set *readSet, int sock) { char *nextChar; struct timeval timeout; int n = 4096; int rc; *headers = malloc(n); nextChar = *headers; *nextChar = '\0'; while (!strstr(*headers, "\r\n\r\n")) { FD_ZERO(readSet); FD_SET(sock, readSet); timeout.tv_sec = TIMEOUT_SECS; timeout.tv_usec = 0; rc = select(sock + 1, readSet, NULL, NULL, &timeout); if (rc == 0) { close(sock); free(*headers); *headers = NULL; return FTPERR_SERVER_TIMEOUT; } else if (rc < 0) { close(sock); free(*headers); *headers = NULL; return FTPERR_SERVER_IO_ERROR; } if (read(sock, nextChar, 1) != 1) { close(sock); free(*headers); *headers = NULL; return FTPERR_SERVER_IO_ERROR; } nextChar++; *nextChar = '\0'; if (nextChar - *headers == n) { n += 4096; *headers = realloc (*headers, sizeof(**headers)*n); } } return 0; } static char *find_header (char *headers, char *to_find) { char *start, *end, *searching_for, *retval; if (asprintf(&searching_for, "\r\n%s:", to_find) == -1) return NULL; if ((start = strstr(headers, searching_for)) == NULL) { free(searching_for); return NULL; } /* Trim off what we were searching for so we only return the value. */ start += strlen(searching_for); free(searching_for); while (isspace(*start) && *start) start++; if (start == NULL) return NULL; /* Now find the end of the header. */ end = strstr (start, "\r\n"); if (end == NULL) return NULL; retval = strndup (start, end-start); return retval; } static char *find_status_code (char *headers) { char *start, *end, *retval; start = headers; /* Skip ahead to the first whitespace in the header. */ while (!isspace(*start) && *start) start++; if (start == NULL) return NULL; /* Now skip over the whitespace. What's next is the status code number, * followed by a text description of the code. */ while (isspace(*start) && *start) start++; if (start == NULL) return NULL; if ((end = strstr (start, "\r\n")) == NULL) return NULL; retval = strndup (start, end-start); return retval; } /* extraHeaders is either NULL or a string with extra headers separated * by '\r\n', ending with '\r\n'. */ int httpGetFileDesc(char * hostname, int port, char * remotename, char *extraHeaders, long long *size) { char * buf, *headers = NULL; char *status; char *hstr; char *contlen; int family; struct in_addr addr; struct in6_addr addr6; int sock; int rc; struct sockaddr_in destPort; struct sockaddr_in6 destPort6; fd_set readSet; if (port < 0) port = 80; family = AF_INET; rc = getHostAddress(hostname, &addr, family); if (rc) { family = AF_INET6; rc = getHostAddress(hostname, &addr6, family); if (rc) return rc; } sock = socket(family, SOCK_STREAM, IPPROTO_IP); if (sock < 0) { return FTPERR_FAILED_CONNECT; } if (family == AF_INET) { destPort.sin_family = family; destPort.sin_port = htons(port); destPort.sin_addr = addr; if (connect(sock, (struct sockaddr *) &destPort, sizeof(destPort))) { close(sock); return FTPERR_FAILED_CONNECT; } } else if (family == AF_INET6) { destPort6.sin6_family = family; destPort6.sin6_port = htons(port); destPort6.sin6_addr = addr6; if (connect(sock, (struct sockaddr *) &destPort6, sizeof(destPort6))) { close(sock); return FTPERR_FAILED_CONNECT; } } if (extraHeaders) hstr = extraHeaders; else hstr = ""; buf = alloca(strlen(remotename) + strlen(hostname) + strlen(hstr) + 25); sprintf(buf, "GET %s HTTP/1.0\r\nHost: %s\r\n%s\r\n", remotename, hostname, hstr); rc = write(sock, buf, strlen(buf)); rc = read_headers (&headers, &readSet, sock); if (rc < 0) return rc; status = find_status_code (headers); if (status == NULL) { close(sock); if (status) free(status); if (headers) free(headers); return FTPERR_SERVER_IO_ERROR; } else if (!strncmp(status, "200", 3)) { contlen = find_header(headers, "Content-Length"); if (contlen == NULL) { *size = 0; } else { errno = 0; *size = strtoll(contlen, NULL, 10); if ((errno == ERANGE && (*size == LLONG_MIN || *size == LLONG_MAX)) || (errno != 0 && *size == 0)) { logMessage(ERROR, "%s: %d: %m", __func__, __LINE__); abort(); } } if (*size < 0) *size = 0; if (status) free(status); if (headers) free(headers); return sock; } else if (!strncmp(status, "301", 3) || !strncmp(status, "302", 3) || !strncmp(status, "303", 3) || !strncmp(status, "307", 3)) { struct iurlinfo ui; char *redir_loc = find_header (headers, "Location"); int retval; if (status) free(status); if (headers) free(headers); if (redir_loc == NULL) { logMessage(WARNING, "got a redirect response, but Location header is NULL"); close(sock); return FTPERR_FILE_NOT_FOUND; } logMessage(INFO, "redirecting to %s", redir_loc); convertURLToUI(redir_loc, &ui); retval = httpGetFileDesc (ui.address, -1, ui.prefix, extraHeaders, size); free(redir_loc); return retval; } else if (!strncmp(status, "403", 3)) { if (status) free(status); if (headers) free(headers); close(sock); return FTPERR_PERMISSION_DENIED; } else if (!strncmp(status, "404", 3)) { if (status) free(status); if (headers) free(headers); close(sock); return FTPERR_FILE_NOT_FOUND; } else { if (status) free(status); if (headers) free(headers); close(sock); logMessage(ERROR, "bad HTTP response code: %s", status); return FTPERR_BAD_SERVER_RESPONSE; } } /* vim:set shiftwidth=4 softtabstop=4: */