Skip to content

Commit

Permalink
Use Effection for Ursa async (WIP)
Browse files Browse the repository at this point in the history
Also use Effection in some of Ark: evalArkFs becomes a generator.

Make launch an Exp, not a Statement, as we need to be able to await
it.

ArkPromise becomes ArkOperation.

Add NativeOperation, and bind Effection's sleep() using it.

Use Effection in the JavaScript back-end too, using the "obvious"
translation (basically, the Effection "Rosetta Stone")). Standalone
scripts now require Effection.

Change 'launch.ursa' test to await one launched expression but not the
other, to test that the un-awaited one is terminated early.
  • Loading branch information
rrthomas committed Aug 3, 2024
1 parent 5fb6486 commit c6b3577
Show file tree
Hide file tree
Showing 14 changed files with 119 additions and 80 deletions.
3 changes: 2 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
"@prettier/sync": "^0.5.2",
"@sc3d/stacktracey": "^2.1.9",
"argparse": "^2.0.1",
"effection": "^3.0.3",
"env-paths": "^3.0.0",
"fs-extra": "^11.2.0",
"get-source": "^2.0.12",
Expand Down Expand Up @@ -64,7 +65,7 @@
"patch-diff-ohm": "patch -p0 --output=src/grammar/ursa.ohm-bundle.d.part-patched.ts < src/grammar/ursa.ohm-bundle.d.ts.diff",
"patch-ohm": "npm run patch-diff-ohm && sed -e 's/this: NonterminalNode/this: ThisNode/' < src/grammar/ursa.ohm-bundle.d.part-patched.ts > src/grammar/ursa.ohm-bundle.d.ts",
"generate": "npm run generate-only && npm run patch-ohm && mkdir -p lib/grammar/ && cp src/grammar/*ohm-bundle.js src/grammar/*ohm-bundle.d.ts lib/grammar/",
"pre-compile-prelude": "echo null > src/ark/prelude.json && echo 'new ArkObject(new Map())' > src/ark/compiler/prelude.js",
"pre-compile-prelude": "echo null > src/ark/prelude.json && echo 'function* gen() { return new ArkObject(new Map()) }; gen' > src/ark/compiler/prelude.js",
"update-ohm-patch": "npm run generate-only && npm run patch-diff-ohm && diff -u src/grammar/ursa.ohm-bundle.d.ts src/grammar/ursa.ohm-bundle.d.part-patched.ts > src/grammar/ursa.ohm-bundle.d.ts.diff; npm run patch-ohm",
"compile-prelude": "npm run pre-compile-prelude && ./bin/test-run.sh compile --output=src/ark/prelude.json src/ursa/prelude.ursa && ./bin/test-run.sh --target=js compile --output=src/ark/compiler/prelude.js src/ursa/prelude.ursa",
"test": "npm run generate && npm run build && ava 'src/**/basics.test.ts' 'src/**/examples.test.ts' 'src/**/cli.test.ts' 'src/**/fsmap.test.ts'",
Expand Down
38 changes: 20 additions & 18 deletions src/ark/compiler/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ import fs from 'fs'
import path from 'path'
import {fileURLToPath} from 'url'
import util from 'util'

import {call, Operation, run} from 'effection'
import getSource from 'get-source'
import {
CodeWithSourceMap, Position, SourceMapConsumer,
Expand All @@ -31,7 +33,7 @@ import {
globals, jsGlobals,
ArkBoolean, ArkBooleanVal, ArkList, ArkMap, ArkNull,
ArkNumber, ArkNullVal, ArkNumberVal, ArkObject, ArkString,
ArkStringVal, ArkUndefined, ArkVal, NativeFn, ArkPromise,
ArkStringVal, ArkUndefined, ArkVal, NativeFn, ArkOperation,
} from '../data.js'
import {ArkExp} from '../code.js'
import {debug} from '../util.js'
Expand All @@ -54,8 +56,9 @@ class UrsaStackTracey extends StackTracey {

// Compile prelude and add it to globals
export const preludeJs = fs.readFileSync(path.join(__dirname, 'prelude.js'), {encoding: 'utf-8'})
const prelude = await evalArkJs(preludeJs) as ArkObject
prelude.properties.forEach((val, sym) => jsGlobals.set(sym, val))
const prelude = await run(() => evalArkJs(preludeJs))
const preludeObj = await run(prelude as () => Operation<ArkObject>)
preludeObj.properties.forEach((val, sym) => jsGlobals.set(sym, val))

// runtimeContext records the values that are needed by JavaScript at
// runtime, and prevents the TypeScript compiler throwing away their
Expand All @@ -70,7 +73,7 @@ const runtimeContext: Record<string, unknown> = {
ArkObject,
ArkList,
ArkMap,
ArkPromise,
ArkOperation,
NativeFn,
jsGlobals,
}
Expand Down Expand Up @@ -121,13 +124,13 @@ export function flatToJs(insts: ArkInsts, file: string | null = null): CodeWithS
return sourceNode(letAssign(inst.id, valToJs(inst.val)))
} else if (inst instanceof ArkLaunchBlockCloseInst) {
return sourceNode([
`return ${inst.id.description!}\n`,
'})())\n',
`yield* ${inst.blockId.description!}\n`,
'})\n',
])
} else if (inst instanceof ArkFnBlockCloseInst) {
env = env.popFrame()
if (inst.matchingOpen instanceof ArkGeneratorBlockOpenInst) {
return sourceNode(['}()\nreturn new NativeFn([\'x\'], async (x) => {\nconst {value, done} = await gen.next(x)\nreturn done ? ArkNull() : value\n})\n})\n'])
return sourceNode(['}()\nreturn new NativeFn([\'x\'], (x) => {\nconst {value, done} = gen.next(x)\nreturn done ? ArkNull() : value\n})\n})\n'])
}
return sourceNode(['})\n'])
} else if (inst instanceof ArkIfBlockOpenInst) {
Expand All @@ -142,20 +145,20 @@ export function flatToJs(insts: ArkInsts, file: string | null = null): CodeWithS
} else if (inst instanceof ArkLoopBlockOpenInst) {
return sourceNode([letAssign(inst.matchingClose.id, 'ArkNull()'), 'for (;;) {\n'])
} else if (inst instanceof ArkLaunchBlockOpenInst) {
return sourceNode([letAssign(inst.matchingClose.id, 'new ArkPromise((async function() {')])
return sourceNode([letAssign(inst.matchingClose.id, 'yield* spawn(function* () {')])
} else if (inst instanceof ArkGeneratorBlockOpenInst) {
env = env.pushFrame(
new Frame(inst.params.map((p) => new Location(p, false)), [], inst.name),
)
return sourceNode([
letAssign(inst.matchingClose.id, `new NativeFn([${inst.params.map((p) => `'${p}'`).join(', ')}], function (${inst.params.join(', ')}) {\nconst gen = async function*() {`),
letAssign(inst.matchingClose.id, `new NativeFn([${inst.params.map((p) => `'${p}'`).join(', ')}], function (${inst.params.join(', ')}) {\nconst gen = function* () {`),
])
} else if (inst instanceof ArkFnBlockOpenInst) {
env = env.pushFrame(
new Frame(inst.params.map((p) => new Location(p, false)), [], inst.name),
)
return sourceNode([
letAssign(inst.matchingClose.id, `new NativeFn([${inst.params.map((p) => `'${p}'`).join(', ')}], async function(${inst.params.join(', ')}) {`),
letAssign(inst.matchingClose.id, `new NativeFn([${inst.params.map((p) => `'${p}'`).join(', ')}], function* (${inst.params.join(', ')}) {`),
])
} else if (inst instanceof ArkLetBlockOpenInst) {
return sourceNode([
Expand All @@ -166,7 +169,7 @@ export function flatToJs(insts: ArkInsts, file: string | null = null): CodeWithS
} else if (inst instanceof ArkBlockOpenInst) {
return sourceNode([`let ${inst.matchingClose.id.description!}\n`, '{\n'])
} else if (inst instanceof ArkAwaitInst) {
return sourceNode(letAssign(inst.id, `await ${inst.argId.description}.promise`))
return sourceNode(letAssign(inst.id, `yield* ${inst.argId.description}`))
} else if (inst instanceof ArkBreakInst) {
return sourceNode([`${assign(inst.argId.description!, inst.loopInst.matchingClose.id.description!)}\n`, 'break\n'])
} else if (inst instanceof ArkContinueInst) {
Expand All @@ -178,7 +181,7 @@ export function flatToJs(insts: ArkInsts, file: string | null = null): CodeWithS
} else if (inst instanceof ArkLetCopyInst) {
return sourceNode(letAssign(inst.id, inst.argId.description!))
} else if (inst instanceof ArkCallInst) {
return sourceNode(letAssign(inst.id, `await ${inst.fnId.description}.body(${inst.argIds.map((id) => id.description).join(', ')})`))
return sourceNode(letAssign(inst.id, `yield* ${inst.fnId.description}.body(${inst.argIds.map((id) => id.description).join(', ')})`))
} else if (inst instanceof ArkSetInst) {
return sourceNode([
`if (${inst.lexpId.description} !== ArkUndefined && ${inst.lexpId.description}.constructor !== ArkNullVal && ${inst.valId.description}.constructor !== ${inst.lexpId.description}.constructor) {\n`,
Expand Down Expand Up @@ -222,11 +225,10 @@ export function flatToJs(insts: ArkInsts, file: string | null = null): CodeWithS
}

const sourceNode = new SourceNode(1, 1, 'src/ursa/flat-to-js.ts', [
// FIXME: work out how to eval ESM, so we can use top-level await.
'"use strict";\n',
'(async () => {\n',
'(function* () {\n',
instsToJs(insts),
`return ${insts.id.description}\n})()`,
`return ${insts.id.description}\n})`,
])
const jsCode = sourceNode.toStringWithSourceMap({file: file ?? undefined})
if (process.env.DEBUG) {
Expand All @@ -240,7 +242,7 @@ export function arkToJs(exp: ArkExp, file: string | null = null): CodeWithSource
return flatToJs(insts, file)
}

export async function evalArkJs(source: CodeWithSourceMap | string, file = '(Compiled Ark)'): Promise<ArkVal> {
export function* evalArkJs(source: CodeWithSourceMap | string, file = '(Compiled Ark)') {
let jsSource: string
if (typeof source === 'string') {
jsSource = source
Expand All @@ -253,7 +255,7 @@ export async function evalArkJs(source: CodeWithSourceMap | string, file = '(Com
}
try {
// eslint-disable-next-line no-eval
return await (eval(jsSource) as Promise<ArkVal>)
return eval(jsSource) as GeneratorFunction
} catch (e) {
assert(e instanceof Error)
const dirtyStack = new UrsaStackTracey(e).withSources()
Expand Down Expand Up @@ -293,7 +295,7 @@ export async function evalArkJs(source: CodeWithSourceMap | string, file = '(Com
newError.message = `${prefix}\n${message}`
let consumer
if (typeof source !== 'string') {
consumer = await new SourceMapConsumer(source.map.toJSON())
consumer = yield* call(new SourceMapConsumer(source.map.toJSON()))
}
for (const [i, frame] of stack.items.slice(1).entries()) {
let fnLocation
Expand Down
47 changes: 34 additions & 13 deletions src/ark/data.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,10 @@

import assert from 'assert'

import {
action, call, Operation, Reject, Resolve, sleep,
} from 'effection'

import {FsMap} from './fsmap.js'
import programVersion from '../version.js'
import {debug} from './util.js'
Expand Down Expand Up @@ -147,8 +151,8 @@ export function ArkString(s: string) {
return ConcreteInterned.value<ArkStringVal, string>(ArkStringVal, s)
}

export class ArkPromise extends ArkVal {
constructor(public promise: Promise<ArkVal>) {
export class ArkOperation extends ArkVal {
constructor(public operation: Operation<ArkVal>) {
super()
}
}
Expand All @@ -170,15 +174,28 @@ export abstract class ArkClosure extends ArkCallable {
export abstract class ArkGeneratorClosure extends ArkClosure {}

export class NativeFn extends ArkCallable {
constructor(params: string[], public body: (...args: ArkVal[]) => ArkVal) {
public body: (...args: ArkVal[]) => Operation<ArkVal>

constructor(params: string[], public innerBody: (...args: ArkVal[]) => ArkVal) {
super(params)
// eslint-disable-next-line require-yield
this.body = function* gen(...args: ArkVal[]) { return innerBody(...args) }
}
}

export class NativeOperation extends ArkCallable {
constructor(params: string[], public body: (...args: ArkVal[]) => Operation<ArkVal>) {
super(params)
}
}

// ts-unused-exports:disable-next-line
export class NativeAsyncFn extends ArkCallable {
constructor(params: string[], public body: (...args: ArkVal[]) => Promise<ArkVal>) {
public body: (...args: ArkVal[]) => Operation<ArkVal>

constructor(params: string[], innerBody: (...args: ArkVal[]) => Promise<ArkVal>) {
super(params)
this.body = (...args: ArkVal[]) => call(() => innerBody(...args))
}
}

Expand Down Expand Up @@ -366,17 +383,21 @@ export const globals = new ArkObject(new Map<string, ArkVal>([
debug(obj)
return ArkNull()
})],
['fs', new NativeFn(['path'], (path: ArkVal) => new NativeObject(new FsMap(toJs(path) as string)))],

['Promise', new NativeAsyncFn(
['fs', new NativeFn(['path'], (path: ArkVal) => new NativeObject(new FsMap((path as ArkStringVal).val)))],
['sleep', new NativeOperation(['ms'], function* gen(ms: ArkVal) {
yield* sleep((ms as ArkNumberVal).val)
return ArkNull()
})],
['action', new NativeFn(
['resolve', 'reject'],
(fn: ArkVal) => Promise.resolve(new ArkPromise(
new Promise(
toJs(fn) as
(resolve: (value: unknown) => void, reject: (reason?: unknown) => void) => void,
).then((x) => fromJs(x)),
)),
function* gen(fn: ArkVal) {
const result = yield* action(
toJs(fn) as (resolve: Resolve<unknown>, reject: Reject) => Operation<void>,
)
return fromJs(result)
},
)],
// FIXME: should be able to use js.fetch directly.
['fetch', new NativeAsyncFn(
['url', 'options'],
async (url: ArkVal, options: ArkVal) => new NativeObject(
Expand Down
30 changes: 19 additions & 11 deletions src/ark/interpreter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,9 @@
// © Reuben Thomas 2023-2024
// Released under the MIT license.

import {
Operation, run, spawn,
} from 'effection'
import {Interval} from 'ohm-js'

import {
Expand All @@ -15,8 +18,8 @@ import {
} from './flatten.js'
import {
ArkAbstractObjectBase, ArkBoolean, ArkList, ArkMap, ArkNull, ArkNullVal,
ArkObject, ArkPromise, ArkUndefined, ArkVal, NativeAsyncFn, NativeFn,
ArkRef, ArkValRef,
ArkObject, ArkOperation, ArkUndefined, ArkVal, NativeAsyncFn, NativeFn,
NativeOperation, ArkRef, ArkValRef,
} from './data.js'
import {
ArkCapture, ArkContinuation, ArkFlatClosure, ArkFlatGeneratorClosure,
Expand Down Expand Up @@ -101,6 +104,10 @@ function makeLocals(names: string[], vals: ArkVal[]): ArkRef[] {
}

async function evalFlat(outerArk: ArkState): Promise<ArkVal> {
return run(() => doEvalFlat(outerArk))
}

function* doEvalFlat(outerArk: ArkState): Operation<ArkVal> {
let ark: ArkState | undefined = outerArk
let inst = ark.inst
let prevInst
Expand Down Expand Up @@ -157,9 +164,10 @@ async function evalFlat(outerArk: ArkState): Promise<ArkVal> {
),
ark.outerState,
)
const result = Promise.resolve(new ArkPromise(evalFlat(innerArk)))
const operation = yield* spawn(() => doEvalFlat(innerArk))
const result = new ArkOperation(operation)
mem.set(inst.id, result)
// The Promise becomes the result of the entire block.
// The ArkOperation becomes the result of the entire block.
mem.set(inst.matchingClose.id, result)
inst = inst.matchingClose.next
} else if (inst instanceof ArkCallableBlockOpenInst) {
Expand All @@ -184,8 +192,9 @@ async function evalFlat(outerArk: ArkState): Promise<ArkVal> {
} else if (inst instanceof ArkBlockOpenInst) {
inst = inst.next
} else if (inst instanceof ArkAwaitInst) {
const promise = (mem.get(inst.argId)! as ArkPromise).promise
mem.set(inst.id, await promise)
const operation = (mem.get(inst.argId)! as ArkOperation).operation
const result = yield* operation
mem.set(inst.id, result)
inst = inst.next
} else if (inst instanceof ArkBreakInst) {
const result = mem.get(inst.argId)!
Expand Down Expand Up @@ -273,11 +282,10 @@ async function evalFlat(outerArk: ArkState): Promise<ArkVal> {
inst = inst.next
}
}
} else if (callable instanceof NativeFn) {
mem.set(inst.id, callable.body(...args))
inst = inst.next
} else if (callable instanceof NativeAsyncFn) {
mem.set(inst.id, await callable.body(...args))
} else if (callable instanceof NativeFn
|| callable instanceof NativeAsyncFn
|| callable instanceof NativeOperation) {
mem.set(inst.id, yield* callable.body(...args))
inst = inst.next
} else {
throw new ArkRuntimeError(ark, 'Invalid call', inst.sourceLoc)
Expand Down
4 changes: 2 additions & 2 deletions src/ark/serialize.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

import {
globals, ArkVal,
ArkConcreteVal, ArkNull, ArkPromise, ArkList, ArkMap, ArkObject,
ArkConcreteVal, ArkNull, ArkOperation, ArkList, ArkMap, ArkObject,
ArkUndefined, NativeObject,
} from './data.js'
import {
Expand Down Expand Up @@ -87,7 +87,7 @@ export function valToJs(val: ArkVal | ArkExp, externalSyms = globals) {
return ['yield', doValToJs(val.exp)]
} else if (val instanceof ArkReturn) {
return ['return', doValToJs(val.exp)]
} else if (val instanceof ArkPromise) {
} else if (val instanceof ArkOperation) {
// FIXME: Can we properly serialize a promise?
return ['promise']
} else if (val === ArkNull()) {
Expand Down
2 changes: 1 addition & 1 deletion src/grammar/ursa.ohm
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,7 @@ Ursa {
| For
| await Exp -- await
| yield Exp? -- yield
| launch Exp -- launch
| LogicExp

Assignment
Expand All @@ -119,7 +120,6 @@ Ursa {
| Use
| break Exp? -- break
| continue -- continue
| launch Exp -- launch
| return Exp? -- return
| Exp

Expand Down
4 changes: 2 additions & 2 deletions src/grammar/ursa.ohm-bundle.d.ts.diff
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
--- src/grammar/ursa.ohm-bundle.d.ts 2024-07-25 19:26:08.992907449 +0100
+++ src/grammar/ursa.ohm-bundle.d.part-patched.ts 2024-07-25 19:26:09.128908027 +0100
--- src/grammar/ursa.ohm-bundle.d.ts 2024-07-27 22:31:25.130617483 +0100
+++ src/grammar/ursa.ohm-bundle.d.part-patched.ts 2024-07-27 22:31:25.270617985 +0100
@@ -4,14 +4,33 @@
import {
BaseActionDict,
Expand Down
7 changes: 5 additions & 2 deletions src/testutil.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,15 @@ import util from 'util'
import fs from 'fs'
import path from 'path'
import tmp from 'tmp'
import {run as effectionRun, Operation} from 'effection'
import test, {ExecutionContext, Macro} from 'ava'
import {ExecaError, Options as ExecaOptions, execa} from 'execa'
import {compareSync, Difference} from 'dir-compare'

import {debug} from './ark/util.js'
import {flatToJs, evalArkJs} from './ark/compiler/index.js'
import {expToInsts} from './ark/flatten.js'
import {ArkObject, toJs} from './ark/data.js'
import {ArkObject, ArkVal, toJs} from './ark/data.js'
import {ArkExp} from './ark/code.js'
import {ArkState} from './ark/interpreter.js'
import {compile as doArkCompile} from './ark/reader.js'
Expand Down Expand Up @@ -52,7 +53,9 @@ function doTestGroup(
const flat = expToInsts(compiled)
const jsSource = flatToJs(flat)
const resArk = await new ArkState(flat.insts[0]).run()
const resJs = await evalArkJs(jsSource, title)
const resJs = await effectionRun(
await effectionRun(() => evalArkJs(jsSource, title)) as () => Operation<ArkVal>,
)
if (resArk instanceof ArkObject) {
assert(typeof expected === 'object')
// Remove methods of ArkObject
Expand Down
Loading

0 comments on commit c6b3577

Please sign in to comment.