Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
E
erp5
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Ayush Tiwari
erp5
Commits
37e6839a
Commit
37e6839a
authored
Nov 22, 2017
by
Ayush Tiwari
Committed by
Ayush Tiwari
Dec 21, 2017
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Diff Tool: Find Diff betweeen any 2 erp5 objects
parent
8858ca86
Changes
9
Show whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
476 additions
and
3 deletions
+476
-3
product/ERP5/ERP5Site.py
product/ERP5/ERP5Site.py
+1
-0
product/ERP5/Tool/DiffTool.py
product/ERP5/Tool/DiffTool.py
+224
-0
product/ERP5/__init__.py
product/ERP5/__init__.py
+3
-1
product/ERP5Type/Base.py
product/ERP5Type/Base.py
+3
-1
product/ERP5Type/Core/Folder.py
product/ERP5Type/Core/Folder.py
+4
-1
product/ERP5Type/interfaces/json_representable.py
product/ERP5Type/interfaces/json_representable.py
+52
-0
product/ERP5Type/interfaces/patchable.py
product/ERP5Type/interfaces/patchable.py
+47
-0
product/ERP5Type/mixin/json_representable.py
product/ERP5Type/mixin/json_representable.py
+89
-0
product/ERP5Type/mixin/patchable.py
product/ERP5Type/mixin/patchable.py
+53
-0
No files found.
product/ERP5/ERP5Site.py
View file @
37e6839a
...
...
@@ -1989,6 +1989,7 @@ class ERP5Generator(PortalGenerator):
addERP5Tool
(
p
,
'portal_simulation'
,
'Simulation Tool'
)
addERP5Tool
(
p
,
'portal_deliveries'
,
'Delivery Tool'
)
addERP5Tool
(
p
,
'portal_orders'
,
'Order Tool'
)
addERP5Tool
(
p
,
'portal_diff'
,
'Diff Tool'
)
def
setupTemplateTool
(
self
,
p
,
**
kw
):
"""
...
...
product/ERP5/Tool/DiffTool.py
0 → 100644
View file @
37e6839a
# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2017 Nexedi SARL and Contributors. All Rights Reserved.
# Ayush Tiwari <ayush.tiwari@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################
import
jsonpatch
from
deepdiff
import
DeepDiff
from
AccessControl
import
ClassSecurityInfo
from
Products.ERP5Type.Globals
import
InitializeClass
from
Products.ERP5Type.Tool.BaseTool
import
BaseTool
from
Products.ERP5Type
import
Permissions
class
DiffTool
(
BaseTool
):
"""
A portal tool that provides all kinds of utilities to
compare objects.
"""
id
=
'portal_diff'
title
=
'Diff Tool'
meta_type
=
'ERP5 Diff Tool'
portal_type
=
'Portal Diff Tool'
allowed_types
=
()
# Declarative Security
security
=
ClassSecurityInfo
()
def
diffPortalObject
(
self
,
old
,
new
,
path
=
None
,
patch_format
=
"deepdiff"
):
"""
Returns a PortalPatch instance with the appropriate format
original -- original object
new -- new object
path -- optional path to specify which property to diff
patch_format -- optional format (rfc6902 or deepdiff)
"""
return
PortalPatch
(
old
,
new
,
patch_format
)
class
PortalPatch
:
"""
Provides an abstraction to a patch that
depends on the patch format.
In the case of deepdiff, the abstraction can
lead to a commutative merge system.
In the case of rfc6902, the abstraction can not
lead to a commutative merge system but may be
useful to some UI applications.
"""
def
__init__
(
self
,
old
,
new
,
patch_format
=
"deepdiff"
):
"""
Intialises the class from a deepdiff or
a rfc6902 patch. deepdiff is the default.
"""
self
.
old_value
=
old
self
.
new_value
=
new
self
.
patch_format
=
patch_format
def
getPortalPatchOperationList
(
self
):
"""
List all PortalPatchOperation instances in the PortalPatch
"""
patch
=
self
.
asDeepDiffPatch
()
# In general, we are using `tree` view, so basically for us all operations
# currently are `values_changed` from old to new value or to none
change_list
=
patch
.
values
()
# Here we can have the change_list as nested list also, for example:
#
# change_list =
# {
# 'iterable_item_removed': set([<root[2] t1:'c', t2:Not Present>]),
# 'values_changed': set([<root[1] t1:'b', t2:'e'>, <root[0] t1:'a', t2:'d'>])
# }
# We can see here that the values are basically change from one value to
# another, so to get the list of operation(s), we have to flatten all the
# values in one list
flatten_change_list
=
[
item
for
sublist
in
change_list
for
item
in
sublist
]
return
flatten_change_list
def
patchPortalObject
(
self
,
object
):
"""
Apply patch to an object by applying
one by one each PortalPatchItem
"""
pass
def
asDeepDiffPatch
(
self
):
"""
Returns a Json patch with deep diff extensions
"""
# Use try-except as it's easier to ask forgiveness than permission
# `_asDict` is available only for objects, so in that case, we convert the
# ERP5-fied objects into dict and then work on them.
# In all other cases, we let `deepdiff` do its work on checking the type
try
:
src
=
self
.
old_value
.
_asDict
()
except
AttributeError
:
src
=
self
.
old_value
try
:
dst
=
self
.
new_value
.
_asDict
()
except
AttributeError
:
dst
=
self
.
new_value
# For now, we prefer having 'tree' view as it provides us with node level
# where on each node we have value changed(atleast for list and dictionary)
ddiff
=
DeepDiff
(
src
,
dst
,
view
=
'tree'
)
return
ddiff
def
asStrippedHTML
(
self
):
"""
Returns an HTML representation of the whole patch
that can be embedded
"""
pass
def
asHTML
(
self
):
"""
Returns an HTML representation of the whole patch
that can be displayed in a standalone way
"""
pass
class
PortalPatchOperation
:
"""
Provides an abstraction to a patch operation that
depends on the patch format.
In the case of deepdiff, each operation defines
actually a desired state in a declarative way.
In the case of rfc6902, each operation is defined
in an imperative manner.
"""
def
patchPortalObject
(
object
,
unified_diff_selection
=
None
):
"""
Apply patch to an object
unified_diff_selection -- a selection of lines in the unified diff
that will be applied
"""
pass
def
getOperation
(
self
):
"""
Returns one of "replace", "add" or "remove"
(hopefully, this can also be used for deepdiff format)
set_item_added, values_changed, etc.
"""
pass
def
getPath
(
self
):
"""
Returns a path representing the value that is changed
(hopefully, this can also be used for deepdiff format)
"""
pass
def
getOldValue
(
self
):
"""
Returns the old value
"""
pass
def
getNewValue
(
self
):
"""
Returns the new value
"""
pass
def
getUnifiedDiff
(
self
):
"""
Returns a unified diff of the value changed
(this is useful for a text value) or None if
there is no such change.
(see String difference 2 in deepdiff)
"""
pass
def
asStrippedHTML
(
self
):
"""
Returns an HTML representation of the change
that can be embedded
"""
pass
def
asHTML
(
self
):
"""
Returns an HTML representation that can be displayed
in a standalone way
"""
pass
InitializeClass
(
DiffTool
)
product/ERP5/__init__.py
View file @
37e6839a
...
...
@@ -51,7 +51,8 @@ from Tool import CategoryTool, SimulationTool, RuleTool, IdTool, TemplateTool,\
GadgetTool
,
ContributionRegistryTool
,
IntrospectionTool
,
\
AcknowledgementTool
,
SolverTool
,
SolverProcessTool
,
\
ConversionTool
,
RoundingTool
,
UrlRegistryTool
,
InterfaceTool
,
\
CertificateAuthorityTool
,
InotifyTool
,
TaskDistributionTool
CertificateAuthorityTool
,
InotifyTool
,
TaskDistributionTool
,
\
DiffTool
import
ERP5Site
from
Document
import
PythonScript
,
SQLMethod
object_classes
=
(
ERP5Site
.
ERP5Site
,
...
...
@@ -85,6 +86,7 @@ portal_tools = ( CategoryTool.CategoryTool,
InotifyTool
.
InotifyTool
,
TaskDistributionTool
.
TaskDistributionTool
,
InterfaceTool
.
InterfaceTool
,
DiffTool
.
DiffTool
)
content_classes
=
()
content_constructors
=
()
...
...
product/ERP5Type/Base.py
View file @
37e6839a
...
...
@@ -86,6 +86,7 @@ from Products.ERP5Type.Accessor.TypeDefinition import asDate
from
Products.ERP5Type.Message
import
Message
from
Products.ERP5Type.ConsistencyMessage
import
ConsistencyMessage
from
Products.ERP5Type.UnrestrictedMethod
import
UnrestrictedMethod
,
super_user
from
Products.ERP5Type.mixin.json_representable
import
JSONRepresentableMixin
from
zope.interface
import
classImplementsOnly
,
implementedBy
...
...
@@ -675,7 +676,8 @@ class Base( CopyContainer,
ActiveObject
,
OFS
.
History
.
Historical
,
ERP5PropertyManager
,
PropertyTranslatableBuiltInDictMixIn
PropertyTranslatableBuiltInDictMixIn
,
JSONRepresentableMixin
,
):
"""
This is the base class for all ERP5 Zope objects.
...
...
product/ERP5Type/Core/Folder.py
View file @
37e6839a
...
...
@@ -48,6 +48,8 @@ from Products.ERP5Type.Utils import sortValueList
from
Products.ERP5Type
import
Permissions
from
Products.ERP5Type.Globals
import
InitializeClass
from
Products.ERP5Type.Accessor
import
Base
as
BaseAccessor
from
Products.ERP5Type.mixin.json_representable
import
JSONRepresentableMixin
try
:
from
Products.CMFCore.CMFBTreeFolder
import
CMFBTreeFolder
except
ImportError
:
...
...
@@ -524,7 +526,8 @@ HBTREE_HANDLER = 2
InitializeClass
(
FolderMixIn
)
class
Folder
(
CopyContainer
,
CMFBTreeFolder
,
CMFHBTreeFolder
,
Base
,
FolderMixIn
):
class
Folder
(
CopyContainer
,
CMFBTreeFolder
,
CMFHBTreeFolder
,
Base
,
FolderMixIn
,
JSONRepresentableMixin
,):
"""
A Folder is a subclass of Base but not of XMLObject.
Folders are not considered as documents and are therefore
...
...
product/ERP5Type/interfaces/json_representable.py
0 → 100644
View file @
37e6839a
# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2017 Nexedi SA and Contributors. All Rights Reserved.
# Ayush Tiwari <ayush.tiwari@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################
from
zope.interface
import
Interface
class
IJSONRepresentable
(
Interface
):
"""
An interface for objects that can be converted to JSON
and back to an ERP5 object.
This can be useful if we want to use JSON as much
as possible in ERP5 in the future and ensure
that certain objects (not all) can be converted to JSON
back and forth
"""
def
asJSON
():
"""
Returns a JSON representation based on
propertysheets and portal properties
"""
def
fromJSON
():
"""
Updates an object based on a JSON representation
"""
product/ERP5Type/interfaces/patchable.py
0 → 100644
View file @
37e6839a
# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2017 Nexedi SA and Contributors. All Rights Reserved.
# Ayush Tiwari <ayush.tiwari@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################
from
zope.interface
import
Interface
class
IPatchable
(
Interface
):
"""
An interface for objects that can be patched. This is useful
to raise an exception for some objects that are not meant to be
supported by BusinessTemplatePatchItem
"""
def
patch
(
patch
,
path
=
None
,
unified_diff_selection
=
None
):
"""
Apply a PortalPatch or PortalPatchOperation to an object
unified_diff_selection -- a selection of lines in the unified diff
that will be applied
path -- optional path in case one wants to patch only some properties
"""
product/ERP5Type/mixin/json_representable.py
0 → 100644
View file @
37e6839a
# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2017 Nexedi SA and Contributors. All Rights Reserved.
# Ayush Tiwari <ayush.tiwari@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################
import
json
import
xmltodict
import
zope.interface
from
OFS
import
XMLExportImport
from
StringIO
import
StringIO
from
AccessControl
import
ClassSecurityInfo
from
Products.ERP5Type.interfaces.json_representable
import
IJSONRepresentable
from
Products.ERP5Type
import
Permissions
from
Products.ERP5Type.Globals
import
InitializeClass
class
JSONRepresentableMixin
:
"""
An implementation for IJSONRepresentable
"""
# Declarative Security
security
=
ClassSecurityInfo
()
security
.
declareObjectProtected
(
Permissions
.
AccessContentsInformation
)
zope
.
interface
.
implements
(
IJSONRepresentable
)
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'asJSON'
)
def
asJSON
(
self
):
"""
Generate a JSON representable content for ERP5 object
Currently we use `XMLExportImport` to first convert the object to its XML
respresentation and then use xmltodict to convert it to dict and JSON
format finally
"""
dict_value
=
self
.
_asDict
()
# Convert the XML to json representation
return
json
.
dumps
(
dict_value
)
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'asDict'
)
def
_asDict
(
self
):
"""
Gets the dict representation of the object
"""
# Use OFS exportXML to first export to xml
f
=
StringIO
()
XMLExportImport
.
exportXML
(
self
.
_p_jar
,
self
.
_p_oid
,
f
)
# Get the value of exported XML
xml_value
=
f
.
getvalue
()
return
xmltodict
.
parse
(
xml_value
)
def
fromJSON
(
self
,
val
):
"""
Updates an object, based on a JSON representation
"""
dict_value
=
json
.
loads
(
val
)
# Convert the dict_value to XML representation
xml_value
=
xmltodict
.
unparse
(
dict_value
)
f
=
StringIO
(
xml_value
)
return
XMLExportImport
.
importXML
(
self
.
_p_jar
,
f
)
InitializeClass
(
JSONRepresentableMixin
)
product/ERP5Type/mixin/patchable.py
0 → 100644
View file @
37e6839a
# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2017 Nexedi SA and Contributors. All Rights Reserved.
# Ayush Tiwari <ayush.tiwari@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################
import
zope.interface
from
Products.ERP5Type.interfaces.patchable
import
IPatchable
from
AccessControl
import
ClassSecurityInfo
from
Products.ERP5Type
import
Permissions
from
Products.ERP5Type.Globals
import
InitializeClass
class
PatchableMixin
:
"""
An implementation of IPatchable
"""
zope
.
interface
.
implements
(
IPatchable
)
# Declarative Security
security
=
ClassSecurityInfo
()
security
.
declareObjectProtected
(
Permissions
.
AccessContentsInformation
)
security
.
declareProtected
(
Permissions
.
AccessContentsInformation
,
'patch'
)
def
patch
(
patch
,
path
=
None
,
unified_diff_selection
=
None
):
pass
InitializeClass
(
PatchableMixin
)
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