From e432921a543a0fd9f616d354d1e483894dfccb15 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Jakub=20Sko=C5=99epa?= <jakub@skorepa.info>
Date: Mon, 11 Apr 2016 21:20:59 +0200
Subject: [PATCH] Implement simple article queriing utility
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Example usage:

node query.js --sql "select metadata.tags as tags, metadata.title as title where count(metadata.tags) == 1 and contains(metadata.tags, 'Článek');"

prints list of articles which have only tag "Článek" - usefull for
listing not yet tagged articles
---
 package.json      |   1 +
 query.js          | 152 ++++++++++++++++++++++++++++++++++++++++++++++
 sitegin/config.js |  43 +++++++------
 3 files changed, 178 insertions(+), 18 deletions(-)
 create mode 100644 query.js

diff --git a/package.json b/package.json
index e9e8f0ed..3c1432e3 100644
--- a/package.json
+++ b/package.json
@@ -23,6 +23,7 @@
     "nodegit": "^0.11.7",
     "nunjucks": "^2.3.0",
     "nunjucks-date-filter": "^0.1.1",
+    "sqlite-parser": "^0.14.3",
     "syntax-error": "^1.1.5",
     "toml": "^2.3.0",
     "toml-js": "0.0.8",
diff --git a/query.js b/query.js
new file mode 100644
index 00000000..178240ca
--- /dev/null
+++ b/query.js
@@ -0,0 +1,152 @@
+var sqliteParser = require('sqlite-parser');
+var cli = require('cli');
+var options = cli.parse({
+ contentdir: [null, 'Allows to specify arbitrary content directory.', 'string', 'content'],
+ sql: ['q', 'SQL querry', 'string']
+});
+
+var ast = sqliteParser(options.sql).statement[0];
+
+console.log(JSON.stringify(ast,null,'  '));
+
+var getProp = function(prop, article) {
+  var ret = article;
+  prop.split('.').forEach(function(p) {
+    if(ret === undefined) return;
+    ret = ret[p];
+  });
+  return ret;
+}
+
+var functions = {
+  'contains': function(context, args) {
+    if(args.length !== 2) throw new Error('Wrong number of arguments for function contains');
+    var prop = evalExpr(context, args[0]);
+    if(!Array.isArray(prop)) return false;
+    var val = evalExpr(context, args[1]);
+    var ret = false;
+    prop.forEach(function(el) {
+      if(el == val) ret = true;
+    })
+    return ret;
+  },
+  'count': function(context, args) {
+    if(args.length !== 1) throw new Error('Wrong number of arguments for function count');
+    var prop = evalExpr(context, args[0]);
+    if(prop === undefined) return 0;
+    if(!Array.isArray(prop)) return 1;
+    return prop.length;
+  }
+}
+
+var binaryOperators = {
+  'and': function(context, left, right) {
+    return evalExpr(context, left) && evalExpr(context, right);
+  },
+  'or': function(context, left, right) {
+    return evalExpr(context, left) || evalExpr(context, right);
+  },
+  '==': function(context, left, right) {
+    return evalExpr(context, left) == evalExpr(context, right);
+  },
+  '>': function(context, left, right) {
+    return evalExpr(context, left) > evalExpr(context, right);
+  },
+  '<': function(context, left, right) {
+    return evalExpr(context, left) < evalExpr(context, right);
+  },
+  '>=': function(context, left, right) {
+    return evalExpr(context, left) >= evalExpr(context, right);
+  },
+  '<=': function(context, left, right) {
+    return evalExpr(context, left) <= evalExpr(context, right);
+  },
+  '!=': function(context, left, right) {
+    return evalExpr(context, left) != evalExpr(context, right);
+  },
+}
+
+var literals = {
+  'decimal': function(context, value) {
+    return Number(value);
+  },
+  'string': function(context, value) {
+    return value;
+  }
+}
+
+var evalExpr = function(context, expr) {
+  if(expr.type == 'identifier') {
+    return getProp(expr.name, context);
+  } else if(expr.type == 'expression' && expr.format == 'binary') {
+    if(typeof binaryOperators[expr.operation] === 'function') {
+      return binaryOperators[expr.operation](context, expr.left, expr.right);
+    } else {
+      throw new Error('Unsupported binary operator '+expr.operation);
+    }
+  } else if(expr.type == 'function') {
+    if(typeof functions[expr.name] === 'function') {
+      return functions[expr.name](context, expr.args);
+    } else {
+      throw new Error('Unsupported function '+expr.name);
+    }
+  } else if(expr.type == 'literal') {
+    if(typeof literals[expr.variant] === 'function') {
+      return literals[expr.variant](context, expr.value);
+    } else {
+      throw new Error('Unsupported literal variant '+expr.variant);
+    }
+  } else {
+    throw new Error('Unsupported expression '+JSON.stringify(expr));
+  }
+  throw new Error('Code should not get here. This is really bad.');
+}
+
+var jobs = require('./sitegin/jobs');
+jobs.registerMultiple(
+  {},
+  ['parseHugo', './parseHugo'],
+  ['readFiles', './readFiles']
+)
+.then(function() {
+  return jobs.run('readFiles',{config:{
+    sourceDir: options.contentdir,
+    articlesLocation: 'articles',
+    redirectsLocation: 'redirects'
+  }})
+})
+.then(function(obj) {
+  return jobs.run('parseHugo',obj)
+})
+.then(function(obj){
+  return obj.pages
+})
+.then(function(pages) {
+  return pages.filter(function(page) {
+    return evalExpr(page, ast.where[0]);
+  })
+})
+.then(function(pages) {
+  var results = ast.result;
+  if(results[0].variant == 'star') {
+    pages.forEach(function(page) {
+      console.log(JSON.stringify(page));
+    })
+    return;
+  }
+  pages.forEach(function(page) {
+    var n = {};
+    results.forEach(function(result) {
+      if(result.alias) n[result.alias] = getProp(result.name, page);
+      else n[result.name] = getProp(page, result.name);
+    })
+    console.log(n);
+  })
+})
+.then(function() {
+  process.exit(0);
+})
+.catch(function(e) {
+  console.log(e.stack);
+  process.exit(1);
+})
diff --git a/sitegin/config.js b/sitegin/config.js
index 9b2e93c6..1a6de2ce 100644
--- a/sitegin/config.js
+++ b/sitegin/config.js
@@ -1,12 +1,14 @@
+'use strict';
 /*
  * This file specifies whole configuration of sitegin.
- * In future it might actually read configuration files but for now - if you
- * want to configure sitegin you have to modify this file.
- * (or write config reader which would replace this file - I'd appreciate
- * merge request for this feature ;) )
+ * It reads config file from content/config.toml it also parses command-line
+ * arguments. CLI args take precedence over config options
  */
 var Git = require('nodegit');
 var cli = require('cli');
+var fs = require('fs');
+var path = require('path');
+var toml = require('toml');
 var options = cli.parse({
  noserver: ['n', 'Dont run server'],
  port: ['p', 'Port on which server should run', 'number', 1337],
@@ -30,20 +32,25 @@ module.exports = function() {
     options.uiport = options.port+1;
   }
 
-  return new Promise(function(resolve, reject){
-    function doResolve() {
-      resolve({config: {
-        options: options,
-        builddir: builddir,
-        sourceDir: options.contentdir,
-        staticDir: options.staticdir,
-        themeDir: options.themedir,
-        articlesLocation: 'articles',
-        redirectsLocation: 'redirects',
-        linksPerPage: 6
-      }});
-    }
-    doResolve();
+  return new Promise(function(resolve, reject) {
+    var config = {
+      options: options,
+      builddir: builddir,
+      sourceDir: options.contentdir,
+      staticDir: options.staticdir,
+      themeDir: options.themedir,
+      articlesLocation: 'articles',
+      redirectsLocation: 'redirects',
+      linksPerPage: 6
+    };
+    var configFile = path.join(config.sourceDir, 'config.toml');
+    fs.readFile(configFile,'utf-8',function(err,data) {
+      if(err) return reject('Error reading '+configFile);
+      data = toml.parse(data);
+      console.log(data);
+      for(let attr in config) { data[attr] = config[attr]; }
+      resolve({config: data});
+    })
   })
 }
 module.exports.watch = !options.noserver;
-- 
GitLab