File: //usr/local/src/ssh2-1.4.1/ssh2_sftp.c
/*
+----------------------------------------------------------------------+
| PHP Version 7 |
+----------------------------------------------------------------------+
| Copyright (c) 1997-2016 The PHP Group |
+----------------------------------------------------------------------+
| This source file is subject to version 3.01 of the PHP license, |
| that is bundled with this package in the file LICENSE, and is |
| available through the world-wide-web at the following url: |
| http://www.php.net/license/3_01.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. |
+----------------------------------------------------------------------+
| Author: Sara Golemon <pollita@php.net> |
+----------------------------------------------------------------------+
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "php.h"
#include "php_ssh2.h"
#include "ext/standard/php_string.h"
/* *************************
* Resource Housekeeping *
************************* */
void php_ssh2_sftp_dtor(zend_resource *rsrc)
{
php_ssh2_sftp_data *data = (php_ssh2_sftp_data*)rsrc->ptr;
if (!data) {
return;
}
if (data->session_rsrc->ptr != NULL) {
libssh2_sftp_shutdown(data->sftp);
}
zend_list_delete(data->session_rsrc);
efree(data);
}
/* *****************
* SFTP File Ops *
***************** */
unsigned long php_ssh2_parse_fopen_modes(char *openmode) {
unsigned long flags = 0;
if (strchr(openmode, 'a')) {
flags |= LIBSSH2_FXF_APPEND;
}
if (strchr(openmode, 'w')) {
flags |= LIBSSH2_FXF_WRITE | LIBSSH2_FXF_TRUNC | LIBSSH2_FXF_CREAT;
}
if (strchr(openmode, 'r')) {
flags |= LIBSSH2_FXF_READ;
}
if (strchr(openmode, '+')) {
flags |= LIBSSH2_FXF_READ | LIBSSH2_FXF_WRITE;
}
if (strchr(openmode, 'x')) {
flags |= LIBSSH2_FXF_WRITE | LIBSSH2_FXF_TRUNC | LIBSSH2_FXF_EXCL | LIBSSH2_FXF_CREAT;
}
return flags;
}
static inline int php_ssh2_sftp_attr2ssb(php_stream_statbuf *ssb, LIBSSH2_SFTP_ATTRIBUTES *attrs)
{
memset(ssb, 0, sizeof(php_stream_statbuf));
if (attrs->flags & LIBSSH2_SFTP_ATTR_SIZE) {
ssb->sb.st_size = attrs->filesize;
}
if (attrs->flags & LIBSSH2_SFTP_ATTR_UIDGID) {
ssb->sb.st_uid = attrs->uid;
ssb->sb.st_gid = attrs->gid;
}
if (attrs->flags & LIBSSH2_SFTP_ATTR_PERMISSIONS) {
ssb->sb.st_mode = attrs->permissions;
}
if (attrs->flags & LIBSSH2_SFTP_ATTR_ACMODTIME) {
ssb->sb.st_atime = attrs->atime;
ssb->sb.st_mtime = attrs->mtime;
}
return 0;
}
typedef struct _php_ssh2_sftp_handle_data {
LIBSSH2_SFTP_HANDLE *handle;
zend_resource *sftp_rsrc;
} php_ssh2_sftp_handle_data;
/* {{{ php_ssh2_sftp_stream_write
*/
#if PHP_VERSION_ID < 70400
static size_t php_ssh2_sftp_stream_write(php_stream *stream, const char *buf, size_t count)
#else
static ssize_t php_ssh2_sftp_stream_write(php_stream *stream, const char *buf, size_t count)
#endif
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
ssize_t bytes_written;
bytes_written = libssh2_sftp_write(data->handle, buf, count);
#if PHP_VERSION_ID < 70400
return (size_t)(bytes_written<0 ? 0 : bytes_written);
#else
return bytes_written;
#endif
}
/* }}} */
/* {{{ php_ssh2_sftp_stream_read
*/
#if PHP_VERSION_ID < 70400
static size_t php_ssh2_sftp_stream_read(php_stream *stream, char *buf, size_t count)
#else
static ssize_t php_ssh2_sftp_stream_read(php_stream *stream, char *buf, size_t count)
#endif
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
ssize_t bytes_read;
bytes_read = libssh2_sftp_read(data->handle, buf, count);
stream->eof = (bytes_read <= 0 && bytes_read != LIBSSH2_ERROR_EAGAIN);
#if PHP_VERSION_ID < 70400
return (size_t)(bytes_read<0 ? 0 : bytes_read);
#else
return bytes_read;
#endif
}
/* }}} */
/* {{{ php_ssh2_sftp_stream_close
*/
static int php_ssh2_sftp_stream_close(php_stream *stream, int close_handle)
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
libssh2_sftp_close(data->handle);
zend_list_delete(data->sftp_rsrc);
efree(data);
return 0;
}
/* }}} */
/* {{{ php_ssh2_sftp_stream_seek
*/
static int php_ssh2_sftp_stream_seek(php_stream *stream, zend_off_t offset, int whence, zend_off_t *newoffset)
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
switch (whence) {
case SEEK_END:
{
LIBSSH2_SFTP_ATTRIBUTES attrs;
if (libssh2_sftp_fstat(data->handle, &attrs)) {
return -1;
}
if ((attrs.flags & LIBSSH2_SFTP_ATTR_SIZE) == 0) {
return -1;
}
offset += attrs.filesize;
break;
}
case SEEK_CUR:
{
zend_off_t current_offset = libssh2_sftp_tell(data->handle);
if (current_offset < 0) {
return -1;
}
offset += current_offset;
break;
}
}
libssh2_sftp_seek(data->handle, offset);
if (newoffset) {
*newoffset = offset;
}
return 0;
}
/* }}} */
/* {{{ php_ssh2_sftp_stream_fstat
*/
static int php_ssh2_sftp_stream_fstat(php_stream *stream, php_stream_statbuf *ssb)
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
LIBSSH2_SFTP_ATTRIBUTES attrs;
if (libssh2_sftp_fstat(data->handle, &attrs)) {
return -1;
}
return php_ssh2_sftp_attr2ssb(ssb, &attrs);
}
/* }}} */
static php_stream_ops php_ssh2_sftp_stream_ops = {
php_ssh2_sftp_stream_write,
php_ssh2_sftp_stream_read,
php_ssh2_sftp_stream_close,
NULL, /* flush */
PHP_SSH2_SFTP_STREAM_NAME,
php_ssh2_sftp_stream_seek,
NULL, /* cast */
php_ssh2_sftp_stream_fstat,
NULL, /* set_option */
};
/* {{{ php_ssh2_sftp_stream_opener
*/
static php_stream *php_ssh2_sftp_stream_opener(php_stream_wrapper *wrapper, const char *filename, const char *mode,
int options, zend_string **opened_path, php_stream_context *context STREAMS_DC)
{
php_ssh2_sftp_handle_data *data;
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
LIBSSH2_SFTP_HANDLE *handle;
php_stream *stream;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource;
unsigned long flags;
long perms = 0644;
resource = php_ssh2_fopen_wraper_parse_path(filename, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp || !sftp_rsrc) {
return NULL;
}
flags = php_ssh2_parse_fopen_modes((char *)mode);
handle = libssh2_sftp_open(sftp, SSH2_URL_STR(resource->path), flags, perms);
if (!handle) {
php_error_docref(NULL, E_WARNING, "Unable to open %s on remote host", filename);
php_url_free(resource);
zend_list_delete(sftp_rsrc);
return NULL;
}
data = emalloc(sizeof(php_ssh2_sftp_handle_data));
data->handle = handle;
data->sftp_rsrc = sftp_rsrc;
stream = php_stream_alloc(&php_ssh2_sftp_stream_ops, data, 0, mode);
if (!stream) {
libssh2_sftp_close(handle);
zend_list_delete(sftp_rsrc);
efree(data);
}
php_url_free(resource);
return stream;
}
/* }}} */
/* **********************
* SFTP Directory Ops *
********************** */
/* {{{ php_ssh2_sftp_dirstream_read
*/
#if PHP_VERSION_ID < 70400
static size_t php_ssh2_sftp_dirstream_read(php_stream *stream, char *buf, size_t count)
#else
static ssize_t php_ssh2_sftp_dirstream_read(php_stream *stream, char *buf, size_t count)
#endif
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
php_stream_dirent *ent = (php_stream_dirent*)buf;
int bytesread = libssh2_sftp_readdir(data->handle, ent->d_name, sizeof(ent->d_name) - 1, NULL);
zend_string *basename;
if (bytesread <= 0) {
return 0;
}
ent->d_name[bytesread] = 0;
basename = php_basename(ent->d_name, bytesread, NULL, 0);
if (!basename) {
return 0;
}
bytesread = MIN(sizeof(ent->d_name) - 1, basename->len);
memcpy(ent->d_name, basename->val, bytesread);
ent->d_name[bytesread] = 0;
zend_string_release(basename);
return sizeof(php_stream_dirent);
}
/* }}} */
/* {{{ php_ssh2_sftp_dirstream_close
*/
static int php_ssh2_sftp_dirstream_close(php_stream *stream, int close_handle)
{
php_ssh2_sftp_handle_data *data = (php_ssh2_sftp_handle_data*)stream->abstract;
libssh2_sftp_close(data->handle);
zend_list_delete(data->sftp_rsrc);
efree(data);
return 0;
}
/* }}} */
static php_stream_ops php_ssh2_sftp_dirstream_ops = {
NULL, /* write */
php_ssh2_sftp_dirstream_read,
php_ssh2_sftp_dirstream_close,
NULL, /* flush */
PHP_SSH2_SFTP_DIRSTREAM_NAME,
NULL, /* seek */
NULL, /* cast */
NULL, /* fstat */
NULL, /* set_option */
};
/* {{{ php_ssh2_sftp_dirstream_opener
*/
static php_stream *php_ssh2_sftp_dirstream_opener(php_stream_wrapper *wrapper, const char *filename, const char *mode,
int options, zend_string **opened_path, php_stream_context *context STREAMS_DC)
{
php_ssh2_sftp_handle_data *data;
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
LIBSSH2_SFTP_HANDLE *handle;
php_stream *stream;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource;
resource = php_ssh2_fopen_wraper_parse_path(filename, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp) {
return NULL;
}
handle = libssh2_sftp_opendir(sftp, SSH2_URL_STR(resource->path));
if (!handle) {
php_error_docref(NULL, E_WARNING, "Unable to open %s on remote host", filename);
php_url_free(resource);
zend_list_delete(sftp_rsrc);
return NULL;
}
data = emalloc(sizeof(php_ssh2_sftp_handle_data));
data->handle = handle;
data->sftp_rsrc = sftp_rsrc;
stream = php_stream_alloc(&php_ssh2_sftp_dirstream_ops, data, 0, mode);
if (!stream) {
libssh2_sftp_close(handle);
zend_list_delete(sftp_rsrc);
efree(data);
}
php_url_free(resource);
return stream;
}
/* }}} */
/* ****************
* SFTP Wrapper *
**************** */
/* {{{ php_ssh2_sftp_urlstat
*/
static int php_ssh2_sftp_urlstat(php_stream_wrapper *wrapper, const char *url, int flags, php_stream_statbuf *ssb, php_stream_context *context)
{
LIBSSH2_SFTP_ATTRIBUTES attrs;
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource;
resource = php_ssh2_fopen_wraper_parse_path(url, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp || !resource->path) {
return -1;
}
if (libssh2_sftp_stat_ex(sftp, SSH2_URL_STR(resource->path), SSH2_URL_LEN(resource->path),
(flags & PHP_STREAM_URL_STAT_LINK) ? LIBSSH2_SFTP_LSTAT : LIBSSH2_SFTP_STAT, &attrs)) {
php_url_free(resource);
//zend_list_delete(sftp_rsrcid);
return -1;
}
php_url_free(resource);
/* parse_path addrefs the resource, but we're not holding on to it so we have to delref it before we leave */
//zend_list_delete(sftp_rsrcid);
return php_ssh2_sftp_attr2ssb(ssb, &attrs);
}
/* }}} */
/* {{{ php_ssh2_sftp_unlink
*/
static int php_ssh2_sftp_unlink(php_stream_wrapper *wrapper, const char *url, int options, php_stream_context *context)
{
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource;
int result;
resource = php_ssh2_fopen_wraper_parse_path(url, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp || !resource->path) {
if (resource) {
php_url_free(resource);
}
return 0;
}
result = libssh2_sftp_unlink(sftp, SSH2_URL_STR(resource->path));
php_url_free(resource);
//zend_list_delete(sftp_rsrcid);
/* libssh2 uses 0 for success and the streams API uses 0 for failure, so invert */
return (result == 0) ? -1 : 0;
}
/* }}} */
/* {{{ php_ssh2_sftp_rename
*/
static int php_ssh2_sftp_rename(php_stream_wrapper *wrapper, const char *url_from, const char *url_to, int options, php_stream_context *context)
{
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource, *resource_to;
int result;
if (strncmp(url_from, "ssh2.sftp://", sizeof("ssh2.sftp://") - 1) ||
strncmp(url_to, "ssh2.sftp://", sizeof("ssh2.sftp://") - 1)) {
return 0;
}
resource_to = php_url_parse(url_to);
if (!resource_to || !resource_to->path) {
if (resource_to) {
php_url_free(resource_to);
}
return 0;
}
resource = php_ssh2_fopen_wraper_parse_path(url_from, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp || !resource->path) {
if (resource) {
php_url_free(resource);
}
php_url_free(resource_to);
return 0;
}
result = libssh2_sftp_rename(sftp, SSH2_URL_STR(resource->path), SSH2_URL_STR(resource_to->path));
php_url_free(resource);
php_url_free(resource_to);
//zend_list_delete(sftp_rsrcid);
/* libssh2 uses 0 for success and the streams API uses 0 for failure, so invert */
return (result == 0) ? -1 : 0;
}
/* }}} */
/* {{{ php_ssh2_sftp_mkdir
*/
static int php_ssh2_sftp_mkdir(php_stream_wrapper *wrapper, const char *url, int mode, int options, php_stream_context *context)
{
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource;
int result;
resource = php_ssh2_fopen_wraper_parse_path(url, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp || !resource->path) {
if (resource) {
php_url_free(resource);
}
return 0;
}
if (options & PHP_STREAM_MKDIR_RECURSIVE) {
/* Just attempt to make every directory, some will fail, but we only care about the last success/failure */
char *p = SSH2_URL_STR(resource->path);
while ((p = strchr(p + 1, '/'))) {
libssh2_sftp_mkdir_ex(sftp, SSH2_URL_STR(resource->path), p - SSH2_URL_STR(resource->path), mode);
}
}
result = libssh2_sftp_mkdir(sftp, SSH2_URL_STR(resource->path), mode);
php_url_free(resource);
//zend_list_delete(sftp_rsrcid);
/* libssh2 uses 0 for success and the streams API uses 0 for failure, so invert */
return (result == 0) ? -1 : 0;
}
/* }}} */
/* {{{ php_ssh2_sftp_rmdir
*/
static int php_ssh2_sftp_rmdir(php_stream_wrapper *wrapper, const char *url, int options, php_stream_context *context)
{
LIBSSH2_SESSION *session = NULL;
LIBSSH2_SFTP *sftp = NULL;
zend_resource *rsrc = NULL, *sftp_rsrc = NULL;
php_url *resource;
int result;
resource = php_ssh2_fopen_wraper_parse_path(url, "sftp", context, &session, &rsrc, &sftp, &sftp_rsrc);
if (!resource || !session || !sftp || !resource->path) {
if (resource) {
php_url_free(resource);
}
return 0;
}
result = libssh2_sftp_rmdir(sftp, SSH2_URL_STR(resource->path));
php_url_free(resource);
//zend_list_delete(sftp_rsrcid);
/* libssh2 uses 0 for success and the streams API uses 0 for failure, so invert */
return (result == 0) ? -1 : 0;
}
/* }}} */
static php_stream_wrapper_ops php_ssh2_sftp_wrapper_ops = {
php_ssh2_sftp_stream_opener,
NULL, /* close */
NULL, /* stat */
php_ssh2_sftp_urlstat,
php_ssh2_sftp_dirstream_opener,
PHP_SSH2_SFTP_WRAPPER_NAME,
php_ssh2_sftp_unlink,
php_ssh2_sftp_rename,
php_ssh2_sftp_mkdir,
php_ssh2_sftp_rmdir,
};
php_stream_wrapper php_ssh2_sftp_wrapper = {
&php_ssh2_sftp_wrapper_ops,
NULL,
1,
};
/* *****************
* Userspace API *
***************** */
/* {{{ proto resource ssh2_sftp(resource session)
* Request the SFTP subsystem from an already connected SSH2 server
*/
PHP_FUNCTION(ssh2_sftp)
{
LIBSSH2_SESSION *session;
LIBSSH2_SFTP *sftp;
php_ssh2_sftp_data *data;
zval *zsession;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &zsession) == FAILURE) {
return;
}
if ((session = (LIBSSH2_SESSION *)zend_fetch_resource(Z_RES_P(zsession), PHP_SSH2_SESSION_RES_NAME, le_ssh2_session)) == NULL) {
RETURN_FALSE;
}
sftp = libssh2_sftp_init(session);
if (!sftp) {
char *sess_err = "Unknown";
libssh2_session_last_error(session, &sess_err, NULL, 0);
php_error_docref(NULL, E_WARNING, "Unable to startup SFTP subsystem: %s", sess_err);
RETURN_FALSE;
}
data = emalloc(sizeof(php_ssh2_sftp_data));
data->session = session;
data->sftp = sftp;
data->session_rsrc = Z_RES_P(zsession);
Z_ADDREF_P(zsession);
RETURN_RES(zend_register_resource(data, le_ssh2_sftp));
}
/* }}} */
/* Much of the stuff below can be done via wrapper ops as of PHP5, but is included here for PHP 4.3 users */
/* {{{ proto bool ssh2_sftp_rename(resource sftp, string from, string to)
*/
PHP_FUNCTION(ssh2_sftp_rename)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *src, *dst;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rSS", &zsftp, &src, &dst) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
RETURN_BOOL(!libssh2_sftp_rename_ex(data->sftp, src->val, src->len, dst->val, dst->len,
LIBSSH2_SFTP_RENAME_OVERWRITE | LIBSSH2_SFTP_RENAME_ATOMIC | LIBSSH2_SFTP_RENAME_NATIVE));
}
/* }}} */
/* {{{ proto bool ssh2_sftp_unlink(resource sftp, string filename)
*/
PHP_FUNCTION(ssh2_sftp_unlink)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *filename;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rS", &zsftp, &filename) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
RETURN_BOOL(!libssh2_sftp_unlink_ex(data->sftp, filename->val, filename->len));
}
/* }}} */
/* {{{ proto bool ssh2_sftp_mkdir(resource sftp, string dirname[, int mode[, int recursive]])
*/
PHP_FUNCTION(ssh2_sftp_mkdir)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *dirname;
zend_long mode = 0777;
zend_bool recursive = 0;
char *p;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rS|lb", &zsftp, &dirname, &mode, &recursive) == FAILURE) {
return;
}
if (!dirname) {
RETURN_FALSE;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
if (recursive) {
/* Just attempt to make every directory, some will fail, but we only care about the last success/failure */
p = dirname->val;
while ((p = strchr(p + 1, '/'))) {
if ((p - dirname->val) + 1 == dirname->len) {
break;
}
libssh2_sftp_mkdir_ex(data->sftp, dirname->val, p - dirname->val, mode);
}
}
RETURN_BOOL(!libssh2_sftp_mkdir_ex(data->sftp, dirname->val, dirname->len, mode));
}
/* }}} */
/* {{{ proto bool ssh2_sftp_rmdir(resource sftp, string dirname)
*/
PHP_FUNCTION(ssh2_sftp_rmdir)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *dirname;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rS", &zsftp, &dirname) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
RETURN_BOOL(!libssh2_sftp_rmdir_ex(data->sftp, dirname->val, dirname->len));
}
/* }}} */
/* {{{ proto bool ssh2_sftp_chmod(resource sftp, string filename, int mode)
*/
PHP_FUNCTION(ssh2_sftp_chmod)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *filename;
zend_long mode;
LIBSSH2_SFTP_ATTRIBUTES attrs;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rSl", &zsftp, &filename, &mode) == FAILURE) {
return;
}
if (ZSTR_LEN(filename) < 1) {
RETURN_FALSE;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
attrs.permissions = mode;
attrs.flags = LIBSSH2_SFTP_ATTR_PERMISSIONS;
RETURN_BOOL(!libssh2_sftp_stat_ex(data->sftp, filename->val, filename->len, LIBSSH2_SFTP_SETSTAT, &attrs));
}
/* }}} */
/* {{{ php_ssh2_sftp_stat_func
* In PHP4.3 this is the only way to request stat into, in PHP >= 5 you can use the fopen wrapper approach
* Both methods will return identical structures
* (well, the other one will include other values set to 0 but they don't count)
*/
static void php_ssh2_sftp_stat_func(INTERNAL_FUNCTION_PARAMETERS, int stat_type)
{
php_ssh2_sftp_data *data;
LIBSSH2_SFTP_ATTRIBUTES attrs;
zval *zsftp;
zend_string *path;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rS", &zsftp, &path) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
if (libssh2_sftp_stat_ex(data->sftp, path->val, path->len, stat_type, &attrs)) {
php_error_docref(NULL, E_WARNING, "Failed to stat remote file");
RETURN_FALSE;
}
array_init(return_value);
if (attrs.flags & LIBSSH2_SFTP_ATTR_SIZE) {
add_index_long(return_value, 7, attrs.filesize);
add_assoc_long(return_value, "size", attrs.filesize);
}
if (attrs.flags & LIBSSH2_SFTP_ATTR_UIDGID) {
add_index_long(return_value, 4, attrs.uid);
add_assoc_long(return_value, "uid", attrs.uid);
add_index_long(return_value, 5, attrs.gid);
add_assoc_long(return_value, "gid", attrs.gid);
}
if (attrs.flags & LIBSSH2_SFTP_ATTR_PERMISSIONS) {
add_index_long(return_value, 2, attrs.permissions);
add_assoc_long(return_value, "mode", attrs.permissions);
}
if (attrs.flags & LIBSSH2_SFTP_ATTR_ACMODTIME) {
add_index_long(return_value, 8, attrs.atime);
add_assoc_long(return_value, "atime", attrs.atime);
add_index_long(return_value, 9, attrs.mtime);
add_assoc_long(return_value, "mtime", attrs.mtime);
}
}
/* }}} */
/* {{{ proto array ssh2_sftp_stat(resource sftp, string path)
*/
PHP_FUNCTION(ssh2_sftp_stat)
{
php_ssh2_sftp_stat_func(INTERNAL_FUNCTION_PARAM_PASSTHRU, LIBSSH2_SFTP_STAT);
}
/* }}} */
/* {{{ proto array ssh2_sftp_lstat(resource sftp, string path)
*/
PHP_FUNCTION(ssh2_sftp_lstat)
{
php_ssh2_sftp_stat_func(INTERNAL_FUNCTION_PARAM_PASSTHRU, LIBSSH2_SFTP_LSTAT);
}
/* }}} */
/* {{{ proto bool ssh2_sftp_symlink(resource sftp, string target, string link)
*/
PHP_FUNCTION(ssh2_sftp_symlink)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *targ, *link;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rSS", &zsftp, &targ, &link) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
RETURN_BOOL(!libssh2_sftp_symlink_ex(data->sftp, targ->val, targ->len, link->val, link->len, LIBSSH2_SFTP_SYMLINK));
}
/* }}} */
/* {{{ proto string ssh2_sftp_readlink(resource sftp, string link)
*/
PHP_FUNCTION(ssh2_sftp_readlink)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *link;
int targ_len = 0;
char targ[8192];
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rS", &zsftp, &link) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
if ((targ_len = libssh2_sftp_symlink_ex(data->sftp, link->val, link->len, targ, 8192, LIBSSH2_SFTP_READLINK)) < 0) {
php_error_docref(NULL, E_WARNING, "Unable to read link '%s'", ZSTR_VAL(link));
RETURN_FALSE;
}
RETURN_STRINGL(targ, targ_len);
}
/* }}} */
/* {{{ proto string ssh2_sftp_realpath(resource sftp, string filename)
*/
PHP_FUNCTION(ssh2_sftp_realpath)
{
php_ssh2_sftp_data *data;
zval *zsftp;
zend_string *link;
int targ_len = 0;
char targ[8192];
if (zend_parse_parameters(ZEND_NUM_ARGS(), "rS", &zsftp, &link) == FAILURE) {
return;
}
if ((data = (php_ssh2_sftp_data *)zend_fetch_resource(Z_RES_P(zsftp), PHP_SSH2_SFTP_RES_NAME, le_ssh2_sftp)) == NULL) {
RETURN_FALSE;
}
if (data->session_rsrc->ptr == NULL) {
RETURN_FALSE;
}
if ((targ_len = libssh2_sftp_symlink_ex(data->sftp, link->val, link->len, targ, 8192, LIBSSH2_SFTP_REALPATH)) < 0) {
php_error_docref(NULL, E_WARNING, "Unable to resolve realpath for '%s'", link->val);
RETURN_FALSE;
}
RETURN_STRINGL(targ, targ_len);
}
/* }}} */
/*
* Local variables:
* tab-width: 4
* c-basic-offset: 4
* indent-tabs-mode: t
* End:
*/