php-src/ext/standard/ftp_fopen_wrapper.c

301 lines
8.0 KiB
C
Raw Normal View History

/*
+----------------------------------------------------------------------+
2001-12-11 15:32:16 +00:00
| PHP Version 4 |
+----------------------------------------------------------------------+
2001-12-11 15:32:16 +00:00
| Copyright (c) 1997-2002 The PHP Group |
+----------------------------------------------------------------------+
| This source file is subject to version 2.02 of the PHP license, |
| that is bundled with this package in the file LICENSE, and is |
| available at through the world-wide-web at |
| http://www.php.net/license/2_02.txt. |
| If you did not receive a copy of the PHP license and are unable to |
| obtain it through the world-wide-web, please send a note to |
| license@php.net so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
2002-02-28 08:29:35 +00:00
| Authors: Rasmus Lerdorf <rasmus@php.net> |
| Jim Winstead <jimw@php.net> |
| Hartmut Holzgraefe <hholzgra@php.net> |
+----------------------------------------------------------------------+
*/
/* $Id$ */
#include "php.h"
#include "php_globals.h"
2000-10-13 09:13:01 +00:00
#include "php_network.h"
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#ifdef PHP_WIN32
#include <windows.h>
#include <winsock.h>
#define O_RDONLY _O_RDONLY
#include "win32/param.h"
#else
#include <sys/param.h>
#endif
#include "php_standard.h"
#include <sys/types.h>
#if HAVE_SYS_SOCKET_H
#include <sys/socket.h>
#endif
#ifdef PHP_WIN32
#include <winsock.h>
#else
#include <netinet/in.h>
#include <netdb.h>
#if HAVE_ARPA_INET_H
#include <arpa/inet.h>
#endif
#endif
2001-12-23 00:46:13 +00:00
#if defined(PHP_WIN32) || defined(__riscos__)
#undef AF_UNIX
#endif
#if defined(AF_UNIX)
#include <sys/un.h>
#endif
#include "php_fopen_wrappers.h"
static int php_get_ftp_result(php_stream *stream TSRMLS_DC)
{
char tmp_line[513];
2002-03-15 21:03:08 +00:00
while (php_stream_gets(stream, tmp_line, sizeof(tmp_line)-1) &&
!(isdigit((int) tmp_line[0]) && isdigit((int) tmp_line[1]) &&
isdigit((int) tmp_line[2]) && tmp_line[3] == ' '));
return strtol(tmp_line, NULL, 10);
}
static int php_stream_ftp_stream_stat(php_stream_wrapper *wrapper,
php_stream *stream,
php_stream_statbuf *ssb
TSRMLS_DC)
{
/* For now, we return with a failure code to prevent the underlying
* file's details from being used instead. */
return -1;
}
static php_stream_wrapper_ops ftp_stream_wops = {
2002-03-15 21:03:08 +00:00
php_stream_url_wrap_ftp,
NULL,
php_stream_ftp_stream_stat,
NULL
};
php_stream_wrapper php_stream_ftp_wrapper = {
&ftp_stream_wops,
2002-03-15 21:03:08 +00:00
NULL
};
/* {{{ php_fopen_url_wrap_ftp
*/
php_stream * php_stream_url_wrap_ftp(php_stream_wrapper *wrapper, char *path, char *mode, int options, char **opened_path STREAMS_DC TSRMLS_DC)
{
2002-03-15 21:03:08 +00:00
php_stream *stream=NULL;
php_url *resource=NULL;
char tmp_line[512];
unsigned short portno;
char *scratch;
int result;
int i;
char *tpath, *ttpath;
resource = php_url_parse((char *) path);
2002-03-15 21:03:08 +00:00
if (resource == NULL || resource->path == NULL)
return NULL;
2002-03-15 21:03:08 +00:00
/* use port 21 if one wasn't specified */
if (resource->port == 0)
resource->port = 21;
2002-03-15 21:03:08 +00:00
stream = php_stream_sock_open_host(resource->host, resource->port, SOCK_STREAM, 0, 0);
if (stream == NULL)
goto errexit;
/* Start talking to ftp server */
result = php_get_ftp_result(stream TSRMLS_CC);
if (result > 299 || result < 200)
goto errexit;
/* send the user name */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "USER ");
if (resource->user != NULL) {
php_raw_url_decode(resource->user, strlen(resource->user));
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, resource->user);
} else {
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "anonymous");
}
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "\r\n");
/* get the response */
result = php_get_ftp_result(stream TSRMLS_CC);
/* if a password is required, send it */
if (result >= 300 && result <= 399) {
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "PASS ");
if (resource->pass != NULL) {
php_raw_url_decode(resource->pass, strlen(resource->pass));
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, resource->pass);
} else {
/* if the user has configured who they are,
send that as the password */
if (cfg_get_string("from", &scratch) == SUCCESS) {
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, scratch);
} else {
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "anonymous");
}
}
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "\r\n");
/* read the response */
result = php_get_ftp_result(stream TSRMLS_CC);
}
if (result > 299 || result < 200)
goto errexit;
/* set the connection to be binary */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "TYPE I\r\n");
result = php_get_ftp_result(stream TSRMLS_CC);
if (result > 299 || result < 200)
goto errexit;
/* find out the size of the file (verifying it exists) */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "SIZE ");
php_stream_write_string(stream, resource->path);
php_stream_write_string(stream, "\r\n");
/* read the response */
result = php_get_ftp_result(stream TSRMLS_CC);
if (mode[0] == 'r') {
/* when reading file, it must exist */
if (result > 299 || result < 200) {
errno = ENOENT;
2002-03-15 21:03:08 +00:00
goto errexit;
}
} else {
/* when writing file, it must NOT exist */
if (result <= 299 && result >= 200) {
errno = EEXIST;
2002-03-15 21:03:08 +00:00
goto errexit;
}
}
/* set up the passive connection */
/* We try EPSV first, needed for IPv6 and works on some IPv4 servers */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "EPSV\r\n");
while (php_stream_gets(stream, tmp_line, sizeof(tmp_line)-1) &&
!(isdigit((int) tmp_line[0]) && isdigit((int) tmp_line[1]) &&
isdigit((int) tmp_line[2]) && tmp_line[3] == ' '));
/* check if we got a 229 response */
if (strncmp(tmp_line, "229", 3)) {
/* EPSV failed, let's try PASV */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "PASV\r\n");
while (php_stream_gets(stream, tmp_line, sizeof(tmp_line)-1) &&
!(isdigit((int) tmp_line[0]) && isdigit((int) tmp_line[1]) &&
isdigit((int) tmp_line[2]) && tmp_line[3] == ' '));
/* make sure we got a 227 response */
if (strncmp(tmp_line, "227", 3))
goto errexit;
2001-08-11 17:03:37 +00:00
/* parse pasv command (129, 80, 95, 25, 13, 221) */
tpath = tmp_line;
/* skip over the "227 Some message " part */
for (tpath += 4; *tpath && !isdigit((int) *tpath); tpath++);
if (!*tpath)
goto errexit;
/* skip over the host ip, we just assume it's the same */
for (i = 0; i < 4; i++) {
for (; isdigit((int) *tpath); tpath++);
if (*tpath != ',')
goto errexit;
tpath++;
}
/* pull out the MSB of the port */
portno = (unsigned short) strtol(tpath, &ttpath, 10) * 256;
if (ttpath == NULL) {
/* didn't get correct response from PASV */
goto errexit;
}
tpath = ttpath;
if (*tpath != ',')
goto errexit;
tpath++;
/* pull out the LSB of the port */
portno += (unsigned short) strtol(tpath, &ttpath, 10);
} else {
/* parse epsv command (|||6446|) */
for (i = 0, tpath = tmp_line + 4; *tpath; tpath++) {
if (*tpath == '|') {
i++;
if (i == 3)
break;
}
}
if (i < 3)
goto errexit;
/* pull out the port */
portno = (unsigned short) strtol(tpath + 1, &ttpath, 10);
}
if (ttpath == NULL) {
/* didn't get correct response from EPSV/PASV */
goto errexit;
}
if (mode[0] == 'r') {
/* retrieve file */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "RETR ");
} else {
/* store file */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "STOR ");
}
if (resource->path != NULL) {
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, resource->path);
} else {
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "/");
}
/* close control connection */
2002-03-15 21:03:08 +00:00
php_stream_write_string(stream, "\r\nQUIT\r\n");
php_stream_close(stream);
/* open the data channel */
2002-03-15 21:03:08 +00:00
stream = php_stream_sock_open_host(resource->host, portno, SOCK_STREAM, 0, 0);
if (stream == NULL)
goto errexit;
2002-03-15 21:03:08 +00:00
php_url_free(resource);
2002-03-15 21:03:08 +00:00
return stream;
errexit:
php_url_free(resource);
2002-03-15 21:03:08 +00:00
if (stream)
php_stream_close(stream);
return NULL;
}
/* }}} */
/*
* Local variables:
* tab-width: 4
* c-basic-offset: 4
* End:
* vim600: sw=4 ts=4 fdm=marker
* vim<600: sw=4 ts=4
*/