1998-09-13 14:40:43 +04:00
|
|
|
/* Virtual File System: FISH implementation for transfering files over
|
|
|
|
shell connections.
|
|
|
|
|
|
|
|
Copyright (C) 1998 The Free Software Foundation
|
|
|
|
|
|
|
|
Written by: 1998 Pavel Machek
|
|
|
|
|
|
|
|
Derived from ftpfs.c.
|
|
|
|
|
|
|
|
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, write to the Free Software
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read README.fish for protocol specification.
|
|
|
|
*
|
|
|
|
* Syntax of path is: /#sh:user@host[:Cr]/path
|
|
|
|
* where C means you want compressed connection,
|
|
|
|
* and r means you want to use rsh
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <pwd.h>
|
|
|
|
#include <grp.h>
|
|
|
|
#include <ctype.h> /* For isdigit */
|
|
|
|
#ifdef SCO_FLAVOR
|
|
|
|
# include <sys/timeb.h> /* alex: for struct timeb definition */
|
|
|
|
#endif /* SCO_FLAVOR */
|
|
|
|
#include <time.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#if defined(HAVE_UNISTD_H)
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#if HAVE_SYS_SELECT_H
|
|
|
|
# include <sys/select.h>
|
|
|
|
#endif
|
|
|
|
#include "../src/fs.h"
|
|
|
|
#include "../src/mad.h"
|
|
|
|
#include "../src/setup.h"
|
|
|
|
#include "../src/tty.h" /* enable/disable interrupt key */
|
|
|
|
#include "../src/main.h"
|
|
|
|
#ifndef SCO_FLAVOR
|
|
|
|
# include <sys/time.h> /* alex: this redefines struct timeval */
|
|
|
|
#endif /* SCO_FLAVOR */
|
|
|
|
#include <sys/param.h>
|
|
|
|
|
|
|
|
#include "../src/mem.h"
|
|
|
|
#define WANT_PARSE_LS_LGA
|
|
|
|
#include "vfs.h"
|
|
|
|
#include "tcputil.h"
|
|
|
|
#include "../src/util.h"
|
|
|
|
#include "../src/dialog.h"
|
|
|
|
#include "container.h"
|
|
|
|
#include "fish.h"
|
|
|
|
#ifndef MAXHOSTNAMELEN
|
|
|
|
# define MAXHOSTNAMELEN 64
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Reply codes.
|
|
|
|
*/
|
|
|
|
#define PRELIM 1 /* positive preliminary */
|
|
|
|
#define COMPLETE 2 /* positive completion */
|
|
|
|
#define CONTINUE 3 /* positive intermediate */
|
|
|
|
#define TRANSIENT 4 /* transient negative completion */
|
|
|
|
#define ERROR 5 /* permanent negative completion */
|
|
|
|
|
|
|
|
#define UPLOAD_ZERO_LENGTH_FILE
|
|
|
|
|
|
|
|
static int my_errno;
|
|
|
|
static int code;
|
|
|
|
|
|
|
|
/* Where we store the transactions */
|
|
|
|
static FILE *logfile = NULL;
|
|
|
|
|
|
|
|
/* If true, the directory cache is forced to reload */
|
|
|
|
static int force_expiration = 0;
|
|
|
|
|
|
|
|
static struct linklist *connections_list;
|
|
|
|
|
|
|
|
/* command wait_flag: */
|
|
|
|
#define NONE 0x00
|
|
|
|
#define WAIT_REPLY 0x01
|
|
|
|
#define WANT_STRING 0x02
|
|
|
|
static char reply_str [80];
|
|
|
|
|
|
|
|
static char *fish_get_current_directory(struct connection *bucket);
|
|
|
|
static void free_bucket (void *data);
|
|
|
|
static void connection_destructor(void *data);
|
|
|
|
static void flush_all_directory(struct connection *bucket);
|
|
|
|
static int get_line (int sock, char *buf, int buf_len, char term);
|
1998-09-18 15:02:04 +04:00
|
|
|
static char *get_path (struct connection **bucket, char *path);
|
1998-09-13 14:40:43 +04:00
|
|
|
|
|
|
|
static char *my_get_host_and_username (char *path, char **host, char **user, int *flags, char **pass)
|
|
|
|
{
|
|
|
|
return get_host_and_username (path, host, user, flags, 0, 0, pass);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int decode_reply (char *s, int was_garbage)
|
|
|
|
{
|
|
|
|
if (!sscanf(s, "%d", &code)) {
|
|
|
|
code = 500;
|
|
|
|
return 5;
|
|
|
|
}
|
|
|
|
if (code<100) return was_garbage ? ERROR : (!code ? COMPLETE : PRELIM);
|
|
|
|
return code / 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns a reply code, check /usr/include/arpa/ftp.h for possible values */
|
|
|
|
static int get_reply (int sock, char *string_buf, int string_len)
|
|
|
|
{
|
|
|
|
char answer[1024];
|
|
|
|
int was_garbage = 0;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
if (!get_line(sock, answer, sizeof(answer), '\n')) {
|
|
|
|
if (string_buf)
|
|
|
|
*string_buf = 0;
|
|
|
|
code = 421;
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
if (strncmp(answer, "### ", 4)) {
|
|
|
|
was_garbage = 1;
|
|
|
|
if (string_buf) {
|
|
|
|
strncpy(string_buf, answer, string_len - 1);
|
|
|
|
*(string_buf + string_len - 1) = 0;
|
|
|
|
}
|
|
|
|
} else return decode_reply(answer+4, was_garbage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int got_sigpipe = 0;
|
|
|
|
|
|
|
|
static int command (struct connection *bucket, int wait_reply,
|
|
|
|
char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
char buf[2048]; /* FIXME: buffer exceed ?? */
|
|
|
|
int n, status;
|
|
|
|
|
|
|
|
va_start (ap, fmt);
|
|
|
|
vsprintf (buf, fmt, ap);
|
|
|
|
va_end (ap);
|
|
|
|
n = strlen(buf);
|
|
|
|
buf[n] = 0;
|
|
|
|
|
|
|
|
if (logfile){
|
|
|
|
fwrite (buf, strlen (buf), 1, logfile);
|
|
|
|
fflush (logfile);
|
|
|
|
}
|
|
|
|
|
|
|
|
enable_interrupt_key();
|
|
|
|
status = write(qsockw(bucket), buf, strlen(buf));
|
|
|
|
if (status < 0){
|
|
|
|
code = 421;
|
|
|
|
if (errno == EPIPE){
|
|
|
|
got_sigpipe = 1;
|
|
|
|
}
|
|
|
|
disable_interrupt_key();
|
|
|
|
return TRANSIENT;
|
|
|
|
}
|
|
|
|
disable_interrupt_key();
|
|
|
|
|
|
|
|
if (wait_reply)
|
|
|
|
return get_reply (qsockr(bucket), (wait_reply & WANT_STRING) ? reply_str : NULL, sizeof (reply_str)-1);
|
|
|
|
return COMPLETE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
connection_close (void *data)
|
|
|
|
{
|
|
|
|
struct connection *bucket = data;
|
|
|
|
|
|
|
|
if ((qsockw (bucket) != -1) || (qsockr (bucket) != -1)){
|
|
|
|
print_vfs_message ("fish: Disconnecting from %s", qhost(bucket));
|
|
|
|
command(bucket, NONE, "logout\n");
|
|
|
|
close(qsockw(bucket));
|
|
|
|
close(qsockr(bucket));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
pipeopen(struct connection *bucket, char *path, char *argv[])
|
|
|
|
{
|
|
|
|
int fileset1[2], fileset2[2];
|
|
|
|
FILE *retval;
|
|
|
|
int res;
|
|
|
|
|
|
|
|
if (pipe(fileset1)<0) vfs_die("Could not pipe(): %m.");
|
|
|
|
if (pipe(fileset2)<0) vfs_die("Could not pipe(): %m.");
|
|
|
|
|
|
|
|
if (res = fork()) {
|
|
|
|
if (res<0) vfs_die("Could not fork(): %m.");
|
|
|
|
/* We are the parent */
|
|
|
|
close(fileset1[0]);
|
|
|
|
qsockw(bucket) = fileset1[1];
|
|
|
|
close(fileset2[1]);
|
|
|
|
qsockr(bucket) = fileset2[0];
|
|
|
|
if (!retval) vfs_die( "Could not fdopen(): %m." );
|
|
|
|
} else {
|
|
|
|
close(0);
|
|
|
|
dup(fileset1[0]);
|
|
|
|
close(fileset1[0]); close(fileset1[1]);
|
|
|
|
close(1); close(2);
|
|
|
|
dup(fileset2[1]);
|
|
|
|
dup(fileset2[1]);
|
|
|
|
close(fileset2[0]); close(fileset2[1]);
|
|
|
|
execvp(path, argv);
|
|
|
|
vfs_die("Exec failed.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct connection *
|
|
|
|
open_command_connection (char *host, char *user, int flags, char *netrcpass)
|
|
|
|
{
|
|
|
|
struct connection *bucket;
|
|
|
|
char *argv[100];
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
bucket = xmalloc(sizeof(struct connection),
|
|
|
|
"struct connection");
|
|
|
|
|
1998-09-18 15:02:04 +04:00
|
|
|
if (bucket == NULL) ERRNOR (ENOMEM, NULL);
|
1998-09-13 14:40:43 +04:00
|
|
|
|
|
|
|
qhost(bucket) = strdup (host);
|
|
|
|
quser(bucket) = strdup (user);
|
|
|
|
qcdir(bucket) = NULL;
|
|
|
|
qflags(bucket) = flags;
|
|
|
|
qlock(bucket) = 0;
|
|
|
|
qhome(bucket) = NULL;
|
|
|
|
qupdir(bucket)= 0;
|
|
|
|
qdcache(bucket)=0;
|
|
|
|
bucket->__inode_counter = 0;
|
|
|
|
bucket->lock = 0;
|
|
|
|
bucket->password = 0;
|
|
|
|
|
|
|
|
my_errno = ENOMEM;
|
|
|
|
if ((qdcache(bucket) = linklist_init()) == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
#define XSH (flags == FISH_FLAG_RSH ? "rsh" : "ssh")
|
|
|
|
argv[i++] = XSH;
|
|
|
|
argv[i++] = "-l";
|
|
|
|
argv[i++] = user;
|
|
|
|
argv[i++] = host;
|
|
|
|
if (flags == FISH_FLAG_COMPRESSED)
|
|
|
|
argv[i++] = "-C";
|
|
|
|
argv[i++] = "echo FISH:; /bin/sh";
|
|
|
|
argv[i++] = NULL;
|
|
|
|
|
|
|
|
pipeopen(bucket, XSH, argv );
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
char answer[2048];
|
|
|
|
print_vfs_message( "FISH: Waiting for initial line..." );
|
|
|
|
if (!get_line(qsockr(bucket), answer, sizeof(answer), ':'))
|
|
|
|
goto error_2;
|
|
|
|
print_vfs_message( answer );
|
|
|
|
if (strstr(answer, "assword")) {
|
|
|
|
|
|
|
|
/* Currently, this does not work. ssh reads passwords from
|
|
|
|
/dev/tty, not from stdin :-(. */
|
|
|
|
|
|
|
|
message_1s (1, MSG_ERROR, _("Sorry, we can not do password authenticated connections for now."));
|
|
|
|
my_errno = EPERM;
|
|
|
|
goto error_2;
|
|
|
|
|
|
|
|
if (!bucket->password){
|
|
|
|
char *p, *op;
|
|
|
|
p = copy_strings (" FISH: Password required for ", quser(bucket),
|
|
|
|
" ", NULL);
|
|
|
|
op = vfs_get_password (p);
|
|
|
|
free (p);
|
|
|
|
my_errno = EPERM;
|
|
|
|
if (op == NULL)
|
|
|
|
goto error_2;
|
|
|
|
bucket->password = strdup (op);
|
|
|
|
wipe_password(op);
|
|
|
|
}
|
|
|
|
print_vfs_message( "FISH: Sending password..." );
|
|
|
|
write(qsockw(bucket), bucket->password, strlen(bucket->password));
|
|
|
|
write(qsockw(bucket), "\r\n", 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
print_vfs_message( "FISH: Sending initial line..." );
|
|
|
|
my_errno = ECONNREFUSED;
|
|
|
|
if (command (bucket, WAIT_REPLY, "#FISH\necho; start_fish_server; echo '### 200'\n") != COMPLETE)
|
|
|
|
goto error_2;
|
|
|
|
|
|
|
|
print_vfs_message( "FISH: Handshaking version..." );
|
|
|
|
if (command (bucket, WAIT_REPLY, "#VER 0.0.0\necho '### 000'\n") != COMPLETE)
|
|
|
|
goto error_2;
|
|
|
|
|
|
|
|
print_vfs_message( "FISH: Setting up current directory..." );
|
|
|
|
qhome(bucket) = fish_get_current_directory (bucket);
|
|
|
|
if (!qhome(bucket))
|
|
|
|
qhome(bucket) = strdup ("/");
|
|
|
|
print_vfs_message( "FISH: Connected." );
|
|
|
|
return bucket;
|
|
|
|
|
|
|
|
error_2:
|
|
|
|
close(qsockr(bucket));
|
|
|
|
close(qsockw(bucket));
|
|
|
|
error:
|
|
|
|
free (qhost(bucket));
|
|
|
|
free (quser(bucket));
|
|
|
|
free (bucket);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This routine keeps track of open connections */
|
|
|
|
/* Returns a connected socket to host */
|
|
|
|
static struct connection *
|
|
|
|
open_link (char *host, char *user, int flags, char *netrcpass)
|
|
|
|
{
|
|
|
|
struct connection *bucket;
|
|
|
|
struct linklist *lptr;
|
|
|
|
|
|
|
|
for (lptr = connections_list->next;
|
|
|
|
lptr != connections_list; lptr = lptr->next) {
|
|
|
|
bucket = lptr->data;
|
|
|
|
if ((strcmp (host, qhost(bucket)) == 0) &&
|
|
|
|
(strcmp (user, quser(bucket)) == 0) &&
|
|
|
|
(flags == qflags(bucket)))
|
|
|
|
return bucket;
|
|
|
|
}
|
|
|
|
bucket = open_command_connection(host, user, flags, netrcpass);
|
|
|
|
if (bucket == NULL)
|
|
|
|
return NULL;
|
|
|
|
if (!linklist_insert(connections_list, bucket)) {
|
|
|
|
connection_destructor(bucket);
|
1998-09-18 15:02:04 +04:00
|
|
|
ERRNOR (ENOMEM, NULL);
|
1998-09-13 14:40:43 +04:00
|
|
|
}
|
|
|
|
return bucket;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The returned directory should always contain a trailing slash */
|
|
|
|
static char *fish_get_current_directory(struct connection *bucket)
|
|
|
|
{
|
|
|
|
if (command(bucket, WANT_STRING, "#PWD\npwd; echo '### 200'\n") == COMPLETE)
|
|
|
|
return copy_strings(reply_str, "/", NULL);
|
1998-09-18 15:02:04 +04:00
|
|
|
ERRNOR (EIO, NULL);
|
1998-09-13 14:40:43 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
#define X "fish"
|
|
|
|
#define X_myname "/#sh:"
|
|
|
|
#define X_vfs_ops fish_vfs_ops
|
|
|
|
#define X_fill_names fish_fill_names
|
|
|
|
#define X_hint_reread fish_hint_reread
|
|
|
|
#define X_flushdir fish_flushdir
|
|
|
|
#define X_done fish_done
|
|
|
|
#include "shared_ftp_fish.c"
|
|
|
|
|
1998-09-18 15:02:04 +04:00
|
|
|
static char*
|
|
|
|
get_path (struct connection **bucket, char *path)
|
|
|
|
{
|
|
|
|
char *res;
|
|
|
|
if (res = s_get_path (bucket, path, "/#sh:"))
|
|
|
|
return res;
|
|
|
|
if (res = s_get_path (bucket, path, "/#ssh:"))
|
|
|
|
return res;
|
|
|
|
if (res = s_get_path (bucket, path, "/#rsh:")) {
|
|
|
|
qflags((*bucket)) |= FISH_FLAG_RSH;
|
|
|
|
return res;
|
|
|
|
}
|
1998-09-18 18:28:07 +04:00
|
|
|
return NULL;
|
1998-09-18 15:02:04 +04:00
|
|
|
}
|
|
|
|
|
1998-09-13 14:40:43 +04:00
|
|
|
/*
|
|
|
|
* This is the 'new' code
|
|
|
|
*/
|
1998-09-15 23:41:22 +04:00
|
|
|
/*
|
|
|
|
* Last parameter (resolve_symlinks) is currently not used. Due to
|
|
|
|
* the code sharing (file shared_ftp_fish.c) the fish and ftp interface
|
|
|
|
* have to be the same (Norbert).
|
|
|
|
*/
|
1998-09-13 14:40:43 +04:00
|
|
|
|
|
|
|
static struct dir *
|
1998-09-15 23:41:22 +04:00
|
|
|
retrieve_dir(struct connection *bucket, char *remote_path, int resolve_symlinks)
|
1998-09-13 14:40:43 +04:00
|
|
|
{
|
1998-09-18 18:28:07 +04:00
|
|
|
int has_symlinks;
|
1998-09-13 14:40:43 +04:00
|
|
|
struct linklist *file_list, *p;
|
|
|
|
struct direntry *fe;
|
|
|
|
char buffer[8192];
|
|
|
|
struct dir *dcache;
|
|
|
|
int got_intr = 0;
|
|
|
|
|
|
|
|
for (p = qdcache(bucket)->next;p != qdcache(bucket);
|
|
|
|
p = p->next) {
|
|
|
|
dcache = p->data;
|
|
|
|
if (strcmp(dcache->remote_path, remote_path) == 0) {
|
|
|
|
struct timeval tim;
|
|
|
|
|
|
|
|
gettimeofday(&tim, NULL);
|
|
|
|
if ((tim.tv_sec < dcache->timestamp.tv_sec) && !force_expiration)
|
|
|
|
return dcache;
|
|
|
|
else {
|
|
|
|
force_expiration = 0;
|
|
|
|
p->next->prev = p->prev;
|
|
|
|
p->prev->next = p->next;
|
|
|
|
dir_destructor(dcache);
|
|
|
|
free (p);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
has_symlinks = 0;
|
|
|
|
print_vfs_message("fish: Reading FTP directory...");
|
|
|
|
|
|
|
|
my_errno = ENOMEM;
|
|
|
|
if (!(file_list = linklist_init()))
|
|
|
|
return NULL;
|
|
|
|
if (!(dcache = xmalloc(sizeof(struct dir), "struct dir"))) {
|
|
|
|
linklist_destroy(file_list, NULL);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
gettimeofday(&dcache->timestamp, NULL);
|
|
|
|
dcache->timestamp.tv_sec += 360;
|
|
|
|
dcache->file_list = file_list;
|
|
|
|
dcache->remote_path = strdup(remote_path);
|
|
|
|
dcache->count = 1;
|
|
|
|
|
|
|
|
command(bucket, NONE,
|
|
|
|
"#LIST %s\nls -lLa %s | grep '^[^cbt]' | ( while read p x u g s m d y n; do echo \"P$p $u.$g\n"
|
|
|
|
"S$s\nd$m $d $y\n:$n\n\"; done )\n"
|
|
|
|
"ls -lLa %s | grep '^[cb]' | ( while read p x u g a i m d y n; do echo \"P$p $u.$g\n"
|
|
|
|
"E$a$i\nd$m $d $y\n:$n\n\"; done ); echo '### 200'\n",
|
|
|
|
remote_path, remote_path, remote_path);
|
|
|
|
|
|
|
|
/* Clear the interrupt flag */
|
|
|
|
enable_interrupt_key ();
|
|
|
|
|
|
|
|
fe = NULL;
|
|
|
|
errno = 0;
|
|
|
|
my_errno = ENOMEM;
|
|
|
|
while ((got_intr = get_line_interruptible (buffer, sizeof (buffer), qsockr(bucket))) != EINTR){
|
|
|
|
int eof = (got_intr == 0);
|
|
|
|
|
|
|
|
if (logfile){
|
|
|
|
fputs (buffer, logfile);
|
|
|
|
fputs ("\n", logfile);
|
|
|
|
fflush (logfile);
|
|
|
|
}
|
|
|
|
if (eof) {
|
|
|
|
if (fe)
|
|
|
|
free(fe);
|
|
|
|
my_errno = ECONNRESET;
|
|
|
|
goto error_1;
|
|
|
|
}
|
|
|
|
if (!strncmp(buffer, "### ", 4))
|
|
|
|
break;
|
|
|
|
if ((!buffer[0]) && fe) {
|
|
|
|
if (!linklist_insert(file_list, fe)) {
|
|
|
|
free(fe);
|
|
|
|
goto error_1;
|
|
|
|
}
|
|
|
|
fe = NULL;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!fe) {
|
|
|
|
if (!(fe = xmalloc(sizeof(struct direntry), "struct direntry")))
|
|
|
|
goto error_1;
|
|
|
|
bzero(fe, sizeof(struct direntry));
|
|
|
|
fe->count = 1;
|
|
|
|
fe->bucket = bucket;
|
|
|
|
fe->s.st_ino = bucket->__inode_counter++;
|
|
|
|
fe->s.st_nlink = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(buffer[0]) {
|
|
|
|
case ':': fe->name = strdup(buffer+1); break;
|
|
|
|
case 'S': fe->s.st_size = atoi(buffer+1); break;
|
|
|
|
case 'P': {
|
|
|
|
int i;
|
|
|
|
if ((i = parse_filetype(buffer[1])) ==-1)
|
|
|
|
break;
|
|
|
|
fe->s.st_mode = i;
|
|
|
|
if ((i = parse_filemode(buffer+2)) ==-1)
|
|
|
|
break;
|
|
|
|
fe->s.st_mode |= i;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'd': {
|
|
|
|
split_text(buffer+1);
|
|
|
|
if (!parse_filedate(0, &fe->s.st_ctime))
|
|
|
|
break;
|
|
|
|
fe->s.st_atime = fe->s.st_mtime = fe->s.st_ctime;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'D': {
|
|
|
|
struct tm tim;
|
|
|
|
if (sscanf(buffer+1, "%d %d %d %d %d %d", &tim.tm_year, &tim.tm_mon,
|
|
|
|
&tim.tm_mday, &tim.tm_hour, &tim.tm_min, &tim.tm_sec) != 6)
|
|
|
|
break;
|
|
|
|
fe->s.st_atime = fe->s.st_mtime = fe->s.st_ctime = mktime(&tim);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'E': {
|
|
|
|
int maj, min;
|
|
|
|
if (sscanf(buffer+1, "%d,%d", &maj, &min) != 2)
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_ST_RDEV
|
|
|
|
fe->s.st_rdev = (maj << 8) | min;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
case 'L': fe->linkname = strdup(buffer+1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
disable_interrupt_key();
|
|
|
|
#if 0
|
|
|
|
if (got_intr)
|
|
|
|
vfs_die("fish: reading FTP directory interrupted by user");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (decode_reply(buffer+4, 0) != COMPLETE) {
|
|
|
|
my_errno = EIO;
|
|
|
|
goto error_3;
|
|
|
|
}
|
|
|
|
if (file_list->next == file_list) {
|
|
|
|
my_errno = EACCES;
|
|
|
|
goto error_3;
|
|
|
|
}
|
|
|
|
if (!linklist_insert(qdcache(bucket), dcache)) {
|
|
|
|
my_errno = ENOMEM;
|
|
|
|
goto error_3;
|
|
|
|
}
|
|
|
|
print_vfs_message("fish: got listing");
|
|
|
|
return dcache;
|
|
|
|
error_1:
|
|
|
|
disable_interrupt_key();
|
|
|
|
error_3:
|
|
|
|
free(dcache->remote_path);
|
|
|
|
free(dcache);
|
|
|
|
linklist_destroy(file_list, direntry_destructor);
|
|
|
|
print_vfs_message("fish: failed");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
store_file(struct direntry *fe)
|
|
|
|
{
|
1998-09-18 18:28:07 +04:00
|
|
|
int local_handle, n, total;
|
1998-09-13 14:40:43 +04:00
|
|
|
char buffer[8192];
|
|
|
|
struct stat s;
|
|
|
|
int was_error = 0;
|
|
|
|
|
|
|
|
local_handle = open(fe->local_filename, O_RDONLY);
|
|
|
|
unlink (fe->local_filename);
|
|
|
|
my_errno = EIO;
|
|
|
|
if (local_handle == -1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fstat(local_handle, &s);
|
|
|
|
|
|
|
|
/* Use this as stor: ( dd block ; dd malyblock ) | ( cat > file; cat > /dev/null ) */
|
|
|
|
|
|
|
|
print_vfs_message("FISH: store: sending command..." );
|
|
|
|
if (command (fe->bucket, WAIT_REPLY,
|
|
|
|
"#STOR %d %s\n> %s; echo '### 001'; ( dd bs=4096 count=%d; dd bs=%d count=1 ) 2>/dev/null | ( cat > %s; cat > /dev/null ); echo '### 200'\n",
|
|
|
|
s.st_size, fe->remote_filename,
|
|
|
|
fe->remote_filename,
|
|
|
|
s.st_size / 4096, s.st_size % 4096, fe->remote_filename)
|
|
|
|
!= PRELIM)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
total = 0;
|
|
|
|
|
|
|
|
enable_interrupt_key();
|
|
|
|
while (1) {
|
|
|
|
while ((n = read(local_handle, buffer, sizeof(buffer))) < 0) {
|
|
|
|
if ((errno == EINTR) && got_interrupt)
|
|
|
|
continue;
|
|
|
|
print_vfs_message("FISH: Local read failed, sending zeros" );
|
|
|
|
close(local_handle);
|
|
|
|
local_handle = open( "/dev/zero", O_RDONLY );
|
|
|
|
}
|
|
|
|
if (n == 0)
|
|
|
|
break;
|
|
|
|
while (write(qsockw(fe->bucket), buffer, n) < 0) {
|
|
|
|
if (errno == EINTR) {
|
|
|
|
if (got_interrupt()) {
|
|
|
|
my_errno = EINTR;
|
|
|
|
goto error_return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
my_errno = errno;
|
|
|
|
goto error_return;
|
|
|
|
}
|
|
|
|
total += n;
|
|
|
|
print_vfs_message("fish: storing %s %d (%d)",
|
|
|
|
was_error ? "zeros" : "file", total, s.st_size);
|
|
|
|
}
|
|
|
|
disable_interrupt_key();
|
|
|
|
close(local_handle);
|
1998-09-18 15:02:04 +04:00
|
|
|
if (get_reply (qsockr(fe->bucket), NULL, 0) != COMPLETE)
|
|
|
|
ERRNOR (EIO, 0);
|
1998-09-13 14:40:43 +04:00
|
|
|
return (!was_error);
|
|
|
|
error_return:
|
|
|
|
disable_interrupt_key();
|
|
|
|
close(local_handle);
|
|
|
|
get_reply(qsockr(fe->bucket), NULL, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For _ctl routine */
|
|
|
|
static int remotelocal_handle, remoten = 0, remotestat_size;
|
|
|
|
|
|
|
|
static void
|
|
|
|
fish_abort (struct connection *bucket)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
|
|
|
|
print_vfs_message( "Aborting transfer..." );
|
|
|
|
do {
|
|
|
|
n = remotestat_size - remotetotal;
|
|
|
|
n = (n>8192) ? 8192:n;
|
|
|
|
if (n)
|
|
|
|
if ((n = read(qsockr(remoteent->bucket), remotebuffer, n)) < 0)
|
|
|
|
return;
|
|
|
|
} while (n);
|
|
|
|
|
|
|
|
if (get_reply (qsockr(remoteent->bucket), NULL, 0) != COMPLETE)
|
|
|
|
print_vfs_message( "Error reported after abort." );
|
|
|
|
else
|
|
|
|
print_vfs_message( "Aborted transfer would be successfull." );
|
|
|
|
}
|
|
|
|
|
|
|
|
static int retrieve_file_start(struct direntry *fe)
|
|
|
|
{
|
1998-09-18 15:02:04 +04:00
|
|
|
if (fe->local_filename == NULL)
|
|
|
|
ERRNOR (ENOMEM, 0);
|
1998-09-13 14:40:43 +04:00
|
|
|
if (command(fe->bucket, WANT_STRING,
|
|
|
|
"#RETR %s\nls -l %s | ( read var1 var2 var3 var4 var5 var6; echo $var5 ); echo '### 100'; cat %s; echo '### 200'\n",
|
|
|
|
fe->remote_filename, fe->remote_filename, fe->remote_filename )
|
1998-09-18 15:02:04 +04:00
|
|
|
!= PRELIM) ERRNOR (EPERM, 0);
|
1998-09-13 14:40:43 +04:00
|
|
|
|
|
|
|
remotestat_size = atoi(reply_str);
|
|
|
|
|
|
|
|
remotetotal = 0;
|
|
|
|
remoteent = fe;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int retrieve_file_start2(struct direntry *fe)
|
|
|
|
{
|
|
|
|
remotebuffer = xmalloc (8192, "retrieve_file");
|
|
|
|
remotelocal_handle = open(fe->local_filename, O_RDWR | O_CREAT | O_TRUNC | O_EXCL, 0600);
|
|
|
|
if (remotelocal_handle == -1) {
|
|
|
|
fish_abort(remoteent->bucket);
|
|
|
|
free(remotebuffer);
|
|
|
|
free(fe->local_filename);
|
|
|
|
fe->local_filename = NULL;
|
|
|
|
fe->local_is_temp = 1;
|
1998-09-18 15:02:04 +04:00
|
|
|
ERRNOR (EIO, 0);
|
1998-09-13 14:40:43 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int fish_ctl (void *data, int ctlop, int arg)
|
|
|
|
{
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
switch (ctlop) {
|
|
|
|
case MCCTL_ISREMOTECOPY:
|
|
|
|
return isremotecopy;
|
|
|
|
|
|
|
|
case MCCTL_REMOTECOPYCHUNK:
|
|
|
|
if (!transfer_started)
|
|
|
|
if (!retrieve_file_start2 (remoteent)){
|
|
|
|
return MCERR_TARGETOPEN;
|
|
|
|
} else
|
|
|
|
transfer_started = 1;
|
|
|
|
|
|
|
|
enable_interrupt_key ();
|
|
|
|
if (!remoten) {
|
|
|
|
int v = select_on_two (qsockr(remoteent->bucket), 0);
|
|
|
|
|
|
|
|
if (((v < 0) && (errno == EINTR)) || v == 0){
|
|
|
|
disable_interrupt_key ();
|
|
|
|
return MCERR_DATA_ON_STDIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
n = remotestat_size - remotetotal;
|
|
|
|
n = (n>8192) ? 8192:n;
|
|
|
|
if (n)
|
|
|
|
if ((n = read(qsockr(remoteent->bucket), remotebuffer, n)) < 0){
|
|
|
|
disable_interrupt_key ();
|
|
|
|
if (errno == EINTR)
|
|
|
|
return MCERR_DATA_ON_STDIN;
|
|
|
|
else
|
|
|
|
return MCERR_READ;
|
|
|
|
}
|
|
|
|
if (!n) {
|
|
|
|
if (get_reply (qsockr(remoteent->bucket), NULL, 0) != COMPLETE)
|
|
|
|
my_errno = EIO;
|
|
|
|
close(remotelocal_handle);
|
|
|
|
if (localname){
|
|
|
|
free (localname);
|
|
|
|
localname = NULL;
|
|
|
|
}
|
|
|
|
disable_interrupt_key ();
|
|
|
|
transfer_started = 0;
|
|
|
|
return MCERR_FINISH;
|
|
|
|
}
|
|
|
|
disable_interrupt_key ();
|
|
|
|
remotetotal += n;
|
|
|
|
remoten = n;
|
|
|
|
} else
|
|
|
|
n = remoten;
|
|
|
|
if (write(remotelocal_handle, remotebuffer, remoten) < 0)
|
|
|
|
return MCERR_WRITE;
|
|
|
|
remoten = 0;
|
|
|
|
return n;
|
|
|
|
|
|
|
|
/* We get this message if the transfer was aborted */
|
|
|
|
case MCCTL_FINISHREMOTE:
|
|
|
|
if (localname) {
|
|
|
|
free (localname);
|
|
|
|
localname = NULL;
|
|
|
|
}
|
|
|
|
if (!arg) { /* OK */
|
|
|
|
if (stat (remoteent->local_filename, &remoteent->local_stat) < 0)
|
|
|
|
remoteent->local_stat.st_mtime = 0;
|
|
|
|
} else
|
|
|
|
remoteent->local_stat.st_mtime = 0;
|
|
|
|
transfer_started = 0;
|
|
|
|
fish_abort (remoteent->bucket);
|
|
|
|
my_errno = EINTR;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int retrieve_file(struct direntry *fe)
|
|
|
|
{
|
|
|
|
if (fe->local_filename)
|
|
|
|
return 1;
|
|
|
|
fe->local_stat.st_mtime = 0;
|
|
|
|
fe->local_filename = tempnam (NULL, "fish");
|
|
|
|
fe->local_is_temp = 1;
|
|
|
|
isremotecopy = 0;
|
|
|
|
|
|
|
|
my_errno = ENOMEM;
|
|
|
|
if (!fe->local_filename ||
|
|
|
|
!retrieve_file_start(fe))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Clear the interrupt status */
|
|
|
|
enable_interrupt_key ();
|
|
|
|
my_errno = 0;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
int res;
|
|
|
|
|
|
|
|
res = fish_ctl(NULL, MCCTL_REMOTECOPYCHUNK, 0 );
|
|
|
|
if ((res == MCERR_TARGETOPEN) ||
|
|
|
|
(res == MCERR_WRITE) ||
|
|
|
|
(res == MCERR_DATA_ON_STDIN) ||
|
|
|
|
(res == MCERR_READ)) {
|
|
|
|
my_errno = EIO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (res == MCERR_FINISH)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (remotestat_size == 0)
|
|
|
|
print_vfs_message ("fish: Getting file: %ld bytes transfered",
|
|
|
|
remotetotal);
|
|
|
|
else
|
|
|
|
print_vfs_message ("fish: Getting file: %3d%% (%ld bytes transfered)",
|
|
|
|
remotetotal*100/remotestat_size, remotetotal);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (my_errno) {
|
|
|
|
disable_interrupt_key ();
|
|
|
|
fish_abort( remoteent->bucket );
|
|
|
|
unlink(fe->local_filename);
|
|
|
|
free(fe->local_filename);
|
|
|
|
fe->local_filename = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stat (fe->local_filename, &fe->local_stat) < 0)
|
|
|
|
fe->local_stat.st_mtime = 0;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
send_fish_command(struct connection *bucket, char *cmd, int flags)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
int flush_directory_cache = (flags & OPT_FLUSH) && (normal_flush > 0);
|
|
|
|
|
|
|
|
r = command (bucket, WAIT_REPLY, cmd);
|
|
|
|
vfs_add_noncurrent_stamps (&fish_vfs_ops, (vfsid) bucket, NULL);
|
1998-09-18 15:02:04 +04:00
|
|
|
if (r != COMPLETE) ERRNOR (EPERM, -1);
|
1998-09-13 14:40:43 +04:00
|
|
|
if (flush_directory_cache)
|
|
|
|
flush_all_directory(bucket);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
fish_init (void)
|
|
|
|
{
|
|
|
|
connections_list = linklist_init();
|
|
|
|
logfile = fopen ("/tmp/talk.fish", "w+");
|
|
|
|
}
|
|
|
|
|
|
|
|
#define PREFIX \
|
|
|
|
char buf[999]; \
|
|
|
|
char *remote_path; \
|
|
|
|
struct connection *bucket; \
|
|
|
|
if (!(remote_path = get_path(&bucket, path))) \
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
#define POSTFIX(flags) \
|
|
|
|
free(remote_path); \
|
|
|
|
return send_fish_command(bucket, buf, flags);
|
|
|
|
|
|
|
|
int fish_chmod (char *path, int mode)
|
|
|
|
{
|
|
|
|
PREFIX
|
|
|
|
sprintf(buf, "#CHMOD %4.4o %s\nchmod %4.4o %s; echo '### 000'\n",
|
|
|
|
mode & 07777, remote_path,
|
|
|
|
mode & 07777, remote_path);
|
|
|
|
POSTFIX(OPT_FLUSH);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define FISH_OP(name, chk, string) \
|
|
|
|
int fish_##name (char *path1, char *path2) \
|
|
|
|
{ \
|
|
|
|
char buf[120]; \
|
|
|
|
char *remote_path1 = NULL, *remote_path2 = NULL; \
|
|
|
|
struct connection *bucket1, *bucket2; \
|
|
|
|
if (!(remote_path1 = get_path(&bucket1, path1))) \
|
|
|
|
return -1; \
|
|
|
|
if (!(remote_path2 = get_path(&bucket2, path2))) { \
|
|
|
|
free(remote_path1); \
|
|
|
|
return -1; \
|
|
|
|
} \
|
|
|
|
sprintf(buf, string, path1, path2, path1, path2 ); \
|
|
|
|
free(remote_path1); \
|
|
|
|
free(remote_path2); \
|
|
|
|
return send_fish_command(bucket2, buf, OPT_FLUSH); \
|
|
|
|
}
|
|
|
|
|
1998-09-18 15:02:04 +04:00
|
|
|
#define XTEST if (bucket1 != bucket2) { free(remote_path1); free(remote_path2); ERRNOR (EXDEV, -1); }
|
1998-09-13 14:40:43 +04:00
|
|
|
FISH_OP(rename, XTEST, "#RENAME %s %s\nmv %s %s; echo '*** 000'" );
|
|
|
|
FISH_OP(link, XTEST, "#LINK %s %s\nln %s %s; echo '*** 000'" );
|
|
|
|
FISH_OP(symlink, , "#SYMLINK %s %s\nln -s %s %s; echo '*** 000'" );
|
|
|
|
|
|
|
|
int fish_chown (char *path, int owner, int group)
|
|
|
|
{
|
|
|
|
char *sowner, *sgroup;
|
|
|
|
PREFIX
|
|
|
|
sowner = getpwuid( owner )->pw_name;
|
|
|
|
sgroup = getgrgid( group )->gr_name;
|
|
|
|
sprintf(buf, "#CHOWN %s %s\nchown %s %s; echo '### 000'\n",
|
|
|
|
sowner, remote_path,
|
|
|
|
sowner, remote_path);
|
|
|
|
send_fish_command(bucket, buf, OPT_FLUSH);
|
|
|
|
/* FIXME: what should we report if chgrp succeeds but chown fails? */
|
|
|
|
sprintf(buf, "#CHGRP %s %s\nchgrp %s %s; echo '### 000'\n",
|
|
|
|
sgroup, remote_path,
|
|
|
|
sgroup, remote_path);
|
|
|
|
free(remote_path);
|
|
|
|
POSTFIX(OPT_FLUSH)
|
|
|
|
}
|
|
|
|
|
|
|
|
static int fish_unlink (char *path)
|
|
|
|
{
|
|
|
|
PREFIX
|
|
|
|
sprintf(buf, "#DELE %s\nrm -f %s; echo '### 000'\n", remote_path, remote_path);
|
|
|
|
POSTFIX(OPT_FLUSH);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int fish_mkdir (char *path, mode_t mode)
|
|
|
|
{
|
|
|
|
PREFIX
|
|
|
|
sprintf(buf, "#MKD %s\nmkdir %s; echo '### 000'\n", remote_path, remote_path);
|
|
|
|
POSTFIX(OPT_FLUSH);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int fish_rmdir (char *path)
|
|
|
|
{
|
|
|
|
PREFIX
|
|
|
|
sprintf(buf, "#RMD %s\nrmdir %s; echo '### 000'\n", remote_path, remote_path);
|
|
|
|
POSTFIX(OPT_FLUSH);
|
|
|
|
}
|
|
|
|
|
|
|
|
void fish_set_debug (char *file)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void fish_forget (char *file)
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
struct linklist *l;
|
|
|
|
char *host, *user, *pass, *rp;
|
|
|
|
int port;
|
|
|
|
|
|
|
|
#ifndef BROKEN_PATHS
|
|
|
|
if (strncmp (file, "/#sh:", 5))
|
|
|
|
return; /* Normal: consider cd /bla/#ftp */
|
|
|
|
#else
|
|
|
|
if (!(file = strstr (file, "/#sh:")))
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
file += 6;
|
|
|
|
if (!(rp = fish_get_host_and_username (file, &host, &user, &port, &pass))) {
|
|
|
|
free (host);
|
|
|
|
free (user);
|
|
|
|
if (pass)
|
|
|
|
wipe_password (pass);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we do not care about the path actually */
|
|
|
|
free (rp);
|
|
|
|
|
|
|
|
for (l = connections_list->next; l != connections_list; l = l->next){
|
|
|
|
struct connection *bucket = l->data;
|
|
|
|
|
|
|
|
if ((strcmp (host, qhost (bucket)) == 0) &&
|
|
|
|
(strcmp (user, quser (bucket)) == 0) &&
|
|
|
|
(port == qport (bucket))){
|
|
|
|
|
|
|
|
/* close socket: the child owns it now */
|
|
|
|
close (bucket->sock);
|
|
|
|
bucket->sock = -1;
|
|
|
|
|
|
|
|
/* reopen the connection */
|
|
|
|
bucket->sock = fish_open_socket (bucket);
|
|
|
|
if (bucket->sock != -1)
|
|
|
|
login_server (bucket, pass);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free (host);
|
|
|
|
free (user);
|
|
|
|
if (pass)
|
|
|
|
wipe_password (pass);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
vfs fish_vfs_ops = {
|
|
|
|
s_open,
|
|
|
|
s_close,
|
|
|
|
s_read,
|
|
|
|
s_write,
|
|
|
|
|
|
|
|
s_opendir,
|
|
|
|
s_readdir,
|
|
|
|
s_closedir,
|
|
|
|
s_telldir,
|
|
|
|
s_seekdir,
|
|
|
|
|
|
|
|
s_stat,
|
|
|
|
s_lstat,
|
|
|
|
s_fstat,
|
|
|
|
|
|
|
|
fish_chmod,
|
|
|
|
fish_chown, /* not really implemented but returns success */
|
|
|
|
NULL, /* utime */
|
|
|
|
|
|
|
|
s_readlink,
|
|
|
|
fish_symlink, /* symlink */
|
|
|
|
fish_link, /* link */
|
|
|
|
fish_unlink,
|
|
|
|
|
|
|
|
fish_rename, /* rename */
|
|
|
|
s_chdir,
|
|
|
|
s_errno,
|
|
|
|
s_lseek,
|
|
|
|
NULL, /* mknod */
|
|
|
|
|
|
|
|
s_getid,
|
|
|
|
s_nothingisopen,
|
|
|
|
s_free,
|
|
|
|
|
|
|
|
s_getlocalcopy,
|
|
|
|
s_ungetlocalcopy,
|
|
|
|
|
|
|
|
fish_mkdir,
|
|
|
|
fish_rmdir,
|
|
|
|
fish_ctl,
|
|
|
|
s_setctl,
|
|
|
|
fish_forget
|
|
|
|
#ifdef HAVE_MMAP
|
|
|
|
, NULL,
|
|
|
|
NULL
|
|
|
|
#endif
|
|
|
|
};
|