mathjs/test/unit-tests/expression/node/SymbolNode.test.js
Jos de Jong c6cbf55389 Fix #2150, #660: give a clear error "Error: Undefined function ..." instead when
evaluating a non-existing function, and expose internal functions `FunctionNode.onUndefinedFunction(name)` and `SymbolNode.onUndefinedSymbol(name)`
2021-04-10 13:38:04 +02:00

180 lines
5.4 KiB
JavaScript

// test SymbolNode
import assert from 'assert'
import math from '../../../../src/defaultInstance.js'
const Node = math.Node
const ConstantNode = math.ConstantNode
const SymbolNode = math.SymbolNode
const OperatorNode = math.OperatorNode
describe('SymbolNode', function () {
it('should create a SymbolNode', function () {
const n = new SymbolNode('sqrt')
assert(n instanceof SymbolNode)
assert(n instanceof Node)
assert.strictEqual(n.type, 'SymbolNode')
})
it('should have isSymbolNode', function () {
const node = new SymbolNode('a')
assert(node.isSymbolNode)
})
it('should throw an error when calling without new operator', function () {
assert.throws(function () { SymbolNode('sqrt') }, SyntaxError)
})
it('should throw an error when calling with wrong arguments', function () {
assert.throws(function () { console.log(new SymbolNode()) }, TypeError)
assert.throws(function () { console.log(new SymbolNode(2)) }, TypeError)
})
it('should throw an error when evaluating an undefined symbol', function () {
const scope = {}
const s = new SymbolNode('foo')
assert.throws(function () { s.compile().evaluate(scope) }, /Error: Undefined symbol foo/)
})
it('should compile a SymbolNode', function () {
const s = new SymbolNode('a')
const expr = s.compile()
const scope = { a: 5 }
assert.strictEqual(expr.evaluate(scope), 5)
assert.throws(function () { expr.evaluate({}) }, Error)
const s2 = new SymbolNode('sqrt')
const expr2 = s2.compile()
const scope2 = {}
assert.strictEqual(expr2.evaluate(scope2), math.sqrt)
})
it('should filter a SymbolNode', function () {
const n = new SymbolNode('x')
assert.deepStrictEqual(n.filter(function (node) { return node instanceof SymbolNode }), [n])
assert.deepStrictEqual(n.filter(function (node) { return node.name === 'x' }), [n])
assert.deepStrictEqual(n.filter(function (node) { return node.name === 'q' }), [])
assert.deepStrictEqual(n.filter(function (node) { return node instanceof ConstantNode }), [])
})
it('should run forEach on a SymbolNode', function () {
const a = new SymbolNode('a')
a.forEach(function () {
assert.ok(false, 'should not execute, symbol has no childs')
})
})
it('should map a SymbolNode', function () {
const a = new SymbolNode('a')
const b = a.map(function () {
assert.ok(false, 'should not execute, symbol has no childs')
return undefined
})
assert.notStrictEqual(b, a)
assert.deepStrictEqual(b, a)
})
it('should transform a SymbolNode', function () {
const a = new SymbolNode('x')
const b = new SymbolNode('y')
const c = a.transform(function (node) {
return node instanceof SymbolNode && node.name === 'x' ? b : node
})
assert.deepStrictEqual(c, b)
// no match should leave the symbol as is
const d = a.transform(function (node) {
return node instanceof SymbolNode && node.name === 'q' ? b : node
})
assert.deepStrictEqual(d, a)
})
it('should clone a SymbolNode', function () {
const a = new SymbolNode('x')
const b = a.clone()
assert(b instanceof SymbolNode)
assert.deepStrictEqual(a, b)
assert.notStrictEqual(a, b)
assert.strictEqual(a.name, b.name)
})
it('test equality another Node', function () {
const a = new SymbolNode('a')
const b = new SymbolNode('b')
const aEqual = new SymbolNode('a')
const aFake = {
name: 'a'
}
assert.strictEqual(a.equals(null), false)
assert.strictEqual(a.equals(undefined), false)
assert.strictEqual(a.equals(aEqual), true)
assert.strictEqual(a.equals(b), false)
assert.strictEqual(a.equals(aFake), false)
assert.strictEqual(a.equals(new ConstantNode(2)), false)
})
it('should stringify a SymbolNode', function () {
const s = new SymbolNode('foo')
assert.strictEqual(s.toString(), 'foo')
})
it('should stringigy a SymbolNode with custom toString', function () {
// Also checks if the custom functions get passed on to the children
const customFunction = function (node, options) {
if (node.type === 'SymbolNode') {
return 'symbol(' + node.name + ')'
}
}
const n = new SymbolNode('a')
assert.strictEqual(n.toString({ handler: customFunction }), 'symbol(a)')
})
it('toJSON and fromJSON', function () {
const a = new SymbolNode('a')
const json = a.toJSON()
assert.deepStrictEqual(json, {
mathjs: 'SymbolNode',
name: 'a'
})
const parsed = SymbolNode.fromJSON(json)
assert.deepStrictEqual(parsed, a)
})
it('should LaTeX a SymbolNode', function () {
assert.strictEqual(new SymbolNode('foo').toTex(), ' foo')
assert.strictEqual(new SymbolNode('Infinity').toTex(), '\\infty')
})
it('should LaTeX a SymbolNode with custom toTex', function () {
// Also checks if the custom functions get passed on to the children
const customFunction = function (node, options) {
if (node.type === 'SymbolNode') {
return 'symbol(' + node.name + ')'
}
}
const n = new SymbolNode('a')
assert.strictEqual(n.toTex({ handler: customFunction }), 'symbol(a)')
})
it('should LaTeX a SymbolNode without breaking \\cdot', function () {
const a = new ConstantNode(1)
const b = new SymbolNode('Epsilon')
const mult = new OperatorNode('*', 'multiply', [a, b])
assert.strictEqual(mult.toTex(), '1\\cdot E')
})
})