Skip to content

Commit

Permalink
Merge branch 'master' into fix-#302-assertion-errors
Browse files Browse the repository at this point in the history
  • Loading branch information
PalumboN authored Nov 19, 2024
2 parents 99d9641 + 25de9db commit 9d9a306
Show file tree
Hide file tree
Showing 7 changed files with 205 additions and 12 deletions.
6 changes: 3 additions & 3 deletions .github/workflows/node.js.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,9 @@ jobs:

steps:
- uses: actions/checkout@v3
- uses: webfactory/[email protected]
with:
ssh-private-key: ${{ secrets.SSH_PRIVATE_KEY }}
# - uses: webfactory/[email protected]
# with:
# ssh-private-key: ${{ secrets.SSH_PRIVATE_KEY }}
- name: Read .nvmrc
run: echo "##[set-output name=NVMRC;]$(cat .nvmrc)"
id: nvm
Expand Down
2 changes: 1 addition & 1 deletion scripts/fetchLanguage.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { existsSync, mkdirSync } from 'fs'
import gitClient from 'simple-git'
import { wollokVersion } from '../package.json'

const WOLLOK_LANGUAGE_REPO = 'git@github.com:uqbar-project/wollok-language.git'
const WOLLOK_LANGUAGE_REPO = 'https://github.com/uqbar-project/wollok-language.git'
const WOLLOK_LANGUAGE_TAG = wollokVersion.includes(':') ? wollokVersion.split(':')[1] : `v${wollokVersion}`
const WOLLOK_LANGUAGE_FOLDER = 'language'

Expand Down
12 changes: 9 additions & 3 deletions src/helpers.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { BOOLEAN_MODULE, CLOSURE_EVALUATE_METHOD, CLOSURE_TO_STRING_METHOD, INITIALIZE_METHOD, KEYWORDS, NUMBER_MODULE, OBJECT_MODULE, STRING_MODULE, VOID_WKO, WOLLOK_BASE_PACKAGE } from './constants'
import { BOOLEAN_MODULE, CLOSURE_EVALUATE_METHOD, CLOSURE_MODULE, CLOSURE_TO_STRING_METHOD, INITIALIZE_METHOD, KEYWORDS, NUMBER_MODULE, OBJECT_MODULE, STRING_MODULE, VOID_WKO, WOLLOK_BASE_PACKAGE } from './constants'
import { getPotentiallyUninitializedLazy } from './decorators'
import { count, is, isEmpty, last, List, match, notEmpty, otherwise, valueAsListOrEmpty, when, excludeNullish } from './extensions'
import { RuntimeObject, RuntimeValue } from './interpreter/runtimeModel'
Expand Down Expand Up @@ -217,7 +217,7 @@ export const usesField = (node: Node, field: Field): boolean =>
match(node)(
when(Singleton)(node => {
if (!node.isClosure()) return false
const applyMethod = node.methods.find(method => method.name === CLOSURE_EVALUATE_METHOD)
const applyMethod = node.methods.find(isApplyMethodForClosures)
return !!applyMethod && usesField(applyMethod, field)
}),
when(Variable)(node => usesField(node.value, field)),
Expand Down Expand Up @@ -441,8 +441,14 @@ export const getNodeDefinition = (environment: Environment) => (node: Node): Nod
}
}

export const isApplyMethodForClosures = (method: Method): boolean =>
method.name === CLOSURE_EVALUATE_METHOD && method.parent.fullyQualifiedName.startsWith(`${CLOSURE_MODULE}#`) // TODO: Maybe re-define isClosure() ?

export const superMethodDefinition = (superNode: Super, methodModule: Module): Method | undefined => {
const currentMethod = superNode.ancestors.find(is(Method))!
function isValidMethod(node: Node): node is Method {
return node.is(Method) && !isApplyMethodForClosures(node)
}
const currentMethod = superNode.ancestors.find(isValidMethod)!
return methodModule.lookupMethod(currentMethod.name, superNode.args.length, { lookupStartFQN: currentMethod.parent.fullyQualifiedName })
}

Expand Down
3 changes: 2 additions & 1 deletion src/linker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,6 @@ export const assignScopes = (root: Node): void => {
? parent?.parent.scope
: parent?.scope
assign(node, { scope: new LocalScope(containerScope) })

parent?.scope?.register(...scopeContribution(node))
})

Expand All @@ -116,7 +115,9 @@ export const assignScopes = (root: Node): void => {
node.scope.include(new LocalScope(undefined, ...contributions))
}
}
})

root.forEach((node, _parent) => {
if (node.is(Module)) {
node.scope.include(...node.hierarchy.slice(1).map(supertype => supertype.scope))
}
Expand Down
2 changes: 1 addition & 1 deletion src/validator/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -156,7 +156,7 @@ export const shouldOnlyInheritFromMixin = error<Mixin>(node => node.supertypes.e
}))

export const shouldUseOverrideKeyword = warning<Method>(node =>
node.isOverride || !superclassMethod(node) || node.name == INITIALIZE_METHOD
node.isOverride || node.isSynthetic || !superclassMethod(node) || node.name == INITIALIZE_METHOD
)

export const possiblyReturningBlock = warning<Method>(node => {
Expand Down
131 changes: 129 additions & 2 deletions test/interpreter.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { expect, should, use } from 'chai'
import { restore } from 'sinon'
import sinonChai from 'sinon-chai'
import { EXCEPTION_MODULE, Evaluation, REPL, WRENatives, buildEnvironment } from '../src'
import { DirectedInterpreter, interprete, Interpreter, getStackTraceSanitized } from '../src/interpreter/interpreter'
import { DirectedInterpreter, getStackTraceSanitized, interprete, Interpreter } from '../src/interpreter/interpreter'
import link from '../src/linker'
import { Body, Class, Field, Literal, Method, Package, ParameterizedType, Reference, Return, Send, Singleton, SourceIndex, SourceMap } from '../src/model'
import { WREEnvironment } from './utils'
Expand Down Expand Up @@ -268,6 +268,134 @@ describe('Wollok Interpreter', () => {
it('for closure', () => {
checkSuccessfulResult('{1 + 2}', '{1 + 2}')
})

it('should be able to execute sentences related to a hierarchy defined in different packages', () => {
const replEnvironment = buildEnvironment([{
name: 'jefeDeDepartamento.wlk', content: `
import medico.*
class Jefe inherits Medico {
const subordinados = #{}
override method atenderA(unaPersona) {
subordinados.anyOne().atenderA(unaPersona)
}
}
`,
}, {
name: 'medico.wlk', content: `
import persona.*
class Medico inherits Persona {
const dosis
override method contraerEnfermedad(unaEnfermedad) {
super(unaEnfermedad)
self.atenderA(self)
}
method atenderA(unaPersona) {
unaPersona.recibirMedicamento(dosis)
}
}
`,
}, {
name: 'persona.wlk', content: `
class Persona {
const enfermedades = []
method contraerEnfermedad(unaEnfermedad) {
enfermedades.add(unaEnfermedad)
}
method saludar() = "hola"
}
`,
}, {
name: REPL, content: `
import medico.*
object testit {
method test() = new Medico(dosis = 200).saludar()
}
`,
}])
interpreter = new Interpreter(Evaluation.build(replEnvironment, WRENatives))
const { error, result } = interprete(interpreter, 'testit.test()')
expect(error).to.be.undefined
expect(result).to.equal('"hola"')
})

it('should be able to execute sentences related to a hierarchy defined in different packages - 2', () => {
const replEnvironment = buildEnvironment([{
name: 'medico.wlk', content: `
import persona.*
class Medico inherits Persona {
const dosis
override method contraerEnfermedad(unaEnfermedad) {
super(unaEnfermedad)
self.atenderA(self)
}
method atenderA(unaPersona) {
unaPersona.recibirMedicamento(dosis)
}
}
`,
}, {
name: 'pediatra.wlk', content: `
import jefeDeDepartamento.*
class Pediatra inherits Jefe {
const property fechaIngreso = new Date()
method esNuevo() = fechaIngreso.year() < 2022
}
`,
}, {
name: 'jefeDeDepartamento.wlk', content: `
import medico.*
class Jefe inherits Medico {
const subordinados = #{}
override method atenderA(unaPersona) {
subordinados.anyOne().atenderA(unaPersona)
}
}
`,
}, {
name: 'persona.wlk', content: `
class Persona {
const enfermedades = []
method contraerEnfermedad(unaEnfermedad) {
enfermedades.add(unaEnfermedad)
}
method saludar() = "hola"
}
`,
}, {
name: REPL, content: `
import pediatra.*
object testit {
method test() = new Pediatra(dosis = 200).saludar()
}
`,
}])
interpreter = new Interpreter(Evaluation.build(replEnvironment, WRENatives))
const { error, result } = interprete(interpreter, 'testit.test()')
expect(error).to.be.undefined
expect(result).to.equal('"hola"')
})

})

describe('sanitize stack trace', () => {
Expand Down Expand Up @@ -539,7 +667,6 @@ describe('Wollok Interpreter', () => {
expect(errored).to.be.false
})


})

describe('DirectedInterpreter', () => {
Expand Down
61 changes: 60 additions & 1 deletion test/linker.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { expect, should, use } from 'chai'
import { GAME_MODULE, OBJECT_MODULE } from '../src'
import { getPotentiallyUninitializedLazy } from '../src/decorators'
import link, { canBeReferenced, linkSentenceInNode } from '../src/linker'
import { Body, Class, Closure, Describe, Environment, Field, Import, Method, Mixin, NamedArgument, Node, Package, Parameter, ParameterizedType, Reference, Return, Sentence, Singleton, Test, Variable } from '../src/model'
import { Body, Class, Closure, Describe, Environment, Field, Import, Method, Mixin, NamedArgument, Node, Package, Parameter, ParameterizedType, Reference, Return, Sentence, Singleton, Test, Variable, Literal } from '../src/model'
import * as parse from '../src/parser'
import { linkerAssertions } from './assertions'
import { environmentWithEntities, WREEnvironment } from './utils'
Expand Down Expand Up @@ -464,6 +464,65 @@ describe('Wollok linker', () => {
C.methods[0].sentences[0].should.target(C.fields[0])
})

it('should target references to members inherited from superclass in different packages', () => {
const environment = link([
new Package({
name: 'aaa',
imports: [
new Import({ isGeneric: true, entity: new Reference({ name: 'bbb' }) }),
],
members: [
new Class({
name: 'C',
supertypes: [new ParameterizedType({ reference: new Reference({ name: 'B' }) })],
members: [
new Method({
name: 'm2',
body: new Body({ sentences: [new Literal({ value: '2' })] }),
}),
],
}),
],
}),
new Package({
name: 'bbb',
imports: [
new Import({ isGeneric: true, entity: new Reference({ name: 'zzz' }) }),
],
members: [
new Class({
name: 'B',
supertypes: [new ParameterizedType({ reference: new Reference({ name: 'A' }) })],
members: [
new Method({
name: 'm',
body: new Body({ sentences: [new Reference({ name: 'x' })] }),
}),
],
}),
],
}),
new Package({
name: 'zzz',
members: [
new Class({
name: 'A', members: [
new Field({ name: 'x', isConstant: false }),
],
}),
],
}),
], WREEnvironment)

const C = environment.getNodeByFQN<Class>('aaa.C')
const B = environment.getNodeByFQN<Class>('bbb.B')
const A = environment.getNodeByFQN<Class>('zzz.A')

C.supertypes[0].reference.should.target(B)
B.supertypes[0].reference.should.target(A)
B.methods[0].sentences[0].should.target(A.fields[0])
})

it('should target references overriden on mixins to members inherited from superclass', () => {
const environment = link([
new Package({
Expand Down

0 comments on commit 9d9a306

Please sign in to comment.