Merge branch '2242_fish_external_scripts'
* 2242_fish_external_scripts: Applied MC indentation policy. Updated protocol description in README.fish Ticket #2242 (improved FISH)
Этот коммит содержится в:
Коммит
a6bc7200ca
@ -616,6 +616,8 @@ lib/vfs/mc-vfs/extfs/urar
|
||||
lib/vfs/mc-vfs/extfs/uzip
|
||||
lib/vfs/mc-vfs/extfs/uzoo
|
||||
|
||||
lib/vfs/mc-vfs/fish/Makefile
|
||||
|
||||
misc/syntax/Makefile
|
||||
|
||||
doc/Makefile
|
||||
|
@ -29,6 +29,23 @@
|
||||
#define CHARSETS_INDEX "mc.charsets"
|
||||
#define MC_LIB_EXT "mc.ext"
|
||||
|
||||
#define FISH_PREFIX "fish"
|
||||
|
||||
#define FISH_LS_FILE "ls"
|
||||
#define FISH_EXISTS_FILE "fexists"
|
||||
#define FISH_MKDIR_FILE "mkdir"
|
||||
#define FISH_UNLINK_FILE "unlink"
|
||||
#define FISH_CHOWN_FILE "chown"
|
||||
#define FISH_CHMOD_FILE "chmod"
|
||||
#define FISH_RMDIR_FILE "rmdir"
|
||||
#define FISH_LN_FILE "ln"
|
||||
#define FISH_MV_FILE "mv"
|
||||
#define FISH_HARDLINK_FILE "hardlink"
|
||||
#define FISH_GET_FILE "get"
|
||||
#define FISH_SEND_FILE "send"
|
||||
#define FISH_APPEND_FILE "append"
|
||||
#define FISH_INFO_FILE "info"
|
||||
|
||||
#define MC_EXTFS_DIR "extfs.d"
|
||||
|
||||
#define MC_BASHRC_FILE "bashrc"
|
||||
|
@ -3,9 +3,9 @@ SAMBA_CFLAGS = -DCONFIGDIR=\""@configdir@"\"
|
||||
SAMBA_SUBDIRS = samba
|
||||
endif
|
||||
|
||||
DIST_SUBDIRS = extfs
|
||||
DIST_SUBDIRS = extfs fish
|
||||
|
||||
SUBDIRS = extfs $(SAMBA_SUBDIRS)
|
||||
SUBDIRS = extfs fish $(SAMBA_SUBDIRS)
|
||||
AM_CFLAGS = $(GLIB_CFLAGS) -I$(top_srcdir) $(SAMBA_CFLAGS)
|
||||
|
||||
AM_CPPFLAGS = -DLIBEXECDIR=\""$(libexecdir)/@PACKAGE@/"\"
|
||||
@ -13,9 +13,9 @@ AM_CPPFLAGS = -DLIBEXECDIR=\""$(libexecdir)/@PACKAGE@/"\"
|
||||
BASICFILES = \
|
||||
cpio.c \
|
||||
direntry.c \
|
||||
extfs.c \
|
||||
gc.c \
|
||||
local.c \
|
||||
extfs.c \
|
||||
gc.c \
|
||||
local.c \
|
||||
tar.c \
|
||||
sfs.c \
|
||||
utilvfs.c \
|
||||
@ -23,12 +23,13 @@ BASICFILES = \
|
||||
|
||||
VFSHDRS = \
|
||||
fish.h \
|
||||
ftpfs.h \
|
||||
gc.h \
|
||||
fishdef.h \
|
||||
ftpfs.h \
|
||||
gc.h \
|
||||
local.h \
|
||||
mcfs.h \
|
||||
mcfsutil.h \
|
||||
netutil.h \
|
||||
mcfs.h \
|
||||
mcfsutil.h \
|
||||
netutil.h \
|
||||
smbfs.h \
|
||||
utilvfs.h \
|
||||
vfs.h \
|
||||
@ -49,7 +50,7 @@ NETFILES = netutil.c fish.c ftpfs.c mcfs.c mcfsutil.c $(SMB_NETFILES)
|
||||
|
||||
NONETFILES = $(BASICFILES) $(UNDEL_FILES)
|
||||
|
||||
EXTRA_DIST = HACKING README README.fish \
|
||||
EXTRA_DIST = HACKING README \
|
||||
$(VFSHDRS) $(BASICFILES) $(NETFILES) $(SMBFILES) $(UNDELFILES)
|
||||
|
||||
dist-hook:
|
||||
|
@ -6,6 +6,8 @@
|
||||
|
||||
Written by: 1998 Pavel Machek
|
||||
Spaces fix: 2000 Michal Svec
|
||||
2010 Andrew Borobin
|
||||
2010 Ilia Maslakov
|
||||
|
||||
Derived from ftpfs.c.
|
||||
|
||||
@ -57,6 +59,7 @@
|
||||
#include "lib/tty/tty.h" /* enable/disable interrupt key */
|
||||
#include "lib/strescape.h"
|
||||
#include "lib/unixcompat.h"
|
||||
#include "lib/fileloc.h"
|
||||
|
||||
#include "src/wtools.h" /* message() */
|
||||
#include "src/main.h" /* print_vfs_message */
|
||||
@ -67,6 +70,7 @@
|
||||
#include "gc.h" /* vfs_stamp_create */
|
||||
#include "netutil.h"
|
||||
#include "fish.h"
|
||||
#include "fishdef.h"
|
||||
|
||||
int fish_directory_timeout = 900;
|
||||
|
||||
@ -75,7 +79,7 @@ int fish_directory_timeout = 900;
|
||||
#define DO_FREE_RESOURCE 4
|
||||
|
||||
#define FISH_FLAG_COMPRESSED 1
|
||||
#define FISH_FLAG_RSH 2
|
||||
#define FISH_FLAG_RSH 2
|
||||
|
||||
#define OPT_FLUSH 1
|
||||
#define OPT_IGNORE_ERROR 2
|
||||
@ -83,23 +87,58 @@ int fish_directory_timeout = 900;
|
||||
/*
|
||||
* 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 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 */
|
||||
|
||||
/* command wait_flag: */
|
||||
#define NONE 0x00
|
||||
#define WAIT_REPLY 0x01
|
||||
#define WANT_STRING 0x02
|
||||
|
||||
/* environment flags */
|
||||
#define FISH_HAVE_HEAD 1
|
||||
#define FISH_HAVE_SED 2
|
||||
#define FISH_HAVE_AWK 4
|
||||
#define FISH_HAVE_PERL 8
|
||||
#define FISH_HAVE_LSQ 16
|
||||
#define FISH_HAVE_DATE_MDYT 32
|
||||
|
||||
static char reply_str[80];
|
||||
|
||||
static struct vfs_class vfs_fish_ops;
|
||||
|
||||
static int
|
||||
fish_command (struct vfs_class *me, struct vfs_s_super *super,
|
||||
int wait_reply, const char *fmt, ...) __attribute__ ((format (__printf__, 4, 5)));
|
||||
|
||||
static char *
|
||||
fish_load_script_from_file (const char *hostname, const char *script_name, const char *def_content)
|
||||
{
|
||||
char *scr_filename = NULL;
|
||||
char *scr_content;
|
||||
gsize scr_len = 0;
|
||||
|
||||
/* 1st: scan user directory */
|
||||
scr_filename = g_build_path (PATH_SEP_STR, home_dir, MC_USERCONF_DIR, FISH_PREFIX, hostname,
|
||||
script_name, (char *) NULL);
|
||||
/* silent about user dir */
|
||||
g_file_get_contents (scr_filename, &scr_content, &scr_len, NULL);
|
||||
g_free (scr_filename);
|
||||
/* 2nd: scan system dir */
|
||||
if (scr_content == NULL)
|
||||
{
|
||||
g_free (scr_content);
|
||||
scr_filename = g_build_path (PATH_SEP_STR, LIBEXECDIR, FISH_PREFIX, script_name, (char *) NULL);
|
||||
g_file_get_contents (scr_filename, &scr_content, &scr_len, NULL);
|
||||
g_free (scr_filename);
|
||||
}
|
||||
|
||||
if (scr_content != NULL)
|
||||
return scr_content;
|
||||
|
||||
g_free (scr_content);
|
||||
return g_strdup (def_content);
|
||||
}
|
||||
|
||||
static int
|
||||
fish_decode_reply (char *s, int was_garbage)
|
||||
@ -195,6 +234,21 @@ fish_free_archive (struct vfs_class *me, struct vfs_s_super *super)
|
||||
g_free (SUP.user);
|
||||
g_free (SUP.cwdir);
|
||||
g_free (SUP.password);
|
||||
g_free (SUP.scr_ls);
|
||||
g_free (SUP.scr_exists);
|
||||
g_free (SUP.scr_mkdir);
|
||||
g_free (SUP.scr_unlink);
|
||||
g_free (SUP.scr_chown);
|
||||
g_free (SUP.scr_chmod);
|
||||
g_free (SUP.scr_rmdir);
|
||||
g_free (SUP.scr_ln);
|
||||
g_free (SUP.scr_mv);
|
||||
g_free (SUP.scr_hardlink);
|
||||
g_free (SUP.scr_get);
|
||||
g_free (SUP.scr_send);
|
||||
g_free (SUP.scr_append);
|
||||
g_free (SUP.scr_info);
|
||||
g_free (SUP.scr_env);
|
||||
}
|
||||
|
||||
static void
|
||||
@ -234,6 +288,57 @@ fish_pipeopen (struct vfs_s_super *super, const char *path, const char *argv[])
|
||||
}
|
||||
}
|
||||
|
||||
static char *
|
||||
fish_set_env (int flags)
|
||||
{
|
||||
GString *tmp;
|
||||
|
||||
tmp = g_string_sized_new (128);
|
||||
g_string_assign (tmp, "export ");
|
||||
|
||||
if ((flags & FISH_HAVE_HEAD) != 0)
|
||||
g_string_append (tmp, "FISH_HAVE_HEAD=1 ");
|
||||
|
||||
if ((flags & FISH_HAVE_SED) != 0)
|
||||
g_string_append (tmp, "FISH_HAVE_SED=1 ");
|
||||
|
||||
if ((flags & FISH_HAVE_AWK) != 0)
|
||||
g_string_append (tmp, "FISH_HAVE_AWK=1 ");
|
||||
|
||||
if ((flags & FISH_HAVE_PERL) != 0)
|
||||
g_string_append (tmp, "FISH_HAVE_PERL=1 ");
|
||||
|
||||
if ((flags & FISH_HAVE_LSQ) != 0)
|
||||
g_string_append (tmp, "FISH_HAVE_LSQ=1 ");
|
||||
|
||||
if ((flags & FISH_HAVE_DATE_MDYT) != 0)
|
||||
g_string_append (tmp, "FISH_HAVE_DATE_MDYT=1 ");
|
||||
|
||||
return g_string_free (tmp, FALSE);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
fish_info (struct vfs_class *me, struct vfs_s_super *super)
|
||||
{
|
||||
char buffer[8192];
|
||||
if (fish_command (me, super, NONE, SUP.scr_info) == COMPLETE)
|
||||
{
|
||||
while (1)
|
||||
{
|
||||
int res;
|
||||
res = vfs_s_get_line_interruptible (me, buffer, sizeof (buffer), SUP.sockr);
|
||||
if ((!res) || (res == EINTR))
|
||||
ERRNOR (ECONNRESET, FALSE);
|
||||
if (!strncmp (buffer, "### ", 4))
|
||||
break;
|
||||
SUP.host_flags = atol (buffer);
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
ERRNOR (E_PROTO, FALSE);
|
||||
}
|
||||
|
||||
|
||||
/* The returned directory should always contain a trailing slash */
|
||||
static char *
|
||||
fish_getcwd (struct vfs_class *me, struct vfs_s_super *super)
|
||||
@ -337,16 +442,20 @@ fish_open_archive_int (struct vfs_class *me, struct vfs_s_super *super)
|
||||
ERRNOR (E_PROTO, -1);
|
||||
|
||||
print_vfs_message (_("fish: Handshaking version..."));
|
||||
if (fish_command (me, super, WAIT_REPLY, "#VER 0.0.0\necho '### 000'\n") != COMPLETE)
|
||||
if (fish_command (me, super, WAIT_REPLY, "#VER 0.0.3\necho '### 000'\n") != COMPLETE)
|
||||
ERRNOR (E_PROTO, -1);
|
||||
|
||||
/* Set up remote locale to C, otherwise dates cannot be recognized */
|
||||
if (fish_command
|
||||
(me, super, WAIT_REPLY,
|
||||
"LANG=C; LC_ALL=C; LC_TIME=C\n"
|
||||
"export LANG; export LC_ALL; export LC_TIME\n" "echo '### 200'\n") != COMPLETE)
|
||||
"export LANG=C LC_ALL=C LC_TIME=C\n"
|
||||
"echo '### 200'\n") != COMPLETE)
|
||||
ERRNOR (E_PROTO, -1);
|
||||
|
||||
print_vfs_message (_("fish: Getting host info..."));
|
||||
if (fish_info (me, super))
|
||||
SUP.scr_env = fish_set_env (SUP.host_flags);
|
||||
|
||||
print_vfs_message (_("fish: Setting up current directory..."));
|
||||
SUP.cwdir = fish_getcwd (me, super);
|
||||
print_vfs_message (_("fish: Connected, home %s."), SUP.cwdir);
|
||||
@ -381,6 +490,20 @@ fish_open_archive (struct vfs_class *me, struct vfs_s_super *super,
|
||||
SUP.cwdir = NULL;
|
||||
if (password)
|
||||
SUP.password = password;
|
||||
SUP.scr_ls = fish_load_script_from_file (host, FISH_LS_FILE, FISH_LS_DEF_CONTENT);
|
||||
SUP.scr_exists = fish_load_script_from_file (host, FISH_EXISTS_FILE, FISH_EXISTS_FILE);
|
||||
SUP.scr_mkdir = fish_load_script_from_file (host, FISH_MKDIR_FILE, FISH_MKDIR_FILE);
|
||||
SUP.scr_unlink = fish_load_script_from_file (host, FISH_UNLINK_FILE, FISH_UNLINK_FILE);
|
||||
SUP.scr_chown = fish_load_script_from_file (host, FISH_CHOWN_FILE, FISH_CHOWN_FILE);
|
||||
SUP.scr_chmod = fish_load_script_from_file (host, FISH_CHMOD_FILE, FISH_CHMOD_FILE);
|
||||
SUP.scr_rmdir = fish_load_script_from_file (host, FISH_RMDIR_FILE, FISH_RMDIR_FILE);
|
||||
SUP.scr_ln = fish_load_script_from_file (host, FISH_LN_FILE, FISH_LN_FILE);
|
||||
SUP.scr_mv = fish_load_script_from_file (host, FISH_MV_FILE, FISH_MV_FILE);
|
||||
SUP.scr_hardlink = fish_load_script_from_file (host, FISH_HARDLINK_FILE, FISH_HARDLINK_FILE);
|
||||
SUP.scr_get = fish_load_script_from_file (host, FISH_GET_FILE, FISH_GET_FILE);
|
||||
SUP.scr_send = fish_load_script_from_file (host, FISH_SEND_FILE,FISH_SEND_FILE);
|
||||
SUP.scr_append = fish_load_script_from_file (host, FISH_APPEND_FILE, FISH_APPEND_FILE);
|
||||
SUP.scr_info = fish_load_script_from_file (host, FISH_INFO_FILE, FISH_INFO_FILE);
|
||||
return fish_open_archive_int (me, super);
|
||||
}
|
||||
|
||||
@ -424,16 +547,15 @@ fish_dir_load (struct vfs_class *me, struct vfs_s_inode *dir, char *remote_path)
|
||||
int reply_code;
|
||||
gchar *shell_commands;
|
||||
|
||||
#if 0
|
||||
/*
|
||||
* Simple FISH debug interface :]
|
||||
*/
|
||||
#if 0
|
||||
if (!(MEDATA->logfile))
|
||||
{
|
||||
MEDATA->logfile = fopen ("/tmp/mc-FISH.sh", "w");
|
||||
}
|
||||
#endif /* 0 */
|
||||
|
||||
#endif
|
||||
logfile = MEDATA->logfile;
|
||||
|
||||
print_vfs_message (_("fish: Reading directory %s..."), remote_path);
|
||||
@ -441,100 +563,8 @@ fish_dir_load (struct vfs_class *me, struct vfs_s_inode *dir, char *remote_path)
|
||||
gettimeofday (&dir->timestamp, NULL);
|
||||
dir->timestamp.tv_sec += fish_directory_timeout;
|
||||
quoted_path = strutils_shell_escape (remote_path);
|
||||
/* *INDENT-OFF* */
|
||||
shell_commands = g_strconcat (
|
||||
"#LIST /%s\n"
|
||||
"if `perl -v > /dev/null 2>&1` ; then\n"
|
||||
"perl -e '\n"
|
||||
"use strict;\n"
|
||||
"use POSIX;\n"
|
||||
"use Fcntl;\n"
|
||||
"use POSIX \":fcntl_h\"; #S_ISLNK was here until 5.6\n"
|
||||
"import Fcntl \":mode\" unless defined &S_ISLNK; #and is now here\n"
|
||||
"my $dirname = $ARGV[0];\n"
|
||||
"if (opendir ( DIR, $dirname )) {\n"
|
||||
"while( (my $filename = readdir(DIR))){\n"
|
||||
"my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size,$atime,$mtime,$ctime,$blksize,$blocks) = lstat(\"$dirname/$filename\");\n"
|
||||
"my $mloctime= strftime(\"%%m-%%d-%%Y %%H:%%M\", localtime $mtime);\n"
|
||||
,
|
||||
"my $strutils_shell_escape_regex = s/([;<>\\*\\|`&\\$!#\\(\\)\\[\\]\\{\\}:'\\''\"\\ \\\\])/\\\\$1/g;\n"
|
||||
"my $e_filename = $filename;\n"
|
||||
"$e_filename =~ $strutils_shell_escape_regex;\n"
|
||||
"if (S_ISLNK($mode) ) {\n"
|
||||
"my $linkname = readlink (\"$dirname/$filename\");\n"
|
||||
"$linkname =~ $strutils_shell_escape_regex;\n"
|
||||
"\n"
|
||||
"printf(\"R%%o %%o $uid.$gid\\n" "S$size\\n"
|
||||
"d$mloctime\\n"
|
||||
":\\\"$e_filename\\\" -> \\\"$linkname\\\"\\n"
|
||||
"\\n\", S_IMODE($mode), S_IFMT($mode));\n"
|
||||
"} else {\n"
|
||||
"printf(\"R%%o %%o $uid.$gid\\n"
|
||||
"S$size\\n"
|
||||
"d$mloctime\\n"
|
||||
":\\\"$e_filename\\\"\\n"
|
||||
"\\n\", S_IMODE($mode), S_IFMT($mode));\n"
|
||||
"}}\n"
|
||||
,
|
||||
"printf(\"### 200\\n\");\n"
|
||||
"closedir(DIR);\n"
|
||||
"} else {\n"
|
||||
"printf(\"### 500\\n\");\n"
|
||||
"}\n"
|
||||
"exit 0\n"
|
||||
"' /%s ||\n" /* ARGV[0] - path to browse */
|
||||
" echo '### 500'\n" /* do not hang if perl is to eval it */
|
||||
"elif `ls -1 /%s >/dev/null 2>&1` ; then\n"
|
||||
"if `ls -Q /%s >/dev/null 2>&1`; then\n"
|
||||
"LSOPT=\"-Qlan\";\n"
|
||||
"ADD=0;\n"
|
||||
"else\n"
|
||||
"LSOPT=\"-lan\";\n"
|
||||
"ADD=1;\n"
|
||||
"fi\n"
|
||||
"ls $LSOPT /%s 2>/dev/null | grep '^[^cbt]' | (\n"
|
||||
"while read p l u g s m d y n n2 n3; do\n"
|
||||
"if test \"$m\" = \"0\" ; then \n"
|
||||
"s=$d; m=$y; d=$n y=$n2; n=$n3\n"
|
||||
"else\n"
|
||||
"n=$n\" \"$n2\" \"$n3\n"
|
||||
"fi\n"
|
||||
"if [ $ADD = 0 ]; then\n"
|
||||
"echo \"P$p $u.$g\nS$s\nd$m $d $y\n:$n\n\"\n"
|
||||
"elif `sed --version >/dev/null 2>&1` ; then\n"
|
||||
"file=`echo $n | sed -e 's#^\\(.*\\) -> \\(.*\\)#\\1\" -> \"\\2#'`\n",
|
||||
"echo \"P$p $u $g\nS$s\nd$m $d $y\n:\"$file\"\n\"\n"
|
||||
"else\n"
|
||||
"echo \"P$p $u $g\nS$s\nd$m $d $y\n:\"$n\"\n\"\n"
|
||||
"fi\n"
|
||||
"done )\n"
|
||||
"ls $LSOPT /%s 2>/dev/null | grep '^[cb]' | (\n"
|
||||
"while read p l u g a i m d y n n2 n3; do\n"
|
||||
"if test \"$a\" = \"0\" ; then \n"
|
||||
"a=$m; i=$d; m=$y; d=$n y=$n2; n=$n3\n"
|
||||
"else\n"
|
||||
"n=$n\" \"$n2\" \"$n3\n"
|
||||
"fi\n"
|
||||
"if [ $ADD = 0 ]; then\n"
|
||||
"echo \"P$p $u.$g\nE$a$i\nd$m $d $y\n:$n\n\"\n"
|
||||
"elif `sed --version >/dev/null 2>&1` ; then\n"
|
||||
"file=`echo $n | sed -e 's#^\\(.*\\) -> \\(.*\\)#\\1\" -> \"\\2#'`\n"
|
||||
"echo \"P$p $u $g\nS$s\nd$m $d $y\n:\"$file\"\n\"\n"
|
||||
"else\n"
|
||||
"echo \"P$p $u $g\nS$s\nd$m $d $y\n:\"$n\"\n\"\n"
|
||||
"fi\n"
|
||||
"done)\n"
|
||||
"echo '### 200'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n"
|
||||
,
|
||||
(char *) NULL);
|
||||
/* *INDENT-ON* */
|
||||
|
||||
fish_command (me, super, NONE, shell_commands,
|
||||
quoted_path, quoted_path, quoted_path, quoted_path, quoted_path, quoted_path);
|
||||
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s;\n", SUP.scr_ls, (char *) NULL);
|
||||
fish_command (me, super, NONE, shell_commands, quoted_path);
|
||||
g_free (shell_commands);
|
||||
g_free (quoted_path);
|
||||
ent = vfs_s_generate_entry (me, NULL, dir, 0);
|
||||
@ -714,6 +744,7 @@ fish_dir_load (struct vfs_class *me, struct vfs_s_inode *dir, char *remote_path)
|
||||
static int
|
||||
fish_file_store (struct vfs_class *me, struct vfs_s_fh *fh, char *name, char *localname)
|
||||
{
|
||||
gchar *shell_commands = NULL;
|
||||
struct vfs_s_super *super = FH_SUPER;
|
||||
int n, total;
|
||||
char buffer[8192];
|
||||
@ -767,50 +798,17 @@ fish_file_store (struct vfs_class *me, struct vfs_s_fh *fh, char *name, char *lo
|
||||
/* FIXME: File size is limited to ULONG_MAX */
|
||||
if (!fh->u.fish.append)
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
n = fish_command (me, super, WAIT_REPLY,
|
||||
"#STOR %lu /%s\n"
|
||||
"echo '### 001'\n"
|
||||
"file=/%s\n"
|
||||
"res=`exec 3>&1\n"
|
||||
"(\n"
|
||||
"head -c %lu -q - || echo DD >&3\n"
|
||||
") 2>/dev/null | (\n"
|
||||
"cat > $file\n"
|
||||
"cat > /dev/null\n"
|
||||
")`; [ \"$res\" = DD ] && {\n"
|
||||
"> \"$file\"\n"
|
||||
"rest=%lu\n"
|
||||
"while [ $rest -gt 0 ]\n"
|
||||
"do\n"
|
||||
" cnt=`expr \\( $rest + 255 \\) / 256`\n"
|
||||
" n=`dd bs=256 count=$cnt | tee -a \"$file\" | wc -c`\n"
|
||||
" rest=`expr $rest - $n`\n"
|
||||
"done\n"
|
||||
"}; echo '### 200'\n",
|
||||
(unsigned long) s.st_size, quoted_name,
|
||||
quoted_name, (unsigned long) s.st_size, (unsigned long) s.st_size);
|
||||
/* *INDENT-ON* */
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s FISH_FILESIZE=%lu;\n",
|
||||
SUP.scr_append, (char *) NULL);
|
||||
n = fish_command (me, super, WAIT_REPLY, shell_commands, quoted_name, (unsigned long) s.st_size);
|
||||
g_free (shell_commands);
|
||||
}
|
||||
else
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
n = fish_command (me, super, WAIT_REPLY,
|
||||
"#STOR %lu /%s\n"
|
||||
"echo '### 001'\n"
|
||||
"{\n"
|
||||
"file=/%s\n"
|
||||
"rest=%lu\n"
|
||||
"while [ $rest -gt 0 ]\n"
|
||||
"do\n"
|
||||
" cnt=`expr \\( $rest + 255 \\) / 256`\n"
|
||||
" n=`dd bs=256 count=$cnt | tee -a $file | wc -c`\n"
|
||||
" rest=`expr $rest - $n`\n"
|
||||
"done\n"
|
||||
"}; echo '### 200'\n",
|
||||
(unsigned long) s.st_size, quoted_name,
|
||||
quoted_name, (unsigned long) s.st_size);
|
||||
/* *INDENT-ON* */
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s FISH_FILESIZE=%lu;\n",
|
||||
SUP.scr_send, (char *) NULL);
|
||||
n = fish_command (me, super, WAIT_REPLY, shell_commands, quoted_name, (unsigned long) s.st_size);
|
||||
g_free (shell_commands);
|
||||
}
|
||||
if (n != PRELIM)
|
||||
{
|
||||
@ -864,6 +862,8 @@ fish_file_store (struct vfs_class *me, struct vfs_s_fh *fh, char *name, char *lo
|
||||
static int
|
||||
fish_linear_start (struct vfs_class *me, struct vfs_s_fh *fh, off_t offset)
|
||||
{
|
||||
gchar *shell_commands = NULL;
|
||||
struct vfs_s_super *super = FH_SUPER;
|
||||
char *name;
|
||||
char *quoted_name;
|
||||
if (offset)
|
||||
@ -881,23 +881,10 @@ fish_linear_start (struct vfs_class *me, struct vfs_s_fh *fh, off_t offset)
|
||||
* with exit status of 0 use `cat' to send the file contents to the
|
||||
* standard output (i.e. over the network).
|
||||
*/
|
||||
/* *INDENT-OFF* */
|
||||
offset = fish_command (me, FH_SUPER, WANT_STRING,
|
||||
"#RETR /%s\n"
|
||||
"if dd if=/%s of=/dev/null bs=1 count=1 2>/dev/null ;\n"
|
||||
"then\n"
|
||||
"ls -ln /%s 2>/dev/null | (\n"
|
||||
"read p l u g s r\n"
|
||||
"echo $s\n"
|
||||
")\n"
|
||||
"echo '### 100'\n"
|
||||
"cat /%s\n"
|
||||
"echo '### 200'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
quoted_name, quoted_name, quoted_name, quoted_name);
|
||||
/* *INDENT-ON* */
|
||||
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s;\n", SUP.scr_get, (char *) NULL);
|
||||
offset = fish_command (me, super, WANT_STRING, shell_commands, quoted_name);
|
||||
g_free (shell_commands);
|
||||
g_free (quoted_name);
|
||||
if (offset != PRELIM)
|
||||
ERRNOR (E_REMOTE, 0);
|
||||
@ -1030,92 +1017,114 @@ fish_send_command (struct vfs_class *me, struct vfs_s_super *super, const char *
|
||||
rpath = strutils_shell_escape (crpath); \
|
||||
g_free (mpath);
|
||||
|
||||
#define POSTFIX(flags) \
|
||||
g_free (rpath); \
|
||||
return fish_send_command (me, super, buf, flags);
|
||||
static int
|
||||
fish_rename (struct vfs_class *me, const char *path1, const char *path2)
|
||||
{
|
||||
gchar *shell_commands = NULL;
|
||||
char buf[BUF_LARGE];
|
||||
const char *crpath1, *crpath2;
|
||||
char *rpath1, *rpath2, *mpath1, *mpath2;
|
||||
struct vfs_s_super *super, *super2;
|
||||
crpath1 = vfs_s_get_path_mangle (me, mpath1 = g_strdup(path1), &super, 0);
|
||||
if (crpath1 == NULL)
|
||||
{
|
||||
g_free (mpath1);
|
||||
return -1;
|
||||
}
|
||||
crpath2 = vfs_s_get_path_mangle (me, mpath2 = g_strdup(path2), &super2, 0);
|
||||
if (crpath2 == NULL)
|
||||
{
|
||||
g_free (mpath1);
|
||||
g_free (mpath2);
|
||||
return -1;
|
||||
}
|
||||
rpath1 = strutils_shell_escape (crpath1);
|
||||
g_free (mpath1);
|
||||
rpath2 = strutils_shell_escape (crpath2);
|
||||
g_free (mpath2);
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILEFROM=%s FISH_FILETO=%s;\n",
|
||||
SUP.scr_mv, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath1, rpath2);
|
||||
g_free (shell_commands);
|
||||
g_free (rpath1);
|
||||
g_free (rpath2);
|
||||
return fish_send_command(me, super2, buf, OPT_FLUSH);
|
||||
}
|
||||
|
||||
static int
|
||||
fish_chmod (struct vfs_class *me, const char *path, int mode)
|
||||
fish_link (struct vfs_class *me, const char *path1, const char *path2)
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
PREFIX
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#CHMOD %4.4o /%s\n"
|
||||
"if chmod %4.4o /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
mode & 07777, rpath, mode & 07777, rpath);
|
||||
POSTFIX (OPT_FLUSH);
|
||||
/* *INDENT-ON* */
|
||||
gchar *shell_commands = NULL;
|
||||
char buf[BUF_LARGE];
|
||||
const char *crpath1, *crpath2;
|
||||
char *rpath1, *rpath2, *mpath1, *mpath2;
|
||||
struct vfs_s_super *super, *super2;
|
||||
crpath1 = vfs_s_get_path_mangle (me, mpath1 = g_strdup(path1), &super, 0);
|
||||
if (crpath1 == NULL)
|
||||
{
|
||||
g_free (mpath1);
|
||||
return -1;
|
||||
}
|
||||
crpath2 = vfs_s_get_path_mangle (me, mpath2 = g_strdup(path2), &super2, 0);
|
||||
if (crpath2 == NULL)
|
||||
{
|
||||
g_free (mpath1);
|
||||
g_free (mpath2);
|
||||
return -1;
|
||||
}
|
||||
rpath1 = strutils_shell_escape (crpath1);
|
||||
g_free (mpath1);
|
||||
rpath2 = strutils_shell_escape (crpath2);
|
||||
g_free (mpath2);
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILEFROM=%s FISH_FILETO=%s;\n",
|
||||
SUP.scr_hardlink, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath1, rpath2);
|
||||
g_free (shell_commands);
|
||||
g_free (rpath1);
|
||||
g_free (rpath2);
|
||||
return fish_send_command(me, super2, buf, OPT_FLUSH);
|
||||
}
|
||||
|
||||
/* *INDENT-OFF* */
|
||||
#define FISH_OP(name, string) \
|
||||
static int fish_##name (struct vfs_class *me, const char *path1, const char *path2) \
|
||||
{ \
|
||||
char buf[BUF_LARGE]; \
|
||||
const char *crpath1, *crpath2; \
|
||||
char *rpath1, *rpath2, *mpath1, *mpath2; \
|
||||
struct vfs_s_super *super1, *super2; \
|
||||
crpath1 = vfs_s_get_path_mangle (me, mpath1 = g_strdup(path1), &super1, 0); \
|
||||
if (crpath1 == NULL) \
|
||||
{ \
|
||||
g_free (mpath1); \
|
||||
return -1; \
|
||||
} \
|
||||
crpath2 = vfs_s_get_path_mangle (me, mpath2 = g_strdup(path2), &super2, 0); \
|
||||
if (crpath2 == NULL) \
|
||||
{ \
|
||||
g_free (mpath1); \
|
||||
g_free (mpath2); \
|
||||
return -1; \
|
||||
} \
|
||||
rpath1 = strutils_shell_escape (crpath1); \
|
||||
g_free (mpath1); \
|
||||
rpath2 = strutils_shell_escape (crpath2); \
|
||||
g_free (mpath2); \
|
||||
g_snprintf (buf, sizeof(buf), string "\n", rpath1, rpath2, rpath1, rpath2); \
|
||||
g_free (rpath1); \
|
||||
g_free (rpath2); \
|
||||
return fish_send_command(me, super2, buf, OPT_FLUSH); \
|
||||
}
|
||||
|
||||
FISH_OP (rename,
|
||||
"#RENAME /%s /%s\n"
|
||||
"if mv /%s /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n")
|
||||
|
||||
FISH_OP (link,
|
||||
"#LINK /%s /%s\n"
|
||||
"if ln /%s /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n")
|
||||
/* *INDENT-ON* */
|
||||
|
||||
static int
|
||||
fish_symlink (struct vfs_class *me, const char *setto, const char *path)
|
||||
{
|
||||
char *qsetto;
|
||||
PREFIX qsetto = strutils_shell_escape (setto);
|
||||
/* *INDENT-OFF* */
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#SYMLINK %s /%s\n"
|
||||
"if ln -s %s /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
qsetto, rpath, qsetto, rpath);
|
||||
/* *INDENT-ON* */
|
||||
gchar *shell_commands = NULL;
|
||||
char buf[BUF_LARGE];
|
||||
const char *crpath;
|
||||
char *rpath, *mpath = g_strdup (path);
|
||||
struct vfs_s_super *super;
|
||||
crpath = vfs_s_get_path_mangle (me, mpath, &super, 0);
|
||||
if (crpath == NULL)
|
||||
{
|
||||
g_free (mpath);
|
||||
return -1;
|
||||
}
|
||||
rpath = strutils_shell_escape (crpath);
|
||||
g_free (mpath);
|
||||
|
||||
qsetto = strutils_shell_escape (setto);
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILEFROM=%s FISH_FILETO=%s;\n",
|
||||
SUP.scr_ln, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, qsetto, rpath);
|
||||
g_free (shell_commands);
|
||||
g_free (qsetto);
|
||||
POSTFIX (OPT_FLUSH);
|
||||
g_free (rpath);
|
||||
return fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
}
|
||||
|
||||
static int
|
||||
fish_chmod (struct vfs_class *me, const char *path, int mode)
|
||||
{
|
||||
gchar *shell_commands = NULL;
|
||||
PREFIX
|
||||
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s FISH_FILEMODE=%4.4o;\n",
|
||||
SUP.scr_chmod, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath, mode & 07777);
|
||||
g_free (shell_commands);
|
||||
g_free (rpath);
|
||||
return fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
}
|
||||
|
||||
static int
|
||||
@ -1136,55 +1145,44 @@ fish_chown (struct vfs_class *me, const char *path, int owner, int group)
|
||||
sowner = pw->pw_name;
|
||||
sgroup = gr->gr_name;
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
gchar *shell_commands = NULL;
|
||||
|
||||
PREFIX
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#CHOWN %s:%s /%s\n"
|
||||
"if chown %s:%s /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
sowner, sgroup, rpath, sowner, sgroup, rpath);
|
||||
/* *INDENT-ON* */
|
||||
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s FISH_FILEOWNER=%s FISH_FILEGROUP=%s;\n",
|
||||
SUP.scr_chown, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath, sowner, sgroup);
|
||||
g_free (shell_commands);
|
||||
fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
/* FIXME: what should we report if chgrp succeeds but chown fails? */
|
||||
/* fish_send_command(me, super, buf, OPT_FLUSH); */
|
||||
POSTFIX (OPT_FLUSH)}
|
||||
g_free (rpath);
|
||||
return fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
fish_unlink (struct vfs_class *me, const char *path)
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
gchar *shell_commands = NULL;
|
||||
PREFIX
|
||||
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#DELE /%s\n"
|
||||
"if rm -f /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
rpath, rpath);
|
||||
/* *INDENT-ON* */
|
||||
|
||||
POSTFIX (OPT_FLUSH);
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s;\n", SUP.scr_unlink, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath);
|
||||
g_free (shell_commands);
|
||||
g_free (rpath);
|
||||
return fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
}
|
||||
|
||||
static int
|
||||
fish_exists (struct vfs_class *me, const char *path)
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
gchar *shell_commands = NULL;
|
||||
PREFIX
|
||||
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#ISEXISTS /%s\n"
|
||||
"ls -l /%s >/dev/null 2>/dev/null\n"
|
||||
"echo '### '$?\n",
|
||||
rpath, rpath);
|
||||
/* *INDENT-ON* */
|
||||
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s;\n", SUP.scr_exists, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath);
|
||||
g_free (shell_commands);
|
||||
g_free (rpath);
|
||||
|
||||
return (fish_send_command (me, super, buf, OPT_FLUSH) == 0) ? 1 : 0;
|
||||
@ -1194,20 +1192,14 @@ fish_exists (struct vfs_class *me, const char *path)
|
||||
static int
|
||||
fish_mkdir (struct vfs_class *me, const char *path, mode_t mode)
|
||||
{
|
||||
gchar *shell_commands = NULL;
|
||||
int ret_code;
|
||||
|
||||
/* *INDENT-OFF* */
|
||||
PREFIX (void) mode;
|
||||
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#MKD /%s\n"
|
||||
"if mkdir /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
rpath, rpath);
|
||||
/* *INDENT-ON* */
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s;\n", SUP.scr_mkdir, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath);
|
||||
g_free (shell_commands);
|
||||
|
||||
g_free (rpath);
|
||||
ret_code = fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
@ -1225,18 +1217,14 @@ fish_mkdir (struct vfs_class *me, const char *path, mode_t mode)
|
||||
static int
|
||||
fish_rmdir (struct vfs_class *me, const char *path)
|
||||
{
|
||||
/* *INDENT-OFF* */
|
||||
gchar *shell_commands = NULL;
|
||||
PREFIX
|
||||
g_snprintf (buf, sizeof (buf),
|
||||
"#RMD /%s\n"
|
||||
"if rmdir /%s 2>/dev/null; then\n"
|
||||
"echo '### 000'\n"
|
||||
"else\n"
|
||||
"echo '### 500'\n"
|
||||
"fi\n",
|
||||
rpath, rpath);
|
||||
/* *INDENT-ON* */
|
||||
POSTFIX (OPT_FLUSH);
|
||||
|
||||
shell_commands = g_strconcat (SUP.scr_env, "FISH_FILENAME=%s;\n", SUP.scr_rmdir, (char *) NULL);
|
||||
g_snprintf (buf, sizeof (buf), shell_commands, rpath);
|
||||
g_free (shell_commands);
|
||||
g_free (rpath);
|
||||
return fish_send_command (me, super, buf, OPT_FLUSH);
|
||||
}
|
||||
|
||||
static int
|
||||
|
10
lib/vfs/mc-vfs/fish/Makefile.am
Обычный файл
10
lib/vfs/mc-vfs/fish/Makefile.am
Обычный файл
@ -0,0 +1,10 @@
|
||||
fishdir = $(libexecdir)/@PACKAGE@/fish
|
||||
|
||||
# Files to install and distribute other than fish scripts
|
||||
FISH_MISC = README.fish
|
||||
|
||||
fish_DATA = $(FISH_MISC)
|
||||
fish_SCRIPTS = ls mkdir fexists unlink chown chmod rmdir ln mv hardlink get send append info
|
||||
fishconfdir = $(sysconfdir)/@PACKAGE@
|
||||
|
||||
EXTRA_DIST = $(FISH_MISC)
|
@ -1,5 +1,5 @@
|
||||
|
||||
FIles transferred over SHell protocol (V 0.0.2)
|
||||
FIles transferred over SHell protocol (V 0.0.3)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
This protocol was designed for transferring files over a remote shell
|
||||
@ -18,6 +18,19 @@ implementation. Fish commands always have priority: server is
|
||||
expected to execute fish command if it understands it. If it does not,
|
||||
however, it can try the luck and execute shell command.
|
||||
|
||||
Since version 4.7.3, the scripts that FISH sends to host machines after
|
||||
a command is transmitted are no longer hardwired in the Midnight
|
||||
Commander source code.
|
||||
|
||||
First, mc looks for system-wide set of scripts, then it checks whether
|
||||
current user has host-specific overrides in his per-user mc
|
||||
configuration folder. User-defined overrides take priority over
|
||||
sytem-wide scripts if they exist. The order in which the folders are
|
||||
traversed is as follows:
|
||||
|
||||
/usr/libexec/mc/fish
|
||||
~/.mc/fish/<hostname>/
|
||||
|
||||
Server's reply is multiline, but always ends with
|
||||
|
||||
### 000<optional text>
|
||||
@ -158,6 +171,11 @@ chown user /file/name; echo '### 000'
|
||||
#CHGRP group /file/name
|
||||
chgrp group /file/name; echo '### 000'
|
||||
|
||||
#INFO
|
||||
...collect info about host into $result ...
|
||||
echo $result
|
||||
echo '### 200'
|
||||
|
||||
#READ <offset> <size> /path/and/filename
|
||||
cat /path/and/filename | ( dd bs=4096 count=<offset/4096> > /dev/null;
|
||||
dd bs=<offset%4096> count=1 > /dev/null;
|
||||
@ -173,6 +191,27 @@ end at eof.
|
||||
Hmm, shall we define these ones if we know our client is not going to
|
||||
use them?
|
||||
|
||||
you can use follow parameters:
|
||||
FISH_FILESIZE
|
||||
FISH_FILENAME
|
||||
FISH_FILEMODE
|
||||
FISH_FILEOWNER
|
||||
FISH_FILEGROUPE
|
||||
FISH_FILEFROM
|
||||
FISH_FILETO
|
||||
|
||||
NB:
|
||||
'FISH_FILESIZE' used if we operate with single file name in 'unlink', 'rmdir', 'chmod', etc...
|
||||
'FISH_FILEFROM','FISH_FILETO' used if we operate with two files in 'ln', 'hardlink', 'mv' etc...
|
||||
'FISH_FILEOWNER', 'FISH_FILEGROUPE' is a new user/group in chown
|
||||
|
||||
also flags:
|
||||
FISH_HAVE_HEAD
|
||||
FISH_HAVE_SED
|
||||
FISH_HAVE_AWK
|
||||
FISH_HAVE_PERL
|
||||
FISH_HAVE_LSQ
|
||||
FISH_HAVE_DATE_MDYT
|
||||
|
||||
That's all, folks!
|
||||
pavel@ucw.cz
|
18
lib/vfs/mc-vfs/fish/append
Исполняемый файл
18
lib/vfs/mc-vfs/fish/append
Исполняемый файл
@ -0,0 +1,18 @@
|
||||
#APPE $FISH_FILESIZE $FISH_FILENAME
|
||||
echo "### 001"
|
||||
FILENAME="/$FISH_FILENAME"
|
||||
res=`exec 3>&1
|
||||
(
|
||||
head -c $FISH_FILESIZE -q - || echo DD >&3
|
||||
) 2>/dev/null | (
|
||||
cat > "${FILENAME}"
|
||||
cat > /dev/null
|
||||
)`; [ "$res" = DD ] && {
|
||||
> "${FILENAME}"
|
||||
while [ $FISH_FILESIZE -gt 0 ]
|
||||
do
|
||||
cnt=`expr \\( $FISH_FILESIZE + 255 \\) / 256`
|
||||
n=`dd bs=256 count=$cnt | tee -a "${FILENAME}" | wc -c`
|
||||
FISH_FILESIZE=`expr $FISH_FILESIZE - $n`
|
||||
done
|
||||
}; echo "### 200"
|
6
lib/vfs/mc-vfs/fish/chmod
Исполняемый файл
6
lib/vfs/mc-vfs/fish/chmod
Исполняемый файл
@ -0,0 +1,6 @@
|
||||
#CHMOD $FISH_FILEMODE $FISH_FILENAME
|
||||
if chmod ${FISH_FILEMODE} "/${FISH_FILENAME}" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
6
lib/vfs/mc-vfs/fish/chown
Исполняемый файл
6
lib/vfs/mc-vfs/fish/chown
Исполняемый файл
@ -0,0 +1,6 @@
|
||||
#CHOWN $FISH_FILEOWNER:$FISH_FILEGROUP $FISH_FILENAME
|
||||
if chown ${FISH_FILEOWNER}:${FISH_FILEGROUP} "/${FISH_FILENAME}" ; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
3
lib/vfs/mc-vfs/fish/fexists
Исполняемый файл
3
lib/vfs/mc-vfs/fish/fexists
Исполняемый файл
@ -0,0 +1,3 @@
|
||||
#ISEXISTS $FISH_FILENAME
|
||||
ls -l "/${FISH_FILENAME}" >/dev/null 2>/dev/null
|
||||
echo '### '$?
|
14
lib/vfs/mc-vfs/fish/get
Исполняемый файл
14
lib/vfs/mc-vfs/fish/get
Исполняемый файл
@ -0,0 +1,14 @@
|
||||
#RETR $FISH_FILENAME
|
||||
FILENAME="/${FISH_FILENAME}"
|
||||
export LC_TIME=C
|
||||
if dd if="${FILENAME}" of=/dev/null bs=1 count=1 2>/dev/null ; then
|
||||
ls -ln "${FILENAME}" 2>/dev/null | (
|
||||
read p l u g s r
|
||||
echo $s
|
||||
)
|
||||
echo "### 100"
|
||||
cat "${FILENAME}"
|
||||
echo "### 200"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
8
lib/vfs/mc-vfs/fish/hardlink
Исполняемый файл
8
lib/vfs/mc-vfs/fish/hardlink
Исполняемый файл
@ -0,0 +1,8 @@
|
||||
#LINK $FISH_FILEFROM $FISH_FILETO
|
||||
FILEFROM="/${FISH_FILEFROM}"
|
||||
FILETO="/${FISH_FILETO}"
|
||||
if ln "${FILEFROM}" "${FILETO}" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
38
lib/vfs/mc-vfs/fish/info
Исполняемый файл
38
lib/vfs/mc-vfs/fish/info
Исполняемый файл
@ -0,0 +1,38 @@
|
||||
export LC_TIME=C
|
||||
#FISH_HAVE_HEAD 1
|
||||
#FISH_HAVE_SED 2
|
||||
#FISH_HAVE_AWK 4
|
||||
#FISH_HAVE_PERL 8
|
||||
#FISH_HAVE_LSQ 16
|
||||
#FISH_HAVE_DATE_MDYT 32
|
||||
res=0
|
||||
if `echo yes| head -c 1 > /dev/null 2>&1` ; then
|
||||
res=`expr $res + 1`
|
||||
fi
|
||||
if `sed --version >/dev/null 2>&1` ; then
|
||||
res=`expr $res + 2`
|
||||
fi
|
||||
if `awk --version > /dev/null 2>&1` ; then
|
||||
res=`expr $res + 4`
|
||||
fi
|
||||
if `perl -v > /dev/null 2>&1` ; then
|
||||
res=`expr $res + 8`
|
||||
fi
|
||||
if `ls -Q / >/dev/null 2>&1` ; then
|
||||
res=`expr $res + 16`
|
||||
fi
|
||||
dat=`ls -lan / 2>/dev/null | head -n 3|tail -n 1 | (
|
||||
while read p l u g s rec; do
|
||||
if [ -n "$g" ]; then
|
||||
if [ -n "$l" ]; then
|
||||
echo "$rec"
|
||||
fi
|
||||
fi
|
||||
done
|
||||
) |cut -c1 2>/dev/null`
|
||||
r=`echo "0123456789"| grep "$dat"`
|
||||
if [ -z "$r" ]; then
|
||||
res=`expr $res + 32`
|
||||
fi
|
||||
echo $res
|
||||
echo "### 200"
|
8
lib/vfs/mc-vfs/fish/ln
Исполняемый файл
8
lib/vfs/mc-vfs/fish/ln
Исполняемый файл
@ -0,0 +1,8 @@
|
||||
#SYMLINK $FISH_FILEFROM $FISH_FILETO
|
||||
FILEFROM="${FISH_FILEFROM}"
|
||||
FILETO="/${FISH_FILETO}"
|
||||
if ln -s "${FILEFROM}" "${FILETO}" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
160
lib/vfs/mc-vfs/fish/ls
Исполняемый файл
160
lib/vfs/mc-vfs/fish/ls
Исполняемый файл
@ -0,0 +1,160 @@
|
||||
#LIST /${FISH_DIR}
|
||||
export LC_TIME=C
|
||||
fish_list_lsq ()
|
||||
{
|
||||
FISH_DIR="$1"
|
||||
ls -Qlan "${FISH_DIR}" 2>/dev/null | grep '^[^cbt]' | (
|
||||
while read p l u g s m d y n; do
|
||||
echo "P$p $u.$g"
|
||||
echo "S$s"
|
||||
echo "d$m $d $y"
|
||||
echo ":$n"
|
||||
echo
|
||||
done
|
||||
)
|
||||
|
||||
ls -Qlan "${FISH_DIR}" 2>/dev/null | grep '^[cb]' | (
|
||||
while read p l u g a i m d y n; do
|
||||
echo "P$p $u.$g"
|
||||
echo "E$a$i"
|
||||
echo "d$m $d $y"
|
||||
echo ":$n"
|
||||
echo
|
||||
done
|
||||
)
|
||||
echo '### 200'
|
||||
}
|
||||
|
||||
fish_list_sed ()
|
||||
{
|
||||
FISH_DIR="$1"
|
||||
ls -lan "${FISH_DIR}" 2>/dev/null | grep '^[^cbt]' | (
|
||||
while read p l u g s rec; do
|
||||
if [ -n "$g" ]; then
|
||||
if [ -n "$FISH_HAVE_DATE_MDYT" ]; then
|
||||
filename=`echo "$rec"| sed 's/[^[:space:]]\+ \+[^[:space:]]\+ \+[^[:space:]]\+ //'`
|
||||
filedate=`echo "$rec"| sed 's/\([^[:space:]]\+ \+[^[:space:]]\+ \+[^[:space:]]\+\) .*/\1/'`
|
||||
else
|
||||
filename=`echo "$rec"| sed 's/[^[:space:]]\+ \+[^[:space:]]\+ //'`
|
||||
filedate=`echo "$rec"| sed 's/\([^[:space:]]\+ \+[^[:space:]]\+\) .*/\1/'`
|
||||
fi
|
||||
pfile=\"`echo "$filename" | sed -e 's#^\(.*\) -> \(.*\)#\1" -> "\2#'`\"
|
||||
echo "P$p $u.$g"
|
||||
echo "S$s"
|
||||
if [ -n "$FISH_HAVE_DATE_MDYT" ]; then
|
||||
echo "d$filedate"
|
||||
else
|
||||
echo "D$filedate"
|
||||
fi
|
||||
echo ":$pfile"
|
||||
echo
|
||||
fi
|
||||
done
|
||||
)
|
||||
ls -lan "${FISH_DIR}" 2>/dev/null | grep '^[cb]' | (
|
||||
while read p l u g a i rec; do
|
||||
if [ -n "$g" ]; then
|
||||
if [ -n "$FISH_HAVE_DATE_MDYT" ]; then
|
||||
filename=`echo "$rec"| sed 's/[^[:space:]]\+ \+[^[:space:]]\+ \+[^[:space:]]\+ //'`
|
||||
filedate=`echo "$rec"| sed 's/\([^[:space:]]\+ \+[^[:space:]]\+ \+[^[:space:]]\+\) .*/\1/'`
|
||||
else
|
||||
filename=`echo "$rec"| sed 's/[^[:space:]]\+ \+[^[:space:]]\+ //'`
|
||||
filedate=`echo "$rec"| sed 's/\([^[:space:]]\+ \+[^[:space:]]\+\) .*/\1/'`
|
||||
fi
|
||||
pfile=\"`echo "$filename" | sed -e 's#^\(.*\) -> \(.*\)#\1" -> "\2#'`\"
|
||||
echo "P$p $u.$g"
|
||||
echo "E$a$i"
|
||||
if [ -n "$FISH_HAVE_DATE_MDYT" ]; then
|
||||
echo "d$filedate"
|
||||
else
|
||||
echo "D$filedate"
|
||||
fi
|
||||
echo ":$pfile"
|
||||
echo
|
||||
fi
|
||||
done
|
||||
)
|
||||
echo '### 200'
|
||||
}
|
||||
|
||||
fish_list_poor_ls ()
|
||||
{
|
||||
FISH_DIR="$1"
|
||||
ls -lan "${FISH_DIR}" 2>/dev/null | grep '^[^cbt]' | (
|
||||
while read p l u g s m d y n n2 n3; do
|
||||
if [ -n "$g" ]; then
|
||||
if [ "$m" = "0" ]; then
|
||||
s=$d; m=$y; d=$n; y=$n2; n=$n3
|
||||
else
|
||||
n=$n" "$n2" "$n3
|
||||
fi
|
||||
echo "P$p $u $g"
|
||||
echo "S$s"
|
||||
echo "d$m $d $y"
|
||||
echo ":"$n
|
||||
echo
|
||||
fi
|
||||
done
|
||||
)
|
||||
ls -lan "${FISH_DIR}" 2>/dev/null | grep '^[cb]' | (
|
||||
while read p l u g a i m d y n n2 n3; do
|
||||
if [ -n "$g" ]; then
|
||||
if [ "$a" = "0" ]; then
|
||||
a=$m; i=$d; m=$y; d=$n; y=$n2; n=$n3
|
||||
else
|
||||
n=$n" "$n2" "$n3
|
||||
fi
|
||||
echo "P$p $u $g"
|
||||
echo "S$s"
|
||||
echo "d$m $d $y"
|
||||
echo ":"$n
|
||||
echo
|
||||
fi
|
||||
done
|
||||
)
|
||||
echo '### 200'
|
||||
}
|
||||
|
||||
fish_list_perl ()
|
||||
{
|
||||
FISH_DIR=$1
|
||||
perl -e '
|
||||
use strict;
|
||||
use POSIX;
|
||||
use Fcntl;
|
||||
use POSIX ":fcntl_h"; #S_ISLNK was here until 5.6
|
||||
import Fcntl ":mode" unless defined &S_ISLNK; #and is now here
|
||||
my $dirname = $ARGV[0];
|
||||
if (opendir (DIR, $dirname)) {
|
||||
while((my $filename = readdir (DIR))){
|
||||
my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size,$atime,$mtime,$ctime,$blksize,$blocks) = lstat("$dirname/$filename");
|
||||
my $mloctime= strftime("%%m-%%d-%%Y %%H:%%M", localtime $mtime);
|
||||
my $strutils_shell_escape_regex = s/([;<>\*\|`&\$!#\(\)\[\]\{\}:'\''"\ \\])/\\$1/g;
|
||||
my $e_filename = $filename;
|
||||
$e_filename =~ $strutils_shell_escape_regex;
|
||||
if (S_ISLNK ($mode)) {
|
||||
my $linkname = readlink ("$dirname/$filename");
|
||||
$linkname =~ $strutils_shell_escape_regex;
|
||||
printf("R%%o %%o $uid.$gid\nS$size\nd$mloctime\n:\"$e_filename\" -> \"$linkname\"\n\n", S_IMODE($mode), S_IFMT($mode));
|
||||
} else {
|
||||
printf("R%%o %%o $uid.$gid\nS$size\nd$mloctime\n:\"$e_filename\"\n\n", S_IMODE($mode), S_IFMT($mode));
|
||||
}
|
||||
}
|
||||
printf("### 200\n");
|
||||
closedir(DIR);
|
||||
} else {
|
||||
printf("### 500\n");
|
||||
}
|
||||
exit 0
|
||||
' "/${FISH_DIR}"
|
||||
}
|
||||
|
||||
if [ -n "${FISH_HAVE_PERL}" ]; then
|
||||
fish_list_perl "/${FISH_FILENAME}"
|
||||
elif [ -n "${FISH_HAVE_LSQ}" ]; then
|
||||
fish_list_lsq "/${FISH_FILENAME}"
|
||||
elif [ -n "${FISH_HAVE_SED}" ]; then
|
||||
fish_list_sed "/${FISH_FILENAME}"
|
||||
else
|
||||
fish_list_poor_ls "/${FISH_FILENAME}"
|
||||
fi
|
6
lib/vfs/mc-vfs/fish/mkdir
Исполняемый файл
6
lib/vfs/mc-vfs/fish/mkdir
Исполняемый файл
@ -0,0 +1,6 @@
|
||||
#MKD $FISH_FILENAME
|
||||
if mkdir "/$FISH_FILENAME" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
6
lib/vfs/mc-vfs/fish/mv
Исполняемый файл
6
lib/vfs/mc-vfs/fish/mv
Исполняемый файл
@ -0,0 +1,6 @@
|
||||
#RENAME $FISH_FILEFROM $FISH_FILETO
|
||||
if mv "/${FISH_FILEFROM}" "/${FISH_FILETO}" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
6
lib/vfs/mc-vfs/fish/rmdir
Исполняемый файл
6
lib/vfs/mc-vfs/fish/rmdir
Исполняемый файл
@ -0,0 +1,6 @@
|
||||
#RMD $FISH_FILENAME
|
||||
if rmdir "/${FISH_FILENAME}" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
10
lib/vfs/mc-vfs/fish/send
Исполняемый файл
10
lib/vfs/mc-vfs/fish/send
Исполняемый файл
@ -0,0 +1,10 @@
|
||||
#STOR $FISH_FILESIZE $FISH_FILENAME
|
||||
FILENAME="/${FISH_FILENAME}"
|
||||
echo "### 001"
|
||||
{
|
||||
while [ $FISH_FILESIZE -gt 0 ]; do
|
||||
cnt=`expr \\( $FISH_FILESIZE + 255 \\) / 256`
|
||||
n=`dd bs=256 count=$cnt | tee -a "${FILENAME}" | wc -c`
|
||||
FISH_FILESIZE=`expr $FISH_FILESIZE - $n`
|
||||
done
|
||||
}; echo "### 200"
|
6
lib/vfs/mc-vfs/fish/unlink
Исполняемый файл
6
lib/vfs/mc-vfs/fish/unlink
Исполняемый файл
@ -0,0 +1,6 @@
|
||||
#DELE $FISH_FILENAME
|
||||
if rm -f "/${FISH_FILENAME}" 2>/dev/null; then
|
||||
echo "### 000"
|
||||
else
|
||||
echo "### 500"
|
||||
fi
|
236
lib/vfs/mc-vfs/fishdef.h
Обычный файл
236
lib/vfs/mc-vfs/fishdef.h
Обычный файл
@ -0,0 +1,236 @@
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief Header: FISH script defaults
|
||||
*/
|
||||
|
||||
#ifndef MC_FISH_DEF_H
|
||||
#define MC_FISH_DEF_H
|
||||
|
||||
/* default 'ls' script */
|
||||
#define FISH_LS_DEF_CONTENT "" \
|
||||
"#LIST /${FISH_DIR}\n" \
|
||||
"export LC_TIME=C\n" \
|
||||
"FISH_DIR=\"/${FISH_PARAM1}\"\n" \
|
||||
"ls -lan \"${FISH_DIR}\" 2>/dev/null | grep '^[^cbt]' | (\n" \
|
||||
"while read p l u g s m d y n n2 n3; do\n" \
|
||||
" if [ -n \"$g\" ]; then\n" \
|
||||
" if [ \"$m\" = \"0\" ]; then\n" \
|
||||
" s=$d; m=$y; d=$n; y=$n2; n=$n3\n" \
|
||||
" else\n" \
|
||||
" n=$n\" \"$n2\" \"$n3\n" \
|
||||
" fi\n" \
|
||||
" echo \"P$p $u $g\"\n" \
|
||||
" echo \"S$s\"\n" \
|
||||
" echo \"d$m $d $y\"\n" \
|
||||
" echo \":\"$n\n" \
|
||||
" echo\n" \
|
||||
" fi\n" \
|
||||
"done\n" \
|
||||
")\n" \
|
||||
"ls -lan \"${FISH_DIR}\" 2>/dev/null | grep '^[cb]' | (\n" \
|
||||
"while read p l u g a i m d y n n2 n3; do\n" \
|
||||
" if [ -n \"$g\" ]; then\n" \
|
||||
" if [ \"$a\" = \"0\" ]; then\n" \
|
||||
" a=$m; i=$d; m=$y; d=$n; y=$n2; n=$n3\n" \
|
||||
" else\n" \
|
||||
" n=$n\" \"$n2\" \"$n3\n" \
|
||||
" fi\n" \
|
||||
" echo \"P$p $u $g\"\n" \
|
||||
" echo \"S$s\"\n" \
|
||||
" echo \"d$m $d $y\"\n" \
|
||||
" echo \":\"$n\n" \
|
||||
" echo\n" \
|
||||
" fi\n" \
|
||||
"done\n" \
|
||||
")\n" \
|
||||
"echo \"### 200\"\n"
|
||||
|
||||
/* default file exisits script */
|
||||
#define FISH_EXISTS_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"#ISEXISTS $FILENAME\n" \
|
||||
"ls -l \"${FILENAME}\" >/dev/null 2>/dev/null\n" \
|
||||
"echo '### '$?\n"
|
||||
|
||||
/* default 'mkdir' script */
|
||||
#define FISH_MKDIR_DEF_CONTENT "" \
|
||||
"FILENAME=/${FISH_PARAM1}\n" \
|
||||
"#MKD $FILENAME\n" \
|
||||
"if mkdir \"$FILENAME\" 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'unlink' script */
|
||||
#define FISH_UNLINK_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"#DELE $FILENAME\n" \
|
||||
"if rm -f \"${FILENAME}\" 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'chown' script */
|
||||
#define FISH_CHOWN_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"NEWUSER=\"${FISH_PARAM2}\"\n" \
|
||||
"NEWGROUP=\"${FISH_PARAM3}\"\n" \
|
||||
"#CHOWN $NEWUSER:$NEWGROUP $FILENAME\n" \
|
||||
"if chown ${NEWUSER}:${NEWGROUP} \"${FILENAME}\" ; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'chmod' script */
|
||||
#define FISH_CHMOD_DEF_CONTENT "" \
|
||||
"FISH_FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"FISH_MODE=\"${FISH_PARAM2}\"\n" \
|
||||
"#CHMOD $FISH_MODE $FISH_FILENAME\n" \
|
||||
"if chmod ${FISH_MODE} \"${FISH_FILENAME} 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'rmdir' script */
|
||||
#define FISH_RMDIR_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"#RMD $FILENAME\n" \
|
||||
"if rmdir \"${FILENAME}\" 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'ln -s' symlink script */
|
||||
#define FISH_LN_DEF_CONTENT "" \
|
||||
"FILEFROM=\"${FISH_PARAM1}\"\n" \
|
||||
"FILETO=\"/${FISH_PARAM2}\"\n" \
|
||||
"#SYMLINK $FILEFROM $FILETO\n" \
|
||||
"if ln -s \"${FILEFROM}\" \"${FILETO}\" 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'mv' script */
|
||||
#define FISH_MV_DEF_CONTENT "" \
|
||||
"FILEFROM=\"/${FISH_PARAM1}\"\n" \
|
||||
"FILETO=\"/${FISH_PARAM2}\"\n" \
|
||||
"#RENAME $FILEFROM $FILETO\n" \
|
||||
"if mv \"${FILEFROM}\" \"${FILETO}\" 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'ln' hardlink script */
|
||||
#define FISH_HARDLINK_DEF_CONTENT "" \
|
||||
"FILEFROM=\"/${FISH_PARAM1}\"\n" \
|
||||
"FILETO=\"/${FISH_PARAM2}\"\n" \
|
||||
"#LINK $FILEFROM $FILETO\n" \
|
||||
"if ln \"${FILEFROM}\" \"${FILETO}\" 2>/dev/null; then\n" \
|
||||
" echo \"### 000\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'retr' script */
|
||||
#define FISH_GET_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"export LC_TIME=C\n" \
|
||||
"#RETR $FILENAME\n" \
|
||||
"if dd if=\"${FILENAME}\" of=/dev/null bs=1 count=1 2>/dev/null ; then\n" \
|
||||
" ls -ln \"${FILENAME}\" 2>/dev/null | (\n" \
|
||||
" read p l u g s r\n" \
|
||||
" echo $s\n" \
|
||||
" )\n" \
|
||||
" echo \"### 100\"\n" \
|
||||
" cat \"${FILENAME}\"\n" \
|
||||
" echo \"### 200\"\n" \
|
||||
"else\n" \
|
||||
" echo \"### 500\"\n" \
|
||||
"fi\n"
|
||||
|
||||
/* default 'stor' script */
|
||||
#define FISH_SEND_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"FILESIZE=${FISH_PARAM2}\n" \
|
||||
"#STOR $FILESIZE $FILENAME\n" \
|
||||
"echo \"### 001\"\n" \
|
||||
"{\n" \
|
||||
" while [ $FILESIZE -gt 0 ]; do\n" \
|
||||
" cnt=`expr \\( $FILESIZE + 255 \\) / 256`\n" \
|
||||
" n=`dd bs=256 count=$cnt | tee -a \"${FILENAME}\" | wc -c`\n" \
|
||||
" FILESIZE=`expr $FILESIZE - $n`\n" \
|
||||
" done\n" \
|
||||
"}; echo \"### 200\"\n"
|
||||
|
||||
/* default 'appe' script */
|
||||
#define FISH_APPEND_DEF_CONTENT "" \
|
||||
"FILENAME=\"/${FISH_PARAM1}\"\n" \
|
||||
"FILESIZE=${FISH_PARAM2}\n" \
|
||||
"#APPE $FILESIZE $FILENAME\n" \
|
||||
"echo \"### 001\"\n" \
|
||||
"res=`exec 3>&1\n" \
|
||||
"(\n" \
|
||||
" head -c $FILESIZE -q - || echo DD >&3\n" \
|
||||
") 2>/dev/null | (\n" \
|
||||
" cat > \"${FILENAME}\"\n" \
|
||||
" cat > /dev/null\n" \
|
||||
")`; [ \"$res\" = DD ] && {\n" \
|
||||
" > \"${FILENAME}\"\n" \
|
||||
" while [ $FILESIZE -gt 0 ]\n" \
|
||||
" do\n" \
|
||||
" cnt=`expr \\( $FILESIZE + 255 \\) / 256`\n" \
|
||||
" n=`dd bs=256 count=$cnt | tee -a \"${FILENAME}\" | wc -c`\n" \
|
||||
" FILESIZE=`expr $FILESIZE - $n`\n" \
|
||||
" done\n" \
|
||||
"}; echo \"### 200\"\n"
|
||||
|
||||
/* default 'info' script */
|
||||
#define FISH_INFO_DEF_CONTENT "" \
|
||||
"export LC_TIME=C\n" \
|
||||
"#FISH_HAVE_HEAD 1\n" \
|
||||
"#FISH_HAVE_SED 2\n" \
|
||||
"#FISH_HAVE_AWK 4\n" \
|
||||
"#FISH_HAVE_PERL 8\n" \
|
||||
"#FISH_HAVE_LSQ 16\n" \
|
||||
"#FISH_HAVE_DATE_MDYT 32\n" \
|
||||
"res=0\n" \
|
||||
"if `echo yes| head -c 1 > /dev/null 2>&1` ; then\n" \
|
||||
" res=`expr $res + 1`\n" \
|
||||
"fi\n" \
|
||||
"if `sed --version >/dev/null 2>&1` ; then\n" \
|
||||
" res=`expr $res + 2`\n" \
|
||||
"fi\n" \
|
||||
"if `awk --version > /dev/null 2>&1` ; then\n" \
|
||||
" res=`expr $res + 4`\n" \
|
||||
"fi\n" \
|
||||
"if `perl -v > /dev/null 2>&1` ; then\n" \
|
||||
" res=`expr $res + 8`\n" \
|
||||
"fi\n" \
|
||||
"if `ls -Q / >/dev/null 2>&1` ; then\n" \
|
||||
" res=`expr $res + 16`\n" \
|
||||
"fi\n" \
|
||||
"dat=`ls -lan / 2>/dev/null | head -n 3|tail -n 1 | (\n" \
|
||||
" while read p l u g s rec; do\n" \
|
||||
" if [ -n \"$g\" ]; then\n" \
|
||||
" if [ -n \"$l\" ]; then\n" \
|
||||
" echo \"$rec\"\n" \
|
||||
" fi\n" \
|
||||
" fi\n" \
|
||||
" done\n" \
|
||||
") |head -c 1`\n" \
|
||||
"r=`echo \"0123456789\"| grep \"$dat\"`\n" \
|
||||
"if [ -z \"$r\" ]; then\n" \
|
||||
" res=`expr $res + 32`\n" \
|
||||
"fi\n" \
|
||||
"echo $res\n" \
|
||||
"echo \"### 200\"\n"
|
||||
|
||||
#endif
|
@ -34,48 +34,70 @@
|
||||
|
||||
|
||||
/* Single connection or archive */
|
||||
struct vfs_s_super {
|
||||
struct vfs_s_super
|
||||
{
|
||||
struct vfs_s_super **prevp, *next;
|
||||
struct vfs_class *me;
|
||||
struct vfs_s_inode *root;
|
||||
char *name; /* My name, whatever it means */
|
||||
int fd_usage; /* Number of open files */
|
||||
int ino_usage; /* Usage count of this superblock */
|
||||
int want_stale; /* If set, we do not flush cache properly */
|
||||
char *name; /* My name, whatever it means */
|
||||
int fd_usage; /* Number of open files */
|
||||
int ino_usage; /* Usage count of this superblock */
|
||||
int want_stale; /* If set, we do not flush cache properly */
|
||||
|
||||
union {
|
||||
struct {
|
||||
int sockr, sockw;
|
||||
char *cwdir;
|
||||
char *host, *user;
|
||||
char *password;
|
||||
int flags;
|
||||
} fish;
|
||||
struct {
|
||||
int sock;
|
||||
char *cwdir;
|
||||
char *host, *user;
|
||||
char *password;
|
||||
int port;
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
int sockr, sockw;
|
||||
char *cwdir;
|
||||
char *host, *user;
|
||||
char *password;
|
||||
int flags;
|
||||
char *scr_ls;
|
||||
char *scr_chmod;
|
||||
char *scr_exists;
|
||||
char *scr_mkdir;
|
||||
char *scr_unlink;
|
||||
char *scr_chown;
|
||||
char *scr_rmdir;
|
||||
char *scr_ln;
|
||||
char *scr_mv;
|
||||
char *scr_hardlink;
|
||||
char *scr_get;
|
||||
char *scr_send;
|
||||
char *scr_append;
|
||||
char *scr_info;
|
||||
int host_flags;
|
||||
char *scr_env;
|
||||
} fish;
|
||||
struct
|
||||
{
|
||||
int sock;
|
||||
char *cwdir;
|
||||
char *host, *user;
|
||||
char *password;
|
||||
int port;
|
||||
|
||||
char *proxy; /* proxy server, NULL if no proxy */
|
||||
int failed_on_login; /* used to pass the failure reason to upper levels */
|
||||
int use_passive_connection;
|
||||
int remote_is_amiga; /* No leading slash allowed for AmiTCP (Amiga) */
|
||||
int isbinary;
|
||||
int cwd_deferred; /* current_directory was changed but CWD command hasn't
|
||||
been sent yet */
|
||||
int strict; /* ftp server doesn't understand
|
||||
"LIST -la <path>"; use "CWD <path>"/
|
||||
"LIST" instead */
|
||||
int ctl_connection_busy;
|
||||
} ftp;
|
||||
struct {
|
||||
int fd;
|
||||
struct stat st;
|
||||
int type; /* Type of the archive */
|
||||
struct defer_inode *deferred; /* List of inodes for which another entries may appear */
|
||||
} arch;
|
||||
char *proxy; /* proxy server, NULL if no proxy */
|
||||
int failed_on_login; /* used to pass the failure reason to upper levels */
|
||||
int use_passive_connection;
|
||||
int remote_is_amiga; /* No leading slash allowed for AmiTCP (Amiga) */
|
||||
int isbinary;
|
||||
int cwd_deferred; /* current_directory was changed but CWD command hasn't
|
||||
been sent yet */
|
||||
int strict; /* ftp server doesn't understand
|
||||
* "LIST -la <path>"; use "CWD <path>"/
|
||||
* "LIST" instead
|
||||
*/
|
||||
int ctl_connection_busy;
|
||||
} ftp;
|
||||
struct
|
||||
{
|
||||
int fd;
|
||||
struct stat st;
|
||||
int type; /* Type of the archive */
|
||||
struct defer_inode *deferred; /* List of inodes for which another entries may appear */
|
||||
} arch;
|
||||
} u;
|
||||
};
|
||||
|
||||
@ -83,42 +105,48 @@ struct vfs_s_super {
|
||||
* Single virtual file - directory entry. The same inode can have many
|
||||
* entries (i.e. hard links), but usually has only one.
|
||||
*/
|
||||
struct vfs_s_entry {
|
||||
struct vfs_s_entry **prevp, *next; /* Pointers in the entry list */
|
||||
struct vfs_s_inode *dir; /* Directory we are in, i.e. our parent */
|
||||
char *name; /* Name of this entry */
|
||||
struct vfs_s_inode *ino; /* ... and its inode */
|
||||
struct vfs_s_entry
|
||||
{
|
||||
struct vfs_s_entry **prevp, *next; /* Pointers in the entry list */
|
||||
struct vfs_s_inode *dir; /* Directory we are in, i.e. our parent */
|
||||
char *name; /* Name of this entry */
|
||||
struct vfs_s_inode *ino; /* ... and its inode */
|
||||
};
|
||||
|
||||
/* Single virtual file - inode */
|
||||
struct vfs_s_inode {
|
||||
struct vfs_s_super *super; /* Archive the file is on */
|
||||
struct vfs_s_entry *ent; /* Our entry in the parent directory -
|
||||
use only for directories because they
|
||||
cannot be hardlinked */
|
||||
struct vfs_s_inode
|
||||
{
|
||||
struct vfs_s_super *super; /* Archive the file is on */
|
||||
struct vfs_s_entry *ent; /* Our entry in the parent directory -
|
||||
use only for directories because they
|
||||
cannot be hardlinked */
|
||||
struct vfs_s_entry *subdir; /* If this is a directory, its entry */
|
||||
struct stat st; /* Parameters of this inode */
|
||||
char *linkname; /* Symlink's contents */
|
||||
char *localname; /* Filename of local file, if we have one */
|
||||
struct timeval timestamp; /* Subclass specific */
|
||||
long data_offset; /* Subclass specific */
|
||||
struct stat st; /* Parameters of this inode */
|
||||
char *linkname; /* Symlink's contents */
|
||||
char *localname; /* Filename of local file, if we have one */
|
||||
struct timeval timestamp; /* Subclass specific */
|
||||
long data_offset; /* Subclass specific */
|
||||
};
|
||||
|
||||
/* Data associated with an open file */
|
||||
struct vfs_s_fh {
|
||||
struct vfs_s_fh
|
||||
{
|
||||
struct vfs_s_inode *ino;
|
||||
long pos; /* This is for module's use */
|
||||
int handle; /* This is for module's use, but if != -1, will be mc_close()d */
|
||||
int changed; /* Did this file change? */
|
||||
int linear; /* Is that file open with O_LINEAR? */
|
||||
union {
|
||||
struct {
|
||||
off_t got, total;
|
||||
int append;
|
||||
} fish;
|
||||
struct {
|
||||
int sock, append;
|
||||
} ftp;
|
||||
long pos; /* This is for module's use */
|
||||
int handle; /* This is for module's use, but if != -1, will be mc_close()d */
|
||||
int changed; /* Did this file change? */
|
||||
int linear; /* Is that file open with O_LINEAR? */
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
off_t got, total;
|
||||
int append;
|
||||
} fish;
|
||||
struct
|
||||
{
|
||||
int sock, append;
|
||||
} ftp;
|
||||
} u;
|
||||
};
|
||||
|
||||
@ -126,82 +154,69 @@ struct vfs_s_fh {
|
||||
* One of our subclasses (tar, cpio, fish, ftpfs) with data and methods.
|
||||
* Extends vfs_class. Stored in the "data" field of vfs_class.
|
||||
*/
|
||||
struct vfs_s_subclass {
|
||||
struct vfs_s_subclass
|
||||
{
|
||||
struct vfs_s_super *supers;
|
||||
int inode_counter;
|
||||
int flags; /* whether the subclass is remove, read-only etc */
|
||||
int flags; /* whether the subclass is remove, read-only etc */
|
||||
dev_t rdev;
|
||||
FILE *logfile;
|
||||
int flush; /* if set to 1, invalidate directory cache */
|
||||
int flush; /* if set to 1, invalidate directory cache */
|
||||
|
||||
int (*init_inode) (struct vfs_class *me, struct vfs_s_inode *ino); /* optional */
|
||||
void (*free_inode) (struct vfs_class *me, struct vfs_s_inode *ino); /* optional */
|
||||
int (*init_entry) (struct vfs_class *me, struct vfs_s_entry *entry); /* optional */
|
||||
int (*init_inode) (struct vfs_class * me, struct vfs_s_inode * ino); /* optional */
|
||||
void (*free_inode) (struct vfs_class * me, struct vfs_s_inode * ino); /* optional */
|
||||
int (*init_entry) (struct vfs_class * me, struct vfs_s_entry * entry); /* optional */
|
||||
|
||||
void *(*archive_check) (struct vfs_class *me, const char *name, char *op); /* optional */
|
||||
int (*archive_same) (struct vfs_class *me, struct vfs_s_super *psup,
|
||||
const char *archive_name, char *op, void *cookie);
|
||||
int (*open_archive) (struct vfs_class *me, struct vfs_s_super *psup,
|
||||
const char *archive_name, char *op);
|
||||
void (*free_archive) (struct vfs_class *me,
|
||||
struct vfs_s_super *psup);
|
||||
void *(*archive_check) (struct vfs_class * me, const char *name, char *op); /* optional */
|
||||
int (*archive_same) (struct vfs_class * me, struct vfs_s_super * psup,
|
||||
const char *archive_name, char *op, void *cookie);
|
||||
int (*open_archive) (struct vfs_class * me, struct vfs_s_super * psup,
|
||||
const char *archive_name, char *op);
|
||||
void (*free_archive) (struct vfs_class * me, struct vfs_s_super * psup);
|
||||
|
||||
int (*fh_open) (struct vfs_class *me, struct vfs_s_fh *fh, int flags,
|
||||
int mode);
|
||||
int (*fh_close) (struct vfs_class *me, struct vfs_s_fh *fh);
|
||||
int (*fh_open) (struct vfs_class * me, struct vfs_s_fh * fh, int flags, int mode);
|
||||
int (*fh_close) (struct vfs_class * me, struct vfs_s_fh * fh);
|
||||
|
||||
struct vfs_s_entry *(*find_entry) (struct vfs_class *me,
|
||||
struct vfs_s_inode *root,
|
||||
const char *path, int follow, int flags);
|
||||
int (*dir_load) (struct vfs_class *me, struct vfs_s_inode *ino,
|
||||
char *path);
|
||||
int (*dir_uptodate) (struct vfs_class *me, struct vfs_s_inode *ino);
|
||||
int (*file_store) (struct vfs_class *me, struct vfs_s_fh *fh,
|
||||
char *path, char *localname);
|
||||
struct vfs_s_entry *(*find_entry) (struct vfs_class * me,
|
||||
struct vfs_s_inode * root,
|
||||
const char *path, int follow, int flags);
|
||||
int (*dir_load) (struct vfs_class * me, struct vfs_s_inode * ino, char *path);
|
||||
int (*dir_uptodate) (struct vfs_class * me, struct vfs_s_inode * ino);
|
||||
int (*file_store) (struct vfs_class * me, struct vfs_s_fh * fh, char *path, char *localname);
|
||||
|
||||
int (*linear_start) (struct vfs_class *me, struct vfs_s_fh *fh,
|
||||
off_t from);
|
||||
int (*linear_read) (struct vfs_class *me, struct vfs_s_fh *fh,
|
||||
void *buf, int len);
|
||||
void (*linear_close) (struct vfs_class *me, struct vfs_s_fh *fh);
|
||||
int (*linear_start) (struct vfs_class * me, struct vfs_s_fh * fh, off_t from);
|
||||
int (*linear_read) (struct vfs_class * me, struct vfs_s_fh * fh, void *buf, int len);
|
||||
void (*linear_close) (struct vfs_class * me, struct vfs_s_fh * fh);
|
||||
};
|
||||
|
||||
|
||||
/* entries and inodes */
|
||||
struct vfs_s_inode *vfs_s_new_inode (struct vfs_class *me,
|
||||
struct vfs_s_super *super,
|
||||
struct stat *initstat);
|
||||
struct vfs_s_super *super, struct stat *initstat);
|
||||
struct vfs_s_entry *vfs_s_new_entry (struct vfs_class *me, const char *name,
|
||||
struct vfs_s_inode *inode);
|
||||
struct vfs_s_inode *inode);
|
||||
void vfs_s_free_entry (struct vfs_class *me, struct vfs_s_entry *ent);
|
||||
void vfs_s_insert_entry (struct vfs_class *me, struct vfs_s_inode *dir,
|
||||
struct vfs_s_entry *ent);
|
||||
void vfs_s_insert_entry (struct vfs_class *me, struct vfs_s_inode *dir, struct vfs_s_entry *ent);
|
||||
struct stat *vfs_s_default_stat (struct vfs_class *me, mode_t mode);
|
||||
|
||||
struct vfs_s_entry *vfs_s_generate_entry (struct vfs_class *me, const char *name,
|
||||
struct vfs_s_inode *parent,
|
||||
mode_t mode);
|
||||
struct vfs_s_inode *parent, mode_t mode);
|
||||
struct vfs_s_inode *vfs_s_find_inode (struct vfs_class *me,
|
||||
const struct vfs_s_super *super,
|
||||
const char *path, int follow, int flags);
|
||||
struct vfs_s_inode *vfs_s_find_root (struct vfs_class *me,
|
||||
struct vfs_s_entry *entry);
|
||||
const struct vfs_s_super *super,
|
||||
const char *path, int follow, int flags);
|
||||
struct vfs_s_inode *vfs_s_find_root (struct vfs_class *me, struct vfs_s_entry *entry);
|
||||
|
||||
/* outside interface */
|
||||
void vfs_s_init_class (struct vfs_class *vclass,
|
||||
struct vfs_s_subclass *sub);
|
||||
void vfs_s_init_class (struct vfs_class *vclass, struct vfs_s_subclass *sub);
|
||||
const char *vfs_s_get_path_mangle (struct vfs_class *me, char *inname,
|
||||
struct vfs_s_super **archive, int flags);
|
||||
struct vfs_s_super **archive, int flags);
|
||||
void vfs_s_invalidate (struct vfs_class *me, struct vfs_s_super *super);
|
||||
char *vfs_s_fullpath (struct vfs_class *me, struct vfs_s_inode *ino);
|
||||
|
||||
/* network filesystems support */
|
||||
int vfs_s_select_on_two (int fd1, int fd2);
|
||||
int vfs_s_get_line (struct vfs_class *me, int sock, char *buf, int buf_len,
|
||||
char term);
|
||||
int vfs_s_get_line_interruptible (struct vfs_class *me, char *buffer,
|
||||
int size, int fd);
|
||||
|
||||
int vfs_s_get_line (struct vfs_class *me, int sock, char *buf, int buf_len, char term);
|
||||
int vfs_s_get_line_interruptible (struct vfs_class *me, char *buffer, int size, int fd);
|
||||
/* misc */
|
||||
int vfs_s_retrieve_file (struct vfs_class *me, struct vfs_s_inode *ino);
|
||||
|
||||
|
Загрузка…
x
Ссылка в новой задаче
Block a user