blob: b0682091eea1fc582c6fde7734939cd9927ed018 [file] [log] [blame]
// AsterixSDK
// Core Object for REST API communication
// Handles callbacks to client applications, communication to REST API, and
// endpoint selection. Initializes an RPC consumer object.
//
// Usage:
// var a = new AsterixSDK();
function AsterixSDK() {
// Asterix SDK request handler initialization
// TODO Depending on configuration, may need multiples of these...
this.xhr = new easyXDM.Rpc({
remote: "http://localhost:19002/sdk/static/client.html"
}, {
remote: {
post: {}
}
});
// Asterix SDK => send
// Posts a message containing an API endpoint, json data,
// and a UI callback function.
//
// Usage:
// var a = AsterixSDK();
// a.send(
// http://localhost:19002/XYZ,
// json
// );
myThis = this;
this.send = function (endpoint, data, extras) {
this.extras = extras;
this.xhr.post(
endpoint,
data,
this.branch
);
};
// Set callback functions
this.callbacks = {};
this.callback = function(fn, mode) {
if (mode == "sync" || mode == "async") {
this.callbacks[mode] = fn;
}
return this;
};
// Set branching method on send
this.branch = function(response) {
if (response && response["error-code"]) {
alert("Error [Code" + response["error-code"][0] + "]: " + response["error-code"][1]);
} else if (response && response["results"]) {
var fn_callback = myThis.callbacks["sync"];
fn_callback(response, myThis.extras);
} else if (response["handle"]) {
var fn_callback = myThis.callbacks["async"];
fn_callback(response, myThis.extras);
} else if (response["status"]) {
var fn_callback = myThis.callbacks["sync"];
fn_callback(response, myThis.extras);
}
}
// Asterix SDK => requestHandler
// Handlers remote requests to Asterix REST API
// using the easyXDM RPC protocol
// Usually should only be called by client side, could be overridden
// TODO Get rid of jQuery ajax notation in favor of xmlhttprequest pure js
this.requestHandler = function() {
var rpc = new easyXDM.Rpc({}, {
local: {
post: {
method: function(url, data, fn, fnError){
$.ajax({
type : 'GET',
url : url,
data : data,
dataType : "json",
success : function(res) {
fn(res);
}
});
}
}
}
});
}
}
// Temporary AsterixExpression Placeholder
function AExpression () {
this._properties = {};
this._success = function() {};
return this;
}
AExpression.prototype.bind = function(options) {
var options = options || {};
if (options.hasOwnProperty("dataverse")) {
this._properties["dataverse"] = options["dataverse"];
}
if (options.hasOwnProperty("success")) {
this._success = options["success"];
}
if (options.hasOwnProperty("return")) {
this._properties["return"] = " return " + options["return"].val();
}
};
AExpression.prototype.run = function() {
var success_fn = this._success;
$.ajax({
type : 'GET',
url : "http://localhost:19002/query",
data : {"query" : this.val()},
dataType : "json",
success : function(data) {
success_fn(data);
}
});
return this;
};
AExpression.prototype.val = function() {
var value = "";
// If there is a dataverse defined, provide it.
if (this._properties.hasOwnProperty("dataverse")) {
value += "use dataverse " + this._properties["dataverse"] + ";\n";
};
if (this._properties.hasOwnProperty("value")) {
value += this._properties["value"];
}
return value;
};
// @param expressionValue [String]
AExpression.prototype.set = function(expressionValue) {
this._properties["value"] = expressionValue;
return this;
};
AExpression.prototype.error = function(msg) {
return "Asterix FunctionExpression Error: " + msg;
};
// FunctionExpression
// Parent: AsterixExpression
//
// @param options [Various],
// @key function [String], a function to be applid to the expression
// @key expression [AsterixExpression or AsterixClause] an AsterixExpression/Clause to which the fn will be applied
function FunctionExpression(options) {
// Initialize superclass
AExpression.call(this);
// Possible to initialize a function epxression without inputs, or with them
this.bind(options);
// Return object
return this;
}
FunctionExpression.prototype = Object.create(AExpression.prototype);
FunctionExpression.prototype.constructor = FunctionExpression;
FunctionExpression.prototype.bind = function(options) {
AExpression.prototype.bind.call(this, options);
var options = options || {};
if (options.hasOwnProperty("function")) {
this._properties["function"] = options["function"];
}
if (options.hasOwnProperty("expression")) {
this._properties["expression"] = options["expression"];
}
return this;
};
FunctionExpression.prototype.val = function () {
return this._properties["function"] + "(" + this._properties["expression"].val() + ")";
};
// FLWOGR ::= ( ForClause | LetClause ) ( Clause )* "return" Expression
// Clause ::= ForClause | LetClause | WhereClause | OrderbyClause | GroupClause | LimitClause | DistinctClause
//
// WhereClause ::= "where" Expression
// OrderbyClause ::= "order" "by" Expression ( ( "asc" ) | ( "desc" ) )? ( "," Expression ( ( "asc" ) | ( "desc" ) )? )*
//
// GroupClause ::= "group" "by" ( Variable ":=" )? Expression ( "," ( Variable ":=" )? Expression )* ( "decor" Variable ":=" Expression ( "," "decor" Variable ":=" Expression )* )? "with" VariableRef ( "," VariableRef )*
// LimitClause ::= "limit" Expression ( "offset" Expression )?
// DistinctClause ::= "distinct" "by" Expression ( "," Expression )*
// FLWOGRExpression
//
// FLWOGRExpression ::= ( ForClause | LetClause ) ( Clause )* "return" Expression
function FLWOGRExpression (options) {
// Initialize superclass
AExpression.call(this);
this._properties["clauses"] = [];
this._properties["minSize"] = 0;
// Bind options and return
this.bind(options);
return this;
}
FLWOGRExpression.prototype = Object.create(AExpression.prototype);
FLWOGRExpression.prototype.constructor = FLWOGRExpression;
FLWOGRExpression.prototype.bind = function(options) {
AExpression.prototype.bind.call(this, options);
var options = options || {};
if (options instanceof SetStatement) {
this._properties["clauses"].push(options);
this._properties["minSize"] += 1;
}
if (this._properties["clauses"].length <= this._properties["minSize"]) {
// Needs to start with for or let clause
if (options instanceof ForClause || options instanceof LetClause) {
this._properties["clauses"].push(options);
}
} else {
if (options instanceof AQLClause) {
this._properties["clauses"].push(options);
}
}
return this;
};
FLWOGRExpression.prototype.val = function() {
var value = AExpression.prototype.val.call(this);
var clauseValues = [];
for (var c in this._properties["clauses"]) {
clauseValues.push(this._properties["clauses"][c].val());
}
return value + clauseValues.join("\n");// + ";";
};
// AQLClause
//
// Base Clause ::= ForClause | LetClause | WhereClause | OrderbyClause | GroupClause | LimitClause | DistinctClause
function AQLClause() {
this._properties = {};
this._properties["clause"] = "";
}
AQLClause.prototype.val = function() {
var value = this._properties["clause"];
if (this._properties.hasOwnProperty("return")) {
value += " return " + this._properties["return"].val();
}
return value;
};
AQLClause.prototype.bind = function(options) {
var options = options || {};
if (options.hasOwnProperty("return")) {
this._properties["return"] = options["return"];
}
return this;
};
AQLClause.prototype.set = function(value) {
this._properties["clause"] = value;
return this;
};
// ForClause
//
// Grammar:
// "for" Variable ( "at" Variable )? "in" ( Expression )
//
// @param for_variable [String], REQUIRED, first variable in clause
// @param at_variable [String], NOT REQUIRED, first variable in clause
// @param expression [AsterixExpression], REQUIRED, expression to evaluate
//
// TODO Error Checking
function ForClause(for_variable, at_variable, expression) {
AQLClause.call(this);
// at_variable is optional, check if defined
var at = typeof at_variable ? at_variable : null;
// Prepare clause
this._properties["clause"] = "for " + for_variable;
if (at != null) {
this._properties["clause"] += " at " + at_variable;
}
this._properties["clause"] += " in " + expression.val();
return this;
}
ForClause.prototype = Object.create(AQLClause.prototype);
ForClause.prototype.constructor = ForClause;
// LetClause
//
// Grammar:
// LetClause ::= "let" Variable ":=" Expression
//
// @param let_variable [String]
// @param expression [AExpression]
//
// TODO Vigorous error checking
function LetClause(let_variable, expression) {
AQLClause.call(this);
this._properties["clause"] = "let $" + let_variable + " := ";
this._properties["clause"] += expression.val();
return this;
}
LetClause.prototype = Object.create(AQLClause.prototype);
LetClause.prototype.constructor = LetClause;
// ReturnClause
//
// Grammar:
// return [AQLExpression]
function ReturnClause(expression) {
AQLClause.call(this);
this._properties["clause"] = "return ";
if (expression instanceof AExpression || expression instanceof AQLClause) {
this._properties["clause"] += expression.val();
} else if ( Object.getPrototypeOf( expression ) === Object.prototype ) {
this._properties["clause"] += "{";
var returnStatements = [];
for (returnValue in expression) {
if (expression[returnValue] instanceof AExpression) {
returnStatements.push('"' + returnValue + '" ' + " : " + expression[returnValue].val());
} else if (typeof expression[returnValue] == "string") {
returnStatements.push('"' + returnValue + '" ' + " : " + expression[returnValue]);
}
}
this._properties["clause"] += returnStatements.join(",\n");
this._properties["clause"] += "}";
} else {
this._properties["clause"] += new AExpression().set(expression).val();
}
return this;
}
ReturnClause.prototype = Object.create(AQLClause.prototype);
ReturnClause.prototype.constructor = ReturnClause;
ReturnClause.prototype.val = function () {
return this._properties["clause"];
};
// WhereClause
//
// Grammar:
// ::= "where" Expression
//
// @param expression [BooleanExpression], pushes this expression onto the stack
function WhereClause(expression) {
AQLClause.call(this);
this._properties["stack"] = [];
this.bind(expression);
return this;
}
WhereClause.prototype = Object.create(AQLClause.prototype);
WhereClause.prototype.constructor = WhereClause;
WhereClause.prototype.bind = function(expression) {
if (expression instanceof AExpression) {
this._properties["stack"].push(expression);
}
};
WhereClause.prototype.val = function() {
var value = "where ";
var count = this._properties["stack"].length - 1;
while (count >= 0) {
value += this._properties["stack"][count].val() + " ";
count -= 1;
}
return value;
};
// LimitClause
// Grammar:
// LimitClause ::= "limit" Expression ( "offset" Expression )?
//
// @param limitExpression [REQUIRED, AQLExpression]
// @param offsetExpression [OPTIONAL, AQLExpression]
function LimitClause(limitExpression, offsetExpression) {
AQLClause.call(this);
// limitExpression required
this._properties["clause"] = "limit " + limitExpression.val();
// Optional: Offset
var offset = typeof offsetExpression ? offsetExpression : null;
if (offset != null) {
this._properties["clause"] += " offset " + offsetExpression.val();
}
return this;
}
LimitClause.prototype = Object.create(AQLClause.prototype);
LimitClause.prototype.constructor = LimitClause;
// OrderbyClause
//
// Grammar:
// OrderbyClause ::= "order" "by" Expression ( ( "asc" ) | ( "desc" ) )? ( "," Expression ( ( "asc" ) | ( "desc" ) )? )*
//
// @params AQLExpressions and asc/desc strings, in any quantity. At least one required.
function OrderbyClause() {
AQLClause.call(this);
// At least one argument expression is required, and first should be expression
if (arguments.length == 0 || !(arguments[0] instanceof AExpression)) {
// TODO Not sure which error to throw for an empty OrderBy but this should fail.
alert("Order By Error");
this._properties["clause"] = null;
return this;
}
var expc = 0;
var expressions = [];
while (expc < arguments.length) {
var expression = "";
if (arguments[expc] instanceof AExpression) {
expression += arguments[expc].val();
}
var next = expc + 1;
if (next < arguments.length && (arguments[next] == "asc" || arguments[next] == "desc")) {
expc++;
expression += " " + arguments[expc];
}
expressions.push(expression);
expc++;
}
this._properties["clause"] = "order by " + expressions.join(", ");
return this;
}
OrderbyClause.prototype = Object.create(AQLClause.prototype);
OrderbyClause.prototype.constructor = OrderbyClause;
// GroupClause
//
// Grammar:
// GroupClause ::= "group" "by" ( Variable ":=" )? Expression ( "," ( Variable ":=" )? Expression )* ( "decor" Variable ":=" Expression ( "," "decor" Variable ":=" Expression )* )? "with" VariableRef ( "," VariableRef )*
function GroupClause() {
AQLClause.call(this);
if (arguments.length == 0) {
// TODO Not sure which error to throw for an empty GroupBy but this should fail.
alert("Group Error");
this._properties["clause"] = null;
return this;
}
var expc = 0;
var expressions = [];
var variableRefs = [];
var isDecor = false;
while (expc < arguments.length) {
if (arguments[expc] instanceof AExpression) {
isDecor = false;
expressions.push(arguments[expc].val());
} else if (typeof arguments[expc] == "string") {
// Special keywords, decor & with
if (arguments[expc] == "decor") {
isDecor = true;
} else if (arguments[expc] == "with") {
isDecor = false;
expc++;
while (expc < arguments.length) {
variableRefs.push("$" + arguments[expc]);
expc++;
}
// Variables and variable refs
} else {
var nextc = expc + 1;
var expression = "";
if (isDecor) {
expression += "decor ";
isDecor = false;
}
expression += "$" + arguments[expc] + " := " + arguments[nextc].val();
expressions.push(expression);
expc++;
}
}
expc++;
}
this._properties["clause"] = "group by " + expressions.join(", ") + " with " + variableRefs.join(", ");
return this;
}
GroupClause.prototype = Object.create(AQLClause.prototype);
GroupClause.prototype.constructor = GroupClause;
// BooleanExpression
//
// TODO
function BooleanExpression(expression) {
this.value = expression;
alert("Debugging Bool: " + arguments.length + " " + expression);
}
BooleanExpression.prototype.val = function() {
return this.value;
}
// SetStatement
//
// Grammar
// "set" Identifier StringLiteral
function SetStatement (identifier, stringLiteral) {
AExpression.call(this);
var statement = "set " + identifier + ' "' + stringLiteral + '";';
AExpression.prototype.set.call(this, statement);
return this;
}
SetStatement.prototype = Object.create(AExpression.prototype);
SetStatement.prototype.constructor = SetStatement;
// Quantified Expression
//
// Grammar
// QuantifiedExpression ::= ( ( "some" ) | ( "every" ) ) Variable "in" Expression ( "," Variable "in" Expression )* "satisfies" Expression
//
// @param String some/every
// @param [AExpression]
// @param [Aexpression] satisfiesExpression
function QuantifiedExpression (keyword, expressions, satisfiesExpression) {
AExpression.call(this);
var expression = keyword + " ";
var varsInExpressions = [];
for (var varInExpression in expressions) {
varsInExpressions.push(varInExpression + " in " + expressions[varInExpression].val());
}
expression += varsInExpressions.join(", ") + " satisfies " + satisfiesExpression.val();
AExpression.prototype.set.call(this, expression);
return this;
}
QuantifiedExpression.prototype = Object.create(AExpression.prototype);
QuantifiedExpression.prototype.constructor = QuantifiedExpression;
QuantifiedExpression.prototype.val = function() {
var value = AExpression.prototype.val.call(this);
return "(" + value + ")";
};
// Functions that can be used to call core expressions/clauses more cleanly
function AFLWOGR () {
}
function AClause () {
}
function ALetClause () {
}
function AWhereClause () {
}
function AOrderbyClause () {
}
function AGroupClause () {
}
function ALimitClause () {
}
function ADistinctClause () {
}
function AVariable () {
}