Skip to content

Latest commit

 

History

History
executable file
·
217 lines (125 loc) · 11.9 KB

README.md

File metadata and controls

executable file
·
217 lines (125 loc) · 11.9 KB

Knowledge Base

PART1

First , I created a basic knowledge base (KB) to store and retrieve facts in the following order

  1. Implement storing facts in the KB
  2. Implement retrieving facts from the KB

Introduction

The KB supports two main interfaces for now:

  • Assert: Adds facts to the KB
  • Ask: Asks queries and returns a list of bindings for facts.

Functions

  • main.py: Contains code for testing the KB, which will be implemented as the KnowledgeBase class
  • student_code.py: Contains the KnowledgeBase class
  • logical_classes.py: Contains classes for each type of logical component, e.g., Fact, Rule, etc.
  • util.py: Contains several useful helper functions
  • read.py: Contains functions that read statements from files or terminal.

There are also two data files that contain the facts and rules to be inserted into the KB:

  • statements_kb.txt
  • statements_kb2.txt

The provided tests use statements_kb.txt, and you may use statements_kb2.txt to generate your own tests.

Storing facts

Storing facts simply puts any facts received into a list of facts.

Retrieving facts

The key idea is to find any facts in the KB that match the fact "asked" for. Since the queried fact may contain a variable, matching facts might not be exact matches. To help in finding matching facts, I have a match method in util.py. If a pair of facts match, then this method will return the bindings (in the data structure Bindings) that make the statements unify.

Bindings is a list of pairs (bindings), where each pair is a variable (e.g., '?X') and a value (e.g., 'red'). Since it is a list, there may be multiple pairs. Actually, there needs to be exactly one binding for each variable. For example, in asking for '(color ?X red)', there will be only one binding, the one for '?X'. But the query for '(color ?X ?Y)'' will result in bindings for '?X' and '?Y'. See test 5 for an example of bindings containing more than one variable.

Since there may be many facts that match a queried fact, kb_ask needs to return a list of bindings (in the data structure ListOfBindings) or False if no facts match. ListOfBindings is exactly as the name implies, a list of Bindings, packaged up in a class with convenient accessors and such. See tests 3 and 5 for examples of multiple bindings being returned from kb_ask.

PART2

  1. Implemented the forward-chaining inferences that occur upon asserting facts and rules into the KB - i.e., implement the InferenceEnginer.fc_infer method.
  2. Implemented the Retract interface to remove facts from the KB - i.e., implement the KnowledgeBase.kb_retract method.

Introduction

Extended the knowledge base (KB) and created an inference engine. The KB will now support three main interfaces:

  • Assert: Adds facts or rules to the KB. After you add facts or rules to the KB, the forward-chaining algorithm is used to infer other facts or rules.
  • Ask: Asks queries and returns a list of bindings for facts.
  • Retract: Removes asserted facts from the knowledge base. Also, removes all other facts or rules that are dependent on the removed fact or rule.

The end result of this assignment is a KB that can be used to model a world/game/thing with a static set of rules. Most board games and established businesses (during a short period of time) fell into this category. In this type of KB, asserted rules should be treated as laws, laying the foundation of the game/business logic; they are unquestionable and therefore must never be removed from the KB. Asserted facts could be treated as factual observations about the state of the world/game/thing, situations which hold until they cease to be valid, when they are retracted by us, the users.

Rule currying in fc_infer

The key idea is that we don't just infer new facts - we can also infer new rules.

When we add a new fact to the KB, we check to see if it triggers any rule(s). When we add a new rule, we check to see if it's triggered by existing facts.

However, a rule might have multiple statements on its left-hand side (LHS), and we don't want to iterate each of these statements every time we add a new fact to the KB. Instead, we'll employ a cool trick. Whenever we add a new rule, we'll only check the first element of the LHS of that rule against the facts in our KB. (If we add a new fact, we'll reverse this - we'll examine each rule in our KB and check the first element of its LHS against this new fact.) If there's a match with this first element, we'll add a new rule paired with bindings for that match.

For example, imagine a box-world. Consider a rule stating that if a box ?x is larger than another box ?y, and box ?x is on box ?y, then box ?y is covered. Formally, that looks like:

((sizeIsLess(?y, ?x), on(?x, ?y)) => covered(?y))

Now imagine that we know that box A is bigger than box B; i.e., we have the fact sizeIsLess(B, A) in the KB. The above rule then matches, with the bindings ((?x: A, ?y: B)). With that binding in place, we can now infer a new rule that uses it:

(on(A, B)) => covered(B)

If we find the fact on(A, B) in the KB, then we can use this rule to infer the fact covered(B). If we don't have that fact, however, we now have a simple rule that will let us make the inference easily if we see that fact in the future.

Removing rules and facts inferred from a removed fact

When you remove a fact, you also need to remove all facts and rules that were inferred using this fact. However, a given fact/rule might be supported by multiple facts - so, you'll need to check whether the facts/rules inferred from this fact are also supported by other facts (or if they were directly asserted).

As a simplification, you can assume that no rules will create circular dependencies. E.g., imagine a situation like A => B, B => C, and C => B. Removing A would mean removing B and C, since they depend on A via those rules. However, implementing that would get messy, since B and C depend on each other. You will NOT be given scenarios like this.

Appendix: File Breakdown

Below is a description of each included file and the classes contained within each including a listing of their attributes. Each file has documentation in the code reflecting the information below (in most cases they are exactly the same).

Attributes of each class are listed in the following format (Note: If you see a type like Fact|Rule the | type is or and means that the type can be either Fact or Rule):

  • field_name (type) - text description

logical_classes.py

This file defines all basic structure classes.

Fact

Represents a fact in our knowledge base (KB). Has a statement containing the content of the fact, e.g., (isa Sorceress Wizard) and fields tracking which facts/rules in the KB it supports and is supported by.

Attributes

  • name (str): 'fact', the name of this class
  • statement (Statement): statement of this fact, basically what the fact actually says
  • asserted (bool): flag indicating if fact was asserted instead of inferred from other rules in the KB
  • supported_by (listof Fact|Rule): Facts/Rules that allow inference of the statement
  • supports_facts (listof Fact): Facts that this fact supports
  • supports_rules (listof Rule): Rules that this fact supports

Rule

Represents a rule in our KB. Has a list of statements (the left-hand side or LHS) containing the statements that need to be in our KB for us to infer the right-hand-side or RHS statement. Also has fields tracking which facts/rules in the KB it supports and is supported by.

Attributes

  • name (str): 'rule', the name of this class
  • lhs (listof Statement): LHS statements of this rule
  • rhs (Statement): RHS statement of this rule
  • asserted (bool): flag indicating if rule was asserted instead of inferred from other rules/facts in the KB
  • supported_by (listof Fact|Rule): Facts/Rules that allow inference of the statement
  • supports_facts (listof Fact): Facts that this rule supports
  • supports_rules (listof Rule): Rules that this rule supports

Statement

Represents a statement in our KB, e.g., (attacked Ai Nosliw), (diamonds Loot), (isa Sorceress Wizard), etc. These statements show up in Facts or on the LHS and RHS of Rules.

Attributes

  • predicate (str) - the predicate of the statement, e.g., isa, hero, needs
  • terms (listof Term) - list of terms (Variable or Constant) in the statement, e.g., 'Nosliw' or '?d'

Term

Represents a term (a Variable or a Constant) in our KB. It could be thought of as a super class of Variable and Constant, though there is no actual inheritance implemented in the code.

Attributes

  • term (Variable|Constant) - the Variable or Constant that this term holds (represents)

Variable

Represents a variable used in statements, e.g., ?x.

Attributes

  • element (str): the name of the variable, e.g., '?x'

Constant

Represents a constant used in statements.

Attributes

  • element (str): the value of the constant, e.g., 'Nosliw'

Binding

Represents a binding of a constant to a variable, e.g., 'Nosliw' might be bound to '?d'.

Attributes

  • variable (str): the name of the variable associated with this binding, e.g., '?d'
  • constant (str): the value of the variable, e.g., 'Nosliw'

Bindings

Represents Binding(s) used while matching two statements.

Attributes

  • bindings (listof Bindings) - bindings involved in match
  • bindings_dict (dictof Bindings) - bindings involved in match where key is bound variable and value is bound value, e.g., some_bindings.bindings_dict['?d'] => 'Nosliw'

Methods

  • add_binding(variable, value) ((Variable, Constant) => void) - Add a binding from a variable to a value.
  • bound_to(variable) ((Variable) => Variable|Constant|False) - Check if variable is bound. If so, return value bound to it, else False.
  • test_and_bind(variable_verm,value_term) ((Term, Term) => bool) - Check if variable_term already bound. If so, return whether or not passed-in value_term matches bound value. If not, add binding between variable_terma and value_term, and return True.

ListOfBindings

Container for multiple Bindings

Methods

  • add_bindings(bindings, facts_rules) - ((Bindings, listof Fact|Rule) => void) - Add given bindings to list of Bindings along with associated rules or facts.

read.py

This file has no classes but defines useful helper functions for reading input from the user or a file.

Functions

  • read_tokenize(file) - ((str) => (listof Fact, listof Rule)) - Takes a filename, reads the file, and returns a fact list and a rule list.
  • parse_input(e) - ((str) => (int, str | listof str)) - Parses input (cleaning it as it does so), assigning labels and splitting rules into LHS and RHS.

util.py

This file has no classes but defines useful helper functions.

Functions

  • is_var(var) ((str|Variable|Constant|Term) => bool) - Check whether an element is a variable (either instance of Variable, instance of Term (where .term is a Variable) or a string starting with '?', e.g., '?d').
  • match(state1, state2, bindings=None) ((Statement, Statement, Bindings) => Bindings|False) - Match two statements, and return the associated bindings or False if there is no binding.
  • match_recursive(terms1, terms2, bindings) ((listof Term, listof Term, Bindings) => Bindings|False) - recursive helper for match
  • instantiate(statement, bindings) ((Statement, Bindings) => Statement|Term) - Generate Statement from given statement and bindings. Constructed statement has bound values for variables if they exist in bindings.
  • printv(message, level, verbose, data=[]) ((str, int, int, listof any) => void) - Prints message if verbose > level. If data provided, then formats message with given data.

KnowledgeBase

Represents a knowledge base and contains the two methods described in the writeup (Assert and Ask).