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
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
Romain Courteaud
erp5
Commits
049fd1cb
Commit
049fd1cb
authored
May 09, 2024
by
Rafael Monnerat
Browse files
Options
Browse Files
Download
Plain Diff
Update from upstream/master
parents
5e36aa7e
233d8641
Changes
48
Hide whitespace changes
Inline
Side-by-side
Showing
48 changed files
with
909 additions
and
377 deletions
+909
-377
CHANGES.erp5.util.txt
CHANGES.erp5.util.txt
+6
-0
bt5/erp5_accounting/TestTemplateItem/portal_components/test.erp5.testAccounting.py
...emplateItem/portal_components/test.erp5.testAccounting.py
+6
-2
bt5/erp5_administration/ExtensionTemplateItem/portal_components/extension.erp5.ERP5Administration.py
...em/portal_components/extension.erp5.ERP5Administration.py
+4
-1
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/Base_viewSecurity.py
...tem/portal_skins/erp5_administration/Base_viewSecurity.py
+0
-11
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/Base_viewSecurityMappingAsUser.py
...ins/erp5_administration/Base_viewSecurityMappingAsUser.py
+0
-27
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/Base_viewSecurityMappingAsUser.xml
...ns/erp5_administration/Base_viewSecurityMappingAsUser.xml
+0
-71
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpAlarmToolConfiguration.xml
...p5_administration/ERP5Site_dumpAlarmToolConfiguration.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpBuilderList.xml
...al_skins/erp5_administration/ERP5Site_dumpBuilderList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpInstalledBusinessTemplateList.xml
...nistration/ERP5Site_dumpInstalledBusinessTemplateList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpOrderBuilderList.xml
...ins/erp5_administration/ERP5Site_dumpOrderBuilderList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalSkinsContent.xml
...s/erp5_administration/ERP5Site_dumpPortalSkinsContent.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalTypeActionList.xml
...erp5_administration/ERP5Site_dumpPortalTypeActionList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalTypeList.xml
...skins/erp5_administration/ERP5Site_dumpPortalTypeList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalTypeRoleList.xml
...s/erp5_administration/ERP5Site_dumpPortalTypeRoleList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPropertySheetList.xml
...ns/erp5_administration/ERP5Site_dumpPropertySheetList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpRuleTesterList.xml
...skins/erp5_administration/ERP5Site_dumpRuleTesterList.xml
+23
-0
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpSkinProperty.xml
...l_skins/erp5_administration/ERP5Site_dumpSkinProperty.xml
+23
-0
bt5/erp5_base/DocumentTemplateItem/portal_components/document.erp5.Person.py
...entTemplateItem/portal_components/document.erp5.Person.py
+2
-2
bt5/erp5_base/DocumentTemplateItem/portal_components/document.erp5.RoleDefinition.py
...ateItem/portal_components/document.erp5.RoleDefinition.py
+20
-5
bt5/erp5_corporate_identity/bt/template_action_path_list
bt5/erp5_corporate_identity/bt/template_action_path_list
+0
-1
bt5/erp5_simulation/DocumentTemplateItem/portal_components/document.erp5.SpecialiseEquivalenceTester.py
...l_components/document.erp5.SpecialiseEquivalenceTester.py
+1
-1
bt5/erp5_test_result/DocumentTemplateItem/portal_components/document.erp5.CloudPerformanceUnitTestDistributor.py
...ents/document.erp5.CloudPerformanceUnitTestDistributor.py
+3
-3
bt5/erp5_ui_test_core/ExtensionTemplateItem/portal_components/extension.erp5.ERP5Zuite.py
...emplateItem/portal_components/extension.erp5.ERP5Zuite.py
+2
-1
bt5/erp5_web/ActionTemplateItem/portal_types/Web%20Page/render.xml
...web/ActionTemplateItem/portal_types/Web%20Page/render.xml
+0
-0
bt5/erp5_web/SkinTemplateItem/portal_skins/erp5_web/WebPage_viewPreview.xml
...emplateItem/portal_skins/erp5_web/WebPage_viewPreview.xml
+1
-1
bt5/erp5_web/SkinTemplateItem/portal_skins/erp5_web/WebPage_viewPreview/iframe_content.xml
...tal_skins/erp5_web/WebPage_viewPreview/iframe_content.xml
+0
-0
bt5/erp5_web/SkinTemplateItem/portal_skins/erp5_web/WebPage_viewPreview/text_content.xml
...ortal_skins/erp5_web/WebPage_viewPreview/text_content.xml
+0
-0
bt5/erp5_web/bt/template_action_path_list
bt5/erp5_web/bt/template_action_path_list
+1
-0
bt5/erp5_web_renderjs_ui_test/PathTemplateItem/portal_tests/renderjs_ui_relation_field_zuite/testMultiRelationFieldEmptySearchWithListbox.zpt
...ld_zuite/testMultiRelationFieldEmptySearchWithListbox.zpt
+25
-5
bt5/erp5_web_renderjs_ui_test/PathTemplateItem/portal_tests/renderjs_ui_relation_field_zuite/testMultiRelationFieldSearchWithListbox.zpt
...n_field_zuite/testMultiRelationFieldSearchWithListbox.zpt
+25
-10
erp5/util/testnode/__init__.py
erp5/util/testnode/__init__.py
+1
-1
product/ERP5/ERP5Site.py
product/ERP5/ERP5Site.py
+2
-17
product/ERP5/bootstrap/erp5_core/ExtensionTemplateItem/portal_components/extension.erp5.PythonCodeUtils.py
...eItem/portal_components/extension.erp5.PythonCodeUtils.py
+11
-1
product/ERP5/bootstrap/erp5_core/ExtensionTemplateItem/portal_components/extension.erp5.StandardSecurity.py
...Item/portal_components/extension.erp5.StandardSecurity.py
+171
-0
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5Type_asSecurityGroupId.py
...Item/portal_skins/erp5_core/ERP5Type_asSecurityGroupId.py
+0
-95
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5Type_asSecurityGroupIdSet.xml
.../portal_skins/erp5_core/ERP5Type_asSecurityGroupIdSet.xml
+28
-0
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5User_getSecurityCategoryValueFromAssignment.xml
..._core/ERP5User_getSecurityCategoryValueFromAssignment.xml
+28
-0
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5User_getUserSecurityCategoryValueList.py
...ns/erp5_core/ERP5User_getUserSecurityCategoryValueList.py
+15
-0
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5User_getUserSecurityCategoryValueList.xml
...s/erp5_core/ERP5User_getUserSecurityCategoryValueList.xml
+2
-2
product/ERP5Form/MultiRelationField.py
product/ERP5Form/MultiRelationField.py
+2
-0
product/ERP5Security/ERP5GroupManager.py
product/ERP5Security/ERP5GroupManager.py
+138
-46
product/ERP5Security/tests/testERP5Security.py
product/ERP5Security/tests/testERP5Security.py
+97
-46
product/ERP5Type/Core/RoleInformation.py
product/ERP5Type/Core/RoleInformation.py
+46
-11
product/ERP5Type/ERP5Type.py
product/ERP5Type/ERP5Type.py
+1
-0
product/ERP5Type/tests/CodingStyleTestCase.py
product/ERP5Type/tests/CodingStyleTestCase.py
+11
-9
product/ERP5Type/tests/ERP5TypeLiveTestCase.py
product/ERP5Type/tests/ERP5TypeLiveTestCase.py
+5
-6
product/Formulator/Widget.py
product/Formulator/Widget.py
+1
-1
setup.py
setup.py
+1
-1
No files found.
CHANGES.erp5.util.txt
View file @
049fd1cb
Changes
Changes
=======
=======
0.4.76 (2024-05-07)
-------------------
* testnode:
- remove unused 'zip_binary' config
0.4.75 (2023-11-15)
0.4.75 (2023-11-15)
-------------------
-------------------
...
...
bt5/erp5_accounting/TestTemplateItem/portal_components/test.erp5.testAccounting.py
View file @
049fd1cb
...
@@ -4322,7 +4322,7 @@ class TestTransactions(AccountingTestCase):
...
@@ -4322,7 +4322,7 @@ class TestTransactions(AccountingTestCase):
self
.
assertFalse
(
invoice_line
.
getGroupingReference
())
self
.
assertFalse
(
invoice_line
.
getGroupingReference
())
self
.
assertFalse
(
payment_line
.
getGroupingReference
())
self
.
assertFalse
(
payment_line
.
getGroupingReference
())
def
test_roundDebitCredit_
raises
_if_big_difference
(
self
):
def
test_roundDebitCredit_
does_nothing
_if_big_difference
(
self
):
invoice
=
self
.
_makeOne
(
invoice
=
self
.
_makeOne
(
portal_type
=
'Sale Invoice Transaction'
,
portal_type
=
'Sale Invoice Transaction'
,
lines
=
(
dict
(
source_value
=
self
.
account_module
.
goods_sales
,
lines
=
(
dict
(
source_value
=
self
.
account_module
.
goods_sales
,
...
@@ -4330,7 +4330,11 @@ class TestTransactions(AccountingTestCase):
...
@@ -4330,7 +4330,11 @@ class TestTransactions(AccountingTestCase):
dict
(
source_value
=
self
.
account_module
.
receivable
,
dict
(
source_value
=
self
.
account_module
.
receivable
,
source_credit
=
100.000001
)))
source_credit
=
100.000001
)))
invoice
.
newContent
(
portal_type
=
'Invoice Line'
,
quantity
=
1
,
price
=
100
)
invoice
.
newContent
(
portal_type
=
'Invoice Line'
,
quantity
=
1
,
price
=
100
)
self
.
assertRaises
(
invoice
.
AccountingTransaction_roundDebitCredit
)
self
.
assertEqual
(
sorted
([
m
.
getQuantity
()
for
m
in
invoice
.
getMovementList
(
portal_type
=
'Sale Invoice Transaction Line'
)]),
[
-
100.032345
,
100.000001
])
def
test_roundDebitCredit_when_payable_is_different_total_price
(
self
):
def
test_roundDebitCredit_when_payable_is_different_total_price
(
self
):
invoice
=
self
.
_makeOne
(
invoice
=
self
.
_makeOne
(
...
...
bt5/erp5_administration/ExtensionTemplateItem/portal_components/extension.erp5.ERP5Administration.py
View file @
049fd1cb
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
import
sys
import
sys
from
AccessControl
import
getSecurityManager
from
ZODB.POSException
import
ConflictError
from
ZODB.POSException
import
ConflictError
from
zExceptions
import
ExceptionFormatter
from
zExceptions
import
ExceptionFormatter
,
Unauthorized
from
Products.CMFActivity.ActiveResult
import
ActiveResult
from
Products.CMFActivity.ActiveResult
import
ActiveResult
from
zLOG
import
LOG
,
INFO
from
zLOG
import
LOG
,
INFO
...
@@ -20,6 +21,8 @@ def dumpWorkflowChain(self, ignore_default=False,
...
@@ -20,6 +21,8 @@ def dumpWorkflowChain(self, ignore_default=False,
# - keep_order : set to True if you would like to keep original order,
# - keep_order : set to True if you would like to keep original order,
# default is to sort alphabetically
# default is to sort alphabetically
# - batch_mode : used to directly return the sctructure instead of return string
# - batch_mode : used to directly return the sctructure instead of return string
if
not
getSecurityManager
().
getUser
().
has_role
(
'Manager'
):
raise
Unauthorized
if
ignore_id_set
is
None
:
if
ignore_id_set
is
None
:
ignore_id_set
=
set
()
ignore_id_set
=
set
()
workflow_tool
=
self
.
getPortalObject
().
portal_workflow
workflow_tool
=
self
.
getPortalObject
().
portal_workflow
...
...
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/Base_viewSecurity.py
View file @
049fd1cb
from
AccessControl
import
getSecurityManager
from
AccessControl
import
getSecurityManager
from
zExceptions
import
Unauthorized
from
pprint
import
pformat
from
pprint
import
pformat
u
=
getSecurityManager
().
getUser
()
u
=
getSecurityManager
().
getUser
()
...
@@ -37,14 +36,4 @@ except AttributeError:
...
@@ -37,14 +36,4 @@ except AttributeError:
print
()
print
()
print
(
'Local roles on document:
\
n
'
,
pformat
(
context
.
get_local_roles
()))
print
(
'Local roles on document:
\
n
'
,
pformat
(
context
.
get_local_roles
()))
print
(
'''
----------------
Security mapping
----------------'''
)
if
u
.
getId
()
is
not
None
:
try
:
print
(
context
.
Base_viewSecurityMappingAsUser
(
u
.
getId
()))
except
Unauthorized
:
print
(
"user doesn't have permission to security mapping in this context"
)
return
printed
return
printed
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/Base_viewSecurityMappingAsUser.py
deleted
100644 → 0
View file @
5e36aa7e
group_id_list_generator
=
getattr
(
context
,
'ERP5Type_asSecurityGroupId'
)
security_category_dict
=
{}
# XXX This is a duplicate of logic present deep inside ERP5GroupManager.getGroupsForPrincipal()
# Please refactor into an accessible method so this code can be removed
def
getDefaultSecurityCategoryMapping
():
return
((
'ERP5Type_getSecurityCategoryFromAssignment'
,
context
.
getPortalObject
().
getPortalAssignmentBaseCategoryList
()
),)
getSecurityCategoryMapping
=
getattr
(
context
,
'ERP5Type_getSecurityCategoryMapping'
,
getDefaultSecurityCategoryMapping
)
# XXX end of code duplication
for
method_id
,
base_category_list
in
getSecurityCategoryMapping
():
try
:
security_category_dict
.
setdefault
(
tuple
(
base_category_list
),
[]).
extend
(
getattr
(
context
,
method_id
)(
base_category_list
,
login
,
context
,
''
))
except
Exception
:
# XXX: it is not possible to log message with traceback from python script
print
(
'It was not possible to invoke method %s with base_category_list %s'
%
(
method_id
,
base_category_list
))
for
base_category_list
,
category_value_list
in
security_category_dict
.
items
():
print
(
'base_category_list: %s'
%
(
base_category_list
,))
for
category_dict
in
category_value_list
:
print
(
'-> category_dict: %s'
%
category_dict
)
print
(
'--> %s'
%
group_id_list_generator
(
category_order
=
base_category_list
,
**
category_dict
))
return
printed
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/Base_viewSecurityMappingAsUser.xml
deleted
100644 → 0
View file @
5e36aa7e
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"PythonScript"
module=
"Products.PythonScripts.PythonScript"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_bind_names
</string>
</key>
<value>
<object>
<klass>
<global
name=
"_reconstructor"
module=
"copy_reg"
/>
</klass>
<tuple>
<global
name=
"NameAssignments"
module=
"Shared.DC.Scripts.Bindings"
/>
<global
name=
"object"
module=
"__builtin__"
/>
<none/>
</tuple>
<state>
<dictionary>
<item>
<key>
<string>
_asgns
</string>
</key>
<value>
<dictionary>
<item>
<key>
<string>
name_container
</string>
</key>
<value>
<string>
container
</string>
</value>
</item>
<item>
<key>
<string>
name_context
</string>
</key>
<value>
<string>
context
</string>
</value>
</item>
<item>
<key>
<string>
name_m_self
</string>
</key>
<value>
<string>
script
</string>
</value>
</item>
<item>
<key>
<string>
name_subpath
</string>
</key>
<value>
<string>
traverse_subpath
</string>
</value>
</item>
</dictionary>
</value>
</item>
</dictionary>
</state>
</object>
</value>
</item>
<item>
<key>
<string>
_params
</string>
</key>
<value>
<string>
login
</string>
</value>
</item>
<item>
<key>
<string>
_proxy_roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
<string>
Member
</string>
</tuple>
</value>
</item>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
Base_viewSecurityMappingAsUser
</string>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpAlarmToolConfiguration.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpAlarmToolConfiguration
</string>
</value>
<value>
<string>
ERP5Site_dumpAlarmToolConfiguration
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpBuilderList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpBuilderList
</string>
</value>
<value>
<string>
ERP5Site_dumpBuilderList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpInstalledBusinessTemplateList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string>
ignore_business_template_list=None
</string>
</value>
<value>
<string>
ignore_business_template_list=None
</string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpInstalledBusinessTemplateList
</string>
</value>
<value>
<string>
ERP5Site_dumpInstalledBusinessTemplateList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpOrderBuilderList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpOrderBuilderList
</string>
</value>
<value>
<string>
ERP5Site_dumpOrderBuilderList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalSkinsContent.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string>
ignore_custom=True, include_workflow_scripts=True, ignore_folder_list=None, ignore_skin_list=None
</string>
</value>
<value>
<string>
ignore_custom=True, include_workflow_scripts=True, ignore_folder_list=None, ignore_skin_list=None
</string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpPortalSkinsContent
</string>
</value>
<value>
<string>
ERP5Site_dumpPortalSkinsContent
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalTypeActionList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpPortalTypeActionList
</string>
</value>
<value>
<string>
ERP5Site_dumpPortalTypeActionList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalTypeList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpPortalTypeList
</string>
</value>
<value>
<string>
ERP5Site_dumpPortalTypeList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPortalTypeRoleList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpPortalTypeRoleList
</string>
</value>
<value>
<string>
ERP5Site_dumpPortalTypeRoleList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpPropertySheetList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string>
ignore_property_sheet_list=None
</string>
</value>
<value>
<string>
ignore_property_sheet_list=None
</string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpPropertySheetList
</string>
</value>
<value>
<string>
ERP5Site_dumpPropertySheetList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpRuleTesterList.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string></string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpRuleTesterList
</string>
</value>
<value>
<string>
ERP5Site_dumpRuleTesterList
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_administration/SkinTemplateItem/portal_skins/erp5_administration/ERP5Site_dumpSkinProperty.xml
View file @
049fd1cb
...
@@ -52,6 +52,12 @@
...
@@ -52,6 +52,12 @@
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string>
ignore_skin_folder_list=None
</string>
</value>
<value>
<string>
ignore_skin_folder_list=None
</string>
</value>
</item>
</item>
<item>
<key>
<string>
guard
</string>
</key>
<value>
<persistent>
<string
encoding=
"base64"
>
AAAAAAAAAAI=
</string>
</persistent>
</value>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Site_dumpSkinProperty
</string>
</value>
<value>
<string>
ERP5Site_dumpSkinProperty
</string>
</value>
...
@@ -59,4 +65,21 @@
...
@@ -59,4 +65,21 @@
</dictionary>
</dictionary>
</pickle>
</pickle>
</record>
</record>
<record
id=
"2"
aka=
"AAAAAAAAAAI="
>
<pickle>
<global
name=
"Guard"
module=
"Products.DCWorkflow.Guard"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
roles
</string>
</key>
<value>
<tuple>
<string>
Manager
</string>
</tuple>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
</ZopeData>
bt5/erp5_base/DocumentTemplateItem/portal_components/document.erp5.Person.py
View file @
049fd1cb
...
@@ -95,7 +95,7 @@ class Person(EncryptedPasswordMixin, Node, LoginAccountProviderMixin, ERP5UserMi
...
@@ -95,7 +95,7 @@ class Person(EncryptedPasswordMixin, Node, LoginAccountProviderMixin, ERP5UserMi
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'getTitle'
)
'getTitle'
)
def
getTitle
(
self
,
**
kw
):
# pylint: disable=super-on-old-class
def
getTitle
(
self
,
**
kw
):
"""
"""
Returns the title if it exists or a combination of
Returns the title if it exists or a combination of
first name, middle name and last name
first name, middle name and last name
...
@@ -109,7 +109,7 @@ class Person(EncryptedPasswordMixin, Node, LoginAccountProviderMixin, ERP5UserMi
...
@@ -109,7 +109,7 @@ class Person(EncryptedPasswordMixin, Node, LoginAccountProviderMixin, ERP5UserMi
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'getTranslatedTitle'
)
'getTranslatedTitle'
)
def
getTranslatedTitle
(
self
,
**
kw
):
# pylint: disable=super-on-old-class
def
getTranslatedTitle
(
self
,
**
kw
):
"""
"""
Returns the title if it exists or a combination of
Returns the title if it exists or a combination of
first name, middle name and last name
first name, middle name and last name
...
...
bt5/erp5_base/DocumentTemplateItem/portal_components/document.erp5.RoleDefinition.py
View file @
049fd1cb
...
@@ -29,8 +29,10 @@ import zope.interface
...
@@ -29,8 +29,10 @@ import zope.interface
from
AccessControl
import
ClassSecurityInfo
,
Unauthorized
from
AccessControl
import
ClassSecurityInfo
,
Unauthorized
from
Products.ERP5Type
import
Permissions
,
PropertySheet
,
interfaces
from
Products.ERP5Type
import
Permissions
,
PropertySheet
,
interfaces
from
Products.ERP5Type.XMLObject
import
XMLObject
from
Products.ERP5Type.XMLObject
import
XMLObject
from
Products.ERP5Type.ERP5Type
\
from
Products.ERP5Type.ERP5Type
import
(
import
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
,
)
@
zope
.
interface
.
implementer
(
interfaces
.
ILocalRoleGenerator
)
@
zope
.
interface
.
implementer
(
interfaces
.
ILocalRoleGenerator
)
class
RoleDefinition
(
XMLObject
):
class
RoleDefinition
(
XMLObject
):
...
@@ -60,8 +62,21 @@ class RoleDefinition(XMLObject):
...
@@ -60,8 +62,21 @@ class RoleDefinition(XMLObject):
security
.
declarePrivate
(
"getLocalRolesFor"
)
security
.
declarePrivate
(
"getLocalRolesFor"
)
def
getLocalRolesFor
(
self
,
ob
,
user_name
=
None
):
def
getLocalRolesFor
(
self
,
ob
,
user_name
=
None
):
group_id_generator
=
getattr
(
ob
,
group_id_generator
=
getattr
(
ob
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
)
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
,
None
)
if
group_id_generator
is
None
:
group_id_list
=
getattr
(
ob
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
,
)(
category_dict
=
{
'agent'
:
[(
x
,
False
)
for
x
in
self
.
getAgentValueList
()],
},
)
else
:
# BBB
group_id_list
=
group_id_generator
(
category_order
=
(
'agent'
,),
agent
=
self
.
getAgentList
(),
)
role_list
=
self
.
getRoleName
(),
role_list
=
self
.
getRoleName
(),
return
{
group_id
:
role_list
return
{
group_id
:
role_list
for
group_id
in
group_id_generator
(
category_order
=
(
'agent'
,),
for
group_id
in
group_id_list
}
agent
=
self
.
getAgentList
())}
bt5/erp5_corporate_identity/bt/template_action_path_list
View file @
049fd1cb
...
@@ -10,5 +10,4 @@ Web Page | print_leaflet
...
@@ -10,5 +10,4 @@ Web Page | print_leaflet
Web Page | print_letter
Web Page | print_letter
Web Page | print_release
Web Page | print_release
Web Page | print_slideshow
Web Page | print_slideshow
Web Page | render
Web Page | view_slideshow
Web Page | view_slideshow
\ No newline at end of file
bt5/erp5_simulation/DocumentTemplateItem/portal_components/document.erp5.SpecialiseEquivalenceTester.py
View file @
049fd1cb
...
@@ -43,7 +43,7 @@ class SpecialiseEquivalenceTester(CategoryMembershipEquivalenceTester):
...
@@ -43,7 +43,7 @@ class SpecialiseEquivalenceTester(CategoryMembershipEquivalenceTester):
movement_specialise_type
=
()
movement_specialise_type
=
()
movement_exclude_specialise_type
=
()
movement_exclude_specialise_type
=
()
def
_compare
(
self
,
prevision_movement
,
decision_movement
,
sorted
=
lambda
x
:
x
):
# pylint: disable=
super-on-old-class,
redefined-builtin
def
_compare
(
self
,
prevision_movement
,
decision_movement
,
sorted
=
lambda
x
:
x
):
# pylint: disable=redefined-builtin
return
super
(
SpecialiseEquivalenceTester
,
self
).
_compare
(
return
super
(
SpecialiseEquivalenceTester
,
self
).
_compare
(
prevision_movement
,
decision_movement
,
sorted
)
prevision_movement
,
decision_movement
,
sorted
)
...
...
bt5/erp5_test_result/DocumentTemplateItem/portal_components/document.erp5.CloudPerformanceUnitTestDistributor.py
View file @
049fd1cb
...
@@ -67,7 +67,7 @@ class CloudPerformanceUnitTestDistributor(ERP5ProjectUnitTestDistributor):
...
@@ -67,7 +67,7 @@ class CloudPerformanceUnitTestDistributor(ERP5ProjectUnitTestDistributor):
test_node
.
setAggregateList
(
test_suite_list
)
test_node
.
setAggregateList
(
test_suite_list
)
security
.
declarePublic
(
"startTestSuite"
)
security
.
declarePublic
(
"startTestSuite"
)
def
startTestSuite
(
self
,
title
,
computer_guid
=
None
,
**
kw
):
# pylint: disable=super-on-old-class
def
startTestSuite
(
self
,
title
,
computer_guid
=
None
,
**
kw
):
"""
"""
give the list of test suite to start. We will take all test suites
give the list of test suite to start. We will take all test suites
associated to the testnode. Then we add the test node title to the
associated to the testnode. Then we add the test node title to the
...
@@ -82,7 +82,7 @@ class CloudPerformanceUnitTestDistributor(ERP5ProjectUnitTestDistributor):
...
@@ -82,7 +82,7 @@ class CloudPerformanceUnitTestDistributor(ERP5ProjectUnitTestDistributor):
return
json
.
dumps
(
config_list
)
return
json
.
dumps
(
config_list
)
security
.
declarePublic
(
"generateConfiguration"
)
security
.
declarePublic
(
"generateConfiguration"
)
def
generateConfiguration
(
self
,
test_suite_title
,
batch_mode
=
0
):
# pylint: disable=super-on-old-class
def
generateConfiguration
(
self
,
test_suite_title
,
batch_mode
=
0
):
"""
"""
return the list of configuration to create instances, in the case of ERP5 unit tests,
return the list of configuration to create instances, in the case of ERP5 unit tests,
we will have only one configuration (unlike scalability tests). But for API consistency,
we will have only one configuration (unlike scalability tests). But for API consistency,
...
@@ -94,6 +94,6 @@ class CloudPerformanceUnitTestDistributor(ERP5ProjectUnitTestDistributor):
...
@@ -94,6 +94,6 @@ class CloudPerformanceUnitTestDistributor(ERP5ProjectUnitTestDistributor):
return
super
(
CloudPerformanceUnitTestDistributor
,
self
)
\
return
super
(
CloudPerformanceUnitTestDistributor
,
self
)
\
.
generateConfiguration
(
"ERP5-Cloud-Reliability"
,
batch_mode
)
.
generateConfiguration
(
"ERP5-Cloud-Reliability"
,
batch_mode
)
def
_getTestSuiteFromTitle
(
self
,
suite_title
):
# pylint: disable=super-on-old-class
def
_getTestSuiteFromTitle
(
self
,
suite_title
):
return
super
(
CloudPerformanceUnitTestDistributor
,
return
super
(
CloudPerformanceUnitTestDistributor
,
self
).
_getTestSuiteFromTitle
(
suite_title
.
split
(
"|"
)[
0
])
self
).
_getTestSuiteFromTitle
(
suite_title
.
split
(
"|"
)[
0
])
bt5/erp5_ui_test_core/ExtensionTemplateItem/portal_components/extension.erp5.ERP5Zuite.py
View file @
049fd1cb
...
@@ -13,7 +13,8 @@ def waitForActivities(self, delay=100, count=None):
...
@@ -13,7 +13,8 @@ def waitForActivities(self, delay=100, count=None):
activity_tool
=
self
.
getPortalObject
().
portal_activities
activity_tool
=
self
.
getPortalObject
().
portal_activities
assert
not
(
assert
not
(
activity_tool
.
isSubscribed
()
activity_tool
.
isSubscribed
()
and
getCurrentNode
()
in
activity_tool
.
getProcessingNodeList
())
and
getCurrentNode
()
in
activity_tool
.
getProcessingNodeList
()),
\
'still subscribed to activities'
if
count
is
not
None
:
# BBB
if
count
is
not
None
:
# BBB
# completely arbitrary conversion factor: count used to default to 1000
# completely arbitrary conversion factor: count used to default to 1000
...
...
bt5/erp5_
corporate_identity
/ActionTemplateItem/portal_types/Web%20Page/render.xml
→
bt5/erp5_
web
/ActionTemplateItem/portal_types/Web%20Page/render.xml
View file @
049fd1cb
File moved
bt5/erp5_
corporate_identity/SkinTemplateItem/portal_skins/erp5_corporate_identity
/WebPage_viewPreview.xml
→
bt5/erp5_
web/SkinTemplateItem/portal_skins/erp5_web
/WebPage_viewPreview.xml
View file @
049fd1cb
...
@@ -98,7 +98,7 @@
...
@@ -98,7 +98,7 @@
</item>
</item>
<item>
<item>
<key>
<string>
title
</string>
</key>
<key>
<string>
title
</string>
</key>
<value>
<string></string>
</value>
<value>
<string>
Preview
</string>
</value>
</item>
</item>
<item>
<item>
<key>
<string>
unicode_mode
</string>
</key>
<key>
<string>
unicode_mode
</string>
</key>
...
...
bt5/erp5_
corporate_identity/SkinTemplateItem/portal_skins/erp5_corporate_identity
/WebPage_viewPreview/iframe_content.xml
→
bt5/erp5_
web/SkinTemplateItem/portal_skins/erp5_web
/WebPage_viewPreview/iframe_content.xml
View file @
049fd1cb
File moved
bt5/erp5_
corporate_identity/SkinTemplateItem/portal_skins/erp5_corporate_identity
/WebPage_viewPreview/text_content.xml
→
bt5/erp5_
web/SkinTemplateItem/portal_skins/erp5_web
/WebPage_viewPreview/text_content.xml
View file @
049fd1cb
File moved
bt5/erp5_web/bt/template_action_path_list
View file @
049fd1cb
...
@@ -14,6 +14,7 @@ Static Web Site | view
...
@@ -14,6 +14,7 @@ Static Web Site | view
Web Page Module | view
Web Page Module | view
Web Page | list
Web Page | list
Web Page | local_permission
Web Page | local_permission
Web Page | render
Web Page | view
Web Page | view
Web Page | view_editor
Web Page | view_editor
Web Page | web_view
Web Page | web_view
...
...
bt5/erp5_web_renderjs_ui_test/PathTemplateItem/portal_tests/renderjs_ui_relation_field_zuite/testMultiRelationFieldEmptySearchWithListbox.zpt
View file @
049fd1cb
...
@@ -248,21 +248,41 @@
...
@@ -248,21 +248,41 @@
<tr>
<tr>
<td
colspan=
"3"
><b>
Check multi relation field display
</b></td>
<td
colspan=
"3"
><b>
Check multi relation field display
</b></td>
</tr>
</tr>
<tr>
<td>
waitForElementPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[3]//input
</td>
<td></td>
</tr>
<tr>
<td>
assertValue
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[3]//input
</td>
<td></td>
</tr>
<tr>
<tr>
<td>
assertValue
</td>
<td>
assertValue
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[1]//input
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[1]//input
</td>
<td>
A New Foo
</td>
<td>
A New Foo
</td>
</tr>
</tr>
<tr>
<tr>
<td>
assertValue
</td>
<td>
waitForElementPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//
input
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//
a
</td>
<td>
2
</td>
<td></td>
</tr>
</tr>
<tr>
<tr>
<td>
assert
Value
</td>
<td>
assert
ElementPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[
3]//input
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[
2]//a
</td>
<td></td>
<td></td>
</tr>
</tr>
<tr>
<td>
pause
</td>
<td>
100
</td>
<td></td>
</tr>
<tr>
<td>
assertValue
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//input
</td>
<td>
2
</td>
</tr>
<tr>
<tr>
<td>
assertElementNotPresent
</td>
<td>
assertElementNotPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[4]
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[4]
</td>
...
...
bt5/erp5_web_renderjs_ui_test/PathTemplateItem/portal_tests/renderjs_ui_relation_field_zuite/testMultiRelationFieldSearchWithListbox.zpt
View file @
049fd1cb
...
@@ -249,28 +249,43 @@
...
@@ -249,28 +249,43 @@
<td></td>
<td></td>
</tr>
</tr>
<tr>
<td>
waitForElementPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[3]//input
</td>
<td></td>
</tr>
<tr>
<td>
assertValue
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[3]//input
</td>
<td></td>
</tr>
<tr>
<tr>
<td>
assertValue
</td>
<td>
assertValue
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[1]//input
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[1]//input
</td>
<td>
A New Foo
</td>
<td>
A New Foo
</td>
</tr>
</tr>
<tr>
<tr>
<td>
assertValue
</td>
<td>
waitForElementPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//
input
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//
a
</td>
<td>
2
</td>
<td></td>
</tr>
</tr>
<tr>
<tr>
<td>
assertElementPresent
</td>
<td>
assertElementPresent
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[3]
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//a
</td>
<td></td>
</tr>
<tr>
<td>
pause
</td>
<td>
100
</td>
<td></td>
<td></td>
</tr>
</tr>
<tr>
<td>
assertValue
</td>
<td>
//div[@data-gadget-scope='field_my_bar_category_title_list']//fieldset[2]//input
</td>
<td>
2
</td>
</tr>
<tal:block
metal:use-macro=
"here/Zuite_CommonTemplateForRenderjsUi/macros/save"
/>
<tal:block
metal:use-macro=
"here/Zuite_CommonTemplateForRenderjsUi/macros/save"
/>
...
...
erp5/util/testnode/__init__.py
View file @
049fd1cb
...
@@ -71,7 +71,7 @@ def main(*args):
...
@@ -71,7 +71,7 @@ def main(*args):
for
key
in
(
'slapos_directory'
,
'working_directory'
,
'test_suite_directory'
,
for
key
in
(
'slapos_directory'
,
'working_directory'
,
'test_suite_directory'
,
'log_directory'
,
'run_directory'
,
'srv_directory'
,
'proxy_host'
,
'log_directory'
,
'run_directory'
,
'srv_directory'
,
'proxy_host'
,
'software_directory'
,
'software_directory'
,
'proxy_port'
,
'git_binary'
,
'
zip_binary'
,
'
node_quantity'
,
'proxy_port'
,
'git_binary'
,
'node_quantity'
,
'test_node_title'
,
'ipv4_address'
,
'ipv6_address'
,
'test_suite_master_url'
,
'test_node_title'
,
'ipv4_address'
,
'ipv6_address'
,
'test_suite_master_url'
,
'slapos_binary'
,
'httpd_ip'
,
'httpd_port'
,
'httpd_software_access_port'
,
'slapos_binary'
,
'httpd_ip'
,
'httpd_port'
,
'httpd_software_access_port'
,
'computer_id'
,
'server_url'
,
'shared_part_list'
,
'keep_log_days'
,
'computer_id'
,
'server_url'
,
'shared_part_list'
,
'keep_log_days'
,
...
...
product/ERP5/ERP5Site.py
View file @
049fd1cb
...
@@ -1614,27 +1614,12 @@ class ERP5Site(ResponseHeaderGenerator, FolderMixIn, PortalObjectBase, CacheCook
...
@@ -1614,27 +1614,12 @@ class ERP5Site(ResponseHeaderGenerator, FolderMixIn, PortalObjectBase, CacheCook
def
getPortalSecurityCategoryMapping
(
self
):
def
getPortalSecurityCategoryMapping
(
self
):
"""
"""
Returns a list of pairs composed of a script id and a list of base
DEPRECATED: implement ERP5User_getUserSecurityCategoryValueList instead.
category ids to use for computing security groups.
This is used during indexation, so involved scripts must not rely on
catalog at any point in their execution.
Example:
(
('script_1', ['base_category_1', 'base_category_2', ...]),
('script_2', ['base_category_1', 'base_category_3', ...])
)
"""
"""
return
getattr
(
return
getattr
(
self
,
self
,
'ERP5Type_getSecurityCategoryMapping'
,
'ERP5Type_getSecurityCategoryMapping'
,
lambda
:
(
# BBB
lambda
:
(),
(
'ERP5Type_getSecurityCategoryFromAssignment'
,
self
.
getPortalAssignmentBaseCategoryList
(),
),
),
)()
)()
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
...
...
product/ERP5/bootstrap/erp5_core/ExtensionTemplateItem/portal_components/extension.erp5.PythonCodeUtils.py
View file @
049fd1cb
from
six
import
string_types
as
basestring
from
six
import
string_types
as
basestring
import
re
import
re
import
json
import
json
import
sys
from
zExceptions
import
ExceptionFormatter
from
Products.ERP5Type.Utils
import
checkPythonSourceCode
from
Products.ERP5Type.Utils
import
checkPythonSourceCode
...
@@ -47,7 +49,15 @@ def checkPythonSourceCodeAsJSON(self, data, REQUEST=None):
...
@@ -47,7 +49,15 @@ def checkPythonSourceCodeAsJSON(self, data, REQUEST=None):
else
:
else
:
body
=
data
[
'code'
]
body
=
data
[
'code'
]
message_list
=
checkPythonSourceCode
(
body
.
encode
(
'utf8'
),
data
.
get
(
'portal_type'
))
try
:
message_list
=
checkPythonSourceCode
(
body
.
encode
(
'utf8'
),
data
.
get
(
'portal_type'
))
except
Exception
:
message_list
=
[{
'type'
:
'E'
,
'row'
:
0
,
'column'
:
0
,
'text'
:
'pylint failed:
\
n
%s'
%
''
.
join
(
ExceptionFormatter
.
format_exception
(
*
sys
.
exc_info
())),
}]
for
message_dict
in
message_list
:
for
message_dict
in
message_list
:
if
is_script
:
if
is_script
:
message_dict
[
'row'
]
=
message_dict
[
'row'
]
-
4
message_dict
[
'row'
]
=
message_dict
[
'row'
]
-
4
...
...
product/ERP5/bootstrap/erp5_core/ExtensionTemplateItem/portal_components/extension.erp5.StandardSecurity.py
View file @
049fd1cb
...
@@ -24,8 +24,177 @@
...
@@ -24,8 +24,177 @@
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
#
##############################################################################
##############################################################################
from
collections
import
defaultdict
from
itertools
import
product
import
six
from
DateTime
import
DateTime
from
DateTime
import
DateTime
_STOP_RECURSION_PORTAL_TYPE_SET
=
(
'Base Category'
,
'ERP5 Site'
)
def
getSecurityCategoryValueFromAssignment
(
self
,
rule_dict
):
"""
This function returns a list of dictionaries which represent
the security groups the user represented by self is member of,
based on the applicable Assignment objects it contains.
rule_dict (dict)
Keys: tuples listings base category names set on the Assignment
and which represent security groups assigned to the user.
Values: tuples describing which combinations of the base categories
whose parent categories the user is also member of.
Example call to illustrate argument and return value structures:
getSecurityCategoryValueFromAssignment(
rule_dict={
('function', ): ((), ('function', )),
('group', ): ((), ),
('function', 'group'): ((), ('function', )),
}
)
[
{
'function': (
(<Category function/accountant/chief>, False),
(<Category function/accountant/chief>, True),
(<Category function/accountant>, True),
),
}
{
'group': ((<Category group/nexedi>, False), ),
},
{
'function': (
(<Category function/accountant/chief>, False),
(<Category function/accountant/chief>, True),
(<Category function/accountant>, True),
),
'group': ((<Category group/nexedi>, False), ),
},
]
"""
base_category_set
=
set
(
sum
((
tuple
(
x
)
for
x
in
rule_dict
),
()))
recursive_base_category_set
=
set
(
sum
((
sum
((
tuple
(
y
)
for
y
in
x
),
())
for
x
in
six
.
itervalues
(
rule_dict
)),
()))
category_value_set_dict
=
defaultdict
(
set
)
parent_category_value_dict
=
{}
assignment_membership_dict_list
=
[]
now
=
DateTime
()
for
assignment_value
in
self
.
objectValues
(
portal_type
=
'Assignment'
):
if
assignment_value
.
getValidationState
()
==
'open'
and
(
not
assignment_value
.
hasStartDate
()
or
assignment_value
.
getStartDate
()
<=
now
)
and
(
not
assignment_value
.
hasStopDate
()
or
assignment_value
.
getStopDate
()
>=
now
):
assignment_membership_dict
=
{}
for
base_category
in
base_category_set
:
category_value_list
=
assignment_value
.
getAcquiredValueList
(
base_category
)
if
category_value_list
:
assignment_membership_dict
[
base_category
]
=
tuple
(
set
(
category_value_list
))
category_value_set_dict
[
base_category
].
update
(
category_value_list
)
if
base_category
in
recursive_base_category_set
:
for
category_value
in
category_value_list
:
while
True
:
parent_category_value
=
category_value
.
getParentValue
()
if
(
parent_category_value
in
parent_category_value_dict
or
parent_category_value
.
getPortalType
()
in
_STOP_RECURSION_PORTAL_TYPE_SET
):
break
parent_category_value_dict
[
category_value
]
=
parent_category_value
category_value
=
parent_category_value
if
assignment_membership_dict
:
assignment_membership_dict_list
.
append
(
assignment_membership_dict
)
result
=
[]
for
base_category_list
,
recursion_list
in
six
.
iteritems
(
rule_dict
):
result_entry
=
set
()
for
assignment_membership_dict
in
assignment_membership_dict_list
:
assignment_category_list
=
[]
for
base_category
in
base_category_list
:
category_set
=
set
()
for
category_value
in
assignment_membership_dict
.
get
(
base_category
,
()):
for
recursion_base_category_set
in
recursion_list
:
if
base_category
in
recursion_base_category_set
:
while
True
:
category_set
.
add
((
category_value
,
True
))
try
:
category_value
=
parent_category_value_dict
[
category_value
]
except
KeyError
:
break
else
:
category_set
.
add
((
category_value
,
False
))
assignment_category_list
.
append
((
base_category
,
tuple
(
category_set
)))
if
assignment_category_list
:
result_entry
.
add
(
tuple
(
assignment_category_list
))
for
result_item
in
result_entry
:
result
.
append
(
dict
(
result_item
))
return
result
def
asSecurityGroupIdSet
(
category_dict
,
key_sort
=
sorted
):
"""
The script takes the following parameters:
category_dict (dict)
keys: base category names.
values: list of categories composing the security groups the document is member of.
key_sort ((dict) -> key vector)
Function receiving the value of category_dict and returning the list of keys to
use to construct security group names, in the order in which these group names
will be used.
May return keys which are not part of category_dict.
Defaults to a lexicographic sort of all keys.
The External Method pointing at this implementation should be overridden (and
called using skinSuper) in order to provide this argument with a custom value.
Example call:
context.ERP5Type_asSecurityGroupIdSet(
category_dict={
'site': [(<Category france/lille>, False)],
'group': [(<Category nexedi>, False)],
'function': [(<Category accounting/accountant>, True)],
}
)
This will generate a string like 'ACT*_NXD_LIL' where "LIL", "NXD" and "ACT" are
the codification of respecively "france/lille", "nexedi" and "accounting/accountant"
categories.
If the category points to a document portal type (ex. trade condition, project, etc.),
and if no codification property is defined for this type of object,
the security ID group is generated by considering the object reference or
the object ID.
ERP5Type_asSecurityGroupIdSet can also return a list of users whenever a
category points to a Person instance. This is useful to implement user based local
role assignments instead of abstract security based local roles.
"""
list_of_list
=
[]
user_list
=
[]
associative_list
=
[]
for
base_category_id
in
key_sort
(
category_dict
):
try
:
category_list
=
category_dict
[
base_category_id
]
except
KeyError
:
continue
for
category_value
,
is_child_category
in
category_list
:
if
category_value
.
getPortalType
()
==
'Person'
:
user_name
=
category_value
.
Person_getUserId
()
if
user_name
is
not
None
:
user_list
.
append
(
user_name
)
associative_list
=
[]
elif
not
user_list
:
associative_list
.
append
(
(
category_value
.
getProperty
(
'codification'
)
or
category_value
.
getProperty
(
'reference'
)
or
category_value
.
getId
()
)
+
(
'*'
if
is_child_category
else
''
),
)
if
associative_list
:
list_of_list
.
append
(
associative_list
)
associative_list
=
[]
if
user_list
:
return
user_list
return
[
'_'
.
join
(
x
)
for
x
in
product
(
*
list_of_list
)
if
x
]
def
getSecurityCategoryFromAssignment
(
def
getSecurityCategoryFromAssignment
(
self
,
self
,
base_category_list
,
base_category_list
,
...
@@ -35,6 +204,8 @@ def getSecurityCategoryFromAssignment(
...
@@ -35,6 +204,8 @@ def getSecurityCategoryFromAssignment(
child_category_list
=
None
child_category_list
=
None
):
):
"""
"""
DEPRECATED: use getSecurityCategoryValueFromAssignment for better performance.
This script returns a list of dictionaries which represent
This script returns a list of dictionaries which represent
the security groups which a person is member of. It extracts
the security groups which a person is member of. It extracts
the categories from the current user assignment.
the categories from the current user assignment.
...
...
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5Type_asSecurityGroupId.py
deleted
100644 → 0
View file @
5e36aa7e
"""
This script is used to convert a list of categories into an security
identifier (security ID). It is invoked by two classes in ERP5:
- ERP5Type.py to convert security definitions made of
multiple categories into security ID strings
- ERP5GroupManager.py to convert an assignment definition
into a single security ID string. It should be noted here
that ERP5GroupManager.py also tries to invoke ERP5Type_asSecurityGroupIdList
(DEPRECATED) in order associate a user to multiple security groups.
In this case ERP5Type_asSecurityGroupId is not invoked.
The script takes the following parameters:
category_order - list of base_categories we want to use to generate the group id
kw - keys should be base categories, values should be value
of corresponding relative urls (obtained by getBaseCategory())
Example call:
context.ERP5TypeSecurity_asGroupId(category_order=('site', 'group', 'function'),
site='france/lille', group='nexedi', function='accounting/accountant')
This will generate a string like 'LIL_NXD_ACT' where "LIL", "NXD" and "ACT" are
the codification of respecively "france/lille", "nexedi" and "accounting/accountant" categories
If the category points to a document portal type (ex. trade condition, project, etc.),
and if no codification property is defined for this type of object,
the security ID group is generated by considering the object reference or
the object ID.
ERP5Type_asSecurityGroupId can also return a list of users whenever a category points
to a Person instance. This is useful to implement user based local role assignments
instead of abstract security based local roles.
"""
portal
=
context
.
getPortalObject
()
getCategoryValue
=
portal
.
portal_categories
.
getCategoryValue
# sort the category list lexicographically
# this prevents us to choose the exact order we want,
# but also prevents some human mistake to break everything by creating site_function instead of function_site
if
category_order
not
in
(
None
,
''
):
category_order
=
list
(
category_order
)
category_order
.
sort
()
else
:
category_order
=
[]
# Prepare a cartesian product
from
Products.ERP5Type.Utils
import
cartesianProduct
list_of_list
=
[]
user_list
=
[]
for
base_category
in
category_order
:
# It is acceptable for a category not to be defined
try
:
category_list
=
kw
[
base_category
]
except
KeyError
:
continue
associative_list
=
[]
if
isinstance
(
category_list
,
str
):
category_list
=
[
category_list
]
for
category
in
category_list
:
if
category
[
-
1
]
==
'*'
:
category
=
category
[:
-
1
]
is_child_category
=
1
else
:
is_child_category
=
0
category_path
=
'%s/%s'
%
(
base_category
,
category
)
category_object
=
getCategoryValue
(
category_path
)
if
category_object
is
None
:
raise
RuntimeError
(
"Security definition error (category %r not found)"
%
(
category_path
,))
portal_type
=
category_object
.
getPortalType
()
if
portal_type
==
'Person'
:
# We define a person here
user_name
=
category_object
.
Person_getUserId
()
if
user_name
is
not
None
:
user_list
.
append
(
user_name
)
else
:
category_code
=
(
category_object
.
getProperty
(
'codification'
)
or
category_object
.
getProperty
(
'reference'
)
or
category_object
.
getId
())
if
is_child_category
:
category_code
+=
'*'
associative_list
.
append
(
category_code
)
# Prevent making a cartesian product with an empty set
if
associative_list
:
list_of_list
.
append
(
associative_list
)
# Return a list of users if any was defined
if
user_list
:
return
user_list
# Compute the cartesian product and return the codes
# return filter(lambda x: x, map(lambda x: '_'.join(x), cartesianProduct(list_of_list)))
return
[
'_'
.
join
(
x
)
for
x
in
cartesianProduct
(
list_of_list
)
if
x
]
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5Type_asSecurityGroupIdSet.xml
0 → 100644
View file @
049fd1cb
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"ExternalMethod"
module=
"Products.ExternalMethod.ExternalMethod"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_function
</string>
</key>
<value>
<string>
asSecurityGroupIdSet
</string>
</value>
</item>
<item>
<key>
<string>
_module
</string>
</key>
<value>
<string>
StandardSecurity
</string>
</value>
</item>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5Type_asSecurityGroupIdSet
</string>
</value>
</item>
<item>
<key>
<string>
title
</string>
</key>
<value>
<string></string>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5User_getSecurityCategoryValueFromAssignment.xml
0 → 100644
View file @
049fd1cb
<?xml version="1.0"?>
<ZopeData>
<record
id=
"1"
aka=
"AAAAAAAAAAE="
>
<pickle>
<global
name=
"ExternalMethod"
module=
"Products.ExternalMethod.ExternalMethod"
/>
</pickle>
<pickle>
<dictionary>
<item>
<key>
<string>
_function
</string>
</key>
<value>
<string>
getSecurityCategoryValueFromAssignment
</string>
</value>
</item>
<item>
<key>
<string>
_module
</string>
</key>
<value>
<string>
StandardSecurity
</string>
</value>
</item>
<item>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5User_getSecurityCategoryValueFromAssignment
</string>
</value>
</item>
<item>
<key>
<string>
title
</string>
</key>
<value>
<string></string>
</value>
</item>
</dictionary>
</pickle>
</record>
</ZopeData>
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5User_getUserSecurityCategoryValueList.py
0 → 100644
View file @
049fd1cb
"""
Override this script to customise user security group generation.
It is called on the context of the ERP5 document which represents the user.
This is called when a user object is being prepared by PAS, typically at the
end of traversal but also when calling getUser and getUserById API, in order
to list the security groups the user is member of.
When called by PAS, this script is called in a super-user security context.
"""
return
context
.
ERP5User_getSecurityCategoryValueFromAssignment
(
rule_dict
=
{
tuple
(
context
.
getPortalObject
().
getPortalAssignmentBaseCategoryList
()):
((),
)
},
)
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5
Type_asSecurityGroupId
.xml
→
product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/ERP5
User_getUserSecurityCategoryValueList
.xml
View file @
049fd1cb
...
@@ -50,11 +50,11 @@
...
@@ -50,11 +50,11 @@
</item>
</item>
<item>
<item>
<key>
<string>
_params
</string>
</key>
<key>
<string>
_params
</string>
</key>
<value>
<string>
category_order, **kw
</string>
</value>
<value>
<string></string>
</value>
</item>
</item>
<item>
<item>
<key>
<string>
id
</string>
</key>
<key>
<string>
id
</string>
</key>
<value>
<string>
ERP5
Type_asSecurityGroupId
</string>
</value>
<value>
<string>
ERP5
User_getUserSecurityCategoryValueList
</string>
</value>
</item>
</item>
</dictionary>
</dictionary>
</pickle>
</pickle>
...
...
product/ERP5Form/MultiRelationField.py
View file @
049fd1cb
...
@@ -367,6 +367,8 @@ class MultiRelationStringFieldWidget(Widget.LinesTextAreaWidget,
...
@@ -367,6 +367,8 @@ class MultiRelationStringFieldWidget(Widget.LinesTextAreaWidget,
)
)
return
html_string
return
html_string
render_pdf
=
Widget
.
LinesTextAreaWidget
.
render_pdf
def
render_autocomplete
(
self
,
field
,
key
):
def
render_autocomplete
(
self
,
field
,
key
):
"""
"""
Use jquery-ui autocompletion for all relation fields by default, requiring
Use jquery-ui autocompletion for all relation fields by default, requiring
...
...
product/ERP5Security/ERP5GroupManager.py
View file @
049fd1cb
...
@@ -18,6 +18,7 @@
...
@@ -18,6 +18,7 @@
from
collections
import
defaultdict
from
collections
import
defaultdict
from
contextlib
import
contextmanager
from
contextlib
import
contextmanager
from
threading
import
local
from
threading
import
local
import
warnings
from
Products.ERP5Type.Globals
import
InitializeClass
from
Products.ERP5Type.Globals
import
InitializeClass
from
AccessControl
import
ClassSecurityInfo
from
AccessControl
import
ClassSecurityInfo
from
Products.PageTemplates.PageTemplateFile
import
PageTemplateFile
from
Products.PageTemplates.PageTemplateFile
import
PageTemplateFile
...
@@ -25,8 +26,10 @@ from Products.PluggableAuthService.plugins.BasePlugin import BasePlugin
...
@@ -25,8 +26,10 @@ from Products.PluggableAuthService.plugins.BasePlugin import BasePlugin
from
Products.PluggableAuthService.utils
import
classImplements
from
Products.PluggableAuthService.utils
import
classImplements
from
Products.PluggableAuthService.interfaces.plugins
import
IGroupsPlugin
from
Products.PluggableAuthService.interfaces.plugins
import
IGroupsPlugin
from
Products.ERP5Type.Cache
import
CachingMethod
from
Products.ERP5Type.Cache
import
CachingMethod
from
Products.ERP5Type.ERP5Type
\
from
Products.ERP5Type.ERP5Type
import
(
import
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
,
)
from
Products.ERP5Type.UnrestrictedMethod
import
UnrestrictedMethod
from
Products.ERP5Type.UnrestrictedMethod
import
UnrestrictedMethod
from
Products.ZSQLCatalog.SQLCatalog
import
SimpleQuery
from
Products.ZSQLCatalog.SQLCatalog
import
SimpleQuery
from
ZODB.POSException
import
ConflictError
from
ZODB.POSException
import
ConflictError
...
@@ -115,60 +118,149 @@ class ERP5GroupManager(BasePlugin):
...
@@ -115,60 +118,149 @@ class ERP5GroupManager(BasePlugin):
user_value
=
getattr
(
principal
,
'getUserValue'
,
lambda
:
None
)()
user_value
=
getattr
(
principal
,
'getUserValue'
,
lambda
:
None
)()
if
user_value
is
None
:
if
user_value
is
None
:
return
()
return
()
security_category_dict
=
defaultdict
(
list
)
category_mapping
=
self
.
getPortalSecurityCategoryMapping
()
for
method_name
,
base_category_list
in
self
.
getPortalSecurityCategoryMapping
():
if
category_mapping
:
# BBB
base_category_list
=
tuple
(
base_category_list
)
warnings
.
warn
(
security_category_list
=
security_category_dict
[
base_category_list
]
'Consider migrating ERP5Type_getSecurityCategoryMapping to '
try
:
'ERP5User_getUserSecurityCategoryValueList to get better '
# The called script may want to distinguish if it is called
'performance'
,
# from here or from _updateLocalRolesOnSecurityGroups.
DeprecationWarning
,
# Currently, passing portal_type='' (instead of 'Person')
)
# is the only way to make the difference.
has_relative_urls
=
True
security_category_list
.
extend
(
security_category_dict
=
defaultdict
(
list
)
getattr
(
self
,
method_name
)(
for
method_name
,
base_category_list
in
category_mapping
:
base_category_list
,
base_category_list
=
tuple
(
base_category_list
)
user_id
,
security_category_list
=
security_category_dict
[
base_category_list
]
user_value
,
''
,
)
)
except
ConflictError
:
raise
except
Exception
:
LOG
(
'ERP5GroupManager'
,
WARNING
,
'could not get security categories from %s'
%
(
method_name
,
),
error
=
True
,
)
# Get group names from category values
# XXX try ERP5Type_asSecurityGroupIdList first for compatibility
generator_name
=
'ERP5Type_asSecurityGroupIdList'
group_id_list_generator
=
getattr
(
self
,
generator_name
,
None
)
if
group_id_list_generator
is
None
:
generator_name
=
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
group_id_list_generator
=
getattr
(
self
,
generator_name
,
None
)
security_group_list
=
[]
for
base_category_list
,
category_value_list
in
six
.
iteritems
(
security_category_dict
):
for
category_dict
in
category_value_list
:
try
:
try
:
group_id_list
=
group_id_list_generator
(
# The called script may want to distinguish if it is called
category_order
=
base_category_list
,
# from here or from _updateLocalRolesOnSecurityGroups.
**
category_dict
# Currently, passing portal_type='' (instead of 'Person')
# is the only way to make the difference.
security_category_list
.
extend
(
getattr
(
self
,
method_name
)(
base_category_list
,
user_id
,
user_value
,
''
,
)
)
)
if
isinstance
(
group_id_list
,
str
):
group_id_list
=
[
group_id_list
]
security_group_list
.
extend
(
group_id_list
)
except
ConflictError
:
except
ConflictError
:
raise
raise
except
Exception
:
except
Exception
:
LOG
(
LOG
(
'ERP5GroupManager'
,
'ERP5GroupManager'
,
WARNING
,
WARNING
,
'could not get security
groups from %s'
%
(
generator
_name
,
),
'could not get security
categories from %s'
%
(
method
_name
,
),
error
=
True
,
error
=
True
,
)
)
return
tuple
(
security_group_list
)
else
:
has_relative_urls
=
False
try
:
getUserSecurityCategoryValueList
=
user_value
.
ERP5User_getUserSecurityCategoryValueList
except
AttributeError
:
# BBB
security_category_value_dict_list
=
[]
else
:
security_category_value_dict_list
=
getUserSecurityCategoryValueList
()
security_group_set
=
set
()
# XXX try ERP5Type_asSecurityGroupIdList first for compatibility
generator_name
=
'ERP5Type_asSecurityGroupIdList'
group_id_list_generator
=
getattr
(
self
,
generator_name
,
None
)
if
group_id_list_generator
is
None
:
generator_name
=
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
group_id_list_generator
=
getattr
(
self
,
generator_name
,
None
)
if
group_id_list_generator
is
None
:
if
has_relative_urls
:
# Convert security_category_dict to security_category_value_dict_list
# Differences with direct security_category_value_dict_list production:
# - incomplete deduplication
# - extra intermediate sorting
getCategoryValue
=
self
.
portal_categories
.
getCategoryValue
security_category_value_dict_list
=
(
dict
(
x
)
for
x
in
{
tuple
(
sorted
(
(
(
base_category
,
tuple
(
(
getCategoryValue
(
base_category
+
'/'
+
relative_url
.
rstrip
(
'*'
)),
relative_url
.
endswith
(
'*'
),
)
for
relative_url
in
(
(
relative_url_list
,
)
if
isinstance
(
relative_url_list
,
str
)
else
sorted
(
relative_url_list
)
)
)
)
for
(
base_category
,
relative_url_list
,
)
in
six
.
iteritems
(
security_dict
)
),
# Avoid comparing persistent objects, for performance purposes:
# these are stored by path.
key
=
lambda
x
:
x
[
0
]
))
for
security_category_list
in
six
.
itervalues
(
security_category_dict
)
for
security_dict
in
security_category_list
}
)
for
security_category_value_dict
in
security_category_value_dict_list
:
security_group_set
.
update
(
getattr
(
self
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
)(
category_dict
=
security_category_value_dict
,
),
)
else
:
# BBB
warnings
.
warn
(
'Consider migrating %s to %s to get better performance'
%
(
generator_name
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
,
),
DeprecationWarning
,
)
if
not
has_relative_urls
:
# Convert security_category_value_dict_list to security_category_dict
# Differences with direct security_category_dict generation:
# - the order of items in the tuples used as keys is random
# - repetitions will be missing (which is arguably an improvement)
security_category_dict
=
{
tuple
(
security_category_value_dict
):
[
{
base_category
:
[
category_value
.
getRelativeUrl
()
+
(
'*'
if
parent
else
''
)
for
category_value
,
parent
in
category_value_list
]
for
base_category
,
category_value_list
in
six
.
iteritems
(
security_category_value_dict
,
)
}
]
for
security_category_value_dict
in
security_category_value_dict_list
}
# Get group names from category values
for
base_category_list
,
category_value_list
in
six
.
iteritems
(
security_category_dict
):
for
category_dict
in
category_value_list
:
try
:
group_id_list
=
group_id_list_generator
(
category_order
=
base_category_list
,
**
category_dict
)
if
isinstance
(
group_id_list
,
str
):
group_id_list
=
[
group_id_list
]
security_group_set
.
update
(
group_id_list
)
except
ConflictError
:
raise
except
Exception
:
LOG
(
'ERP5GroupManager'
,
WARNING
,
'could not get security groups from %s'
%
(
generator_name
,
),
error
=
True
,
)
return
tuple
(
security_group_set
)
if
not
NO_CACHE_MODE
and
getattr
(
_CACHE_ENABLED_LOCAL
,
'value'
,
True
):
if
not
NO_CACHE_MODE
and
getattr
(
_CACHE_ENABLED_LOCAL
,
'value'
,
True
):
_getGroupsForPrincipal
=
CachingMethod
(
_getGroupsForPrincipal
,
_getGroupsForPrincipal
=
CachingMethod
(
_getGroupsForPrincipal
,
...
...
product/ERP5Security/tests/testERP5Security.py
View file @
049fd1cb
...
@@ -189,23 +189,22 @@ class UserManagementTestCase(ERP5TypeTestCase):
...
@@ -189,23 +189,22 @@ class UserManagementTestCase(ERP5TypeTestCase):
return
dummy_document
return
dummy_document
class
RoleManagementTestCase
(
UserManagementTestCase
):
class
RoleManagementTestCase
Base
(
UserManagementTestCase
):
"""Test case with required configuration to test role definitions.
"""Test case with required configuration to test role definitions.
"""
"""
def
afterSetUp
(
self
):
def
afterSetUp
(
self
):
"""Initialize requirements of security configuration.
"""Initialize requirements of security configuration.
"""
"""
super
(
RoleManagementTestCase
,
self
).
afterSetUp
()
super
(
RoleManagementTestCase
Base
,
self
).
afterSetUp
()
# create a security configuration script
# create a security configuration script
skin_folder
=
self
.
portal
.
portal_skins
.
custom
skin_folder
=
self
.
portal
.
portal_skins
.
custom
if
'ERP5Type_getSecurityCategoryMapping'
not
in
skin_folder
.
objectIds
():
if
self
.
_security_configuration_script_id
not
in
skin_folder
.
objectIds
():
createZODBPythonScript
(
createZODBPythonScript
(
skin_folder
,
'ERP5Type_getSecurityCategoryMapping'
,
''
,
skin_folder
,
"""return ((
self
.
_security_configuration_script_id
,
'ERP5Type_getSecurityCategoryFromAssignment',
''
,
context.getPortalObject().getPortalAssignmentBaseCategoryList()
self
.
_security_configuration_script_body
,
),)
)
"""
)
# configure group, site, function categories
# configure group, site, function categories
category_tool
=
self
.
getCategoryTool
()
category_tool
=
self
.
getCategoryTool
()
for
bc
in
[
'group'
,
'site'
,
'function'
]:
for
bc
in
[
'group'
,
'site'
,
'function'
]:
...
@@ -241,6 +240,29 @@ class RoleManagementTestCase(UserManagementTestCase):
...
@@ -241,6 +240,29 @@ class RoleManagementTestCase(UserManagementTestCase):
self
.
tic
()
self
.
tic
()
class
RoleManagementTestCaseOld
(
RoleManagementTestCaseBase
):
"""
RoleManagementTestCaseBase variant using the deprecated security declaration API.
"""
_security_configuration_script_id
=
'ERP5Type_getSecurityCategoryMapping'
_security_configuration_script_body
=
"""return ((
'ERP5Type_getSecurityCategoryFromAssignment',
context.getPortalObject().getPortalAssignmentBaseCategoryList()
),)"""
class
RoleManagementTestCase
(
RoleManagementTestCaseBase
):
"""
RoleManagementTestCaseBase variant using the current security declaration API.
"""
_security_configuration_script_id
=
'ERP5User_getUserSecurityCategoryValueList'
_security_configuration_script_body
=
"""return context.ERP5User_getSecurityCategoryValueFromAssignment(
rule_dict={
tuple(context.getPortalObject().getPortalAssignmentBaseCategoryList()): ((), )
},
)"""
class
TestUserManagement
(
UserManagementTestCase
):
class
TestUserManagement
(
UserManagementTestCase
):
"""Tests User Management in ERP5Security.
"""Tests User Management in ERP5Security.
"""
"""
...
@@ -1184,7 +1206,7 @@ class TestUserManagementExternalAuthentication(TestUserManagement):
...
@@ -1184,7 +1206,7 @@ class TestUserManagementExternalAuthentication(TestUserManagement):
self
.
assertIn
(
login
,
response
.
getBody
())
self
.
assertIn
(
login
,
response
.
getBody
())
class
TestLocalRoleManagement
(
RoleManagementTestCase
):
class
_TestLocalRoleManagementMixIn
(
object
):
"""Tests Local Role Management with ERP5Security.
"""Tests Local Role Management with ERP5Security.
"""
"""
...
@@ -1194,24 +1216,29 @@ class TestLocalRoleManagement(RoleManagementTestCase):
...
@@ -1194,24 +1216,29 @@ class TestLocalRoleManagement(RoleManagementTestCase):
def
afterSetUp
(
self
):
def
afterSetUp
(
self
):
"""Called after setup completed.
"""Called after setup completed.
"""
"""
super
(
TestLocalRoleManagement
,
self
).
afterSetUp
()
super
(
_TestLocalRoleManagementMixIn
,
self
).
afterSetUp
()
# any member can add organisations
# any member can add organisations
self
.
portal
.
organisation_module
.
manage_permission
(
self
.
portal
.
organisation_module
.
manage_permission
(
'Add portal content'
,
roles
=
[
'Member'
,
'Manager'
],
acquire
=
1
)
'Add portal content'
,
roles
=
[
'Member'
,
'Manager'
],
acquire
=
1
)
self
.
username
=
'usérn@me'
self
.
username
=
'usérn@me'
# create a user and open an assignement
user_list
=
self
.
portal
.
acl_users
.
getUserById
(
self
.
username
)
pers
=
self
.
getPersonModule
().
newContent
(
portal_type
=
'Person'
,
if
not
user_list
:
user_id
=
self
.
username
)
# create a user and open an assignement
assignment
=
pers
.
newContent
(
portal_type
=
'Assignment'
,
pers
=
self
.
getPersonModule
().
newContent
(
portal_type
=
'Person'
,
group
=
'subcat'
,
user_id
=
self
.
username
)
site
=
'subcat'
,
assignment
=
pers
.
newContent
(
portal_type
=
'Assignment'
,
function
=
'subcat'
)
group
=
'subcat'
,
assignment
.
open
()
site
=
'subcat'
,
pers
.
newContent
(
portal_type
=
'ERP5 Login'
,
function
=
'subcat'
)
reference
=
self
.
username
,
assignment
.
open
()
password
=
self
.
username
).
validate
()
pers
.
newContent
(
portal_type
=
'ERP5 Login'
,
reference
=
self
.
username
,
password
=
self
.
username
).
validate
()
else
:
user
,
=
user_list
pers
=
user
.
getUserValue
()
self
.
person
=
pers
self
.
person
=
pers
self
.
tic
()
self
.
tic
()
...
@@ -1229,6 +1256,15 @@ class TestLocalRoleManagement(RoleManagementTestCase):
...
@@ -1229,6 +1256,15 @@ class TestLocalRoleManagement(RoleManagementTestCase):
def
_makeOne
(
self
):
def
_makeOne
(
self
):
return
self
.
getOrganisationModule
().
newContent
(
portal_type
=
'Organisation'
)
return
self
.
getOrganisationModule
().
newContent
(
portal_type
=
'Organisation'
)
def
_createOrGetObject
(
self
,
container
,
content_id
,
new_content_kw
):
try
:
return
container
[
content_id
]
except
KeyError
:
return
container
.
newContent
(
id
=
content_id
,
**
new_content_kw
)
def
getBusinessTemplateList
(
self
):
def
getBusinessTemplateList
(
self
):
"""List of BT to install. """
"""List of BT to install. """
return
(
'erp5_base'
,
'erp5_web'
,
'erp5_ingestion'
,
'erp5_dms'
,
'erp5_administration'
)
return
(
'erp5_base'
,
'erp5_web'
,
'erp5_ingestion'
,
'erp5_dms'
,
'erp5_administration'
)
...
@@ -1251,11 +1287,6 @@ class TestLocalRoleManagement(RoleManagementTestCase):
...
@@ -1251,11 +1287,6 @@ class TestLocalRoleManagement(RoleManagementTestCase):
def
testSimpleLocalRole
(
self
):
def
testSimpleLocalRole
(
self
):
"""Test simple case of setting a role.
"""Test simple case of setting a role.
"""
"""
def
viewSecurity
():
return
self
.
publish
(
self
.
portal
.
absolute_url_path
()
+
'/Base_viewSecurity'
,
basic
=
'%s:%s'
%
(
self
.
username
,
self
.
username
),
)
self
.
_getTypeInfo
().
newContent
(
portal_type
=
'Role Information'
,
self
.
_getTypeInfo
().
newContent
(
portal_type
=
'Role Information'
,
role_name
=
'Assignor'
,
role_name
=
'Assignor'
,
description
=
'desc.'
,
description
=
'desc.'
,
...
@@ -1269,35 +1300,37 @@ class TestLocalRoleManagement(RoleManagementTestCase):
...
@@ -1269,35 +1300,37 @@ class TestLocalRoleManagement(RoleManagementTestCase):
self
.
assertIn
(
'Assignor'
,
user
.
getRolesInContext
(
obj
))
self
.
assertIn
(
'Assignor'
,
user
.
getRolesInContext
(
obj
))
self
.
assertNotIn
(
'Assignee'
,
user
.
getRolesInContext
(
obj
))
self
.
assertNotIn
(
'Assignee'
,
user
.
getRolesInContext
(
obj
))
person_value
=
self
.
person
user_id
=
person_value
.
getUserId
()
getUserById
=
self
.
portal
.
acl_users
.
getUserById
def
assertRoleItemsEqual
(
expected_role_set
):
self
.
assertItemsEqual
(
getUserById
(
user_id
).
getGroups
(),
expected_role_set
)
# check if assignment change is effective immediately
# check if assignment change is effective immediately
assertRoleItemsEqual
([
'F1_G1_S1'
])
self
.
login
()
self
.
login
()
res
=
viewSecurity
()
self
.
assertEqual
([
x
for
x
in
res
.
body
.
splitlines
()
if
x
.
startswith
(
'-->'
)],
[
"--> ['F1_G1_S1']"
],
res
.
body
)
assignment
=
self
.
person
.
newContent
(
portal_type
=
'Assignment'
,
assignment
=
self
.
person
.
newContent
(
portal_type
=
'Assignment'
,
group
=
'subcat'
,
group
=
'subcat'
,
site
=
'subcat'
,
site
=
'subcat'
,
function
=
'another_subcat'
)
function
=
'another_subcat'
)
assignment
.
open
()
assignment
.
open
()
res
=
viewSecurity
()
assertRoleItemsEqual
([
'F1_G1_S1'
,
'F2_G1_S1'
])
self
.
assertEqual
([
x
for
x
in
res
.
body
.
splitlines
()
if
x
.
startswith
(
'-->'
)],
[
"--> ['F1_G1_S1']"
,
"--> ['F2_G1_S1']"
],
res
.
body
)
assignment
.
setGroup
(
'another_subcat'
)
assignment
.
setGroup
(
'another_subcat'
)
res
=
viewSecurity
()
assertRoleItemsEqual
([
'F1_G1_S1'
,
'F2_G1_S1'
])
self
.
assertEqual
([
x
for
x
in
res
.
body
.
splitlines
()
if
x
.
startswith
(
'-->'
)],
[
"--> ['F1_G1_S1']"
,
"--> ['F2_G2_S1']"
],
res
.
body
)
self
.
abort
()
self
.
abort
()
def
testLocalRolesGroupId
(
self
):
def
testLocalRolesGroupId
(
self
):
"""Assigning a role with local roles group id.
"""Assigning a role with local roles group id.
"""
"""
self
.
portal
.
portal_categories
.
local_role_group
.
newContent
(
portal_type
=
'Category'
,
reference
=
'Alternate'
,
id
=
'Alternate'
)
self
.
_getTypeInfo
().
newContent
(
portal_type
=
'Role Information'
,
self
.
_getTypeInfo
().
newContent
(
portal_type
=
'Role Information'
,
role_name
=
'Assignor'
,
role_name
=
'Assignor'
,
local_role_group_value
=
self
.
portal
.
portal_categories
.
local_role_group
.
Alternate
,
local_role_group_value
=
self
.
_createOrGetObject
(
container
=
self
.
portal
.
portal_categories
.
local_role_group
,
content_id
=
'Alternate'
,
new_content_kw
=
{
'portal_type'
:
'Category'
,
'reference'
:
'Alternate'
,
},
),
role_category
=
self
.
defined_category
)
role_category
=
self
.
defined_category
)
self
.
loginAsUser
(
self
.
username
)
self
.
loginAsUser
(
self
.
username
)
...
@@ -1433,11 +1466,19 @@ class TestLocalRoleManagement(RoleManagementTestCase):
...
@@ -1433,11 +1466,19 @@ class TestLocalRoleManagement(RoleManagementTestCase):
self
.
assertEqual
(
response
.
getStatus
(),
401
)
self
.
assertEqual
(
response
.
getStatus
(),
401
)
class
TestKeyAuthentication
(
RoleManagementTestCase
):
class
TestLocalRoleManagementOld
(
_TestLocalRoleManagementMixIn
,
RoleManagementTestCaseOld
):
pass
class
TestLocalRoleManagement
(
_TestLocalRoleManagementMixIn
,
RoleManagementTestCase
):
pass
class
_TestKeyAuthenticationMixIn
(
object
):
def
getBusinessTemplateList
(
self
):
def
getBusinessTemplateList
(
self
):
"""This test also uses web and dms
"""This test also uses web and dms
"""
"""
return
super
(
TestKeyAuthenticatio
n
,
self
).
getBusinessTemplateList
()
+
(
return
super
(
_TestKeyAuthenticationMixI
n
,
self
).
getBusinessTemplateList
()
+
(
'erp5_core_proxy_field_legacy'
,
# for erp5_web
'erp5_core_proxy_field_legacy'
,
# for erp5_web
'erp5_base'
,
'erp5_web'
,
'erp5_ingestion'
,
'erp5_dms'
,
'erp5_administration'
)
'erp5_base'
,
'erp5_web'
,
'erp5_ingestion'
,
'erp5_dms'
,
'erp5_administration'
)
...
@@ -1449,14 +1490,16 @@ class TestKeyAuthentication(RoleManagementTestCase):
...
@@ -1449,14 +1490,16 @@ class TestKeyAuthentication(RoleManagementTestCase):
# add key authentication PAS plugin
# add key authentication PAS plugin
portal
=
self
.
portal
portal
=
self
.
portal
uf
=
portal
.
acl_users
uf
=
portal
.
acl_users
uf
.
manage_addProduct
[
'ERP5Security'
].
addERP5KeyAuthPlugin
(
try
:
erp5_auth_key_plugin
=
getattr
(
uf
,
"erp5_auth_key"
)
except
AttributeError
:
uf
.
manage_addProduct
[
'ERP5Security'
].
addERP5KeyAuthPlugin
(
id
=
"erp5_auth_key"
,
\
id
=
"erp5_auth_key"
,
\
title
=
"ERP5 Auth key"
,
\
title
=
"ERP5 Auth key"
,
\
encryption_key
=
'fdgfhkfjhltylutyu'
,
encryption_key
=
'fdgfhkfjhltylutyu'
,
cookie_name
=
'__key'
,
\
cookie_name
=
'__key'
,
\
default_cookie_name
=
'__ac'
)
default_cookie_name
=
'__ac'
)
erp5_auth_key_plugin
=
getattr
(
uf
,
"erp5_auth_key"
)
erp5_auth_key_plugin
=
getattr
(
uf
,
"erp5_auth_key"
)
erp5_auth_key_plugin
.
manage_activateInterfaces
(
erp5_auth_key_plugin
.
manage_activateInterfaces
(
interfaces
=
[
'IExtractionPlugin'
,
interfaces
=
[
'IExtractionPlugin'
,
'IAuthenticationPlugin'
,
'IAuthenticationPlugin'
,
...
@@ -1546,6 +1589,14 @@ class TestKeyAuthentication(RoleManagementTestCase):
...
@@ -1546,6 +1589,14 @@ class TestKeyAuthentication(RoleManagementTestCase):
self
.
assertEqual
(
response
.
getStatus
(),
200
)
self
.
assertEqual
(
response
.
getStatus
(),
200
)
class
TestKeyAuthenticationOld
(
_TestKeyAuthenticationMixIn
,
RoleManagementTestCaseOld
):
pass
class
TestKeyAuthentication
(
_TestKeyAuthenticationMixIn
,
RoleManagementTestCase
):
pass
class
TestOwnerRole
(
UserManagementTestCase
):
class
TestOwnerRole
(
UserManagementTestCase
):
def
_createZodbUser
(
self
,
login
,
role_list
=
None
):
def
_createZodbUser
(
self
,
login
,
role_list
=
None
):
if
role_list
is
None
:
if
role_list
is
None
:
...
...
product/ERP5Type/Core/RoleInformation.py
View file @
049fd1cb
...
@@ -28,6 +28,7 @@
...
@@ -28,6 +28,7 @@
""" Information about customizable roles.
""" Information about customizable roles.
"""
"""
from
collections
import
defaultdict
from
six
import
string_types
as
basestring
from
six
import
string_types
as
basestring
import
zope.interface
import
zope.interface
from
AccessControl
import
ClassSecurityInfo
from
AccessControl
import
ClassSecurityInfo
...
@@ -37,12 +38,28 @@ from Products.ERP5Type.Globals import InitializeClass
...
@@ -37,12 +38,28 @@ from Products.ERP5Type.Globals import InitializeClass
from
Products.CMFCore.Expression
import
Expression
from
Products.CMFCore.Expression
import
Expression
from
Products.ERP5Type
import
interfaces
,
Permissions
,
PropertySheet
from
Products.ERP5Type
import
interfaces
,
Permissions
,
PropertySheet
from
Products.ERP5Type.ERP5Type
\
from
Products.ERP5Type.ERP5Type
import
(
import
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
,
)
from
Products.ERP5Type.Permissions
import
AccessContentsInformation
from
Products.ERP5Type.Permissions
import
AccessContentsInformation
from
Products.ERP5Type.XMLObject
import
XMLObject
from
Products.ERP5Type.XMLObject
import
XMLObject
import
six
import
six
def
_toSecurityGroupIdGenerationScriptV2
(
getCategoryValue
,
category_definition_dict
,
):
result
=
{}
for
base_category
,
relative_url_list
in
six
.
iteritems
(
category_definition_dict
):
result
[
base_category
]
=
result_value_list
=
[]
if
isinstance
(
relative_url_list
,
str
):
relative_url_list
=
(
relative_url_list
,
)
for
relative_url
in
relative_url_list
:
category_value
=
getCategoryValue
(
base_category
+
'/'
+
relative_url
.
rstrip
(
'*'
))
assert
category_value
is
not
None
,
(
base_category
,
relative_url
,
category_definition_dict
)
result_value_list
.
append
((
category_value
,
relative_url
.
endswith
(
'*'
)))
return
result
@
zope
.
interface
.
implementer
(
interfaces
.
ILocalRoleGenerator
)
@
zope
.
interface
.
implementer
(
interfaces
.
ILocalRoleGenerator
)
class
RoleInformation
(
XMLObject
):
class
RoleInformation
(
XMLObject
):
...
@@ -143,7 +160,7 @@ class RoleInformation(XMLObject):
...
@@ -143,7 +160,7 @@ class RoleInformation(XMLObject):
# defined categories)
# defined categories)
category_result
=
[{}]
category_result
=
[{}]
group_id_role_dict
=
{}
group_id_role_dict
=
defaultdict
(
set
)
role_list
=
self
.
getRoleNameList
()
role_list
=
self
.
getRoleNameList
()
if
isinstance
(
category_result
,
dict
):
if
isinstance
(
category_result
,
dict
):
...
@@ -153,26 +170,44 @@ class RoleInformation(XMLObject):
...
@@ -153,26 +170,44 @@ class RoleInformation(XMLObject):
for
role
,
group_id_list
in
six
.
iteritems
(
category_result
):
for
role
,
group_id_list
in
six
.
iteritems
(
category_result
):
if
role
in
role_list
:
if
role
in
role_list
:
for
group_id
in
group_id_list
:
for
group_id
in
group_id_list
:
group_id_role_dict
.
setdefault
(
group_id
,
set
())
.
add
(
role
)
group_id_role_dict
[
group_id
]
.
add
(
role
)
else
:
else
:
group_id_generator
=
getattr
(
ob
,
group_id_generator
=
getattr
(
ob
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
)
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
,
None
)
# Prepare definition dict once only
# Prepare definition dict once only
category_definition_dict
=
{}
category_definition_dict
=
defaultdict
(
list
)
for
c
in
self
.
getRoleCategoryList
():
for
c
in
self
.
getRoleCategoryList
():
bc
,
value
=
c
.
split
(
'/'
,
1
)
bc
,
value
=
c
.
split
(
'/'
,
1
)
category_definition_dict
.
setdefault
(
bc
,
[]).
append
(
value
)
category_definition_dict
[
bc
].
append
(
value
)
if
group_id_generator
is
None
:
group_id_generator
=
getattr
(
ob
,
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
)
getCategoryValue
=
self
.
getPortalObject
().
portal_categories
.
getCategoryValue
category_definition_dict
=
_toSecurityGroupIdGenerationScriptV2
(
getCategoryValue
=
getCategoryValue
,
category_definition_dict
=
category_definition_dict
,
)
category_result
=
[
_toSecurityGroupIdGenerationScriptV2
(
getCategoryValue
=
getCategoryValue
,
category_definition_dict
=
category_dict
,
)
for
category_dict
in
category_result
]
else
:
# BBB
for
category_dict
in
category_result
:
category_dict
.
setdefault
(
'category_order'
,
category_order_list
)
group_id_generator_
=
group_id_generator
group_id_generator
=
lambda
category_dict
:
group_id_generator_
(
**
category_dict
)
# category_result is a list of dicts that represents the resolved
# category_result is a list of dicts that represents the resolved
# categories we create a category_value_dict from each of these
# categories we create a category_value_dict from each of these
# dicts aggregated with category_order and statically defined
# dicts aggregated with category_order and statically defined
# categories
# categories
for
category_dict
in
category_result
:
for
category_dict
in
category_result
:
category_value_dict
=
{
'category_order'
:
category_order_list
}
category_value_dict
=
category_dict
.
copy
()
category_value_dict
.
update
(
category_dict
)
category_value_dict
.
update
(
category_definition_dict
)
category_value_dict
.
update
(
category_definition_dict
)
group_id_list
=
group_id_generator
(
**
category_value_dict
)
group_id_list
=
group_id_generator
(
category_dict
=
category_value_dict
)
if
group_id_list
:
if
group_id_list
:
if
isinstance
(
group_id_list
,
str
):
if
isinstance
(
group_id_list
,
str
):
# Single group is defined (this is usually for group membership)
# Single group is defined (this is usually for group membership)
...
...
product/ERP5Type/ERP5Type.py
View file @
049fd1cb
...
@@ -45,6 +45,7 @@ from Products.ERP5Type.dynamic.accessor_holder import getPropertySheetValueList,
...
@@ -45,6 +45,7 @@ from Products.ERP5Type.dynamic.accessor_holder import getPropertySheetValueList,
import
six
import
six
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
=
'ERP5Type_asSecurityGroupId'
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT
=
'ERP5Type_asSecurityGroupId'
ERP5TYPE_SECURITY_GROUP_ID_GENERATION_SCRIPT_V2
=
'ERP5Type_asSecurityGroupIdSet'
from
.TranslationProviderBase
import
TranslationProviderBase
from
.TranslationProviderBase
import
TranslationProviderBase
from
Products.ERP5Type.Accessor.Constant
import
PropertyGetter
as
ConstantGetter
from
Products.ERP5Type.Accessor.Constant
import
PropertyGetter
as
ConstantGetter
...
...
product/ERP5Type/tests/CodingStyleTestCase.py
View file @
049fd1cb
...
@@ -136,6 +136,8 @@ class CodingStyleTestCase(ERP5TypeTestCase):
...
@@ -136,6 +136,8 @@ class CodingStyleTestCase(ERP5TypeTestCase):
"""
"""
self
.
maxDiff
=
None
self
.
maxDiff
=
None
template_tool
=
self
.
portal
.
portal_templates
template_tool
=
self
.
portal
.
portal_templates
diff_line_list
=
[]
diff_files
=
[]
for
bt_title
in
self
.
getTestedBusinessTemplateList
():
for
bt_title
in
self
.
getTestedBusinessTemplateList
():
bt
=
template_tool
.
getInstalledBusinessTemplate
(
bt_title
,
strict
=
True
)
bt
=
template_tool
.
getInstalledBusinessTemplate
(
bt_title
,
strict
=
True
)
# run migrations on the business template, except for the BT used to
# run migrations on the business template, except for the BT used to
...
@@ -158,7 +160,6 @@ class CodingStyleTestCase(ERP5TypeTestCase):
...
@@ -158,7 +160,6 @@ class CodingStyleTestCase(ERP5TypeTestCase):
p
.
strip
()
for
p
in
self
.
rebuild_business_template_ignored_path
.
splitlines
()
p
.
strip
()
for
p
in
self
.
rebuild_business_template_ignored_path
.
splitlines
()
if
p
and
not
p
.
strip
().
startswith
(
"#"
)}
if
p
and
not
p
.
strip
().
startswith
(
"#"
)}
diff_line_list
=
[]
def
get_difference
(
path
,
has_old
=
True
,
has_new
=
True
):
def
get_difference
(
path
,
has_old
=
True
,
has_new
=
True
):
old
=
(
old
=
(
os
.
path
.
join
(
bt_base_path
,
path
)
os
.
path
.
join
(
bt_base_path
,
path
)
...
@@ -214,13 +215,14 @@ class CodingStyleTestCase(ERP5TypeTestCase):
...
@@ -214,13 +215,14 @@ class CodingStyleTestCase(ERP5TypeTestCase):
for
diff
in
get_differences
(
sub_dcmp
,
os
.
path
.
join
(
base
,
sub_path
)):
for
diff
in
get_differences
(
sub_dcmp
,
os
.
path
.
join
(
base
,
sub_path
)):
yield
diff
yield
diff
diff_files
=
list
(
get_differences
(
filecmp
.
dircmp
(
bt_dir
,
export_dir
),
bt_local_path
))
diff_files
.
extend
(
list
(
get_differences
(
filecmp
.
dircmp
(
bt_dir
,
export_dir
),
bt_local_path
)))
# dump a diff in log directory, to help debugging
from
Products.ERP5Type.tests.runUnitTest
import
log_directory
# dump a diff in log directory, to help debugging
if
log_directory
and
diff_line_list
:
from
Products.ERP5Type.tests.runUnitTest
import
log_directory
with
open
(
os
.
path
.
join
(
log_directory
,
'%s.diff'
%
self
.
id
()),
'w'
)
as
f
:
if
log_directory
and
diff_line_list
:
f
.
writelines
(
diff_line_list
)
with
open
(
os
.
path
.
join
(
log_directory
,
'%s.diff'
%
self
.
id
()),
'w'
)
as
f
:
self
.
assertEqual
(
diff_files
,
[])
f
.
writelines
(
diff_line_list
)
self
.
assertEqual
(
diff_files
,
[])
def
test_run_upgrader
(
self
):
def
test_run_upgrader
(
self
):
...
@@ -288,7 +290,7 @@ class CodingStyleTestCase(ERP5TypeTestCase):
...
@@ -288,7 +290,7 @@ class CodingStyleTestCase(ERP5TypeTestCase):
'category'
:
action_category
,
'category'
:
action_category
,
'action_name'
:
action_name
,
'action_name'
:
action_name
,
})
})
self
.
assertEqual
(
duplicate_action_list
,
[])
self
.
assertEqual
(
duplicate_action_list
,
[])
def
test_workflow_consistency
(
self
):
def
test_workflow_consistency
(
self
):
self
.
maxDiff
=
None
self
.
maxDiff
=
None
...
...
product/ERP5Type/tests/ERP5TypeLiveTestCase.py
View file @
049fd1cb
...
@@ -291,12 +291,11 @@ def runLiveTest(test_list, verbosity=1, stream=None, request_server_url=None, **
...
@@ -291,12 +291,11 @@ def runLiveTest(test_list, verbosity=1, stream=None, request_server_url=None, **
if
stream
is
None
:
if
stream
is
None
:
output
=
StringIO
()
output
=
StringIO
()
def
print_and_write
(
data
):
def
print_and_write
(
data
):
sys
.
stdout
.
write
(
data
)
sys
.
stderr
.
write
(
data
)
sys
.
stdout
.
flush
()
sys
.
stderr
.
flush
()
return
StringIO
.
write
(
output
,
data
)
return
output
.
write
(
data
)
output
.
write
=
print_and_write
print_and_write
(
"**Running Live Test:
\
n
"
)
output
.
write
(
"**Running Live Test:
\
n
"
)
ZopeTestCase
.
_print
=
print_and_write
ZopeTestCase
.
_print
=
output
.
write
with
warnings
.
catch_warnings
():
with
warnings
.
catch_warnings
():
warnings
.
simplefilter
(
kw
[
'warnings'
])
warnings
.
simplefilter
(
kw
[
'warnings'
])
...
...
product/Formulator/Widget.py
View file @
049fd1cb
...
@@ -86,7 +86,7 @@ def convertToString(value):
...
@@ -86,7 +86,7 @@ def convertToString(value):
return
str
(
value
)
return
str
(
value
)
return
value
return
value
class
Widget
:
class
Widget
(
object
)
:
"""A field widget that knows how to display itself as HTML.
"""A field widget that knows how to display itself as HTML.
"""
"""
...
...
setup.py
View file @
049fd1cb
...
@@ -2,7 +2,7 @@ from setuptools import setup, find_packages
...
@@ -2,7 +2,7 @@ from setuptools import setup, find_packages
import
glob
import
glob
import
os
import
os
version
=
'0.4.7
5
'
version
=
'0.4.7
6
'
name
=
'erp5.util'
name
=
'erp5.util'
long_description
=
open
(
"README.erp5.util.txt"
).
read
()
+
"
\
n
"
long_description
=
open
(
"README.erp5.util.txt"
).
read
()
+
"
\
n
"
...
...
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