Switch branches/tags
Nothing to show
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
executable file 416 lines (343 sloc) 14.4 KB
#!/usr/bin/env python
# -*- coding: iso-8859-1 -*-
"""Several base datatypes that inherit from native types
(unicode,list,dict, etc) or python defined types (datetime), but adds
support for general properties. The properties are set when
instansiated (in the constructor call, as named arguments). Once an
object has been instansiated, you cannot add any more properties."""
import datetime, re
import xml.etree.cElementTree as ET
class AbstractStructure(object):
def __new__(cls):
obj = super(AbstractStructure,cls).__new__(cls)
return obj
def __init__(self, *args, **kwargs):
for (key,val) in kwargs.items():
# Declare this instance ready for usage. Note that derived
# objects must do their own initialization first, before
# calling the superclass constructor (i.e. this function),
# since this effectively "seals" the instance.
# (we need to call object.__setattr__ directly to bypass our
# own __setattr__ implementation)
object.__setattr__(self, '__initialized', True)
def __setattr__(self,name,value):
if object.__getattribute__(self,'__initialized'):
# initialization phase is over -- no new attributes should
# be created. Check to see if the attribute exists -- if it
# doesn't, we raise an AttributeError (with a sensible
# error message)
except AttributeError:
raise AttributeError("Can't set attribute '%s' on object '%s' after initialization" % (name, self.__class__.__name__))
# Still in initialization phase -- ok to create new
# attributes
class UnicodeStructure(AbstractStructure,unicode):
"""En UnicodeStructure är bärare av ett värde (en textsträng av
något slag). Den kan även ha andra egenskaper (ordningsnummer,
ikraftträdandedatum etc)."""
# immutable objects (like strings, unicode, etc) must provide a __new__ method
def __new__(cls,arg=u'', *args, **kwargs):
if not isinstance(arg,unicode):
raise TypeError("%r is not unicode" % arg)
obj = unicode.__new__(cls, arg)
return obj
#def __init__(self, *args, **kwargs):
# print "UnicodeStructure.__init__ called"
# super(UnicodeStructure,self).__init__(*args, **kwargs)
class IntStructure(AbstractStructure,int):
"""En IntStructure är en int som även kan ha andra egenskaper
(ordningsnummer, ikraftträdandedatum etc)."""
# immutable objects must provide a __new__ method
def __new__(cls,arg=0, *args, **kwargs):
if not isinstance(arg,int):
raise TypeError("%r is not int" % arg)
obj = int.__new__(cls, arg)
return obj
class DateStructure(AbstractStructure,
"""En DateStructure är ett som även kan ha andra
egenskaper (ordningsnummer, ikraftträdandedatum etc)."""
# immutable objects must provide a __new__ method
def __new__(cls,, *args, **kwargs):
if not isinstance(arg,
raise TypeError("%r is not" % arg)
obj =, arg.year, arg.month,
return obj
class CompoundStructure(AbstractStructure,list):
"""En CompoundStructure fungerar som en lista av andra Structureobjekt.
Den kan också ha egna egenskaper (kapitelrubrik, paragrafnummer etc)"""
def __new__(cls,arg=[], *args, **kwargs):
# ideally, we'd like to do just "obj = list.__new__(cls,arg)"
# but that doesn't seem to work
obj = list.__new__(cls)
return obj
class MapStructure(AbstractStructure,dict):
"""En MapStructure är en anpassad map/dictionary"""
def __new__(cls,arg={}, *args, **kwargs):
# ideally, we'd like to do just "obj = dict.__new__(cls,arg)"
# but that doesn't seem to work
obj = dict.__new__(cls, arg)
return obj
# Abstrakta klasser avsedda att användas vid multipelt arv, som
# lägger till vanliga egenskaper
class TemporalStructure(object):
"""En TemporalStructure har ett antal olika tidsegenskaper
(ikraftträder, upphör, meddelas) som anger de tidsmässiga ramarna
för vad det nu är frågan om"""
def __init__(self):
self.entryintoforce = None
self.expires = None
def in_effect(self,date=None):
if not date:
date =
return (date >= self.entryintoforce) and (date <= self.expires)
class OrdinalStructure(object):
"""En OrdinalStructure har ett explicit
ordningsnummer. Ordningsnumret behöver inte vara strikt numeriskt,
utan kan även vara exv '6 a', när lagstiftaren tyckt sig behöva en
ny paragraf mellan 6 och 7"""
def __init__(self):
self.ordinal = None
# FIXME: gör en ordentlig smart mestadels-numerisk jämförelse ("2 a" < "20", "2" < "10")
def __lt__(self,other):
return self.ordinal < other.ordinal
def __le__(self,other):
return self.ordinal <= other.ordinal
def __eq__(self,other):
return self.ordinal == other.ordinal
def __ne__(self,other):
return self.ordinal != other.ordinal
def __gt__(self,other):
return self.ordinal > other.ordinal
def __ge__(self,other):
return self.ordinal == other.ordinal
import Util
class PredicateType(object):
"""Inheriting from this gives the subclass a predicate attribute,
which describes the RDF predicate to which the class is the RDF
subject (eg. if you want to model the title of a document, you
would inherit from UnicodeStructure and this, and then set
.predicate to
def __init__(self, *args, **kwargs):
if 'predicate' in kwargs:
self.predicate = kwargs['predicate']
# switch the full uriref
# (http://rinfo.lagrummet...#paragraf) to one using a
# namespace prefix, if we know of one:
shorten = False
for (prefix, ns) in Util.ns.items():
if kwargs['predicate'].startswith(ns):
predicateuri = kwargs['predicate']
kwargs['predicate'] = kwargs['predicate'].replace(ns, prefix+":")
# print "Shorten predicate %s to: %s" % (predicateuri, kwargs['predicate'])
shorten = True
#if not shorten:
# print "Couldn't shorten predicate: %s" % self.predicate
# From the RDF Schema spec: 'This is the class of
# everything. All other classes are subclasses of this
# class.'
from rdflib import RDFS
self.predicate = RDFS.Resource
super(PredicateType,self).__init__(*args, **kwargs)
def serialize(root):
t = __serializeNode(root)
return ET.tostring(t,'utf-8').decode('utf-8')
def __indentTree(elem, level=0):
i = "\r\n" + level*" "
if len(elem):
if not elem.text or not elem.text.strip():
elem.text = i + " "
for e in elem:
indent(e, level+1)
if not e.tail or not e.tail.strip():
e.tail = i + " "
if not e.tail or not e.tail.strip():
e.tail = i
if level and (not elem.tail or not elem.tail.strip()):
elem.tail = i
def __serializeNode(node):
# print "serializing: %s" % node.__class__.__name__
e = ET.Element(node.__class__.__name__)
if hasattr(node,'__dict__'):
for key in [x for x in node.__dict__.keys() if not x.startswith('__')]:
val = node.__dict__[key]
if (isinstance(val,unicode) or isinstance(val,str)):
if isinstance(node,unicode):
if node:
e.text = node
elif isinstance(node,str):
if str:
e.text = node
elif isinstance(node,int):
e.text = unicode(node)
elif isinstance(node,list):
for x in node:
elif isinstance(node,dict):
for x in node.keys():
k = ET.Element("Key")
v = ET.Element("Value")
e.text = repr(node)
# raise TypeError("Can't serialize %r (%r)" % (type(node), node))
return e
def deserialize(xmlstr, caller_globals):
"""This function is highly insecure -- use only with trusted data"""
# print "Caller globals()"
# print repr(caller_globals.keys())
# print "Callee globals()"
# print repr(globals().keys())
# print repr(locals().keys())
if (isinstance(xmlstr,unicode)):
xmlstr = xmlstr.encode('utf-8')
t = ET.fromstring(xmlstr)
return __deserializeNode(t,caller_globals)
def __deserializeNode(elem,caller_globals):
# print "element %r, attrs %r" % (elem.tag, elem.attrib)
#kwargs = elem.attrib specialcasing first -- classobjects for
# these native objects can't be created by the"caller_globals[elem.tag]" call below
if elem.tag == 'int':
i = 0;
classobj = i.__class__
elif elem.tag == 'str':
i = ''
classobj = i.__class__
elif elem.tag == 'unicode':
i = u''
classobj = i.__class__
# print "creating classobj for %s" % elem.tag
classobj = caller_globals[elem.tag]
testclass = classobj(**elem.attrib)
if isinstance(testclass, str):
c = classobj(str(elem.text),**elem.attrib)
elif isinstance(classobj(**elem.attrib), int):
c = classobj(int(elem.text),**elem.attrib)
elif isinstance(testclass, unicode):
if elem.text:
c = classobj(unicode(elem.text),**elem.attrib)
c = classobj(**elem.attrib)
elif isinstance(testclass,
m = re.match(r'\w+\((\d+), (\d+), (\d+)\)',elem.text)
basedate =,int(,int(
c = classobj(basedate,**elem.attrib)
elif isinstance(testclass, dict):
c = classobj(**elem.attrib)
# FIXME: implement this
c = classobj(**elem.attrib)
for subelem in elem:
# print "Recursing"
c.append(__deserializeNode(subelem, caller_globals))
return c
# in-place prettyprint formatter
def indent(elem, level=0):
i = "\n" + level*" "
if len(elem):
if not elem.text or not elem.text.strip():
elem.text = i + " "
for elem in elem:
indent(elem, level+1)
if not elem.tail or not elem.tail.strip():
elem.tail = i
if level and (not elem.tail or not elem.tail.strip()):
elem.tail = i
# Exempel på andra mixinklasser och ärvda klasser
class EvenMixin():
def iseven(self):
return (len(self.keyword) % 2 == 0)
class DerivedUnicode(UnicodeStructure, EvenMixin):
# an example on how to customize object initialization, while still
# letting the base class do it's initialization
def __init__(self, *args, **kwargs):
if kwargs['keyword']:
self.keyword = kwargs['keyword'].upper()
del kwargs['keyword']
super(DerivedUnicode,self).__init__(*args, **kwargs)
class DerivedList(CompoundStructure, EvenMixin): pass
class DerivedDict(MapStructure, EvenMixin): pass
class DerivedInt(IntStructure, EvenMixin): pass
class DerivedDate(DateStructure, EvenMixin): pass
class RDFString(PredicateType,UnicodeStructure):
# N.B: if we inherit from (UnicodeStructure,PredicateType)
# instead, PredicateType.__init__ never gets called! But this way,
# AbstractStructure.__init__ never gets called. I think i must
# read descrintro again...
class UnicodeSubject(PredicateType,UnicodeStructure): pass
class Stycke(CompoundStructure): pass
class Paragraph(CompoundStructure): pass
class Sektion(CompoundStructure): pass
class Section(CompoundStructure): pass
if __name__ == '__main__':
# print "Testing DerivedUnicode"
u = DerivedUnicode(u'blahonga', keyword=u'myunicode')
# print "\trepr(u): %s" % repr(u)
# print "\tu[1:4]: %r" % u[1:4]
# print "\tu.keyword: %r" % u.keyword
# print "\tu.iseven: %r" % u.iseven()
# print "Testing DerivedList"
l = DerivedList(['x','y','z'], keyword=u'mylist')
# print "\tl[1]: %r" % l[1]
# print "\tl.keyword: %r" % l.keyword
# print "\tl.iseven: %r" % l.iseven()
# print "Testing DerivedDict"
d = DerivedDict({'a':'foo','b':'bar'}, keyword=u'mydict')
# print "\td['a']: %r" % d['a']
# print "\td.keyword: %r" % d.keyword
# print "\td.iseven: %r" % d.iseven()
# print "Testing DerivedInt"
i = DerivedInt(42, keyword=u'myint')
# print "\ti: %r" % i
# print "\ti+5: %r" % (i+5)
# print "\ti.keyword: %r" % d.keyword
# print "\ti.iseven: %r" % d.iseven()
# print "Testing DerivedDate"
nativedate =,3,15)
dt = DerivedDate(nativedate, keyword=u'mydate')
# print "\tdt: %r" % dt
# print "\tdt.keyword: %r" % dt.keyword
# print "\tdt.iseven: %r" % dt.iseven()
# print "Testing RDFString"
r = RDFString(u'Typisk dokumentrubrik', keyword=u'mysubject')
# print "\trepr(r): %s" % repr(r)
# print "\tr[1:4]: %r" % r[1:4]
# print "\tr.keyword: %r" % r.keyword
# print "\tr.predicate: %r" % r.predicate
from rdflib import URIRef
r.predicate = URIRef('')
# print "\tr.predicate: %r" % r.predicate
c = DerivedList([u,l,d,i,dt,r])
x = serialize(c)
print x
y = deserialize(x,globals())
print serialize(y)