@@ -129,7 +129,12 @@ def create_stream(self):
129
129
stream .channel = channel
130
130
131
131
def nudge (self ):
132
- shell_channel = self .channels ['shell' ]
132
+ # Use a transient shell channel to prevent leaking
133
+ # shell responses to the front-end.
134
+ kernel = self .kernel_manager .get_kernel (self .kernel_id )
135
+ shell_channel = kernel .connect_shell ()
136
+
137
+ # The IOPub used by the client.
133
138
iopub_channel = self .channels ['iopub' ]
134
139
135
140
future = Future ()
@@ -141,18 +146,20 @@ def finish():
141
146
loop .remove_timeout (timeout )
142
147
loop .remove_timeout (nudge_handle )
143
148
iopub_channel .stop_on_recv ()
144
- shell_channel .stop_on_recv ()
149
+ if not shell_channel .closed ():
150
+ shell_channel .close ()
145
151
146
152
def on_shell_reply (msg ):
147
153
if not info_future .done ():
148
154
self .log .debug ("Nudge: shell info reply received: %s" , self .kernel_id )
149
- shell_channel .stop_on_recv ()
155
+ if not shell_channel .closed ():
156
+ shell_channel .close ()
150
157
self .log .debug ("Nudge: resolving shell future" )
151
- info_future .set_result (msg )
158
+ info_future .set_result (None )
152
159
if iopub_future .done ():
153
160
finish ()
154
161
self .log .debug ("Nudge: resolving main future in shell handler" )
155
- future .set_result (info_future . result () )
162
+ future .set_result (None )
156
163
157
164
def on_iopub (msg ):
158
165
if not iopub_future .done ():
@@ -163,7 +170,7 @@ def on_iopub(msg):
163
170
if info_future .done ():
164
171
finish ()
165
172
self .log .debug ("Nudge: resolving main future in iopub handler" )
166
- future .set_result (info_future . result () )
173
+ future .set_result (None )
167
174
168
175
def on_timeout ():
169
176
self .log .warning ("Nudge: Timeout waiting for kernel_info_reply: %s" , self .kernel_id )
@@ -177,16 +184,18 @@ def on_timeout():
177
184
178
185
# Nudge the kernel with kernel info requests until we get an IOPub message
179
186
def nudge (count ):
180
- count += 1
181
- nonlocal nudge_handle
187
+ # Do not nudge busy kernels as kernel info requests sent to shell are
188
+ # queued behind execution requests.
189
+ if kernel .execution_state == 'busy' :
190
+ future .set_result (None )
182
191
if not future .done ():
183
192
log = self .log .warning if count % 10 == 0 else self .log .debug
184
193
log ("Nudging attempt %s on kernel %s" % (count , self .kernel_id ))
185
194
self .session .send (shell_channel , "kernel_info_request" )
195
+ nonlocal nudge_handle
186
196
nudge_handle = loop .call_later (0.5 , nudge , count )
187
197
188
198
nudge_handle = loop .call_later (0 , nudge , count = 0 )
189
-
190
199
timeout = loop .add_timeout (loop .time () + self .kernel_info_timeout , on_timeout )
191
200
return future
192
201
0 commit comments