2010-10-28 20:18:16 +08:00
|
|
|
## File System
|
|
|
|
|
|
|
|
File I/O is provided by simple wrappers around standard POSIX functions. To
|
|
|
|
use this module do `require('fs')`. All the methods have asynchronous and
|
2010-11-22 06:22:34 +08:00
|
|
|
synchronous forms.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
The asynchronous form always take a completion callback as its last argument.
|
|
|
|
The arguments passed to the completion callback depend on the method, but the
|
|
|
|
first argument is always reserved for an exception. If the operation was
|
|
|
|
completed successfully, then the first argument will be `null` or `undefined`.
|
|
|
|
|
|
|
|
Here is an example of the asynchronous version:
|
|
|
|
|
|
|
|
var fs = require('fs');
|
|
|
|
|
|
|
|
fs.unlink('/tmp/hello', function (err) {
|
|
|
|
if (err) throw err;
|
|
|
|
console.log('successfully deleted /tmp/hello');
|
|
|
|
});
|
|
|
|
|
|
|
|
Here is the synchronous version:
|
|
|
|
|
|
|
|
var fs = require('fs');
|
|
|
|
|
|
|
|
fs.unlinkSync('/tmp/hello')
|
|
|
|
console.log('successfully deleted /tmp/hello');
|
|
|
|
|
|
|
|
With the asynchronous methods there is no guaranteed ordering. So the
|
|
|
|
following is prone to error:
|
|
|
|
|
|
|
|
fs.rename('/tmp/hello', '/tmp/world', function (err) {
|
|
|
|
if (err) throw err;
|
|
|
|
console.log('renamed complete');
|
|
|
|
});
|
|
|
|
fs.stat('/tmp/world', function (err, stats) {
|
|
|
|
if (err) throw err;
|
|
|
|
console.log('stats: ' + JSON.stringify(stats));
|
|
|
|
});
|
|
|
|
|
|
|
|
It could be that `fs.stat` is executed before `fs.rename`.
|
|
|
|
The correct way to do this is to chain the callbacks.
|
|
|
|
|
|
|
|
fs.rename('/tmp/hello', '/tmp/world', function (err) {
|
|
|
|
if (err) throw err;
|
|
|
|
fs.stat('/tmp/world', function (err, stats) {
|
|
|
|
if (err) throw err;
|
|
|
|
console.log('stats: ' + JSON.stringify(stats));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
In busy processes, the programmer is _strongly encouraged_ to use the
|
|
|
|
asynchronous versions of these calls. The synchronous versions will block
|
|
|
|
the entire process until they complete--halting all connections.
|
|
|
|
|
|
|
|
### fs.rename(path1, path2, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous rename(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.renameSync(path1, path2)
|
|
|
|
|
|
|
|
Synchronous rename(2).
|
|
|
|
|
|
|
|
### fs.truncate(fd, len, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous ftruncate(2). No arguments other than a possible exception are
|
2010-11-18 00:04:21 +08:00
|
|
|
given to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.truncateSync(fd, len)
|
|
|
|
|
|
|
|
Synchronous ftruncate(2).
|
|
|
|
|
2011-04-02 08:46:18 +08:00
|
|
|
### fs.chown(path, mode, [callback])
|
|
|
|
|
|
|
|
Asycnronous chown(2). No arguments other than a possible exception are given
|
|
|
|
to the completion callback.
|
|
|
|
|
|
|
|
### fs.chownSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous chown(2).
|
|
|
|
|
|
|
|
### fs.fchown(path, mode, [callback])
|
|
|
|
|
|
|
|
Asycnronous fchown(2). No arguments other than a possible exception are given
|
|
|
|
to the completion callback.
|
|
|
|
|
|
|
|
### fs.fchownSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous fchown(2).
|
|
|
|
|
|
|
|
### fs.lchown(path, mode, [callback])
|
|
|
|
|
|
|
|
Asycnronous lchown(2). No arguments other than a possible exception are given
|
|
|
|
to the completion callback.
|
|
|
|
|
|
|
|
### fs.lchownSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous lchown(2).
|
|
|
|
|
2010-10-28 20:18:16 +08:00
|
|
|
### fs.chmod(path, mode, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous chmod(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.chmodSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous chmod(2).
|
2010-11-22 06:22:34 +08:00
|
|
|
|
2011-04-02 08:46:18 +08:00
|
|
|
### fs.fchmod(fd, mode, [callback])
|
|
|
|
|
|
|
|
Asynchronous fchmod(2). No arguments other than a possible exception
|
|
|
|
are given to the completion callback.
|
|
|
|
|
|
|
|
### fs.fchmodSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous fchmod(2).
|
|
|
|
|
|
|
|
### fs.lchmod(fd, mode, [callback])
|
|
|
|
|
|
|
|
Asynchronous lchmod(2). No arguments other than a possible exception
|
|
|
|
are given to the completion callback.
|
|
|
|
|
|
|
|
### fs.lchmodSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous lchmod(2).
|
|
|
|
|
2010-10-28 20:18:16 +08:00
|
|
|
### fs.stat(path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous stat(2). The callback gets two arguments `(err, stats)` where
|
2011-03-27 22:37:23 +08:00
|
|
|
`stats` is a [`fs.Stats`](#fs.Stats) object. It looks like this:
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2010-11-18 10:25:14 +08:00
|
|
|
{ dev: 2049,
|
|
|
|
ino: 305352,
|
|
|
|
mode: 16877,
|
|
|
|
nlink: 12,
|
|
|
|
uid: 1000,
|
|
|
|
gid: 1000,
|
|
|
|
rdev: 0,
|
|
|
|
size: 4096,
|
|
|
|
blksize: 4096,
|
|
|
|
blocks: 8,
|
|
|
|
atime: '2009-06-29T11:11:55Z',
|
|
|
|
mtime: '2009-06-29T11:11:40Z',
|
|
|
|
ctime: '2009-06-29T11:11:40Z' }
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2011-03-27 22:37:23 +08:00
|
|
|
See the [fs.Stats](#fs.Stats) section below for more information.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.lstat(path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous lstat(2). The callback gets two arguments `(err, stats)` where
|
|
|
|
`stats` is a `fs.Stats` object. lstat() is identical to stat(), except that if
|
|
|
|
path is a symbolic link, then the link itself is stat-ed, not the file that it
|
2010-11-18 00:04:21 +08:00
|
|
|
refers to.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.fstat(fd, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous fstat(2). The callback gets two arguments `(err, stats)` where
|
2010-11-18 00:04:21 +08:00
|
|
|
`stats` is a `fs.Stats` object.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.statSync(path)
|
|
|
|
|
|
|
|
Synchronous stat(2). Returns an instance of `fs.Stats`.
|
|
|
|
|
|
|
|
### fs.lstatSync(path)
|
|
|
|
|
|
|
|
Synchronous lstat(2). Returns an instance of `fs.Stats`.
|
|
|
|
|
|
|
|
### fs.fstatSync(fd)
|
|
|
|
|
|
|
|
Synchronous fstat(2). Returns an instance of `fs.Stats`.
|
|
|
|
|
|
|
|
### fs.link(srcpath, dstpath, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous link(2). No arguments other than a possible exception are given to
|
2010-11-18 00:04:21 +08:00
|
|
|
the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2011-01-28 06:20:50 +08:00
|
|
|
### fs.linkSync(srcpath, dstpath)
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
Synchronous link(2).
|
|
|
|
|
|
|
|
### fs.symlink(linkdata, path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous symlink(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.symlinkSync(linkdata, path)
|
|
|
|
|
|
|
|
Synchronous symlink(2).
|
|
|
|
|
|
|
|
### fs.readlink(path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous readlink(2). The callback gets two arguments `(err,
|
|
|
|
resolvedPath)`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.readlinkSync(path)
|
|
|
|
|
|
|
|
Synchronous readlink(2). Returns the resolved path.
|
|
|
|
|
|
|
|
### fs.realpath(path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous realpath(2). The callback gets two arguments `(err,
|
2010-11-18 00:04:21 +08:00
|
|
|
resolvedPath)`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.realpathSync(path)
|
|
|
|
|
|
|
|
Synchronous realpath(2). Returns the resolved path.
|
|
|
|
|
|
|
|
### fs.unlink(path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous unlink(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.unlinkSync(path)
|
|
|
|
|
|
|
|
Synchronous unlink(2).
|
|
|
|
|
|
|
|
### fs.rmdir(path, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous rmdir(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.rmdirSync(path)
|
|
|
|
|
|
|
|
Synchronous rmdir(2).
|
|
|
|
|
|
|
|
### fs.mkdir(path, mode, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous mkdir(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.mkdirSync(path, mode)
|
|
|
|
|
|
|
|
Synchronous mkdir(2).
|
|
|
|
|
|
|
|
### fs.readdir(path, [callback])
|
|
|
|
|
|
|
|
Asynchronous readdir(3). Reads the contents of a directory.
|
|
|
|
The callback gets two arguments `(err, files)` where `files` is an array of
|
|
|
|
the names of the files in the directory excluding `'.'` and `'..'`.
|
|
|
|
|
|
|
|
### fs.readdirSync(path)
|
|
|
|
|
|
|
|
Synchronous readdir(3). Returns an array of filenames excluding `'.'` and
|
|
|
|
`'..'`.
|
|
|
|
|
|
|
|
### fs.close(fd, [callback])
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Asynchronous close(2). No arguments other than a possible exception are given
|
2010-11-18 00:04:21 +08:00
|
|
|
to the completion callback.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.closeSync(fd)
|
|
|
|
|
|
|
|
Synchronous close(2).
|
|
|
|
|
2011-02-15 21:41:17 +08:00
|
|
|
### fs.open(path, flags, [mode], [callback])
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
Asynchronous file open. See open(2). Flags can be 'r', 'r+', 'w', 'w+', 'a',
|
2011-02-15 21:41:17 +08:00
|
|
|
or 'a+'. `mode` defaults to 0666. The callback gets two arguments `(err, fd)`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2011-02-15 21:41:17 +08:00
|
|
|
### fs.openSync(path, flags, [mode])
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Synchronous open(2).
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2010-10-29 18:38:13 +08:00
|
|
|
### fs.utimes(path, atime, mtime, callback)
|
|
|
|
### fs.utimesSync(path, atime, mtime)
|
|
|
|
|
|
|
|
Change file timestamps.
|
|
|
|
|
|
|
|
### fs.futimes(path, atime, mtime, callback)
|
|
|
|
### fs.futimesSync(path, atime, mtime)
|
|
|
|
|
|
|
|
Change file timestamps with the difference that if filename refers to a
|
|
|
|
symbolic link, then the link is not dereferenced.
|
|
|
|
|
|
|
|
|
2010-10-28 20:18:16 +08:00
|
|
|
### fs.write(fd, buffer, offset, length, position, [callback])
|
|
|
|
|
|
|
|
Write `buffer` to the file specified by `fd`.
|
|
|
|
|
|
|
|
`offset` and `length` determine the part of the buffer to be written.
|
|
|
|
|
|
|
|
`position` refers to the offset from the beginning of the file where this data
|
|
|
|
should be written. If `position` is `null`, the data will be written at the
|
|
|
|
current position.
|
|
|
|
See pwrite(2).
|
|
|
|
|
2011-03-29 15:53:32 +08:00
|
|
|
The callback will be given three arguments `(err, written, buffer)` where `written`
|
|
|
|
specifies how many _bytes_ were written into `buffer`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2011-03-19 22:27:25 +08:00
|
|
|
Note that it is unsafe to use `fs.write` multiple times on the same file
|
|
|
|
without waiting for the callback. For this scenario,
|
|
|
|
`fs.createWriteStream` is strongly recommended.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.writeSync(fd, buffer, offset, length, position)
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Synchronous version of buffer-based `fs.write()`. Returns the number of bytes
|
2010-11-18 00:04:21 +08:00
|
|
|
written.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.writeSync(fd, str, position, encoding='utf8')
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Synchronous version of string-based `fs.write()`. Returns the number of bytes
|
2010-11-18 00:04:21 +08:00
|
|
|
written.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.read(fd, buffer, offset, length, position, [callback])
|
|
|
|
|
|
|
|
Read data from the file specified by `fd`.
|
|
|
|
|
|
|
|
`buffer` is the buffer that the data will be written to.
|
|
|
|
|
|
|
|
`offset` is offset within the buffer where writing will start.
|
|
|
|
|
|
|
|
`length` is an integer specifying the number of bytes to read.
|
|
|
|
|
|
|
|
`position` is an integer specifying where to begin reading from in the file.
|
|
|
|
If `position` is `null`, data will be read from the current file position.
|
|
|
|
|
2011-03-29 15:53:32 +08:00
|
|
|
The callback is given the three arguments, `(err, bytesRead, buffer)`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.readSync(fd, buffer, offset, length, position)
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Synchronous version of buffer-based `fs.read`. Returns the number of
|
2010-11-18 00:04:21 +08:00
|
|
|
`bytesRead`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.readSync(fd, length, position, encoding)
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
Synchronous version of string-based `fs.read`. Returns the number of
|
2010-11-18 00:04:21 +08:00
|
|
|
`bytesRead`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
### fs.readFile(filename, [encoding], [callback])
|
|
|
|
|
|
|
|
Asynchronously reads the entire contents of a file. Example:
|
|
|
|
|
|
|
|
fs.readFile('/etc/passwd', function (err, data) {
|
|
|
|
if (err) throw err;
|
|
|
|
console.log(data);
|
|
|
|
});
|
|
|
|
|
|
|
|
The callback is passed two arguments `(err, data)`, where `data` is the
|
|
|
|
contents of the file.
|
|
|
|
|
|
|
|
If no encoding is specified, then the raw buffer is returned.
|
|
|
|
|
|
|
|
|
|
|
|
### fs.readFileSync(filename, [encoding])
|
|
|
|
|
|
|
|
Synchronous version of `fs.readFile`. Returns the contents of the `filename`.
|
|
|
|
|
|
|
|
If `encoding` is specified then this function returns a string. Otherwise it
|
|
|
|
returns a buffer.
|
|
|
|
|
|
|
|
|
|
|
|
### fs.writeFile(filename, data, encoding='utf8', [callback])
|
|
|
|
|
2011-03-19 22:22:50 +08:00
|
|
|
Asynchronously writes data to a file, replacing the file if it already exists.
|
|
|
|
`data` can be a string or a buffer.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
fs.writeFile('message.txt', 'Hello Node', function (err) {
|
|
|
|
if (err) throw err;
|
|
|
|
console.log('It\'s saved!');
|
|
|
|
});
|
|
|
|
|
|
|
|
### fs.writeFileSync(filename, data, encoding='utf8')
|
|
|
|
|
|
|
|
The synchronous version of `fs.writeFile`.
|
|
|
|
|
|
|
|
### fs.watchFile(filename, [options], listener)
|
|
|
|
|
|
|
|
Watch for changes on `filename`. The callback `listener` will be called each
|
2011-02-08 05:11:03 +08:00
|
|
|
time the file is accessed.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
The second argument is optional. The `options` if provided should be an object
|
|
|
|
containing two members a boolean, `persistent`, and `interval`, a polling
|
2011-02-08 05:11:03 +08:00
|
|
|
value in milliseconds. The default is `{ persistent: true, interval: 0 }`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
The `listener` gets two arguments the current stat object and the previous
|
|
|
|
stat object:
|
|
|
|
|
|
|
|
fs.watchFile(f, function (curr, prev) {
|
|
|
|
console.log('the current mtime is: ' + curr.mtime);
|
|
|
|
console.log('the previous mtime was: ' + prev.mtime);
|
|
|
|
});
|
|
|
|
|
2010-11-22 06:22:34 +08:00
|
|
|
These stat objects are instances of `fs.Stat`.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
2011-02-08 05:11:03 +08:00
|
|
|
If you want to be notified when the file was modified, not just accessed
|
|
|
|
you need to compare `curr.mtime` and `prev.mtime.
|
|
|
|
|
|
|
|
|
2010-10-28 20:18:16 +08:00
|
|
|
### fs.unwatchFile(filename)
|
|
|
|
|
|
|
|
Stop watching for changes on `filename`.
|
|
|
|
|
|
|
|
## fs.Stats
|
|
|
|
|
|
|
|
Objects returned from `fs.stat()` and `fs.lstat()` are of this type.
|
|
|
|
|
|
|
|
- `stats.isFile()`
|
|
|
|
- `stats.isDirectory()`
|
|
|
|
- `stats.isBlockDevice()`
|
|
|
|
- `stats.isCharacterDevice()`
|
|
|
|
- `stats.isSymbolicLink()` (only valid with `fs.lstat()`)
|
|
|
|
- `stats.isFIFO()`
|
|
|
|
- `stats.isSocket()`
|
|
|
|
|
|
|
|
|
|
|
|
## fs.ReadStream
|
|
|
|
|
|
|
|
`ReadStream` is a `Readable Stream`.
|
|
|
|
|
|
|
|
### fs.createReadStream(path, [options])
|
|
|
|
|
|
|
|
Returns a new ReadStream object (See `Readable Stream`).
|
|
|
|
|
|
|
|
`options` is an object with the following defaults:
|
|
|
|
|
2010-11-18 10:25:14 +08:00
|
|
|
{ flags: 'r',
|
|
|
|
encoding: null,
|
2011-02-20 20:41:14 +08:00
|
|
|
fd: null,
|
2010-11-18 10:25:14 +08:00
|
|
|
mode: 0666,
|
2011-02-20 20:45:25 +08:00
|
|
|
bufferSize: 64 * 1024
|
|
|
|
}
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
`options` can include `start` and `end` values to read a range of bytes from
|
|
|
|
the file instead of the entire file. Both `start` and `end` are inclusive and
|
2011-04-06 05:37:40 +08:00
|
|
|
start at 0.
|
2010-10-28 20:18:16 +08:00
|
|
|
|
|
|
|
An example to read the last 10 bytes of a file which is 100 bytes long:
|
|
|
|
|
|
|
|
fs.createReadStream('sample.txt', {start: 90, end: 99});
|
|
|
|
|
|
|
|
|
|
|
|
## fs.WriteStream
|
|
|
|
|
|
|
|
`WriteStream` is a `Writable Stream`.
|
|
|
|
|
|
|
|
### Event: 'open'
|
|
|
|
|
|
|
|
`function (fd) { }`
|
|
|
|
|
|
|
|
`fd` is the file descriptor used by the WriteStream.
|
|
|
|
|
2011-04-15 04:45:32 +08:00
|
|
|
### file.bytesWritten
|
|
|
|
|
|
|
|
The number of bytes written so far. Does not include data that is still queued
|
|
|
|
for writing.
|
|
|
|
|
2010-10-28 20:18:16 +08:00
|
|
|
### fs.createWriteStream(path, [options])
|
|
|
|
|
|
|
|
Returns a new WriteStream object (See `Writable Stream`).
|
|
|
|
|
|
|
|
`options` is an object with the following defaults:
|
|
|
|
|
2010-11-18 10:25:14 +08:00
|
|
|
{ flags: 'w',
|
|
|
|
encoding: null,
|
|
|
|
mode: 0666 }
|