1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
|
FIles transferred over SSH
~~~~~~~~~~~~~~~~~~~~~~~~~~
This protocol was designed for transferring files over a remote shell
connection (rsh and compatibles). It can be as well used for transfers over
rsh, and there may be other uses.
Since version 4.8.31 Midnight Commander doesn't support FISH commands with form
#FISH_COMMAND [arg1] [arg2] ... [argN]
and sends requests as shell scripts only.
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
directory. User-defined overrides take priority over sytem-wide scripts
if they exist. The order in which the directories are traversed is as follows:
/usr/libexec/mc/shell
~/.local/share/mc/shell/<hostname>/
Server's reply is multiline, but always ends with
### 000<optional text>
line. ### is prefix to mark this line, 000 is return code. Return
codes are superset to those used in ftp.
There are few new exit codes defined:
000 don't know; if there were no previous lines, this marks COMPLETE
success, if they were, it marks failure.
001 don't know; if there were no previous lines, this marks
PRELIMinary success, if they were, it marks failure
Connecting
~~~~~~~~~~
MC uses "echo SHELL:;/bin/sh" as command executed on remote machine.
Actions
~~~~~~~
Get info about host into $result
echo $result
echo '### 200'
Script: info
--------------------------------------------------------------------------------
List directory or get status information about single file.
ls -lLa $1 | grep '^[^cbt]' | ( while read p x u g s m d y n; do echo "P$p $u.$g
S$s
d$m $d $y
:$n
"; done )
ls -lLa $1 | grep '^[cb]' | ( while read p x u g a i m d y n; do echo "P$p $u.$g
E$a$i
dD$m $d $y
:$n
"; done )
echo '### 200'
Output is in following form (any line except :<filename> may be omitted):
P<unix permissions> <owner>.<group>
S<size>
d<3-letters month name> <day> <year or HH:MM>
D<year> <month> <day> <hour> <minute> <second>[.1234]
E<major-of-device>,<minor>
:<filename>
L<filename symlink points to>
<blank line to separate items>
Unix permissions are of form X---------
where X is type of file:
'-' a regular file
'd' a directory
'c' a character device
'b' a block device
'l' a symbolic link
'p' a FIFO
's' a socket.
'd' has three fields:
month (one of strings Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec)
day of month
single number indicating year, or HH:MM field (assume current year in such
case). As you've probably noticed, this is pretty broken; it is for
compatibility with ls listing.
Script: ls
--------------------------------------------------------------------------------
Get file
ls -l /some/name | ( read a b c d x e; echo $x ); echo '### 100'; cat /some/name; echo '### 200'
Server sends line with filesize on it, followed by line with ### 100
indicating partial success, then it sends binary data (exactly filesize bytes)
and follows them with (with no preceding newline) ### 200.
Note that there's no way to abort running RETR command - except closing
the connection.
Script: get
--------------------------------------------------------------------------------
Put file
> /file/name; echo '### 001'; ( dd bs=4096 count=<size/4096>; dd bs=<size%4096> count=1 ) 2>/dev/null | ( cat > %s; cat > /dev/null ); echo '### 200'
This command is for storing /file/name, which is exactly size bytes big.
You probably think I went crazy. Well, I did not: that strange cat > /dev/null
has purpose to discard any extra data which was not written to disk (due to for
example out of space condition).
[Why? Imagine uploading file with "rm -rf /" line in it.]
Script: send
--------------------------------------------------------------------------------
Change directory
cd /somewhere; echo '### 000'
It is specified here, but I'm not sure how wise idea is to use this one:
it breaks stateless-ness of the protocol.
--------------------------------------------------------------------------------
Change mode
chmod 1234 file; echo '### 000'
Script: chmod
--------------------------------------------------------------------------------
Change own
chown user /file/name; echo '### 000'
Script: chown
--------------------------------------------------------------------------------
Remove file
rm -f /some/path; echo '### 000'
Sctipt: unlink
--------------------------------------------------------------------------------
Make directory:
mkdir /some/path; echo '### 000'
Script: mkdir
--------------------------------------------------------------------------------
Remove directory
rmdir /some/path; echo '### 000'
Script: rmdir
--------------------------------------------------------------------------------
Rename/move file
mv /path/a /path/b; echo '### 000'
Script: mv
--------------------------------------------------------------------------------
Make link
ln /path/a /path/b; echo '### 000'
Script: hardlink
--------------------------------------------------------------------------------
Make symbolic link:
ln -s /path/a /path/b; echo '### 000'
Script: ln
--------------------------------------------------------------------------------
You can use following parameters:
SHELL_FILESIZE
SHELL_FILENAME
SHELL_FILEMODE
SHELL_FILEOWNER
SHELL_FILEGROUPE
SHELL_FILEFROM
SHELL_FILETO
NB:
'SHELL_FILESIZE' is used if we operate with single file name in 'unlink', 'rmdir', 'chmod', etc...
'SHELL_FILEFROM','SHELL_FILETO' are used if we operate with two files in 'ln', 'hardlink', 'mv' etc...
'SHELL_FILEOWNER', 'SHELL_FILEGROUPE' are a new user/group in chown
and flags:
SHELL_HAVE_HEAD
SHELL_HAVE_SED
SHELL_HAVE_AWK
SHELL_HAVE_PERL
SHELL_HAVE_LSQ
SHELL_HAVE_DATE_MDYT
That's all, folks!
pavel@ucw.cz
aborodin@vmail.ru
|