Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
erp5
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
alecs_myu
erp5
Commits
eb896b21
Commit
eb896b21
authored
Apr 19, 2018
by
Klaus Wölfel
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
jupyter: store parameters for notebook in notebook context
parent
6394ffbf
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
131 additions
and
62 deletions
+131
-62
bt5/erp5_data_notebook/ExtensionTemplateItem/portal_components/extension.erp5.JupyterCompile.py
...teItem/portal_components/extension.erp5.JupyterCompile.py
+122
-55
bt5/erp5_data_notebook/ExtensionTemplateItem/portal_components/extension.erp5.JupyterCompile.xml
...eItem/portal_components/extension.erp5.JupyterCompile.xml
+9
-7
No files found.
bt5/erp5_data_notebook/ExtensionTemplateItem/portal_components/extension.erp5.JupyterCompile.py
View file @
eb896b21
...
...
@@ -28,6 +28,19 @@ from ipykernel.jsonutil import json_clean, encode_images
import
threading
display_data_wrapper_lock
=
threading
.
Lock
()
# Well known unserializable types
from
Record
import
Record
well_known_unserializable_type_tuple
=
(
ModuleType
,
Record
)
# ZBigArray may not be availableK
try
:
from
wendelin.bigarray.array_zodb
import
ZBigArray
# FIXME ZBigArrays are regular ZODB objects and must be serializable
# FIXME the bug is probably in CanSerialize()
# FIXME -> see https://lab.nexedi.com/nexedi/erp5/commit/5fb16acd#note_33582 for details
well_known_unserializable_type_tuple
=
tuple
(
list
(
well_known_unserializable_type_tuple
)
+
[
ZBigArray
])
except
ImportError
:
pass
def
Base_executeJupyter
(
self
,
python_expression
=
None
,
reference
=
None
,
\
title
=
None
,
request_reference
=
False
,
**
kw
):
# Check permissions for current user and display message to non-authorized user
...
...
@@ -66,14 +79,22 @@ def Base_executeJupyter(self, python_expression=None, reference=None, \
title
=
title
,
reference
=
reference
,
batch_mode
=
True
)
# Add new Data Notebook Line to the Data Notebook
data_notebook_line
=
data_notebook
.
DataNotebook_addDataNotebookLine
(
# By default, store_history is True
store_history
=
kw
.
get
(
'store_history'
,
True
)
# Klaus
store_history
=
False
data_notebook_line
=
None
if
store_history
:
# Add new Data Notebook Line to the Data Notebook
data_notebook_line
=
data_notebook
.
DataNotebook_addDataNotebookLine
(
notebook_code
=
python_expression
,
batch_mode
=
True
)
# Gets the context associated to the data notebook being used
old_notebook_context
=
data_notebook
.
getNotebookContext
()
old_notebook_context
=
None
if
data_notebook
.
hasNotebookContext
():
old_notebook_context
=
data_notebook
.
getNotebookContext
().
copy
()
if
not
old_notebook_context
:
old_notebook_context
=
self
.
Base_createNotebookContext
()
...
...
@@ -83,6 +104,9 @@ def Base_executeJupyter(self, python_expression=None, reference=None, \
new_notebook_context
=
final_result
[
'notebook_context'
]
# Klaus
new_notebook_context
[
"variables"
]
=
old_notebook_context
[
"variables"
]
result
=
{
u'code_result'
:
final_result
[
'result_string'
],
u'print_result'
:
final_result
[
'print_result'
],
...
...
@@ -110,6 +134,7 @@ def Base_executeJupyter(self, python_expression=None, reference=None, \
transaction
.
abort
()
exception_dict
=
getErrorMessageForException
(
self
,
e
,
new_notebook_context
)
result
.
update
(
exception_dict
)
result
[
'notebook_context'
]
=
None
return
json
.
dumps
(
result
)
# Catch exception while seriaizing the result to be passed to jupyter frontend
...
...
@@ -128,10 +153,11 @@ def Base_executeJupyter(self, python_expression=None, reference=None, \
u'status'
:
u'error'
,
u'mime_type'
:
result
[
'mime_type'
]}
serialized_result
=
json
.
dumps
(
result
)
data_notebook_line
.
edit
(
notebook_code_result
=
result
[
'code_result'
],
mime_type
=
result
[
'mime_type'
])
if
data_notebook_line
is
not
None
:
data_notebook_line
.
edit
(
notebook_code_result
=
result
[
'code_result'
],
mime_type
=
result
[
'mime_type'
])
return
serialized_result
...
...
@@ -147,7 +173,7 @@ def mergeTracebackListIntoResultDict(result_dict, error_result_dict_list):
def
matplotlib_pre_run
():
matplotlib
.
interactive
(
Tru
e
)
matplotlib
.
interactive
(
Fals
e
)
rc
=
{
'figure.figsize'
:
(
6.0
,
4.0
),
'figure.facecolor'
:
(
1
,
1
,
1
,
0
),
'figure.edgecolor'
:
(
1
,
1
,
1
,
0
),
...
...
@@ -176,7 +202,10 @@ def matplotlib_post_run(data_list):
class
Displayhook
(
object
):
def
hook
(
self
,
value
):
if
value
is
not
None
:
self
.
result
=
repr
(
value
)
if
getattr
(
value
,
'_repr_html_'
,
None
)
is
not
None
:
self
.
result
=
{
'data'
:{
'text/html'
:
value
.
_repr_html_
()},
'metadata'
:{}}
else
:
self
.
result
=
repr
(
value
)
def
pre_run
(
self
):
self
.
old_hook
=
sys
.
displayhook
sys
.
displayhook
=
self
.
hook
...
...
@@ -277,6 +306,20 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
print_fixer
=
PrintFixer
()
environment_collector
=
EnvironmentParser
()
ast_node
=
import_fixer
.
visit
(
ast_node
)
# Whenever we have new imports we need to warn the user about the
# environment
if
(
import_fixer
.
warning_module_names
!=
[]):
warning
=
(
"print '"
"WARNING: You imported from the modules %s without "
"using the environment object, which is not recomended. "
"Your import was automatically converted to use such method. "
"The setup functions were named as *module*_setup. "
"'"
)
%
(
', '
.
join
(
import_fixer
.
warning_module_names
))
tree
=
ast
.
parse
(
warning
)
tree
.
body
[
0
].
lineno
=
ast_node
.
body
[
-
1
].
lineno
+
5
ast_node
.
body
.
append
(
tree
.
body
[
0
])
ast_node
=
print_fixer
.
visit
(
ast_node
)
ast
.
fix_missing_locations
(
ast_node
)
...
...
@@ -335,6 +378,7 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
'_print'
:
CustomPrint
()}
user_context
.
update
(
inject_variable_dict
)
user_context
.
update
(
notebook_context
[
'variables'
])
user_context
[
'parameter_dict'
]
=
notebook_context
[
'parameter_dict'
]
# Getting the environment setup defined in the current code cell
current_setup_dict
=
environment_collector
.
getEnvironmentSetupDict
()
...
...
@@ -381,6 +425,7 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
del
notebook_context
[
'setup'
][
key
]
# Running all the setup functions that we got
failed_setup_key_list
=
[]
for
key
,
value
in
notebook_context
[
'setup'
].
iteritems
():
try
:
code
=
compile
(
value
[
'code'
],
'<string>'
,
'exec'
)
...
...
@@ -388,12 +433,15 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
# An error happened, so we show the user the stacktrace along with a
# note that the exception happened in a setup function's code.
except
Exception
as
e
:
failed_setup_key_list
.
append
(
key
)
if
value
[
'func_name'
]
in
user_context
:
del
user_context
[
value
[
'func_name'
]]
error_return_dict
=
getErrorMessageForException
(
self
,
e
,
notebook_context
)
additional_information
=
"An error happened when trying to run the one of your setup functions:"
error_return_dict
[
'traceback'
].
insert
(
0
,
additional_information
)
setup_error_return_dict_list
.
append
(
error_return_dict
)
for
failed_setup_key
in
failed_setup_key_list
:
del
notebook_context
[
'setup'
][
failed_setup_key
]
# Iterating over envinronment.define calls captured by the environment collector
# that are functions and saving them as setup functions.
...
...
@@ -420,8 +468,8 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
user_context
[
'_volatile_variable_list'
]
+=
variable
if
environment_collector
.
showEnvironmentSetup
():
inject_variable_dict
.
write
(
"%s
\
n
"
%
str
(
notebook_context
[
'setup'
]))
inject_variable_dict
[
'_print'
]
.
write
(
"%s
\
n
"
%
str
(
notebook_context
[
'setup'
]))
# Execute the nodes with 'exec' mode
for
node
in
to_run_exec
:
mod
=
ast
.
Module
([
node
])
...
...
@@ -429,10 +477,12 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
try
:
exec
(
code
,
user_context
,
user_context
)
except
Exception
as
e
:
error_message
=
getErrorMessageForException
(
self
,
e
,
notebook_context
)
# Abort the current transaction. As a consequence, the notebook lines
# are not added if an exception occurs.
transaction
.
abort
()
return
mergeTracebackListIntoResultDict
(
getErrorMessageForException
(
self
,
e
,
notebook_context
),
log
(
error_message
)
return
mergeTracebackListIntoResultDict
(
error_message
,
setup_error_return_dict_list
)
# Execute the interactive nodes with 'single' mode
...
...
@@ -456,32 +506,36 @@ def Base_runJupyterCode(self, jupyter_code, old_notebook_context):
volatile_variable_list
=
current_setup_dict
.
keys
()
+
inject_variable_dict
.
keys
()
+
user_context
.
get
(
'_volatile_variable_list'
,
[])
volatile_variable_list
.
append
(
'__builtins__'
)
for
key
,
val
in
user_context
.
items
():
if
not
key
in
globals_dict
.
keys
()
and
not
isinstance
(
val
,
ModuleType
)
and
not
key
in
volatile_variable_list
:
if
canSerialize
(
val
):
notebook_context
[
'variables'
][
key
]
=
val
else
:
del
user_context
[
key
]
message
=
(
"Cannot serialize the variable named %s whose value is %s, "
"thus it will not be stored in the context. "
"You should move it's definition to a function and "
"use the environment object to load it.
\
n
"
)
%
(
key
,
val
)
inject_variable_dict
[
'_print'
].
write
(
message
)
# Klaus
#for key, val in user_context.items():
# if not key in globals_dict.keys() and not isinstance(val, well_known_unserializable_type_tuple) and not key in volatile_variable_list:
# if canSerialize(val):
# notebook_context['variables'][key] = val
# else:
# del user_context[key]
# message = (
# "Cannot serialize the variable named %s whose value is %s, "
# "thus it will not be stored in the context. "
# "You should move it's definition to a function and "
# "use the environment object to load it.\n"
# ) % (key, val)
# inject_variable_dict['_print'].write(message)
# Deleting from the variable storage the keys that are not in the user
# context anymore (i.e., variables that are deleted by the user).
for
key
in
notebook_context
[
'variables'
].
keys
():
if
not
key
in
user_context
:
del
notebook_context
[
'variables'
][
key
]
#
for key in notebook_context['variables'].keys():
#
if not key in user_context:
#
del notebook_context['variables'][key]
if
inject_variable_dict
.
get
(
'_print'
)
is
not
None
:
output
=
inject_variable_dict
[
'_print'
].
getCapturedOutputString
()
displayhook_result
=
{
"data"
:{},
"metadata"
:{}}
if
displayhook
.
result
is
not
None
:
displayhook_result
[
"data"
][
"text/plain"
]
=
displayhook
.
result
if
isinstance
(
displayhook
.
result
,
str
):
displayhook_result
[
"data"
][
"text/plain"
]
=
displayhook
.
result
elif
isinstance
(
displayhook
.
result
,
dict
):
displayhook_result
=
displayhook
.
result
result
=
{
'result_string'
:
output
,
'print_result'
:
{
"data"
:{
"text/plain"
:
output
},
"metadata"
:{}},
...
...
@@ -506,7 +560,7 @@ class EnvironmentDefinitionError(TypeError):
def
canSerialize
(
obj
):
container_type_tuple
=
(
list
,
tuple
,
dict
,
set
,
frozenset
)
# if object is a container, we need to check its elements for presence of
# objects that cannot be put inside the zodb
if
isinstance
(
obj
,
container_type_tuple
):
...
...
@@ -524,7 +578,11 @@ def canSerialize(obj):
# Need to unwrap the variable, otherwise we get a TypeError, because
# objects cannot be pickled while inside an acquisition wrapper.
unwrapped_obj
=
Acquisition
.
aq_base
(
obj
)
writer
=
ObjectWriter
(
unwrapped_obj
)
try
:
writer
=
ObjectWriter
(
unwrapped_obj
)
except
:
# Ignore any exceptions, otherwise Jupyter becomes permanent unusble state.
return
False
for
obj
in
writer
:
try
:
writer
.
serialize
(
obj
)
...
...
@@ -727,6 +785,7 @@ class ImportFixer(ast.NodeTransformer):
def
__init__
(
self
):
self
.
import_func_dict
=
{}
self
.
warning_module_names
=
[]
def
visit_FunctionDef
(
self
,
node
):
"""
...
...
@@ -835,24 +894,26 @@ class ImportFixer(ast.NodeTransformer):
if
not
self
.
import_func_dict
.
get
(
name
):
final_module_names
.
append
(
name
)
log
(
"module_names[0]: "
+
module_names
[
0
])
log
(
"result_name: "
+
result_name
)
if
final_module_names
:
# try to import module before it is added to environment
# this way if user tries to import non existent module Exception
# is immediately raised and doesn't block next Jupyter cell execution
exec
(
test_import_string
)
empty_function
=
self
.
newEmptyFunction
(
"%s_setup"
%
result_name
)
dotless_result_name
=
""
for
character
in
result_name
:
if
character
==
'.'
:
dotless_result_name
=
dotless_result_name
+
'_dot_'
else
:
dotless_result_name
=
dotless_result_name
+
character
empty_function
=
self
.
newEmptyFunction
(
"%s_setup"
%
dotless_result_name
)
return_dict
=
self
.
newReturnDict
(
final_module_names
)
log
(
return_dict
)
empty_function
.
body
=
[
node
,
return_dict
]
environment_set
=
self
.
newEnvironmentSetCall
(
"%s_setup"
%
result_name
)
warning
=
self
.
newImportWarningCall
(
root_module_name
,
result_name
)
return
[
empty_function
,
environment_set
,
warning
]
environment_set
=
self
.
newEnvironmentSetCall
(
"%s_setup"
%
dotless_
result_name
)
self
.
newImportWarningCall
(
root_module_name
,
dotless_
result_name
)
return
[
empty_function
,
environment_set
]
else
:
return
node
...
...
@@ -872,7 +933,11 @@ class ImportFixer(ast.NodeTransformer):
"""
return_dict
=
"return {"
for
name
in
module_names
:
return_dict
=
return_dict
+
"'%s': %s, "
%
(
name
,
name
)
if
name
.
find
(
'.'
)
!=
-
1
:
base_name
=
name
[:
name
.
find
(
'.'
)]
else
:
base_name
=
name
return_dict
=
return_dict
+
"'%s': %s, "
%
(
base_name
,
base_name
)
return_dict
=
return_dict
+
'}'
return
ast
.
parse
(
return_dict
).
body
[
0
]
...
...
@@ -887,18 +952,10 @@ class ImportFixer(ast.NodeTransformer):
def
newImportWarningCall
(
self
,
module_name
,
function_name
):
"""
Return an AST.Expr representanting a print statement with a warning to an
user about the import of a module named `module_name` and instructs him
on how to fix it.
Adds a new module to the warning to the user about the importing of new
modules.
"""
warning
=
(
"print '"
"WARNING: Your imported from the module %s without "
"using the environment object, which is not recomended. "
"Your import was automatically converted to use such method."
"The setup function was named as: %s_setup.
\
\
n"
"'"
)
%
(
module_name
,
function_name
)
tree
=
ast
.
parse
(
warning
)
return
tree
.
body
[
0
]
self
.
warning_module_names
.
append
(
module_name
)
def
renderAsHtml
(
self
,
renderable_object
):
...
...
@@ -947,11 +1004,12 @@ def getErrorMessageForException(self, exception, notebook_context):
'traceback'
:
traceback_text
}
def
createNotebookContext
(
self
):
def
createNotebookContext
(
self
,
parameter_dict
=
{}
):
"""
Function to create an empty notebook context.
"""
return
{
'variables'
:
{},
'setup'
:
{}}
nbc
=
{
'variables'
:
{},
'setup'
:
{},
'parameter_dict'
:
parameter_dict
}
return
nbc
class
ObjectProcessor
(
object
):
'''
...
...
@@ -1148,3 +1206,12 @@ def erp5PivotTableUI(self, df):
iframe_host
=
self
.
REQUEST
[
'HTTP_X_FORWARDED_HOST'
].
split
(
','
)[
0
]
url
=
"https://%s/erp5/Base_displayPivotTableFrame?key=%s"
%
(
iframe_host
,
key
)
return
IFrame
(
src
=
url
,
width
=
'100%'
,
height
=
'500'
)
def
Base_checkExistingReference
(
self
,
reference
):
existing_notebook
=
self
.
portal_catalog
.
getResultValue
(
owner
=
self
.
portal_membership
.
getAuthenticatedMember
().
getUserName
(),
portal_type
=
'Data Notebook'
,
reference
=
reference
)
if
not
existing_notebook
is
None
:
return
True
return
False
\ No newline at end of file
bt5/erp5_data_notebook/ExtensionTemplateItem/portal_components/extension.erp5.JupyterCompile.xml
View file @
eb896b21
...
...
@@ -46,13 +46,15 @@
<key>
<string>
text_content_warning_message
</string>
</key>
<value>
<tuple>
<string>
W:312, 10: Use of exec (exec-used)
</string>
<string>
W:355, 10: Use of exec (exec-used)
</string>
<string>
W:368, 10: Use of exec (exec-used)
</string>
<string>
W:453, 6: No exception type(s) specified (bare-except)
</string>
<string>
W:706, 6: Use of exec (exec-used)
</string>
<string>
W:932, 2: Redefining name \'IFrame\' from outer scope (line 4) (redefined-outer-name)
</string>
<string>
W: 18, 0: Unused log imported from Products.ERP5Type.Log (unused-import)
</string>
<string>
W:432, 10: Use of exec (exec-used)
</string>
<string>
W:478, 10: Use of exec (exec-used)
</string>
<string>
W:493, 10: Use of exec (exec-used)
</string>
<string>
W:286, 2: Unused variable \'globals_dict\' (unused-variable)
</string>
<string>
W:583, 4: No exception type(s) specified (bare-except)
</string>
<string>
W:591, 6: No exception type(s) specified (bare-except)
</string>
<string>
W:901, 6: Use of exec (exec-used)
</string>
<string>
W:1007, 0: Dangerous default value {} as argument (dangerous-default-value)
</string>
<string>
W:1138, 2: Redefining name \'IFrame\' from outer scope (line 4) (redefined-outer-name)
</string>
</tuple>
</value>
</item>
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment