mirror of
https://github.com/josdejong/mathjs.git
synced 2025-12-08 19:46:04 +00:00
evaluating a non-existing function, and expose internal functions `FunctionNode.onUndefinedFunction(name)` and `SymbolNode.onUndefinedSymbol(name)`
180 lines
5.4 KiB
JavaScript
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')
|
|
})
|
|
})
|