2014-11-12 05:48:26 +03:00
|
|
|
|
/**********************************************************************
|
|
|
|
|
|
*
|
|
|
|
|
|
*
|
|
|
|
|
|
*
|
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
2014-12-28 03:47:44 +03:00
|
|
|
|
var pathlib = require('path')
|
2014-12-16 04:50:34 +03:00
|
|
|
|
var events = require('events')
|
|
|
|
|
|
|
|
|
|
|
|
var fse = require('fs.extra')
|
|
|
|
|
|
var glob = require('glob')
|
|
|
|
|
|
var promise = require('promise')
|
2014-11-14 21:57:55 +03:00
|
|
|
|
|
|
|
|
|
|
|
2014-11-12 05:48:26 +03:00
|
|
|
|
define(function(require){ var module = {}
|
|
|
|
|
|
console.log('>>> file')
|
|
|
|
|
|
|
|
|
|
|
|
//var DEBUG = DEBUG != null ? DEBUG : true
|
|
|
|
|
|
|
2014-11-14 21:57:55 +03:00
|
|
|
|
var tasks = require('lib/tasks')
|
2014-11-12 05:48:26 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-11-14 21:57:55 +03:00
|
|
|
|
/*********************************************************************/
|
2014-11-12 05:48:26 +03:00
|
|
|
|
|
2014-11-14 21:57:55 +03:00
|
|
|
|
var INDEX_DIR = '.ImageGrid'
|
2014-11-12 05:48:26 +03:00
|
|
|
|
|
|
|
|
|
|
|
2014-12-16 04:50:34 +03:00
|
|
|
|
/*********************************************************************/
|
|
|
|
|
|
// Queue
|
|
|
|
|
|
//
|
|
|
|
|
|
// Task
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-11-12 05:48:26 +03:00
|
|
|
|
/*********************************************************************/
|
2014-12-10 19:59:05 +03:00
|
|
|
|
// things we need...
|
|
|
|
|
|
// - load latest by pattern
|
|
|
|
|
|
// - merge
|
|
|
|
|
|
// - load latest base
|
|
|
|
|
|
// - merge diffs later than base
|
|
|
|
|
|
// - find index(s) in subtree
|
|
|
|
|
|
// - load index
|
|
|
|
|
|
// - data version
|
|
|
|
|
|
// - join indexes
|
|
|
|
|
|
// - take care of different base paths in images
|
|
|
|
|
|
//
|
2014-12-16 04:50:34 +03:00
|
|
|
|
//
|
|
|
|
|
|
//
|
2014-12-12 03:56:58 +03:00
|
|
|
|
// Might also be a nice idea to generic import:
|
|
|
|
|
|
// - get all .ImageGrid/*.json
|
2014-12-16 04:50:34 +03:00
|
|
|
|
// - group by ([a-z]*).* <20> pattern with <keyword>
|
2014-12-12 03:56:58 +03:00
|
|
|
|
// - sort by name, descending
|
|
|
|
|
|
// - split at first non-diff
|
|
|
|
|
|
// - merge diff's in reverse tail to head
|
|
|
|
|
|
//
|
2014-12-16 04:50:34 +03:00
|
|
|
|
// ...and output to format:
|
2014-12-12 03:56:58 +03:00
|
|
|
|
// {
|
|
|
|
|
|
// <keyword>: <data>,
|
|
|
|
|
|
// ...
|
|
|
|
|
|
// }
|
|
|
|
|
|
//
|
2014-11-12 05:48:26 +03:00
|
|
|
|
|
2014-12-28 20:46:06 +03:00
|
|
|
|
var guaranteeGlobEvents =
|
|
|
|
|
|
module.guaranteeGlobEvents =
|
|
|
|
|
|
function guaranteeGlobEvents(glob, all_matches){
|
|
|
|
|
|
all_matches = all_matches == null ? true : false
|
|
|
|
|
|
var visited = []
|
|
|
|
|
|
|
|
|
|
|
|
return glob
|
|
|
|
|
|
// keep track of visited matches...
|
|
|
|
|
|
.on('match', function(path){
|
|
|
|
|
|
all_matches && visited.push(path)
|
|
|
|
|
|
})
|
|
|
|
|
|
// trigger new handlers...
|
|
|
|
|
|
.on('newListener', function(evt, func){
|
|
|
|
|
|
// trigger the 'end' handler if we have already finished...
|
|
|
|
|
|
if(evt == 'end' && this.found != null){
|
|
|
|
|
|
func.call(this, this.found)
|
|
|
|
|
|
|
|
|
|
|
|
// trigger the 'match' handler for each match already found...
|
|
|
|
|
|
} else if(all_matches && evt == 'match' && visited.length > 0){
|
|
|
|
|
|
visited.forEach(function(path){
|
|
|
|
|
|
func.call(this, path)
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// XXX return a promise rather than an event emitter (???)
|
|
|
|
|
|
// XXX glob has a problem: if a match happens fast enough and we are slow
|
|
|
|
|
|
// enough to register a 'match' handler, then that match(s) will get
|
|
|
|
|
|
// missed...
|
2014-11-14 21:57:55 +03:00
|
|
|
|
function listIndexes(base){
|
2014-12-28 20:46:06 +03:00
|
|
|
|
return guaranteeGlobEvents(glob(base +'/**/'+ INDEX_DIR))
|
2014-11-14 21:57:55 +03:00
|
|
|
|
}
|
2014-11-12 05:48:26 +03:00
|
|
|
|
|
|
|
|
|
|
|
2014-12-28 20:46:06 +03:00
|
|
|
|
// XXX return a promise rather than an event emitter (???)
|
2014-12-16 04:50:34 +03:00
|
|
|
|
function listJSON(path, pattern){
|
|
|
|
|
|
pattern = pattern || '*'
|
2014-12-28 20:46:06 +03:00
|
|
|
|
return guaranteeGlobEvents(glob(path +'/'+ pattern +'.json'))
|
2014-12-16 04:50:34 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var loadFile = promise.denodeify(fse.readFile)
|
2014-12-28 20:46:06 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// XXX handle errors...
|
2014-12-16 04:50:34 +03:00
|
|
|
|
function loadJSON(path){
|
2014-12-28 03:47:44 +03:00
|
|
|
|
return loadFile(path).then(JSON.parse)
|
2014-12-16 04:50:34 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// json file name format:
|
|
|
|
|
|
// [<timestamp>-]<keyword>[-diff].json
|
|
|
|
|
|
//
|
|
|
|
|
|
// events emited:
|
|
|
|
|
|
// - queued <path> - json file path queued for loading
|
|
|
|
|
|
// - loaded <path> - done loading json file path
|
|
|
|
|
|
// - index <path> <data> - done loding index at path
|
2014-12-28 05:09:28 +03:00
|
|
|
|
//
|
|
|
|
|
|
// NOTE: logger must be an event emitter...
|
2014-12-16 04:50:34 +03:00
|
|
|
|
//
|
2014-12-28 03:47:44 +03:00
|
|
|
|
// XXX test with:
|
2014-12-28 04:33:57 +03:00
|
|
|
|
// requirejs(['file'],
|
|
|
|
|
|
// function(m){
|
|
|
|
|
|
// f = m.loadIndex("L:/mnt/hdd15 (photo)/NTFS1/media/img/others") })
|
|
|
|
|
|
// .done(function(d){ console.log(d) })
|
|
|
|
|
|
// XXX need to do better error handling...
|
2014-12-28 05:09:28 +03:00
|
|
|
|
// XXX a bit overcomplicated, see if this can be split into more generic
|
|
|
|
|
|
// sections...
|
2014-12-28 03:47:44 +03:00
|
|
|
|
var loadIndex =
|
|
|
|
|
|
module.loadIndex =
|
2014-12-28 04:33:57 +03:00
|
|
|
|
function(path, logger){
|
2014-12-16 04:50:34 +03:00
|
|
|
|
var p = path.split(INDEX_DIR)
|
|
|
|
|
|
var last = p.slice(-1)[0].trim()
|
|
|
|
|
|
|
2014-12-28 04:33:57 +03:00
|
|
|
|
return new promise(function(resolve, reject){
|
|
|
|
|
|
// we've got an index...
|
|
|
|
|
|
if(p.length > 1 && /^\/*$/.test(last)){
|
|
|
|
|
|
listJSON(path)
|
2014-12-28 20:46:06 +03:00
|
|
|
|
// XXX handle errors...
|
|
|
|
|
|
.on('error', function(err){
|
|
|
|
|
|
logger && logger.emit('error', err)
|
|
|
|
|
|
})
|
2014-12-28 04:33:57 +03:00
|
|
|
|
.on('end', function(files){
|
|
|
|
|
|
var res = {}
|
|
|
|
|
|
var index = {}
|
|
|
|
|
|
var root = {}
|
|
|
|
|
|
|
|
|
|
|
|
// group by keyword...
|
|
|
|
|
|
files
|
|
|
|
|
|
.sort()
|
|
|
|
|
|
.reverse()
|
|
|
|
|
|
.forEach(function(n){
|
|
|
|
|
|
var b = pathlib.basename(n)
|
|
|
|
|
|
var s = b.split(/[-.]/g).slice(0, -1)
|
|
|
|
|
|
|
|
|
|
|
|
// <keyword>.json / non-diff
|
|
|
|
|
|
// NOTE: this is a special case, we add this to
|
|
|
|
|
|
// a seporate index and then concat it to
|
|
|
|
|
|
// the final list if needed...
|
|
|
|
|
|
if(s.length == 1){
|
|
|
|
|
|
var k = s[0]
|
|
|
|
|
|
root[k] = n
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
|
|
// <timestamp>-<keyword>[-diff].json / diff / non-diff
|
|
|
|
|
|
} else {
|
|
|
|
|
|
var k = s[1]
|
|
|
|
|
|
var d = s[2] == 'diff'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// new keyword...
|
|
|
|
|
|
if(index[k] == null){
|
|
|
|
|
|
index[k] = [[d, n]]
|
|
|
|
|
|
logger && logger.emit('queued', n)
|
|
|
|
|
|
|
|
|
|
|
|
// do not add anything past the latest non-diff
|
|
|
|
|
|
// for each keyword...
|
|
|
|
|
|
} else if(index[k].slice(-1)[0][0] == true){
|
|
|
|
|
|
index[k].push([d, n])
|
|
|
|
|
|
logger && logger.emit('queued', n)
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// add root files where needed...
|
2014-12-28 20:46:06 +03:00
|
|
|
|
Object.keys(root)
|
|
|
|
|
|
.forEach(function(k){
|
|
|
|
|
|
var n = root[k]
|
2014-12-28 04:33:57 +03:00
|
|
|
|
|
2014-12-28 20:46:06 +03:00
|
|
|
|
// no diffs...
|
|
|
|
|
|
if(index[k] == null){
|
|
|
|
|
|
index[k] = [[false, n]]
|
|
|
|
|
|
logger && logger.emit('queued', n)
|
2014-12-16 04:50:34 +03:00
|
|
|
|
|
2014-12-28 20:46:06 +03:00
|
|
|
|
// add root file if no base is found...
|
|
|
|
|
|
} else if(index[k].slice(-1)[0][0] == true){
|
|
|
|
|
|
index[k].push([false, n])
|
|
|
|
|
|
logger && logger.emit('queued', n)
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2014-12-28 03:47:44 +03:00
|
|
|
|
|
2014-12-28 04:33:57 +03:00
|
|
|
|
// load...
|
|
|
|
|
|
promise
|
|
|
|
|
|
.all(Object.keys(index).map(function(k){
|
|
|
|
|
|
// get relevant paths...
|
|
|
|
|
|
var diffs = index[k]
|
|
|
|
|
|
var latest = diffs.splice(-1)[0][1]
|
|
|
|
|
|
|
|
|
|
|
|
// load latest...
|
|
|
|
|
|
return loadJSON(latest)
|
|
|
|
|
|
.then(function(data){
|
|
|
|
|
|
// handle diffs...
|
|
|
|
|
|
return promise
|
|
|
|
|
|
.all(diffs
|
|
|
|
|
|
.reverse()
|
|
|
|
|
|
.map(function(p){
|
|
|
|
|
|
p = p[1]
|
|
|
|
|
|
// load diff...
|
|
|
|
|
|
return loadJSON(p)
|
2014-12-28 20:46:06 +03:00
|
|
|
|
// XXX handle errors...
|
2014-12-28 04:33:57 +03:00
|
|
|
|
.done(function(json){
|
|
|
|
|
|
// merge...
|
|
|
|
|
|
for(var k in json){
|
|
|
|
|
|
data[k] = json[k]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
logger && logger.emit('loaded', p)
|
|
|
|
|
|
})
|
|
|
|
|
|
}))
|
|
|
|
|
|
.then(function(){
|
|
|
|
|
|
res[k] = data
|
|
|
|
|
|
|
|
|
|
|
|
logger && logger.emit('loaded', latest)
|
|
|
|
|
|
})
|
|
|
|
|
|
})
|
|
|
|
|
|
}))
|
|
|
|
|
|
.then(function(){
|
|
|
|
|
|
logger && logger.emit('index', path, res)
|
|
|
|
|
|
|
|
|
|
|
|
var d = {}
|
|
|
|
|
|
d[path] = res
|
|
|
|
|
|
|
|
|
|
|
|
resolve(d)
|
|
|
|
|
|
})
|
2014-12-28 03:47:44 +03:00
|
|
|
|
})
|
2014-12-16 04:50:34 +03:00
|
|
|
|
|
2014-12-28 04:33:57 +03:00
|
|
|
|
// no explicit index given -- find all in sub tree...
|
|
|
|
|
|
} else {
|
|
|
|
|
|
var res = {}
|
|
|
|
|
|
|
2014-12-28 05:09:28 +03:00
|
|
|
|
// XXX handle 'error' event...
|
2014-12-28 04:33:57 +03:00
|
|
|
|
listIndexes(path)
|
2014-12-28 20:46:06 +03:00
|
|
|
|
// XXX handle errors...
|
|
|
|
|
|
.on('error', function(err){
|
|
|
|
|
|
logger && logger.emit('error', err)
|
|
|
|
|
|
})
|
2014-12-28 05:09:28 +03:00
|
|
|
|
// collect the found indexes...
|
|
|
|
|
|
.on('match', function(path){
|
|
|
|
|
|
loadIndex(path, logger)
|
|
|
|
|
|
.done(function(obj){ res[path] = obj[path] })
|
|
|
|
|
|
})
|
|
|
|
|
|
// done...
|
|
|
|
|
|
.on('end', function(paths){
|
|
|
|
|
|
resolve(res)
|
2014-12-28 03:47:44 +03:00
|
|
|
|
})
|
2014-12-28 04:33:57 +03:00
|
|
|
|
}
|
|
|
|
|
|
})
|
2014-12-16 04:50:34 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-11-12 05:48:26 +03:00
|
|
|
|
/**********************************************************************
|
|
|
|
|
|
* vim:set ts=4 sw=4 : */
|
|
|
|
|
|
return module })
|