-
Notifications
You must be signed in to change notification settings - Fork 0
File System
Status: Proposal
Email comments to: gruehle@adobe.com
The file system APIs in Brackets are a bit chaotic, and usage is inconsistent. Here are just a few issues:
- No centralized file system model. This makes it difficult to add file watchers, and update all file references in the app after operations like rename.
- Inefficient. We constantly hit the disk to read timestamps, content, etc.
- Two APIs for file i/o:
brackets.fs
andNativeFileSystem
- Inconsistent use of
fullPath
vs.FileEntry
- Incorrect creation of
FileEntry
objects (these should never be directly instantiated) - No way to use alternate storage--Dropbox, Google Drive, etc.
I'm pretty sure there are many more...
The high-level goals of the new file system are:
- Clean and consistent API.
- High Performance. Cache wherever possible.
- Ability to swap out the low-level I/O routines.
Here is a block diagram of the major parts:
Clients only interact with the blue boxes. The green boxes represent low-level file i/o implementations, which can be added as extensions. The red box replaces the old FileIndexManager
functionality, and does not have a public API (indexed files are accessed through FileSystem
).
There are a few basic rules for using the new file system.
- Persist full pathnames, but convert into
File
andDirectory
objects for in-memory use. CreatingFile
andDirectory
objects is cheap so there is no need to worry about performance. - There is a forced 1:1 relationship between
File
andDirectory
objects and their representation on disk. If two pieces of code ask theFileSystem
for the same file path, they will both be returned the sameFile
object. - All asyc operations return promises. If the operation succeeds, the promise is resolved. If there was an error, the promise is rejected with the error code.
- To get the contents of a
Directory
, callFileSystem.getDirectoryContents()
. There is noreaddir()
onDirectory
. - Listen for
"change"
events onFileSystem
to be notified if a file or directory changes.
A prototype implementation can be found in the glenn/file-system
branch.
Most of the basic functionality works. Unit tests are completely broken (this is the next thing I want to work on), and many extensions are broken.
File watchers require node 0.10. You will need to update the shell in order to get proper file watching (directory watching works fine in 0.8).
##API##
The main module is FileSystem
. This is the public API for getting files and directories, showing open/save dialogs, and getting the list of all the files in the project.
###FileSystemEntry###
This is an abstract representation of a FileSystem entry, and the base class for the File
and Directory
classes. FileSystemEntry objects are never created directly by client code. Use FileSystem.getFileForPath()
, FileSystem.getDirectoryForPath()
, or FileSystem.getDirectoryContents()
to create the entry.
/src/filesystem/FileSystemEntry.js
###File###
This class represents a file on disk (this could be a local disk or cloud storage). This is a subclass of FileSystemEntry
.
###Directory###
This class represents a directory on disk (this could be a local disk or cloud storage). This is a subclass of FileSystemEntry
.
##Performance## The main performance gains come from caching. The stats and contents of files and directories are cached in memory. This has a huge impact on i/o-intensive operations like "find in files", and generally speeds up many other parts of the system.
##File Watchers##
FileSystem
dispatches a "change"
event whenever a file or directory changes on disk. For now, the prototype refreshes the entire file tree whenever a directory changes. This is overkill, but was easy to implement. Ideally, only the directory that changed will be updated.
##Next Steps##
- Get unit tests working.
- Add a third low-level implementation (Google Drive, SkyDrive, etc.) to vet the API for completeness.
- Write a porting guide.
- Consider adding scaffolding APIs to ease the transition. For example, the file system entry uses
getPath()
to enforce read-only access. Most existing code that usesfullPath
just needs to change togetPath()
, so we could add afullPath
getter that emits a deprecation warning. - Address TODOs in code.
- Testing and bug fixing.