File: vfs-fsapi.man

package info (click to toggle)
tclvfs 1.3-2
  • links: PTS
  • area: main
  • in suites: etch, etch-m68k
  • size: 1,240 kB
  • ctags: 416
  • sloc: tcl: 3,670; xml: 2,882; sh: 2,833; ansic: 1,264; makefile: 58
file content (413 lines) | stat: -rw-r--r-- 13,578 bytes parent folder | download | duplicates (6)
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
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
[comment {-*- tcl -*- doctools manpage}]
[manpage_begin vfs-fsapi n 1.0]
[copyright {2001-2003 Vince Darley <vincentdarley@users.sourceforge.net>}]
[copyright {2003 Andreas Kupries <andreas_kupries@users.sourceforge.net>}]
[moddesc   {Tcl-level Virtual Filesystems}]
[titledesc {API for the implementation of a filesystem in Tcl}]
[require Tcl 8.4]
[require vfs [opt 1.2.1]]
[description]

This document explains the API used by the package [package vfs]
to communicate with filesystem implementations written in tcl.

[section {HANDLER OVERVIEW}]

The package [package vfs] intercepts every filesystem operation which
falls within a given mount point, and passes the operation on to the
mount point's [cmd vfshandler] command in the interpreter which
registered it.

[para]

If the handler takes appropriate action for each of the cases it is
called for, a complete, perfect virtual filesystem will be achieved,
indistinguishable to Tcl from the native filesystem.

(CAVEATS: Right now [package vfs] does not expose to Tcl all the
permission-related flags of [cmd glob]).

[para]
[list_begin definitions]

[call [cmd vfshandler] [arg subcmd] [arg root] [arg relative] [arg actualpath] [arg args]...]

The first argument specifies the operation to perform on behalf of the
filesystem code in the tcl core, the remainder specify the file path
on which to operate, in different forms, and parts, and any additional
arguments which may be required to carry out the action.

[nl]

To demonstrate the treatment of a path by the generic layer we use
[file C:/foo/bar/mount.zip/xxx/yyy] as an example and additionally
assume that the following conditions are true:

[list_begin enum]
[enum]
[file mount.zip] is a zip archive which has been mounted on top of
itself,
[enum]
said zip archive contains a file with path [file xxx/yyy],
[enum]
the current working directory of the application is inside of
directory [file xxx],
[enum]
and the command executed is [cmd {file exists yyy}].
[list_end]
[nl]

The file separator between [arg root] and [arg relative] is omitted.
Most filesystem operations need only the [arg relative] argument for
their correct operation, but some actually require the other parts of
the path.


[list_begin definitions]

[lst_item [arg subcmd]]

This argument of the handler can be one of the following
[method access], [method createdirectory], [method deletefile],
[method fileattributes], [method matchindirectory], [method open],
[method removedirectory], [method stat], or [method utime].

[nl]

The generic layer expects that the subcommands of a handler signal
error conditions by calling [cmd {vfs::filesystem posixerror}] with
the appropriate posix error code instead of throwing a tcl error. If
the latter is done nevertheless it will be treated as an unknown posix
error.

[nl]

There are three exceptions to the rule above: If any of [method open]
(when an interpreter is given), [method matchindirectory], and

[method fileattributes] (for a set or get operation only) throw a tcl
error, this error will be passed up to the caller of the filesystem
command which invoked the handler. Note that this does not preclude
the ability of these subcommands to use the command

[cmd {vfs::filesystem posixerror}] to report more regular filesystem
errors.


[lst_item [arg root]]

Part of the specification of the path to operate upon. It contains the
part of the path which lies outside this filesystem's mount point. For
example outlined above its value will be [file C:/foo/bar/mount.zip].

[lst_item [arg relative]]

Part of the specification of the path to operate upon. It contains the
part of the path which lies inside this filesystem. For example
outlined above its value will be [file xxx/yyy].

[lst_item [arg actualpath]]

Part of the specification of the path to operate upon. It contains the
original (unnormalized) name of the path which was used in the current
command wherever it originated (in Tcl or C). For example outlined
above its value will be [file yyy].

[list_end]
[list_end]

[section {HANDLER METHODS}]
[list_begin definitions]


[call [cmd vfshandler] [method access] [arg root] [arg relative] [arg actualpath] [arg mode]]

Signal a posix error if the specified access [arg mode] (an integer
number) is not compatible with the file or directory described by the
path. The generic layer will ignore any non-empty return value.

[nl]

The command [cmd vfs::accessMode] (see section

[sectref {HANDLER ENVIRONMENT}]) can be used to convert the integer
[arg mode] into an easier to check string value.


[call [cmd vfshandler] [method createdirectory] [arg root] [arg relative] [arg actualpath]]

Create a directory with the given name.  The command can assume that
all sub-directories in the path exist and are valid, and that the
actual desired path does not yet exist (Tcl takes care of all of that
for us).


[call [cmd vfshandler] [method deletefile] [arg root] [arg relative] [arg actualpath]]

Delete the given file.


[call [cmd vfshandler] [method fileattributes] [arg root] [arg relative] [arg actualpath] [opt [arg index]] [opt [arg value]]]

The command has to return a list containing the names of all
acceptable attributes, if neither [arg index] nor [arg value] were
specified.

[nl]

The command has to return the value of the [arg index]'th attribute if
the [arg index] is specified, but not the [arg value]. The attributes
are counted in the same order as their names appear in the list
returned by a call where neither [arg index] nor [arg value] were
specified. The first attribute is has the index [const 0].

[nl]

The command has to set the value of the [arg index]'th attribute to
[arg value] if both [arg index] and [arg value] were specified for the
call.


[call [cmd vfshandler] [method matchindirectory] [arg root] [arg relative] [arg actualpath] [arg pattern] [arg types]]

Return the list of files or directories in the given path which match
the glob [arg pattern] and are compatible with the specified list of
[arg types]. The specified path is always the name of an existing
directory.

[nl]

[emph Note:] As Tcl generates requests for directory-only matches from
the filesystems involved when performing any type of recursive
globbing this subcommand absolutely has to handle such (and file-only)
requests correctly or bad things (TM) will happen.

[nl]

The commands [cmd vfs::matchDirectories] and [cmd vfs::matchFiles]
(see section [sectref {HANDLER ENVIRONMENT}]) can aid the
implementation greatly in this task.


[call [cmd vfshandler] [method open] [arg root] [arg relative] [arg actualpath] [arg mode] [arg permissions]]

Either returns a list describing the successfully opened file, or
throws an error describing how the operation failed.

[nl]

The list returned upon success contains at least one and at most two
elements. The first, obligatory, element is always the handle of the
channel which was created to allow access to the contents of the
file.

[nl]

If specified the second element will be interpreted as a callback,
i.e. a command prefix. This prefix will always be executed as is,
i.e. without additional arguments. Any required arguments have to be
returned as part of the result of the call to [method open].

[nl]

If present the specified callback will be evaluated just before the
channel is closed [emph {by the generic filesystem layer}]. The
callback itself [emph {must not}] call [cmd close].

[nl]

The channel however is live enough to allow [cmd seek] and [cmd read]
operations. In addition all available data will have been flushed into
it already. This means, for example, that the callback can seek to the
beginning of the said channel, read its contents and then store the
gathered data elsewhere. In other words, this callback is not only
crucial to the cleanup of any resources associated with an opened
file, but also for the ability to implement a filesystem which can be
written to.

[nl]

Under normal circumstances return code and any errors thrown by the
callback itself are ignored. In that case errors have to be signaled
asychronously, for example by calling [cmd bgerror].

However if, through a call of the subcommand [method internalerror],
an error handling script has been specified for the file system, all
errors thrown here will be passed to that script for further action.


[list_begin definitions]

[lst_item [arg mode]]
can be any of [const r], [const w], [const a], [const w+], or [const a+].

[lst_item [arg permissions]]
determines the native mode the openend file is created with. Relevant
only of the open [arg mode] actually requests the creation of a
non-existing file, i.e. is not [const r].

[list_end]
[nl]

[call [cmd vfshandler] [method removedirectory] [arg root] [arg relative] [arg actualpath] [arg recursive]]

Delete the given directory. Argument [arg recursive] is a boolean. If
the specified value is [const true] then even if the directory is
non-empty, an attempt has to be made to recursively delete it and its
contents.  If the spcified value is [const false] and the directory is
non-empty, a posix error ([const EEXIST]) has to be thrown.


[call [cmd vfshandler] [method stat] [arg root] [arg relative] [arg actualpath]]

The result has to be a list of keys and values, in a format acceptable
to the builtin command [cmd {array set}]. It describes the contents of
a stat structure. The order of the keys in the list is not important.

[nl]

Given this the subcommand should use something like 

[example {return [list dev 0 type file mtime 1234 ...].}]

as the last command of its implementation.

[nl]

The following keys and their values have to be supplied by the
filesystem:

[list_begin definitions]
[lst_item [const dev]]

A long integer number, the device number of the path stat was called for.

[lst_item [const ino]]

A long integer number, the inode number of the path stat was called for.

Each path handled by the filesystem should be uniquely identified by
the combination of device and inode number. Violating this principle
will cause higher-level algorithms which(have to) keep track of device
and inode information to fail in all manners possible.

[nl]

An example of such an algorithm would be a directory walker using
device/inode information to keep itself out of infinite loops
generated through symbolic links. Returning non-unique device/inode
information will most likely cause such a walker to skip over paths
under the wrong assumption of having them seen already.

[lst_item [const mode]]

An integer number, the access mode of the path. It is this mode which
is checked by the subcommand [method access].

[lst_item [const nlink]]

A long integer number, the number of hard links to the specified path.

[lst_item [const uid]]

A long integer number, the id of the user owning the virtual path.

[lst_item [const gid]]

A long integer number, the id of the user group the virtual path
belongs to.

[lst_item [const size]]

A long integer number, the true size of the virtual path, in bytes.

[lst_item [const atime]]

A long integer number, the time of the latest access to the path, in
seconds since the epoch. Convertible into a readable date/time by the
command [cmd {clock format}].

[lst_item [const mtime]]

A long integer number, the time of the latest modification of the
path, in seconds since the epoch. Convertible into a readable
date/time by the command [cmd {clock format}].

[lst_item [const ctime]]

A long integer number, the time of the path was created, in seconds
since the epoch. Convertible into a readable date/time by the command
[cmd {clock format}].

[lst_item [const type]]

A string, either [const directory], or [const file], describing the
type of the given path.

[list_end]
[nl]

[call [cmd vfshandler] [method utime] [arg root] [arg relative] [arg actualpath] [arg actime] [arg mtime]]

Set the access and modification times of the given file (these are
read with [method stat]).

[list_end]


[section {HANDLER ENVIRONMENT}]

The implementation of a filesystem handler can rely on the
existence of the following utility commands:

[list_begin definitions]
[call [cmd vfs::accessMode] [arg mode]]

This commands converts an access [arg mode] given as integer into a
string, one of [const F], [const X], [const W], [const XW], [const R],
[const RX], and [const RW].


[call [cmd vfs::matchDirectories] [arg types]]

Checks if the glob types specification ask for the inclusion of
directories. Returns a boolean result. [const true] is returned if
types does ask for directories, else [const false].


[call [cmd vfs::matchFiles] [arg types]]

Checks if the glob types specification ask for the inclusion of
files. Returns a boolean result. [const true] is returned if types
does ask for directories, else [const false].


[call [cmd vfs::matchCorrectTypes] [arg types] [arg filelist] [opt [arg inDir]]]

Returns that subset of the [arg filelist] which are compatible with
the [arg types] given. The elements of [arg filelist] are either
absolute paths, or names of files in the directory [arg indir].  The
latter interpretation is taken if and only if the argument [arg indir]
is specified.


[list_end]

[section {FILESYSTEM DEBUGGING}]

To debug a problem in the implementation of a filesystem use code as
shown below. This registers the command [cmd report] as the error
handler for the filesystem, which in turn prints out the error stack
provided by tcl.

[para]
[example {vfs::filesystem internalerror report

proc report {} {
    puts stderr $::errorInfo
}}]

[see_also vfs vfs-filesystems]
[keywords vfs filesystem file]
[manpage_end]