kopia lustrzana https://github.com/backface/turtlestitch
659 wiersze
20 KiB
JavaScript
659 wiersze
20 KiB
JavaScript
/*
|
|
|
|
cloud.js
|
|
|
|
a backend API for SNAP!
|
|
|
|
written by Jens Mönig
|
|
|
|
Copyright (C) 2015 by Jens Mönig
|
|
|
|
This file is part of Snap!.
|
|
|
|
Snap! is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Affero General Public License as
|
|
published by the Free Software Foundation, either version 3 of
|
|
the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
// Global settings /////////////////////////////////////////////////////
|
|
|
|
/*global modules, IDE_Morph, SnapSerializer, hex_sha512, alert, nop,
|
|
localize*/
|
|
|
|
modules.cloud = '2015-December-15';
|
|
|
|
// Global stuff
|
|
|
|
var Cloud;
|
|
var SnapCloud = new Cloud(
|
|
'https://snap.apps.miosoft.com/SnapCloud'
|
|
);
|
|
|
|
// Cloud /////////////////////////////////////////////////////////////
|
|
|
|
function Cloud(url) {
|
|
this.username = null;
|
|
this.password = null; // hex_sha512 hashed
|
|
this.url = url;
|
|
this.session = null;
|
|
this.limo = null;
|
|
this.route = null;
|
|
this.api = {};
|
|
}
|
|
|
|
Cloud.prototype.clear = function () {
|
|
this.username = null;
|
|
this.password = null;
|
|
this.session = null;
|
|
this.limo = null;
|
|
this.route = null;
|
|
this.api = {};
|
|
};
|
|
|
|
Cloud.prototype.hasProtocol = function () {
|
|
return this.url.toLowerCase().indexOf('http') === 0;
|
|
};
|
|
|
|
Cloud.prototype.setRoute = function (username) {
|
|
var routes = 20,
|
|
userNum = 0,
|
|
i;
|
|
|
|
for (i = 0; i < username.length; i += 1) {
|
|
userNum += username.charCodeAt(i);
|
|
}
|
|
userNum = userNum % routes + 1;
|
|
this.route = '.sc1m' +
|
|
(userNum < 10 ? '0' : '') +
|
|
userNum;
|
|
};
|
|
|
|
// Cloud: Snap! API
|
|
|
|
Cloud.prototype.signup = function (
|
|
username,
|
|
email,
|
|
callBack,
|
|
errorCall
|
|
) {
|
|
// both callBack and errorCall are two-argument functions
|
|
var request = new XMLHttpRequest(),
|
|
myself = this;
|
|
try {
|
|
request.open(
|
|
"GET",
|
|
(this.hasProtocol() ? '' : 'http://')
|
|
+ this.url + 'SignUp'
|
|
+ '?Username='
|
|
+ encodeURIComponent(username)
|
|
+ '&Email='
|
|
+ encodeURIComponent(email),
|
|
true
|
|
);
|
|
request.setRequestHeader(
|
|
"Content-Type",
|
|
"application/x-www-form-urlencoded"
|
|
);
|
|
request.withCredentials = true;
|
|
request.onreadystatechange = function () {
|
|
if (request.readyState === 4) {
|
|
if (request.responseText) {
|
|
if (request.responseText.indexOf('ERROR') === 0) {
|
|
errorCall.call(
|
|
this,
|
|
request.responseText,
|
|
'Signup'
|
|
);
|
|
} else {
|
|
callBack.call(
|
|
null,
|
|
request.responseText,
|
|
'Signup'
|
|
);
|
|
}
|
|
} else {
|
|
errorCall.call(
|
|
null,
|
|
myself.url + 'SignUp',
|
|
localize('could not connect to:')
|
|
);
|
|
}
|
|
}
|
|
};
|
|
request.send(null);
|
|
} catch (err) {
|
|
errorCall.call(this, err.toString(), 'Snap!Cloud');
|
|
}
|
|
};
|
|
|
|
Cloud.prototype.getPublicProject = function (
|
|
id,
|
|
callBack,
|
|
errorCall
|
|
) {
|
|
// id is Username=username&projectName=projectname,
|
|
// where the values are url-component encoded
|
|
// callBack is a single argument function, errorCall take two args
|
|
var request = new XMLHttpRequest(),
|
|
myself = this;
|
|
try {
|
|
request.open(
|
|
"GET",
|
|
(this.hasProtocol() ? '' : 'http://')
|
|
+ this.url + 'RawPublic'
|
|
+ '?'
|
|
+ id,
|
|
true
|
|
);
|
|
request.setRequestHeader(
|
|
"Content-Type",
|
|
"application/x-www-form-urlencoded"
|
|
);
|
|
request.withCredentials = true;
|
|
request.onreadystatechange = function () {
|
|
if (request.readyState === 4) {
|
|
if (request.responseText) {
|
|
if (request.responseText.indexOf('ERROR') === 0) {
|
|
errorCall.call(
|
|
this,
|
|
request.responseText
|
|
);
|
|
} else {
|
|
callBack.call(
|
|
null,
|
|
request.responseText
|
|
);
|
|
}
|
|
} else {
|
|
errorCall.call(
|
|
null,
|
|
myself.url + 'Public',
|
|
localize('could not connect to:')
|
|
);
|
|
}
|
|
}
|
|
};
|
|
request.send(null);
|
|
} catch (err) {
|
|
errorCall.call(this, err.toString(), 'Snap!Cloud');
|
|
}
|
|
};
|
|
|
|
Cloud.prototype.resetPassword = function (
|
|
username,
|
|
callBack,
|
|
errorCall
|
|
) {
|
|
// both callBack and errorCall are two-argument functions
|
|
var request = new XMLHttpRequest(),
|
|
myself = this;
|
|
try {
|
|
request.open(
|
|
"GET",
|
|
(this.hasProtocol() ? '' : 'http://')
|
|
+ this.url + 'ResetPW'
|
|
+ '?Username='
|
|
+ encodeURIComponent(username),
|
|
true
|
|
);
|
|
request.setRequestHeader(
|
|
"Content-Type",
|
|
"application/x-www-form-urlencoded"
|
|
);
|
|
request.withCredentials = true;
|
|
request.onreadystatechange = function () {
|
|
if (request.readyState === 4) {
|
|
if (request.responseText) {
|
|
if (request.responseText.indexOf('ERROR') === 0) {
|
|
errorCall.call(
|
|
this,
|
|
request.responseText,
|
|
'Reset Password'
|
|
);
|
|
} else {
|
|
callBack.call(
|
|
null,
|
|
request.responseText,
|
|
'Reset Password'
|
|
);
|
|
}
|
|
} else {
|
|
errorCall.call(
|
|
null,
|
|
myself.url + 'ResetPW',
|
|
localize('could not connect to:')
|
|
);
|
|
}
|
|
}
|
|
};
|
|
request.send(null);
|
|
} catch (err) {
|
|
errorCall.call(this, err.toString(), 'Snap!Cloud');
|
|
}
|
|
};
|
|
|
|
Cloud.prototype.login = function (
|
|
username,
|
|
password,
|
|
callBack,
|
|
errorCall
|
|
) {
|
|
// both callBack and errorCall are two-argument functions
|
|
var request = new XMLHttpRequest(),
|
|
usr = JSON.stringify({'__h': password, '__u': username}),
|
|
myself = this;
|
|
this.setRoute(username);
|
|
try {
|
|
request.open(
|
|
"POST",
|
|
(this.hasProtocol() ? '' : 'http://') +
|
|
this.url +
|
|
'?SESSIONGLUE=' +
|
|
this.route,
|
|
true
|
|
);
|
|
request.setRequestHeader(
|
|
"Content-Type",
|
|
"application/json; charset=utf-8"
|
|
);
|
|
// glue this session to a route:
|
|
request.setRequestHeader('SESSIONGLUE', this.route);
|
|
request.withCredentials = true;
|
|
request.onreadystatechange = function () {
|
|
if (request.readyState === 4) {
|
|
if (request.responseText) {
|
|
myself.api = myself.parseAPI(request.responseText);
|
|
myself.session = request.getResponseHeader('MioCracker')
|
|
.split(';')[0];
|
|
// set the cookie identifier:
|
|
myself.limo = this.getResponseHeader("miocracker")
|
|
.substring(
|
|
9,
|
|
this.getResponseHeader("miocracker").indexOf("=")
|
|
);
|
|
if (myself.api.logout) {
|
|
myself.username = username;
|
|
myself.password = password;
|
|
callBack.call(null, myself.api, 'Snap!Cloud');
|
|
} else {
|
|
errorCall.call(
|
|
null,
|
|
request.responseText,
|
|
'connection failed'
|
|
);
|
|
}
|
|
} else {
|
|
errorCall.call(
|
|
null,
|
|
myself.url,
|
|
localize('could not connect to:')
|
|
);
|
|
}
|
|
}
|
|
};
|
|
request.send(usr);
|
|
} catch (err) {
|
|
errorCall.call(this, err.toString(), 'Snap!Cloud');
|
|
}
|
|
};
|
|
|
|
Cloud.prototype.reconnect = function (
|
|
callBack,
|
|
errorCall
|
|
) {
|
|
if (!(this.username && this.password)) {
|
|
this.message('You are not logged in');
|
|
return;
|
|
}
|
|
this.login(
|
|
this.username,
|
|
this.password,
|
|
callBack,
|
|
errorCall
|
|
);
|
|
};
|
|
|
|
Cloud.prototype.saveProject = function (ide, callBack, errorCall) {
|
|
var myself = this,
|
|
pdata,
|
|
media,
|
|
size,
|
|
mediaSize;
|
|
|
|
ide.serializer.isCollectingMedia = true;
|
|
pdata = ide.serializer.serialize(ide.stage);
|
|
media = ide.hasChangedMedia ?
|
|
ide.serializer.mediaXML(ide.projectName) : null;
|
|
ide.serializer.isCollectingMedia = false;
|
|
ide.serializer.flushMedia();
|
|
|
|
mediaSize = media ? media.length : 0;
|
|
size = pdata.length + mediaSize;
|
|
if (mediaSize > 10485760) {
|
|
new DialogBoxMorph().inform(
|
|
'Snap!Cloud - Cannot Save Project',
|
|
'The media inside this project exceeds 10 MB.\n' +
|
|
'Please reduce the size of costumes or sounds.\n',
|
|
ide.world(),
|
|
ide.cloudIcon(null, new Color(180, 0, 0))
|
|
);
|
|
throw new Error('Project media exceeds 10 MB size limit');
|
|
}
|
|
|
|
// check if serialized data can be parsed back again
|
|
try {
|
|
ide.serializer.parse(pdata);
|
|
} catch (err) {
|
|
ide.showMessage('Serialization of program data failed:\n' + err);
|
|
throw new Error('Serialization of program data failed:\n' + err);
|
|
}
|
|
if (media !== null) {
|
|
try {
|
|
ide.serializer.parse(media);
|
|
} catch (err) {
|
|
ide.showMessage('Serialization of media failed:\n' + err);
|
|
throw new Error('Serialization of media failed:\n' + err);
|
|
}
|
|
}
|
|
ide.serializer.isCollectingMedia = false;
|
|
ide.serializer.flushMedia();
|
|
|
|
ide.showMessage('Uploading ' + Math.round(size / 1024) + ' KB...');
|
|
myself.reconnect(
|
|
function () {
|
|
myself.callService(
|
|
'saveProject',
|
|
function (response, url) {
|
|
callBack.call(null, response, url);
|
|
myself.disconnect();
|
|
ide.hasChangedMedia = false;
|
|
},
|
|
errorCall,
|
|
[
|
|
ide.projectName,
|
|
pdata,
|
|
media,
|
|
pdata.length,
|
|
media ? media.length : 0
|
|
]
|
|
);
|
|
},
|
|
errorCall
|
|
);
|
|
};
|
|
|
|
Cloud.prototype.getProjectList = function (callBack, errorCall) {
|
|
var myself = this;
|
|
this.reconnect(
|
|
function () {
|
|
myself.callService(
|
|
'getProjectList',
|
|
function (response, url) {
|
|
callBack.call(null, response, url);
|
|
myself.disconnect();
|
|
},
|
|
errorCall
|
|
);
|
|
},
|
|
errorCall
|
|
);
|
|
};
|
|
|
|
Cloud.prototype.changePassword = function (
|
|
oldPW,
|
|
newPW,
|
|
callBack,
|
|
errorCall
|
|
) {
|
|
var myself = this;
|
|
this.reconnect(
|
|
function () {
|
|
myself.callService(
|
|
'changePassword',
|
|
function (response, url) {
|
|
callBack.call(null, response, url);
|
|
myself.disconnect();
|
|
},
|
|
errorCall,
|
|
[hex_sha512(oldPW), hex_sha512(newPW)]
|
|
);
|
|
},
|
|
errorCall
|
|
);
|
|
};
|
|
|
|
Cloud.prototype.logout = function (callBack, errorCall) {
|
|
this.clear();
|
|
this.callService(
|
|
'logout',
|
|
callBack,
|
|
errorCall
|
|
);
|
|
};
|
|
|
|
Cloud.prototype.disconnect = function () {
|
|
this.callService(
|
|
'logout',
|
|
nop,
|
|
nop
|
|
);
|
|
};
|
|
|
|
// Cloud: backend communication
|
|
|
|
Cloud.prototype.callURL = function (url, callBack, errorCall) {
|
|
// both callBack and errorCall are optional two-argument functions
|
|
var request = new XMLHttpRequest(),
|
|
stickyUrl,
|
|
myself = this;
|
|
try {
|
|
// set the Limo. Also set the glue as a query paramter for backup.
|
|
stickyUrl = url +
|
|
'&SESSIONGLUE=' +
|
|
this.route +
|
|
'&_Limo=' +
|
|
this.limo;
|
|
request.open('GET', stickyUrl, true);
|
|
request.withCredentials = true;
|
|
request.setRequestHeader(
|
|
"Content-Type",
|
|
"application/x-www-form-urlencoded"
|
|
);
|
|
request.setRequestHeader('MioCracker', this.session);
|
|
// Set the glue as a request header.
|
|
request.setRequestHeader('SESSIONGLUE', this.route);
|
|
request.onreadystatechange = function () {
|
|
if (request.readyState === 4) {
|
|
if (request.responseText) {
|
|
var responseList = myself.parseResponse(
|
|
request.responseText
|
|
);
|
|
callBack.call(null, responseList, url);
|
|
} else {
|
|
errorCall.call(
|
|
null,
|
|
url,
|
|
'no response from:'
|
|
);
|
|
}
|
|
}
|
|
};
|
|
request.send(null);
|
|
} catch (err) {
|
|
errorCall.call(this, err.toString(), url);
|
|
}
|
|
};
|
|
|
|
Cloud.prototype.callService = function (
|
|
serviceName,
|
|
callBack,
|
|
errorCall,
|
|
args
|
|
) {
|
|
// both callBack and errorCall are optional two-argument functions
|
|
var request = new XMLHttpRequest(),
|
|
service = this.api[serviceName],
|
|
myself = this,
|
|
stickyUrl,
|
|
postDict;
|
|
|
|
if (!this.session) {
|
|
errorCall.call(null, 'You are not connected', 'Cloud');
|
|
return;
|
|
}
|
|
if (!service) {
|
|
errorCall.call(
|
|
null,
|
|
'service ' + serviceName + ' is not available',
|
|
'API'
|
|
);
|
|
return;
|
|
}
|
|
if (args && args.length > 0) {
|
|
postDict = {};
|
|
service.parameters.forEach(function (parm, idx) {
|
|
postDict[parm] = args[idx];
|
|
});
|
|
}
|
|
try {
|
|
stickyUrl = this.url +
|
|
'/' +
|
|
service.url +
|
|
'&SESSIONGLUE=' +
|
|
this.route +
|
|
'&_Limo=' +
|
|
this.limo;
|
|
request.open(service.method, stickyUrl, true);
|
|
request.withCredentials = true;
|
|
request.setRequestHeader(
|
|
"Content-Type",
|
|
"application/x-www-form-urlencoded"
|
|
);
|
|
request.setRequestHeader('MioCracker', this.session);
|
|
request.setRequestHeader('SESSIONGLUE', this.route);
|
|
request.onreadystatechange = function () {
|
|
if (request.readyState === 4) {
|
|
var responseList = [];
|
|
if (request.responseText &&
|
|
request.responseText.indexOf('ERROR') === 0) {
|
|
errorCall.call(
|
|
this,
|
|
request.responseText,
|
|
localize('Service:') + ' ' + localize(serviceName)
|
|
);
|
|
return;
|
|
}
|
|
if (serviceName === 'login') {
|
|
myself.api = myself.parseAPI(request.responseText);
|
|
}
|
|
if (serviceName === 'getRawProject') {
|
|
responseList = request.responseText;
|
|
} else {
|
|
responseList = myself.parseResponse(
|
|
request.responseText
|
|
);
|
|
}
|
|
callBack.call(null, responseList, service.url);
|
|
}
|
|
};
|
|
request.send(this.encodeDict(postDict));
|
|
} catch (err) {
|
|
errorCall.call(this, err.toString(), service.url);
|
|
}
|
|
};
|
|
|
|
// Cloud: payload transformation
|
|
|
|
Cloud.prototype.parseAPI = function (src) {
|
|
var api = {},
|
|
services;
|
|
services = src.split(" ");
|
|
services.forEach(function (service) {
|
|
var entries = service.split("&"),
|
|
serviceDescription = {},
|
|
parms;
|
|
entries.forEach(function (entry) {
|
|
var pair = entry.split("="),
|
|
key = decodeURIComponent(pair[0]).toLowerCase(),
|
|
val = decodeURIComponent(pair[1]);
|
|
if (key === "service") {
|
|
api[val] = serviceDescription;
|
|
} else if (key === "parameters") {
|
|
parms = val.split(",");
|
|
if (!(parms.length === 1 && !parms[0])) {
|
|
serviceDescription.parameters = parms;
|
|
}
|
|
} else {
|
|
serviceDescription[key] = val;
|
|
}
|
|
});
|
|
});
|
|
return api;
|
|
};
|
|
|
|
Cloud.prototype.parseResponse = function (src) {
|
|
var ans = [],
|
|
lines;
|
|
if (!src) {return ans; }
|
|
lines = src.split(" ");
|
|
lines.forEach(function (service) {
|
|
var entries = service.split("&"),
|
|
dict = {};
|
|
entries.forEach(function (entry) {
|
|
var pair = entry.split("="),
|
|
key = decodeURIComponent(pair[0]),
|
|
val = decodeURIComponent(pair[1]);
|
|
dict[key] = val;
|
|
});
|
|
ans.push(dict);
|
|
});
|
|
return ans;
|
|
};
|
|
|
|
Cloud.prototype.parseDict = function (src) {
|
|
var dict = {};
|
|
if (!src) {return dict; }
|
|
src.split("&").forEach(function (entry) {
|
|
var pair = entry.split("="),
|
|
key = decodeURIComponent(pair[0]),
|
|
val = decodeURIComponent(pair[1]);
|
|
dict[key] = val;
|
|
});
|
|
return dict;
|
|
};
|
|
|
|
Cloud.prototype.encodeDict = function (dict) {
|
|
var str = '',
|
|
pair,
|
|
key;
|
|
if (!dict) {return null; }
|
|
for (key in dict) {
|
|
if (dict.hasOwnProperty(key)) {
|
|
pair = encodeURIComponent(key)
|
|
+ '='
|
|
+ encodeURIComponent(dict[key]);
|
|
if (str.length > 0) {
|
|
str += '&';
|
|
}
|
|
str += pair;
|
|
}
|
|
}
|
|
return str;
|
|
};
|
|
|
|
// Cloud: user messages (to be overridden)
|
|
|
|
Cloud.prototype.message = function (string) {
|
|
alert(string);
|
|
};
|