2009-11-06 21:21:18 +03:00
|
|
|
/*
|
|
|
|
Copyright 2003-2009 Aris Adamantiadis
|
|
|
|
|
|
|
|
This file is part of the SSH Library
|
|
|
|
|
|
|
|
You are free to copy this file, modify it in any way, consider it being public
|
|
|
|
domain. This does not apply to the rest of the library though, but it is
|
|
|
|
allowed to cut-and-paste working code from this file to any license of
|
|
|
|
program.
|
|
|
|
The goal is to show the API in action. It's not a reference on how terminal
|
|
|
|
clients must be made or how a client should react.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <sys/statvfs.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2013-07-22 12:41:58 +04:00
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
2009-11-06 21:21:18 +03:00
|
|
|
|
|
|
|
#include <libssh/libssh.h>
|
|
|
|
#include <libssh/sftp.h>
|
|
|
|
|
|
|
|
#include "examples_common.h"
|
|
|
|
#ifdef WITH_SFTP
|
|
|
|
|
2021-05-09 21:00:24 +03:00
|
|
|
#ifndef BUF_SIZE
|
|
|
|
#define BUF_SIZE 65536
|
|
|
|
#endif
|
|
|
|
|
2012-02-18 15:28:03 +04:00
|
|
|
static int verbosity;
|
|
|
|
static char *destination;
|
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
static void do_sftp(ssh_session session) {
|
|
|
|
sftp_session sftp = sftp_new(session);
|
2009-11-06 21:21:18 +03:00
|
|
|
sftp_dir dir;
|
|
|
|
sftp_attributes file;
|
|
|
|
sftp_statvfs_t sftpstatvfs;
|
|
|
|
struct statvfs sysstatvfs;
|
|
|
|
sftp_file fichier;
|
|
|
|
sftp_file to;
|
2018-09-11 16:14:17 +03:00
|
|
|
int len = 1;
|
2009-11-06 21:21:18 +03:00
|
|
|
unsigned int i;
|
2021-05-09 21:00:24 +03:00
|
|
|
char data[BUF_SIZE] = {0};
|
2009-11-06 21:21:18 +03:00
|
|
|
char *lnk;
|
|
|
|
|
|
|
|
unsigned int count;
|
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
if (!sftp) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "sftp error initialising channel: %s\n",
|
2018-09-11 16:14:17 +03:00
|
|
|
ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
|
|
|
if (sftp_init(sftp)) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "error initialising sftp: %s\n",
|
2018-09-11 16:14:17 +03:00
|
|
|
ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("Additional SFTP extensions provided by the server:\n");
|
|
|
|
count = sftp_extensions_get_count(sftp);
|
|
|
|
for (i = 0; i < count; i++) {
|
2018-09-11 16:14:17 +03:00
|
|
|
printf("\t%s, version: %s\n",
|
|
|
|
sftp_extensions_get_name(sftp, i),
|
|
|
|
sftp_extensions_get_data(sftp, i));
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* test symlink and readlink */
|
|
|
|
if (sftp_symlink(sftp, "/tmp/this_is_the_link",
|
2018-09-11 16:14:17 +03:00
|
|
|
"/tmp/sftp_symlink_test") < 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Could not create link (%s)\n",
|
|
|
|
ssh_get_error(session));
|
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
lnk = sftp_readlink(sftp, "/tmp/sftp_symlink_test");
|
|
|
|
if (lnk == NULL) {
|
2018-09-11 16:14:17 +03:00
|
|
|
fprintf(stderr, "Could not read link (%s)\n", ssh_get_error(session));
|
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
printf("readlink /tmp/sftp_symlink_test: %s\n", lnk);
|
|
|
|
|
|
|
|
sftp_unlink(sftp, "/tmp/sftp_symlink_test");
|
|
|
|
|
|
|
|
if (sftp_extension_supported(sftp, "statvfs@openssh.com", "2")) {
|
2018-09-11 16:14:17 +03:00
|
|
|
sftpstatvfs = sftp_statvfs(sftp, "/tmp");
|
|
|
|
if (sftpstatvfs == NULL) {
|
|
|
|
fprintf(stderr, "statvfs failed (%s)\n", ssh_get_error(session));
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("sftp statvfs:\n"
|
|
|
|
"\tfile system block size: %llu\n"
|
|
|
|
"\tfundamental fs block size: %llu\n"
|
|
|
|
"\tnumber of blocks (unit f_frsize): %llu\n"
|
|
|
|
"\tfree blocks in file system: %llu\n"
|
|
|
|
"\tfree blocks for non-root: %llu\n"
|
|
|
|
"\ttotal file inodes: %llu\n"
|
|
|
|
"\tfree file inodes: %llu\n"
|
|
|
|
"\tfree file inodes for to non-root: %llu\n"
|
|
|
|
"\tfile system id: %llu\n"
|
|
|
|
"\tbit mask of f_flag values: %llu\n"
|
|
|
|
"\tmaximum filename length: %llu\n",
|
|
|
|
(unsigned long long) sftpstatvfs->f_bsize,
|
|
|
|
(unsigned long long) sftpstatvfs->f_frsize,
|
|
|
|
(unsigned long long) sftpstatvfs->f_blocks,
|
|
|
|
(unsigned long long) sftpstatvfs->f_bfree,
|
|
|
|
(unsigned long long) sftpstatvfs->f_bavail,
|
|
|
|
(unsigned long long) sftpstatvfs->f_files,
|
|
|
|
(unsigned long long) sftpstatvfs->f_ffree,
|
|
|
|
(unsigned long long) sftpstatvfs->f_favail,
|
|
|
|
(unsigned long long) sftpstatvfs->f_fsid,
|
|
|
|
(unsigned long long) sftpstatvfs->f_flag,
|
|
|
|
(unsigned long long) sftpstatvfs->f_namemax);
|
|
|
|
|
|
|
|
sftp_statvfs_free(sftpstatvfs);
|
|
|
|
|
|
|
|
if (statvfs("/tmp", &sysstatvfs) < 0) {
|
|
|
|
fprintf(stderr, "statvfs failed (%s)\n", strerror(errno));
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("sys statvfs:\n"
|
|
|
|
"\tfile system block size: %llu\n"
|
|
|
|
"\tfundamental fs block size: %llu\n"
|
|
|
|
"\tnumber of blocks (unit f_frsize): %llu\n"
|
|
|
|
"\tfree blocks in file system: %llu\n"
|
|
|
|
"\tfree blocks for non-root: %llu\n"
|
|
|
|
"\ttotal file inodes: %llu\n"
|
|
|
|
"\tfree file inodes: %llu\n"
|
|
|
|
"\tfree file inodes for to non-root: %llu\n"
|
|
|
|
"\tfile system id: %llu\n"
|
|
|
|
"\tbit mask of f_flag values: %llu\n"
|
|
|
|
"\tmaximum filename length: %llu\n",
|
|
|
|
(unsigned long long) sysstatvfs.f_bsize,
|
|
|
|
(unsigned long long) sysstatvfs.f_frsize,
|
|
|
|
(unsigned long long) sysstatvfs.f_blocks,
|
|
|
|
(unsigned long long) sysstatvfs.f_bfree,
|
|
|
|
(unsigned long long) sysstatvfs.f_bavail,
|
|
|
|
(unsigned long long) sysstatvfs.f_files,
|
|
|
|
(unsigned long long) sysstatvfs.f_ffree,
|
|
|
|
(unsigned long long) sysstatvfs.f_favail,
|
|
|
|
(unsigned long long) sysstatvfs.f_fsid,
|
|
|
|
(unsigned long long) sysstatvfs.f_flag,
|
|
|
|
(unsigned long long) sysstatvfs.f_namemax);
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* the connection is made */
|
|
|
|
/* opening a directory */
|
2018-09-11 16:14:17 +03:00
|
|
|
dir = sftp_opendir(sftp, "./");
|
|
|
|
if (!dir) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Directory not opened(%s)\n", ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
2009-11-06 21:21:18 +03:00
|
|
|
/* reading the whole directory, file by file */
|
2018-09-11 16:14:17 +03:00
|
|
|
while ((file = sftp_readdir(sftp, dir))) {
|
2010-02-24 02:26:20 +03:00
|
|
|
fprintf(stderr, "%30s(%.8o) : %s(%.5d) %s(%.5d) : %.10llu bytes\n",
|
2018-09-11 16:14:17 +03:00
|
|
|
file->name,
|
|
|
|
file->permissions,
|
|
|
|
file->owner,
|
|
|
|
file->uid,
|
|
|
|
file->group,
|
|
|
|
file->gid,
|
|
|
|
(long long unsigned int) file->size);
|
2009-11-06 21:21:18 +03:00
|
|
|
sftp_attributes_free(file);
|
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
|
|
|
/* when file = NULL, an error has occured OR the directory listing is end of
|
|
|
|
* file */
|
|
|
|
if (!sftp_dir_eof(dir)) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Error: %s\n", ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
|
|
|
if (sftp_closedir(dir)) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Error: %s\n", ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
/* this will open a file and copy it into your /home directory */
|
2018-09-11 16:14:17 +03:00
|
|
|
/* the small buffer size was intended to stress the library. of course, you
|
|
|
|
* can use a buffer till 20kbytes without problem */
|
2009-11-06 21:21:18 +03:00
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
fichier = sftp_open(sftp, "/usr/bin/ssh", O_RDONLY, 0);
|
|
|
|
if (!fichier) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Error opening /usr/bin/ssh: %s\n",
|
2018-09-11 16:14:17 +03:00
|
|
|
ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
2009-11-06 21:21:18 +03:00
|
|
|
/* open a file for writing... */
|
2018-09-11 16:14:17 +03:00
|
|
|
to = sftp_open(sftp, "ssh-copy", O_WRONLY | O_CREAT, 0700);
|
|
|
|
if (!to) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Error opening ssh-copy for writing: %s\n",
|
2018-09-11 16:14:17 +03:00
|
|
|
ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
sftp_close(fichier);
|
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
|
|
|
while ((len = sftp_read(fichier, data, 4096)) > 0) {
|
|
|
|
if (sftp_write(to, data, len) != len) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Error writing %d bytes: %s\n",
|
2018-09-11 16:14:17 +03:00
|
|
|
len, ssh_get_error(session));
|
2018-09-11 15:22:01 +03:00
|
|
|
sftp_close(to);
|
|
|
|
sftp_close(fichier);
|
|
|
|
goto end;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
|
2009-11-06 21:21:18 +03:00
|
|
|
printf("finished\n");
|
2018-09-11 16:14:17 +03:00
|
|
|
if (len < 0) {
|
2009-11-06 21:21:18 +03:00
|
|
|
fprintf(stderr, "Error reading file: %s\n", ssh_get_error(session));
|
2018-09-11 16:14:17 +03:00
|
|
|
}
|
|
|
|
|
2009-11-06 21:21:18 +03:00
|
|
|
sftp_close(fichier);
|
|
|
|
sftp_close(to);
|
|
|
|
printf("fichiers ferm\n");
|
2018-09-11 16:14:17 +03:00
|
|
|
to = sftp_open(sftp, "/tmp/grosfichier", O_WRONLY|O_CREAT, 0644);
|
|
|
|
|
|
|
|
for (i = 0; i < 1000; ++i) {
|
2021-05-09 21:00:24 +03:00
|
|
|
len = sftp_write(to, data, sizeof(data));
|
2018-09-11 16:14:17 +03:00
|
|
|
printf("wrote %d bytes\n", len);
|
2021-05-09 21:00:24 +03:00
|
|
|
if (len != sizeof(data)) {
|
2018-09-11 16:14:17 +03:00
|
|
|
printf("chunk %d : %d (%s)\n", i, len, ssh_get_error(session));
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-11 15:22:01 +03:00
|
|
|
sftp_close(to);
|
|
|
|
end:
|
2009-11-06 21:21:18 +03:00
|
|
|
/* close the sftp session */
|
|
|
|
sftp_free(sftp);
|
|
|
|
printf("sftp session terminated\n");
|
|
|
|
}
|
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
static void usage(const char *argv0) {
|
|
|
|
fprintf(stderr, "Usage : %s [-v] remotehost\n"
|
|
|
|
"sample sftp test client - libssh-%s\n"
|
|
|
|
"Options :\n"
|
|
|
|
" -v : increase log verbosity\n",
|
|
|
|
argv0,
|
|
|
|
ssh_version(0));
|
|
|
|
exit(0);
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
static int opts(int argc, char **argv) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
while ((i = getopt(argc, argv, "v")) != -1) {
|
|
|
|
switch(i) {
|
|
|
|
case 'v':
|
|
|
|
verbosity++;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "unknown option %c\n", optopt);
|
|
|
|
usage(argv[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
destination = argv[optind];
|
|
|
|
if (destination == NULL) {
|
2009-11-06 21:21:18 +03:00
|
|
|
usage(argv[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
2018-09-11 16:14:17 +03:00
|
|
|
return 0;
|
2009-11-06 21:21:18 +03:00
|
|
|
}
|
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
ssh_session session;
|
2009-11-06 21:21:18 +03:00
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
if (opts(argc, argv) < 0) {
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
session = connect_ssh(destination, NULL, verbosity);
|
|
|
|
if (session == NULL) {
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
2009-11-06 21:21:18 +03:00
|
|
|
|
2018-09-11 16:14:17 +03:00
|
|
|
do_sftp(session);
|
|
|
|
ssh_disconnect(session);
|
|
|
|
ssh_free(session);
|
|
|
|
return 0;
|
|
|
|
}
|
2009-11-06 21:21:18 +03:00
|
|
|
|
|
|
|
#endif
|