"""Planning (Chapters 10-11) """ from .utils import Expr class Action: """ Defines an action schema using preconditions and effects Use this to describe actions in PDDL action is an Expr where variables are given as arguments(args) Precondition and effect are both lists with positive and negated literals Example: precond_pos = [expr("Human(person)"), expr("Hungry(Person)")] precond_neg = [expr("Eaten(food)")] effect_add = [expr("Eaten(food)")] effect_rem = [expr("Hungry(person)")] eat = Action(expr("Eat(person, food)"), [precond_pos, precond_neg], [effect_add, effect_rem]) """ def __init__(self, action, precond, effect): self.name = action.op self.args = action.args self.precond_pos = precond[0] self.precond_neg = precond[1] self.effect_add = effect[0] self.effect_rem = effect[1] def __call__(self, kb, args): return self.act(kb, args) def __str__(self): return "{}{!s}".format(self.name, self.args) def substitute(self, e, args): """Replaces variables in expression with their respective Propostional symbol""" new_args = list(e.args) for num, x in enumerate(e.args): for i in range(len(self.args)): if self.args[i] == x: new_args[num] = args[i] return Expr(e.op, *new_args) def check_precond(self, kb, args): """Checks if the precondition is satisfied in the current state""" # check for positive clauses for clause in self.precond_pos: if self.substitute(clause, args) not in kb.clauses: return False # check for negative clauses for clause in self.precond_neg: if self.substitute(clause, args) in kb.clauses: return False return True def act(self, kb, args): """Executes the action on the state's kb""" # check if the preconditions are satisfied if not self.check_precond(kb, args): raise Exception("Action pre-conditions not satisfied") # remove negative literals for clause in self.effect_rem: kb.retract(self.substitute(clause, args)) # add positive literals for clause in self.effect_add: kb.tell(self.substitute(clause, args))