134 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
		
		
			
		
	
	
			134 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
|  | # graceful-fs
 | ||
|  | 
 | ||
|  | graceful-fs functions as a drop-in replacement for the fs module, | ||
|  | making various improvements. | ||
|  | 
 | ||
|  | The improvements are meant to normalize behavior across different | ||
|  | platforms and environments, and to make filesystem access more | ||
|  | resilient to errors. | ||
|  | 
 | ||
|  | ## Improvements over [fs module](https://nodejs.org/api/fs.html)
 | ||
|  | 
 | ||
|  | * Queues up `open` and `readdir` calls, and retries them once | ||
|  |   something closes if there is an EMFILE error from too many file | ||
|  |   descriptors. | ||
|  | * fixes `lchmod` for Node versions prior to 0.6.2. | ||
|  | * implements `fs.lutimes` if possible. Otherwise it becomes a noop. | ||
|  | * ignores `EINVAL` and `EPERM` errors in `chown`, `fchown` or | ||
|  |   `lchown` if the user isn't root. | ||
|  | * makes `lchmod` and `lchown` become noops, if not available. | ||
|  | * retries reading a file if `read` results in EAGAIN error. | ||
|  | 
 | ||
|  | On Windows, it retries renaming a file for up to one second if `EACCESS` | ||
|  | or `EPERM` error occurs, likely because antivirus software has locked | ||
|  | the directory. | ||
|  | 
 | ||
|  | ## USAGE
 | ||
|  | 
 | ||
|  | ```javascript | ||
|  | // use just like fs | ||
|  | var fs = require('graceful-fs') | ||
|  | 
 | ||
|  | // now go and do stuff with it... | ||
|  | fs.readFileSync('some-file-or-whatever') | ||
|  | ``` | ||
|  | 
 | ||
|  | ## Global Patching
 | ||
|  | 
 | ||
|  | If you want to patch the global fs module (or any other fs-like | ||
|  | module) you can do this: | ||
|  | 
 | ||
|  | ```javascript | ||
|  | // Make sure to read the caveat below. | ||
|  | var realFs = require('fs') | ||
|  | var gracefulFs = require('graceful-fs') | ||
|  | gracefulFs.gracefulify(realFs) | ||
|  | ``` | ||
|  | 
 | ||
|  | This should only ever be done at the top-level application layer, in | ||
|  | order to delay on EMFILE errors from any fs-using dependencies.  You | ||
|  | should **not** do this in a library, because it can cause unexpected | ||
|  | delays in other parts of the program. | ||
|  | 
 | ||
|  | ## Changes
 | ||
|  | 
 | ||
|  | This module is fairly stable at this point, and used by a lot of | ||
|  | things.  That being said, because it implements a subtle behavior | ||
|  | change in a core part of the node API, even modest changes can be | ||
|  | extremely breaking, and the versioning is thus biased towards | ||
|  | bumping the major when in doubt. | ||
|  | 
 | ||
|  | The main change between major versions has been switching between | ||
|  | providing a fully-patched `fs` module vs monkey-patching the node core | ||
|  | builtin, and the approach by which a non-monkey-patched `fs` was | ||
|  | created. | ||
|  | 
 | ||
|  | The goal is to trade `EMFILE` errors for slower fs operations.  So, if | ||
|  | you try to open a zillion files, rather than crashing, `open` | ||
|  | operations will be queued up and wait for something else to `close`. | ||
|  | 
 | ||
|  | There are advantages to each approach.  Monkey-patching the fs means | ||
|  | that no `EMFILE` errors can possibly occur anywhere in your | ||
|  | application, because everything is using the same core `fs` module, | ||
|  | which is patched.  However, it can also obviously cause undesirable | ||
|  | side-effects, especially if the module is loaded multiple times. | ||
|  | 
 | ||
|  | Implementing a separate-but-identical patched `fs` module is more | ||
|  | surgical (and doesn't run the risk of patching multiple times), but | ||
|  | also imposes the challenge of keeping in sync with the core module. | ||
|  | 
 | ||
|  | The current approach loads the `fs` module, and then creates a | ||
|  | lookalike object that has all the same methods, except a few that are | ||
|  | patched.  It is safe to use in all versions of Node from 0.8 through | ||
|  | 7.0. | ||
|  | 
 | ||
|  | ### v4
 | ||
|  | 
 | ||
|  | * Do not monkey-patch the fs module.  This module may now be used as a | ||
|  |   drop-in dep, and users can opt into monkey-patching the fs builtin | ||
|  |   if their app requires it. | ||
|  | 
 | ||
|  | ### v3
 | ||
|  | 
 | ||
|  | * Monkey-patch fs, because the eval approach no longer works on recent | ||
|  |   node. | ||
|  | * fixed possible type-error throw if rename fails on windows | ||
|  | * verify that we *never* get EMFILE errors | ||
|  | * Ignore ENOSYS from chmod/chown | ||
|  | * clarify that graceful-fs must be used as a drop-in | ||
|  | 
 | ||
|  | ### v2.1.0
 | ||
|  | 
 | ||
|  | * Use eval rather than monkey-patching fs. | ||
|  | * readdir: Always sort the results | ||
|  | * win32: requeue a file if error has an OK status | ||
|  | 
 | ||
|  | ### v2.0
 | ||
|  | 
 | ||
|  | * A return to monkey patching | ||
|  | * wrap process.cwd | ||
|  | 
 | ||
|  | ### v1.1
 | ||
|  | 
 | ||
|  | * wrap readFile | ||
|  | * Wrap fs.writeFile. | ||
|  | * readdir protection | ||
|  | * Don't clobber the fs builtin | ||
|  | * Handle fs.read EAGAIN errors by trying again | ||
|  | * Expose the curOpen counter | ||
|  | * No-op lchown/lchmod if not implemented | ||
|  | * fs.rename patch only for win32 | ||
|  | * Patch fs.rename to handle AV software on Windows | ||
|  | * Close #4 Chown should not fail on einval or eperm if non-root | ||
|  | * Fix isaacs/fstream#1 Only wrap fs one time | ||
|  | * Fix #3 Start at 1024 max files, then back off on EMFILE | ||
|  | * lutimes that doens't blow up on Linux | ||
|  | * A full on-rewrite using a queue instead of just swallowing the EMFILE error | ||
|  | * Wrap Read/Write streams as well | ||
|  | 
 | ||
|  | ### 1.0
 | ||
|  | 
 | ||
|  | * Update engines for node 0.6 | ||
|  | * Be lstat-graceful on Windows | ||
|  | * first |