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
Léo-Paul Géneau
erp5
Commits
f97e908a
Commit
f97e908a
authored
Sep 28, 2020
by
Tatuya Kamada
Browse files
Options
Browse Files
Download
Plain Diff
erp5_tabular_data_tool: Add portal_tabular (Tabular Data Tool)
See merge request
nexedi/erp5!1222
parents
92062f69
43affc49
Changes
17
Hide whitespace changes
Inline
Side-by-side
Showing
17 changed files
with
1134 additions
and
0 deletions
+1134
-0
bt5/erp5_tabular_data_tool/DocumentTemplateItem/portal_components/tool.erp5.TabularDataTool.py
...mplateItem/portal_components/tool.erp5.TabularDataTool.py
+196
-0
bt5/erp5_tabular_data_tool/DocumentTemplateItem/portal_components/tool.erp5.TabularDataTool.xml
...plateItem/portal_components/tool.erp5.TabularDataTool.xml
+127
-0
bt5/erp5_tabular_data_tool/PortalTypeTemplateItem/portal_types/Tabular%20Data%20Tool.xml
...alTypeTemplateItem/portal_types/Tabular%20Data%20Tool.xml
+60
-0
bt5/erp5_tabular_data_tool/ToolTemplateItem/portal_tabular.xml
...rp5_tabular_data_tool/ToolTemplateItem/portal_tabular.xml
+20
-0
bt5/erp5_tabular_data_tool/bt/dependency_list
bt5/erp5_tabular_data_tool/bt/dependency_list
+1
-0
bt5/erp5_tabular_data_tool/bt/template_document_id_list
bt5/erp5_tabular_data_tool/bt/template_document_id_list
+1
-0
bt5/erp5_tabular_data_tool/bt/template_format_version
bt5/erp5_tabular_data_tool/bt/template_format_version
+1
-0
bt5/erp5_tabular_data_tool/bt/template_portal_type_id_list
bt5/erp5_tabular_data_tool/bt/template_portal_type_id_list
+1
-0
bt5/erp5_tabular_data_tool/bt/template_tool_id_list
bt5/erp5_tabular_data_tool/bt/template_tool_id_list
+1
-0
bt5/erp5_tabular_data_tool/bt/template_update_tool
bt5/erp5_tabular_data_tool/bt/template_update_tool
+1
-0
bt5/erp5_tabular_data_tool/bt/title
bt5/erp5_tabular_data_tool/bt/title
+1
-0
bt5/erp5_tabular_data_tool_test/TestTemplateItem/portal_components/test.erp5.testTabularDataTool.py
...teItem/portal_components/test.erp5.testTabularDataTool.py
+591
-0
bt5/erp5_tabular_data_tool_test/TestTemplateItem/portal_components/test.erp5.testTabularDataTool.xml
...eItem/portal_components/test.erp5.testTabularDataTool.xml
+127
-0
bt5/erp5_tabular_data_tool_test/bt/dependency_list
bt5/erp5_tabular_data_tool_test/bt/dependency_list
+3
-0
bt5/erp5_tabular_data_tool_test/bt/template_format_version
bt5/erp5_tabular_data_tool_test/bt/template_format_version
+1
-0
bt5/erp5_tabular_data_tool_test/bt/template_test_id_list
bt5/erp5_tabular_data_tool_test/bt/template_test_id_list
+1
-0
bt5/erp5_tabular_data_tool_test/bt/title
bt5/erp5_tabular_data_tool_test/bt/title
+1
-0
No files found.
bt5/erp5_tabular_data_tool/DocumentTemplateItem/portal_components/tool.erp5.TabularDataTool.py
0 → 100644
View file @
f97e908a
from
AccessControl
import
ClassSecurityInfo
from
Products.ERP5Type
import
Permissions
from
Products.ERP5Type.Globals
import
InitializeClass
from
Products.ERP5Type.Tool.BaseTool
import
BaseTool
import
pandas
as
pd
MARKER
=
object
()
DEFAULT_FILL_STRING
=
''
DEFAULT_FILL_NUMBER
=
0.0
DEFAULT_FILL_UID
=
-
1
class
TabularDataTool
(
BaseTool
):
"""
Tabular Data Tool is a Tool to handle tabular data (2-dimentional table data) in erp5.
The tabular data type is currently a plain pandas DataFrame. It is intended to
use with ERP5 Reports to create spread sheets (ODS/Excel).
With Tabular Data Tool, we can do those things on reporting:
- transform dict-list, portal-catalog-result and document-list into a tabular data
- joining/grouping/sorting the tabular data
- using vectorizing function to optimise time consuming reports
"""
id
=
'portal_tabular'
title
=
'Tabular Data Tool'
meta_type
=
'ERP5 Tabular Data Tool'
portal_type
=
'Tabular Data Tool'
allowed_types
=
()
# Declarative Security
security
=
ClassSecurityInfo
()
# The empty DataFrame marker
def
_createEmptyDataFrame
(
self
):
return
pd
.
DataFrame
()
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'fillna'
)
def
fillna
(
self
,
tabular
,
fillna_on_string
=
MARKER
,
fillna_on_number
=
MARKER
,
fillna_on_uid
=
MARKER
):
"""
Fill NA/NaN values on the given tabular (a dataframe).
Keyword arguments:
tabular -- the dataframe to fill empty values
fillna_on_string -- Fill given value for empty string
fillna_on_number -- Fill given value for empty numbers (int or float)
fillna_on_uid -- Fill given value for empty uid, which overrides fillna_on_numbers for uid
"""
if
tabular
is
None
:
return
tabular
string_filler
=
DEFAULT_FILL_STRING
if
fillna_on_string
is
MARKER
else
fillna_on_string
number_filler
=
DEFAULT_FILL_NUMBER
if
fillna_on_number
is
MARKER
else
fillna_on_number
uid_filler
=
DEFAULT_FILL_UID
if
fillna_on_uid
is
MARKER
else
fillna_on_uid
fillna_setting_dict
=
{}
string_column_list
=
tabular
.
select_dtypes
(
include
=
[
object
]).
columns
.
tolist
()
string_none_column_list
=
[]
if
fillna_on_string
is
None
:
# pandas.dataframe.fillna(None) raises error, so use .replace() instead
string_none_column_list
=
string_column_list
string_column_list
=
[]
number_column_list
=
tabular
.
select_dtypes
(
include
=
[
int
,
float
]).
columns
.
tolist
()
uid_column_list
=
[
column
for
column
in
tabular
.
columns
.
tolist
()
if
column
.
endswith
(
'_uid'
)]
number_column_list
=
list
(
set
(
number_column_list
)
-
set
(
uid_column_list
))
string_column_list
=
list
(
set
(
string_column_list
)
-
set
(
uid_column_list
))
fillna_setting_dict
.
update
(
dict
.
fromkeys
(
string_column_list
,
string_filler
))
fillna_setting_dict
.
update
(
dict
.
fromkeys
(
number_column_list
,
number_filler
))
fillna_setting_dict
.
update
(
dict
.
fromkeys
(
uid_column_list
,
uid_filler
))
filled_tabular
=
tabular
.
fillna
(
value
=
fillna_setting_dict
)
if
string_none_column_list
:
replace_dict
=
dict
.
fromkeys
(
string_none_column_list
,
{
""
:
None
})
filled_tabular
=
filled_tabular
.
replace
(
replace_dict
)
return
filled_tabular
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'getTabular'
)
def
getTabular
(
self
,
data_list
,
fillna_on_string
=
MARKER
,
fillna_on_number
=
MARKER
,
fillna_on_uid
=
MARKER
,
additional_property_name_list
=
MARKER
):
"""
Create a tabular from dict list with filling values for convenience in ERP5.
Currently the tabular data type is pandas DataFrame.
Keyword arguments:
data_list -- A dict list or brain_list to convert it to dataframe
fillna_on_string -- Fill given value for empty string
fillna_on_numbers -- Fill given value for empty numbers (int or float)
fillna_on_uid -- Fill given value for empty uid, which overrides fillna_on_numbers for uid
additional_property_name_list -- Additional property name list of brain object
"""
if
data_list
is
None
or
len
(
data_list
)
==
0
:
return
self
.
_createEmptyDataFrame
()
df
=
None
if
isinstance
(
data_list
[
0
],
dict
):
df
=
pd
.
DataFrame
.
from_dict
(
data_list
)
# Expect Shared.ZRDB.Results
elif
getattr
(
data_list
,
'dictionaries'
,
None
)
is
not
None
:
if
additional_property_name_list
is
not
MARKER
:
new_data_dict_list
=
[]
for
(
data_
,
record_dict
)
in
zip
(
data_list
,
data_list
.
dictionaries
()):
property_dict
=
self
.
_getPropertyDict
(
data_
,
additional_property_name_list
,
False
,
False
)
property_dict
.
update
(
record_dict
)
new_data_dict_list
.
append
(
property_dict
)
df
=
self
.
getTabular
(
new_data_dict_list
)
else
:
df
=
self
.
getTabular
(
data_list
.
dictionaries
())
if
df
is
not
None
:
return
self
.
fillna
(
df
,
fillna_on_string
=
fillna_on_string
,
fillna_on_number
=
fillna_on_number
,
fillna_on_uid
=
fillna_on_uid
)
raise
ValueError
(
"The type is not supported, {}"
.
format
(
data_list
))
def
_getPropertyDict
(
self
,
document
,
property_name_list
,
is_category_as_uid
,
add_category_title
):
portal_type
=
document
.
getPortalType
()
base_type
=
self
.
getPortalObject
().
portal_types
[
portal_type
]
base_category_list
=
base_type
.
getInstanceBaseCategoryList
()
if
property_name_list
is
MARKER
:
property_name_list
=
list
(
base_type
.
getInstancePropertySet
())
else
:
# If property_name_list is explicitly passed, respect the name list in category
base_category_list
=
list
(
set
(
base_category_list
).
intersection
(
set
(
property_name_list
)))
property_dict
=
{}
for
property_name
in
property_name_list
:
property_dict
[
property_name
]
=
document
.
getProperty
(
property_name
)
for
base_category_id
in
base_category_list
:
category_name
=
"{}_uid"
.
format
(
base_category_id
)
if
is_category_as_uid
else
base_category_id
property_dict
[
category_name
]
=
document
.
getProperty
(
category_name
)
if
add_category_title
:
category_title
=
"{}_title"
.
format
(
base_category_id
)
property_dict
[
category_title
]
=
document
.
getProperty
(
category_title
)
return
property_dict
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'getFromDocumentList'
)
def
getFromDocumentList
(
self
,
document_list
,
property_name_list
=
MARKER
,
fillna_on_string
=
MARKER
,
fillna_on_number
=
MARKER
,
fillna_on_uid
=
MARKER
,
is_category_as_uid
=
True
,
add_category_title
=
True
):
"""
Create tabular data from document list
Keyword arguments:
property_name_list -- Explict to specify the property name list of document
If not specified all properties and categories of the portal type is applied
fillna_on_string -- Fill given value for empty string
fillna_on_numbers -- Fill given value for empty numbers (int or float)
fillna_on_uid -- Fill given value for empty uid, which overrides fill_on_numbers for uid
is_category_as_uid -- Store uid on {base_category}_uid, instead of store the url on {base_category}
add_category_title -- Add {category}_title columns into the tabular
Note: It takes O(N) time.
"""
if
document_list
is
None
or
len
(
document_list
)
==
0
:
return
self
.
_createEmptyDataFrame
()
# raise Attribute error if given document does not have portal_type
portal_type_set
=
set
([
x
.
getPortalType
()
for
x
in
document_list
])
if
len
(
portal_type_set
)
!=
1
:
raise
ValueError
(
'Given documents do not have the same portal type:{}'
.
format
(
portal_type_set
))
dict_list
=
[
self
.
_getPropertyDict
(
x
,
property_name_list
,
is_category_as_uid
,
add_category_title
)
for
x
in
document_list
]
return
self
.
getTabular
(
dict_list
,
fillna_on_string
=
fillna_on_string
,
fillna_on_number
=
fillna_on_number
,
fillna_on_uid
=
fillna_on_uid
)
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'searchResults'
)
def
searchResults
(
self
,
select_dict
=
None
,
fillna_on_string
=
MARKER
,
fillna_on_number
=
MARKER
,
fillna_on_uid
=
MARKER
,
**
kw
):
"""
Create tabular data based on the portal_catalog.searchResult()
select_dict -- If select_dict is None, specify all the column ids of catalog table.
"""
if
select_dict
is
None
:
portal
=
self
.
getPortalObject
()
catalog_id
=
portal
.
portal_catalog
.
getDefaultErp5CatalogId
()
default_erp5_catalog
=
portal
.
portal_catalog
[
catalog_id
]
column_id_set
=
set
(
default_erp5_catalog
.
getResultColumnIds
())
sql_search_result_keys_set
=
set
(
default_erp5_catalog
.
getSqlSearchResultKeysList
())
all_column_id_set
=
column_id_set
-
sql_search_result_keys_set
select_dict
=
dict
([(
column_id
,
None
)
for
column_id
in
all_column_id_set
if
column_id
.
startswith
(
'catalog.'
)])
search_result
=
portal
.
portal_catalog
(
select_dict
=
select_dict
,
**
kw
)
return
self
.
getTabular
(
search_result
,
fillna_on_string
=
fillna_on_string
,
fillna_on_number
=
fillna_on_number
,
fillna_on_uid
=
fillna_on_uid
)
__call__
=
getTabular
InitializeClass
(
TabularDataTool
)
\ No newline at end of file
bt5/erp5_tabular_data_tool/DocumentTemplateItem/portal_components/tool.erp5.TabularDataTool.xml
0 → 100644
View file @
f97e908a
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"Tool Component"
module=
"erp5.portal_type"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_recorded_property_dict
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<key>
<string>
default_reference
</string>
</key>
<value>
<string>
TabularDataTool
</string>
</value>
</item>
<item>
<key>
<string>
description
</string>
</key>
<value>
<none/>
</value>
</item>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
tool.erp5.TabularDataTool
</string>
</value>
</item>
<item>
<key>
<string>
portal_type
</string>
</key>
<value>
<string>
Tool Component
</string>
</value>
</item>
<item>
<key>
<string>
sid
</string>
</key>
<value>
<none/>
</value>
</item>
<item>
<key>
<string>
text_content_error_message
</string>
</key>
<value>
<tuple/>
</value>
</item>
<item>
<key>
<string>
text_content_warning_message
</string>
</key>
<value>
<tuple/>
</value>
</item>
<item>
<key>
<string>
version
</string>
</key>
<value>
<string>
erp5
</string>
</value>
</item>
<item>
<key>
<string>
workflow_history
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAM=
</string>
</persistent>
</value>
</item>
</dictionary>
</pickle>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"PersistentMapping"
module=
"Persistence.mapping"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
data
</string>
</key>
<value>
<dictionary/>
</value>
</item>
</dictionary>
</pickle>
</record>
<record
id=
"3"
aka=
"AAAAAAAAAAM="
>
<pickle>
<global
name=
"PersistentMapping"
module=
"Persistence.mapping"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
data
</string>
</key>
<value>
<dictionary>
<item>
<key>
<string>
component_validation_workflow
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAQ=
</string>
</persistent>
</value>
</item>
</dictionary>
</value>
</item>
</dictionary>
</pickle>
</record>
<record
id=
"4"
aka=
"AAAAAAAAAAQ="
>
<pickle>
<global
name=
"WorkflowHistoryList"
module=
"Products.ERP5Type.Workflow"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_log
</string>
</key>
<value>
<list>
<dictionary>
<item>
<key>
<string>
action
</string>
</key>
<value>
<string>
validate
</string>
</value>
</item>
<item>
<key>
<string>
validation_state
</string>
</key>
<value>
<string>
validated
</string>
</value>
</item>
</dictionary>
</list>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
bt5/erp5_tabular_data_tool/PortalTypeTemplateItem/portal_types/Tabular%20Data%20Tool.xml
0 → 100644
View file @
f97e908a
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"Base Type"
module=
"erp5.portal_type"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
acquire_local_roles
</string>
</key>
<value>
<int>
1
</int>
</value>
</item>
<item>
<key>
<string>
content_icon
</string>
</key>
<value>
<string>
folder_icon.gif
</string>
</value>
</item>
<item>
<key>
<string>
description
</string>
</key>
<value>
<string>
Tabular Data Tool is for handling tabular data (2-dimentional table data) .
</string>
</value>
</item>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
Tabular Data Tool
</string>
</value>
</item>
<item>
<key>
<string>
init_script
</string>
</key>
<value>
<none/>
</value>
</item>
<item>
<key>
<string>
permission
</string>
</key>
<value>
<none/>
</value>
</item>
<item>
<key>
<string>
portal_type
</string>
</key>
<value>
<string>
Base Type
</string>
</value>
</item>
<item>
<key>
<string>
type_class
</string>
</key>
<value>
<string>
TabularDataTool
</string>
</value>
</item>
<item>
<key>
<string>
type_interface
</string>
</key>
<value>
<tuple/>
</value>
</item>
<item>
<key>
<string>
type_mixin
</string>
</key>
<value>
<tuple/>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
bt5/erp5_tabular_data_tool/ToolTemplateItem/portal_tabular.xml
0 → 100644
View file @
f97e908a
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"Tabular Data Tool"
module=
"erp5.portal_type"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
portal_tabular
</string>
</value>
</item>
<item>
<key>
<string>
portal_type
</string>
</key>
<value>
<string>
Tabular Data Tool
</string>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
bt5/erp5_tabular_data_tool/bt/dependency_list
0 → 100644
View file @
f97e908a
erp5_base
\ No newline at end of file
bt5/erp5_tabular_data_tool/bt/template_document_id_list
0 → 100644
View file @
f97e908a
tool.erp5.TabularDataTool
\ No newline at end of file
bt5/erp5_tabular_data_tool/bt/template_format_version
0 → 100644
View file @
f97e908a
1
\ No newline at end of file
bt5/erp5_tabular_data_tool/bt/template_portal_type_id_list
0 → 100644
View file @
f97e908a
Tabular Data Tool
\ No newline at end of file
bt5/erp5_tabular_data_tool/bt/template_tool_id_list
0 → 100644
View file @
f97e908a
portal_tabular
\ No newline at end of file
bt5/erp5_tabular_data_tool/bt/template_update_tool
0 → 100644
View file @
f97e908a
1
\ No newline at end of file
bt5/erp5_tabular_data_tool/bt/title
0 → 100644
View file @
f97e908a
erp5_tabular_data_tool
\ No newline at end of file
bt5/erp5_tabular_data_tool_test/TestTemplateItem/portal_components/test.erp5.testTabularDataTool.py
0 → 100644
View file @
f97e908a
# coding: utf-8
##############################################################################
#
# Copyright (c) 2002-2020 Nexedi SA and Contributors. All Rights Reserved.
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsibility of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# guarantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
##############################################################################
from
Products.ERP5Type.tests.ERP5TypeTestCase
import
ERP5TypeTestCase
from
Products.ERP5Type.tests.Sequence
import
SequenceList
from
Products.ERP5Type.tests.utils
import
createZODBPythonScript
import
math
class
TestTabularDataTool
(
ERP5TypeTestCase
):
"""
Test for Tabular Data Tool
"""
#maxDiff = None
def
getTitle
(
self
):
return
"TestTabularDataTool"
def
getBusinessTemplateList
(
self
):
"""
Tuple of Business Templates we need to install
"""
return
(
'erp5_trade'
,)
def
beforeTearDown
(
self
):
"""
Remove this test related documents (for livetest in the real environment)
"""
organisation_module
=
self
.
portal
.
organisation_module
internal_packing_list_module
=
self
.
portal
.
internal_packing_list_module
product_module
=
self
.
portal
.
product_module
delete_module_list
=
[
organisation_module
,
internal_packing_list_module
,
product_module
]
def
isDelete
(
document
):
if
document
.
getTitle
()
is
None
:
return
False
return
document
.
getTitle
().
startswith
(
self
.
_getTestDocumentTitlePrefix
())
for
module
in
delete_module_list
:
delete_id_list
=
[
id_
for
id_
in
module
.
objectIds
()
if
isDelete
(
module
[
id_
])]
module
.
manage_delObjects
(
delete_id_list
)
rule_reference_script
=
'InternalPackingList_getRuleReference'
if
self
.
portal
.
portal_skins
.
custom
.
get
(
rule_reference_script
,
None
)
is
not
None
:
self
.
portal
.
portal_skins
.
custom
.
manage_delObjects
([
rule_reference_script
])
self
.
commit
()
def
_getTestDocumentTitlePrefix
(
self
):
return
self
.
getTitle
()
def
_getTestDocumentTitle
(
self
,
reference
):
return
"{}_{}"
.
format
(
self
.
_getTestDocumentTitlePrefix
(),
reference
)
def
testSearchResults
(
self
):
"""
Test portal_tabular.searchResults() which uses portal_catalog.searchResults()
as the given data.
"""
organisationA_title
=
self
.
_getTestDocumentTitle
(
'organisationA'
)
organisation_description
=
'(description default value is empty string)'
self
.
portal
.
organisation_module
.
newContent
(
portal_type
=
'Organisation'
,
description
=
organisation_description
,
title
=
organisationA_title
)
self
.
portal
.
organisation_module
.
newContent
(
portal_type
=
'Organisation'
,
description
=
organisation_description
,
title
=
self
.
_getTestDocumentTitle
(
'organisationB'
))
self
.
tic
()
result_list
=
self
.
portal
.
portal_catalog
(
portal_type
=
'Organisation'
,
title
=
self
.
_getTestDocumentTitle
(
'organisation%'
))
self
.
assertEqual
(
len
(
result_list
),
2
)
organisation_tabular
=
self
.
portal
.
portal_tabular
.
searchResults
(
portal_type
=
'Organisation'
,
title
=
self
.
_getTestDocumentTitle
(
'organisation%'
),
fillna_on_string
=
None
)
self
.
assertEqual
(
len
(
organisation_tabular
),
2
)
self
.
assertTrue
(
'title'
in
organisation_tabular
.
columns
.
tolist
())
organisationA_tabular
=
organisation_tabular
[
organisation_tabular
[
'title'
]
==
organisationA_title
]
self
.
assertEqual
(
len
(
organisationA_tabular
),
1
)
# only 'description' has empty string default value, so check this separatedly
self
.
assertEqual
(
organisationA_tabular
.
iloc
[
0
][
'description'
],
organisation_description
)
catalog_column_list
=
[
'uid'
,
'security_uid'
,
'owner'
,
'viewable_owner'
,
'path'
,
'relative_url'
,
'parent_uid'
,
'id'
,
'description'
,
'title'
,
'meta_type'
,
'portal_type'
,
'opportunity_state'
,
'corporate_registration_code'
,
'ean13_code'
,
'validation_state'
,
'simulation_state'
,
'causality_state'
,
'invoice_state'
,
'payment_state'
,
'event_state'
,
'immobilisation_state'
,
'reference'
,
'grouping_reference'
,
'grouping_date'
,
'source_reference'
,
'destination_reference'
,
'string_index'
,
'int_index'
,
'float_index'
,
'has_cell_content'
,
'creation_date'
,
'modification_date'
,
'indexation_timestamp'
]
self
.
assertTrue
(
len
(
organisationA_tabular
.
columns
.
tolist
())
>=
len
(
catalog_column_list
))
# catalog result has both uid and path by default so ignore them
select_dict
=
dict
.
fromkeys
(
set
(
catalog_column_list
)
-
set
([
'uid'
,
'path'
]),
None
)
catalog_search_result
=
self
.
portal
.
portal_catalog
.
searchResults
(
portal_type
=
'Organisation'
,
title
=
self
.
_getTestDocumentTitle
(
'organisation%'
),
select_dict
=
select_dict
)
organisationA_result
=
[
r
for
r
in
catalog_search_result
if
r
.
getTitle
()
==
organisationA_title
][
0
]
for
catalog_column
in
catalog_column_list
:
self
.
assertEqual
(
organisationA_tabular
.
iloc
[
0
][
catalog_column
],
organisationA_result
.
getProperty
(
catalog_column
),
"Not equal on {} expect:{}, result:{}"
.
format
(
catalog_column
,
organisationA_tabular
.
iloc
[
0
][
catalog_column
],
organisationA_result
.
getProperty
(
catalog_column
)
))
def
stepSetTestDocumentTitleNode1
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
node_title
=
self
.
_getTestDocumentTitle
(
'node1'
))
def
stepSetTestDocumentTitleResource1
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
resource_title
=
self
.
_getTestDocumentTitle
(
'resource1'
))
def
stepAddNode
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
node_title
=
sequence
.
get
(
'node_title'
)
node_value
=
self
.
portal
.
organisation_module
.
newContent
(
portal_type
=
'Organisation'
,
title
=
node_title
)
sequence
.
edit
(
node_value
=
node_value
)
def
stepAddResource
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
resource_title
=
sequence
.
get
(
'resource_title'
)
resource_value
=
self
.
portal
.
product_module
.
newContent
(
portal_type
=
'Product'
,
title
=
resource_title
)
sequence
.
edit
(
resource_value
=
resource_value
)
def
stepSetTestDocumentTitleMovement1
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
movement_title
=
self
.
_getTestDocumentTitle
(
'movement1'
))
def
stepAddInternalPackingList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
movement_title
=
sequence
.
get
(
'movement_title'
)
resource_value
=
sequence
.
get
(
'resource_value'
)
node_value
=
sequence
.
get
(
'node_value'
)
ipl
=
self
.
portal
.
internal_packing_list_module
.
newContent
(
portal_type
=
'Internal Packing List'
,
source_value
=
node_value
,
title
=
movement_title
)
ipl
.
newContent
(
portal_type
=
'Internal Packing List Line'
,
title
=
movement_title
,
quantity
=
10
,
resource_value
=
resource_value
)
sequence
.
edit
(
internal_packing_list
=
ipl
)
def
stepConfirmInternalPackingList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
internal_packing_list
=
sequence
.
get
(
'internal_packing_list'
)
internal_packing_list
.
confirm
()
self
.
assertEqual
(
internal_packing_list
.
getSimulationState
(),
'confirmed'
)
def
stepGetInventoryList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
resource_value
=
sequence
.
get
(
'resource_value'
)
node_value
=
sequence
.
get
(
'node_value'
)
title_prefix
=
sequence
.
get
(
'title_prefix'
,
None
)
if
title_prefix
is
None
:
title_prefix
=
self
.
_getTestDocumentTitlePrefix
()
inventory_list
=
self
.
portal
.
portal_simulation
.
getInventoryList
(
title
=
"{}%"
.
format
(
title_prefix
),
portal_type
=
'Internal Packing List Line'
,
resource_uid
=
resource_value
.
getUid
(),
node_uid
=
node_value
.
getUid
())
sequence
.
edit
(
inventory_list
=
inventory_list
)
def
stepGetAllResourceInventoryList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
title_prefix
=
sequence
.
get
(
'title_prefix'
,
None
)
node_value
=
sequence
.
get
(
'node_value'
)
if
title_prefix
is
None
:
title_prefix
=
self
.
_getTestDocumentTitlePrefix
()
inventory_list
=
self
.
portal
.
portal_simulation
.
getInventoryList
(
title
=
"{}%"
.
format
(
title_prefix
),
portal_type
=
'Internal Packing List Line'
,
node_uid
=
node_value
.
getUid
())
sequence
.
edit
(
inventory_list
=
inventory_list
)
def
stepCheckWithGetFromDocumentListAgainstInventoryList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
node_title
=
sequence
.
get
(
'node_title'
)
movement_title
=
sequence
.
get
(
'movement_title'
)
inventory_list
=
sequence
.
get
(
'inventory_list'
)
self
.
assertEqual
(
len
(
inventory_list
),
1
)
inventory_tabular
=
self
.
portal
.
portal_tabular
.
getFromDocumentList
(
inventory_list
,
property_name_list
=
[
"node_title"
,
"title"
])
self
.
assertEqual
(
len
(
inventory_tabular
),
1
)
self
.
assertTrue
(
'node_title'
in
inventory_tabular
.
columns
.
tolist
())
self
.
assertTrue
(
'title'
in
inventory_tabular
.
columns
.
tolist
())
# if explicitly specifies property_name_list,
# the tabular only has the specified columns
self
.
assertEqual
(
len
(
inventory_tabular
.
columns
.
tolist
()),
2
)
self
.
assertEqual
(
inventory_tabular
.
iloc
[
0
][
'node_title'
],
node_title
)
self
.
assertEqual
(
inventory_tabular
.
iloc
[
0
][
'title'
],
movement_title
)
default_inventory_tabular
=
self
.
portal
.
portal_tabular
.
getFromDocumentList
(
inventory_list
)
self
.
assertEqual
(
len
(
default_inventory_tabular
),
1
)
# The inventory list is ragarged as Internal Packing Line List, so does not have node_title
self
.
assertFalse
(
'node_title'
in
default_inventory_tabular
.
columns
.
tolist
())
self
.
assertTrue
(
'title'
in
default_inventory_tabular
.
columns
.
tolist
())
# If do not specify the property_names, it automatically add
# all the Sale Packing List Line property and category names based on the portal type
# configuration, so the number of columns is bigger than 2. (can be > 200)
self
.
assertTrue
(
len
(
default_inventory_tabular
.
columns
.
tolist
())
>
2
)
self
.
assertEqual
(
default_inventory_tabular
.
iloc
[
0
][
'title'
],
movement_title
)
self
.
assertTrue
(
'source_title'
in
default_inventory_tabular
.
columns
.
tolist
())
# The inventory list is regarded as Sale Packing Line List, so it has source_title
self
.
assertEqual
(
default_inventory_tabular
.
iloc
[
0
][
'source_title'
],
node_title
)
def
stepCheckWithGetTabularAgainstInventoryList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
inventory_list
=
sequence
.
get
(
'inventory_list'
)
node_title
=
sequence
.
get
(
'node_title'
)
node_value
=
sequence
.
get
(
'node_value'
)
resource_value
=
sequence
.
get
(
'resource_value'
)
resource_title
=
sequence
.
get
(
'resource_title'
)
movement_title
=
sequence
.
get
(
'movement_title'
)
# Test with portal_tabular() whitch is an alias of portal_tabular.getTabular()
# at first check without passing additonal_property_name_list parameter
inventory_tabular_without_name
=
self
.
portal
.
portal_tabular
(
inventory_list
)
property_name_list
=
[
'node_title'
,
'title'
,
'resource_title'
]
self
.
assertEqual
(
len
(
inventory_tabular_without_name
),
1
)
for
property_name
in
property_name_list
:
self
.
assertTrue
(
property_name
not
in
inventory_tabular_without_name
.
columns
.
tolist
(),
"{} is in the columns:{}"
.
format
(
property_name
,
inventory_tabular_without_name
.
columns
.
tolist
()))
# They have uid because Resource_zGetInventoryList returns them in the brain
self
.
assertEqual
(
inventory_tabular_without_name
.
iloc
[
0
][
'node_uid'
],
node_value
.
getUid
())
self
.
assertEqual
(
inventory_tabular_without_name
.
iloc
[
0
][
'resource_uid'
],
resource_value
.
getUid
())
# check with additional_property_name_list parameter
inventory_tabular_with_name
=
self
.
portal
.
portal_tabular
(
inventory_list
,
additional_property_name_list
=
property_name_list
)
self
.
assertEqual
(
len
(
inventory_tabular_with_name
),
1
)
self
.
assertEqual
(
len
(
inventory_tabular_with_name
.
columns
.
tolist
()),
len
(
inventory_tabular_without_name
.
columns
.
tolist
())
+
len
(
property_name_list
))
for
property_name
in
property_name_list
:
self
.
assertTrue
(
property_name
in
inventory_tabular_with_name
.
columns
.
tolist
(),
"{} is not in the columns:{}"
.
format
(
property_name
,
inventory_tabular_with_name
.
columns
.
tolist
()))
self
.
assertEqual
(
inventory_tabular_with_name
.
iloc
[
0
][
'node_title'
],
node_title
)
self
.
assertEqual
(
inventory_tabular_with_name
.
iloc
[
0
][
'title'
],
movement_title
)
self
.
assertEqual
(
inventory_tabular_with_name
.
iloc
[
0
][
'resource_title'
],
resource_title
)
def
testGetInventoryList
(
self
):
sequence_list
=
SequenceList
()
sequence_string
=
"""
DisableSimulationOnInternalPackingList
SetTestDocumentTitleNode1
AddNode
SetTestDocumentTitleResource1
AddResource
SetTestDocumentTitleMovement1
AddInternalPackingList
ConfirmInternalPackingList
Tic
GetInventoryList
CheckWithGetFromDocumentListAgainstInventoryList
CheckWithGetTabularAgainstInventoryList
"""
sequence_list
.
addSequenceString
(
sequence_string
)
sequence_list
.
play
(
self
)
def
testFillnaOnString
(
self
):
"""
Test fillna_on_string parameter
"""
dict_list
=
[{
"test"
:
"a"
},
{
"test"
:
None
},
{
"test"
:
""
},
{
"test"
:
"d"
},
{
"test"
:
None
}]
# by default it is empty
df1
=
self
.
portal
.
portal_tabular
(
dict_list
)
empty_value
=
""
self
.
assertEqual
(
df1
.
iloc
[
0
][
"test"
],
"a"
)
self
.
assertEqual
(
df1
.
iloc
[
1
][
"test"
],
empty_value
)
self
.
assertEqual
(
df1
.
iloc
[
2
][
"test"
],
""
)
self
.
assertEqual
(
df1
.
iloc
[
3
][
"test"
],
"d"
)
self
.
assertEqual
(
df1
.
iloc
[
4
][
"test"
],
empty_value
)
empty_value_2
=
"""[No Value]"""
df2
=
self
.
portal
.
portal_tabular
(
dict_list
,
fillna_on_string
=
empty_value_2
)
self
.
assertEqual
(
df2
.
iloc
[
0
][
"test"
],
"a"
)
self
.
assertEqual
(
df2
.
iloc
[
1
][
"test"
],
empty_value_2
)
self
.
assertEqual
(
df2
.
iloc
[
2
][
"test"
],
""
)
self
.
assertEqual
(
df2
.
iloc
[
3
][
"test"
],
"d"
)
self
.
assertEqual
(
df2
.
iloc
[
4
][
"test"
],
empty_value_2
)
def
testFillnaOnNumber
(
self
):
"""
Test fillna_on_number
"""
dict_list
=
[{
"test"
:
0
},
{
"test"
:
None
},
{
"test"
:
2
},
{
"test"
:
3
},
{
"test"
:
None
}]
# by default it is empty
df1
=
self
.
portal
.
portal_tabular
(
dict_list
)
empty_value
=
0.0
self
.
assertEqual
(
df1
.
iloc
[
0
][
"test"
],
0
)
self
.
assertEqual
(
df1
.
iloc
[
1
][
"test"
],
empty_value
)
self
.
assertEqual
(
df1
.
iloc
[
2
][
"test"
],
2
)
self
.
assertEqual
(
df1
.
iloc
[
3
][
"test"
],
3
)
self
.
assertEqual
(
df1
.
iloc
[
4
][
"test"
],
empty_value
)
empty_value_2
=
-
1
df2
=
self
.
portal
.
portal_tabular
(
dict_list
,
fillna_on_number
=
empty_value_2
)
self
.
assertEqual
(
df2
.
iloc
[
0
][
"test"
],
0
)
self
.
assertEqual
(
df2
.
iloc
[
1
][
"test"
],
empty_value_2
)
self
.
assertEqual
(
df2
.
iloc
[
2
][
"test"
],
2
)
self
.
assertEqual
(
df2
.
iloc
[
3
][
"test"
],
3
)
self
.
assertEqual
(
df2
.
iloc
[
4
][
"test"
],
empty_value_2
)
def
testGetFromDocumentList
(
self
):
"""
Check to specify accessor
"""
organisation_list
=
[]
for
_title
in
(
'testAccesssor1'
,
'testAccesssor2'
):
organisation_title
=
self
.
_getTestDocumentTitle
(
_title
)
organisation
=
self
.
portal
.
organisation_module
.
newContent
(
portal_type
=
'Organisation'
,
title
=
organisation_title
,
activity_code
=
'test activity code'
)
organisation_list
.
append
(
organisation
)
organisation_tabular
=
self
.
portal
.
portal_tabular
.
getFromDocumentList
(
organisation_list
)
self
.
assertEqual
(
len
(
organisation_tabular
),
2
)
self
.
assertEqual
(
organisation_tabular
.
iloc
[
0
][
'activity_code'
],
'test activity code'
)
def
testFillnaOnUid
(
self
):
"""
Check fillna_on_uid parameter
"""
node_title
=
self
.
_getTestDocumentTitle
(
"test_node_fillna_on_uid_1"
)
node_value
=
self
.
portal
.
organisation_module
.
newContent
(
portal_type
=
'Organisation'
,
title
=
node_title
)
spl_title
=
self
.
_getTestDocumentTitle
(
"test_spl_fillna_on_uid_1"
)
spl
=
self
.
portal
.
sale_packing_list_module
.
newContent
(
portal_type
=
'Sale Packing List'
,
source_value
=
node_value
,
title
=
spl_title
)
sale_packing_list_list
=
[
spl
]
sale_packing_list_tabular
=
self
.
portal
.
portal_tabular
.
getFromDocumentList
(
sale_packing_list_list
)
self
.
assertEqual
(
len
(
sale_packing_list_tabular
),
1
)
self
.
assertEqual
(
sale_packing_list_tabular
.
iloc
[
0
][
'destination_uid'
],
-
1
)
self
.
assertEqual
(
sale_packing_list_tabular
.
iloc
[
0
][
'source_uid'
],
node_value
.
getUid
())
empty_uid
=
-
99999999
sale_packing_list_tabular2
=
self
.
portal
.
portal_tabular
.
getFromDocumentList
(
sale_packing_list_list
,
fillna_on_uid
=
empty_uid
)
self
.
assertEqual
(
len
(
sale_packing_list_tabular2
),
1
)
self
.
assertEqual
(
sale_packing_list_tabular2
.
iloc
[
0
][
'destination_uid'
],
empty_uid
)
self
.
assertEqual
(
sale_packing_list_tabular2
.
iloc
[
0
][
'source_uid'
],
node_value
.
getUid
())
def
testFillna
(
self
):
"""
Test portal_tabular.fillna() method
"""
dict_list
=
[{
'test'
:
0
},
{
'test'
:
None
},
{
'test'
:
2
},
{
'test'
:
3
},
{
'test'
:
None
}]
# by default it is empty
df1
=
self
.
portal
.
portal_tabular
(
dict_list
)
numeric_empty_value
=
0.0
self
.
assertEqual
(
df1
.
iloc
[
0
][
'test'
],
0
)
self
.
assertEqual
(
df1
.
iloc
[
1
][
'test'
],
numeric_empty_value
)
self
.
assertEqual
(
df1
.
iloc
[
2
][
'test'
],
2
)
self
.
assertEqual
(
df1
.
iloc
[
3
][
'test'
],
3
)
self
.
assertEqual
(
df1
.
iloc
[
4
][
'test'
],
numeric_empty_value
)
# add new column into the dataframe
df1
[
'test_numeric'
]
=
[
None
,
100
,
None
,
200
,
300
]
# So by default Na value of numeric is NaN in pandas dataframe
self
.
assertTrue
(
math
.
isnan
(
df1
.
iloc
[
0
][
'test_numeric'
]))
self
.
assertEqual
(
df1
.
iloc
[
1
][
'test_numeric'
],
100
)
self
.
assertTrue
(
math
.
isnan
(
df1
.
iloc
[
2
][
'test_numeric'
]))
self
.
assertEqual
(
df1
.
iloc
[
3
][
'test_numeric'
],
200
)
self
.
assertEqual
(
df1
.
iloc
[
4
][
'test_numeric'
],
300
)
df1
[
'test_string'
]
=
[
'x'
,
None
,
'y'
,
None
,
'z'
]
# The default Na value of string is None in pandas dataframe
self
.
assertEqual
(
df1
.
iloc
[
0
][
'test_string'
],
'x'
)
self
.
assertEqual
(
df1
.
iloc
[
1
][
'test_string'
],
None
)
self
.
assertEqual
(
df1
.
iloc
[
2
][
'test_string'
],
'y'
)
self
.
assertEqual
(
df1
.
iloc
[
3
][
'test_string'
],
None
)
self
.
assertEqual
(
df1
.
iloc
[
4
][
'test_string'
],
'z'
)
# Fill Na/NaN values with portal_tabular.fillna()
df2
=
self
.
portal
.
portal_tabular
.
fillna
(
df1
)
self
.
assertEqual
(
df2
.
iloc
[
0
][
'test_numeric'
],
numeric_empty_value
)
self
.
assertEqual
(
df2
.
iloc
[
1
][
'test_numeric'
],
100
)
self
.
assertEqual
(
df2
.
iloc
[
2
][
'test_numeric'
],
numeric_empty_value
)
self
.
assertEqual
(
df2
.
iloc
[
3
][
'test_numeric'
],
200
)
self
.
assertEqual
(
df2
.
iloc
[
4
][
'test_numeric'
],
300
)
string_empty_value
=
''
# Test they are filled with the default string filler ''
self
.
assertEqual
(
df2
.
iloc
[
0
][
'test_string'
],
'x'
)
self
.
assertEqual
(
df2
.
iloc
[
1
][
'test_string'
],
string_empty_value
)
self
.
assertEqual
(
df2
.
iloc
[
2
][
'test_string'
],
'y'
)
self
.
assertEqual
(
df2
.
iloc
[
3
][
'test_string'
],
string_empty_value
)
self
.
assertEqual
(
df2
.
iloc
[
4
][
'test_string'
],
'z'
)
# portal_tabular.fillna() with specifying the empty values
numeric_empty_value_2
=
-
1
string_empty_value_2
=
"""[No Value]"""
df3
=
self
.
portal
.
portal_tabular
.
fillna
(
df1
,
fillna_on_number
=
numeric_empty_value_2
,
fillna_on_string
=
string_empty_value_2
)
# Test with the specified filler
self
.
assertEqual
(
df3
.
iloc
[
0
][
'test_numeric'
],
numeric_empty_value_2
)
self
.
assertEqual
(
df3
.
iloc
[
1
][
'test_numeric'
],
100
)
self
.
assertEqual
(
df3
.
iloc
[
2
][
'test_numeric'
],
numeric_empty_value_2
)
self
.
assertEqual
(
df3
.
iloc
[
3
][
'test_numeric'
],
200
)
self
.
assertEqual
(
df3
.
iloc
[
4
][
'test_numeric'
],
300
)
self
.
assertEqual
(
df3
.
iloc
[
0
][
'test_string'
],
'x'
)
self
.
assertEqual
(
df3
.
iloc
[
1
][
'test_string'
],
string_empty_value_2
)
self
.
assertEqual
(
df3
.
iloc
[
2
][
'test_string'
],
'y'
)
self
.
assertEqual
(
df3
.
iloc
[
3
][
'test_string'
],
string_empty_value_2
)
self
.
assertEqual
(
df3
.
iloc
[
4
][
'test_string'
],
'z'
)
def
stepDisableSimulationOnInternalPackingList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
# Do not care simulation in this test, so return None
# for the rule reference of Internal Packing List
script_container
=
self
.
portal
.
portal_skins
.
custom
code
=
'return None'
createZODBPythonScript
(
script_container
,
'InternalPackingList_getRuleReference'
,
'**kw'
,
code
)
def
stepSetTestDocumentTitleNode2
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
node_title
=
self
.
_getTestDocumentTitle
(
'node2'
))
def
stepSetTestDocumentTitleResource2
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
resource_title
=
self
.
_getTestDocumentTitle
(
'resource2'
))
def
stepSetTestDocumentTitleTestReportMovement1
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
movement_title
=
self
.
_getTestDocumentTitle
(
'test_report_movement1'
))
def
stepSetTestDocumentTitleTestReportMovement2
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
movement_title
=
self
.
_getTestDocumentTitle
(
'test_report_movement2'
))
def
stepSetTestDocumentTitleTestReportMovement3
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
movement_title
=
self
.
_getTestDocumentTitle
(
'test_report_movement3'
))
def
stepSetTestDocumentTitleTestReportMovement4
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
sequence
.
edit
(
movement_title
=
self
.
_getTestDocumentTitle
(
'test_report_movement4'
))
def
stepConfirmAllInternalPackingList
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
title
=
"{}_{}%"
.
format
(
self
.
_getTestDocumentTitlePrefix
(),
'test_report_movement'
)
result
=
self
.
portal
.
portal_catalog
(
portal_type
=
'Internal Packing List'
,
title
=
title
)
for
r
in
result
:
internal_packing_list
=
r
.
getObject
()
internal_packing_list
.
confirm
()
self
.
assertEqual
(
internal_packing_list
.
getSimulationState
(),
'confirmed'
)
def
stepSetTitlePrefix
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
title_prefix
=
"{}_{}"
.
format
(
self
.
_getTestDocumentTitlePrefix
(),
'test_report_movement'
)
sequence
.
edit
(
title_prefix
=
title_prefix
)
def
stepCheckToCreateReport
(
self
,
sequence
=
None
,
sequence_list
=
None
,
**
kw
):
inventory_list
=
sequence
.
get
(
'inventory_list'
)
additional_property_name_list
=
[
'resource_title'
,
'node_title'
,
'title'
]
inventory_tabular
=
self
.
portal
.
portal_tabular
(
inventory_list
,
additional_property_name_list
=
additional_property_name_list
)
self
.
assertEqual
(
len
(
inventory_tabular
),
4
)
node_value
=
sequence
.
get
(
'node_value'
)
for
i
in
range
(
4
):
self
.
assertEqual
(
inventory_tabular
.
iloc
[
i
][
'node_title'
],
node_value
.
getTitle
())
self
.
assertEqual
(
inventory_tabular
.
iloc
[
i
][
'inventory'
],
-
10
)
"""
node1, resource1, movement1, -10
node1, resource1, movement2, -10
node1, resource2, movement3, -10
node1, resource2, movement4, -10
"""
# Aggregate the result grouping by resource_title and node_title,
# then sorting by resource_title, and summing the inventory.
report_tabular
=
inventory_tabular
.
groupby
(
[
'resource_title'
,
'node_title'
,
]
).
agg
(
dict
(
inventory
=
'sum'
)).
reset_index
().
sort_values
(
by
=
[
'resource_title'
])
self
.
assertEqual
(
len
(
report_tabular
),
2
)
self
.
assertEqual
(
report_tabular
.
iloc
[
0
][
'inventory'
],
-
20
)
self
.
assertEqual
(
report_tabular
.
iloc
[
1
][
'inventory'
],
-
20
)
title_prefix
=
self
.
_getTestDocumentTitlePrefix
()
# The result is ordered by resource title
self
.
assertEqual
(
report_tabular
.
iloc
[
0
][
'resource_title'
],
"{}_{}"
.
format
(
title_prefix
,
'resource1'
))
self
.
assertEqual
(
report_tabular
.
iloc
[
1
][
'resource_title'
],
"{}_{}"
.
format
(
title_prefix
,
'resource2'
))
# Turn into dict to show in listbox for example
report_dict_list
=
report_tabular
.
to_dict
(
"records"
)
for
report_dict
in
report_dict_list
:
self
.
assertTrue
(
isinstance
(
report_dict
,
dict
))
self
.
assertEqual
(
report_dict
[
'node_title'
],
node_value
.
getTitle
())
self
.
assertEqual
(
report_dict
[
'inventory'
],
-
20
)
self
.
assertTrue
(
'resource_title'
in
report_dict
)
def
testReport
(
self
):
"""
Test tabular data result for reports such as groupby, join, and sorting
"""
sequence_list
=
SequenceList
()
sequence_string
=
"""
DisableSimulationOnInternalPackingList
Tic
SetTestDocumentTitleNode1
AddNode
SetTestDocumentTitleResource1
AddResource
SetTestDocumentTitleTestReportMovement1
AddInternalPackingList
SetTestDocumentTitleTestReportMovement2
AddInternalPackingList
SetTestDocumentTitleResource2
AddResource
SetTestDocumentTitleTestReportMovement3
AddInternalPackingList
SetTestDocumentTitleTestReportMovement4
AddInternalPackingList
ConfirmAllInternalPackingList
Tic
SetTitlePrefix
GetAllResourceInventoryList
CheckToCreateReport
"""
sequence_list
.
addSequenceString
(
sequence_string
)
sequence_list
.
play
(
self
)
def
testEncoding
(
self
):
"""
Test portal_tabular returns utf-8 string not unicode string
"""
dict_list
=
[{
'test'
:
None
},
{
'test'
:
'あ'
},
{
'test'
:
''
},
{
'test'
:
'ç'
},
{
'test'
:
None
}]
# by default it is empty
df1
=
self
.
portal
.
portal_tabular
(
dict_list
)
string_empty_value
=
''
self
.
assertEqual
(
df1
.
iloc
[
0
][
'test'
],
string_empty_value
)
self
.
assertEqual
(
df1
.
iloc
[
1
][
'test'
],
'あ'
)
self
.
assertEqual
(
df1
.
iloc
[
2
][
'test'
],
string_empty_value
)
self
.
assertEqual
(
df1
.
iloc
[
3
][
'test'
],
'ç'
)
self
.
assertEqual
(
df1
.
iloc
[
4
][
'test'
],
string_empty_value
)
def
testEmptyTabular
(
self
):
"""
Test updating empty tabular does not update different empty tabular
"""
empty_tabular
=
self
.
portal
.
portal_tabular
(
data_list
=
[])
empty_tabular
[
'a'
]
=
[
1
,
2
,
3
]
empty_tabular2
=
self
.
portal
.
portal_tabular
(
data_list
=
[])
self
.
assertNotEquals
(
len
(
empty_tabular
.
columns
),
len
(
empty_tabular2
.
columns
))
self
.
assertNotEquals
(
len
(
empty_tabular
),
len
(
empty_tabular2
))
\ No newline at end of file
bt5/erp5_tabular_data_tool_test/TestTemplateItem/portal_components/test.erp5.testTabularDataTool.xml
0 → 100644
View file @
f97e908a
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"Test Component"
module=
"erp5.portal_type"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_recorded_property_dict
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<key>
<string>
default_reference
</string>
</key>
<value>
<string>
testTabularDataTool
</string>
</value>
</item>
<item>
<key>
<string>
description
</string>
</key>
<value>
<none/>
</value>
</item>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
test.erp5.testTabularDataTool
</string>
</value>
</item>
<item>
<key>
<string>
portal_type
</string>
</key>
<value>
<string>
Test Component
</string>
</value>
</item>
<item>
<key>
<string>
sid
</string>
</key>
<value>
<none/>
</value>
</item>
<item>
<key>
<string>
text_content_error_message
</string>
</key>
<value>
<tuple/>
</value>
</item>
<item>
<key>
<string>
text_content_warning_message
</string>
</key>
<value>
<tuple/>
</value>
</item>
<item>
<key>
<string>
version
</string>
</key>
<value>
<string>
erp5
</string>
</value>
</item>
<item>
<key>
<string>
workflow_history
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAM=
</string>
</persistent>
</value>
</item>
</dictionary>
</pickle>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"PersistentMapping"
module=
"Persistence.mapping"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
data
</string>
</key>
<value>
<dictionary/>
</value>
</item>
</dictionary>
</pickle>
</record>
<record
id=
"3"
aka=
"AAAAAAAAAAM="
>
<pickle>
<global
name=
"PersistentMapping"
module=
"Persistence.mapping"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
data
</string>
</key>
<value>
<dictionary>
<item>
<key>
<string>
component_validation_workflow
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAQ=
</string>
</persistent>
</value>
</item>
</dictionary>
</value>
</item>
</dictionary>
</pickle>
</record>
<record
id=
"4"
aka=
"AAAAAAAAAAQ="
>
<pickle>
<global
name=
"WorkflowHistoryList"
module=
"Products.ERP5Type.Workflow"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_log
</string>
</key>
<value>
<list>
<dictionary>
<item>
<key>
<string>
action
</string>
</key>
<value>
<string>
validate
</string>
</value>
</item>
<item>
<key>
<string>
validation_state
</string>
</key>
<value>
<string>
validated
</string>
</value>
</item>
</dictionary>
</list>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
bt5/erp5_tabular_data_tool_test/bt/dependency_list
0 → 100644
View file @
f97e908a
erp5_trade
erp5_full_text_mroonga_catalog
erp5_tabular_data_tool
\ No newline at end of file
bt5/erp5_tabular_data_tool_test/bt/template_format_version
0 → 100644
View file @
f97e908a
1
\ No newline at end of file
bt5/erp5_tabular_data_tool_test/bt/template_test_id_list
0 → 100644
View file @
f97e908a
test.erp5.testTabularDataTool
\ No newline at end of file
bt5/erp5_tabular_data_tool_test/bt/title
0 → 100644
View file @
f97e908a
erp5_tabular_data_tool_test
\ No newline at end of file
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