mirror of
https://github.com/Jermolene/TiddlyWiki5
synced 2024-12-02 14:29:55 +00:00
70a120d4a6
Now we get decent visual indication of sync errors, for instance. Still work to do to coalesce alerts so that the screen doesn’t fill up with them after an interval. And probably we should add a button to clear all alerts.
229 lines
6.7 KiB
JavaScript
229 lines
6.7 KiB
JavaScript
/*\
|
|
title: $:/plugins/tiddlywiki/filesystem/filesystemadaptor.js
|
|
type: application/javascript
|
|
module-type: syncadaptor
|
|
|
|
A sync adaptor module for synchronising with the local filesystem via node.js APIs
|
|
|
|
\*/
|
|
(function(){
|
|
|
|
/*jslint node: true, browser: true */
|
|
/*global $tw: false */
|
|
"use strict";
|
|
|
|
// Get a reference to the file system
|
|
var fs = !$tw.browser ? require("fs") : null;
|
|
|
|
|
|
function FileSystemAdaptor(syncer) {
|
|
var self = this;
|
|
this.syncer = syncer;
|
|
this.watchers = {};
|
|
this.pending = {};
|
|
this.logger = new $tw.utils.Logger("FileSystem");
|
|
this.setwatcher = function(filename, title) {
|
|
return undefined;
|
|
return this.watchers[filename] = this.watchers[filename] ||
|
|
fs.watch(filename, {persistent: false}, function(e) {
|
|
self.logger.log("Error:",e,filename);
|
|
if(e === "change") {
|
|
var tiddlers = $tw.loadTiddlersFromFile(filename).tiddlers;
|
|
for(var t in tiddlers) {
|
|
if(tiddlers[t].title) {
|
|
$tw.wiki.addTiddler(tiddlers[t]);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
for(var f in $tw.boot.files) {
|
|
var fileInfo = $tw.boot.files[f];
|
|
this.setwatcher(fileInfo.filepath, f);
|
|
}
|
|
// Create the <wiki>/tiddlers folder if it doesn't exist
|
|
// TODO: we should create the path recursively
|
|
if(!fs.existsSync($tw.boot.wikiTiddlersPath)) {
|
|
fs.mkdirSync($tw.boot.wikiTiddlersPath);
|
|
}
|
|
}
|
|
|
|
FileSystemAdaptor.prototype.getTiddlerInfo = function(tiddler) {
|
|
return {};
|
|
};
|
|
|
|
$tw.config.typeInfo = {
|
|
"text/vnd.tiddlywiki": {
|
|
fileType: "application/x-tiddler",
|
|
extension: ".tid"
|
|
},
|
|
"image/jpeg" : {
|
|
hasMetaFile: true
|
|
}
|
|
};
|
|
|
|
$tw.config.typeTemplates = {
|
|
"application/x-tiddler": "$:/core/templates/tid-tiddler"
|
|
};
|
|
|
|
FileSystemAdaptor.prototype.getTiddlerFileInfo = function(tiddler,callback) {
|
|
// See if we've already got information about this file
|
|
var self = this,
|
|
title = tiddler.fields.title,
|
|
fileInfo = $tw.boot.files[title];
|
|
// Get information about how to save tiddlers of this type
|
|
var type = tiddler.fields.type || "text/vnd.tiddlywiki",
|
|
typeInfo = $tw.config.typeInfo[type];
|
|
if(!typeInfo) {
|
|
typeInfo = $tw.config.typeInfo["text/vnd.tiddlywiki"];
|
|
}
|
|
var extension = typeInfo.extension || "";
|
|
if(!fileInfo) {
|
|
// If not, we'll need to generate it
|
|
// Start by getting a list of the existing files in the directory
|
|
fs.readdir($tw.boot.wikiTiddlersPath,function(err,files) {
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
// Assemble the new fileInfo
|
|
fileInfo = {};
|
|
fileInfo.filepath = $tw.boot.wikiTiddlersPath + "/" + self.generateTiddlerFilename(title,extension,files);
|
|
fileInfo.type = typeInfo.fileType || tiddler.fields.type;
|
|
fileInfo.hasMetaFile = typeInfo.hasMetaFile;
|
|
// Save the newly created fileInfo
|
|
$tw.boot.files[title] = fileInfo;
|
|
self.pending[fileInfo.filepath] = title;
|
|
// Pass it to the callback
|
|
callback(null,fileInfo);
|
|
});
|
|
} else {
|
|
// Otherwise just invoke the callback
|
|
callback(null,fileInfo);
|
|
}
|
|
};
|
|
|
|
/*
|
|
Given a tiddler title and an array of existing filenames, generate a new legal filename for the title, case insensitively avoiding the array of existing filenames
|
|
*/
|
|
FileSystemAdaptor.prototype.generateTiddlerFilename = function(title,extension,existingFilenames) {
|
|
// First remove any of the characters that are illegal in Windows filenames
|
|
var baseFilename = title.replace(/\<|\>|\:|\"|\/|\\|\||\?|\*|\^/g,"_");
|
|
// Truncate the filename if it is too long
|
|
if(baseFilename.length > 200) {
|
|
baseFilename = baseFilename.substr(0,200) + extension;
|
|
}
|
|
// Start with the base filename plus the extension
|
|
var filename = baseFilename + extension,
|
|
count = 1;
|
|
// Add a discriminator if we're clashing with an existing filename
|
|
while(existingFilenames.indexOf(filename) !== -1) {
|
|
filename = baseFilename + " " + (count++) + extension;
|
|
}
|
|
return filename;
|
|
};
|
|
|
|
/*
|
|
Save a tiddler and invoke the callback with (err,adaptorInfo,revision)
|
|
*/
|
|
FileSystemAdaptor.prototype.saveTiddler = function(tiddler,callback) {
|
|
var self = this;
|
|
this.getTiddlerFileInfo(tiddler,function(err,fileInfo) {
|
|
var template, content, encoding;
|
|
function _finish() {
|
|
if(self.pending[fileInfo.filepath]) {
|
|
self.setwatcher(fileInfo.filepath, tiddler.fields.title);
|
|
delete self.pending[fileInfo.filepath];
|
|
}
|
|
callback(null, {}, 0);
|
|
}
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
if(self.watchers[fileInfo.filepath]) {
|
|
self.watchers[fileInfo.filepath].close();
|
|
delete self.watchers[fileInfo.filepath];
|
|
self.pending[fileInfo.filepath] = tiddler.fields.title;
|
|
}
|
|
if(fileInfo.hasMetaFile) {
|
|
// Save the tiddler as a separate body and meta file
|
|
var typeInfo = $tw.config.contentTypeInfo[fileInfo.type];
|
|
fs.writeFile(fileInfo.filepath,tiddler.fields.text,{encoding: typeInfo.encoding},function(err) {
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
content = $tw.wiki.renderTiddler("text/plain","$:/core/templates/tiddler-metadata",{variables: {currentTiddler: tiddler.fields.title}});
|
|
fs.writeFile(fileInfo.filepath + ".meta",content,{encoding: "utf8"},function (err) {
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
self.logger.log("Saved file",fileInfo.filepath);
|
|
_finish();
|
|
});
|
|
});
|
|
} else {
|
|
// Save the tiddler as a self contained templated file
|
|
template = $tw.config.typeTemplates[fileInfo.type];
|
|
content = $tw.wiki.renderTiddler("text/plain",template,{variables: {currentTiddler: tiddler.fields.title}});
|
|
fs.writeFile(fileInfo.filepath,content,{encoding: "utf8"},function (err) {
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
self.logger.log("Saved file",fileInfo.filepath);
|
|
_finish();
|
|
});
|
|
}
|
|
});
|
|
};
|
|
|
|
/*
|
|
Load a tiddler and invoke the callback with (err,tiddlerFields)
|
|
|
|
We don't need to implement loading for the file system adaptor, because all the tiddler files will have been loaded during the boot process.
|
|
*/
|
|
FileSystemAdaptor.prototype.loadTiddler = function(title,callback) {
|
|
callback(null,null);
|
|
};
|
|
|
|
/*
|
|
Delete a tiddler and invoke the callback with (err)
|
|
*/
|
|
FileSystemAdaptor.prototype.deleteTiddler = function(title,callback) {
|
|
var self = this,
|
|
fileInfo = $tw.boot.files[title];
|
|
// Only delete the tiddler if we have writable information for the file
|
|
if(fileInfo) {
|
|
if(this.watchers[fileInfo.filepath]) {
|
|
this.watchers[fileInfo.filepath].close();
|
|
delete this.watchers[fileInfo.filepath];
|
|
}
|
|
delete this.pending[fileInfo.filepath];
|
|
// Delete the file
|
|
fs.unlink(fileInfo.filepath,function(err) {
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
self.logger.log("Deleted file",fileInfo.filepath);
|
|
// Delete the metafile if present
|
|
if(fileInfo.hasMetaFile) {
|
|
fs.unlink(fileInfo.filepath + ".meta",function(err) {
|
|
if(err) {
|
|
return callback(err);
|
|
}
|
|
callback(null);
|
|
});
|
|
} else {
|
|
callback(null);
|
|
}
|
|
});
|
|
} else {
|
|
callback(null);
|
|
}
|
|
};
|
|
|
|
if(fs) {
|
|
exports.adaptorClass = FileSystemAdaptor;
|
|
}
|
|
|
|
})();
|