Mercurial > hg > Members > aka > jupyter_CbC_kernel
annotate jupyter_c_kernel/kernel.py @ 35:e05900c7967e
Remove logging
author | Brendan Rius <brendan@omixy.com> |
---|---|
date | Sat, 30 Apr 2016 22:47:49 +0100 |
parents | b5bc8483ac3e |
children | 8acbab5a9f21 |
rev | line source |
---|---|
34 | 1 from queue import Queue, Empty |
33 | 2 from threading import Thread |
3 | |
0 | 4 from ipykernel.kernelbase import Kernel |
5 import subprocess | |
6 import tempfile | |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
7 import os |
34 | 8 |
9 | |
10 class JupyterSubprocess(subprocess.Popen): | |
11 def __init__(self, cmd, write_to_stdout, write_to_stderr): | |
12 self._write_to_stdout = write_to_stdout | |
13 self._write_to_stderr = write_to_stderr | |
14 | |
15 super().__init__(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE) | |
16 | |
17 self._stdout_queue = Queue() | |
18 self._stdout_thread = Thread(target=JupyterSubprocess._enqueue_output, args=(self.stdout, self._stdout_queue)) | |
19 self._stdout_thread.daemon = True | |
20 self._stdout_thread.start() | |
21 | |
22 self._stderr_queue = Queue() | |
23 self._stderr_thread = Thread(target=JupyterSubprocess._enqueue_output, args=(self.stderr, self._stderr_queue)) | |
24 self._stderr_thread.daemon = True | |
25 self._stderr_thread.start() | |
26 | |
27 @staticmethod | |
28 def _enqueue_output(contents, queue): | |
29 for line in iter(contents.readline, b''): | |
30 queue.put(line) | |
31 contents.close() | |
32 | |
35 | 33 def write_contents(self): |
34 | 34 try: |
35 stdout_contents = self._stdout_queue.get_nowait() | |
36 except Empty: | |
35 | 37 pass |
34 | 38 else: |
39 self._write_to_stdout(stdout_contents) | |
40 try: | |
41 stderr_contents = self._stderr_queue.get_nowait() | |
42 except Empty: | |
35 | 43 pass |
34 | 44 else: |
45 self._write_to_stderr(stderr_contents) | |
0 | 46 |
47 | |
48 class CKernel(Kernel): | |
20 | 49 implementation = 'jupyter_c_kernel' |
0 | 50 implementation_version = '1.0' |
51 language = 'c' | |
52 language_version = 'C11' | |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
53 language_info = {'name': 'c', |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
54 'mimetype': 'text/plain', |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
55 'file_extension': 'c'} |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
56 banner = "C kernel.\n" \ |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
57 "Uses gcc, compiles in C11, and creates source code files and executables in temporary folder.\n" |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
58 |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
59 def __init__(self, *args, **kwargs): |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
60 super(CKernel, self).__init__(*args, **kwargs) |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
61 self.files = [] |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
62 |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
63 def cleanup_files(self): |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
64 """Remove all the temporary files created by the kernel""" |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
65 for file in self.files: |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
66 os.remove(file) |
0 | 67 |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
68 def new_temp_file(self, **kwargs): |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
69 """Create a new temp file to be deleted when the kernel shuts down""" |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
70 # We don't want the file to be deleted when closed, but only when the kernel stops |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
71 kwargs['delete'] = False |
9 | 72 kwargs['mode'] = 'w' |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
73 file = tempfile.NamedTemporaryFile(**kwargs) |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
74 self.files.append(file.name) |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
75 return file |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
76 |
34 | 77 def _write_to_stdout(self, contents): |
78 self.send_response(self.iopub_socket, 'stream', {'name': 'stdout', 'text': contents}) | |
33 | 79 |
34 | 80 def _write_to_stderr(self, contents): |
81 self.send_response(self.iopub_socket, 'stream', {'name': 'stderr', 'text': contents}) | |
33 | 82 |
34 | 83 def create_jupyter_subprocess(self, cmd): |
84 return JupyterSubprocess(cmd, | |
85 lambda contents: self._write_to_stdout(contents.decode()), | |
86 lambda contents: self._write_to_stderr(contents.decode())) | |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
87 |
34 | 88 def compile_with_gcc(self, source_filename, binary_filename): |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
89 args = ['gcc', source_filename, '-std=c11', '-o', binary_filename] |
34 | 90 return self.create_jupyter_subprocess(args) |
0 | 91 |
92 def do_execute(self, code, silent, store_history=True, | |
93 user_expressions=None, allow_stdin=False): | |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
94 with self.new_temp_file(suffix='.c') as source_file: |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
95 source_file.write(code) |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
96 source_file.flush() |
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
97 with self.new_temp_file(suffix='.out') as binary_file: |
34 | 98 p = self.compile_with_gcc(source_file.name, binary_file.name) |
33 | 99 while p.poll() is None: |
35 | 100 p.write_contents() |
101 p.write_contents() | |
33 | 102 if p.returncode != 0: # Compilation failed |
34 | 103 self._write_to_stderr( |
104 "[C kernel] GCC exited with code {}, the executable will not be executed".format( | |
105 p.returncode)) | |
33 | 106 return {'status': 'ok', 'execution_count': self.execution_count, 'payload': [], |
107 'user_expressions': {}} | |
0 | 108 |
34 | 109 p = self.create_jupyter_subprocess([binary_file.name]) |
33 | 110 while p.poll() is None: |
35 | 111 p.write_contents() |
112 p.write_contents() | |
33 | 113 |
114 if p.returncode != 0: | |
34 | 115 self._write_to_stderr("[C kernel] Executable exited with code {}".format(p.returncode)) |
2
b46b2e5b6c08
Improve error management by separating stdout and stderr
Brendan Rius <brendan@omixy.com>
parents:
0
diff
changeset
|
116 return {'status': 'ok', 'execution_count': self.execution_count, 'payload': [], 'user_expressions': {}} |
7
aa54c85303b6
Remove temporary files when shutting down the kernel
Brendan Rius <brendan@omixy.com>
parents:
4
diff
changeset
|
117 |
aa54c85303b6
Remove temporary files when shutting down the kernel
Brendan Rius <brendan@omixy.com>
parents:
4
diff
changeset
|
118 def do_shutdown(self, restart): |
aa54c85303b6
Remove temporary files when shutting down the kernel
Brendan Rius <brendan@omixy.com>
parents:
4
diff
changeset
|
119 """Cleanup the created source code files and executables when shutting down the kernel""" |
aa54c85303b6
Remove temporary files when shutting down the kernel
Brendan Rius <brendan@omixy.com>
parents:
4
diff
changeset
|
120 self.cleanup_files() |