hacked stylesheets, fixed base_url
This commit is contained in:
parent
c995197c0b
commit
330eb5346c
|
@ -38,5 +38,8 @@
|
||||||
"forEachAsync": "~5.0.5",
|
"forEachAsync": "~5.0.5",
|
||||||
"node-uuid": "~1.4.2",
|
"node-uuid": "~1.4.2",
|
||||||
"markdown-it": "~3.0.2"
|
"markdown-it": "~3.0.2"
|
||||||
|
},
|
||||||
|
"resolutions": {
|
||||||
|
"bluebird": "~2.6.2"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
205
deardesi.js
205
deardesi.js
|
@ -31,6 +31,10 @@
|
||||||
, 12: 'December'
|
, 12: 'December'
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function firstCap(str) {
|
||||||
|
return str.replace(/^./, function ($1) { return $1.toUpperCase(); });
|
||||||
|
}
|
||||||
|
|
||||||
function pad(str) {
|
function pad(str) {
|
||||||
str = str.toString();
|
str = str.toString();
|
||||||
if (str.length < 2) {
|
if (str.length < 2) {
|
||||||
|
@ -170,7 +174,8 @@
|
||||||
themename = desi.config.themes.default;
|
themename = desi.config.themes.default;
|
||||||
}
|
}
|
||||||
if (!layout) {
|
if (!layout) {
|
||||||
layout = 'post.html';
|
// TODO make configurable
|
||||||
|
layout = 'posts.html';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -210,7 +215,7 @@
|
||||||
;
|
;
|
||||||
|
|
||||||
desi.urls = desi.config.urls = {};
|
desi.urls = desi.config.urls = {};
|
||||||
if (development) {
|
if (true || development) {
|
||||||
desi.urls.base_path = desi.config.development.base_path;
|
desi.urls.base_path = desi.config.development.base_path;
|
||||||
desi.urls.url = desi.config.development.url;
|
desi.urls.url = desi.config.development.url;
|
||||||
desi.urls.development_url = desi.config.development.url;
|
desi.urls.development_url = desi.config.development.url;
|
||||||
|
@ -230,10 +235,25 @@
|
||||||
droot = getDirty(cacheByPath, cacheBySha1, [desi.meta.root], dthemes);
|
droot = getDirty(cacheByPath, cacheBySha1, [desi.meta.root], dthemes);
|
||||||
dfiles = getDirty(cacheByPath, cacheBySha1, desi.meta.collections, dthemes);
|
dfiles = getDirty(cacheByPath, cacheBySha1, desi.meta.collections, dthemes);
|
||||||
|
|
||||||
|
/*
|
||||||
|
if (!droot.length) {
|
||||||
|
console.error("no root files to get");
|
||||||
|
}
|
||||||
|
if (!dfiles.length) {
|
||||||
|
console.error("no content files to get");
|
||||||
|
}
|
||||||
|
if (!dthemes.length) {
|
||||||
|
console.error("no theme files to get");
|
||||||
|
}
|
||||||
|
if (!droot || !dfiles || !droot) {
|
||||||
|
throw new Error("didn't read files");
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
return PromiseA.all([
|
return PromiseA.all([
|
||||||
fsapi.getContents(Object.keys(droot))
|
Object.keys(droot).length ? fsapi.getContents(Object.keys(droot)) : PromiseA.resolve([])
|
||||||
, fsapi.getContents(Object.keys(dfiles))
|
, Object.keys(dfiles).length ? fsapi.getContents(Object.keys(dfiles)) : PromiseA.resolve([])
|
||||||
, fsapi.getContents(Object.keys(dthemes))
|
, Object.keys(dthemes).length ? fsapi.getContents(Object.keys(dthemes)) : PromiseA.resolve([])
|
||||||
]).then(function (arr) {
|
]).then(function (arr) {
|
||||||
// TODO XXX display errors in html
|
// TODO XXX display errors in html
|
||||||
function noErrors(o) {
|
function noErrors(o) {
|
||||||
|
@ -260,6 +280,8 @@
|
||||||
console.info('getting config, data, caches...');
|
console.info('getting config, data, caches...');
|
||||||
|
|
||||||
return PromiseA.all([fsapi.getConfig(), fsapi.getData(), fsapi.getCache(), fsapi.getPartials()]).then(function (arr) {
|
return PromiseA.all([fsapi.getConfig(), fsapi.getData(), fsapi.getCache(), fsapi.getPartials()]).then(function (arr) {
|
||||||
|
console.info('config');
|
||||||
|
console.log(arr[0]);
|
||||||
var config = arr[0]
|
var config = arr[0]
|
||||||
, data = arr[1]
|
, data = arr[1]
|
||||||
, cache = arr[2]
|
, cache = arr[2]
|
||||||
|
@ -268,10 +290,16 @@
|
||||||
, themenames = Object.keys(config.themes)
|
, themenames = Object.keys(config.themes)
|
||||||
.filter(function (k) { return 'default' !== k; })
|
.filter(function (k) { return 'default' !== k; })
|
||||||
//.map(function (n) { return path.join(n, 'layouts'); })
|
//.map(function (n) { return path.join(n, 'layouts'); })
|
||||||
|
, assetnames = Object.keys(config.assets)
|
||||||
;
|
;
|
||||||
|
|
||||||
console.info('loaded config, data, caches.');
|
console.info('loaded config, data, caches, partials');
|
||||||
console.log(arr);
|
console.log({
|
||||||
|
config: arr[0]
|
||||||
|
, data: arr[1]
|
||||||
|
, cache: arr[2]
|
||||||
|
, partials: arr[3]
|
||||||
|
});
|
||||||
console.info('last update: ' + (cache.lastUpdate && new Date(cache.lastUpdate) || 'never'));
|
console.info('last update: ' + (cache.lastUpdate && new Date(cache.lastUpdate) || 'never'));
|
||||||
|
|
||||||
// TODO make document configurability
|
// TODO make document configurability
|
||||||
|
@ -295,7 +323,21 @@
|
||||||
, extensions: ['md', 'markdown', 'htm', 'html', 'jade']
|
, extensions: ['md', 'markdown', 'htm', 'html', 'jade']
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
, fsapi.getMeta(
|
||||||
|
assetnames
|
||||||
|
, { dotfiles: false
|
||||||
|
//, extensions: ['md', 'markdown', 'htm', 'html', 'jade', 'css', 'js', 'yml']
|
||||||
|
}
|
||||||
|
)
|
||||||
]).then(function (things) {
|
]).then(function (things) {
|
||||||
|
console.info('loaded theme meta, root meta, collection meta');
|
||||||
|
console.log({
|
||||||
|
theme: things[0]
|
||||||
|
, root: things[1]
|
||||||
|
, collection: things[2]
|
||||||
|
, asset: things[3]
|
||||||
|
});
|
||||||
|
|
||||||
function noErrors(map) {
|
function noErrors(map) {
|
||||||
Object.keys(map).forEach(function (path) {
|
Object.keys(map).forEach(function (path) {
|
||||||
map[path] = map[path].filter(function (m) {
|
map[path] = map[path].filter(function (m) {
|
||||||
|
@ -316,11 +358,25 @@
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|
||||||
var themes = noErrors(things[0])
|
var themes = noErrors(things[0])
|
||||||
, root = noErrors(things[1])[config.rootdir]
|
, root = noErrors(things[1])[config.rootdir]
|
||||||
, collections = noErrors(things[2])
|
, collections = noErrors(things[2])
|
||||||
|
, assets = noErrors(things[3])
|
||||||
;
|
;
|
||||||
|
|
||||||
|
if (!themes[Object.keys(themes)[0]].length) {
|
||||||
|
console.error('Missing THEMES!');
|
||||||
|
throw new Error('It seems that your themes directory is missing');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!root.length) {
|
||||||
|
console.error('Missing ROOT!');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!collections[Object.keys(collections)[0]].length) {
|
||||||
|
console.error('Missing Collections!');
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
config: config
|
config: config
|
||||||
, data: data
|
, data: data
|
||||||
|
@ -329,10 +385,43 @@
|
||||||
themes: themes
|
themes: themes
|
||||||
, collections: collections
|
, collections: collections
|
||||||
, root: root
|
, root: root
|
||||||
|
, assets: assets
|
||||||
}
|
}
|
||||||
, partials: partials
|
, partials: partials
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
}).then(runDesi).then(function (desi) {
|
||||||
|
var files = {}
|
||||||
|
;
|
||||||
|
|
||||||
|
// copy assets -> easy!
|
||||||
|
// TODO check cache
|
||||||
|
Object.keys(desi.meta.assets).forEach(function (key) {
|
||||||
|
var assets = desi.meta.assets[key]
|
||||||
|
;
|
||||||
|
|
||||||
|
// TODO fix compiled_path + base_path
|
||||||
|
assets.forEach(function (asset) {
|
||||||
|
console.log(asset);
|
||||||
|
files[path.join(asset.relativePath, asset.name)] = path.join(desi.config.compiled_path, 'assets', asset.relativePath, asset.name);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return Object.keys(files).length && fsapi.copy(files).then(function (copied) {
|
||||||
|
if (copied.error) {
|
||||||
|
console.error(copied.error);
|
||||||
|
throw new Error(copied.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (copied.errors && copied.errors.length) {
|
||||||
|
console.error("Errors copying assets...");
|
||||||
|
copied.errors.forEach(function (err) {
|
||||||
|
console.error(err);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return desi;
|
||||||
|
}) || PromiseA.resolve(desi);
|
||||||
}).then(runDesi).then(function (desi) {
|
}).then(runDesi).then(function (desi) {
|
||||||
return readFrontmatter(desi.content.root.concat(desi.content.themes.concat(desi.content.collections))).then(function () {
|
return readFrontmatter(desi.content.root.concat(desi.content.themes.concat(desi.content.collections))).then(function () {
|
||||||
return desi;
|
return desi;
|
||||||
|
@ -343,26 +432,28 @@
|
||||||
|
|
||||||
desi.content.root.forEach(function (page) {
|
desi.content.root.forEach(function (page) {
|
||||||
var name = path.basename(page.path, path.extname(page.path))
|
var name = path.basename(page.path, path.extname(page.path))
|
||||||
|
, nindex
|
||||||
;
|
;
|
||||||
|
|
||||||
//if (-1 === desi.data.navigation.indexOf(name) && 'index' !== name)
|
//if (-1 === desi.data.navigation.indexOf(name) && 'index' !== name)
|
||||||
if (-1 === desi.data.navigation.indexOf(name)) {
|
nindex = desi.data.navigation.indexOf(name);
|
||||||
|
if (-1 === nindex) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
desi.navigation.push({
|
desi.navigation[nindex] = {
|
||||||
title: page.yml && page.yml.title || name.replace(/^./, function ($1) { return $1.toUpperCase(); })
|
title: page.yml && page.yml.title || firstCap(name)
|
||||||
, href: '/' + name
|
, href: desi.urls.base_path + '/' + name
|
||||||
, path: '/' + name
|
, path: desi.urls.base_path + '/' + name
|
||||||
, name: name
|
, name: name
|
||||||
, active: false // placeholder
|
, active: false // placeholder
|
||||||
});
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
desi.content.root.forEach(function (page) {
|
desi.content.root.forEach(function (page) {
|
||||||
page.yml = page.yml || {};
|
page.yml = page.yml || {};
|
||||||
page.yml.layout = page.yml.layout || 'default';
|
// TODO make default layout configurable
|
||||||
|
page.yml.layout = page.yml.layout || '_root';
|
||||||
|
|
||||||
if (!page.relativePath) {
|
if (!page.relativePath) {
|
||||||
page.relativePath = path.dirname(page.path);
|
page.relativePath = path.dirname(page.path);
|
||||||
|
@ -371,6 +462,11 @@
|
||||||
|
|
||||||
page.relativePath = page.relativePath.replace(desi.config.rootdir, '').replace(/^\//, '');
|
page.relativePath = page.relativePath.replace(desi.config.rootdir, '').replace(/^\//, '');
|
||||||
page.path = path.join(page.relativePath, page.name);
|
page.path = path.join(page.relativePath, page.name);
|
||||||
|
|
||||||
|
// TODO make bare root routes configurable
|
||||||
|
page.yml.permalink = page.yml.permalink || page.path.replace(/\.\w+$/, '');
|
||||||
|
|
||||||
|
page.yml.title = page.yml.title || firstCap(page.name.replace(/\.\w+$/, ''));
|
||||||
});
|
});
|
||||||
|
|
||||||
desi.content.collections = desi.content.collections.filter(function (article) {
|
desi.content.collections = desi.content.collections.filter(function (article) {
|
||||||
|
@ -395,13 +491,24 @@
|
||||||
|
|
||||||
// TODO read the config for this collection for how to create premalink
|
// TODO read the config for this collection for how to create premalink
|
||||||
if (!yml.permalink) {
|
if (!yml.permalink) {
|
||||||
|
if (page.name) {
|
||||||
|
page.htmlname = page.name.replace(/\.\w+$/, '.html');
|
||||||
|
}
|
||||||
page.path = page.path || path.join(page.relativePath, page.name);
|
page.path = page.path || path.join(page.relativePath, page.name);
|
||||||
|
page.htmlpath = page.path.replace(/\.\w+$/, '.html');
|
||||||
// TODO strip '_root' or whatever
|
// TODO strip '_root' or whatever
|
||||||
// strip .html, .md, .jade, etc
|
// strip .html, .md, .jade, etc
|
||||||
yml.permalink = path.join(desi.urls.base_path, path.basename(page.path, path.extname(page.path)));
|
yml.permalink = page.htmlpath;
|
||||||
|
console.info('1', yml.permalink);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!/\.html?$/.test(yml.permalink)) {
|
||||||
|
console.info(page.yml.permalink);
|
||||||
|
yml.permalink = path.join(yml.permalink, 'index.html');
|
||||||
|
}
|
||||||
|
|
||||||
//yml.permalinkBase = path.join(path.dirname(yml.permalink), path.basename(yml.permalink, path.extname(yml.permalink)));
|
//yml.permalinkBase = path.join(path.dirname(yml.permalink), path.basename(yml.permalink, path.extname(yml.permalink)));
|
||||||
yml.permalink = path.join(path.dirname(yml.permalink), path.basename(yml.permalink, path.extname(yml.permalink)));
|
//yml.permalink = path.join(path.dirname(yml.permalink), path.basename(yml.permalink, path.extname(yml.permalink)));
|
||||||
|
|
||||||
if (!page.yml.uuid) {
|
if (!page.yml.uuid) {
|
||||||
// TODO only do this if it's going to be saved
|
// TODO only do this if it's going to be saved
|
||||||
|
@ -439,7 +546,7 @@
|
||||||
//entity.second = entity.published_at.second;
|
//entity.second = entity.published_at.second;
|
||||||
|
|
||||||
// The root index is the one exception
|
// The root index is the one exception
|
||||||
if (/^\/?index$/.test(entity.yml.permalink)) {
|
if (/^\/?index(\.html?)?$/.test(entity.yml.permalink)) {
|
||||||
entity.yml.permalink = '';
|
entity.yml.permalink = '';
|
||||||
console.info('found index', entity);
|
console.info('found index', entity);
|
||||||
}
|
}
|
||||||
|
@ -500,7 +607,7 @@
|
||||||
set = yearsArr[yindex];
|
set = yearsArr[yindex];
|
||||||
|
|
||||||
if (!set.months[mindex]) {
|
if (!set.months[mindex]) {
|
||||||
set.months[mindex] = { month: f.month, pages: [] };
|
set.months[mindex] = { month: months[parseInt(f.month, 10)], pages: [] };
|
||||||
}
|
}
|
||||||
set = set.months[mindex];
|
set = set.months[mindex];
|
||||||
|
|
||||||
|
@ -554,16 +661,20 @@
|
||||||
// TODO less / sass / etc
|
// TODO less / sass / etc
|
||||||
compiled.push({ contents: entity.body || entity.contents, path: path.join(desi.config.compiled_path, 'themes', entity.path) });
|
compiled.push({ contents: entity.body || entity.contents, path: path.join(desi.config.compiled_path, 'themes', entity.path) });
|
||||||
if (/stylesheets.*\.css/.test(entity.path) && (!/google/.test(entity.path) || /obsid/.test(entity.path))) {
|
if (/stylesheets.*\.css/.test(entity.path) && (!/google/.test(entity.path) || /obsid/.test(entity.path))) {
|
||||||
|
// TODO XXX move to a partial
|
||||||
desi.assets.push(
|
desi.assets.push(
|
||||||
'<link href="' + entity.path + '" type="text/css" rel="stylesheet" media="all">'
|
'<link href="' + desi.urls.base_path + '/themes/' + entity.path + '" type="text/css" rel="stylesheet" media="all">'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
desi.navigation.filter(function (n) {
|
||||||
|
return n;
|
||||||
|
});
|
||||||
console.log(desi.navigation);
|
console.log(desi.navigation);
|
||||||
function compileContentEntity(entity, i, arr) {
|
function compileContentEntity(entity, i, arr) {
|
||||||
console.log("compiling " + (i + 1) + "/" + arr.length + " " + (entity.path || entity.name));
|
console.log("compiling " + (i + 1) + "/" + arr.length + " " + (entity.path || entity.name));
|
||||||
|
|
||||||
var child = ''
|
var previous = ''
|
||||||
, layers
|
, layers
|
||||||
, view
|
, view
|
||||||
;
|
;
|
||||||
|
@ -572,7 +683,6 @@
|
||||||
|
|
||||||
view = {
|
view = {
|
||||||
page: entity.yml // data for just *this* page
|
page: entity.yml // data for just *this* page
|
||||||
, content: child // processed content for just *this* page
|
|
||||||
//, data: desi.data // data.yml
|
//, data: desi.data // data.yml
|
||||||
// https://github.com/janl/mustache.js/issues/415
|
// https://github.com/janl/mustache.js/issues/415
|
||||||
, data: num2str(desi.data)
|
, data: num2str(desi.data)
|
||||||
|
@ -590,6 +700,7 @@
|
||||||
// TODO concat theme, widget, and site assets
|
// TODO concat theme, widget, and site assets
|
||||||
, assets: desi.assets.join('\n')
|
, assets: desi.assets.join('\n')
|
||||||
};
|
};
|
||||||
|
|
||||||
//console.log('rel:', view.relative_url);
|
//console.log('rel:', view.relative_url);
|
||||||
view.site.author = desi.data.author;
|
view.site.author = desi.data.author;
|
||||||
view.site.navigation = JSON.parse(JSON.stringify(desi.navigation));
|
view.site.navigation = JSON.parse(JSON.stringify(desi.navigation));
|
||||||
|
@ -601,14 +712,20 @@
|
||||||
});
|
});
|
||||||
// backwards compat
|
// backwards compat
|
||||||
view.site['navigation?to_pages'] = view.site.navigation;
|
view.site['navigation?to_pages'] = view.site.navigation;
|
||||||
|
view.site['navigation?to__root'] = view.site.navigation;
|
||||||
|
view.data.navigation = view.site.navigation;
|
||||||
|
view.data['navigation?to_pages'] = view.site.navigation;
|
||||||
|
view.data['navigation?to__root'] = view.site.navigation;
|
||||||
|
|
||||||
layers.forEach(function (current) {
|
layers.forEach(function (current) {
|
||||||
// TODO meta.layout
|
// TODO meta.layout
|
||||||
var body = (current.body || current.contents || '').trim()
|
var body = (current.body || current.contents || '').trim()
|
||||||
, html
|
, html
|
||||||
|
, curview = {}
|
||||||
;
|
;
|
||||||
|
|
||||||
current.path = current.path || entity.relativePath + '/' + entity.name;
|
// TODO move to normalization
|
||||||
|
current.path = current.path || (entity.relativePath + '/' + entity.name);
|
||||||
|
|
||||||
if (/\.(html|htm)$/.test(current.path)) {
|
if (/\.(html|htm)$/.test(current.path)) {
|
||||||
html = body;
|
html = body;
|
||||||
|
@ -622,29 +739,42 @@
|
||||||
console.error('unknown parser for ' + (entity.path));
|
console.error('unknown parser for ' + (entity.path));
|
||||||
}
|
}
|
||||||
|
|
||||||
view.content = child;
|
view.content = previous;
|
||||||
|
view.page.content = previous;
|
||||||
|
|
||||||
child = Mustache.render(html, view, desi.partials);
|
// to prevent perfect object equality (and potential template caching)
|
||||||
|
Object.keys(view).forEach(function (key) {
|
||||||
|
curview[key] = view[key];
|
||||||
|
});
|
||||||
|
previous = Mustache.render(html, curview, desi.partials);
|
||||||
});
|
});
|
||||||
|
|
||||||
// TODO add html meta-refresh redirects
|
console.log({ contents: previous });
|
||||||
compiled.push({ contents: child, path: path.join(desi.config.compiled_path, entity.yml.permalink, 'index.html') });
|
|
||||||
|
// NOTE: by now, all permalinks should be in the format /path/to/page.html or /path/to/page/index.html
|
||||||
|
compiled.push({ contents: previous, path: path.join(desi.config.compiled_path, entity.yml.permalink/*, 'index.html'*/) });
|
||||||
entity.yml.redirects = entity.yml.redirects || [];
|
entity.yml.redirects = entity.yml.redirects || [];
|
||||||
if (entity.yml.permalink) {
|
if (/\/index.html$/.test(entity.yml.permalink)) {
|
||||||
entity.yml.redirects.push(entity.yml.permalink + '.html');
|
entity.yml.redirects.push(entity.yml.permalink.replace(/\/index.html$/, '.html'));
|
||||||
|
} else {
|
||||||
|
entity.yml.redirects.push(entity.yml.permalink.replace(/\.html?$/, '/index.html'));
|
||||||
}
|
}
|
||||||
entity.yml.redirects.forEach(function (redirect) {
|
entity.yml.redirects.forEach(function (redirect) {
|
||||||
child =
|
var content
|
||||||
|
;
|
||||||
|
|
||||||
|
// TODO move to partial
|
||||||
|
content =
|
||||||
'<html>'
|
'<html>'
|
||||||
+ '<head>'
|
+ '<head>'
|
||||||
+ '<title>Redirecting to ' + entity.yml.title + '</title>'
|
+ '<title>Redirecting to ' + entity.yml.title + '</title>'
|
||||||
+ '<meta http-equiv="refresh" content="0;URL=\''
|
+ '<meta http-equiv="refresh" content="0;URL=\''
|
||||||
+ desi.urls.url + path.join(entity.yml.permalink)
|
+ desi.urls.url + path.join(desi.urls.base_path, entity.yml.permalink)
|
||||||
+ '\'" />'
|
+ '\'" />'
|
||||||
+ '</head>'
|
+ '</head>'
|
||||||
+ '<body>'
|
+ '<body>'
|
||||||
+ '<p>This page has moved to a <a href="'
|
+ '<p>This page has moved to a <a href="'
|
||||||
+ desi.urls.url + path.join(entity.yml.permalink)
|
+ desi.urls.url + path.join(desi.urls.base_path, entity.yml.permalink)
|
||||||
+'">'
|
+'">'
|
||||||
+ entity.yml.title
|
+ entity.yml.title
|
||||||
+ '</a>.</p>'
|
+ '</a>.</p>'
|
||||||
|
@ -652,7 +782,7 @@
|
||||||
+ '</html>'
|
+ '</html>'
|
||||||
;
|
;
|
||||||
|
|
||||||
compiled.push({ contents: child, path: path.join(desi.config.compiled_path, redirect) });
|
compiled.push({ contents: content, path: path.join(desi.config.compiled_path, redirect) });
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -680,9 +810,9 @@
|
||||||
|
|
||||||
// because some servers / proxies are terrible at handling large uploads (>= 100k)
|
// because some servers / proxies are terrible at handling large uploads (>= 100k)
|
||||||
// (vagrant? or express? one of the two is CRAZY slow)
|
// (vagrant? or express? one of the two is CRAZY slow)
|
||||||
console.info('saving compiled files');
|
console.info('saving compiled files', desi.compiled);
|
||||||
while (compiled.length) {
|
while (compiled.length) {
|
||||||
batches.push(compiled.splice(0, 1));
|
batches.push(compiled.splice(0, 500));
|
||||||
}
|
}
|
||||||
|
|
||||||
now = Date.now();
|
now = Date.now();
|
||||||
|
@ -716,6 +846,7 @@
|
||||||
}).catch(function (e) {
|
}).catch(function (e) {
|
||||||
console.error('A great and uncatchable error has befallen the land. Read ye here for das detalles..');
|
console.error('A great and uncatchable error has befallen the land. Read ye here for das detalles..');
|
||||||
console.error(e.message);
|
console.error(e.message);
|
||||||
|
console.error(e);
|
||||||
throw e;
|
throw e;
|
||||||
});
|
});
|
||||||
}('undefined' !== typeof exports && exports || window));
|
}('undefined' !== typeof exports && exports || window));
|
||||||
|
|
|
@ -246,6 +246,20 @@
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
fsapi.copy = function (files) {
|
||||||
|
var body = { files: files };
|
||||||
|
body = JSON.stringify(body); // this is more or less instant for a few MiB of posts
|
||||||
|
return request.post('/api/fs/copy', body).then(function (resp) {
|
||||||
|
var response = JSON.parse(resp)
|
||||||
|
;
|
||||||
|
|
||||||
|
// not accurate for utf8/unicode, but close enough
|
||||||
|
response.size = body.length;
|
||||||
|
return response;
|
||||||
|
});
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
fsapi.putFiles = function (files) {
|
fsapi.putFiles = function (files) {
|
||||||
var body = { files: files };
|
var body = { files: files };
|
||||||
body = JSON.stringify(body); // this is more or less instant for a few MiB of posts
|
body = JSON.stringify(body); // this is more or less instant for a few MiB of posts
|
||||||
|
|
88
lib/fsapi.js
88
lib/fsapi.js
|
@ -9,6 +9,7 @@ var PromiseA = require('bluebird').Promise
|
||||||
, safeResolve = require('./deardesi-utils').safeResolve
|
, safeResolve = require('./deardesi-utils').safeResolve
|
||||||
, sha1sum = function (str) { return require('secret-utils').hashsum('sha1', str); }
|
, sha1sum = function (str) { return require('secret-utils').hashsum('sha1', str); }
|
||||||
, mkdirp = PromiseA.promisify(require('mkdirp'))
|
, mkdirp = PromiseA.promisify(require('mkdirp'))
|
||||||
|
, fsExtra = PromiseA.promisifyAll(require('fs.extra'))
|
||||||
;
|
;
|
||||||
|
|
||||||
function strip(prefix, pathname) {
|
function strip(prefix, pathname) {
|
||||||
|
@ -157,6 +158,92 @@ function getfs(blogdir, filepaths) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function makeAllDirs(dirpaths) {
|
||||||
|
var errors = []
|
||||||
|
;
|
||||||
|
|
||||||
|
return forEachAsync(dirpaths, function (pathname) {
|
||||||
|
return mkdirp(pathname).catch(function (e) {
|
||||||
|
// TODO exclude attempting to write files to this dir?
|
||||||
|
errors.push({
|
||||||
|
type: 'directory'
|
||||||
|
|
||||||
|
, directory: pathname
|
||||||
|
|
||||||
|
, message: e.message
|
||||||
|
, code: e.code
|
||||||
|
, errno: e.errno
|
||||||
|
, status: e.status
|
||||||
|
, syscall: e.syscall
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
}).then(function () {
|
||||||
|
return errors;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function copyfs(blogdir, files) {
|
||||||
|
// TODO switch format to { source: ..., dest: ..., opts: ... } ?
|
||||||
|
var results = { errors: [] }
|
||||||
|
, dirpaths = {}
|
||||||
|
, sources = Object.keys(files)
|
||||||
|
;
|
||||||
|
|
||||||
|
return forEachAsync(sources, function (source) {
|
||||||
|
/*
|
||||||
|
var nsource = safeResolve(blogdir, source)
|
||||||
|
;
|
||||||
|
*/
|
||||||
|
|
||||||
|
var dest = safeResolve(blogdir, files[source])
|
||||||
|
, pathname = path.dirname(dest)
|
||||||
|
//, filename = path.basename(dest)
|
||||||
|
;
|
||||||
|
|
||||||
|
dirpaths[pathname] = true;
|
||||||
|
|
||||||
|
return Promise.resolve();
|
||||||
|
}).then(function () {
|
||||||
|
// TODO is it better to do this lazy-like or as a batch?
|
||||||
|
// I figure as batch when there may be hundreds of files,
|
||||||
|
// likely within 2 or 3 directories
|
||||||
|
return makeAllDirs(Object.keys(dirpaths)).then(function (errors) {
|
||||||
|
errors.forEach(function (e) {
|
||||||
|
results.errors.push(e);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}).then(function () {
|
||||||
|
// TODO allow delete?
|
||||||
|
return forEachAsync(sources, function (source) {
|
||||||
|
return fsExtra.copyAsync(safeResolve(blogdir, source), safeResolve(blogdir, files[source])).catch(function (e) {
|
||||||
|
results.errors.push({
|
||||||
|
type: 'file'
|
||||||
|
|
||||||
|
, source: source
|
||||||
|
, destination: files[source]
|
||||||
|
|
||||||
|
, message: e.message
|
||||||
|
, code: e.code
|
||||||
|
, errno: e.errno
|
||||||
|
, status: e.status
|
||||||
|
, syscall: e.syscall
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}).catch(function (e) {
|
||||||
|
results.error = {
|
||||||
|
message: e.message
|
||||||
|
, code: e.code
|
||||||
|
, errno: e.errno
|
||||||
|
, status: e.status
|
||||||
|
, syscall: e.syscall
|
||||||
|
};
|
||||||
|
}).then(function () {
|
||||||
|
return results;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function putfs(blogdir, files) {
|
function putfs(blogdir, files) {
|
||||||
var results = { errors: [] }
|
var results = { errors: [] }
|
||||||
, dirpaths = {}
|
, dirpaths = {}
|
||||||
|
@ -246,6 +333,7 @@ walkDirs('blog', ['posts'], { contents: false }).then(function (stats) {
|
||||||
*/
|
*/
|
||||||
|
|
||||||
module.exports.walk = { walkDirs: walkDirs, walkDir: walkDir };
|
module.exports.walk = { walkDirs: walkDirs, walkDir: walkDir };
|
||||||
|
module.exports.copyfs = copyfs;
|
||||||
module.exports.getfs = getfs;
|
module.exports.getfs = getfs;
|
||||||
module.exports.putfs = putfs;
|
module.exports.putfs = putfs;
|
||||||
module.exports.walkDir = walkDir;
|
module.exports.walkDir = walkDir;
|
||||||
|
|
|
@ -38,6 +38,7 @@
|
||||||
"connect-send-json": "^1.0.0",
|
"connect-send-json": "^1.0.0",
|
||||||
"escape-string-regexp": "^1.0.2",
|
"escape-string-regexp": "^1.0.2",
|
||||||
"foreachasync": "^5.0.5",
|
"foreachasync": "^5.0.5",
|
||||||
|
"fs.extra": "^1.2.1",
|
||||||
"json2yaml": "^1.0.3",
|
"json2yaml": "^1.0.3",
|
||||||
"markdown": "^0.5.0",
|
"markdown": "^0.5.0",
|
||||||
"markdown-it": "^3.0.2",
|
"markdown-it": "^3.0.2",
|
||||||
|
|
34
server.js
34
server.js
|
@ -10,6 +10,7 @@ var connect = require('connect')
|
||||||
, send = require('connect-send-json')
|
, send = require('connect-send-json')
|
||||||
|
|
||||||
, app = connect()
|
, app = connect()
|
||||||
|
, fsapi = require('./lib/fsapi')
|
||||||
, walk = require('./lib/fsapi').walk
|
, walk = require('./lib/fsapi').walk
|
||||||
, getfs = require('./lib/fsapi').getfs
|
, getfs = require('./lib/fsapi').getfs
|
||||||
, putfs = require('./lib/fsapi').putfs
|
, putfs = require('./lib/fsapi').putfs
|
||||||
|
@ -23,15 +24,7 @@ var connect = require('connect')
|
||||||
app
|
app
|
||||||
.use(send.json())
|
.use(send.json())
|
||||||
.use(query())
|
.use(query())
|
||||||
.use(function (req, res, next) {
|
|
||||||
console.log('before parse');
|
|
||||||
next();
|
|
||||||
})
|
|
||||||
.use(bodyParser.json({ limit: 10 * 1024 * 1024 })) // 10mb
|
.use(bodyParser.json({ limit: 10 * 1024 * 1024 })) // 10mb
|
||||||
.use(function (req, res, next) {
|
|
||||||
console.log('after parse');
|
|
||||||
next();
|
|
||||||
})
|
|
||||||
.use(require('compression')())
|
.use(require('compression')())
|
||||||
.use('/api/fs/walk', function (req, res, next) {
|
.use('/api/fs/walk', function (req, res, next) {
|
||||||
if (!(/^GET$/i.test(req.method) || /^GET$/i.test(req.query._method))) {
|
if (!(/^GET$/i.test(req.method) || /^GET$/i.test(req.query._method))) {
|
||||||
|
@ -112,6 +105,25 @@ app
|
||||||
res.json(results);
|
res.json(results);
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
|
.use('/api/fs/copy', function (req, res, next) {
|
||||||
|
if (!(/^POST|PUT$/i.test(req.method) || /^POST|PUT$/i.test(req.query._method))) {
|
||||||
|
next();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var opts = {}
|
||||||
|
, files = req.body.files
|
||||||
|
;
|
||||||
|
|
||||||
|
if ('object' !== typeof files || !Object.keys(files).length) {
|
||||||
|
res.json({ error: "please specify POST w/ req.body.files" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return fsapi.copyfs(blogdir, files, opts).then(function (results) {
|
||||||
|
res.json(results);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
|
||||||
.use('/api/fs', function (req, res, next) {
|
.use('/api/fs', function (req, res, next) {
|
||||||
next();
|
next();
|
||||||
|
@ -119,12 +131,12 @@ app
|
||||||
})
|
})
|
||||||
.use('/api/fs/static', serveStatic('.'))
|
.use('/api/fs/static', serveStatic('.'))
|
||||||
|
|
||||||
.use(serveStatic(blogdir))
|
|
||||||
.use(serveStatic('.'))
|
.use(serveStatic('.'))
|
||||||
|
.use(serveStatic(blogdir))
|
||||||
;
|
;
|
||||||
|
|
||||||
module.exports = app;
|
module.exports = app;
|
||||||
|
|
||||||
require('http').createServer().on('request', app).listen(80, function () {
|
require('http').createServer().on('request', app).listen(process.argv[2] || 65080, function () {
|
||||||
console.log('listening 80');
|
console.log('listening ' + (process.argv[2] || 65080));
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue