Open
Description
Hello,
While I have been using Symbolics and SymbolicUtils for some time now [and greatly appreciate, having switched from Sympy], I am a novice in the use of Metatheory.
My first attempt to set up Metatheory using TermInterface as described in the documentation is the code below. The test expression is part of a much larger expression which I previously simplified using Symbolics.
Several questions regarding implementation regarding which I would appreciate any insights:
- Does the symbolic expression - Metatheory handle indexed coefficients such as c[0:n]?
- What does the expression
@commutative_monoid (*) 1
mean? - In the
@theory
term-rewriting, does my addition(a / c) + (b / c) --> (a + b) / c
make sense? - What is the symbolic expression
:h
in and the[4]
correspond to inhcall = MyExpr(:h, [4], "hello")
- What does
[2]
correspond to inex = MyExpr(:f, [MyExpr(:z, [2]), hcall])
- How do I set up the following code fragment
hcall = MyExpr(:h, [4], "hello")
ex = MyExpr(:f, [MyExpr(:z, [2]), hcall])
g = EGraph(ex; keepmeta = true)
for my use case?
Julia 1.9.4
Metatheory 2.0.2
# test_metatheory.jl
using Metatheory
import Metatheory: @rule
using Metatheory.EGraphs
using Metatheory.Library
using TermInterface
using Test
# Custom Expr type
struct MetaExpr
head::Any
args::Vector{Any}
meta_str::String # additional metadata
end
# Define equality for MetaExpr expression
function Base.:(==)(a::MetaExpr, b::MetaExpr)
a.head == b.head && a.args == b.args && a.meta_str == b.meta_str
end
# create a term that is in the same closure of types of x
function TermInterface.similarterm(
x::MetaExpr,
head,
args;
metadata = nothing,
exprhead = :call)
MetaExpr(head, args, isnothing(metadata) ? "" : metadata)
end
# Add metadata information for compatibility with SymbolicUtils.jl
function EGraphs.egraph_reconstruct_expression(
::Type{MetaExpr},
op,
args;
metadata = nothing,
exprhead = nothing)
MetaExpr(op, args, (isnothing(metadata) ? () : metadata))
end
begin
MetaExpr(head, args) = MetaExpr(head, args, "")
MetaExpr(head) = MetaExpr(head, [])
# Override istree()
TermInterface.istree(::MetaExpr) = true
# Specify a node's represented operation
TermInterface.exprhead(::MetaExpr) = :call
# Specify how to extract the children nodes
TermInterface.arguments(e::MetaExpr) = e.args
# Specify that all expressions of type MetaExpr can be represented (and matched against)
# by a pattern that is represented by a :call Expr.
TermInterface.metadata(e::MetaExpr) = e.meta_str
cmmttv_monoid = @commutative_monoid (*) 1;
t = @theory a b c begin
a + 0 --> a
a + b --> b + a
a + inv(a) --> 0 # inverse
a + (b + c) --> (a + b) + c
a * (b + c) --> (a * b) + (a * c)
(a * b) + (a * c) --> a * (b + c)
a * a --> a^2
a --> a^1
a^b * a^c --> a^(b+c)
(a / c) + (b / c) --> (a + b) / c
a::Number + b::Number => a + b
a::Number * b::Number => a * b
end
t = cmmttv_monoid ∪ t;
n = 6
@variables z
@variables c[0:n]
# Test expression - simplify to rational expression
# Known simplification result:
# expr =
# (-c[0]*c[1]*c[2]*(c[4]^2)*c[6]*(z^2) + c[0]*c[1]*c[2]*c[4]*(c[5]^2)*(z^2) + c[0]*c[1]*(c[3]^2)*c[4]*c[6]*(z^2)
# - c[0]*c[1]*(c[3]^2)*(c[5]^2)*(z^2) + c[0]*(c[2]^2)*c[3]*c[4]*c[6]*(z^2) - c[0]*(c[2]^2)*(c[4]^2)*c[5]*(z^2)
# - c[0]*c[2]*(c[3]^3)*c[6]*(z^2) + c[0]*c[2]*c[3]*(c[4]^3)*(z^2) + c[0]*(c[3]^4)*c[5]*(z^2) - c[0]*(c[3]^3)*(c[4]^2)*(z^2))
# / ((-c[1]*c[3]*c[5] + c[1]*(c[4]^2) + (c[2]^2)*c[5] - 2.0c[2]*c[3]*c[4] + c[3]^3)*c[3])
expr =
:(
((((-c[0]*(c[4]^3)) / c[3] + (-c[0]*c[2]*(c[5]^2)) / c[3] + (c[0]*c[2]*(c[4]^2)*c[5]) / (c[3]^2) +
c[0]*c[4]*c[5])*(z^3)) / (-c[3] + (c[2]*c[4]) / c[3]) + (-c[0]*c[4]*c[5]*(z^3)) / c[3] +
c[0]*c[6]*(z^3)) / (-c[3] + (c[1]*c[5]) / c[3] + ((c[1]*(c[4]^2)) / c[3] + ((c[2]^2)*c[5]) / c[3] +
(-c[1]*c[2]*c[4]*c[5]) / (c[3]^2) - c[2]*c[4]) / (-c[3] + (c[2]*c[4]) / c[3]))
)
@show expr
println("")
#= How to set up the following code for the above expr?
hcall = MetaExpr(:h, [4], "hello")
ex = MetaExpr(:f, [MetaExpr(:z, [2]), hcall])
g = EGraph(ex; keepmeta = true)
=#
end
Metadata
Assignees
Labels
No labels