Hi, welcome to my regex decision equivalence project!
Essentially, this project is an implementation of two regular expression decision procedures: one by automata construction and one by Brzozowski derivatives.
It will allow you to tell if two regular expressions represent the same regular language. For example, it is a theorem in regular language theory that "(1 + x)(1 + x)(xxx)* = x*" where "s*" represents the language {"", "s", "ss", "sss", "ssss", "sssss", ...}. This program will compute that theorem to be true.
This was implemented in OCaml. This program also includes a relatively basic interpreter for my language of regular expressions.
Ast.ml contains the type for regular expressions, as well as functions that flatten regular expressions, so we don't have something like Sum [Sum _; Sum _]. We also have a functions that normalize expressions, which comes in useful for ensuring termination from recursively running Brzowski derivatives.
Automata.ml contains the types of DFAs, NFAs with epsilon closures, and NFAs without. Here contains most of the functions related to converting expressions to automaton, as well as converting automatons from NFA to DFA. I use Thompson's construction algorithm to convert from expressions to NFAs with epsilon closures and powerset construction to go from that to DFAs.
Derivative.ml contains functions that take Brzozowski derivatives of expressions. There is a lot of commented code, where I attempted to linearize regexes to guarantee termination from repeated derivatives. This was before I introduced a function to normalize expressions. Getting Brzowski derivatives to terminate turned out to be quite tricky.
Parse.ml contains functions that allow conversion from strings to regexes. For example, using [parse_exp "a^ * b^ + c"] will output Sum [Prod [Star (Char 'a'); Star (Char 'b')]; Char 'c']. It might be a little confusing, but here, "*" represents multiplication or concatenation while "^" represents the Kleene star operation. Parse.ml depends on menhir being installed to do the parsing.
Unionfind.ml contains definitions of the unionfind data structure and functions associated with it.
Equivalence.ml contains the main decision functions. It relies in the near-linear time bisimulation algorithm by Hopcroft and Karp, and it uses the unionfind data.There are two decision functions here, [decide_w_automaton] and [decide_w_brzowski]. These two functions both take in two strings, and will output true or false if those strings correspond to eqiuvalent regular languages.
To see a quick demo, you can run [dune exec bin/main.exe], which will show a couple examples of the decision algorithms being used. [dune test] will also show a couple more examples in the form of ounit tests.
To run your own examples, you could do so in utop. To get to the using the decision functions, in the root of the directory, input the following commands into terminal: "dune utop" "open Regex.Equivalence"
The syntax I use for regexes is a little different from what you may encounter in other regex engines. "0" represents the emptyset. "1" represents the empty string, so does "". "a + b" represents {"a", "b"} "a * b" represents concatenation, so in this case {"ab"} "a^" represents the Kleene star operation, so this would be {"", "a", "aa", "aaa", ...}.