Integrating deep and relational learning through differentiable logic programming.
- This is the official implementation of the concept of Deep Learning with Relational Logic Representations
- instantiated through the framework of Lifted Relational Neural Networks
- This is the (Java) backend, you can find Python frontend at PyNeuralogic
At the core of the framework, there is a custom language you can use to write differentiable programs encoding your learning scenarios, similarly to classic Deep Learning (DL) frameworks (e.g. TensorFlow). However, the language follows a logic programming paradigm and is declarative in nature (it's similar to Datalog). This means that instead of directly encoding the computation graph, you just declare:
- the inputs (and their numeric values, if any)
- i.e. the observed facts/data = objects, structures, knowledge graphs, relational databases, ...
- e.g.
atom(oxygen_1)
,0.3 stable(oxygen)
,8 protons(oxygen)
,1.7 energy(oxygen,leve2)
,[1.2,0,-1] features(oxygen)
,[[0,2.1],[1.7,-1]] bond(oxygen_1,hydrogen_2,covalent_1)
- the outputs (and their expected values - for supervised learning)
- i.e. the queries = classification labels, regression targets, ...
- e.g.
1 class
,4.7 target(molecule_3)
,0 relation(carbon,xenon,fluor)
- a set of rules applicable in your domain (and their learnable parameters
W
)- i.e. the generic prior/knowledge/bias which you want to use.
- these rules will be used to (automatically) infer (link) the outputs from the inputs
- e.g. explicit/interpretable rules such as
0.99 covalent(B) <= oxygen(X), hydrogen(Y), bond(X,Y,B).
- or more implicit/flexible concepts such as
embed(X) <= W_1 embed(Y), bond(X,Y,_).
- with these you can easily encode a lot of diverse structured deep learning models!
- i.e. the generic prior/knowledge/bias which you want to use.
Consider a simple program for learning with molecular data1, encoding a generic idea that some hidden representation (predicate h(.)
) of any chemical atom (variable X
) is somewhat dependent on the other atoms (a(Y)
) bound to it (b(X,Y)
), with a parameterized rule as:
W_h1 h(X) <= W_a a(Y), W_b b(X,Y).
Additionally, let's assume that representation of a molecule (q
) follows from representations of all the contained atoms (h(X)
), i.e.:
W_q q <= W_h2 h(X).
These 2 rules, parameterized with the tensors W_*
's, then form a learning program, which can be used to classify molecules. Actually, it directly encodes a popular idea known as Graph Neural Networks.
Execution of this program ("template") for 2 input molecule samples will generate 2 parameterized computational graphs as follows:
Each computation node in the graphs is associated with some (differentiable) activation function defined by a user (or settings).
The parameters W_*
in the program are then automatically optimized to reflect the expected output values (A_q
) through gradient descent.
For detailed syntax and semantics, please check out the concept of "Lifted Relational Neural Networks". For a deep dive into the principles in full scientific context, please see my dissertation thesis or the book on Deep learning with relational logic representations.
1: Note that NeuraLogic is by no means designed or limited to learning with chemical data/models/knowledge, but we use it as an example domain here for consistency.
While the framework can be used to encode anything from MLPs, CNNs, RNNs, etc., it is not well suited for classic deep learning with regular data based on large homogeneous tensor operations. The framework is rather meant for efficient encoding of deep relational learning scenarios2, i.e. using dynamic (weight-sharing) neural networks to learn from data with irregular structure(s). That is why we exploit the declarative language with first-order expressiveness, as it allows for compact encoding of complex relational scenarios (similarly to how Prolog can be elegant for relational problems, while not so much for classic programming).
The framework is mostly optimized for quick, high-level prototyping of learning scenarios with sparse, irregular, relational data and complex, dynamically structured models. Particularly, you may find it beneficial for encoding various:
- Graph neural networks
- where you can use it to go well beyond the existing models
- Knowledge base completions
- with complex models requiring e.g. chained inference
- learning with Relational background knowledge/bias
- which can be as expressive as the models themselves
- approaches from Neural-symbolic integration
- for combining (fuzzy) logic inference with neural networks
- and other more crazy ideas, such as learning with
- hypergraphs and ontologies
- recursion
- e.g. for latent type hierarchies
- and generic latent logic programs
2: if you come from deep learning background, you may be familiar with related terms such as "Geometric deep learning" or "Graph representation learning" (but this framework is not limited to graphs only).
Java ≥ 1.8
- continuously tested with JDK 1.8 on the latest Ubuntu, Windows and macOS3
- download a release into some directory
DIR
- or build from source with Maven or IntelliJ IDEA
- clone this repository (or just download the Resources/datasets directory) within
DIR
git clone https://github.com/GustikS/NeuraLogic
- try some trivial examples from terminal in
DIR
- a simple familiar XOR problem
- scalar:
java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/neural/xor/naive
- vectorized:
java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/neural/xor/vectorized
- scalar:
- a simple relational problem (Example 2 from this paper)
java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/simple/family
- molecule classification problem (mutagenesis)
java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/relational/molecules/mutagenesis -ts 100
- knowledge-base completion problem (countries)
java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/relational/kbs/nations
- a simple familiar XOR problem
- you can check various exported settings and results in
DIR/target
- if you have Graphviz installed (
which dot
), you can observe the internal computation structures in debug mode:java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/neural/xor/naive -iso -1 -prune -1 -debug all
- this should show a graph of the 1) worflow, 2) template, 3) inference graphs, 4) neural networks + weight updates, and 5) final learned template
java -jar NeuraLogic.jar -sd ./NeuraLogic/Resources/datasets/simple/family -iso -1 -prune -1 -debug all
- note we turn off network pruning and compression here so that you can observe direct correspondence to the original example
- if you have Graphviz installed (
3: if you don't have Java in your system already, get it either from Oracle or OpenJDK
- for simple usage, it is enough to get the runtime environment (JRE)
- for developers - you can clone and open this project directly in IntelliJ IDEA
usage: java -jar NeuraLogic.jar
-lc,--logColors <INT> colored output on console, best on white background {0,INT} (default: 1)
-sf,--sourcesFile <FILE> path to json Sources specification file (default: sources.json)
-sd,--sourcesDir <DIR> path to directory with all the standardly-named Source files for learning (default: .)
-t,--template <FILE> Template file containing weighted rules for leaning (default: template.txt)
-q,--trainQueries <FILE> file containing labeled training Queries (default: trainQueries.txt)
-e,--trainExamples <FILE> file containing, possibly labeled, input training Facts (default: trainExamples.txt)
-vq,--valQueries <FILE> file containing labeled validation Queries (default: valQueries.txt)
-ve,--valExamples <FILE> file containing, possibly labeled, input validation Facts (default: valExamples.txt)
-te,--testExamples <FILE> file with, possibly labeled, test Examples (default: testExamples.txt)
-tq,--testQueries <FILE> file with test Queries (default: testQueries.txt)
-fp,--foldPrefix <STRING> folds folder names prefix (default: fold_)
-xval,--crossvalidation <INT> number of folds to split for crossvalidation (default: 5)
-set,--settingsFile <FILE> path to json file with all the Settings (default: settings.json)
-out,--outputFolder <DIR> output folder for logging and exporting (default: ./target/out)
-mode,--pipelineMode <ENUM> main mode of the program {complete, neuralization, debug} (default: complete)
-debug,--debugMode <ENUM> debug some objects within the Pipeline during the run {template, grounding, neuralization, samples, model, all} (default: all)
-lim,--limitExamples <INT> limit examples to some smaller number, used e.g. for debugging {-1,INT} (default: -1)
-seed,--randomSeed <INT> int seed for random generator (default: 0)
-gm,--groundingMode <ENUM> groundings mode {independent, sequential, global} (default: independent)
-dist,--distribution <ENUM> distribution for weight initialization {uniform, normal, longtail, constant} (default: uniform)
-init,--initialization <ENUM> algorithm for weight initialization {simple, glorot, he} (default: simple)
-opt,--optimizer <ENUM> optimization algorithm {sgd, adam} (default: ADAM)
-lr,--learningRate <FLOAT> initial learning rate (default: 0.001)
-ts,--trainingSteps <INT> cumulative number of epochae in neural training (default: 3000)
-rec,--recalculationEpocha <INT> recalculate true training and validation error+stats every {INT} epochae (default: 10)
-decay,--learnRateDecay <FLOAT> learning rate decay geometric coefficient {-1,FLOAT} (default: 0.95)
-decays,--decaySteps <INT> learning rate decays every {-1,INT} steps (default: 100)
-preft,--preferTraining <INT> turn on to force best training model selection as opposed to (default) selecting best validation error model {0,1} (default: 0)
-atomf,--atomFunction <ENUM> activation function for atom neurons {sigmoid,tanh,relu,identity,...} (default: tanh)
-rulef,--ruleFunction <ENUM> activation function for rule neurons {sigmoid,tanh,relu,identity,...} (default: tanh)
-aggf,--aggFunction <ENUM> aggregation function for aggregation neurons {avg,max,sum,...} (default: avg)
-em,--evaluationMode <ENUM> evaluation metrics are either for {regression, classification, kbc} (default: classification)
-ef,--errorFunction <ENUM> type of error function {MSE, XEnt} (default: AVG(SQUARED_DIFF))
-iso,--isoCompression <INT> iso-value network compression (lifting), number of decimal digits (default: 12)
-isoinits,--isoInitializations <INT> number of iso-value initializations for network compression (default: 1)
-isocheck,--losslessCompression <INT> lossless compression isomorphism extra check? {0,1} (default: 0)
-prune,--chainPruning <INT> linear chain network pruning {0,1} (default: 1)
To integrate within your project, you can import as a dependency directly from Maven central or Github packages.
<dependency>
<groupId>io.github.gustiks</groupId>
<artifactId>neuralogic-cli</artifactId>
<version>0.3.0</version>
</dependency>
For the above shown usage, the Main class in the CLI module is your starting point.
Note that this is a multi-module Maven project (with a parent pom.xml). Hence, you can also choose to use the submodules independently (e.g. 'Algebra' for simple math, 'Logic' for first-order inference, 'Neural' for deep learning in Java, 'Pipelines' for creating "MLOps" workflows, etc.).
Module | Description |
---|---|
Algebra | value definitions (scalar/vector/matrix...) with respective mathematical operations and functions |
CLI | simple command line interface to the framework, based on Apache commons' CLI |
Drawing | visualization of templates, inference and neural networks, based on DOT language (Graphviz) |
Frontend | Python scripts for calling some high-level functionalities, reading results, and exporting neural nets to Dynet, based on Py4J (moved to PyNeuralogic) |
Learning | high-level (supervised) machine learning definitions and functions |
Logging | simple logging + (color) formatting, based on the default java.util.logging library |
Logic | subsumption engine providing efficient relational logic grounding, credit to Ondrej Kuzelka |
Logical | the logical part of the integration containing logic-based structures and computation - i.e. weighted logic grounding, extension of the Logic module |
Neural | the neural part of the integration containing neural-based structures and computation - i.e. neural nets processing, backprop and classic DL stuff |
Neuralization | the process of conversion from the logical to the neural structures |
Parsing | definition and parsing of the NeuraLogic language into internal representation, based on ANTLR |
Pipelines | high-level "MLOps" library for creating generic ML workflows (custom made) |
Resources | test resources, templates, datasets, etc. |
Settings | central configuration/validation of all settings and input sources (files) |
Utilities | generic utilities (maths, java DIY, etc.), with Gson serialization and JMH benchmarking |
Workflow | NeuraLogic-specific building blocks for typical ML worflows, based on the Pipelines module |
Beyond Graph Neural Networks with Lifted Relational Neural Networks Machine Learning Journal, 2021
- also presented at IJCLR 2021 and AAAI 2023
Lossless compression of structured convolutional models via lifting ICLR, 2021
Lifted Relational Neural Networks Journal of Artificial Intelligence Research, 2018
Stacked structure learning for lifted relational neural networks Inductive Logic Programming, 2017
Learning predictive categories using lifted relational neural networks Inductive Logic Programming, 2016
Lifted Relational Neural Networks (short version) Cognitive Computing workshop @NIPS, 2015
Deep Learning with Relational Logic Representations (2022)
- A book providing full context for the framework, based on my (freely available) dissertation thesis
Lifted Relational Neural Networks: From Graphs to Deep Relational Learning (2023)
- A book chapter about the framework in the Compendium of Neurosymbolic Artificial Intelligence
Prague Machine learning meetups 2017
Much of the Logic and Utils modules comes from Ondrej Kuzelka's repositories Logic and Utils
For experiments from the paper "Beyond GNNs with LRNNs" (2020), please see a GNNwLRNNs repository.
For experiments from the paper "Lossless Compression of Structured Convolutional Models via Lifting" (2020), please see a NeuraLifting repository.
This is a second generation of the framework4, but it is still work in (very slow) progress. The framework is meant for academic purposes, developed at Intelligent Data Analysis lab at Czech Technical University in Prague. In case of any questions or anything interesting, feel free to contact me at souregus@gmail.com, but please do not expect any sort of professional software support.
4: for reference, you can find previous implementation (2014-2018) in this repository