2010-10-20 20:11:15 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
Copyright 2010, Google Inc.
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are
|
|
|
|
met:
|
|
|
|
|
2011-08-04 22:37:14 +02:00
|
|
|
* Redistributions of source code must retain the above copyright
|
2010-10-20 20:11:15 +02:00
|
|
|
notice, this list of conditions and the following disclaimer.
|
2011-08-04 22:37:14 +02:00
|
|
|
* Redistributions in binary form must reproduce the above
|
2010-10-20 20:11:15 +02:00
|
|
|
copyright notice, this list of conditions and the following disclaimer
|
|
|
|
in the documentation and/or other materials provided with the
|
|
|
|
distribution.
|
2011-08-04 22:37:14 +02:00
|
|
|
* Neither the name of Google Inc. nor the names of its
|
2010-10-20 20:11:15 +02:00
|
|
|
contributors may be used to endorse or promote products derived from
|
|
|
|
this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
2011-08-04 22:37:14 +02:00
|
|
|
*/
|
2010-10-20 20:11:15 +02:00
|
|
|
|
2010-10-08 03:39:38 +02:00
|
|
|
var SchemaAlignment = {};
|
|
|
|
|
|
|
|
SchemaAlignment.autoAlign = function() {
|
2011-08-04 22:37:14 +02:00
|
|
|
var protograph = {};
|
|
|
|
|
|
|
|
var columns = theProject.columnModel.columns;
|
|
|
|
|
|
|
|
var typedCandidates = [];
|
|
|
|
var candidates = [];
|
|
|
|
|
|
|
|
for (var c = 0; c < columns.length; c++) {
|
|
|
|
var column = columns[c];
|
|
|
|
var typed = (column.reconConfig) &&
|
|
|
|
ReconciliationManager.isFreebaseIdOrMid(column.reconConfig.identifierSpace) &&
|
|
|
|
ReconciliationManager.isFreebaseId(column.reconConfig.schemaSpace);
|
|
|
|
|
|
|
|
var candidate = {
|
|
|
|
status: "unbound",
|
|
|
|
typed: typed,
|
|
|
|
index: c,
|
|
|
|
column: column
|
|
|
|
};
|
|
|
|
|
|
|
|
candidates.push(candidate);
|
|
|
|
if (typed) {
|
|
|
|
typedCandidates.push(candidate);
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (typedCandidates.length > 0) {
|
|
|
|
|
|
|
|
} else {
|
|
|
|
var queries = {};
|
|
|
|
for (var i = 0; i < candidates.length; i++) {
|
|
|
|
var candidate = candidates[i];
|
|
|
|
var name = SchemaAlignment._cleanName(candidate.column.name);
|
|
|
|
var key = "t" + i + ":search";
|
|
|
|
queries[key] = {
|
|
|
|
"query" : name,
|
|
|
|
"limit" : 10,
|
|
|
|
"type" : "/type/type,/type/property",
|
|
|
|
"type_strict" : "any"
|
|
|
|
};
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
|
|
|
|
SchemaAlignment._batchSearch(queries, function(result) {
|
|
|
|
console.log(result);
|
|
|
|
});
|
|
|
|
}
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SchemaAlignment._batchSearch = function(queries, onDone) {
|
2011-08-04 22:37:14 +02:00
|
|
|
var keys = [];
|
|
|
|
for (var n in queries) {
|
|
|
|
if (queries.hasOwnProperty(n)) {
|
|
|
|
keys.push(n);
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var result = {};
|
|
|
|
var args = [];
|
|
|
|
var makeBatch = function(keyBatch) {
|
|
|
|
var batch = {};
|
|
|
|
for (var k = 0; k < keyBatch.length; k++) {
|
|
|
|
var key = keyBatch[k];
|
|
|
|
batch[key] = queries[key];
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
|
2013-03-12 22:46:18 +01:00
|
|
|
// TODO: New API doesn't accept multiple queries
|
|
|
|
// args.push("https://www.googleapis.com/freebase/v1/search?key=" + Freebase.API_KEY + "&" +
|
2013-05-26 14:02:12 +02:00
|
|
|
args.push("http://api.freebase.com/api/service/search?" + // FIXME:
|
2011-08-04 22:37:14 +02:00
|
|
|
$.param({ "queries" : JSON.stringify(batch) }) + "&callback=?");
|
|
|
|
|
|
|
|
args.push(null); // no data
|
|
|
|
args.push(function(data) {
|
|
|
|
for (var k = 0; k < keyBatch.length; k++) {
|
|
|
|
var key = keyBatch[k];
|
|
|
|
result[key] = data[key];
|
|
|
|
}
|
2010-10-08 03:39:38 +02:00
|
|
|
});
|
2011-08-04 22:37:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
for (var i = 0; i < keys.length; i += 10) {
|
|
|
|
makeBatch(keys.slice(i, i + 10));
|
|
|
|
}
|
|
|
|
|
|
|
|
args.push(function() {
|
|
|
|
onDone(result);
|
|
|
|
});
|
|
|
|
|
|
|
|
Ajax.chainGetJSON.apply(null, args);
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SchemaAlignment._cleanName = function(s) {
|
2011-08-04 22:37:14 +02:00
|
|
|
return s.replace(/\W/g, " ").replace(/\s+/g, " ").toLowerCase();
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SchemaAlignment.createNewRootNode = function() {
|
2011-08-04 22:37:14 +02:00
|
|
|
var rootNode = null;
|
|
|
|
var links = [];
|
|
|
|
var columns = theProject.columnModel.columns;
|
|
|
|
for (var i = 0; i < columns.length; i++) {
|
|
|
|
var column = columns[i];
|
|
|
|
var target = {
|
|
|
|
nodeType: "cell-as-topic",
|
|
|
|
columnName: column.name,
|
|
|
|
createForNoReconMatch: true
|
|
|
|
};
|
|
|
|
if ((column.reconConfig) &&
|
|
|
|
ReconciliationManager.isFreebaseIdOrMid(column.reconConfig.identifierSpace) &&
|
|
|
|
ReconciliationManager.isFreebaseId(column.reconConfig.schemaSpace) &&
|
|
|
|
(column.reconConfig.type)) {
|
|
|
|
|
|
|
|
target.type = {
|
|
|
|
id: column.reconConfig.type.id,
|
|
|
|
name: column.reconConfig.type.name
|
|
|
|
};
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
|
|
|
|
if (column.name == theProject.columnModel.keyColumnName) {
|
|
|
|
rootNode = target;
|
|
|
|
} else {
|
|
|
|
links.push({
|
|
|
|
property: null,
|
|
|
|
target: target
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rootNode = rootNode || { nodeType: "cell-as-topic" };
|
|
|
|
rootNode.links = links;
|
|
|
|
|
|
|
|
return rootNode;
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
function SchemaAlignmentDialog(protograph, onDone) {
|
2011-08-04 22:37:14 +02:00
|
|
|
this._onDone = onDone;
|
|
|
|
this._hasUnsavedChanges = false;
|
|
|
|
|
|
|
|
this._createDialog();
|
|
|
|
this._reset(protograph, true);
|
2010-10-25 03:07:25 +02:00
|
|
|
}
|
|
|
|
|
2010-10-25 22:05:48 +02:00
|
|
|
SchemaAlignmentDialog.prototype._reset = function(protograph, initial) {
|
2011-08-04 22:37:14 +02:00
|
|
|
this._originalProtograph = protograph || { rootNodes: [] };
|
|
|
|
this._protograph = cloneDeep(this._originalProtograph); // this is what can be munched on
|
|
|
|
|
|
|
|
if (!this._protograph.rootNodes.length) {
|
|
|
|
this._protograph.rootNodes.push(SchemaAlignment.createNewRootNode());
|
|
|
|
}
|
|
|
|
|
|
|
|
$(this._nodeTable).empty();
|
|
|
|
|
|
|
|
this._nodeUIs = [];
|
|
|
|
for (var i = 0; i < this._protograph.rootNodes.length; i++) {
|
|
|
|
this._nodeUIs.push(new SchemaAlignmentDialog.UINode(
|
|
|
|
this,
|
|
|
|
this._protograph.rootNodes[i],
|
|
|
|
this._nodeTable,
|
|
|
|
{
|
|
|
|
expanded: true,
|
|
|
|
mustBeCellTopic: true
|
|
|
|
}
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
this.preview(initial);
|
2010-10-25 03:07:25 +02:00
|
|
|
};
|
2010-10-08 03:39:38 +02:00
|
|
|
|
2010-10-25 03:07:25 +02:00
|
|
|
SchemaAlignmentDialog.prototype._save = function(onDone) {
|
2011-08-04 22:37:14 +02:00
|
|
|
var self = this;
|
|
|
|
var protograph = this.getJSON();
|
|
|
|
|
|
|
|
Refine.postProcess(
|
|
|
|
"freebase",
|
|
|
|
"save-protograph",
|
|
|
|
{},
|
|
|
|
{ protograph: JSON.stringify(protograph) },
|
|
|
|
{},
|
|
|
|
{
|
|
|
|
onDone: function() {
|
|
|
|
theProject.overlayModels.freebaseProtograph = protograph;
|
|
|
|
|
|
|
|
self._elmts.statusIndicator.hide();
|
|
|
|
self._hasUnsavedChanges = false;
|
|
|
|
|
|
|
|
if (onDone) onDone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
2010-10-25 03:07:25 +02:00
|
|
|
SchemaAlignmentDialog.prototype._createDialog = function() {
|
2011-08-04 22:37:14 +02:00
|
|
|
var self = this;
|
|
|
|
var frame = $(DOM.loadHTML("freebase", "scripts/dialogs/schema-alignment/schema-alignment-dialog.html"));
|
|
|
|
var elmts = this._elmts = DOM.bind(frame);
|
|
|
|
|
|
|
|
this._level = DialogSystem.showDialog(frame);
|
|
|
|
|
|
|
|
var dismiss = function() {
|
|
|
|
DialogSystem.dismissUntil(self._level - 1);
|
|
|
|
};
|
|
|
|
|
|
|
|
elmts.saveButton.click(function() {
|
|
|
|
self._save();
|
|
|
|
});
|
|
|
|
elmts.saveAndLoadButton.click(function() {
|
|
|
|
self._save(function() {
|
|
|
|
dismiss();
|
|
|
|
FreebaseExtension.handlers.loadIntoFreebase();
|
2010-10-25 22:05:48 +02:00
|
|
|
});
|
2011-08-04 22:37:14 +02:00
|
|
|
});
|
|
|
|
elmts.resetButton.click(function() {
|
|
|
|
self._reset(null);
|
|
|
|
});
|
|
|
|
elmts.closeButton.click(function() {
|
|
|
|
if (!self._hasUnsavedChanges || window.confirm("There are unsaved changes. Close anyway?")) {
|
|
|
|
dismiss();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#schema-alignment-tabs").tabs();
|
|
|
|
$("#schema-alignment-tabs-preview-mqllike").css("display", "");
|
|
|
|
$("#schema-alignment-tabs-preview-tripleloader").css("display", "");
|
|
|
|
|
|
|
|
this._previewPanes = $(".schema-alignment-dialog-preview");
|
|
|
|
|
|
|
|
this._canvas = $(".schema-alignment-dialog-canvas");
|
|
|
|
this._nodeTable = $('<table></table>').addClass("schema-alignment-table-layout").appendTo(this._canvas)[0];
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SchemaAlignmentDialog.prototype.getJSON = function() {
|
2011-08-04 22:37:14 +02:00
|
|
|
var rootNodes = [];
|
|
|
|
for (var i = 0; i < this._nodeUIs.length; i++) {
|
|
|
|
var node = this._nodeUIs[i].getJSON();
|
|
|
|
if (node !== null) {
|
|
|
|
rootNodes.push(node);
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
rootNodes: rootNodes
|
|
|
|
};
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
2010-10-25 22:05:48 +02:00
|
|
|
SchemaAlignmentDialog.prototype.preview = function(initial) {
|
2011-08-04 22:37:14 +02:00
|
|
|
var self = this;
|
|
|
|
|
|
|
|
this._previewPanes.empty();
|
|
|
|
if (!(initial)) {
|
|
|
|
this._elmts.statusIndicator.show().text("There are unsaved changes.");
|
|
|
|
this._hasUnsavedChanges = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
var protograph = this.getJSON();
|
|
|
|
$.post(
|
2012-10-13 19:47:08 +02:00
|
|
|
"command/freebase/preview-protograph?" + $.param({ project: theProject.id }),
|
2011-08-04 22:37:14 +02:00
|
|
|
{ protograph: JSON.stringify(protograph), engine: JSON.stringify(ui.browsingEngine.getJSON()) },
|
|
|
|
function(data) {
|
|
|
|
if ("mqllike" in data) {
|
|
|
|
$(self._previewPanes[0]).text(JSON.stringify(data.mqllike, null, 2));
|
|
|
|
}
|
|
|
|
if ("tripleloader" in data) {
|
|
|
|
$(self._previewPanes[1]).text(data.tripleloader);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"json"
|
|
|
|
);
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SchemaAlignmentDialog._findColumn = function(cellIndex) {
|
2011-08-04 22:37:14 +02:00
|
|
|
var columns = theProject.columnModel.columns;
|
|
|
|
for (var i = 0; i < columns.length; i++) {
|
|
|
|
var column = columns[i];
|
|
|
|
if (column.cellIndex == cellIndex) {
|
|
|
|
return column;
|
2010-10-08 03:39:38 +02:00
|
|
|
}
|
2011-08-04 22:37:14 +02:00
|
|
|
}
|
|
|
|
return null;
|
2010-10-08 03:39:38 +02:00
|
|
|
};
|