-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathbehaviour.py
164 lines (134 loc) · 6.03 KB
/
behaviour.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
"""
This is the main entry point to our Python application from PythonSkill.cpp,
where 'robot' is defined so 'import robot' does not fail with ImportError.
The `import robot` bridge from C++ to Python is defined by various Python
wrappers, run once at import time. Please see the `tick` function below for a
bridge from C++ to Python and back to C++ which is run many times at run time.
For example, for the implementation of `robot.TOP_IMAGE_ROWS`,
see the file `VisionDefinitions_wrap.cpp` in the following directory:
@see
https://github.com/UNSWComputing/rUNSWift/tree/main/robot/perception/behaviour/python/wrappers
For info, the following contains the C++ definition of `TOP_IMAGE_ROWS`:
@see
https://github.com/UNSWComputing/rUNSWift/blob/main/robot/perception/vision/VisionDefinitions.hpp
"""
import Simulation # noqa
from pkgutil import iter_modules
from traceback import format_exc
from importlib import import_module
from os import getenv
from os.path import join
from util import Log
from robot import BehaviourRequest, All
from util.world import World
from util.Global import update_global
from util.TeamStatus import update_team_status
from util.FieldGeometry import update_field_geometry
from util.Timer import update_timer
from util.Sonar import update_sonar
from util.GameStatus import update_game_status
from util.NetworkEar import update_network_ear, get_ear_colour
from util.RemoteStiffener import update_remote_stiffener, get_stiffen_command
from util.ObstacleAvoidance import update_obstacle_avoidance
from util.LedOverride import override_request, reset_led_override
from util.Constants import LEDColour
skill_instance = None
headskill_instance = None
def catch_all(tick):
def catcher(blackboard):
try:
return tick(blackboard)
except Exception:
Log.error("Behaviour exception:", exc_info=True)
raise
return catcher
def skill_instance_factory(blackboard, skill, behaviour_packages):
# Load the module and the class we're going to use.
found_skill = False
SkillClass = None
behaviour_root = getenv("BEHAVIOUR_ROOT", "/home/nao/data/behaviours")
for package in behaviour_packages:
look_in = join(behaviour_root, "/".join(package))
seen_modules = [name for _, name, _ in iter_modules([look_in])]
if skill not in seen_modules:
Log.info("%s.py was not in %s, skipping import attempt.", skill, look_in)
continue
skill_path = "%s.%s" % (".".join(package), skill)
try:
skill_module = import_module(skill_path)
# Access the class so we can do some reflection.
SkillClass = getattr(skill_module, skill)
found_skill = True
Log.info("Successfully imported %s from %s", skill, skill_path)
break
except ImportError as e:
Log.error("%s %s", package, e)
Log.error(format_exc())
if not found_skill:
raise ImportError("Skill: %s not found in behaviour packages %s" % (skill, behaviour_packages))
new_world = World(blackboard) # It's a whole new world.
_skill_instance = SkillClass(world=new_world)
return _skill_instance
@catch_all
def tick(blackboard):
"""
This is the main entry point from C++ into our Python behaviours and back.
More specifically it is the bridge from which C++ calls Python inside
the runswift executable, and receives a BehaviourRequest back.
Currently called in `robot/perception/behaviour/python/PythonSkill.cpp`,
the `PythonSkill::execute()` C++ function, and explicitly the line
`behaviour_tick(bb)`.
:param blackboard: The runswift Blackboard, a bunch of things
stored in global memory.
:return: A `robot.BehaviourRequest()` instance, defined in C++ inside
`robot/types/BehaviourRequest.hpp`.
"""
# Update all blackboard dependent helper modules.
update_timer(blackboard)
update_global(blackboard)
update_team_status(blackboard)
update_field_geometry(blackboard)
update_sonar(blackboard)
update_network_ear(blackboard)
update_remote_stiffener(blackboard)
update_game_status(blackboard)
update_obstacle_avoidance(blackboard)
reset_led_override()
global skill_instance
if not skill_instance:
behaviour_packages = [["body", "roles"], ["body", "skills"], ["body", "test"]]
skill_name = blackboard.behaviour.skill
skill_instance = skill_instance_factory(blackboard, skill_name, behaviour_packages)
global headskill_instance
if not headskill_instance:
head_packages = [["head"]]
headskill_name = blackboard.behaviour.headskill
headskill_instance = skill_instance_factory(blackboard, headskill_name, head_packages)
# On every tick of the perception thread, we update the blackboard,
# tick the skill, and then return the resulting behaviour request.
skill_instance.world.update(blackboard)
skill_instance.world.b_request = BehaviourRequest()
skill_instance.world.b_request.actions = All()
skill_instance.tick()
request = skill_instance.world.b_request
headskill_instance.world.update(blackboard)
headskill_instance.world.b_request = request
headskill_instance.tick()
request = headskill_instance.world.b_request
# LED colouring for ball detection
if len(blackboard.vision.balls) > 0:
request.actions.leds.rightEye = LEDColour.red
else:
request.actions.leds.rightEye = LEDColour.off
# Override leds as requested by skills
override_request(request)
# Get right ear colour depending on network activity
request.actions.leds.rightEar = get_ear_colour()
# Remotely stiffen
request.actions.stiffen = get_stiffen_command()
# Obtain behaviour hierarchies of current skill
request.behaviourDebugInfo.bodyBehaviourHierarchy = skill_instance.world.behaviour_hierarchy
skill_instance.world.behaviour_hierarchy = ""
request.behaviourDebugInfo.headBehaviourHierarchy = headskill_instance.world.behaviour_hierarchy
headskill_instance.world.behaviour_hierarchy = ""
return request