# Learning Symmetric Rules with SATNet
This repository contains the source code for reproducing the experiments in the paper [Learning Symmetric Rules with SATNet](https://arxiv.org/abs/2206.13998?context=math) by Sangho Lim, Eun-Gyeol Oh, and [Hongseok Yang](https://sites.google.com/view/hongseokyang/home).

## 1. Dependencies
We only provide the GPU implementation of SymSATNet.
The experiment was conducted in a Linux conda environment, and we used python 3.8 and pytorch 1.8.0 with CUDA 11.1 to install SATNet, SymSATNet packages.
We recommend creating a new conda environment and installing those packages on the environment by the following commands.

    conda create -n satnet python=3.8
    conda activate satnet
    conda install pytorch==1.8.0 torchvision==0.9.0 torchaudio==0.8.0 cudatoolkit=11.1 -c pytorch -c conda-forge

Now, we can install SATNet,

    git clone https://github.com/locuslab/SATNet ./src/SATNet
    cd src/SATNet; python setup.py install; cd ../..

and install SymSATNet by the following.

    cd src/SymSATNet; python setup.py install; cd ../..
    
SATNet and SymSATNet also require the following packages.

    conda install -c conda-forge matplotlib
    conda install -c conda-forge packaging
    conda install -c anaconda ipython
    conda install -c pytorch tqdm


## 2. Related Work
- The implementation of SymSATNet is based on [the original SATNet code](https://github.com/locuslab/SATNet) by [Wang et. al.](http://proceedings.mlr.press/v97/wang19e.html) [2019] available under the MIT License which is also contained in this repository.

- We used the Sudoku generator in [this repository](https://github.com/Kyubyong/sudoku) by Park [2018].


## 3. SymSATNet
SymSATNet is a variant of SATNet. It is an abbreviation of symmetry-aware SATNet.
SymSATNet assumes that some symmetries of the target rules are given a priori although the rules themselves are unknown.
These symmetries can be obtained by a domain expert, or an automatic symmetry-detection algorithm, such as our SymFind.
The given symmetries get incorporated into the SATNet's objective, and become 
the equivariance requirement on the parameter matrix of SATNet, which leads to the substantial reduction in the number of parameters to learn in SymSATNet (via 
a basis of the space of equivariant matrices).

| ![sudoku_symmetry](img/sudoku_symmetry.png) |
|:--:|
| Parameters having symmetries in Sudoku |

| ![cube_symmetry](img/cube_symmetry.png) |
|:--:|
| Parameters having symmetries in Rubik's cube |


## 4. SymFind

SymFind is an automatic symmetry-detection algorithm, which receives a matrix $M$
as an input and returns a permutation group $G$
which closely approximates the symmetries in $M$
: $\forall g \in G, \, gM \approx Mg$.
The symmetry group discovered by SymFind can be used to train SymSATNet.
SymFind can find a permutation group expressible by a particular grammar which covers a suitable range of practical permutation groups.
SymFind calls two subroutines, SumFind and ProdFind, which may also recursively call SymFind to find symmetries for smaller parts of $M$. 

|   |   |
|---|---|
| ![sumfind](img/sumfind.gif) | SumFind clusters entries in the input matrix as blocks since block-shaped clusters commonly arise in matrices equivariant with respect to a direct sum of groups. |

| ![prodfind](img/prodfind.png) |
|---|
| ProdFind exploits a typical pattern of Kronecker product of matrices, and detects the presence of the pattern by applying SVD. |


## 5. Tasks
Our first task is Sudoku. Sudoku has group symmetries represented by $G = (S_3 \wr S_3) \otimes (S_3 \wr S_3) \otimes S_9$
. The following are three examples of permutations in $G$
, which preserve the validity of Sudoku solutions.

| ![sudoku_1](img/sudoku_1.png) |
|:--:|
| Column permutation within a stack |

| ![sudoku_2](img/sudoku_2.png) |
|:--:|
| Stack permutation |

| ![sudoku_3](img/sudoku_3.png) |
|:--:|
| Permutation of number occurences |

Our second task is the completion problem of Rubik's cube.
The player is asked to find a colour assignment of the Rubik's cube that is solvable.
This problem has group symmetries represented by $G = R_{54} \otimes R_6$
. Here $R_{54}$
is the Rubik's cube group, and $R_6$
is the permutation group generated by 3-dimensional $90^\circ$
rotations.

| ![cube](img/cube.png) |
|:--:|
| Completion problem of Rubik's cube |


## 6. Training and Results
The following two plots show the test accuracy of each model in Sudoku and Rubik's cube during the 100 epochs of training.

| ![sudoku_accuracy](img/sudoku_accuracy.png) | ![cube_accraucy](img/cube_accuracy.png) |
|:--:|:--:|
| Test accuracy in Sudoku | Test accuracy in Rubik's cube |

Also, the following two plots show the test accuracy of each model in noisy Sudoku and noisy Rubik's cube with respect to the number of corrupted cells or facelets.

| ![sudoku_robustness](img/sudoku_robustness.png) | ![cube_robustness](img/cube_robustness.png) |
|:--:|:--:|
| Robustness in noisy Sudoku | Robustness in noisy Rubik's cube |

See [main.py](main.py) and [experiment.ipynb](experiment.ipynb) for more information.
