Page Menu
Home
Phabricator
Search
Configure Global Search
Log In
Files
F7159872
pidf.py
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Flag For Later
Size
16 KB
Referenced Files
None
Subscribers
None
pidf.py
View Options
"""PIDF handling according to RFC3863 and RFC4479"""
__all__
=
[
'pidf_namespace'
,
'dm_namespace'
,
'PIDFDocument'
,
'ServiceExtension'
,
'DeviceExtension'
,
'PersonExtension'
,
'StatusExtension'
,
'Note'
,
'DeviceID'
,
'Status'
,
'Basic'
,
'Contact'
,
'ServiceTimestamp'
,
'Service'
,
'DeviceTimestamp'
,
'Device'
,
'PersonTimestamp'
,
'Person'
,
'PIDF'
,
# Extensions
'ExtendedStatus'
,
'StatusType'
,
'DeviceInfo'
]
from
application.python.weakref
import
weakobjectmap
from
sipsimple.payloads
import
ValidationError
,
XMLDocument
,
XMLListRootElement
,
XMLListElement
,
XMLElement
,
XMLAttribute
,
XMLElementID
,
XMLElementChild
from
sipsimple.payloads
import
XMLStringElement
,
XMLLocalizedStringElement
,
XMLDateTimeElement
,
XMLAnyURIElement
from
sipsimple.payloads.datatypes
import
AnyURI
,
ID
,
DateTime
pidf_namespace
=
'urn:ietf:params:xml:ns:pidf'
dm_namespace
=
'urn:ietf:params:xml:ns:pidf:data-model'
class
PIDFDocument
(
XMLDocument
):
content_type
=
'application/pidf+xml'
PIDFDocument
.
register_namespace
(
pidf_namespace
,
prefix
=
None
,
schema
=
'pidf.xsd'
)
PIDFDocument
.
register_namespace
(
dm_namespace
,
prefix
=
'dm'
,
schema
=
'data-model.xsd'
)
## Marker mixin
class
ServiceExtension
(
object
):
pass
class
ServiceItemExtension
(
object
):
pass
class
DeviceExtension
(
object
):
pass
class
PersonExtension
(
object
):
pass
class
StatusExtension
(
object
):
pass
## Attribute value types
class
BasicStatusValue
(
str
):
def
__new__
(
cls
,
value
):
if
value
not
in
(
'closed'
,
'open'
):
raise
ValueError
(
'illegal BasicStatusValue'
)
return
str
.
__new__
(
cls
,
value
)
## General elements
class
Note
(
str
):
def
__new__
(
cls
,
value
,
lang
=
None
):
instance
=
str
.
__new__
(
cls
,
value
)
instance
.
lang
=
lang
return
instance
def
__repr__
(
self
):
return
"
%s
(
%s
, lang=
%r
)"
%
(
self
.
__class__
.
__name__
,
str
.
__repr__
(
self
),
self
.
lang
)
def
__eq__
(
self
,
other
):
if
isinstance
(
other
,
Note
):
return
str
.
__eq__
(
self
,
other
)
and
self
.
lang
==
other
.
lang
elif
isinstance
(
other
,
str
):
return
self
.
lang
is
None
and
str
.
__eq__
(
self
,
other
)
else
:
return
NotImplemented
def
__ne__
(
self
,
other
):
equal
=
self
.
__eq__
(
other
)
return
NotImplemented
if
equal
is
NotImplemented
else
not
equal
class
PIDFNote
(
XMLLocalizedStringElement
):
_xml_tag
=
'note'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
def
__unicode__
(
self
):
return
Note
(
self
.
value
,
self
.
lang
)
class
DMNote
(
XMLLocalizedStringElement
):
_xml_tag
=
'note'
_xml_namespace
=
dm_namespace
_xml_document
=
PIDFDocument
def
__unicode__
(
self
):
return
Note
(
self
.
value
,
self
.
lang
)
class
NoteMap
(
object
):
"""Descriptor to be used for _note_map attributes on XML elements with notes"""
def
__init__
(
self
):
self
.
object_map
=
weakobjectmap
()
def
__get__
(
self
,
obj
,
type
):
if
obj
is
None
:
return
self
try
:
return
self
.
object_map
[
obj
]
except
KeyError
:
return
self
.
object_map
.
setdefault
(
obj
,
{})
def
__set__
(
self
,
obj
,
value
):
raise
AttributeError
(
"cannot set attribute"
)
def
__delete__
(
self
,
obj
):
raise
AttributeError
(
"cannot delete attribute"
)
class
NoteList
(
object
):
def
__init__
(
self
,
xml_element
,
note_type
):
self
.
xml_element
=
xml_element
self
.
note_type
=
note_type
def
__contains__
(
self
,
item
):
if
isinstance
(
item
,
Note
):
item
=
self
.
note_type
(
item
,
item
.
lang
)
elif
isinstance
(
item
,
str
):
item
=
self
.
note_type
(
item
)
return
item
in
iter
(
list
(
self
.
xml_element
.
_note_map
.
values
()))
def
__iter__
(
self
):
return
(
str
(
self
.
xml_element
.
_note_map
[
element
])
for
element
in
self
.
xml_element
.
element
if
element
in
self
.
xml_element
.
_note_map
)
def
__len__
(
self
):
return
len
(
self
.
xml_element
.
_note_map
)
def
__eq__
(
self
,
other
):
if
isinstance
(
other
,
NoteList
):
return
self
is
other
or
(
len
(
self
)
==
len
(
other
)
and
all
(
self_item
==
other_item
for
self_item
,
other_item
in
zip
(
self
,
other
)))
else
:
return
NotImplemented
def
__ne__
(
self
,
other
):
equal
=
self
.
__eq__
(
other
)
return
NotImplemented
if
equal
is
NotImplemented
else
not
equal
def
_parse_element
(
self
,
element
):
self
.
xml_element
.
_note_map
.
clear
()
for
child
in
element
:
if
child
.
tag
==
self
.
note_type
.
qname
:
try
:
note
=
self
.
note_type
.
from_element
(
child
,
xml_document
=
self
.
xml_element
.
_xml_document
)
except
ValidationError
:
pass
else
:
self
.
xml_element
.
_note_map
[
note
.
element
]
=
note
def
_build_element
(
self
):
for
note
in
list
(
self
.
xml_element
.
_note_map
.
values
()):
note
.
to_element
()
def
add
(
self
,
item
):
if
isinstance
(
item
,
Note
):
item
=
self
.
note_type
(
item
,
item
.
lang
)
elif
isinstance
(
item
,
str
):
item
=
self
.
note_type
(
item
)
if
type
(
item
)
is
not
self
.
note_type
:
raise
TypeError
(
"
%s
cannot add notes of type
%s
"
%
(
self
.
xml_element
.
__class__
.
__name__
,
item
.
__class__
.
__name__
))
self
.
xml_element
.
_insert_element
(
item
.
element
)
self
.
xml_element
.
_note_map
[
item
.
element
]
=
item
self
.
xml_element
.
__dirty__
=
True
def
remove
(
self
,
item
):
if
isinstance
(
item
,
Note
):
try
:
item
=
next
((
entry
for
entry
in
list
(
self
.
xml_element
.
_note_map
.
values
())
if
str
(
entry
)
==
item
))
except
StopIteration
:
raise
KeyError
(
item
)
elif
isinstance
(
item
,
str
):
try
:
item
=
next
((
entry
for
entry
in
list
(
self
.
xml_element
.
_note_map
.
values
())
if
entry
==
item
))
except
StopIteration
:
raise
KeyError
(
item
)
if
type
(
item
)
is
not
self
.
note_type
:
raise
KeyError
(
item
)
self
.
xml_element
.
element
.
remove
(
item
.
element
)
del
self
.
xml_element
.
_note_map
[
item
.
element
]
self
.
xml_element
.
__dirty__
=
True
def
update
(
self
,
sequence
):
for
item
in
sequence
:
self
.
add
(
item
)
def
clear
(
self
):
for
item
in
list
(
self
.
xml_element
.
_note_map
.
values
()):
self
.
remove
(
item
)
class
DeviceID
(
XMLStringElement
):
_xml_tag
=
'deviceID'
_xml_namespace
=
dm_namespace
_xml_document
=
PIDFDocument
## Service elements
class
Basic
(
XMLStringElement
):
_xml_tag
=
'basic'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
_xml_value_type
=
BasicStatusValue
class
Status
(
XMLElement
):
_xml_tag
=
'status'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
_xml_extension_type
=
StatusExtension
_xml_children_order
=
{
Basic
.
qname
:
0
}
basic
=
XMLElementChild
(
'basic'
,
type
=
Basic
,
required
=
False
,
test_equal
=
True
)
def
__init__
(
self
,
basic
=
None
):
XMLElement
.
__init__
(
self
)
self
.
basic
=
basic
def
check_validity
(
self
):
if
len
(
self
.
element
)
==
0
:
raise
ValidationError
(
"Status objects must have at least one child"
)
super
(
Status
,
self
)
.
check_validity
()
def
__repr__
(
self
):
return
'
%s
(
%r
)'
%
(
self
.
__class__
.
__name__
,
self
.
basic
)
class
Contact
(
XMLAnyURIElement
):
_xml_tag
=
'contact'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
priority
=
XMLAttribute
(
'priority'
,
type
=
float
,
required
=
False
,
test_equal
=
False
)
class
ServiceTimestamp
(
XMLDateTimeElement
):
_xml_tag
=
'timestamp'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
class
Service
(
XMLListElement
):
_xml_tag
=
'tuple'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
_xml_extension_type
=
ServiceExtension
_xml_item_type
=
(
DeviceID
,
ServiceItemExtension
)
_xml_children_order
=
{
Status
.
qname
:
0
,
None
:
1
,
Contact
.
qname
:
2
,
PIDFNote
.
qname
:
3
,
ServiceTimestamp
.
qname
:
4
}
id
=
XMLElementID
(
'id'
,
type
=
ID
,
required
=
True
,
test_equal
=
True
)
status
=
XMLElementChild
(
'status'
,
type
=
Status
,
required
=
True
,
test_equal
=
True
)
contact
=
XMLElementChild
(
'contact'
,
type
=
Contact
,
required
=
False
,
test_equal
=
True
)
timestamp
=
XMLElementChild
(
'timestamp'
,
type
=
ServiceTimestamp
,
required
=
False
,
test_equal
=
True
)
_note_map
=
NoteMap
()
def
__init__
(
self
,
id
,
notes
=
[],
status
=
None
,
contact
=
None
,
timestamp
=
None
):
XMLListElement
.
__init__
(
self
)
self
.
id
=
id
self
.
status
=
status
self
.
contact
=
contact
self
.
timestamp
=
timestamp
self
.
notes
.
update
(
notes
)
@property
def
notes
(
self
):
return
NoteList
(
self
,
PIDFNote
)
def
__eq__
(
self
,
other
):
if
isinstance
(
other
,
Service
):
return
super
(
Service
,
self
)
.
__eq__
(
other
)
and
self
.
notes
==
other
.
notes
else
:
return
self
.
id
==
other
def
__repr__
(
self
):
return
'
%s
(
%r
,
%r
,
%r
,
%r
,
%r
)'
%
(
self
.
__class__
.
__name__
,
self
.
id
,
list
(
self
.
notes
),
self
.
status
,
self
.
contact
,
self
.
timestamp
)
def
_parse_element
(
self
,
element
):
super
(
Service
,
self
)
.
_parse_element
(
element
)
self
.
notes
.
_parse_element
(
element
)
def
_build_element
(
self
):
super
(
Service
,
self
)
.
_build_element
()
self
.
notes
.
_build_element
()
class
DeviceTimestamp
(
XMLDateTimeElement
):
_xml_tag
=
'timestamp'
_xml_namespace
=
dm_namespace
_xml_document
=
PIDFDocument
class
Device
(
XMLElement
):
_xml_tag
=
'device'
_xml_namespace
=
dm_namespace
_xml_document
=
PIDFDocument
_xml_extension_type
=
DeviceExtension
_xml_children_order
=
{
None
:
0
,
DeviceID
.
qname
:
1
,
DMNote
.
qname
:
2
,
DeviceTimestamp
.
qname
:
3
}
id
=
XMLElementID
(
'id'
,
type
=
ID
,
required
=
True
,
test_equal
=
True
)
device_id
=
XMLElementChild
(
'device_id'
,
type
=
DeviceID
,
required
=
False
,
test_equal
=
True
)
timestamp
=
XMLElementChild
(
'timestamp'
,
type
=
DeviceTimestamp
,
required
=
False
,
test_equal
=
True
)
_note_map
=
NoteMap
()
def
__init__
(
self
,
id
,
device_id
=
None
,
notes
=
[],
timestamp
=
None
):
XMLElement
.
__init__
(
self
)
self
.
id
=
id
self
.
device_id
=
device_id
self
.
timestamp
=
timestamp
self
.
notes
.
update
(
notes
)
@property
def
notes
(
self
):
return
NoteList
(
self
,
DMNote
)
def
__eq__
(
self
,
other
):
if
isinstance
(
other
,
Device
):
return
super
(
Device
,
self
)
.
__eq__
(
other
)
and
self
.
notes
==
other
.
notes
else
:
return
self
.
id
==
other
def
__repr__
(
self
):
return
'
%s
(
%r
,
%r
,
%r
,
%r
)'
%
(
self
.
__class__
.
__name__
,
self
.
id
,
self
.
device_id
,
list
(
self
.
notes
),
self
.
timestamp
)
def
_parse_element
(
self
,
element
):
super
(
Device
,
self
)
.
_parse_element
(
element
)
self
.
notes
.
_parse_element
(
element
)
def
_build_element
(
self
):
super
(
Device
,
self
)
.
_build_element
()
self
.
notes
.
_build_element
()
class
PersonTimestamp
(
XMLDateTimeElement
):
_xml_tag
=
'timestamp'
_xml_namespace
=
dm_namespace
_xml_document
=
PIDFDocument
class
Person
(
XMLElement
):
_xml_tag
=
'person'
_xml_namespace
=
dm_namespace
_xml_document
=
PIDFDocument
_xml_extension_type
=
PersonExtension
_xml_children_order
=
{
None
:
0
,
DMNote
.
qname
:
1
,
PersonTimestamp
.
qname
:
2
}
id
=
XMLElementID
(
'id'
,
type
=
ID
,
required
=
True
,
test_equal
=
True
)
timestamp
=
XMLElementChild
(
'timestamp'
,
type
=
PersonTimestamp
,
required
=
False
,
test_equal
=
True
)
_note_map
=
NoteMap
()
def
__init__
(
self
,
id
,
notes
=
[],
timestamp
=
None
):
XMLElement
.
__init__
(
self
)
self
.
id
=
id
self
.
timestamp
=
timestamp
self
.
notes
.
update
(
notes
)
@property
def
notes
(
self
):
return
NoteList
(
self
,
DMNote
)
def
__eq__
(
self
,
other
):
if
isinstance
(
other
,
Person
):
return
super
(
Person
,
self
)
.
__eq__
(
other
)
and
self
.
notes
==
other
.
notes
else
:
return
self
.
id
==
other
def
__repr__
(
self
):
return
'
%s
(
%r
,
%r
,
%r
)'
%
(
self
.
__class__
.
__name__
,
self
.
id
,
list
(
self
.
notes
),
self
.
timestamp
)
def
_parse_element
(
self
,
element
):
super
(
Person
,
self
)
.
_parse_element
(
element
)
self
.
notes
.
_parse_element
(
element
)
def
_build_element
(
self
):
super
(
Person
,
self
)
.
_build_element
()
self
.
notes
.
_build_element
()
class
PIDF
(
XMLListRootElement
):
_xml_tag
=
'presence'
_xml_namespace
=
pidf_namespace
_xml_document
=
PIDFDocument
_xml_children_order
=
{
Service
.
qname
:
0
,
PIDFNote
.
qname
:
1
,
Person
.
qname
:
2
,
Device
.
qname
:
3
}
_xml_item_type
=
(
Service
,
PIDFNote
,
Person
,
Device
)
entity
=
XMLAttribute
(
'entity'
,
type
=
AnyURI
,
required
=
True
,
test_equal
=
True
)
services
=
property
(
lambda
self
:
(
item
for
item
in
self
if
type
(
item
)
is
Service
))
notes
=
property
(
lambda
self
:
(
item
for
item
in
self
if
type
(
item
)
is
Note
))
persons
=
property
(
lambda
self
:
(
item
for
item
in
self
if
type
(
item
)
is
Person
))
devices
=
property
(
lambda
self
:
(
item
for
item
in
self
if
type
(
item
)
is
Device
))
def
__init__
(
self
,
entity
,
elements
=
[]):
XMLListRootElement
.
__init__
(
self
)
self
.
entity
=
entity
self
.
update
(
elements
)
def
__contains__
(
self
,
item
):
if
isinstance
(
item
,
Note
):
item
=
PIDFNote
(
item
,
item
.
lang
)
return
super
(
PIDF
,
self
)
.
__contains__
(
item
)
def
__iter__
(
self
):
return
(
str
(
item
)
if
type
(
item
)
is
PIDFNote
else
item
for
item
in
super
(
PIDF
,
self
)
.
__iter__
())
def
__repr__
(
self
):
return
'
%s
(
%r
,
%r
)'
%
(
self
.
__class__
.
__name__
,
self
.
entity
,
list
(
self
))
def
add
(
self
,
item
):
if
isinstance
(
item
,
Note
):
item
=
PIDFNote
(
item
,
item
.
lang
)
super
(
PIDF
,
self
)
.
add
(
item
)
def
remove
(
self
,
item
):
if
isinstance
(
item
,
Note
):
try
:
item
=
next
((
entry
for
entry
in
super
(
PIDF
,
self
)
.
__iter__
()
if
type
(
entry
)
is
PIDFNote
and
str
(
entry
)
==
item
))
except
StopIteration
:
raise
KeyError
(
item
)
super
(
PIDF
,
self
)
.
remove
(
item
)
#
# Extensions
#
agp_pidf_namespace
=
'urn:ag-projects:xml:ns:pidf'
PIDFDocument
.
register_namespace
(
agp_pidf_namespace
,
prefix
=
'agp-pidf'
)
class
ExtendedStatusValue
(
str
):
def
__new__
(
cls
,
value
):
if
value
not
in
(
'available'
,
'offline'
,
'away'
,
'busy'
):
raise
ValueError
(
"illegal value for extended status"
)
return
str
.
__new__
(
cls
,
value
)
class
ExtendedStatus
(
XMLStringElement
,
StatusExtension
):
_xml_tag
=
'extended'
_xml_namespace
=
agp_pidf_namespace
_xml_document
=
PIDFDocument
_xml_value_type
=
ExtendedStatusValue
class
StatusType
(
XMLStringElement
,
StatusExtension
):
_xml_tag
=
'type'
_xml_namespace
=
agp_pidf_namespace
_xml_document
=
PIDFDocument
Status
.
register_extension
(
'extended'
,
type
=
ExtendedStatus
)
Status
.
register_extension
(
'type'
,
type
=
StatusType
)
class
Description
(
XMLStringElement
):
_xml_tag
=
'description'
_xml_namespace
=
agp_pidf_namespace
_xml_document
=
PIDFDocument
class
UserAgent
(
XMLStringElement
):
_xml_tag
=
'user-agent'
_xml_namespace
=
agp_pidf_namespace
_xml_document
=
PIDFDocument
class
TimeOffset
(
XMLStringElement
):
_xml_tag
=
'time-offset'
_xml_namespace
=
agp_pidf_namespace
_xml_document
=
PIDFDocument
description
=
XMLAttribute
(
'description'
,
type
=
str
,
required
=
False
,
test_equal
=
True
)
def
__init__
(
self
,
value
=
None
,
description
=
None
):
if
value
is
None
:
value
=
DateTime
.
now
()
.
utcoffset
()
.
seconds
/
60
XMLStringElement
.
__init__
(
self
,
str
(
value
))
self
.
description
=
description
def
__int__
(
self
):
return
int
(
self
.
value
)
class
DeviceInfo
(
XMLElement
,
ServiceExtension
):
_xml_tag
=
'device-info'
_xml_namespace
=
agp_pidf_namespace
_xml_document
=
PIDFDocument
_xml_children_order
=
{
Description
.
qname
:
0
,
UserAgent
.
qname
:
1
}
id
=
XMLElementID
(
'id'
,
type
=
str
,
required
=
True
,
test_equal
=
True
)
description
=
XMLElementChild
(
'description'
,
type
=
Description
,
required
=
False
,
test_equal
=
True
)
user_agent
=
XMLElementChild
(
'user_agent'
,
type
=
UserAgent
,
required
=
False
,
test_equal
=
True
)
time_offset
=
XMLElementChild
(
'time_offset'
,
type
=
TimeOffset
,
required
=
False
,
test_equal
=
True
)
def
__init__
(
self
,
id
,
description
=
None
,
user_agent
=
None
,
time_offset
=
None
):
XMLElement
.
__init__
(
self
)
self
.
id
=
id
self
.
description
=
description
self
.
user_agent
=
user_agent
self
.
time_offset
=
time_offset
def
__repr__
(
self
):
return
'
%s
(
%r
,
%r
,
%r
,
%r
)'
%
(
self
.
__class__
.
__name__
,
self
.
id
,
self
.
description
,
self
.
user_agent
,
self
.
time_offset
)
Service
.
register_extension
(
'device_info'
,
type
=
DeviceInfo
)
File Metadata
Details
Attached
Mime Type
text/x-python
Expires
Sat, Nov 23, 2:18 PM (1 d, 5 h)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
3409225
Default Alt Text
pidf.py (16 KB)
Attached To
Mode
rPYNSIPSIMPLE python3-sipsimple
Attached
Detach File
Event Timeline
Log In to Comment