137535Sdes/*-
2236103Sdes * Copyright (c) 1998-2011 Dag-Erling Sm��rgrav
337535Sdes * All rights reserved.
437535Sdes *
537535Sdes * Redistribution and use in source and binary forms, with or without
637535Sdes * modification, are permitted provided that the following conditions
737535Sdes * are met:
837535Sdes * 1. Redistributions of source code must retain the above copyright
937535Sdes *    notice, this list of conditions and the following disclaimer
1037535Sdes *    in this position and unchanged.
1137535Sdes * 2. Redistributions in binary form must reproduce the above copyright
1237535Sdes *    notice, this list of conditions and the following disclaimer in the
1337535Sdes *    documentation and/or other materials provided with the distribution.
1437535Sdes * 3. The name of the author may not be used to endorse or promote products
1537535Sdes *    derived from this software without specific prior written permission
1637535Sdes *
1737535Sdes * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
1837535Sdes * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
1937535Sdes * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
2037535Sdes * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
2137535Sdes * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
2237535Sdes * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
2337535Sdes * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
2437535Sdes * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
2537535Sdes * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
2637535Sdes * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2737535Sdes */
2837535Sdes
2984203Sdillon#include <sys/cdefs.h>
3084203Sdillon__FBSDID("$FreeBSD$");
3184203Sdillon
3237535Sdes/*
3337571Sdes * Portions of this code were taken from or based on ftpio.c:
3437535Sdes *
3537535Sdes * ----------------------------------------------------------------------------
3637535Sdes * "THE BEER-WARE LICENSE" (Revision 42):
3793150Sphk * <phk@FreeBSD.org> wrote this file.  As long as you retain this notice you
3837535Sdes * can do whatever you want with this stuff. If we meet some day, and you think
3937535Sdes * this stuff is worth it, you can buy me a beer in return.   Poul-Henning Kamp
4037535Sdes * ----------------------------------------------------------------------------
4137535Sdes *
4237535Sdes * Major Changelog:
4337535Sdes *
44236103Sdes * Dag-Erling Sm��rgrav
4537535Sdes * 9 Jun 1998
4637535Sdes *
4737535Sdes * Incorporated into libfetch
4837535Sdes *
4937535Sdes * Jordan K. Hubbard
5037535Sdes * 17 Jan 1996
5137535Sdes *
5237535Sdes * Turned inside out. Now returns xfers as new file ids, not as a special
5337535Sdes * `state' of FTP_t
5437535Sdes *
5537535Sdes * $ftpioId: ftpio.c,v 1.30 1998/04/11 07:28:53 phk Exp $
5637535Sdes *
5737535Sdes */
5837535Sdes
5941862Sdes#include <sys/param.h>
6037535Sdes#include <sys/socket.h>
6137535Sdes#include <netinet/in.h>
6237535Sdes
6337535Sdes#include <ctype.h>
6475891Sarchie#include <err.h>
6555557Sdes#include <errno.h>
6667430Sdes#include <fcntl.h>
6760188Sdes#include <netdb.h>
6837573Sdes#include <stdarg.h>
6997856Sdes#include <stdint.h>
7037535Sdes#include <stdio.h>
7137571Sdes#include <stdlib.h>
7237535Sdes#include <string.h>
7341869Sdes#include <time.h>
7437571Sdes#include <unistd.h>
7537535Sdes
7637535Sdes#include "fetch.h"
7740939Sdes#include "common.h"
7841862Sdes#include "ftperr.h"
7937535Sdes
8070795Sdes#define FTP_ANONYMOUS_USER	"anonymous"
8137535Sdes
8264883Sdes#define FTP_CONNECTION_ALREADY_OPEN	125
8337573Sdes#define FTP_OPEN_DATA_CONNECTION	150
8437573Sdes#define FTP_OK				200
8541869Sdes#define FTP_FILE_STATUS			213
8641863Sdes#define FTP_SERVICE_READY		220
8767890Sdes#define FTP_TRANSFER_COMPLETE		226
8837573Sdes#define FTP_PASSIVE_MODE		227
8960737Sume#define FTP_LPASSIVE_MODE		228
9060737Sume#define FTP_EPASSIVE_MODE		229
9137573Sdes#define FTP_LOGGED_IN			230
9237573Sdes#define FTP_FILE_ACTION_OK		250
93148986Sdes#define FTP_DIRECTORY_CREATED		257 /* multiple meanings */
94148986Sdes#define FTP_FILE_CREATED		257 /* multiple meanings */
95148986Sdes#define FTP_WORKING_DIRECTORY		257 /* multiple meanings */
9637573Sdes#define FTP_NEED_PASSWORD		331
9737573Sdes#define FTP_NEED_ACCOUNT		332
9860188Sdes#define FTP_FILE_OK			350
9955557Sdes#define FTP_SYNTAX_ERROR		500
10063336Sdes#define FTP_PROTOCOL_ERROR		999
10137573Sdes
10240975Sdesstatic struct url cached_host;
10397856Sdesstatic conn_t	*cached_connection;
10437535Sdes
105174761Sdes#define isftpreply(foo)				\
106174761Sdes	(isdigit((unsigned char)foo[0]) &&	\
107174761Sdes	    isdigit((unsigned char)foo[1]) &&	\
108174761Sdes	    isdigit((unsigned char)foo[2]) &&	\
109174761Sdes	    (foo[3] == ' ' || foo[3] == '\0'))
110174761Sdes#define isftpinfo(foo) \
111174761Sdes	(isdigit((unsigned char)foo[0]) &&	\
112174761Sdes	    isdigit((unsigned char)foo[1]) &&	\
113174761Sdes	    isdigit((unsigned char)foo[2]) &&	\
114174761Sdes	    foo[3] == '-')
11555557Sdes
11690267Sdes/*
11790267Sdes * Translate IPv4 mapped IPv6 address to IPv4 address
11890267Sdes */
11960737Sumestatic void
12060737Sumeunmappedaddr(struct sockaddr_in6 *sin6)
12160737Sume{
12290267Sdes	struct sockaddr_in *sin4;
12390267Sdes	u_int32_t addr;
12490267Sdes	int port;
12560737Sume
12690267Sdes	if (sin6->sin6_family != AF_INET6 ||
12790267Sdes	    !IN6_IS_ADDR_V4MAPPED(&sin6->sin6_addr))
12890267Sdes		return;
12990267Sdes	sin4 = (struct sockaddr_in *)sin6;
130221822Sdes	addr = *(u_int32_t *)(uintptr_t)&sin6->sin6_addr.s6_addr[12];
13190267Sdes	port = sin6->sin6_port;
13290267Sdes	memset(sin4, 0, sizeof(struct sockaddr_in));
13390267Sdes	sin4->sin_addr.s_addr = addr;
13490267Sdes	sin4->sin_port = port;
13590267Sdes	sin4->sin_family = AF_INET;
13690267Sdes	sin4->sin_len = sizeof(struct sockaddr_in);
13760737Sume}
13860737Sume
13937571Sdes/*
14055557Sdes * Get server response
14137535Sdes */
14237535Sdesstatic int
143174588Sdesftp_chkerr(conn_t *conn)
14437535Sdes{
145174588Sdes	if (fetch_getln(conn) == -1) {
146174588Sdes		fetch_syserr();
14790267Sdes		return (-1);
14862215Sdes	}
14997856Sdes	if (isftpinfo(conn->buf)) {
15097856Sdes		while (conn->buflen && !isftpreply(conn->buf)) {
151174588Sdes			if (fetch_getln(conn) == -1) {
152174588Sdes				fetch_syserr();
15390267Sdes				return (-1);
15490267Sdes			}
15590267Sdes		}
15690267Sdes	}
15755557Sdes
158174761Sdes	while (conn->buflen &&
159174761Sdes	    isspace((unsigned char)conn->buf[conn->buflen - 1]))
16097856Sdes		conn->buflen--;
16197856Sdes	conn->buf[conn->buflen] = '\0';
16237535Sdes
16397856Sdes	if (!isftpreply(conn->buf)) {
164174588Sdes		ftp_seterr(FTP_PROTOCOL_ERROR);
16590267Sdes		return (-1);
16690267Sdes	}
16755557Sdes
16897856Sdes	conn->err = (conn->buf[0] - '0') * 100
16997856Sdes	    + (conn->buf[1] - '0') * 10
17097856Sdes	    + (conn->buf[2] - '0');
17190267Sdes
17297856Sdes	return (conn->err);
17337535Sdes}
17437535Sdes
17537535Sdes/*
17637573Sdes * Send a command and check reply
17737535Sdes */
17837535Sdesstatic int
179174588Sdesftp_cmd(conn_t *conn, const char *fmt, ...)
18037535Sdes{
18190267Sdes	va_list ap;
18290267Sdes	size_t len;
18390267Sdes	char *msg;
18490267Sdes	int r;
18537573Sdes
18690267Sdes	va_start(ap, fmt);
18790267Sdes	len = vasprintf(&msg, fmt, ap);
18890267Sdes	va_end(ap);
18990267Sdes
19090267Sdes	if (msg == NULL) {
19190267Sdes		errno = ENOMEM;
192174588Sdes		fetch_syserr();
19390267Sdes		return (-1);
19490267Sdes	}
19590267Sdes
196174588Sdes	r = fetch_putln(conn, msg, len);
19790267Sdes	free(msg);
19890267Sdes
19990267Sdes	if (r == -1) {
200174588Sdes		fetch_syserr();
20190267Sdes		return (-1);
20290267Sdes	}
20390267Sdes
204174588Sdes	return (ftp_chkerr(conn));
20537535Sdes}
20637535Sdes
20737535Sdes/*
20863340Sdes * Return a pointer to the filename part of a path
20963340Sdes */
21075891Sarchiestatic const char *
211174588Sdesftp_filename(const char *file, int *len, int *type)
21263340Sdes{
213148986Sdes	const char *s;
21490267Sdes
21590267Sdes	if ((s = strrchr(file, '/')) == NULL)
216148986Sdes		s = file;
21790267Sdes	else
218148986Sdes		s = s + 1;
219148986Sdes	*len = strlen(s);
220148986Sdes	if (*len > 7 && strncmp(s + *len - 7, ";type=", 6) == 0) {
221148986Sdes		*type = s[*len - 1];
222148986Sdes		*len -= 7;
223148986Sdes	} else {
224148986Sdes		*type = '\0';
225148986Sdes	}
226148986Sdes	return (s);
22763340Sdes}
22863340Sdes
22963340Sdes/*
230148986Sdes * Get current working directory from the reply to a CWD, PWD or CDUP
231148986Sdes * command.
232148986Sdes */
233148986Sdesstatic int
234174588Sdesftp_pwd(conn_t *conn, char *pwd, size_t pwdlen)
235148986Sdes{
236148986Sdes	char *src, *dst, *end;
237148986Sdes	int q;
238148986Sdes
239148986Sdes	if (conn->err != FTP_WORKING_DIRECTORY &&
240148986Sdes	    conn->err != FTP_FILE_ACTION_OK)
241148986Sdes		return (FTP_PROTOCOL_ERROR);
242148986Sdes	end = conn->buf + conn->buflen;
243148986Sdes	src = conn->buf + 4;
244148986Sdes	if (src >= end || *src++ != '"')
245148986Sdes		return (FTP_PROTOCOL_ERROR);
246148986Sdes	for (q = 0, dst = pwd; src < end && pwdlen--; ++src) {
247148986Sdes		if (!q && *src == '"')
248148986Sdes			q = 1;
249148986Sdes		else if (q && *src != '"')
250148986Sdes			break;
251148986Sdes		else if (q)
252148986Sdes			*dst++ = '"', q = 0;
253148986Sdes		else
254148986Sdes			*dst++ = *src;
255148986Sdes	}
256148986Sdes	if (!pwdlen)
257148986Sdes		return (FTP_PROTOCOL_ERROR);
258148986Sdes	*dst = '\0';
259148986Sdes#if 0
260148986Sdes	DEBUG(fprintf(stderr, "pwd: [%s]\n", pwd));
261148986Sdes#endif
262148986Sdes	return (FTP_OK);
263148986Sdes}
264148986Sdes
265148986Sdes/*
26690267Sdes * Change working directory to the directory that contains the specified
26790267Sdes * file.
26863340Sdes */
26963340Sdesstatic int
270174588Sdesftp_cwd(conn_t *conn, const char *file)
27163340Sdes{
272148986Sdes	const char *beg, *end;
273148986Sdes	char pwd[PATH_MAX];
274148986Sdes	int e, i, len;
27563340Sdes
276168960Snjl	/* If no slashes in name, no need to change dirs. */
277148986Sdes	if ((end = strrchr(file, '/')) == NULL)
278148986Sdes		return (0);
279174588Sdes	if ((e = ftp_cmd(conn, "PWD")) != FTP_WORKING_DIRECTORY ||
280174588Sdes	    (e = ftp_pwd(conn, pwd, sizeof(pwd))) != FTP_OK) {
281174588Sdes		ftp_seterr(e);
28290267Sdes		return (-1);
28390267Sdes	}
284148986Sdes	for (;;) {
285148986Sdes		len = strlen(pwd);
286168960Snjl
287168960Snjl		/* Look for a common prefix between PWD and dir to fetch. */
288148986Sdes		for (i = 0; i <= len && i <= end - file; ++i)
289148986Sdes			if (pwd[i] != file[i])
290148986Sdes				break;
291148986Sdes#if 0
292148986Sdes		DEBUG(fprintf(stderr, "have: [%.*s|%s]\n", i, pwd, pwd + i));
293148986Sdes		DEBUG(fprintf(stderr, "want: [%.*s|%s]\n", i, file, file + i));
294148986Sdes#endif
295168960Snjl		/* Keep going up a dir until we have a matching prefix. */
296148986Sdes		if (pwd[i] == '\0' && (file[i - 1] == '/' || file[i] == '/'))
297148986Sdes			break;
298174588Sdes		if ((e = ftp_cmd(conn, "CDUP")) != FTP_FILE_ACTION_OK ||
299174588Sdes		    (e = ftp_cmd(conn, "PWD")) != FTP_WORKING_DIRECTORY ||
300174588Sdes		    (e = ftp_pwd(conn, pwd, sizeof(pwd))) != FTP_OK) {
301174588Sdes			ftp_seterr(e);
302148986Sdes			return (-1);
303148986Sdes		}
304148986Sdes	}
305168960Snjl
306168960Snjl#ifdef FTP_COMBINE_CWDS
307168960Snjl	/* Skip leading slashes, even "////". */
308168960Snjl	for (beg = file + i; beg < end && *beg == '/'; ++beg, ++i)
309168960Snjl		/* nothing */ ;
310168960Snjl
311168960Snjl	/* If there is no trailing dir, we're already there. */
312168960Snjl	if (beg >= end)
313168960Snjl		return (0);
314168960Snjl
315168960Snjl	/* Change to the directory all in one chunk (e.g., foo/bar/baz). */
316174588Sdes	e = ftp_cmd(conn, "CWD %.*s", (int)(end - beg), beg);
317168960Snjl	if (e == FTP_FILE_ACTION_OK)
318168960Snjl		return (0);
319168960Snjl#endif /* FTP_COMBINE_CWDS */
320168960Snjl
321168960Snjl	/* That didn't work so go back to legacy behavior (multiple CWDs). */
322148986Sdes	for (beg = file + i; beg < end; beg = file + i + 1) {
323159565Sdes		while (*beg == '/')
324159565Sdes			++beg, ++i;
325148986Sdes		for (++i; file + i < end && file[i] != '/'; ++i)
326148986Sdes			/* nothing */ ;
327174588Sdes		e = ftp_cmd(conn, "CWD %.*s", file + i - beg, beg);
328148986Sdes		if (e != FTP_FILE_ACTION_OK) {
329174588Sdes			ftp_seterr(e);
330148986Sdes			return (-1);
331148986Sdes		}
332148986Sdes	}
33390267Sdes	return (0);
33463340Sdes}
33563340Sdes
33663340Sdes/*
337148986Sdes * Set transfer mode and data type
338148986Sdes */
339148986Sdesstatic int
340174588Sdesftp_mode_type(conn_t *conn, int mode, int type)
341148986Sdes{
342148986Sdes	int e;
343148986Sdes
344148986Sdes	switch (mode) {
345148986Sdes	case 0:
346148986Sdes	case 's':
347148986Sdes		mode = 'S';
348148986Sdes	case 'S':
349148986Sdes		break;
350148986Sdes	default:
351148986Sdes		return (FTP_PROTOCOL_ERROR);
352148986Sdes	}
353174588Sdes	if ((e = ftp_cmd(conn, "MODE %c", mode)) != FTP_OK) {
354154550Sdes		if (mode == 'S') {
355154550Sdes			/*
356154550Sdes			 * Stream mode is supposed to be the default - so
357154550Sdes			 * much so that some servers not only do not
358154550Sdes			 * support any other mode, but do not support the
359154550Sdes			 * MODE command at all.
360154550Sdes			 *
361154550Sdes			 * If "MODE S" fails, it is unlikely that we
362154550Sdes			 * previously succeeded in setting a different
363154550Sdes			 * mode.  Therefore, we simply hope that the
364154550Sdes			 * server is already in the correct mode, and
365154550Sdes			 * silently ignore the failure.
366154550Sdes			 */
367154550Sdes		} else {
368154550Sdes			return (e);
369154550Sdes		}
370154550Sdes	}
371148986Sdes
372148986Sdes	switch (type) {
373148986Sdes	case 0:
374148986Sdes	case 'i':
375148986Sdes		type = 'I';
376148986Sdes	case 'I':
377148986Sdes		break;
378148986Sdes	case 'a':
379148986Sdes		type = 'A';
380148986Sdes	case 'A':
381148986Sdes		break;
382148986Sdes	case 'd':
383148986Sdes		type = 'D';
384148986Sdes	case 'D':
385148986Sdes		/* can't handle yet */
386148986Sdes	default:
387148986Sdes		return (FTP_PROTOCOL_ERROR);
388148986Sdes	}
389174588Sdes	if ((e = ftp_cmd(conn, "TYPE %c", type)) != FTP_OK)
390148986Sdes		return (e);
391148986Sdes
392148986Sdes	return (FTP_OK);
393148986Sdes}
394148986Sdes
395148986Sdes/*
39663340Sdes * Request and parse file stats
39763340Sdes */
39863340Sdesstatic int
399174588Sdesftp_stat(conn_t *conn, const char *file, struct url_stat *us)
40063340Sdes{
40190267Sdes	char *ln;
402148986Sdes	const char *filename;
403148986Sdes	int filenamelen, type;
40490267Sdes	struct tm tm;
40590267Sdes	time_t t;
40690267Sdes	int e;
40763340Sdes
40875292Sdes	us->size = -1;
40990267Sdes	us->atime = us->mtime = 0;
41063340Sdes
411174588Sdes	filename = ftp_filename(file, &filenamelen, &type);
41290267Sdes
413174588Sdes	if ((e = ftp_mode_type(conn, 0, type)) != FTP_OK) {
414174588Sdes		ftp_seterr(e);
41590267Sdes		return (-1);
41690267Sdes	}
417148986Sdes
418174588Sdes	e = ftp_cmd(conn, "SIZE %.*s", filenamelen, filename);
419148986Sdes	if (e != FTP_FILE_STATUS) {
420174588Sdes		ftp_seterr(e);
421148986Sdes		return (-1);
422148986Sdes	}
423174761Sdes	for (ln = conn->buf + 4; *ln && isspace((unsigned char)*ln); ln++)
42490267Sdes		/* nothing */ ;
425174761Sdes	for (us->size = 0; *ln && isdigit((unsigned char)*ln); ln++)
42690267Sdes		us->size = us->size * 10 + *ln - '0';
427174761Sdes	if (*ln && !isspace((unsigned char)*ln)) {
428174588Sdes		ftp_seterr(FTP_PROTOCOL_ERROR);
42990267Sdes		us->size = -1;
43090267Sdes		return (-1);
43190267Sdes	}
43290267Sdes	if (us->size == 0)
43390267Sdes		us->size = -1;
43490267Sdes	DEBUG(fprintf(stderr, "size: [%lld]\n", (long long)us->size));
43590267Sdes
436174588Sdes	e = ftp_cmd(conn, "MDTM %.*s", filenamelen, filename);
437148986Sdes	if (e != FTP_FILE_STATUS) {
438174588Sdes		ftp_seterr(e);
43990267Sdes		return (-1);
44090267Sdes	}
441174761Sdes	for (ln = conn->buf + 4; *ln && isspace((unsigned char)*ln); ln++)
44290267Sdes		/* nothing */ ;
44390267Sdes	switch (strspn(ln, "0123456789")) {
44490267Sdes	case 14:
44590267Sdes		break;
44690267Sdes	case 15:
44790267Sdes		ln++;
44890267Sdes		ln[0] = '2';
44990267Sdes		ln[1] = '0';
45090267Sdes		break;
45190267Sdes	default:
452174588Sdes		ftp_seterr(FTP_PROTOCOL_ERROR);
45390267Sdes		return (-1);
45490267Sdes	}
45590267Sdes	if (sscanf(ln, "%04d%02d%02d%02d%02d%02d",
45690267Sdes	    &tm.tm_year, &tm.tm_mon, &tm.tm_mday,
45790267Sdes	    &tm.tm_hour, &tm.tm_min, &tm.tm_sec) != 6) {
458174588Sdes		ftp_seterr(FTP_PROTOCOL_ERROR);
45990267Sdes		return (-1);
46090267Sdes	}
46190267Sdes	tm.tm_mon--;
46290267Sdes	tm.tm_year -= 1900;
46390267Sdes	tm.tm_isdst = -1;
46490267Sdes	t = timegm(&tm);
46590267Sdes	if (t == (time_t)-1)
46690267Sdes		t = time(NULL);
46790267Sdes	us->mtime = t;
46890267Sdes	us->atime = t;
46990267Sdes	DEBUG(fprintf(stderr,
47090267Sdes	    "last modified: [%04d-%02d-%02d %02d:%02d:%02d]\n",
47190267Sdes	    tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday,
47290267Sdes	    tm.tm_hour, tm.tm_min, tm.tm_sec));
47390267Sdes	return (0);
47463340Sdes}
47563340Sdes
47663340Sdes/*
47767430Sdes * I/O functions for FTP
47867430Sdes */
47967430Sdesstruct ftpio {
48097866Sdes	conn_t	*cconn;		/* Control connection */
48197866Sdes	conn_t	*dconn;		/* Data connection */
48290267Sdes	int	 dir;		/* Direction */
48390267Sdes	int	 eof;		/* EOF reached */
48490267Sdes	int	 err;		/* Error code */
48567430Sdes};
48667430Sdes
487174588Sdesstatic int	 ftp_readfn(void *, char *, int);
488174588Sdesstatic int	 ftp_writefn(void *, const char *, int);
489174588Sdesstatic fpos_t	 ftp_seekfn(void *, fpos_t, int);
490174588Sdesstatic int	 ftp_closefn(void *);
49167430Sdes
49267430Sdesstatic int
493174588Sdesftp_readfn(void *v, char *buf, int len)
49467430Sdes{
49590267Sdes	struct ftpio *io;
49690267Sdes	int r;
49767430Sdes
49890267Sdes	io = (struct ftpio *)v;
49990267Sdes	if (io == NULL) {
50090267Sdes		errno = EBADF;
50190267Sdes		return (-1);
50290267Sdes	}
50397866Sdes	if (io->cconn == NULL || io->dconn == NULL || io->dir == O_WRONLY) {
50490267Sdes		errno = EBADF;
50590267Sdes		return (-1);
50690267Sdes	}
50790267Sdes	if (io->err) {
50890267Sdes		errno = io->err;
50990267Sdes		return (-1);
51090267Sdes	}
51190267Sdes	if (io->eof)
51290267Sdes		return (0);
513174588Sdes	r = fetch_read(io->dconn, buf, len);
51490267Sdes	if (r > 0)
51590267Sdes		return (r);
51690267Sdes	if (r == 0) {
51790267Sdes		io->eof = 1;
51890267Sdes		return (0);
51990267Sdes	}
52090267Sdes	if (errno != EINTR)
52190267Sdes		io->err = errno;
52290267Sdes	return (-1);
52367430Sdes}
52467430Sdes
52567430Sdesstatic int
526174588Sdesftp_writefn(void *v, const char *buf, int len)
52767430Sdes{
52890267Sdes	struct ftpio *io;
52990267Sdes	int w;
53090267Sdes
53190267Sdes	io = (struct ftpio *)v;
53290267Sdes	if (io == NULL) {
53390267Sdes		errno = EBADF;
53490267Sdes		return (-1);
53590267Sdes	}
53697866Sdes	if (io->cconn == NULL || io->dconn == NULL || io->dir == O_RDONLY) {
53790267Sdes		errno = EBADF;
53890267Sdes		return (-1);
53990267Sdes	}
54090267Sdes	if (io->err) {
54190267Sdes		errno = io->err;
54290267Sdes		return (-1);
54390267Sdes	}
544174588Sdes	w = fetch_write(io->dconn, buf, len);
54590267Sdes	if (w >= 0)
54690267Sdes		return (w);
54790267Sdes	if (errno != EINTR)
54890267Sdes		io->err = errno;
54990267Sdes	return (-1);
55067430Sdes}
55167430Sdes
55267430Sdesstatic fpos_t
553174588Sdesftp_seekfn(void *v, fpos_t pos __unused, int whence __unused)
55467430Sdes{
55590267Sdes	struct ftpio *io;
55690267Sdes
55790267Sdes	io = (struct ftpio *)v;
55890267Sdes	if (io == NULL) {
55990267Sdes		errno = EBADF;
56090267Sdes		return (-1);
56190267Sdes	}
56290267Sdes	errno = ESPIPE;
56390267Sdes	return (-1);
56467430Sdes}
56567430Sdes
56667430Sdesstatic int
567174588Sdesftp_closefn(void *v)
56867430Sdes{
56990267Sdes	struct ftpio *io;
57090267Sdes	int r;
57167430Sdes
57290267Sdes	io = (struct ftpio *)v;
57390267Sdes	if (io == NULL) {
57490267Sdes		errno = EBADF;
57590267Sdes		return (-1);
57690267Sdes	}
57790267Sdes	if (io->dir == -1)
57890267Sdes		return (0);
57997866Sdes	if (io->cconn == NULL || io->dconn == NULL) {
58090267Sdes		errno = EBADF;
58190267Sdes		return (-1);
58290267Sdes	}
583174588Sdes	fetch_close(io->dconn);
58490267Sdes	io->dir = -1;
58597866Sdes	io->dconn = NULL;
58690267Sdes	DEBUG(fprintf(stderr, "Waiting for final status\n"));
587174588Sdes	r = ftp_chkerr(io->cconn);
588105903Snjl	if (io->cconn == cached_connection && io->cconn->ref == 1)
589105903Snjl		cached_connection = NULL;
590174588Sdes	fetch_close(io->cconn);
59190267Sdes	free(io);
59290267Sdes	return (r == FTP_TRANSFER_COMPLETE) ? 0 : -1;
59367430Sdes}
59467430Sdes
59567430Sdesstatic FILE *
596174588Sdesftp_setup(conn_t *cconn, conn_t *dconn, int mode)
59767430Sdes{
59890267Sdes	struct ftpio *io;
59990267Sdes	FILE *f;
60067430Sdes
60197866Sdes	if (cconn == NULL || dconn == NULL)
60297866Sdes		return (NULL);
603109967Sdes	if ((io = malloc(sizeof(*io))) == NULL)
60490267Sdes		return (NULL);
60597866Sdes	io->cconn = cconn;
60697866Sdes	io->dconn = dconn;
60790267Sdes	io->dir = mode;
60890267Sdes	io->eof = io->err = 0;
609174588Sdes	f = funopen(io, ftp_readfn, ftp_writefn, ftp_seekfn, ftp_closefn);
61090267Sdes	if (f == NULL)
61190267Sdes		free(io);
61290267Sdes	return (f);
61367430Sdes}
61467430Sdes
61567430Sdes/*
61637608Sdes * Transfer file
61737535Sdes */
61837535Sdesstatic FILE *
619174588Sdesftp_transfer(conn_t *conn, const char *oper, const char *file,
62090267Sdes    int mode, off_t offset, const char *flags)
62137535Sdes{
62290267Sdes	struct sockaddr_storage sa;
62390267Sdes	struct sockaddr_in6 *sin6;
62490267Sdes	struct sockaddr_in *sin4;
625159566Sdes	const char *bindaddr;
626148986Sdes	const char *filename;
627148986Sdes	int filenamelen, type;
62890267Sdes	int low, pasv, verbose;
62990267Sdes	int e, sd = -1;
63090267Sdes	socklen_t l;
63190267Sdes	char *s;
63290267Sdes	FILE *df;
63355544Sdes
63490267Sdes	/* check flags */
63590267Sdes	low = CHECK_FLAG('l');
636226015Sdes	pasv = CHECK_FLAG('p') || !CHECK_FLAG('P');
63790267Sdes	verbose = CHECK_FLAG('v');
63855544Sdes
63990267Sdes	/* passive mode */
640226015Sdes	if ((s = getenv("FTP_PASSIVE_MODE")) != NULL)
641226015Sdes		pasv = (strncasecmp(s, "no", 2) != 0);
64260951Sdes
643148986Sdes	/* isolate filename */
644174588Sdes	filename = ftp_filename(file, &filenamelen, &type);
645148986Sdes
646148986Sdes	/* set transfer mode and data type */
647174588Sdes	if ((e = ftp_mode_type(conn, 0, type)) != FTP_OK)
648148986Sdes		goto ouch;
649148986Sdes
65090267Sdes	/* find our own address, bind, and listen */
651109967Sdes	l = sizeof(sa);
65297856Sdes	if (getsockname(conn->sd, (struct sockaddr *)&sa, &l) == -1)
65390267Sdes		goto sysouch;
65490267Sdes	if (sa.ss_family == AF_INET6)
65590267Sdes		unmappedaddr((struct sockaddr_in6 *)&sa);
65660737Sume
65790267Sdes	/* open data socket */
65890267Sdes	if ((sd = socket(sa.ss_family, SOCK_STREAM, IPPROTO_TCP)) == -1) {
659174588Sdes		fetch_syserr();
66090267Sdes		return (NULL);
66160737Sume	}
66237573Sdes
66390267Sdes	if (pasv) {
66490267Sdes		u_char addr[64];
66590267Sdes		char *ln, *p;
66690267Sdes		unsigned int i;
66790267Sdes		int port;
66837573Sdes
66990267Sdes		/* send PASV command */
67090267Sdes		if (verbose)
671174588Sdes			fetch_info("setting passive mode");
67290267Sdes		switch (sa.ss_family) {
67390267Sdes		case AF_INET:
674174588Sdes			if ((e = ftp_cmd(conn, "PASV")) != FTP_PASSIVE_MODE)
67590267Sdes				goto ouch;
67690267Sdes			break;
67790267Sdes		case AF_INET6:
678174588Sdes			if ((e = ftp_cmd(conn, "EPSV")) != FTP_EPASSIVE_MODE) {
67990267Sdes				if (e == -1)
68090267Sdes					goto ouch;
681174588Sdes				if ((e = ftp_cmd(conn, "LPSV")) !=
68297856Sdes				    FTP_LPASSIVE_MODE)
68390267Sdes					goto ouch;
68490267Sdes			}
68590267Sdes			break;
68690267Sdes		default:
68790267Sdes			e = FTP_PROTOCOL_ERROR; /* XXX: error code should be prepared */
68890267Sdes			goto ouch;
68990267Sdes		}
69037573Sdes
69190267Sdes		/*
69290267Sdes		 * Find address and port number. The reply to the PASV command
69390267Sdes		 * is IMHO the one and only weak point in the FTP protocol.
69490267Sdes		 */
69597856Sdes		ln = conn->buf;
69690267Sdes		switch (e) {
69790267Sdes		case FTP_PASSIVE_MODE:
69890267Sdes		case FTP_LPASSIVE_MODE:
699174761Sdes			for (p = ln + 3; *p && !isdigit((unsigned char)*p); p++)
70090267Sdes				/* nothing */ ;
70190267Sdes			if (!*p) {
70290267Sdes				e = FTP_PROTOCOL_ERROR;
70390267Sdes				goto ouch;
70490267Sdes			}
70590267Sdes			l = (e == FTP_PASSIVE_MODE ? 6 : 21);
70690267Sdes			for (i = 0; *p && i < l; i++, p++)
70790267Sdes				addr[i] = strtol(p, &p, 10);
70890267Sdes			if (i < l) {
70990267Sdes				e = FTP_PROTOCOL_ERROR;
71090267Sdes				goto ouch;
71190267Sdes			}
71290267Sdes			break;
71390267Sdes		case FTP_EPASSIVE_MODE:
71490267Sdes			for (p = ln + 3; *p && *p != '('; p++)
71590267Sdes				/* nothing */ ;
71690267Sdes			if (!*p) {
71790267Sdes				e = FTP_PROTOCOL_ERROR;
71890267Sdes				goto ouch;
71990267Sdes			}
72090267Sdes			++p;
72190267Sdes			if (sscanf(p, "%c%c%c%d%c", &addr[0], &addr[1], &addr[2],
72290267Sdes				&port, &addr[3]) != 5 ||
72390267Sdes			    addr[0] != addr[1] ||
72490267Sdes			    addr[0] != addr[2] || addr[0] != addr[3]) {
72590267Sdes				e = FTP_PROTOCOL_ERROR;
72690267Sdes				goto ouch;
72790267Sdes			}
72890267Sdes			break;
72990267Sdes		}
73060188Sdes
73190267Sdes		/* seek to required offset */
73290267Sdes		if (offset)
733174588Sdes			if (ftp_cmd(conn, "REST %lu", (u_long)offset) != FTP_FILE_OK)
73490267Sdes				goto sysouch;
73590267Sdes
73690267Sdes		/* construct sockaddr for data socket */
737109967Sdes		l = sizeof(sa);
73897856Sdes		if (getpeername(conn->sd, (struct sockaddr *)&sa, &l) == -1)
73990267Sdes			goto sysouch;
74090267Sdes		if (sa.ss_family == AF_INET6)
74190267Sdes			unmappedaddr((struct sockaddr_in6 *)&sa);
74290267Sdes		switch (sa.ss_family) {
74390267Sdes		case AF_INET6:
74490267Sdes			sin6 = (struct sockaddr_in6 *)&sa;
74590267Sdes			if (e == FTP_EPASSIVE_MODE)
74690267Sdes				sin6->sin6_port = htons(port);
74790267Sdes			else {
748176105Sdes				memcpy(&sin6->sin6_addr, addr + 2, 16);
749176105Sdes				memcpy(&sin6->sin6_port, addr + 19, 2);
75090267Sdes			}
75190267Sdes			break;
75290267Sdes		case AF_INET:
75390267Sdes			sin4 = (struct sockaddr_in *)&sa;
75490267Sdes			if (e == FTP_EPASSIVE_MODE)
75590267Sdes				sin4->sin_port = htons(port);
75690267Sdes			else {
757176105Sdes				memcpy(&sin4->sin_addr, addr, 4);
758176105Sdes				memcpy(&sin4->sin_port, addr + 4, 2);
75990267Sdes			}
76090267Sdes			break;
76190267Sdes		default:
76290267Sdes			e = FTP_PROTOCOL_ERROR; /* XXX: error code should be prepared */
76390267Sdes			break;
76490267Sdes		}
76590267Sdes
76690267Sdes		/* connect to data port */
76790267Sdes		if (verbose)
768174588Sdes			fetch_info("opening data connection");
769159566Sdes		bindaddr = getenv("FETCH_BIND_ADDRESS");
770159566Sdes		if (bindaddr != NULL && *bindaddr != '\0' &&
771174588Sdes		    fetch_bind(sd, sa.ss_family, bindaddr) != 0)
772159566Sdes			goto sysouch;
77390267Sdes		if (connect(sd, (struct sockaddr *)&sa, sa.ss_len) == -1)
77490267Sdes			goto sysouch;
77590267Sdes
77690267Sdes		/* make the server initiate the transfer */
77790267Sdes		if (verbose)
778174588Sdes			fetch_info("initiating transfer");
779174588Sdes		e = ftp_cmd(conn, "%s %.*s", oper, filenamelen, filename);
78090267Sdes		if (e != FTP_CONNECTION_ALREADY_OPEN && e != FTP_OPEN_DATA_CONNECTION)
78190267Sdes			goto ouch;
78290267Sdes
78390267Sdes	} else {
78490267Sdes		u_int32_t a;
78590267Sdes		u_short p;
78690267Sdes		int arg, d;
78790267Sdes		char *ap;
78890267Sdes		char hname[INET6_ADDRSTRLEN];
78990267Sdes
79090267Sdes		switch (sa.ss_family) {
79190267Sdes		case AF_INET6:
79290267Sdes			((struct sockaddr_in6 *)&sa)->sin6_port = 0;
79360737Sume#ifdef IPV6_PORTRANGE
79490267Sdes			arg = low ? IPV6_PORTRANGE_DEFAULT : IPV6_PORTRANGE_HIGH;
79590267Sdes			if (setsockopt(sd, IPPROTO_IPV6, IPV6_PORTRANGE,
79690267Sdes				(char *)&arg, sizeof(arg)) == -1)
79790267Sdes				goto sysouch;
79860737Sume#endif
79990267Sdes			break;
80090267Sdes		case AF_INET:
80190267Sdes			((struct sockaddr_in *)&sa)->sin_port = 0;
80290267Sdes			arg = low ? IP_PORTRANGE_DEFAULT : IP_PORTRANGE_HIGH;
80390267Sdes			if (setsockopt(sd, IPPROTO_IP, IP_PORTRANGE,
804109967Sdes				(char *)&arg, sizeof(arg)) == -1)
80590267Sdes				goto sysouch;
80690267Sdes			break;
80790267Sdes		}
80890267Sdes		if (verbose)
809174588Sdes			fetch_info("binding data socket");
81090267Sdes		if (bind(sd, (struct sockaddr *)&sa, sa.ss_len) == -1)
81190267Sdes			goto sysouch;
81290267Sdes		if (listen(sd, 1) == -1)
81390267Sdes			goto sysouch;
81437573Sdes
81590267Sdes		/* find what port we're on and tell the server */
81690267Sdes		if (getsockname(sd, (struct sockaddr *)&sa, &l) == -1)
81790267Sdes			goto sysouch;
81890267Sdes		switch (sa.ss_family) {
81990267Sdes		case AF_INET:
82090267Sdes			sin4 = (struct sockaddr_in *)&sa;
82190267Sdes			a = ntohl(sin4->sin_addr.s_addr);
82290267Sdes			p = ntohs(sin4->sin_port);
823174588Sdes			e = ftp_cmd(conn, "PORT %d,%d,%d,%d,%d,%d",
82490267Sdes			    (a >> 24) & 0xff, (a >> 16) & 0xff,
82590267Sdes			    (a >> 8) & 0xff, a & 0xff,
82690267Sdes			    (p >> 8) & 0xff, p & 0xff);
82790267Sdes			break;
82890267Sdes		case AF_INET6:
82960737Sume#define UC(b)	(((int)b)&0xff)
83090267Sdes			e = -1;
83190267Sdes			sin6 = (struct sockaddr_in6 *)&sa;
83299253Sume			sin6->sin6_scope_id = 0;
83390267Sdes			if (getnameinfo((struct sockaddr *)&sa, sa.ss_len,
83490267Sdes				hname, sizeof(hname),
83590267Sdes				NULL, 0, NI_NUMERICHOST) == 0) {
836174588Sdes				e = ftp_cmd(conn, "EPRT |%d|%s|%d|", 2, hname,
83790267Sdes				    htons(sin6->sin6_port));
83890267Sdes				if (e == -1)
83990267Sdes					goto ouch;
84090267Sdes			}
84190267Sdes			if (e != FTP_OK) {
84290267Sdes				ap = (char *)&sin6->sin6_addr;
843174588Sdes				e = ftp_cmd(conn,
84490267Sdes				    "LPRT %d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d",
84590267Sdes				    6, 16,
84690267Sdes				    UC(ap[0]), UC(ap[1]), UC(ap[2]), UC(ap[3]),
84790267Sdes				    UC(ap[4]), UC(ap[5]), UC(ap[6]), UC(ap[7]),
84890267Sdes				    UC(ap[8]), UC(ap[9]), UC(ap[10]), UC(ap[11]),
84990267Sdes				    UC(ap[12]), UC(ap[13]), UC(ap[14]), UC(ap[15]),
85090267Sdes				    2,
85190267Sdes				    (ntohs(sin6->sin6_port) >> 8) & 0xff,
85290267Sdes				    ntohs(sin6->sin6_port)        & 0xff);
85390267Sdes			}
85490267Sdes			break;
85590267Sdes		default:
85690267Sdes			e = FTP_PROTOCOL_ERROR; /* XXX: error code should be prepared */
85790267Sdes			goto ouch;
85890267Sdes		}
85990267Sdes		if (e != FTP_OK)
86090267Sdes			goto ouch;
86190267Sdes
86290267Sdes		/* seek to required offset */
86390267Sdes		if (offset)
864174588Sdes			if (ftp_cmd(conn, "REST %ju", (uintmax_t)offset) != FTP_FILE_OK)
86590267Sdes				goto sysouch;
86690267Sdes
86790267Sdes		/* make the server initiate the transfer */
86890267Sdes		if (verbose)
869174588Sdes			fetch_info("initiating transfer");
870174588Sdes		e = ftp_cmd(conn, "%s %.*s", oper, filenamelen, filename);
871119123Sdes		if (e != FTP_CONNECTION_ALREADY_OPEN && e != FTP_OPEN_DATA_CONNECTION)
87290267Sdes			goto ouch;
87390267Sdes
87490267Sdes		/* accept the incoming connection and go to town */
87590267Sdes		if ((d = accept(sd, NULL, NULL)) == -1)
87690267Sdes			goto sysouch;
87790267Sdes		close(sd);
87890267Sdes		sd = d;
87960737Sume	}
88037573Sdes
881174588Sdes	if ((df = ftp_setup(conn, fetch_reopen(sd), mode)) == NULL)
88262256Sdes		goto sysouch;
88390267Sdes	return (df);
88437573Sdes
88537573Sdessysouch:
886174588Sdes	fetch_syserr();
88790267Sdes	if (sd >= 0)
88890267Sdes		close(sd);
88990267Sdes	return (NULL);
89041869Sdes
89137573Sdesouch:
89290267Sdes	if (e != -1)
893174588Sdes		ftp_seterr(e);
89490267Sdes	if (sd >= 0)
89590267Sdes		close(sd);
89690267Sdes	return (NULL);
89737535Sdes}
89837535Sdes
89937571Sdes/*
90077238Sdes * Authenticate
90177238Sdes */
90277238Sdesstatic int
903174588Sdesftp_authenticate(conn_t *conn, struct url *url, struct url *purl)
90477238Sdes{
90590267Sdes	const char *user, *pwd, *logname;
90690267Sdes	char pbuf[MAXHOSTNAMELEN + MAXLOGNAME + 1];
90790267Sdes	int e, len;
90877238Sdes
90990267Sdes	/* XXX FTP_AUTH, and maybe .netrc */
91090267Sdes
91190267Sdes	/* send user name and password */
912109697Sdes	if (url->user[0] == '\0')
913174588Sdes		fetch_netrc_auth(url);
91490267Sdes	user = url->user;
915109697Sdes	if (*user == '\0')
91690267Sdes		user = getenv("FTP_LOGIN");
917109697Sdes	if (user == NULL || *user == '\0')
91890267Sdes		user = FTP_ANONYMOUS_USER;
919174588Sdes	if (purl && url->port == fetch_default_port(url->scheme))
920174588Sdes		e = ftp_cmd(conn, "USER %s@%s", user, url->host);
92190267Sdes	else if (purl)
922174588Sdes		e = ftp_cmd(conn, "USER %s@%s@%d", user, url->host, url->port);
92390267Sdes	else
924174588Sdes		e = ftp_cmd(conn, "USER %s", user);
92590267Sdes
92690267Sdes	/* did the server request a password? */
92790267Sdes	if (e == FTP_NEED_PASSWORD) {
92890267Sdes		pwd = url->pwd;
929109697Sdes		if (*pwd == '\0')
93090267Sdes			pwd = getenv("FTP_PASSWORD");
931109697Sdes		if (pwd == NULL || *pwd == '\0') {
93290267Sdes			if ((logname = getlogin()) == 0)
93390267Sdes				logname = FTP_ANONYMOUS_USER;
93490267Sdes			if ((len = snprintf(pbuf, MAXLOGNAME + 1, "%s@", logname)) < 0)
93590267Sdes				len = 0;
93690267Sdes			else if (len > MAXLOGNAME)
93790267Sdes				len = MAXLOGNAME;
938109967Sdes			gethostname(pbuf + len, sizeof(pbuf) - len);
93990267Sdes			pwd = pbuf;
94090267Sdes		}
941174588Sdes		e = ftp_cmd(conn, "PASS %s", pwd);
94277238Sdes	}
94390267Sdes
94490267Sdes	return (e);
94577238Sdes}
94677238Sdes
94777238Sdes/*
94837571Sdes * Log on to FTP server
94937535Sdes */
95097856Sdesstatic conn_t *
951174588Sdesftp_connect(struct url *url, struct url *purl, const char *flags)
95237571Sdes{
95397856Sdes	conn_t *conn;
95497856Sdes	int e, direct, verbose;
95560737Sume#ifdef INET6
95690267Sdes	int af = AF_UNSPEC;
95760737Sume#else
95890267Sdes	int af = AF_INET;
95960737Sume#endif
96037571Sdes
96190267Sdes	direct = CHECK_FLAG('d');
96290267Sdes	verbose = CHECK_FLAG('v');
96390267Sdes	if (CHECK_FLAG('4'))
96490267Sdes		af = AF_INET;
96590267Sdes	else if (CHECK_FLAG('6'))
96690267Sdes		af = AF_INET6;
96760737Sume
96890267Sdes	if (direct)
96990267Sdes		purl = NULL;
97037608Sdes
97190267Sdes	/* check for proxy */
97290267Sdes	if (purl) {
97390267Sdes		/* XXX proxy authentication! */
974174588Sdes		conn = fetch_connect(purl->host, purl->port, af, verbose);
97590267Sdes	} else {
97690267Sdes		/* no proxy, go straight to target */
977174588Sdes		conn = fetch_connect(url->host, url->port, af, verbose);
97890267Sdes		purl = NULL;
97990267Sdes	}
98037608Sdes
98190267Sdes	/* check connection */
982103459Sfenner	if (conn == NULL)
983174588Sdes		/* fetch_connect() has already set an error code */
98490267Sdes		return (NULL);
98590267Sdes
98690267Sdes	/* expect welcome message */
987174588Sdes	if ((e = ftp_chkerr(conn)) != FTP_SERVICE_READY)
98890267Sdes		goto fouch;
98990267Sdes
99090267Sdes	/* authenticate */
991174588Sdes	if ((e = ftp_authenticate(conn, url, purl)) != FTP_LOGGED_IN)
99290267Sdes		goto fouch;
99390267Sdes
994168960Snjl	/* TODO: Request extended features supported, if any (RFC 3659). */
995168960Snjl
99690267Sdes	/* done */
99797856Sdes	return (conn);
99890267Sdes
99937571Sdesfouch:
100090267Sdes	if (e != -1)
1001174588Sdes		ftp_seterr(e);
1002174588Sdes	fetch_close(conn);
100390267Sdes	return (NULL);
100437571Sdes}
100537571Sdes
100637571Sdes/*
100737571Sdes * Disconnect from server
100837571Sdes */
100937571Sdesstatic void
1010174588Sdesftp_disconnect(conn_t *conn)
101137571Sdes{
1012174588Sdes	(void)ftp_cmd(conn, "QUIT");
1013105903Snjl	if (conn == cached_connection && conn->ref == 1)
1014105903Snjl		cached_connection = NULL;
1015174588Sdes	fetch_close(conn);
101637571Sdes}
101737571Sdes
101837571Sdes/*
101937571Sdes * Check if we're already connected
102037571Sdes */
102137571Sdesstatic int
1022174588Sdesftp_isconnected(struct url *url)
102337571Sdes{
102497856Sdes	return (cached_connection
102537571Sdes	    && (strcmp(url->host, cached_host.host) == 0)
102637571Sdes	    && (strcmp(url->user, cached_host.user) == 0)
102737571Sdes	    && (strcmp(url->pwd, cached_host.pwd) == 0)
102837571Sdes	    && (url->port == cached_host.port));
102937571Sdes}
103037571Sdes
103137608Sdes/*
103241869Sdes * Check the cache, reconnect if no luck
103337608Sdes */
103497856Sdesstatic conn_t *
1035174588Sdesftp_cached_connect(struct url *url, struct url *purl, const char *flags)
103637535Sdes{
103797856Sdes	conn_t *conn;
103897856Sdes	int e;
103937535Sdes
104090267Sdes	/* set default port */
104190267Sdes	if (!url->port)
1042174588Sdes		url->port = fetch_default_port(url->scheme);
104390267Sdes
104490267Sdes	/* try to use previously cached connection */
1045174588Sdes	if (ftp_isconnected(url)) {
1046174588Sdes		e = ftp_cmd(cached_connection, "NOOP");
104790267Sdes		if (e == FTP_OK || e == FTP_SYNTAX_ERROR)
1048174588Sdes			return (fetch_ref(cached_connection));
104990267Sdes	}
105090267Sdes
105190267Sdes	/* connect to server */
1052174588Sdes	if ((conn = ftp_connect(url, purl, flags)) == NULL)
105397856Sdes		return (NULL);
105497856Sdes	if (cached_connection)
1055174588Sdes		ftp_disconnect(cached_connection);
1056174588Sdes	cached_connection = fetch_ref(conn);
1057109967Sdes	memcpy(&cached_host, url, sizeof(*url));
105897856Sdes	return (conn);
105937535Sdes}
106037535Sdes
106137571Sdes/*
106267043Sdes * Check the proxy settings
106363713Sdes */
106467043Sdesstatic struct url *
1065174752Sdesftp_get_proxy(struct url * url, const char *flags)
106663713Sdes{
106790267Sdes	struct url *purl;
106890267Sdes	char *p;
106990267Sdes
1070112420Smtm	if (flags != NULL && strchr(flags, 'd') != NULL)
1071112081Sdes		return (NULL);
1072174752Sdes	if (fetch_no_proxy_match(url->host))
1073174752Sdes		return (NULL);
107490267Sdes	if (((p = getenv("FTP_PROXY")) || (p = getenv("ftp_proxy")) ||
107590267Sdes		(p = getenv("HTTP_PROXY")) || (p = getenv("http_proxy"))) &&
107690267Sdes	    *p && (purl = fetchParseURL(p)) != NULL) {
107790267Sdes		if (!*purl->scheme) {
107890267Sdes			if (getenv("FTP_PROXY") || getenv("ftp_proxy"))
107990267Sdes				strcpy(purl->scheme, SCHEME_FTP);
108090267Sdes			else
108190267Sdes				strcpy(purl->scheme, SCHEME_HTTP);
108290267Sdes		}
108390267Sdes		if (!purl->port)
1084174588Sdes			purl->port = fetch_default_proxy_port(purl->scheme);
108590267Sdes		if (strcasecmp(purl->scheme, SCHEME_FTP) == 0 ||
108690267Sdes		    strcasecmp(purl->scheme, SCHEME_HTTP) == 0)
108790267Sdes			return (purl);
108890267Sdes		fetchFreeURL(purl);
108969272Sdes	}
109090267Sdes	return (NULL);
109163713Sdes}
109263713Sdes
109363713Sdes/*
109487315Sdes * Process an FTP request
109537571Sdes */
109637535SdesFILE *
1097174588Sdesftp_request(struct url *url, const char *op, struct url_stat *us,
109890267Sdes    struct url *purl, const char *flags)
109937608Sdes{
110097856Sdes	conn_t *conn;
110197856Sdes	int oflag;
110290267Sdes
110390267Sdes	/* check if we should use HTTP instead */
110490267Sdes	if (purl && strcasecmp(purl->scheme, SCHEME_HTTP) == 0) {
110590267Sdes		if (strcmp(op, "STAT") == 0)
1106174588Sdes			return (http_request(url, "HEAD", us, purl, flags));
110790267Sdes		else if (strcmp(op, "RETR") == 0)
1108174588Sdes			return (http_request(url, "GET", us, purl, flags));
110990267Sdes		/*
111090267Sdes		 * Our HTTP code doesn't support PUT requests yet, so try
111190267Sdes		 * a direct connection.
111290267Sdes		 */
111390267Sdes	}
111490267Sdes
111590267Sdes	/* connect to server */
1116174588Sdes	conn = ftp_cached_connect(url, purl, flags);
111790267Sdes	if (purl)
111890267Sdes		fetchFreeURL(purl);
111997856Sdes	if (conn == NULL)
112090267Sdes		return (NULL);
112190267Sdes
112290267Sdes	/* change directory */
1123174588Sdes	if (ftp_cwd(conn, url->doc) == -1)
1124199801Sattilio		goto errsock;
112590267Sdes
112690267Sdes	/* stat file */
1127174588Sdes	if (us && ftp_stat(conn, url->doc, us) == -1
112890267Sdes	    && fetchLastErrCode != FETCH_PROTO
112990267Sdes	    && fetchLastErrCode != FETCH_UNAVAIL)
1130199801Sattilio		goto errsock;
113190267Sdes
113290267Sdes	/* just a stat */
1133199801Sattilio	if (strcmp(op, "STAT") == 0) {
1134217505Sdes		--conn->ref;
1135199801Sattilio		ftp_disconnect(conn);
113690267Sdes		return (FILE *)1; /* bogus return value */
1137199801Sattilio	}
113890267Sdes	if (strcmp(op, "STOR") == 0 || strcmp(op, "APPE") == 0)
113990267Sdes		oflag = O_WRONLY;
114090267Sdes	else
114190267Sdes		oflag = O_RDONLY;
114287315Sdes
114390267Sdes	/* initiate the transfer */
1144174588Sdes	return (ftp_transfer(conn, op, url->doc, oflag, url->offset, flags));
1145199801Sattilio
1146199801Sattilioerrsock:
1147199801Sattilio	ftp_disconnect(conn);
1148199801Sattilio	return (NULL);
114937608Sdes}
115037608Sdes
115141869Sdes/*
115287315Sdes * Get and stat file
115387315Sdes */
115487315SdesFILE *
115587315SdesfetchXGetFTP(struct url *url, struct url_stat *us, const char *flags)
115687315Sdes{
1157174752Sdes	return (ftp_request(url, "RETR", us, ftp_get_proxy(url, flags), flags));
115887315Sdes}
115987315Sdes
116087315Sdes/*
116163340Sdes * Get file
116263340Sdes */
116363340SdesFILE *
116475891SarchiefetchGetFTP(struct url *url, const char *flags)
116563340Sdes{
116690267Sdes	return (fetchXGetFTP(url, NULL, flags));
116763340Sdes}
116863340Sdes
116963340Sdes/*
117041869Sdes * Put file
117141869Sdes */
117237608SdesFILE *
117375891SarchiefetchPutFTP(struct url *url, const char *flags)
117437535Sdes{
1175174588Sdes	return (ftp_request(url, CHECK_FLAG('a') ? "APPE" : "STOR", NULL,
1176174752Sdes	    ftp_get_proxy(url, flags), flags));
117737535Sdes}
117840975Sdes
117941869Sdes/*
118041869Sdes * Get file stats
118141869Sdes */
118240975Sdesint
118375891SarchiefetchStatFTP(struct url *url, struct url_stat *us, const char *flags)
118440975Sdes{
1185112081Sdes	FILE *f;
118690267Sdes
1187174752Sdes	f = ftp_request(url, "STAT", us, ftp_get_proxy(url, flags), flags);
1188112081Sdes	if (f == NULL)
118990267Sdes		return (-1);
1190175611Sdes	/*
1191175611Sdes	 * When op is "STAT", ftp_request() will return either NULL or
1192175611Sdes	 * (FILE *)1, never a valid FILE *, so we mustn't fclose(f) before
1193175611Sdes	 * returning, as it would cause a segfault.
1194175611Sdes	 */
119590267Sdes	return (0);
119640975Sdes}
119741989Sdes
119841989Sdes/*
119941989Sdes * List a directory
120041989Sdes */
120141989Sdesstruct url_ent *
120285093SdesfetchListFTP(struct url *url __unused, const char *flags __unused)
120341989Sdes{
120490267Sdes	warnx("fetchListFTP(): not implemented");
120590267Sdes	return (NULL);
120641989Sdes}
1207