Mercurial > genshi > mirror
view markup/core.py @ 71:db7c76d337aa trunk
Add back line that was accidentially left commented out in [75].
author | cmlenz |
---|---|
date | Tue, 11 Jul 2006 17:41:47 +0000 |
parents | c40a5dcd2b55 |
children | 1da51d718391 |
line wrap: on
line source
# -*- coding: utf-8 -*- # # Copyright (C) 2006 Edgewall Software # All rights reserved. # # This software is licensed as described in the file COPYING, which # you should have received as part of this distribution. The terms # are also available at http://markup.edgewall.org/wiki/License. # # This software consists of voluntary contributions made by many # individuals. For the exact contribution history, see the revision # history and logs, available at http://markup.edgewall.org/log/. """Core classes for markup processing.""" import htmlentitydefs import re from StringIO import StringIO __all__ = ['Stream', 'Markup', 'escape', 'unescape', 'Namespace', 'QName'] class StreamEventKind(str): """A kind of event on an XML stream.""" class Stream(object): """Represents a stream of markup events. This class is basically an iterator over the events. Also provided are ways to serialize the stream to text. The `serialize()` method will return an iterator over generated strings, while `render()` returns the complete generated text at once. Both accept various parameters that impact the way the stream is serialized. Stream events are tuples of the form: (kind, data, position) where `kind` is the event kind (such as `START`, `END`, `TEXT`, etc), `data` depends on the kind of event, and `position` is a `(line, offset)` tuple that contains the location of the original element or text in the input. """ __slots__ = ['events'] START = StreamEventKind('START') # a start tag END = StreamEventKind('END') # an end tag TEXT = StreamEventKind('TEXT') # literal text PROLOG = StreamEventKind('PROLOG') # XML prolog DOCTYPE = StreamEventKind('DOCTYPE') # doctype declaration START_NS = StreamEventKind('START-NS') # start namespace mapping END_NS = StreamEventKind('END-NS') # end namespace mapping PI = StreamEventKind('PI') # processing instruction COMMENT = StreamEventKind('COMMENT') # comment def __init__(self, events): """Initialize the stream with a sequence of markup events. @param events: a sequence or iterable providing the events """ self.events = events def __iter__(self): return iter(self.events) def render(self, method='xml', encoding='utf-8', filters=None, **kwargs): """Return a string representation of the stream. @param method: determines how the stream is serialized; can be either 'xml' or 'html', or a custom `Serializer` subclass @param encoding: how the output string should be encoded; if set to `None`, this method returns a `unicode` object Any additional keyword arguments are passed to the serializer, and thus depend on the `method` parameter value. """ generator = self.serialize(method=method, filters=filters, **kwargs) output = u''.join(list(generator)) if encoding is not None: return output.encode(encoding) return output def select(self, path): """Return a new stream that contains the events matching the given XPath expression. @param path: a string containing the XPath expression """ from markup.path import Path return Path(path).select(self) def serialize(self, method='xml', filters=None, **kwargs): """Generate strings corresponding to a specific serialization of the stream. Unlike the `render()` method, this method is a generator that returns the serialized output incrementally, as opposed to returning a single string. @param method: determines how the stream is serialized; can be either 'xml' or 'html', or a custom `Serializer` subclass @param filters: list of filters to apply to the stream before serialization. The default is to apply whitespace reduction using `markup.filters.WhitespaceFilter`. """ from markup.filters import WhitespaceFilter from markup import output cls = method if isinstance(method, basestring): cls = {'xml': output.XMLSerializer, 'html': output.HTMLSerializer}[method] else: assert issubclass(cls, output.Serializer) serializer = cls(**kwargs) stream = self if filters is None: filters = [WhitespaceFilter()] for filter_ in filters: stream = filter_(iter(stream)) return serializer.serialize(stream) def __str__(self): return self.render() def __unicode__(self): return self.render(encoding=None) START = Stream.START END = Stream.END TEXT = Stream.TEXT PROLOG = Stream.PROLOG DOCTYPE = Stream.DOCTYPE START_NS = Stream.START_NS END_NS = Stream.END_NS PI = Stream.PI COMMENT = Stream.COMMENT class Attributes(list): """Sequence type that stores the attributes of an element. The order of the attributes is preserved, while accessing and manipulating attributes by name is also supported. >>> attrs = Attributes([('href', '#'), ('title', 'Foo')]) >>> attrs [(u'href', '#'), (u'title', 'Foo')] >>> 'href' in attrs True >>> 'tabindex' in attrs False >>> attrs.get(u'title') 'Foo' >>> attrs.set(u'title', 'Bar') >>> attrs [(u'href', '#'), (u'title', 'Bar')] >>> attrs.remove(u'title') >>> attrs [(u'href', '#')] New attributes added using the `set()` method are appended to the end of the list: >>> attrs.set(u'accesskey', 'k') >>> attrs [(u'href', '#'), (u'accesskey', 'k')] """ __slots__ = [] def __init__(self, attrib=None): """Create the `Attributes` instance. If the `attrib` parameter is provided, it is expected to be a sequence of `(name, value)` tuples. """ if attrib is None: attrib = [] list.__init__(self, [(QName(name), value) for name, value in attrib]) def __contains__(self, name): """Return whether the list includes an attribute with the specified name. """ return name in [attr for attr, _ in self] def get(self, name, default=None): """Return the value of the attribute with the specified name, or the value of the `default` parameter if no such attribute is found. """ for attr, value in self: if attr == name: return value return default def remove(self, name): """Removes the attribute with the specified name. If no such attribute is found, this method does nothing. """ for idx, (attr, _) in enumerate(self): if attr == name: del self[idx] break def set(self, name, value): """Sets the specified attribute to the given value. If an attribute with the specified name is already in the list, the value of the existing entry is updated. Otherwise, a new attribute is appended to the end of the list. """ for idx, (attr, _) in enumerate(self): if attr == name: self[idx] = (attr, value) break else: self.append((QName(name), value)) class Markup(unicode): """Marks a string as being safe for inclusion in HTML/XML output without needing to be escaped. """ __slots__ = [] def __new__(cls, text='', *args): if args: text %= tuple([escape(arg) for arg in args]) return unicode.__new__(cls, text) def __add__(self, other): return Markup(unicode(self) + escape(other)) def __mod__(self, args): if not isinstance(args, (list, tuple)): args = [args] return Markup(unicode.__mod__(self, tuple([escape(arg) for arg in args]))) def __mul__(self, num): return Markup(unicode(self) * num) def __repr__(self): return '<%s "%s">' % (self.__class__.__name__, self) def join(self, seq, escape_quotes=True): return Markup(unicode(self).join([escape(item, quotes=escape_quotes) for item in seq])) def stripentities(self, keepxmlentities=False): """Return a copy of the text with any character or numeric entities replaced by the equivalent UTF-8 characters. If the `keepxmlentities` parameter is provided and evaluates to `True`, the core XML entities (&, ', >, < and ") are not stripped. """ def _replace_entity(match): if match.group(1): # numeric entity ref = match.group(1) if ref.startswith('x'): ref = int(ref[1:], 16) else: ref = int(ref, 10) return unichr(ref) else: # character entity ref = match.group(2) if keepxmlentities and ref in ('amp', 'apos', 'gt', 'lt', 'quot'): return '&%s;' % ref try: codepoint = htmlentitydefs.name2codepoint[ref] return unichr(codepoint) except KeyError: if keepxmlentities: return '&%s;' % ref else: return ref return Markup(re.sub(r'&(?:#((?:\d+)|(?:[xX][0-9a-fA-F]+));?|(\w+);)', _replace_entity, self)) def striptags(self): """Return a copy of the text with all XML/HTML tags removed.""" return Markup(re.sub(r'<[^>]*?>', '', self)) def escape(cls, text, quotes=True): """Create a Markup instance from a string and escape special characters it may contain (<, >, & and \"). If the `quotes` parameter is set to `False`, the \" character is left as is. Escaping quotes is generally only required for strings that are to be used in attribute values. """ if isinstance(text, cls): return text text = unicode(text) if not text or isinstance(text, cls): return cls() text = text.replace('&', '&') \ .replace('<', '<') \ .replace('>', '>') if quotes: text = text.replace('"', '"') return cls(text) escape = classmethod(escape) def unescape(self): """Reverse-escapes &, <, > and \" and returns a `unicode` object.""" if not self: return u'' return unicode(self).replace('"', '"') \ .replace('>', '>') \ .replace('<', '<') \ .replace('&', '&') def plaintext(self, keeplinebreaks=True): """Returns the text as a `unicode` string with all entities and tags removed. """ text = unicode(self.striptags().stripentities()) if not keeplinebreaks: text = text.replace(u'\n', u' ') return text def sanitize(self): from markup.filters import HTMLSanitizer from markup.input import HTMLParser text = StringIO(self.stripentities(keepxmlentities=True)) return Stream(HTMLSanitizer()(HTMLParser(text))) escape = Markup.escape def unescape(text): """Reverse-escapes &, <, > and \" and returns a `unicode` object.""" if not isinstance(text, Markup): return text return text.unescape() class Namespace(object): """Utility class creating and testing elements with a namespace. Internally, namespace URIs are encoded in the `QName` of any element or attribute, the namespace URI being enclosed in curly braces. This class helps create and test these strings. A `Namespace` object is instantiated with the namespace URI. >>> html = Namespace('http://www.w3.org/1999/xhtml') >>> html <Namespace "http://www.w3.org/1999/xhtml"> >>> html.uri u'http://www.w3.org/1999/xhtml' The `Namespace` object can than be used to generate `QName` objects with that namespace: >>> html.body u'{http://www.w3.org/1999/xhtml}body' >>> html.body.localname u'body' >>> html.body.namespace u'http://www.w3.org/1999/xhtml' The same works using item access notation, which is useful for element or attribute names that are not valid Python identifiers: >>> html['body'] u'{http://www.w3.org/1999/xhtml}body' A `Namespace` object can also be used to test whether a specific `QName` belongs to that namespace using the `in` operator: >>> qname = html.body >>> qname in html True >>> qname in Namespace('http://www.w3.org/2002/06/xhtml2') False """ def __init__(self, uri): self.uri = unicode(uri) def __contains__(self, qname): return qname.namespace == self.uri def __eq__(self, other): if isinstance(other, Namespace): return self.uri == other.uri return self.uri == other def __getitem__(self, name): return QName(self.uri + u'}' + name) __getattr__ = __getitem__ def __repr__(self): return '<Namespace "%s">' % self.uri def __str__(self): return self.uri.encode('utf-8') def __unicode__(self): return self.uri class QName(unicode): """A qualified element or attribute name. The unicode value of instances of this class contains the qualified name of the element or attribute, in the form `{namespace}localname`. The namespace URI can be obtained through the additional `namespace` attribute, while the local name can be accessed through the `localname` attribute. >>> qname = QName('foo') >>> qname u'foo' >>> qname.localname u'foo' >>> qname.namespace >>> qname = QName('http://www.w3.org/1999/xhtml}body') >>> qname u'{http://www.w3.org/1999/xhtml}body' >>> qname.localname u'body' >>> qname.namespace u'http://www.w3.org/1999/xhtml' """ __slots__ = ['namespace', 'localname'] def __new__(cls, qname): if isinstance(qname, QName): return qname parts = qname.split(u'}', 1) if qname.find(u'}') > 0: self = unicode.__new__(cls, u'{' + qname) self.namespace = unicode(parts[0]) self.localname = unicode(parts[1]) else: self = unicode.__new__(cls, qname) self.namespace = None self.localname = unicode(qname) return self