refactoring to be closer to the evaluator js in WeScheme
This commit is contained in:
parent
a371c3ffcf
commit
37551af8b1
|
@ -5,6 +5,7 @@
|
||||||
<script src="collects/library.js"></script>
|
<script src="collects/library.js"></script>
|
||||||
<script src="easyXDM-min.js"></script>
|
<script src="easyXDM-min.js"></script>
|
||||||
<script src="repl.js"></script>
|
<script src="repl.js"></script>
|
||||||
|
<script src="index.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>Repl experiment</h1>
|
<h1>Repl experiment</h1>
|
||||||
|
@ -14,6 +15,6 @@
|
||||||
|
|
||||||
<div id="output" style="width:500px; height:300px; border:1px solid black; overflow:scroll"></div>
|
<div id="output" style="width:500px; height:300px; border:1px solid black; overflow:scroll"></div>
|
||||||
<br/>
|
<br/>
|
||||||
<input id="repl" type="text" style="width:500px"></input><img id="break" src="break.png"/>
|
<input id="prompt" type="text" style="width:500px"></input><img id="break" src="break.png"/>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
69
whalesong/repl-prototype/htdocs/index.js
Normal file
69
whalesong/repl-prototype/htdocs/index.js
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
jQuery(document).ready(function() {
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var prompt = jQuery("#prompt");
|
||||||
|
var output = jQuery("#output");
|
||||||
|
var breakButton = jQuery("#break");
|
||||||
|
var resetButton = jQuery("#reset");
|
||||||
|
|
||||||
|
var write = function(dom) {
|
||||||
|
output.append(dom);
|
||||||
|
output.get(0).scrollTop = output.get(0).scrollHeight;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var onBreak = function() {
|
||||||
|
repl.requestBreak(allowInput);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var allowInput = function() {
|
||||||
|
prompt.val('');
|
||||||
|
prompt.removeAttr('disabled');
|
||||||
|
breakButton.hide();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var onReset = function() {
|
||||||
|
repl.reset(allowInput);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var onExpressionEntered = function() {
|
||||||
|
var src = prompt.val();
|
||||||
|
jQuery(this).val("");
|
||||||
|
prompt.attr('disabled', 'true');
|
||||||
|
prompt.val("... evaluating...");
|
||||||
|
breakButton.show();
|
||||||
|
repl.compileAndExecuteProgram('interactions',
|
||||||
|
src,
|
||||||
|
allowInput,
|
||||||
|
onError);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var onError = function(err) {
|
||||||
|
console.log(err);
|
||||||
|
if (err.message) {
|
||||||
|
write(jQuery('<span/>').css('color', 'red').append(err.message));
|
||||||
|
write(jQuery('<br/>'));
|
||||||
|
}
|
||||||
|
allowInput();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
breakButton.hide();
|
||||||
|
breakButton.click(onBreak);
|
||||||
|
resetButton.click(onReset);
|
||||||
|
prompt.attr('disabled', 'true');
|
||||||
|
prompt.val('Please wait, initializing...');
|
||||||
|
prompt.keypress(function(e) {
|
||||||
|
if (e.which == 13 && !prompt.attr('disabled')) {
|
||||||
|
onExpressionEntered();
|
||||||
|
}});
|
||||||
|
var afterReplSetup = function() {
|
||||||
|
prompt.val('');
|
||||||
|
prompt.removeAttr('disabled');
|
||||||
|
};
|
||||||
|
var repl = new plt.runtime.Repl({ write: write }, afterReplSetup);
|
||||||
|
});
|
|
@ -1,106 +1,155 @@
|
||||||
jQuery(document).ready(function() {
|
(function() {
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
// if (! console.log) { console.log = function() { }; }
|
// options: { compilerUrl: string,,
|
||||||
|
// write: (dom-node -> void)
|
||||||
|
// language: string }
|
||||||
|
var Repl = function(options, afterSetup) {
|
||||||
|
this.M = plt.runtime.currentMachine;
|
||||||
|
|
||||||
var repl = jQuery("#repl");
|
this.xhr = new easyXDM.Rpc(
|
||||||
var output = jQuery("#output");
|
{ remote: options.compilerUrl || 'rpc.html' },
|
||||||
var breakButton = jQuery("#break");
|
{ remote: { replCompile: {} } });
|
||||||
var resetButton = jQuery("#reset");
|
|
||||||
|
|
||||||
|
if (options.write) { this.write = options.write; }
|
||||||
|
|
||||||
// The machine.
|
this.language = (options.language ||
|
||||||
var M;
|
'whalesong/wescheme/lang/semantics.rkt');
|
||||||
|
setupMachine(this, afterSetup);
|
||||||
|
};
|
||||||
|
|
||||||
var interactionsCount = 0;
|
// write: dom-node -> void
|
||||||
|
// Expected to be overridden by others via options.write.
|
||||||
var sendOutputToBottom = function() {
|
Repl.prototype.write = function(dom) {
|
||||||
output.get(0).scrollTop = output.get(0).scrollHeight;
|
jQuery(document.body).append(dom)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
var xhr = new easyXDM.Rpc(
|
// Return true if the Repl is currently evaluating.
|
||||||
{ remote: 'rpc.html' },
|
Repl.prototype.isRunning = function() {
|
||||||
{ remote: { replCompile: {} } });
|
return this.M.running;
|
||||||
|
|
||||||
|
|
||||||
var onBreak = function() {
|
|
||||||
if (M.running) {
|
|
||||||
interruptEvaluation(function(){});
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var onReset = function() {
|
|
||||||
if (M.running) {
|
// Request a break on the current evaluation, calling afterBreak
|
||||||
M.params.currentDisplayer =
|
// once the break succeeds. If no evaluation is running, immediately
|
||||||
function(MACHINE, domNode) {};
|
// call afterBreak.
|
||||||
M.params.currentErrorDisplayer =
|
Repl.prototype.requestBreak = function(afterBreak) {
|
||||||
function(MACHINE, domNode) {};
|
if (this.M.running) {
|
||||||
interruptEvaluation(
|
interruptEvaluation(this, afterBreak);
|
||||||
function() {
|
|
||||||
output.empty();
|
|
||||||
setupMachine();
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
output.empty();
|
afterBreak();
|
||||||
setupMachine();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
var setupMachine = function() {
|
var interruptEvaluation = function(that, afterBreak) {
|
||||||
M = plt.runtime.currentMachine;
|
if (! that.M.running) {
|
||||||
|
throw new Error("internal error: trying to interrupt evaluation but nothing is running.");
|
||||||
|
}
|
||||||
|
that.M.scheduleBreak(afterBreak);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// Resets the evaluator, quietly interrupting evaluation if
|
||||||
|
// necessary.
|
||||||
|
Repl.prototype.reset = function(afterReset) {
|
||||||
|
var that = this;
|
||||||
|
if (this.M.running) {
|
||||||
|
this.M.params.currentDisplayer =
|
||||||
|
function(MACHINE, domNode) {};
|
||||||
|
this.M.params.currentErrorDisplayer =
|
||||||
|
function(MACHINE, domNode) {};
|
||||||
|
interruptEvaluation(
|
||||||
|
that,
|
||||||
|
function() {
|
||||||
|
setupMachine(that, afterReset);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setupMachine(that, afterReset);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var setupMachine = function(that, afterSetup) {
|
||||||
|
var M = that.M;
|
||||||
M.reset();
|
M.reset();
|
||||||
// We configure output to send it to the "output" DOM node.
|
|
||||||
|
// We configure the machine's output to send it to the
|
||||||
|
// "output" DOM node.
|
||||||
M.params.currentDisplayer = function(MACHINE, domNode) {
|
M.params.currentDisplayer = function(MACHINE, domNode) {
|
||||||
jQuery(domNode).appendTo(output);
|
that.write(domNode);
|
||||||
sendOutputToBottom();
|
|
||||||
};
|
};
|
||||||
M.params.currentErrorDisplayer = function(MACHINE, domNode) {
|
M.params.currentErrorDisplayer = function(MACHINE, domNode) {
|
||||||
jQuery(domNode).css("color", "red").appendTo(output);
|
that.write(domNode);
|
||||||
sendOutputToBottom();
|
|
||||||
};
|
};
|
||||||
|
// FIXME: add other parameter settings here.
|
||||||
|
|
||||||
// We then want to initialize the language module.
|
// We then want to initialize the language module.
|
||||||
var initializeLanguage = function(afterLanguageInitialization) {
|
M.loadAndInvoke(
|
||||||
// Load up the language.
|
that.language,
|
||||||
M.loadAndInvoke('whalesong/wescheme/lang/semantics.rkt',
|
|
||||||
function() {
|
|
||||||
var semanticsModule =
|
|
||||||
M.modules['whalesong/wescheme/lang/semantics.rkt'];
|
|
||||||
M.params.currentNamespace = semanticsModule.getExports();
|
|
||||||
afterLanguageInitialization();
|
|
||||||
},
|
|
||||||
function(err) {
|
|
||||||
// Nothing should work if we can't get this to work.
|
|
||||||
alert("uh oh!: language could not be loaded.");
|
|
||||||
});
|
|
||||||
};
|
|
||||||
repl.attr('disabled', 'true');
|
|
||||||
repl.val('Please wait, initializing...');
|
|
||||||
initializeLanguage(
|
|
||||||
function() {
|
function() {
|
||||||
repl.val('');
|
var semanticsModule = M.modules[that.language];
|
||||||
repl.removeAttr('disabled');
|
// FIXME: this should be getting the namespace,
|
||||||
// Hook up a simple one-line REPL with enter triggering evaluation.
|
// not the export dictionary...
|
||||||
repl.keypress(function(e) {
|
M.params.currentNamespace = semanticsModule.getExports();
|
||||||
if (e.which == 13 && !repl.attr('disabled')) {
|
afterSetup();
|
||||||
var src = repl.val();
|
},
|
||||||
jQuery(this).val("");
|
function(err) {
|
||||||
repl.attr('disabled', 'true');
|
// Nothing should work if we can't get this to work.
|
||||||
repl.val("... evaluating...");
|
alert("uh oh!: language could not be loaded.");
|
||||||
breakButton.show();
|
|
||||||
compileAndEvaluate(src,
|
|
||||||
function() { repl.removeAttr('disabled');
|
|
||||||
repl.val("");
|
|
||||||
breakButton.hide();});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Repl.prototype.compileAndExecuteProgram = function(programName, code,
|
||||||
|
onDone, onDoneError) {
|
||||||
|
var that = this;
|
||||||
|
this.compileProgram(
|
||||||
|
programName,
|
||||||
|
code,
|
||||||
|
function(compiledCode) {
|
||||||
|
that.executeCompiledProgram(compiledCode,
|
||||||
|
onDone,
|
||||||
|
onDoneError);
|
||||||
|
},
|
||||||
|
onDoneError);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Repl.prototype.compileProgram = function(programName, code,
|
||||||
|
onDone, onDoneError) {
|
||||||
|
this.xhr.replCompile(programName, code, onDone, onDoneError);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Repl.prototype.executeCompiledProgram = function(compiledResult,
|
||||||
|
onDoneSuccess, onDoneFail) {
|
||||||
|
var that = this;
|
||||||
|
// compiledResult.compiledCodes is an array of function chunks.
|
||||||
|
// The evaluation leaves the value register of the machine
|
||||||
|
// to contain the list of values from toplevel evaluation.
|
||||||
|
var compiledCodes = compiledResult.compiledCodes;
|
||||||
|
forEachK(compiledCodes,
|
||||||
|
function(code, k) {
|
||||||
|
// Indirect eval usage here is deliberate.
|
||||||
|
var codeFunction = (0,eval)(code);
|
||||||
|
var onGoodEvaluation = function() {
|
||||||
|
var resultList = that.M.v;
|
||||||
|
while(resultList !== plt.baselib.lists.EMPTY) {
|
||||||
|
print(that, resultList.first);
|
||||||
|
resultList = resultList.rest;
|
||||||
|
};
|
||||||
|
k();
|
||||||
|
};
|
||||||
|
var onBadEvaluation = function(M, err) {
|
||||||
|
onDoneFail(err);
|
||||||
|
};
|
||||||
|
codeFunction(that.M, onGoodEvaluation, onBadEvaluation);
|
||||||
|
},
|
||||||
|
onDoneSuccess);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
// CPS'ed for-each.
|
// CPS'ed for-each.
|
||||||
var forEachK = function(elts, f, after) {
|
var forEachK = function(elts, f, after) {
|
||||||
var n = elts.length;
|
var n = elts.length;
|
||||||
|
@ -115,117 +164,23 @@ jQuery(document).ready(function() {
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
// writeErrorMessage: string -> void
|
// Print: Repl racket-value -> void
|
||||||
// Write out an error message.
|
// Prints the racket value out, followed by a newline,
|
||||||
var writeErrorMessage = function(msg) {
|
// unless VOID is being printed.
|
||||||
M.params.currentErrorDisplayer(M,
|
var print = function(that, elt) {
|
||||||
jQuery("<span/>")
|
var outputPort = that.M.params.currentOutputPort;
|
||||||
.text(''+msg)
|
|
||||||
.css("color", "red"));
|
|
||||||
M.params.currentErrorDisplayer(M, jQuery("<br/>"));
|
|
||||||
sendOutputToBottom();
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Print: Racket value -> void
|
|
||||||
// Prints the racket value out.
|
|
||||||
var print = function(elt) {
|
|
||||||
var outputPort =
|
|
||||||
M.params.currentOutputPort;
|
|
||||||
if (elt !== plt.runtime.VOID) {
|
if (elt !== plt.runtime.VOID) {
|
||||||
outputPort.writeDomNode(
|
outputPort.writeDomNode(
|
||||||
M,
|
that.M,
|
||||||
plt.runtime.toDomNode(elt, M.params['print-mode']));
|
plt.runtime.toDomNode(elt, that.M.params['print-mode']));
|
||||||
outputPort.writeDomNode(M, plt.runtime.toDomNode("\n", 'display'));
|
outputPort.writeDomNode(that.M,
|
||||||
|
plt.runtime.toDomNode("\n", 'display'));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var interruptEvaluation = function(afterBreak) {
|
|
||||||
if (! M.running) {
|
|
||||||
throw new Error("internal error: trying to interrupt evaluation but nothing is running.");
|
|
||||||
}
|
|
||||||
M.scheduleBreak(afterBreak);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////
|
||||||
// In evaluation, we'll send compilation requests to the server,
|
//////////////////////////////////////////////////////////////////////
|
||||||
// and get back bytecode that we should evaluate.
|
// Expose to the outside world as plt.runtime.Repl.
|
||||||
var compileAndEvaluate = function(src, after) {
|
plt.runtime.Repl = Repl;
|
||||||
M.params.currentDisplayer(M, jQuery("<tt/>").text('> ' + src));
|
}());
|
||||||
M.params.currentDisplayer(M, jQuery("<br/>"));
|
|
||||||
var onCompile = function(compiledResult) {
|
|
||||||
if (compiledResult.type === 'repl') {
|
|
||||||
return onGoodReplCompile(compiledResult);
|
|
||||||
} else if (compiledResult.type === 'module') {
|
|
||||||
alert('internal error: module unexpected');
|
|
||||||
after();
|
|
||||||
} else if (compiledResult.type === 'error') {
|
|
||||||
return onCompileTimeError(compiledResult);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
var onCompileTimeError = function(compiledResult) {
|
|
||||||
writeErrorMessage(compiledResult.message);
|
|
||||||
after();
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
var onGoodReplCompile = function(compiledResult) {
|
|
||||||
// compiledResult.compiledCodes is an array of function chunks.
|
|
||||||
// The evaluation leaves the value register of the machine
|
|
||||||
// to contain the list of values from toplevel evaluation.
|
|
||||||
var compiledCodes = compiledResult.compiledCodes;
|
|
||||||
forEachK(compiledCodes,
|
|
||||||
function(code, k) {
|
|
||||||
// Indirect eval usage here is deliberate.
|
|
||||||
var codeFunction = (0,eval)(code);
|
|
||||||
var onGoodEvaluation = function() {
|
|
||||||
var resultList = M.v;
|
|
||||||
while(resultList !== plt.baselib.lists.EMPTY) {
|
|
||||||
print(resultList.first);
|
|
||||||
resultList = resultList.rest;
|
|
||||||
};
|
|
||||||
k();
|
|
||||||
};
|
|
||||||
var onBadEvaluation = function(M, err) {
|
|
||||||
if (err.message) {
|
|
||||||
writeErrorMessage(err.message);
|
|
||||||
}
|
|
||||||
|
|
||||||
after();
|
|
||||||
};
|
|
||||||
codeFunction(M, onGoodEvaluation, onBadEvaluation);
|
|
||||||
},
|
|
||||||
after);
|
|
||||||
};
|
|
||||||
var onServerError = function(err) {
|
|
||||||
writeErrorMessage("internal server error");
|
|
||||||
after();
|
|
||||||
};
|
|
||||||
xhr.replCompile("<interactions" + interactionsCount + ">",
|
|
||||||
src, onCompile, onServerError);
|
|
||||||
interactionsCount = interactionsCount + 1;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
// Things that we need to make as automated tests:
|
|
||||||
//
|
|
||||||
// Make sure: (let () (define (f x) (f x)) (f 42))
|
|
||||||
// is interruptable.
|
|
||||||
//
|
|
||||||
// Test: simple expressions, functions, etc.
|
|
||||||
//
|
|
||||||
// Test: multiple value return, even zero
|
|
||||||
//
|
|
||||||
// Test: require image library, try drawing a few things.
|
|
||||||
//
|
|
||||||
// Test: compile a module.
|
|
||||||
//
|
|
||||||
|
|
||||||
breakButton.hide();
|
|
||||||
breakButton.click(onBreak);
|
|
||||||
resetButton.click(onReset);
|
|
||||||
setupMachine();
|
|
||||||
});
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user