Compare commits

...

2 Commits

Author SHA1 Message Date
Will Webberley
755ff6827b fixed merge commits in fetch from upstream master 2015-07-14 17:58:27 +00:00
Will Webberley
204440c5f2 fixing cenode-cestore intergatrion 2015-06-30 14:17:30 +00:00

277
cenode.js
View File

@ -20,13 +20,13 @@
* Library constants * Library constants
*/ */
var POST_SENTENCES_ENDPOINT = "/sentences"; var POST_SENTENCES_ENDPOINT = "/sentences";
var GET_CARDS_ENDPOINT = "/cards"; var GET_INSTANCES_ENDPOINT = "/instances";
var MODELS = { var MODELS = {
CORE : [ CORE : [
"conceptualise an ~ entity ~ E", "conceptualise an ~ entity ~ E",
"conceptualise an ~ imageable thing ~ I that has the value V as ~ image ~", "conceptualise an ~ imageable thing ~ I that has the value V as ~ image ~",
"conceptualise a ~ timestamp ~ T that is an entity", "conceptualise a ~ timestamp ~ T that is an entity",
"conceptualise an ~ agent ~ A that is an entity and has the value V as ~ address ~", "conceptualise an ~ agent ~ A that is an entity and has the value V as ~ address ~ and has the value W as ~ authorisation type ~ and has the value X as ~ encoded authorisation ~ and has the value Y as ~ get path ~ and has the value Z as ~ post path ~",
"conceptualise an ~ individual ~ I that is an ~ agent ~", "conceptualise an ~ individual ~ I that is an ~ agent ~",
"conceptualise a ~ card ~ C that is an entity and has the timestamp T as ~ timestamp ~ and has the value V as ~ content ~ and has the value W as ~ linked content ~ and has the value V as ~ number of keystrokes ~ and has the timestamp T as ~ start time ~ and has the value W as ~ submit time ~", "conceptualise a ~ card ~ C that is an entity and has the timestamp T as ~ timestamp ~ and has the value V as ~ content ~ and has the value W as ~ linked content ~ and has the value V as ~ number of keystrokes ~ and has the timestamp T as ~ start time ~ and has the value W as ~ submit time ~",
"conceptualise the card C ~ is to ~ the agent A and ~ is from ~ the agent B", "conceptualise the card C ~ is to ~ the agent A and ~ is from ~ the agent B",
@ -162,7 +162,9 @@ var MODELS = {
"there is a forwardall policy named 'p1' that has 'true' as all agents and has the timestamp '0' as start time and has 'true' as enabled", "there is a forwardall policy named 'p1' that has 'true' as all agents and has the timestamp '0' as start time and has 'true' as enabled",
"there is a building named 'North Building'", "there is a building named 'North Building'",
"there is a floor named '2nd Floor'", "there is a floor named '2nd Floor'",
"the room 'N215' is located in the building 'North Building' and is located on the floor '2nd Floor'" "the room 'N215' is located in the building 'North Building' and is located on the floor '2nd Floor'",
"there is an agent named 'Moira' that has 'https://sentinel.ita-ce.com/CeStoreWeb/stores/SHERLOCK' as address and has 'basic' as authorisation type and has 'bW9pcmE6Y3IxbXMwbg== ' as encoded authorisation and has '/special/instances-for-multiple-concepts' as get path and has 'sources/Moira?action=save' as post path",
"there is a tell policy named 'p2' that has 'true' as enabled and has the agent 'Moira' as target"
], ],
SHERLOCK_NODE : [ SHERLOCK_NODE : [
"there is an agent named 'Mycroft' that has 'http://cenode.sentinelstream.net' as address", "there is an agent named 'Mycroft' that has 'http://cenode.sentinelstream.net' as address",
@ -1672,6 +1674,85 @@ function CEAgent(n){
var unsent_tell_cards = {}; var unsent_tell_cards = {};
var unsent_ask_cards = {}; var unsent_ask_cards = {};
var handled_cards = []; var handled_cards = [];
/*
* Utility object to support network tasks.
*/
var net = {
get_remote_instances: function(agent, instance_types, data, callback){
data.conceptNames = instance_types.join(",");
if(util.on_client()){net.make_request_client("GET", agent, data, callback);}
else{net.make_request_node("GET", agent, data, callback);}
},
post_remote_sentences: function(agent, sentences, callback){
var data = sentences.join("\n");
if(util.on_client()){net.make_request_client("POST", agent, data, callback);}
else{net.make_request_node("POST", agent, data, callback);}
},
make_request_client: function(method, agent, data, callback){
var address = node.get_instance_value(agent, "address");
var path = node.get_instance_value(agent, method.toLowerCase()+" path");
if(method == "GET" && data != null){
var params = [];
for(key in data){
params.push(key+"="+data[key]);
}
path+="?"+params.join("&");
}
var xhr = new XMLHttpRequest();
xhr.open(method, address+path);
xhr.onreadystatechange = function(){
if(xhr.readyState==4 && (xhr.status==200 || xhr.status==302) && callback != null){
callback(xhr.responseText);
}
};
if(node.get_instance_value(agent, "authorisation type") == "basic"){
xhr.setRequestHeader("Authorization", "Basic "+node.get_instance_value(agent, "encoded authorisation"));
}
console.log(method+" "+agent.name+path);
if(method == "POST"){
xhr.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
xhr.send(data);
}
else if(method == "GET"){
xhr.setRequestHeader("Accept", "text/plain");
xhr.send();
}
},
make_request_node: function(method, agent, data, callback){
var http = require('http');
var address = node.get_instance_value(agent, "address");
var path = node.get_instance_value(agent, method.toLowerCase()+" path");
if(method == "GET" && data != null){
var params = [];
for(key in data){
params.push(key+"="+data[key]);
}
path+="?"+params.join("&");
}
var options = {
host: address,
path: path,
method: method,
headers: {'Content-Type': 'application/x-www-form-urlencoded', 'Accept': 'text/plain'}
};
if(node.get_instance_value(agent, "authorisation type") == "basic"){
options.headers.Authorization = "Basic "+node.get_instance_value(agent, "encoded authorisation");
}
console.log(method+" "+agent.name+path);
var req = http.request(options, function(response){
var body = '';
response.on('data', function(chunk){body+=data;});
response.on('end', function(){
body = decodeURIComponent(body.replace(/\+/g, ' '));
callback(body);
});
});
if(data != null){req.write(data);}
req.end();
}
}
this.set_name = function(n){ this.set_name = function(n){
name = n; name = n;
@ -1711,8 +1792,10 @@ function CEAgent(n){
for(var j = 0; j < ask_policies.length; j++){ for(var j = 0; j < ask_policies.length; j++){
if(node.get_instance_value(ask_policies[j], "enabled") == 'true'){ if(node.get_instance_value(ask_policies[j], "enabled") == 'true'){
var target_name = node.get_instance_value(ask_policies[j], "target").name; var target_name = node.get_instance_value(ask_policies[j], "target").name;
if(!(target_name in unsent_ask_cards)){unsent_ask_cards[target_name] = [];} if(target_name.toLowerCase() != name.toLowerCase()){
unsent_ask_cards[target_name].push(card); if(!(target_name in unsent_ask_cards)){unsent_ask_cards[target_name] = [];}
unsent_ask_cards[target_name].push(card);
}
} }
} }
@ -1739,8 +1822,10 @@ function CEAgent(n){
for(var j = 0; j < tell_policies.length; j++){ for(var j = 0; j < tell_policies.length; j++){
if(node.get_instance_value(tell_policies[j], "enabled") == 'true'){ if(node.get_instance_value(tell_policies[j], "enabled") == 'true'){
var target_name = node.get_instance_value(tell_policies[j], "target").name; var target_name = node.get_instance_value(tell_policies[j], "target").name;
if(!(target_name in unsent_tell_cards)){unsent_tell_cards[target_name] = [];} if(target_name.toLowerCase() != name.toLowerCase()){
unsent_tell_cards[target_name].push(card); if(!(target_name in unsent_tell_cards)){unsent_tell_cards[target_name] = [];}
unsent_tell_cards[target_name].push(card);
}
} }
} }
} }
@ -1821,21 +1906,22 @@ function CEAgent(n){
var target = node.get_instance_value(tell_policies[i], "target"); var target = node.get_instance_value(tell_policies[i], "target");
var cards = unsent_tell_cards[target.name]; var cards = unsent_tell_cards[target.name];
if(cards){ if(cards){
var data = ""; var sentences = [];
for(var j = 0; j < cards.length; j++){ for(var j = 0; j < cards.length; j++){
var card = cards[j]; var card = cards[j];
var from = node.get_instance_relationship(card, "is from"); var from = node.get_instance_relationship(card, "is from");
if(from.name.toLowerCase() != target.name.toLowerCase()){ // Don't send back a card sent from target agent var to = node.get_instance_relationship(card, "is to");
if(from.name.toLowerCase() != target.name.toLowerCase() && to.name.toLowerCase() != target.name.toLowerCase()){ // Don't send back a card sent from target agent
var rel = {}; var rel = {};
rel.target_name = target.name; rel.target_name = target.name;
rel.target_id = target.id; rel.target_id = target.id;
rel.label = "is to"; rel.label = "is to";
card.relationships.push(rel); card.relationships.push(rel);
data += node.get_instance_ce(card)+"\n"; sentences.push(node.get_instance_ce(card));
} }
} }
if(data != ""){ if(sentences.length > 0){
net.make_request("POST", node.get_instance_value(target, "address"), POST_SENTENCES_ENDPOINT, data, function(resp){ net.post_remote_sentences(target, sentences, function(resp){
unsent_tell_cards[target.name] = []; unsent_tell_cards[target.name] = [];
}); });
} }
@ -1849,7 +1935,7 @@ function CEAgent(n){
var target = node.get_instance_value(ask_policies[i], "target"); var target = node.get_instance_value(ask_policies[i], "target");
var cards = unsent_ask_cards[target.name]; var cards = unsent_ask_cards[target.name];
if(cards){ if(cards){
var data = ""; var sentences = [];
for(var j = 0; j < cards.length; j++){ for(var j = 0; j < cards.length; j++){
var card = cards[j]; var card = cards[j];
var from = node.get_instance_relationship(card, "is from"); var from = node.get_instance_relationship(card, "is from");
@ -1864,11 +1950,11 @@ function CEAgent(n){
rel2.label = "is from"; rel2.label = "is from";
card.relationships.push(rel); card.relationships.push(rel);
card.relationships.push(rel2); card.relationships.push(rel2);
data += node.get_instance_ce(card)+"\n"; sentences.push(node.get_instance_ce(card));
} }
} }
if(data != ""){ if(sentences != []){
net.make_request("POST", node.get_instance_value(target, "address"), POST_SENTENCES_ENDPOINT, data, function(resp){ net.post_remote_sentences(target, sentences, function(resp){
unsent_ask_cards[target.name] = []; unsent_ask_cards[target.name] = [];
}); });
} }
@ -1878,7 +1964,7 @@ function CEAgent(n){
// For each listen policy in place, make a GET request to get cards addressed to THIS agent, and add to node // For each listen policy in place, make a GET request to get cards addressed to THIS agent, and add to node
for(var i = 0; i < listen_policies.length; i++){ for(var i = 0; i < listen_policies.length; i++){
var target = node.get_instance_value(listen_policies[i], "target"); var target = node.get_instance_value(listen_policies[i], "target");
net.make_request("GET", node.get_instance_value(target, "address"), GET_CARDS_ENDPOINT+"?agent="+name, null, function(resp){ net.get_remote_instances(target, ["card"], {agent:name}, function(resp){
var cards = resp.split("\n"); var cards = resp.split("\n");
node.add_sentences(cards); node.add_sentences(cards);
}); });
@ -1902,32 +1988,35 @@ function CEAgent(n){
var card = cards[i]; var card = cards[i];
var to_agent = false; var to_agent = false;
var tos = node.get_instance_relationships(card, "is to"); var tos = node.get_instance_relationships(card, "is to");
var card_timestamp = node.get_instance_value(card, "timestamp").name; var card_timestamp = node.get_instance_value(card, "timestamp");
if(parseInt(card_timestamp) > parseInt(start_time)){ if(card_timestamp != null){
for(var j = 0; j < tos.length; j++){ card_timestamp = card_timestamp.name;
if(tos[j].name == name){ // If card sent to THIS agent if(parseInt(card_timestamp) > parseInt(start_time)){
to_agent = true; for(var j = 0; j < tos.length; j++){
break; if(tos[j].name == name){ // If card sent to THIS agent
} to_agent = true;
} break;
if(to_agent == true){
var from = node.get_instance_relationships(card, "is from")[0];
// Add each other agent as a recipient (if they aren't already), but not THIS agent or the original author
for(var j = 0; j < agents.length; j++){
var agent_is_recipient = false;
for(var k = 0; k < tos.length; k++){
if(tos[k].name.toLowerCase() == agents[j].name.toLowerCase()){
agent_is_recipient = true;
break;
}
} }
if(!agent_is_recipient && agents[j].name.toLowerCase() != name.toLowerCase() && agents[j].name.toLowerCase() != from.name.toLowerCase()){ }
var relationship = {}; if(to_agent == true){
relationship.label = "is to"; var from = node.get_instance_relationships(card, "is from")[0];
relationship.target_name = agents[j].name;
relationship.target_id = agents[j].id; // Add each other agent as a recipient (if they aren't already), but not THIS agent or the original author
card.relationships.push(relationship); for(var j = 0; j < agents.length; j++){
var agent_is_recipient = false;
for(var k = 0; k < tos.length; k++){
if(tos[k].name.toLowerCase() == agents[j].name.toLowerCase()){
agent_is_recipient = true;
break;
}
}
if(!agent_is_recipient && agents[j].name.toLowerCase() != name.toLowerCase() && agents[j].name.toLowerCase() != from.name.toLowerCase()){
var relationship = {};
relationship.label = "is to";
relationship.target_name = agents[j].name;
relationship.target_id = agents[j].id;
card.relationships.push(relationship);
}
} }
} }
} }
@ -2013,52 +2102,7 @@ var util = {
} }
} }
/*
* Utility object to support network tasks.
*/
var net = {
make_request: function(method, node_url, path, data, callback){
if(util.on_client()){net.make_request_client(method, node_url, path, data, callback);}
else{make_request_node(net.method, node_url, path, data, callback);}
},
make_request_client: function(method, node_url, path, data, callback){
console.log(method+" "+path);
var xhr = new XMLHttpRequest();
xhr.open(method, node_url+path);
xhr.onreadystatechange = function(){
if(xhr.readyState==4 && (xhr.status==200 || xhr.status==302) && callback != null){
callback(xhr.responseText);
}
};
if(data != null){
xhr.setRequestHeader("Content-type","application/x-www-form-urlencoded");
xhr.send(data);
}
else{
xhr.send();
}
},
make_request_node: function(method, node_url, path, data, callback){
var http = require('http');
var options = {
host: node_url,
path: path,
method: method,
headers: {'Content-Type': 'application/x-www-form-urlencoded'}
};
var req = http.request(options, function(response){
var body = '';
response.on('data', function(chunk){body+=data;});
response.on('end', function(){
body = decodeURIComponent(body.replace(/\+/g, ' '));
callback(body);
});
});
if(data != null){req.write(data);}
req.end();
}
}
// If running as a Node.js app, export CENode class and MODELS object. // If running as a Node.js app, export CENode class and MODELS object.
if(!util.on_client()){ if(!util.on_client()){
@ -2090,31 +2134,44 @@ if(!util.on_client() && require.main === module){
}); });
} }
function get_cards(request, response){ function get_instances(request, response){
var url = decodeURIComponent(request.url); var url = decodeURIComponent(request.url);
var agent_regex = url.match(/agent=(.*)/); var components = {};
var agent_str = null; url.replace(
new RegExp("([^?=&]+)(=([^&]*))?", "g"),
function($0, $1, $2, $3) { components[$1] = $3; }
);
var agents = []; var agents = [];
if(agent_regex != null){agent_str = agent_regex[1];} if(components.agent != null){
if(agent_str != null){ agents = components.agent.toLowerCase().split(",");
agents = agent_str.toLowerCase().split(","); }
var concept_names = [];
if(components.conceptNames != null){
concept_names = components.conceptNames.toLowerCase().split(",");
}
var instances = [];
for(var i = 0; i < concept_names.length; i++){
instances = instances.concat(node.get_instances(concept_names[i], true));
} }
var cards = node.get_instances("card", true);
var s = ""; var s = "";
for(var i = 0; i < cards.length; i++){ for(var i = 0; i < instances.length; i++){
if(agents == null || agents.length == 0){ if(agents.length > 0){
s += node.get_instance_ce(cards[i])+"\n"; var tos = node.get_instance_relationships(instances[i], "is to");
} if(tos.length > 0){
else{ for(var j = 0; j < tos.length; j++){
var tos = node.get_instance_relationships(cards[i], "is to"); for(var k = 0; k < agents.length; k++){
for(var j = 0; j < tos.length; j++){ if(tos[j].name.toLowerCase() == agents[k]){
for(var k = 0; k < agents.length; k++){ s += node.get_instance_ce(instances[i])+"\n";
if(tos[j].name.toLowerCase() == agents[k]){ }
s += node.get_instance_ce(cards[i])+"\n";
break;
} }
} }
} }
else{
s += node.get_instance_ce(instances[i])+"\n";
}
}
else{
s += node.get_instance_ce(instances[i])+"\n";
} }
} }
response.write(s); response.write(s);
@ -2134,7 +2191,7 @@ if(!util.on_client() && require.main === module){
s+='<p>Add CE sentences to the node:</p><form action="/ui/sentences" enctype="application/x-www-form-urlencoded" method="POST"><textarea name="sentence" style="width:95%;height:100px;"></textarea><br /><br /><input type="submit" /></form></div>'; s+='<p>Add CE sentences to the node:</p><form action="/ui/sentences" enctype="application/x-www-form-urlencoded" method="POST"><textarea name="sentence" style="width:95%;height:100px;"></textarea><br /><br /><input type="submit" /></form></div>';
s+='<div style="width:48%;float:left;"><h2>Node settings</h2><p>Update local agent name:</p><form method="POST" action="/agent_name"><input type="text" name="name" value="'+node.get_agent_name()+'" /><input type="submit" /></form>'; s+='<div style="width:48%;float:left;"><h2>Node settings</h2><p>Update local agent name:</p><form method="POST" action="/agent_name"><input type="text" name="name" value="'+node.get_agent_name()+'" /><input type="submit" /></form>';
s+='<p>Other options:</p><button onclick="window.location=\'/reset\';">Empty model</button>'; s+='<p>Other options:</p><button onclick="window.location=\'/reset\';">Empty model</button>';
s+='<p>Available endpoints on this node server instance:</p><p style="font-family:\'monospace\';font-size:11px;">- POST '+POST_SENTENCES_ENDPOINT+' (body = newline-delimited set of sentences)<br />- GET '+GET_CARDS_ENDPOINT+'?agent=NAME (get all known cards sent to NAME)</p>'; s+='<p>Available endpoints on this node server instance:</p><p style="font-family:\'monospace\';font-size:11px;">- POST '+POST_SENTENCES_ENDPOINT+' (body = newline-delimited set of sentences)<br />- GET '+GET_INSTANCES_ENDPOINT+'?conceptNames=CONCEPTS&agent=NAMES (get all instances with concept name in CONCEPT and, if a card, only those sent to NAMES)</p>';
s+='</div><div style="clear:both;"></div>'; s+='</div><div style="clear:both;"></div>';
s+='<div style="display:inline-block;width:45%;float:left;"><h2>Concepts</h2><textarea style="width:100%;height:300px;" readonly>'+JSON.stringify(con, undefined, 2)+'</textarea></div>'; s+='<div style="display:inline-block;width:45%;float:left;"><h2>Concepts</h2><textarea style="width:100%;height:300px;" readonly>'+JSON.stringify(con, undefined, 2)+'</textarea></div>';
s+='<div style="display:inline-block;width:45%;float:right;"><h2>Instances</h2><textarea style="width:100%;height:300px;" readonly>'+JSON.stringify(ins, undefined, 2)+'</textarea></div>'; s+='<div style="display:inline-block;width:45%;float:right;"><h2>Instances</h2><textarea style="width:100%;height:300px;" readonly>'+JSON.stringify(ins, undefined, 2)+'</textarea></div>';
@ -2142,9 +2199,9 @@ if(!util.on_client() && require.main === module){
response.writeHead(200, {"Content-Type": "text/html"}); response.writeHead(200, {"Content-Type": "text/html"});
response.end(s); response.end(s);
} }
else if(request.url.indexOf(GET_CARDS_ENDPOINT) == 0){ else if(request.url.indexOf(GET_INSTANCES_ENDPOINT) == 0){
response.writeHead(200, {"Content-Type": "text/ce"}); response.writeHead(200, {"Content-Type": "text/plain"});
get_cards(request, response); get_instances(request, response);
} }
else if(request.url == "/reset"){ else if(request.url == "/reset"){
node.reset_all(); node.reset_all();
@ -2158,7 +2215,7 @@ if(!util.on_client() && require.main === module){
} }
else if(request.method == "POST"){ else if(request.method == "POST"){
if(request.url == POST_SENTENCES_ENDPOINT){ if(request.url == POST_SENTENCES_ENDPOINT){
response.writeHead(200, {"Content-Type": "text/ce"}); response.writeHead(200, {"Content-Type": "text/plain"});
post_sentences(request, response); post_sentences(request, response);
} }
else if(request.url == "/ui/sentences"){ else if(request.url == "/ui/sentences"){