Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Z
Zope
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
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
Zope
Commits
e82c919e
Commit
e82c919e
authored
Feb 26, 1999
by
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
First whack at propertysheets.
parent
0c4d2676
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
528 additions
and
2 deletions
+528
-2
lib/python/OFS/PropertyManager.py
lib/python/OFS/PropertyManager.py
+6
-2
lib/python/OFS/PropertySheets.py
lib/python/OFS/PropertySheets.py
+522
-0
No files found.
lib/python/OFS/PropertyManager.py
View file @
e82c919e
...
...
@@ -84,7 +84,7 @@
##############################################################################
"""Property management"""
__version__
=
'$Revision: 1.
9
$'
[
11
:
-
2
]
__version__
=
'$Revision: 1.
10
$'
[
11
:
-
2
]
from
ZPublisher.Converters
import
type_converters
...
...
@@ -95,7 +95,7 @@ from Acquisition import Implicit
from
Globals
import
Persistent
from
DateTime
import
DateTime
from
PropertySheets
import
vps
class
PropertyManager
:
"""
...
...
@@ -159,6 +159,9 @@ class PropertyManager:
_properties
=
({
'id'
:
'title'
,
'type'
:
'string'
,
'mode'
:
'w'
},)
_reserved_names
=
()
__propsets__
=
()
propertysheets
=
vps
()
def
valid_property_id
(
self
,
id
):
if
not
id
or
id
[:
1
]
==
'_'
or
(
' '
in
id
)
\
or
hasattr
(
self
.
aq_base
,
id
):
...
...
@@ -316,3 +319,4 @@ class PropertyManager:
if
REQUEST
is
not
None
:
return
self
.
manage_propertiesForm
(
self
,
REQUEST
)
lib/python/OFS/PropertySheets.py
0 → 100644
View file @
e82c919e
##############################################################################
#
# Zope Public License (ZPL) Version 0.9.4
# ---------------------------------------
#
# Copyright (c) Digital Creations. All rights reserved.
#
# Redistribution and use in source and binary forms, with or
# without modification, are permitted provided that the following
# conditions are met:
#
# 1. Redistributions in source code must retain the above
# copyright notice, this list of conditions, and the following
# disclaimer.
#
# 2. Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions, and the following
# disclaimer in the documentation and/or other materials
# provided with the distribution.
#
# 3. Any use, including use of the Zope software to operate a
# website, must either comply with the terms described below
# under "Attribution" or alternatively secure a separate
# license from Digital Creations.
#
# 4. All advertising materials, documentation, or technical papers
# mentioning features derived from or use of this software must
# display the following acknowledgement:
#
# "This product includes software developed by Digital
# Creations for use in the Z Object Publishing Environment
# (http://www.zope.org/)."
#
# 5. Names associated with Zope or Digital Creations must not be
# used to endorse or promote products derived from this
# software without prior written permission from Digital
# Creations.
#
# 6. Redistributions of any form whatsoever must retain the
# following acknowledgment:
#
# "This product includes software developed by Digital
# Creations for use in the Z Object Publishing Environment
# (http://www.zope.org/)."
#
# 7. Modifications are encouraged but must be packaged separately
# as patches to official Zope releases. Distributions that do
# not clearly separate the patches from the original work must
# be clearly labeled as unofficial distributions.
#
# Disclaimer
#
# THIS SOFTWARE IS PROVIDED BY DIGITAL CREATIONS ``AS IS'' AND
# ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
# FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
# SHALL DIGITAL CREATIONS OR ITS CONTRIBUTORS BE LIABLE FOR ANY
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
# IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
# THE POSSIBILITY OF SUCH DAMAGE.
#
# Attribution
#
# Individuals or organizations using this software as a web site
# must provide attribution by placing the accompanying "button"
# and a link to the accompanying "credits page" on the website's
# main entry point. In cases where this placement of
# attribution is not feasible, a separate arrangment must be
# concluded with Digital Creations. Those using the software
# for purposes other than web sites must provide a corresponding
# attribution in locations that include a copyright using a
# manner best suited to the application environment.
#
# This software consists of contributions made by Digital
# Creations and many individuals on behalf of Digital Creations.
# Specific attributions are listed in the accompanying credits
# file.
#
##############################################################################
"""Property sheets"""
__version__
=
'$Revision: 1.1 $'
[
11
:
-
2
]
import
time
,
string
from
ZPublisher.Converters
import
type_converters
from
Globals
import
HTMLFile
,
MessageDialog
from
string
import
find
,
join
,
lower
,
split
from
Acquisition
import
Implicit
,
Explicit
from
ExtensionClass
import
Base
from
Globals
import
Persistent
class
PropertySheet
(
Persistent
,
Implicit
):
"""A PropertySheet is a container for a set of related properties and
metadata describing those properties. PropertySheets may or may not
provide a web interface for managing its properties."""
_properties
=
()
def
__init__
(
self
,
id
,
md
=
None
):
# Create a new property set, using the given id and namespace
# string. The namespace string should be usable as an xml name-
# space identifier.
self
.
id
=
id
self
.
md
=
md
or
{}
def
xml_namespace
(
self
):
# Return a namespace string usable as an xml namespace
# for this property set.
return
self
.
md
.
get
(
'xmlns'
,
''
)
def
v_self
(
self
):
return
self
def
valid_property_id
(
self
,
id
):
# Return a true value if the given id is valid to use as
# a property id. Note that this method does not consider
# factors other than the actual value of the id, such as
# whether the given id is already in use.
if
not
id
or
(
id
[:
1
]
==
'_'
)
or
(
' '
in
id
):
return
0
return
1
def
hasProperty
(
self
,
id
):
# Return a true value if a property exists with the given id.
for
prop
in
self
.
propertyMap
():
if
id
==
prop
[
'id'
]:
return
1
return
0
def
getProperty
(
self
,
id
,
default
=
None
):
# Return the property with the given id, returning the optional
# second argument or None if no such property is found.
if
self
.
hasProperty
(
id
):
return
getattr
(
self
.
v_self
(),
id
)
return
default
def
setProperty
(
self
,
id
,
value
,
type
=
'string'
,
meta
=
None
):
# Set a new property with the given id, value and optional type.
# Note that different property sets may support different typing
# systems.
if
not
self
.
valid_property_id
(
id
):
raise
'Bad Request'
,
'Invalid property id.'
self
=
self
.
v_self
()
if
meta
is
None
:
meta
=
{}
prop
=
{
'id'
:
id
,
'type'
:
type
,
'meta'
:
meta
}
self
.
_properties
=
self
.
_properties
+
(
prop
,)
setattr
(
self
,
id
,
value
)
def
updateProperty
(
self
,
id
,
value
):
# Update the value of an existing property. If value is a string,
# an attempt will be made to convert the value to the type of the
# existing property.
if
not
self
.
hasProperty
(
id
):
raise
'Bad Request'
,
'The property %s does not exist.'
%
id
if
type
(
value
)
==
type
(
''
):
proptype
=
self
.
propertyInfo
(
id
).
get
(
'type'
,
'string'
)
if
type_converters
.
has_key
(
proptype
):
value
=
type_converters
[
proptype
](
value
)
setattr
(
self
.
v_self
(),
id
,
value
)
def
delProperty
(
self
,
id
):
# Delete the property with the given id. If a property with the
# given id does not exist, a ValueError is raised.
if
not
self
.
hasProperty
(
id
):
raise
ValueError
,
'The property %s does not exist.'
%
id
self
=
self
.
v_self
()
delattr
(
self
,
id
)
self
.
_properties
=
tuple
(
filter
(
lambda
i
,
n
=
id
:
i
[
'id'
]
!=
n
,
self
.
_properties
))
def
propertyIds
(
self
):
# Return a list of property ids.
return
map
(
lambda
i
:
i
[
'id'
],
self
.
propertyMap
())
def
propertyValues
(
self
):
# Return a list of property values.
vself
=
self
.
v_self
()
return
map
(
lambda
i
,
s
=
vself
:
s
.
getProperty
(
i
[
'id'
]),
self
.
propertyMap
())
def
propertyItems
(
self
):
# Return a list of (id, property) tuples.
vself
=
self
.
v_self
()
return
map
(
lambda
i
,
s
=
vself
:
(
i
[
'id'
],
s
.
getProperty
(
i
[
'id'
])),
self
.
propertyMap
())
def
propertyInfo
(
self
,
id
):
# Return a mapping containing property meta-data
for
p
in
self
.
propertyMap
():
if
p
[
'id'
]
==
id
:
return
p
raise
ValueError
,
'Property %s not found.'
%
id
def
propertyMap
(
self
):
# Return a tuple of mappings, giving meta-data for properties.
return
self
.
v_self
().
_properties
def
propdict
(
self
):
dict
=
{}
for
p
in
self
.
propertyMap
():
dict
[
p
[
'id'
]]
=
p
return
dict
def
dav__propstat
(
self
,
url
,
allprop
,
vals
,
join
=
string
.
join
):
# The dav__propstat method returns a chunk of xml containing
# one or more propstat elements indicating property names,
# values, errors and status codes.
propstat
=
'<d:propstat%s>
\
n
'
\
' <d:prop>
\
n
'
\
'%s
\
n
'
\
' </d:prop>
\
n
'
\
' <d:status>HTTP/1.1 %s</d:status>
\
n
'
\
'</d:propstat>'
result
=
[]
if
not
allprop
and
not
vals
:
# propname request
for
name
in
self
.
propertyIds
():
result
.
append
(
' <ns0:%s/>'
%
name
)
result
=
join
(
result
,
'
\
n
'
)
nsdef
=
' xmlns:ns0="%s"'
%
self
.
xml_namespace
()
return
propstat
%
(
nsdef
,
result
,
'200 OK'
)
elif
allprop
:
for
name
,
value
in
self
.
propertyItems
():
prop
=
' <ns0:%s>%s</ns0:%s>'
%
(
name
,
value
,
name
)
result
.
append
(
prop
)
result
=
join
(
result
,
'
\
n
'
)
nsdef
=
' xmlns:ns0="%s"'
%
self
.
xml_namespace
()
return
propstat
%
(
nsdef
,
result
,
'200 OK'
)
else
:
xml_ns
=
self
.
xml_namespace
()
propdict
=
self
.
propdict
()
nsdef
=
' xmlns:ns0="%s"'
%
self
.
xml_namespace
()
for
name
,
ns
in
vals
:
if
ns
==
xml_ns
:
if
propdict
.
has_key
(
name
):
value
=
self
.
getProperty
(
name
)
prop
=
' <ns0:%s>%s</ns0:%s>'
%
(
name
,
value
,
name
)
result
.
append
(
propstat
%
(
nsdef
,
prop
,
'200 OK'
))
else
:
prop
=
' <ns0:%s/>'
%
name
result
.
append
(
propstat
%
(
nsdef
,
prop
,
'404 Not Found'
))
return
join
(
result
,
'
\
n
'
)
# Web interface
manage_propertiesForm
=
HTMLFile
(
'properties'
,
globals
())
def
manage_addProperty
(
self
,
id
,
value
,
type
,
REQUEST
=
None
):
"""Add a new property via the web. Sets a new property with
the given id, type, and value."""
if
type_converters
.
has_key
(
type
):
value
=
type_converters
[
type
](
value
)
self
.
setProperty
(
id
,
value
,
type
)
if
REQUEST
is
not
None
:
return
self
.
manage_propertiesForm
(
self
,
REQUEST
)
def
manage_changeProperties
(
self
,
REQUEST
=
None
,
**
kw
):
"""Change existing object properties by passing either a mapping
object of name:value pairs {'foo':6} or passing name=value
parameters."""
if
REQUEST
is
None
:
props
=
{}
else
:
props
=
REQUEST
if
kw
:
for
name
,
value
in
kw
.
items
():
props
[
name
]
=
value
propdict
=
self
.
propdict
()
vself
=
self
.
v_self
()
for
name
,
value
in
props
.
items
():
if
self
.
hasProperty
(
name
):
if
not
'w'
in
propdict
[
name
].
get
(
'mode'
,
'wd'
):
raise
'BadRequest'
,
'%s cannot be changed.'
%
name
vself
.
updateProperty
(
name
,
value
)
if
REQUEST
is
not
None
:
return
MessageDialog
(
title
=
'Success!'
,
message
=
'Your changes have been saved.'
,
action
=
'manage_propertiesForm'
)
def
manage_delProperties
(
self
,
ids
=
None
,
REQUEST
=
None
):
"""Delete one or more properties specified by 'ids'."""
if
ids
is
None
:
return
MessageDialog
(
title
=
'No property specified'
,
message
=
'No properties were specified!'
,
action
=
'./manage_propertiesForm'
,)
propdict
=
self
.
propdict
()
vself
=
self
.
v_self
()
if
hasattr
(
vself
,
'_reserved_names'
):
nd
=
vself
.
_reserved_names
else
:
nd
=
()
for
id
in
ids
:
if
not
propdict
.
has_key
(
id
):
raise
'BadRequest'
,
(
'The property <em>%s</em> does not exist.'
%
id
)
if
(
not
'd'
in
propdict
[
id
].
get
(
'mode'
,
'wd'
))
or
(
id
in
nd
):
return
MessageDialog
(
title
=
'Cannot delete %s'
%
id
,
message
=
'The property <em>%s</em> cannot be deleted.'
%
id
,
action
=
'manage_propertiesForm'
)
self
.
delProperty
(
id
)
if
REQUEST
is
not
None
:
return
self
.
manage_propertiesForm
(
self
,
REQUEST
)
class
DefaultProperties
(
PropertySheet
):
"""The default property set mimics the behavior of old-style Zope
properties -- it stores its property values in the instance of
its owner."""
id
=
'default'
md
=
{
'xmlns'
:
'http://www.zope.org/propsets/default'
}
def
__init__
(
self
):
pass
def
v_self
(
self
):
return
self
.
aq_parent
.
aq_parent
class
DAVProperties
(
PropertySheet
):
"""WebDAV properties"""
id
=
'webdav'
md
=
{
'xmlns'
:
'DAV:'
}
pm
=
({
'id'
:
'creationdate'
,
'mode'
:
'r'
},
{
'id'
:
'displayname'
,
'mode'
:
'r'
},
{
'id'
:
'resourcetype'
,
'mode'
:
'r'
},
{
'id'
:
'getlastmodified'
,
'mode'
:
'r'
},
{
'id'
:
'getcontenttype'
,
'mode'
:
'r'
},
{
'id'
:
'getcontentlength'
,
'mode'
:
'r'
},
{
'id'
:
'source'
,
'mode'
:
'r'
},
)
def
__init__
(
self
):
pass
def
v_self
(
self
):
return
self
.
aq_parent
.
aq_parent
def
hasProperty
(
self
,
id
):
return
self
.
dav__propmap
.
has_key
(
id
)
def
getProperty
(
self
,
id
,
default
=
None
):
method
=
'dav__%s'
%
id
if
not
hasattr
(
self
,
method
):
return
default
return
getattr
(
self
,
method
)()
def
setProperty
(
self
,
id
,
value
,
type
=
'string'
,
meta
=
None
):
raise
ValueError
,
'Property cannot be set.'
def
updateProperty
(
self
,
id
,
value
):
raise
ValueError
,
'Property cannot be set.'
def
delProperty
(
self
,
id
):
raise
ValueError
,
'Property cannot be deleted.'
def
propertyMap
(
self
):
return
self
.
pm
def
dav__creationdate
(
self
):
return
''
def
dav__displayname
(
self
):
return
absattr
(
self
.
v_self
().
id
)
def
dav__resourcetype
(
self
):
self
=
self
.
v_self
()
if
hasattr
(
aq_base
(
self
),
'isAnObjectManager'
)
and
\
self
.
isAnObjectManager
:
return
'<d:collection/>'
return
''
def
dav__getlastmodified
(
self
):
self
=
self
.
v_self
()
if
hasattr
(
self
,
'_p_mtime'
):
return
rfc1123_date
(
self
.
_p_mtime
)
return
''
def
dav__getcontenttype
(
self
):
self
=
self
.
v_self
()
if
hasattr
(
self
,
'content_type'
):
return
self
.
content_type
return
''
def
dav__getcontentlength
(
self
):
self
=
self
.
v_self
()
if
hasattr
(
self
,
'get_size'
):
return
self
.
get_size
()
return
''
def
dav__source
(
self
):
self
=
self
.
v_self
()
if
hasattr
(
self
,
'meta_type'
)
and
self
.
meta_type
in
\
(
'Document'
,
'DTMLDocument'
,
'DTMLMethod'
,
'ZSQLMethod'
):
url
=
self
.
absolute_url
()
return
'<d:src>%s</d:src>
\
n
'
\
'<d:dst>%s/object_src</d:dst>'
%
(
url
,
url
)
return
''
def
dav__supportedlock
(
self
):
return
'<d:supportedlock>
\
n
'
\
'<d:lockentry>
\
n
'
\
'<d:lockscope><d:exclusive/></d:lockscope>
\
n
'
\
'<d:locktype><d:write/></d:locktype>
\
n
'
\
'</d:lockentry>
\
n
'
\
'</d:supportedlock>
\
n
'
def
dav__lockdiscovery
(
self
):
text
=
[
'<d:lockdiscovery>
\
n
'
]
for
lock
in
self
.
dav__get_locks
():
txt
=
'<d:activelock>
\
n
'
\
'<d:locktype><d:%(type)s/></d:locktype>
\
n
'
\
'<d:lockscope><d:%(scope)s/></d:lockscope>
\
n
'
\
'<d:depth>%(depth)s</d:depth>
\
n
'
\
'<d:owner>%(owner)s</d:owner>
\
n
'
\
'<d:timeout>%(timeout)s</d:timeout>
\
n
'
\
'<d:locktoken>
\
n
'
\
'<d:href>opaquelocktoken:%(token)s</d:href>
\
n
'
\
'</d:locktoken>
\
n
'
\
'</d:activelock>
\
n
'
%
lock
.
__dict__
text
.
append
(
lock
.
dav__activelock
())
text
.
append
(
'</d:lockdiscovery>
\
n
'
)
return
string
.
join
(
text
,
''
)
class
PropertySheets
(
Implicit
):
"""A tricky container to keep property sets from polluting
an object's direct attribute namespace."""
id
=
'propertysheets'
default
=
DefaultProperties
()
webdav
=
DAVProperties
()
def
__init__
(
self
,
parent
):
pass
def
__propsets__
(
self
):
propsets
=
self
.
aq_parent
.
__propsets__
return
(
self
.
default
,
self
.
webdav
)
+
propsets
def
__bobo_traverse__
(
self
,
REQUEST
,
name
=
None
):
for
propset
in
self
.
__propsets__
():
if
propset
.
id
==
name
:
return
propset
.
__of__
(
self
)
raise
'Not Found'
,
name
def
__getitem__
(
self
,
n
):
return
self
.
__propsets__
()[
n
].
__of__
(
self
)
def
get
(
self
,
name
,
default
=
None
):
for
propset
in
self
.
__propsets__
():
if
propset
.
id
==
name
or
propset
.
xml_namespace
()
==
name
:
return
propset
.
__of__
(
self
)
return
default
def
manage_addPropertySheet
(
self
,
id
,
ns
):
""" """
md
=
{
'xmlns'
:
ns
}
ps
=
PropertySet
(
id
,
md
)
self
.
addPropertySet
(
ps
)
return
'OK'
def
addPropertySheet
(
self
,
propset
):
propsets
=
self
.
aq_parent
.
__propsets__
propsets
=
propsets
+
(
propset
,)
self
.
aq_parent
.
__propsets__
=
propsets
def
delPropertySheet
(
self
,
name
):
result
=
[]
for
propset
in
self
.
aq_parent
.
__propsets__
:
if
propset
.
id
!=
name
and
propset
.
xml_namespace
()
!=
name
:
result
.
append
(
propset
)
self
.
aq_parent
.
__propsets__
=
tuple
(
result
)
def
__len__
(
self
):
return
len
(
self
.
__propsets__
())
class
vps
(
Base
):
# The vps object implements a "computed attribute" - it ensures
# that a PropertySheets instance is returned when the propertysheets
# attribute of a PropertyManager is accessed.
def
__init__
(
self
,
c
=
PropertySheets
):
self
.
c
=
c
def
__of__
(
self
,
parent
):
if
hasattr
(
parent
,
'aq_base'
):
parent
=
parent
.
aq_base
return
self
.
c
(
parent
)
def
absattr
(
attr
):
if
callable
(
attr
):
return
attr
()
return
attr
def
aq_base
(
ob
):
if
hasattr
(
ob
,
'aq_base'
):
return
ob
.
aq_base
return
ob
def
rfc1123_date
(
ts
=
None
):
# Return an RFC 1123 format date string, required for
# use in HTTP Date headers per the HTTP 1.1 spec.
if
ts
is
None
:
ts
=
time
.
time
()
ts
=
time
.
asctime
(
time
.
gmtime
(
ts
))
ts
=
string
.
split
(
ts
)
return
'%s, %s %s %s %s GMT'
%
(
ts
[
0
],
ts
[
2
],
ts
[
1
],
ts
[
3
],
ts
[
4
])
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