-
Notifications
You must be signed in to change notification settings - Fork 0
/
world.py
300 lines (240 loc) · 12.6 KB
/
world.py
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
287
288
289
290
291
292
293
294
295
296
297
298
299
"""This module implements the classes needed to represent the fictional world of the game.
The world class includes references to the several components (Items, Locations, Character),
and methods to update according to the detected changes by a language model.
"""
import re
class Component:
"""A class to represent a component of the world.
The components considered in the PAYADOR approach are Items, Locations and Characters.
"""
def __init__ (self, name:str, descriptions: 'list[str]'):
self.name = name
"""the name of the component"""
self.descriptions = descriptions
"""a set of natural language descriptions for the component"""
class Item (Component):
"""A class to represent an Item."""
def __init__ (self, name:str, descriptions: 'list[str]', gettable: bool = True):
super().__init__(name, descriptions)
"""inherited from Component"""
self.gettable = gettable
"""indicates if the Item can be taken by the player"""
class Location (Component):
"""A class to represent a Location in the world."""
def __init__ (self, name:str, descriptions: 'list[str]', items: 'list[Item]' = None, connecting_locations: 'list[Location]' = None):
super().__init__(name, descriptions)
"""inherited from Component"""
self.items = items or []
"""a list of the items available in that location"""
self.connecting_locations = connecting_locations or []
"""a list of the reachable locations from itself."""
self.blocked_locations = {}
"""a dictionary with the name of a location as key and <location,obstacle,symmetric> as value.
A blocked passage between self and a location means that it
will be reachable from [self] after overcoming the [obstacle].
The symmetric variable is a boolean that indicates if, when unblocked,
[self] will also be reachable from [location].
"""
def block_passage(self, location: 'Location', obstacle, symmetric: bool = True):
"""Block a passage between self and location using an obstacle."""
if location in self.connecting_locations:
if location.name not in self.blocked_locations:
self.blocked_locations[location.name] = (location, obstacle, symmetric)
self.connecting_locations = [x for x in self.connecting_locations if x is not location]
else:
raise Exception(f"Error: A blocked passage to {location.name} already exists")
else:
raise Exception(f"Error: Two non-conected locations cannot be blocked")
def unblock_passage(self, location: 'Location'):
"""Unblock a passage between self and location by adding it to the connecting locations of self.
In case that the block was symmetric, self will be added to the connecting locations of location.
"""
if self.blocked_locations[location.name]:
self.connecting_locations += [location]
if self.blocked_locations[location.name][2]:
location.connecting_locations += [self]
del self.blocked_locations[location.name]
else:
raise Exception("Error: That is not a blocked passage")
class Character (Component):
"""A class to represent a character."""
def __init__ (self, name:str, descriptions: 'list[str]', location:Location, inventory: 'list[Item]' = None):
super().__init__(name, descriptions)
"""inherited from Component"""
self.inventory = inventory or []
"""a set of Items the carachter has"""
self.location = location
"""the location of the character"""
def move(self, new_location: Location):
"""Move the character to a new location."""
if new_location in self.location.connecting_locations:
self.location = new_location
else:
raise Exception(f"Error: {new_location.name} is not reachable")
def save_item(self,item: Item, item_location_or_owner):
"""Add an item to the character inventory."""
if item.gettable:
if item not in self.inventory:
self.inventory += [item]
if item_location_or_owner.__class__.__name__ == 'Character':
item_location_or_owner.inventory = [i for i in item_location_or_owner.inventory if i is not item]
elif item_location_or_owner.__class__.__name__ == 'Location':
item_location_or_owner.items = [i for i in item_location_or_owner.items if i is not item]
else:
raise Exception(f"Error: {item.name} is already in your inventory")
else:
raise Exception(f"Error: {item.name} cannot be taken")
def drop_item (self, item: Item):
"""Leave an item in the current location."""
self.inventory = [i for i in self.inventory if i is not item]
self.location.items += [item]
def give_item (self, character: 'Character', item: Item):
"""Give an item to another character."""
try:
character.save_item(item, self)
except Exception as e:
print(e)
class World:
"""A class to represent the fictional world, with references to every component."""
def __init__ (self, player: Character) -> None:
self.items = {}
"""a dictionary of all the Items in the world, with their names as values"""
self.characters = {}
"""a dictionary of all the Characters in the world, with their names as values"""
self.locations = {}
"""a dictionary of all the Locations in the world, with their names as values"""
self.player = player
"""a character for the player"""
def add_location (self,location: Location) -> None:
"""Add a location to the world."""
if location.name in self.locations:
raise Exception(f"Error: Already exists a location called '{location.name}'")
else:
self.locations[location.name] = location
def add_item (self, item: Item) -> None:
"""Add an item to the world."""
if item.name in self.items:
raise Exception(f"Error: Already exists an item called '{item.name}'")
else:
self.items[item.name] = item
def add_character (self, character: Character) -> None:
"""Add a character to the world."""
if character.name in self.characters:
raise Exception(f"Error: Already exists a character called '{character.name}'")
else:
self.characters[character.name] = character
def add_locations (self,locations: 'list[Location]') -> None:
""""Add a set of locations to the world."""
for location in locations:
self.add_location(location)
def add_items (self, items: 'list[Item]') -> None:
"""Add a set of items to the world."""
for item in items:
self.add_item(item)
def add_characters (self, characters: 'list[Character]') -> None:
"""Add a set of characters to the world."""
for character in characters:
self.add_character(character)
def render_world(self, *, detail_components:bool = True) -> str:
"""Return the fictional world as a natural language description, using simple sentences.
The components described are only those the player can see in the current location.
If detail_components is False, then the descriptions for each component are not included.
"""
player_location = self.player.location
reachable_locations = [f"<{p.name}>" for p in player_location.connecting_locations]
blocked_passages = [f"<{p}> blocked by <{player_location.blocked_locations[p][1].name}>" for p in player_location.blocked_locations.keys()]
characters_in_the_scene = [character for character in self.characters.values() if character.location is player_location]
world_description = f'You are in <{player_location.name}>\n'
if reachable_locations:
world_description += f'From <{player_location.name}> you can access: {(", ").join(reachable_locations)}\n'
else:
world_description += f'From <{player_location.name}> you can access: None\n'
if blocked_passages:
world_description += f'From <{player_location.name}> there are blocked passages to: {(", ").join(blocked_passages)}\n'
else:
world_description += f'From <{player_location.name}> there are blocked passages to: None\n'
if self.player.inventory:
world_description += f'You have the following items in your inventory: {(", ").join([f"<{i.name}>" for i in self.player.inventory])}\n'
else:
world_description += f'You have the following items in your inventory: None\n'
if player_location.items:
world_description += f'If you look around, you can see the following items: {(", ").join([f"<{i.name}>" for i in player_location.items])}\n'
else:
world_description += f'If you look around, you can see the following items: None\n'
if characters_in_the_scene:
world_description += f'You can see some people: {(", ").join([f"<{c.name}>" for c in characters_in_the_scene])}'
else:
world_description += f'You can see some people: None'
details = ""
if detail_components:
items_in_the_scene = player_location.items + self.player.inventory + [blocked_values[1] for blocked_values in player_location.blocked_locations.values() if isinstance(blocked_values[1], Item)]
details += "\nHere is a description of each component.\n"
details += f"<{player_location.name}>: This is the player's location. {('. ').join(player_location.descriptions)}.\n"
details += "Characters:\n"
details += f"- <Player>: The player is acting as {self.player.name}. {('. ').join(self.player.descriptions)}.\n"
if len(characters_in_the_scene)>0:
for character in characters_in_the_scene:
details += f"- <{character.name}>: {('. ').join(character.descriptions)}."
if len(character.inventory)>0:
details += f" This character has the following items: {(', ').join([f'<{i.name}>' for i in character.inventory])}\n"
items_in_the_scene+= character.inventory
else:
details += "\n"
if len(items_in_the_scene)>0:
details+="Objects:\n"
for item in items_in_the_scene:
details += f"- <{item.name}>: {('. ').join(item.descriptions)}\n"
return world_description + '\n' + details
def parse_updates (self, updates: str) -> None:
"""Does the changes in the world according to the output of the language model.
The possible changes considered are:
- an object was moved
- a location is now reachable
- the position of the player changed.
"""
self.parse_moved_objects(updates)
self.parse_blocked_passages(updates)
self.parse_location_change(updates)
def parse_moved_objects (self, updates: str) -> None:
"""Parse the output of the language model to update the position of objects.
There are three cases:
- the player has a new item
- the player gave an item to other character
- the player dropped an item.
"""
parsed_objects = re.findall(r".*Moved object:\s*(.+)",updates)
if 'None' not in parsed_objects:
parsed_objects_split = re.findall(r"<[^<>]*?>.*?<[^<>]*?>",parsed_objects[0])
for parsed_object in parsed_objects_split:
pair = re.findall(r"<([^<>]*?)>.*?<([^<>]*?)>",parsed_object)
try:
world_item = self.items[pair[0][0]]
if pair[0][1] == 'Inventory': #(save_item case)
item_location = [character for character in list(self.characters.values()) if world_item in character.inventory]
item_location += [location for location in list(self.locations.values()) if world_item in location.items]
self.player.save_item(world_item, item_location[0])
elif pair[0][1] in self.characters: #(give_item case)
self.player.give_item(self.characters[pair[0][1]], world_item)
else: #(drop_item case)
self.player.drop_item(world_item)
except Exception as e:
print(e)
def parse_blocked_passages (self, updates: str) -> None:
"""Parse the output of the language model to update the reachable locations."""
parsed_blocked_passages = re.findall(r".*Blocked passages now available:\s*(.+)",updates)
if 'None' not in parsed_blocked_passages:
parsed_blocked_passages_split = re.findall(r"<([^<>]*?)>",parsed_blocked_passages[0])
for parsed_passage in parsed_blocked_passages_split:
try:
self.locations[self.player.location.name].unblock_passage(self.locations[parsed_passage])
except Exception as e:
print (e)
def parse_location_change (self, updates: str) -> None:
"""Parse the output of the language model to update the position of the player."""
parsed_location_change = re.findall(r".*Your location changed: (.+)",updates)
if "None" not in parsed_location_change:
parsed_location_change_split = re.findall(r"<([^<>]*?)>",parsed_location_change[0])
try:
self.player.move(self.locations[parsed_location_change_split[0]])
except Exception as e:
print(e)