##// END OF EJS Templates
add missing 'control' socket to ConsoleApp.load_connection_file
MinRK -
Show More
@@ -1,395 +1,395 b''
1 """ A minimal application base mixin for all ZMQ based IPython frontends.
1 """ A minimal application base mixin for all ZMQ based IPython frontends.
2
2
3 This is not a complete console app, as subprocess will not be able to receive
3 This is not a complete console app, as subprocess will not be able to receive
4 input, there is no real readline support, among other limitations. This is a
4 input, there is no real readline support, among other limitations. This is a
5 refactoring of what used to be the IPython/qt/console/qtconsoleapp.py
5 refactoring of what used to be the IPython/qt/console/qtconsoleapp.py
6
6
7 Authors:
7 Authors:
8
8
9 * Evan Patterson
9 * Evan Patterson
10 * Min RK
10 * Min RK
11 * Erik Tollerud
11 * Erik Tollerud
12 * Fernando Perez
12 * Fernando Perez
13 * Bussonnier Matthias
13 * Bussonnier Matthias
14 * Thomas Kluyver
14 * Thomas Kluyver
15 * Paul Ivanov
15 * Paul Ivanov
16
16
17 """
17 """
18
18
19 #-----------------------------------------------------------------------------
19 #-----------------------------------------------------------------------------
20 # Imports
20 # Imports
21 #-----------------------------------------------------------------------------
21 #-----------------------------------------------------------------------------
22
22
23 # stdlib imports
23 # stdlib imports
24 import atexit
24 import atexit
25 import json
25 import json
26 import os
26 import os
27 import shutil
27 import shutil
28 import signal
28 import signal
29 import sys
29 import sys
30 import uuid
30 import uuid
31
31
32
32
33 # Local imports
33 # Local imports
34 from IPython.config.application import boolean_flag
34 from IPython.config.application import boolean_flag
35 from IPython.config.configurable import Configurable
35 from IPython.config.configurable import Configurable
36 from IPython.core.profiledir import ProfileDir
36 from IPython.core.profiledir import ProfileDir
37 from IPython.kernel.blocking import BlockingKernelClient
37 from IPython.kernel.blocking import BlockingKernelClient
38 from IPython.kernel import KernelManager
38 from IPython.kernel import KernelManager
39 from IPython.kernel import tunnel_to_kernel, find_connection_file, swallow_argv
39 from IPython.kernel import tunnel_to_kernel, find_connection_file, swallow_argv
40 from IPython.utils.path import filefind
40 from IPython.utils.path import filefind
41 from IPython.utils.py3compat import str_to_bytes
41 from IPython.utils.py3compat import str_to_bytes
42 from IPython.utils.traitlets import (
42 from IPython.utils.traitlets import (
43 Dict, List, Unicode, CUnicode, Int, CBool, Any, CaselessStrEnum
43 Dict, List, Unicode, CUnicode, Int, CBool, Any, CaselessStrEnum
44 )
44 )
45 from IPython.kernel.zmq.kernelapp import (
45 from IPython.kernel.zmq.kernelapp import (
46 kernel_flags,
46 kernel_flags,
47 kernel_aliases,
47 kernel_aliases,
48 IPKernelApp
48 IPKernelApp
49 )
49 )
50 from IPython.kernel.zmq.session import Session, default_secure
50 from IPython.kernel.zmq.session import Session, default_secure
51 from IPython.kernel.zmq.zmqshell import ZMQInteractiveShell
51 from IPython.kernel.zmq.zmqshell import ZMQInteractiveShell
52 from IPython.kernel.connect import ConnectionFileMixin
52 from IPython.kernel.connect import ConnectionFileMixin
53
53
54 #-----------------------------------------------------------------------------
54 #-----------------------------------------------------------------------------
55 # Network Constants
55 # Network Constants
56 #-----------------------------------------------------------------------------
56 #-----------------------------------------------------------------------------
57
57
58 from IPython.utils.localinterfaces import LOCALHOST, LOCAL_IPS
58 from IPython.utils.localinterfaces import LOCALHOST, LOCAL_IPS
59
59
60 #-----------------------------------------------------------------------------
60 #-----------------------------------------------------------------------------
61 # Globals
61 # Globals
62 #-----------------------------------------------------------------------------
62 #-----------------------------------------------------------------------------
63
63
64
64
65 #-----------------------------------------------------------------------------
65 #-----------------------------------------------------------------------------
66 # Aliases and Flags
66 # Aliases and Flags
67 #-----------------------------------------------------------------------------
67 #-----------------------------------------------------------------------------
68
68
69 flags = dict(kernel_flags)
69 flags = dict(kernel_flags)
70
70
71 # the flags that are specific to the frontend
71 # the flags that are specific to the frontend
72 # these must be scrubbed before being passed to the kernel,
72 # these must be scrubbed before being passed to the kernel,
73 # or it will raise an error on unrecognized flags
73 # or it will raise an error on unrecognized flags
74 app_flags = {
74 app_flags = {
75 'existing' : ({'IPythonConsoleApp' : {'existing' : 'kernel*.json'}},
75 'existing' : ({'IPythonConsoleApp' : {'existing' : 'kernel*.json'}},
76 "Connect to an existing kernel. If no argument specified, guess most recent"),
76 "Connect to an existing kernel. If no argument specified, guess most recent"),
77 }
77 }
78 app_flags.update(boolean_flag(
78 app_flags.update(boolean_flag(
79 'confirm-exit', 'IPythonConsoleApp.confirm_exit',
79 'confirm-exit', 'IPythonConsoleApp.confirm_exit',
80 """Set to display confirmation dialog on exit. You can always use 'exit' or 'quit',
80 """Set to display confirmation dialog on exit. You can always use 'exit' or 'quit',
81 to force a direct exit without any confirmation.
81 to force a direct exit without any confirmation.
82 """,
82 """,
83 """Don't prompt the user when exiting. This will terminate the kernel
83 """Don't prompt the user when exiting. This will terminate the kernel
84 if it is owned by the frontend, and leave it alive if it is external.
84 if it is owned by the frontend, and leave it alive if it is external.
85 """
85 """
86 ))
86 ))
87 flags.update(app_flags)
87 flags.update(app_flags)
88
88
89 aliases = dict(kernel_aliases)
89 aliases = dict(kernel_aliases)
90
90
91 # also scrub aliases from the frontend
91 # also scrub aliases from the frontend
92 app_aliases = dict(
92 app_aliases = dict(
93 ip = 'IPythonConsoleApp.ip',
93 ip = 'IPythonConsoleApp.ip',
94 transport = 'IPythonConsoleApp.transport',
94 transport = 'IPythonConsoleApp.transport',
95 hb = 'IPythonConsoleApp.hb_port',
95 hb = 'IPythonConsoleApp.hb_port',
96 shell = 'IPythonConsoleApp.shell_port',
96 shell = 'IPythonConsoleApp.shell_port',
97 iopub = 'IPythonConsoleApp.iopub_port',
97 iopub = 'IPythonConsoleApp.iopub_port',
98 stdin = 'IPythonConsoleApp.stdin_port',
98 stdin = 'IPythonConsoleApp.stdin_port',
99 existing = 'IPythonConsoleApp.existing',
99 existing = 'IPythonConsoleApp.existing',
100 f = 'IPythonConsoleApp.connection_file',
100 f = 'IPythonConsoleApp.connection_file',
101
101
102
102
103 ssh = 'IPythonConsoleApp.sshserver',
103 ssh = 'IPythonConsoleApp.sshserver',
104 )
104 )
105 aliases.update(app_aliases)
105 aliases.update(app_aliases)
106
106
107 #-----------------------------------------------------------------------------
107 #-----------------------------------------------------------------------------
108 # Classes
108 # Classes
109 #-----------------------------------------------------------------------------
109 #-----------------------------------------------------------------------------
110
110
111 #-----------------------------------------------------------------------------
111 #-----------------------------------------------------------------------------
112 # IPythonConsole
112 # IPythonConsole
113 #-----------------------------------------------------------------------------
113 #-----------------------------------------------------------------------------
114
114
115 classes = [IPKernelApp, ZMQInteractiveShell, KernelManager, ProfileDir, Session]
115 classes = [IPKernelApp, ZMQInteractiveShell, KernelManager, ProfileDir, Session]
116
116
117 try:
117 try:
118 from IPython.kernel.zmq.pylab.backend_inline import InlineBackend
118 from IPython.kernel.zmq.pylab.backend_inline import InlineBackend
119 except ImportError:
119 except ImportError:
120 pass
120 pass
121 else:
121 else:
122 classes.append(InlineBackend)
122 classes.append(InlineBackend)
123
123
124 class IPythonConsoleApp(ConnectionFileMixin):
124 class IPythonConsoleApp(ConnectionFileMixin):
125 name = 'ipython-console-mixin'
125 name = 'ipython-console-mixin'
126
126
127 description = """
127 description = """
128 The IPython Mixin Console.
128 The IPython Mixin Console.
129
129
130 This class contains the common portions of console client (QtConsole,
130 This class contains the common portions of console client (QtConsole,
131 ZMQ-based terminal console, etc). It is not a full console, in that
131 ZMQ-based terminal console, etc). It is not a full console, in that
132 launched terminal subprocesses will not be able to accept input.
132 launched terminal subprocesses will not be able to accept input.
133
133
134 The Console using this mixing supports various extra features beyond
134 The Console using this mixing supports various extra features beyond
135 the single-process Terminal IPython shell, such as connecting to
135 the single-process Terminal IPython shell, such as connecting to
136 existing kernel, via:
136 existing kernel, via:
137
137
138 ipython <appname> --existing
138 ipython <appname> --existing
139
139
140 as well as tunnel via SSH
140 as well as tunnel via SSH
141
141
142 """
142 """
143
143
144 classes = classes
144 classes = classes
145 flags = Dict(flags)
145 flags = Dict(flags)
146 aliases = Dict(aliases)
146 aliases = Dict(aliases)
147 kernel_manager_class = KernelManager
147 kernel_manager_class = KernelManager
148 kernel_client_class = BlockingKernelClient
148 kernel_client_class = BlockingKernelClient
149
149
150 kernel_argv = List(Unicode)
150 kernel_argv = List(Unicode)
151 # frontend flags&aliases to be stripped when building kernel_argv
151 # frontend flags&aliases to be stripped when building kernel_argv
152 frontend_flags = Any(app_flags)
152 frontend_flags = Any(app_flags)
153 frontend_aliases = Any(app_aliases)
153 frontend_aliases = Any(app_aliases)
154
154
155 # create requested profiles by default, if they don't exist:
155 # create requested profiles by default, if they don't exist:
156 auto_create = CBool(True)
156 auto_create = CBool(True)
157 # connection info:
157 # connection info:
158
158
159 sshserver = Unicode('', config=True,
159 sshserver = Unicode('', config=True,
160 help="""The SSH server to use to connect to the kernel.""")
160 help="""The SSH server to use to connect to the kernel.""")
161 sshkey = Unicode('', config=True,
161 sshkey = Unicode('', config=True,
162 help="""Path to the ssh key to use for logging in to the ssh server.""")
162 help="""Path to the ssh key to use for logging in to the ssh server.""")
163
163
164 hb_port = Int(0, config=True,
164 hb_port = Int(0, config=True,
165 help="set the heartbeat port [default: random]")
165 help="set the heartbeat port [default: random]")
166 shell_port = Int(0, config=True,
166 shell_port = Int(0, config=True,
167 help="set the shell (ROUTER) port [default: random]")
167 help="set the shell (ROUTER) port [default: random]")
168 iopub_port = Int(0, config=True,
168 iopub_port = Int(0, config=True,
169 help="set the iopub (PUB) port [default: random]")
169 help="set the iopub (PUB) port [default: random]")
170 stdin_port = Int(0, config=True,
170 stdin_port = Int(0, config=True,
171 help="set the stdin (DEALER) port [default: random]")
171 help="set the stdin (DEALER) port [default: random]")
172 connection_file = Unicode('', config=True,
172 connection_file = Unicode('', config=True,
173 help="""JSON file in which to store connection info [default: kernel-<pid>.json]
173 help="""JSON file in which to store connection info [default: kernel-<pid>.json]
174
174
175 This file will contain the IP, ports, and authentication key needed to connect
175 This file will contain the IP, ports, and authentication key needed to connect
176 clients to this kernel. By default, this file will be created in the security-dir
176 clients to this kernel. By default, this file will be created in the security-dir
177 of the current profile, but can be specified by absolute path.
177 of the current profile, but can be specified by absolute path.
178 """)
178 """)
179 def _connection_file_default(self):
179 def _connection_file_default(self):
180 return 'kernel-%i.json' % os.getpid()
180 return 'kernel-%i.json' % os.getpid()
181
181
182 existing = CUnicode('', config=True,
182 existing = CUnicode('', config=True,
183 help="""Connect to an already running kernel""")
183 help="""Connect to an already running kernel""")
184
184
185 confirm_exit = CBool(True, config=True,
185 confirm_exit = CBool(True, config=True,
186 help="""
186 help="""
187 Set to display confirmation dialog on exit. You can always use 'exit' or 'quit',
187 Set to display confirmation dialog on exit. You can always use 'exit' or 'quit',
188 to force a direct exit without any confirmation.""",
188 to force a direct exit without any confirmation.""",
189 )
189 )
190
190
191
191
192 def build_kernel_argv(self, argv=None):
192 def build_kernel_argv(self, argv=None):
193 """build argv to be passed to kernel subprocess"""
193 """build argv to be passed to kernel subprocess"""
194 if argv is None:
194 if argv is None:
195 argv = sys.argv[1:]
195 argv = sys.argv[1:]
196 self.kernel_argv = swallow_argv(argv, self.frontend_aliases, self.frontend_flags)
196 self.kernel_argv = swallow_argv(argv, self.frontend_aliases, self.frontend_flags)
197 # kernel should inherit default config file from frontend
197 # kernel should inherit default config file from frontend
198 self.kernel_argv.append("--IPKernelApp.parent_appname='%s'" % self.name)
198 self.kernel_argv.append("--IPKernelApp.parent_appname='%s'" % self.name)
199
199
200 def init_connection_file(self):
200 def init_connection_file(self):
201 """find the connection file, and load the info if found.
201 """find the connection file, and load the info if found.
202
202
203 The current working directory and the current profile's security
203 The current working directory and the current profile's security
204 directory will be searched for the file if it is not given by
204 directory will be searched for the file if it is not given by
205 absolute path.
205 absolute path.
206
206
207 When attempting to connect to an existing kernel and the `--existing`
207 When attempting to connect to an existing kernel and the `--existing`
208 argument does not match an existing file, it will be interpreted as a
208 argument does not match an existing file, it will be interpreted as a
209 fileglob, and the matching file in the current profile's security dir
209 fileglob, and the matching file in the current profile's security dir
210 with the latest access time will be used.
210 with the latest access time will be used.
211
211
212 After this method is called, self.connection_file contains the *full path*
212 After this method is called, self.connection_file contains the *full path*
213 to the connection file, never just its name.
213 to the connection file, never just its name.
214 """
214 """
215 if self.existing:
215 if self.existing:
216 try:
216 try:
217 cf = find_connection_file(self.existing)
217 cf = find_connection_file(self.existing)
218 except Exception:
218 except Exception:
219 self.log.critical("Could not find existing kernel connection file %s", self.existing)
219 self.log.critical("Could not find existing kernel connection file %s", self.existing)
220 self.exit(1)
220 self.exit(1)
221 self.log.info("Connecting to existing kernel: %s" % cf)
221 self.log.info("Connecting to existing kernel: %s" % cf)
222 self.connection_file = cf
222 self.connection_file = cf
223 else:
223 else:
224 # not existing, check if we are going to write the file
224 # not existing, check if we are going to write the file
225 # and ensure that self.connection_file is a full path, not just the shortname
225 # and ensure that self.connection_file is a full path, not just the shortname
226 try:
226 try:
227 cf = find_connection_file(self.connection_file)
227 cf = find_connection_file(self.connection_file)
228 except Exception:
228 except Exception:
229 # file might not exist
229 # file might not exist
230 if self.connection_file == os.path.basename(self.connection_file):
230 if self.connection_file == os.path.basename(self.connection_file):
231 # just shortname, put it in security dir
231 # just shortname, put it in security dir
232 cf = os.path.join(self.profile_dir.security_dir, self.connection_file)
232 cf = os.path.join(self.profile_dir.security_dir, self.connection_file)
233 else:
233 else:
234 cf = self.connection_file
234 cf = self.connection_file
235 self.connection_file = cf
235 self.connection_file = cf
236
236
237 # should load_connection_file only be used for existing?
237 # should load_connection_file only be used for existing?
238 # as it is now, this allows reusing ports if an existing
238 # as it is now, this allows reusing ports if an existing
239 # file is requested
239 # file is requested
240 try:
240 try:
241 self.load_connection_file()
241 self.load_connection_file()
242 except Exception:
242 except Exception:
243 self.log.error("Failed to load connection file: %r", self.connection_file, exc_info=True)
243 self.log.error("Failed to load connection file: %r", self.connection_file, exc_info=True)
244 self.exit(1)
244 self.exit(1)
245
245
246 def load_connection_file(self):
246 def load_connection_file(self):
247 """load ip/port/hmac config from JSON connection file"""
247 """load ip/port/hmac config from JSON connection file"""
248 # this is identical to IPKernelApp.load_connection_file
248 # this is identical to IPKernelApp.load_connection_file
249 # perhaps it can be centralized somewhere?
249 # perhaps it can be centralized somewhere?
250 try:
250 try:
251 fname = filefind(self.connection_file, ['.', self.profile_dir.security_dir])
251 fname = filefind(self.connection_file, ['.', self.profile_dir.security_dir])
252 except IOError:
252 except IOError:
253 self.log.debug("Connection File not found: %s", self.connection_file)
253 self.log.debug("Connection File not found: %s", self.connection_file)
254 return
254 return
255 self.log.debug(u"Loading connection file %s", fname)
255 self.log.debug(u"Loading connection file %s", fname)
256 with open(fname) as f:
256 with open(fname) as f:
257 cfg = json.load(f)
257 cfg = json.load(f)
258 self.transport = cfg.get('transport', 'tcp')
258 self.transport = cfg.get('transport', 'tcp')
259 self.ip = cfg.get('ip', LOCALHOST)
259 self.ip = cfg.get('ip', LOCALHOST)
260
260
261 for channel in ('hb', 'shell', 'iopub', 'stdin'):
261 for channel in ('hb', 'shell', 'iopub', 'stdin', 'control'):
262 name = channel + '_port'
262 name = channel + '_port'
263 if getattr(self, name) == 0 and name in cfg:
263 if getattr(self, name) == 0 and name in cfg:
264 # not overridden by config or cl_args
264 # not overridden by config or cl_args
265 setattr(self, name, cfg[name])
265 setattr(self, name, cfg[name])
266 if 'key' in cfg:
266 if 'key' in cfg:
267 self.config.Session.key = str_to_bytes(cfg['key'])
267 self.config.Session.key = str_to_bytes(cfg['key'])
268 if 'signature_scheme' in cfg:
268 if 'signature_scheme' in cfg:
269 self.config.Session.signature_scheme = cfg['signature_scheme']
269 self.config.Session.signature_scheme = cfg['signature_scheme']
270
270
271 def init_ssh(self):
271 def init_ssh(self):
272 """set up ssh tunnels, if needed."""
272 """set up ssh tunnels, if needed."""
273 if not self.existing or (not self.sshserver and not self.sshkey):
273 if not self.existing or (not self.sshserver and not self.sshkey):
274 return
274 return
275 self.load_connection_file()
275 self.load_connection_file()
276
276
277 transport = self.transport
277 transport = self.transport
278 ip = self.ip
278 ip = self.ip
279
279
280 if transport != 'tcp':
280 if transport != 'tcp':
281 self.log.error("Can only use ssh tunnels with TCP sockets, not %s", transport)
281 self.log.error("Can only use ssh tunnels with TCP sockets, not %s", transport)
282 sys.exit(-1)
282 sys.exit(-1)
283
283
284 if self.sshkey and not self.sshserver:
284 if self.sshkey and not self.sshserver:
285 # specifying just the key implies that we are connecting directly
285 # specifying just the key implies that we are connecting directly
286 self.sshserver = ip
286 self.sshserver = ip
287 ip = LOCALHOST
287 ip = LOCALHOST
288
288
289 # build connection dict for tunnels:
289 # build connection dict for tunnels:
290 info = dict(ip=ip,
290 info = dict(ip=ip,
291 shell_port=self.shell_port,
291 shell_port=self.shell_port,
292 iopub_port=self.iopub_port,
292 iopub_port=self.iopub_port,
293 stdin_port=self.stdin_port,
293 stdin_port=self.stdin_port,
294 hb_port=self.hb_port
294 hb_port=self.hb_port
295 )
295 )
296
296
297 self.log.info("Forwarding connections to %s via %s"%(ip, self.sshserver))
297 self.log.info("Forwarding connections to %s via %s"%(ip, self.sshserver))
298
298
299 # tunnels return a new set of ports, which will be on localhost:
299 # tunnels return a new set of ports, which will be on localhost:
300 self.ip = LOCALHOST
300 self.ip = LOCALHOST
301 try:
301 try:
302 newports = tunnel_to_kernel(info, self.sshserver, self.sshkey)
302 newports = tunnel_to_kernel(info, self.sshserver, self.sshkey)
303 except:
303 except:
304 # even catch KeyboardInterrupt
304 # even catch KeyboardInterrupt
305 self.log.error("Could not setup tunnels", exc_info=True)
305 self.log.error("Could not setup tunnels", exc_info=True)
306 self.exit(1)
306 self.exit(1)
307
307
308 self.shell_port, self.iopub_port, self.stdin_port, self.hb_port = newports
308 self.shell_port, self.iopub_port, self.stdin_port, self.hb_port = newports
309
309
310 cf = self.connection_file
310 cf = self.connection_file
311 base,ext = os.path.splitext(cf)
311 base,ext = os.path.splitext(cf)
312 base = os.path.basename(base)
312 base = os.path.basename(base)
313 self.connection_file = os.path.basename(base)+'-ssh'+ext
313 self.connection_file = os.path.basename(base)+'-ssh'+ext
314 self.log.critical("To connect another client via this tunnel, use:")
314 self.log.critical("To connect another client via this tunnel, use:")
315 self.log.critical("--existing %s" % self.connection_file)
315 self.log.critical("--existing %s" % self.connection_file)
316
316
317 def _new_connection_file(self):
317 def _new_connection_file(self):
318 cf = ''
318 cf = ''
319 while not cf:
319 while not cf:
320 # we don't need a 128b id to distinguish kernels, use more readable
320 # we don't need a 128b id to distinguish kernels, use more readable
321 # 48b node segment (12 hex chars). Users running more than 32k simultaneous
321 # 48b node segment (12 hex chars). Users running more than 32k simultaneous
322 # kernels can subclass.
322 # kernels can subclass.
323 ident = str(uuid.uuid4()).split('-')[-1]
323 ident = str(uuid.uuid4()).split('-')[-1]
324 cf = os.path.join(self.profile_dir.security_dir, 'kernel-%s.json' % ident)
324 cf = os.path.join(self.profile_dir.security_dir, 'kernel-%s.json' % ident)
325 # only keep if it's actually new. Protect against unlikely collision
325 # only keep if it's actually new. Protect against unlikely collision
326 # in 48b random search space
326 # in 48b random search space
327 cf = cf if not os.path.exists(cf) else ''
327 cf = cf if not os.path.exists(cf) else ''
328 return cf
328 return cf
329
329
330 def init_kernel_manager(self):
330 def init_kernel_manager(self):
331 # Don't let Qt or ZMQ swallow KeyboardInterupts.
331 # Don't let Qt or ZMQ swallow KeyboardInterupts.
332 if self.existing:
332 if self.existing:
333 self.kernel_manager = None
333 self.kernel_manager = None
334 return
334 return
335 signal.signal(signal.SIGINT, signal.SIG_DFL)
335 signal.signal(signal.SIGINT, signal.SIG_DFL)
336
336
337 # Create a KernelManager and start a kernel.
337 # Create a KernelManager and start a kernel.
338 self.kernel_manager = self.kernel_manager_class(
338 self.kernel_manager = self.kernel_manager_class(
339 ip=self.ip,
339 ip=self.ip,
340 transport=self.transport,
340 transport=self.transport,
341 shell_port=self.shell_port,
341 shell_port=self.shell_port,
342 iopub_port=self.iopub_port,
342 iopub_port=self.iopub_port,
343 stdin_port=self.stdin_port,
343 stdin_port=self.stdin_port,
344 hb_port=self.hb_port,
344 hb_port=self.hb_port,
345 connection_file=self.connection_file,
345 connection_file=self.connection_file,
346 parent=self,
346 parent=self,
347 )
347 )
348 self.kernel_manager.client_factory = self.kernel_client_class
348 self.kernel_manager.client_factory = self.kernel_client_class
349 self.kernel_manager.start_kernel(extra_arguments=self.kernel_argv)
349 self.kernel_manager.start_kernel(extra_arguments=self.kernel_argv)
350 atexit.register(self.kernel_manager.cleanup_ipc_files)
350 atexit.register(self.kernel_manager.cleanup_ipc_files)
351
351
352 if self.sshserver:
352 if self.sshserver:
353 # ssh, write new connection file
353 # ssh, write new connection file
354 self.kernel_manager.write_connection_file()
354 self.kernel_manager.write_connection_file()
355
355
356 # in case KM defaults / ssh writing changes things:
356 # in case KM defaults / ssh writing changes things:
357 km = self.kernel_manager
357 km = self.kernel_manager
358 self.shell_port=km.shell_port
358 self.shell_port=km.shell_port
359 self.iopub_port=km.iopub_port
359 self.iopub_port=km.iopub_port
360 self.stdin_port=km.stdin_port
360 self.stdin_port=km.stdin_port
361 self.hb_port=km.hb_port
361 self.hb_port=km.hb_port
362 self.connection_file = km.connection_file
362 self.connection_file = km.connection_file
363
363
364 atexit.register(self.kernel_manager.cleanup_connection_file)
364 atexit.register(self.kernel_manager.cleanup_connection_file)
365
365
366 def init_kernel_client(self):
366 def init_kernel_client(self):
367 if self.kernel_manager is not None:
367 if self.kernel_manager is not None:
368 self.kernel_client = self.kernel_manager.client()
368 self.kernel_client = self.kernel_manager.client()
369 else:
369 else:
370 self.kernel_client = self.kernel_client_class(
370 self.kernel_client = self.kernel_client_class(
371 ip=self.ip,
371 ip=self.ip,
372 transport=self.transport,
372 transport=self.transport,
373 shell_port=self.shell_port,
373 shell_port=self.shell_port,
374 iopub_port=self.iopub_port,
374 iopub_port=self.iopub_port,
375 stdin_port=self.stdin_port,
375 stdin_port=self.stdin_port,
376 hb_port=self.hb_port,
376 hb_port=self.hb_port,
377 connection_file=self.connection_file,
377 connection_file=self.connection_file,
378 parent=self,
378 parent=self,
379 )
379 )
380
380
381 self.kernel_client.start_channels()
381 self.kernel_client.start_channels()
382
382
383
383
384
384
385 def initialize(self, argv=None):
385 def initialize(self, argv=None):
386 """
386 """
387 Classes which mix this class in should call:
387 Classes which mix this class in should call:
388 IPythonConsoleApp.initialize(self,argv)
388 IPythonConsoleApp.initialize(self,argv)
389 """
389 """
390 self.init_connection_file()
390 self.init_connection_file()
391 default_secure(self.config)
391 default_secure(self.config)
392 self.init_ssh()
392 self.init_ssh()
393 self.init_kernel_manager()
393 self.init_kernel_manager()
394 self.init_kernel_client()
394 self.init_kernel_client()
395
395
General Comments 0
You need to be logged in to leave comments. Login now