File size: 11,553 Bytes
2130399
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
# PySoarLib
## Aaron Mininger
### 2018

This is a python library module with code to help make working with Soar SML in python 
just a little bit easier. 

Methods do have docstrings, to read help information open a python shell and type
```
import pysoarlib
help(pysoarlib)
help(pysoarlib.SoarClient)
```

* [SoarClient](#soarclient)
* [Config Settings](#configsettings)
* [AgentConnector](#agentconnector)
* [IdentifierExtensions](#idextensions)
* [WMInterface](#wminterface)
* [SoarWME](#soarwme)
* [SVSCommands](#svscommands)
* [TimeConnector](#timeconnector)
* [util](#util)

<a name="soarclient"></a>
# SoarClient
Defines a class used for creating a soar agent, sending commands, running it, etc.
There are a number of settings that you can use to configure the client (see following subsection) and
can specify either by including them in the config file or by giving as keyword arguments
in the constructor. 

`SoarClient(config_filename=None, print_handler=None, **kwargs)`     
Will create the soar kernel and agent, as well as source the agent files. 
`print_handler` specifies how output is handled (defaults to python print) and 
`config_filename` names a file with client settings in it


`add_connector(AgentConnector, name:str)`    
Adds the given connector and will invoke callbacks on it (such as on_input_phase)

`get_connector(name:str)`    
Returns a connector of the given name, or None

`has_connector(name:str) -> Boolean`    
Returns true if the given connector exists

`add_print_event_handler(handler)`   
Will call the given handler during each soar print event (handler should be a method taking 1 string)

`connect()`     
Will register callbacks (call before running)

`disconnect()`     
Will deregister callbacks

`start()`     
Will cause the agent to start running in new thread (non-blocking)

`stop()`     
Will stop the agent

`execute_command(cmd:str, print_res:bool=False)`     
Sends the given command to the agent and returns the result as a string. If print_res=True it also prints the output using print_handler

`restart()`    
Completely destroys the agent and creates + sources a new one

`kill()`     
Will stop the agent and destroy the agent/kernel


## Config Settings (kwargs or config file)
<a name="configsettings"></a>

These can be passed as keyword arguments to the SoarClient constructor, 
or you can create a config file that contains these settings. 

| Argument           | Type     | Default    | Description                              |
| ------------------ | -------- | ---------- | ---------------------------------------- |
| `agent_name`       | str      | soaragent  | The soar agent's name |
| `agent_source`     | filename |            | The root soar file to source the agent productions  |
| `smem_source`      | filename |            | The root soar file that sources smem add commands |
| `source_output`    | enum str | summary    | How much detail to print when sourcing files: none, summary, or full |
| `watch_level`      | int      | 1          | Sets the soar watch/trace level, how much to print each DC (0=none) |
| `spawn_debugger`   | bool     | false      | If true, spawns the soar java debugger |
| `start_running`    | bool     | false      | If true, will automatically start running the agent |
| `write_to_stdout`  | bool     | false      | If true, will print all soar output to the print_handler |
| `print_handler`    | method   | print      | A method taking 1 string arg, handles agent output |
| `enable_log`       | bool     | false      | If true, writes all soar/agent output to a file |
| `log_filename`     | filename | agent-log.txt | The name of the log file to create |
| **time settings** <a name="timesettings"></a> |          |            |               |
| `use_time_connector`| bool    | false      | If true, creates a TimeConnector to put time info on the input-link |
| `clock_include_ms` | bool     | true       | Will include milliseconds for elapsed and clock times |
| `sim_clock`        | bool     | false      | If false, the clock shows real time. If true, it advances a fixed amount each DC |
| `clock_step_ms`    | int      | 50         | The number of milliseconds the simulated clock advances each decision cycle |

Instead of passing as arguments, you can include them in a file specified by config_filename
Each line in the file should be 'setting = value'

Example File:    
```
agent_name = Rosie    
agent_source = agent.soar    
spawn_debugger = false    
```


<a name="agentconnector"></a>
# AgentConnector
Defines an abstract base class for creating classes that connect to Soar's input/output links

`AgentConnector(client:SoarClient)`     


`add_output_command(command_name:str)`     
Will register a handler that listens to output link commands with the given name

`add_print_event_handler(handler:func)`     
Will register a print event handler (function taking 1 string argument) that will be called whenever a soar print event occurs. 

`on_init_soar()`     
Event Handler called when init-soar happens (need to release SML working memory objects)

`on_input_phase(input_link:Identifier)`     
Event Handler called every input phase

`on_output_event(command_name, root_id)`     
Event Handler called when a new output link command is created `(<output-link> ^command_name <root_id>)`





<a name="idextensions"></a>
# IdentifierExtensions 
These add a few helper methods to the sml.Identifier class:
(Do not need to import directly, come with any import from module)

`Identifier.GetChildString(attribute:str)`     
Given an attribute, will look for a child WME of the form `(<id> ^attribute <value>)` and return the value as a string

`Identifier.GetChildInt(attribute:str)`     
Given an attribute, will look for a child IntegerWME of the form `(<id> ^attribute <value>)` and return the value as an integer

`Identifier.GetChildFloat(attribute:str)`     
Given an attribute, will look for a child FloatWME of the form `(<id> ^attribute <value>)` and return the value as an float

`Identifier.GetChildId(attribute:str)`     
Given an attribute, will look for a child WME of the form `(<id> ^attribute <child_id>)` and return an Identifier with child_id as the root

`Identifier.GetAllChildIds(attribute:str=None)`     
Given an attribute, returns a list of Identifiers from all child WME's matching `(<id> ^attribute <child_id>)`
If no attribute is specified, all child Identifiers are returned

`Identifier.GetAllChildValues(attribute:str=None)`     
Given an attribute, returns a list of strings from all child WME's matching `(<id> ^attribute <value>)`
If no attribute is specified, all child WME values (non-identifiers) are returned

`Identifier.GetAllChildWmes()`     
Returns a list of (attr, val) tuples representing all wmes rooted at this identifier.
val will either be an Identifier or a string, depending on its type """


<a name="wminterface"></a>
# WMInterface:    
An interface class which defines a standard way of adding/removing structures from working memory:

`is_added()`    
Returns True if the structure is currently added to working memory

`add_to_wm(parent_id)`    
Adds the structure to working memory under the given identifier

`update_wm(parent_id=None)`    
Applies any changes to working memory    
Note, if a `parent_id` is given and the item is not yet added to wm, it will add it

`remove_from_wm()`    
Removes the structure from working memory


<a name="soarwme"></a>
# SoarWME:    
A class which can represent a WMElement with an `(<id> ^att value)` but takes care of actually interfacing with working memory

You can update its value whenever you want, it will not affect working memory. To change working memory, call `add_to_wm`, `update_wm`, and `remove_from_wm` during an event callback (like BEFORE_INPUT_PHASE)


<a name="svscommands"></a>
# SVSCommands:
A collection of helper functions to create string commands that can be send to SVS
Here pos, rot, and scl are lists of 3 numbers (like [1, 2.5, 3.1])

* `add_box(obj_id, pos=None, rot=None, scl=None)`
* `change_pos(obj_id, pos)`
* `change_rot(obj_id, rot)`
* `change_scl(obj_id, scl)`
* `delete(obj_id)`
* `add_tag(obj_id, tag_name, tag_value)`
* `change_tag(obj_id, tag_name, tag_value)`
* `delete_tag(obj_id, tag_name)`

<a name="timeconnector"></a>
# TimeConnector
An AgentConnector that will create time info on the input-link. 
Includes elapsed time since the agent started, and can have a real-time or simulated wall clock. 
It is enabled through the client setting `use-time-connector=True`
There are several settings that control its behavior as described in [Config Settings](#timesettings). 


```
# Will add and update the following on the input-link:
([il] ^time [t])
([t] ^seconds [secs] # real-time seconds elapsed since start of agent
     ^milliseconds [ms] # real-time milliseconds elapsed since start
     ^steps [steps] # number of decision cycles since start of agent
     ^clock [clock])
([clock] ^hour [hr] # 0-23
         ^minute [min] # 0-59
         ^second [sec] # 0-59
         ^millisecond [ms] # 0-999
         ^epoch [sec] # Unix epoch time in seconds)
```

Also, if using a simulated clock, the agent can change the time itself using an output command:
```
([out] ^set-time [cmd])
([cmd] ^hour 9
       ^minute 15
       ^second 30) # optional 
```

<a name="util"></a>
# pysoarlib.util
Package containing several utility functions for reading/writing working memory through sml structures.

#### `parse_wm_printout(text:str)`   

Given a printout of soar's working memory (p S1 -d 4), parses it into a dictionary of wmes, 
where the keys are identifiers, and the values are lists of wme triples rooted at that id.

You can wrap the result with a PrintoutIdentifier(wmes, root_id) which will provide an Identifier-like
iterface for crawling over the graph structure. It provides all the methods in the IdentifierExtensions interface.


#### `extract_wm_graph(root_id, max_depth)`

Recursively explores all working memory reachable from the given root_id (up to max_depth),
builds up a graph structure representing all that information. 

Note: max_depth is optional (defaults to no depth limit), and the function is smart about handling cycles (will not recurse forever)

```
# Returns a WMNode object wrapping the root_id and containing links to children
node.id = root_id (Identifier)
node.symbol = string (The root_id symbol e.g. O34)
node.attributes() - returns a list of child attribute strings
node['attr'] = WMNode   # for child identifiers
node['attr'] = constant # for string, double, or int value
node['attr'] = [ val1, val2, ... ] # for multi-valued attributes 
               (values can be constants or WMNodes)
str(node) - will pretty-print the node and all children recursively
```


#### `update_wm_from_tree(root_id, root_name, input_dict, wme_table)`

Will update working memory using the given `input_dict` as the provided structure rooted at `root_id`. 
Created wme's are stored in the given `wme_table`, which should be a dictionary that is kept across
multiple calls to this function. `root_name` specifies a prefix for each wme name in the wme_table. 

```
# input_dict should have the following structure:
{
  'attr1': getter() <- The value will be the result of calling the given getter function
  'attr2': dict   <- The value will be a child identifier with its own recursive substructure
}
```

#### `remove_tree_from_wm(wme_table)`    
      
Given a wme_table filled by `SoarUtils.update_wm_from_tree`, removes all wmes from working memory