fix: Make search work with automatic mode
This commit is contained in:
parent
6f4c26ecee
commit
8abe689a0a
|
@ -19,6 +19,7 @@ poetry build
|
|||
To run server tests:
|
||||
``` bash
|
||||
python -m pantograph.server
|
||||
python -m pantograph.search
|
||||
```
|
||||
The tests in `pantograph/server.py` also serve as simple interaction examples
|
||||
|
||||
|
|
|
@ -21,7 +21,8 @@ class SearchState:
|
|||
|
||||
@property
|
||||
def next_goal_id(self) -> int:
|
||||
goal_id, _ = max([(i, prio) for i, prio in enumerate(self.priorities) if not self.solved[i]],
|
||||
goal_id, _ = max(
|
||||
((i, prio) for i, prio in enumerate(self.priorities) if not self.solved[i]),
|
||||
key=lambda x: x[1])
|
||||
return goal_id
|
||||
|
||||
|
@ -40,11 +41,20 @@ class SearchResult:
|
|||
steps: int
|
||||
|
||||
class Agent:
|
||||
"""
|
||||
An agent interface for proof search
|
||||
"""
|
||||
|
||||
def next_tactic(self, state: GoalState, goal_id: int, informal_stmt:str, informal_proof:str) -> Optional[Tactic]:
|
||||
def next_tactic(
|
||||
self,
|
||||
state: GoalState,
|
||||
goal_id: int,
|
||||
informal_stmt: str,
|
||||
informal_proof: str) -> Optional[Tactic]:
|
||||
"""
|
||||
Implement this function to generate the next tactic for a goal
|
||||
"""
|
||||
return None
|
||||
|
||||
def guidance(self, state: GoalState) -> list[float]:
|
||||
"""
|
||||
|
@ -65,16 +75,22 @@ class Agent:
|
|||
max_steps: int = 100,
|
||||
max_trials_per_goal: int = 5,
|
||||
verbose: bool = False) -> SearchResult:
|
||||
"""
|
||||
Searches using th
|
||||
"""
|
||||
assert server.is_automatic(), "Search must be run in automatic mode"
|
||||
|
||||
search_stack = [SearchState(state=server.goal_start(target),
|
||||
initial_state = SearchState(
|
||||
state=server.goal_start(target),
|
||||
parent=None,
|
||||
parent_goal_id=None,
|
||||
priorities=[0.0])]
|
||||
priorities=[0.0]
|
||||
)
|
||||
search_stack = [initial_state]
|
||||
"""
|
||||
Executes proof search on this state
|
||||
"""
|
||||
for i_step in range(max_steps):
|
||||
|
||||
assert search_stack, "No states in search stack"
|
||||
|
||||
if verbose:
|
||||
|
@ -84,18 +100,8 @@ class Agent:
|
|||
assert isinstance(search_state, SearchState)
|
||||
|
||||
if search_state.is_solved:
|
||||
# if the state is solved, propagate this solved status
|
||||
if search_state.is_root:
|
||||
if verbose:
|
||||
print("Search complete: Root state solved")
|
||||
self.reset()
|
||||
return SearchResult(success=True, steps=i_step)
|
||||
|
||||
search_stack.pop(-1)
|
||||
assert not search_stack[search_state.parent].solved[search_state.parent_goal_id]
|
||||
search_stack[search_state.parent].solved[search_state.parent_goal_id] = True
|
||||
continue
|
||||
|
||||
# Find the unsolved goal with the highest priority
|
||||
goal_id = search_state.next_goal_id
|
||||
|
||||
|
@ -106,7 +112,8 @@ class Agent:
|
|||
# Generate tactic for this goal
|
||||
tactic = self.next_tactic(search_state.state, goal_id, informal_stmt, informal_proof)
|
||||
|
||||
print("????next tactic: ", tactic)
|
||||
if verbose:
|
||||
print(f"Next tactic: {tactic}")
|
||||
if not tactic:
|
||||
# pop the current state and continue to the next
|
||||
search_stack.pop(-1)
|
||||
|
@ -128,12 +135,16 @@ class Agent:
|
|||
if len(next_goal_state.goals) <= 1 else \
|
||||
self.guidance(next_goal_state)
|
||||
parent = len(search_stack) - 1
|
||||
search_stack.append(SearchState(state=next_goal_state,
|
||||
next_state = SearchState(
|
||||
state=next_goal_state,
|
||||
parent=parent,
|
||||
parent_goal_id=goal_id,
|
||||
priorities=priorities))
|
||||
priorities=priorities
|
||||
)
|
||||
search_stack.append(next_state)
|
||||
|
||||
except TacticFailure as t:
|
||||
if verbose:
|
||||
print(f"Tactic failed: {t}")
|
||||
# try the next tactic. this one failed
|
||||
|
||||
|
@ -163,7 +174,12 @@ class DumbAgent(Agent):
|
|||
"assumption",
|
||||
]
|
||||
|
||||
def next_tactic(self, state: GoalState, goal_id: int, informal_stmt:str, informal_proof:str) -> Optional[Tactic]:
|
||||
def next_tactic(
|
||||
self,
|
||||
state: GoalState,
|
||||
goal_id: int,
|
||||
informal_stmt: str,
|
||||
informal_proof: str) -> Optional[Tactic]:
|
||||
key = (state.state_id, goal_id)
|
||||
i = self.goal_tactic_id_map[key]
|
||||
|
||||
|
@ -188,14 +204,20 @@ class TestSearch(unittest.TestCase):
|
|||
|
||||
server = Server()
|
||||
agent = DumbAgent()
|
||||
flag = agent.search(server=server, target="∀ (p q: Prop), p -> p", verbose=True)
|
||||
flag = agent.search(
|
||||
server=server,
|
||||
target="∀ (p q: Prop), p -> p",
|
||||
verbose=False)
|
||||
#flag = agent.search(server=server, target="∀ (p q: Prop), Or p q -> Or q p", verbose=True)
|
||||
self.assertTrue(flag)
|
||||
def test_solve_big(self):
|
||||
|
||||
server = Server()
|
||||
agent = DumbAgent()
|
||||
flag = agent.search(server=server, target="∀ (p q: Prop), Or p q -> Or q p", verbose=True)
|
||||
flag = agent.search(
|
||||
server=server,
|
||||
target="∀ (p q: Prop), Or p q -> Or q p",
|
||||
verbose=False)
|
||||
self.assertTrue(flag)
|
||||
|
||||
|
||||
|
|
|
@ -51,6 +51,9 @@ class Server:
|
|||
# List of goal states that should be garbage collected
|
||||
self.to_remove_goal_states = []
|
||||
|
||||
def is_automatic(self):
|
||||
return self.options.get("automaticMode", True)
|
||||
|
||||
def restart(self):
|
||||
if self.proc is not None:
|
||||
self.proc.close()
|
||||
|
|
Loading…
Reference in New Issue