From 6657ae24658383a9841450329a518327638651c0 Mon Sep 17 00:00:00 2001 From: "luxagraf@c63593aa-01b0-44d9-8516-4b9c7e931d7f" Date: Sat, 12 Jun 2010 20:33:36 +0000 Subject: reorganized lib --- apps/blog/models.py | 2 +- apps/links/utils.py | 11 +- apps/photos/utils.py | 4 +- apps/photos/views.py | 7 +- apps/projects/models/base.py | 2 +- base_urls.py | 5 - lib/APIClients.py | 104 -- lib/context-processors/__init__.py | 0 lib/context-processors/context_processors.py | 3 - lib/markdown2.py | 1877 -------------------------- lib/pydelicious.py | 1045 -------------- lib/strutils.py | 50 - lib/utils/APIClients.py | 104 ++ lib/utils/GeoClient.py | 292 ---- lib/utils/email_multipart.py | 80 -- lib/utils/markdown2.py | 1877 ++++++++++++++++++++++++++ lib/utils/pydelicious.py | 858 +++++++----- lib/utils/strutils.py | 50 + lib/view_wrapper.py | 10 - templates/includes/map_entry_list.html | 178 +-- 20 files changed, 2677 insertions(+), 3882 deletions(-) delete mode 100644 lib/APIClients.py delete mode 100644 lib/context-processors/__init__.py delete mode 100644 lib/context-processors/context_processors.py delete mode 100755 lib/markdown2.py delete mode 100644 lib/pydelicious.py delete mode 100644 lib/strutils.py create mode 100644 lib/utils/APIClients.py delete mode 100644 lib/utils/GeoClient.py delete mode 100644 lib/utils/email_multipart.py create mode 100755 lib/utils/markdown2.py create mode 100644 lib/utils/strutils.py delete mode 100644 lib/view_wrapper.py diff --git a/apps/blog/models.py b/apps/blog/models.py index c3b5a3a..9995bc1 100644 --- a/apps/blog/models.py +++ b/apps/blog/models.py @@ -6,7 +6,7 @@ from django.contrib.sitemaps import Sitemap from django.template.defaultfilters import truncatewords_html -import markdown2 as markdown +from utils import markdown2 as markdown from tagging.fields import TagField from tagging.models import Tag diff --git a/apps/links/utils.py b/apps/links/utils.py index 9e5aea5..604ecab 100644 --- a/apps/links/utils.py +++ b/apps/links/utils.py @@ -4,14 +4,15 @@ from django.core.exceptions import ObjectDoesNotExist from django.template.defaultfilters import slugify,striptags from django.core.mail import EmailMessage from django.utils.encoding import smart_unicode +from django.conf import settings + -from strutils import safestr,unquotehtml -from APIClients import MagnoliaClient -import pydelicious as delicious +from utils.strutils import safestr,unquotehtml +from utils.APIClients import MagnoliaClient +from utils import pydelicious as delicious +from utils import markdown2 as markdown -import markdown2 as markdown from links.models import Link -from django.conf import settings def flickr_datetime_to_datetime(fdt): from datetime import datetime diff --git a/apps/photos/utils.py b/apps/photos/utils.py index 24991b2..137fe9f 100644 --- a/apps/photos/utils.py +++ b/apps/photos/utils.py @@ -28,13 +28,13 @@ except ImportError: raise ImportError("Could not import the Python Imaging Library.") -from strutils import safestr +from utils.strutils import safestr from photos.models import Photo,PhotoGallery # Flickr Sync stuffs API_KEY = settings.FLICKR_API_KEY -from APIClients import FlickrClient +from utils.APIClients import FlickrClient EXIF_PARAMS = {"Aperture":'f/2.8',"Make":'Apple',"Model":'iPhone',"Exposure":'',"ISO Speed":'',"Focal Length":'',"Shutter Speed":'',"Lens":'','Date and Time (Original)':'2008:07:03 22:44:25'} diff --git a/apps/photos/views.py b/apps/photos/views.py index 5ccec00..861f00e 100644 --- a/apps/photos/views.py +++ b/apps/photos/views.py @@ -5,7 +5,6 @@ from django.http import Http404,HttpResponse from django.core import serializers -from view_wrapper import luxagraf_render from tagging.models import Tag,TaggedItem from photos.models import Photo,PhotoGallery from locations.models import Country, Region @@ -24,8 +23,10 @@ def gallery_list(request,page): return object_list(request, queryset=qs, template_name='archives/photos.html') def gallery(request,slug): - g = PhotoGallery.objects.get(set_slug=slug) - return luxagraf_render(request,'details/photo_galleries.html', {'object': g,}) + context = { + 'object': PhotoGallery.objects.get(set_slug=slug) + } + render_to_response('details/photo_galleries.html', context, context_instance = RequestContext(request)) def photo_json(request, slug): p = PhotoGallery.objects.filter(set_slug=slug) diff --git a/apps/projects/models/base.py b/apps/projects/models/base.py index bc42918..96ccd59 100644 --- a/apps/projects/models/base.py +++ b/apps/projects/models/base.py @@ -6,7 +6,7 @@ from django.contrib.sitemaps import Sitemap from django.template.defaultfilters import truncatewords_html -import markdown2 as markdown +from utils import markdown2 as markdown from photos.models import PhotoGallery diff --git a/base_urls.py b/base_urls.py index 8a3c970..bc6f344 100644 --- a/base_urls.py +++ b/base_urls.py @@ -40,19 +40,14 @@ urlpatterns += patterns('', urlpatterns += patterns('', (r'^colophon/', redirect_to, {'url': '/about/'}), - (r'^admin/doc/', redirect_to, {'url': '/grappelli/help/1/'}), (r'^admin/doc/', include('django.contrib.admindocs.urls')), (r'^admin/filebrowser/', include('filebrowser.urls')), (r'^admin/(.*)', admin.site.root), (r'^grappelli/', include('grappelli.urls')), - (r'^uploads/', include('upload.urls')), (r'^feeds/(?P.*)/$', 'django.contrib.syndication.views.feed', {'feed_dict': feeds}), - (r'^admin/doc/', include('django.contrib.admindocs.urls')), - (r'^admin/(.*)', admin.site.root), (r'^robots.txt$', direct_to_template, {'template': 'archives/robots.html'}), (r'^googleb11655cd59dacf3c.html$', direct_to_template, {'template': 'static/gverify.html'}), (r'^contact/', include('contact_form.urls')), - #(r'^photo-of-the-day/$', 'luxagraf.photos.views.potd_list'), (r'^sitemap.xml$', 'django.contrib.sitemaps.views.sitemap', {'sitemaps': sitemaps}), (r'^writing/', include('blog.urls')), (r'^projects/', include('projects.urls')), diff --git a/lib/APIClients.py b/lib/APIClients.py deleted file mode 100644 index 24ab97b..0000000 --- a/lib/APIClients.py +++ /dev/null @@ -1,104 +0,0 @@ -# APIClients for grabbing data from popular web services -# By Scott Gilbertson -# Copyright is lame, take what you want, except for those portions noted - -# Dependencies: -import sys, urllib -import xml.etree.cElementTree as xml_parser - - -DEBUG = 0 - -""" -base class -- handles GoodReads.com, but works for any rss feed, just send an empty string for anything you don't need -""" -class APIClient: - def __init__(self, base_path, api_key): - self.api_key = api_key - self.base_path = base_path - - def __getattr__(self, method): - def method(_self=self, _method=method, **params): - url = "%s%s?%s&" % (self.base_path, self.api_key, urllib.urlencode(params)) - if DEBUG: print url - data = self.fetch(url) - return data - - return method - - def fetch(self, url): - u = urllib.FancyURLopener(None) - usock = u.open(url) - rawdata = usock.read() - if DEBUG: print rawdata - usock.close() - return xml_parser.fromstring(rawdata) - -""" - Extend APIClient to work with the ma.gnolia.com API - (http://wiki.ma.gnolia.com/Ma.gnolia_API) - Adds some error handling as well -""" -class MagnoliaError(Exception): - def __init__(self, code, message): - self.code = code - self.message = message - - def __str__(self): - return 'Magnolia Error %s: %s' % (self.code, self.message) - - -class MagnoliaClient(APIClient): - def __getattr__(self, method): - def method(_self=self, _method=method, **params): - url = "%s%s?%s&api_key=%s" % (self.base_path, _method, urllib.urlencode(params), self.api_key) - if DEBUG: print url - data = APIClient.fetch(self, url) - return data - return method - - -""" - Extend APIClient to work with the Flickr API - (http://www.flickr.com/services/api/) - Adds error handling as well -""" - -class FlickrError(Exception): - def __init__(self, code, message): - self.code = code - self.message = message - - def __str__(self): - return 'Flickr Error %s: %s' % (self.code, self.message) - -class FlickrClient(APIClient): - def __getattr__(self, method): - def method(_self=self, _method=method, **params): - _method = _method.replace("_", ".") - url = "%s?method=%s&%s&api_key=%s" % (self.base_path, _method, urllib.urlencode(params), self.api_key) - if DEBUG: print url - data = APIClient.fetch(self, url) - return data - return method - -class TumblrClient: - def __init__(self, base_path): - self.base_path = base_path - - def __getattr__(self, method): - def method(_self=self, _method=method, **params): - url = "%s" % (self.base_path) - if DEBUG: print url - data = self.fetch(url) - return data - - return method - - def fetch(self, url): - u = urllib.FancyURLopener(None) - usock = u.open(url) - rawdata = usock.read() - if DEBUG: print rawdata - usock.close() - return xml_parser.fromstring(rawdata) diff --git a/lib/context-processors/__init__.py b/lib/context-processors/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/lib/context-processors/context_processors.py b/lib/context-processors/context_processors.py deleted file mode 100644 index 5691102..0000000 --- a/lib/context-processors/context_processors.py +++ /dev/null @@ -1,3 +0,0 @@ -def media_url(request): - from django.conf import settings - return {'media_url': settings.MEDIA_URL, 'map_key':settings.MAP_API_KEY} diff --git a/lib/markdown2.py b/lib/markdown2.py deleted file mode 100755 index d72f414..0000000 --- a/lib/markdown2.py +++ /dev/null @@ -1,1877 +0,0 @@ -#!/usr/bin/env python -# Copyright (c) 2007-2008 ActiveState Corp. -# License: MIT (http://www.opensource.org/licenses/mit-license.php) - -r"""A fast and complete Python implementation of Markdown. - -[from http://daringfireball.net/projects/markdown/] -> Markdown is a text-to-HTML filter; it translates an easy-to-read / -> easy-to-write structured text format into HTML. Markdown's text -> format is most similar to that of plain text email, and supports -> features such as headers, *emphasis*, code blocks, blockquotes, and -> links. -> -> Markdown's syntax is designed not as a generic markup language, but -> specifically to serve as a front-end to (X)HTML. You can use span-level -> HTML tags anywhere in a Markdown document, and you can use block level -> HTML tags (like
and as well). - -Module usage: - - >>> import markdown2 - >>> markdown2.markdown("*boo!*") # or use `html = markdown_path(PATH)` - u'

boo!

\n' - - >>> markdowner = Markdown() - >>> markdowner.convert("*boo!*") - u'

boo!

\n' - >>> markdowner.convert("**boom!**") - u'

boom!

\n' - -This implementation of Markdown implements the full "core" syntax plus a -number of extras (e.g., code syntax coloring, footnotes) as described on -. -""" - -cmdln_desc = """A fast and complete Python implementation of Markdown, a -text-to-HTML conversion tool for web writers. -""" - -# Dev Notes: -# - There is already a Python markdown processor -# (http://www.freewisdom.org/projects/python-markdown/). -# - Python's regex syntax doesn't have '\z', so I'm using '\Z'. I'm -# not yet sure if there implications with this. Compare 'pydoc sre' -# and 'perldoc perlre'. - -__version_info__ = (1, 0, 1, 13) # first three nums match Markdown.pl -__version__ = '1.0.1.13' -__author__ = "Trent Mick" - -import os -import sys -from pprint import pprint -import re -import logging -try: - from hashlib import md5 -except ImportError: - from md5 import md5 -import optparse -from random import random -import codecs - - - -#---- Python version compat - -if sys.version_info[:2] < (2,4): - from sets import Set as set - def reversed(sequence): - for i in sequence[::-1]: - yield i - def _unicode_decode(s, encoding, errors='xmlcharrefreplace'): - return unicode(s, encoding, errors) -else: - def _unicode_decode(s, encoding, errors='strict'): - return s.decode(encoding, errors) - - -#---- globals - -DEBUG = False -log = logging.getLogger("markdown") - -DEFAULT_TAB_WIDTH = 4 - -# Table of hash values for escaped characters: -def _escape_hash(s): - # Lame attempt to avoid possible collision with someone actually - # using the MD5 hexdigest of one of these chars in there text. - # Other ideas: random.random(), uuid.uuid() - #return md5(s).hexdigest() # Markdown.pl effectively does this. - return 'md5-'+md5(s).hexdigest() -g_escape_table = dict([(ch, _escape_hash(ch)) - for ch in '\\`*_{}[]()>#+-.!']) - - - -#---- exceptions - -class MarkdownError(Exception): - pass - - - -#---- public api - -def markdown_path(path, encoding="utf-8", - html4tags=False, tab_width=DEFAULT_TAB_WIDTH, - safe_mode=None, extras=None, link_patterns=None, - use_file_vars=False): - text = codecs.open(path, 'r', encoding).read() - return Markdown(html4tags=html4tags, tab_width=tab_width, - safe_mode=safe_mode, extras=extras, - link_patterns=link_patterns, - use_file_vars=use_file_vars).convert(text) - -def markdown(text, html4tags=False, tab_width=DEFAULT_TAB_WIDTH, - safe_mode=None, extras=None, link_patterns=None, - use_file_vars=False): - return Markdown(html4tags=html4tags, tab_width=tab_width, - safe_mode=safe_mode, extras=extras, - link_patterns=link_patterns, - use_file_vars=use_file_vars).convert(text) - -class Markdown(object): - # The dict of "extras" to enable in processing -- a mapping of - # extra name to argument for the extra. Most extras do not have an - # argument, in which case the value is None. - # - # This can be set via (a) subclassing and (b) the constructor - # "extras" argument. - extras = None - - urls = None - titles = None - html_blocks = None - html_spans = None - html_removed_text = "[HTML_REMOVED]" # for compat with markdown.py - - # Used to track when we're inside an ordered or unordered list - # (see _ProcessListItems() for details): - list_level = 0 - - _ws_only_line_re = re.compile(r"^[ \t]+$", re.M) - - def __init__(self, html4tags=False, tab_width=4, safe_mode=None, - extras=None, link_patterns=None, use_file_vars=False): - if html4tags: - self.empty_element_suffix = ">" - else: - self.empty_element_suffix = " />" - self.tab_width = tab_width - - # For compatibility with earlier markdown2.py and with - # markdown.py's safe_mode being a boolean, - # safe_mode == True -> "replace" - if safe_mode is True: - self.safe_mode = "replace" - else: - self.safe_mode = safe_mode - - if self.extras is None: - self.extras = {} - elif not isinstance(self.extras, dict): - self.extras = dict([(e, None) for e in self.extras]) - if extras: - if not isinstance(extras, dict): - extras = dict([(e, None) for e in extras]) - self.extras.update(extras) - assert isinstance(self.extras, dict) - self._instance_extras = self.extras.copy() - self.link_patterns = link_patterns - self.use_file_vars = use_file_vars - self._outdent_re = re.compile(r'^(\t|[ ]{1,%d})' % tab_width, re.M) - - def reset(self): - self.urls = {} - self.titles = {} - self.html_blocks = {} - self.html_spans = {} - self.list_level = 0 - self.extras = self._instance_extras.copy() - if "footnotes" in self.extras: - self.footnotes = {} - self.footnote_ids = [] - - def convert(self, text): - """Convert the given text.""" - # Main function. The order in which other subs are called here is - # essential. Link and image substitutions need to happen before - # _EscapeSpecialChars(), so that any *'s or _'s in the - # and tags get encoded. - - # Clear the global hashes. If we don't clear these, you get conflicts - # from other articles when generating a page which contains more than - # one article (e.g. an index page that shows the N most recent - # articles): - self.reset() - - if not isinstance(text, unicode): - #TODO: perhaps shouldn't presume UTF-8 for string input? - text = unicode(text, 'utf-8') - - if self.use_file_vars: - # Look for emacs-style file variable hints. - emacs_vars = self._get_emacs_vars(text) - if "markdown-extras" in emacs_vars: - splitter = re.compile("[ ,]+") - for e in splitter.split(emacs_vars["markdown-extras"]): - if '=' in e: - ename, earg = e.split('=', 1) - try: - earg = int(earg) - except ValueError: - pass - else: - ename, earg = e, None - self.extras[ename] = earg - - # Standardize line endings: - text = re.sub("\r\n|\r", "\n", text) - - # Make sure $text ends with a couple of newlines: - text += "\n\n" - - # Convert all tabs to spaces. - text = self._detab(text) - - # Strip any lines consisting only of spaces and tabs. - # This makes subsequent regexen easier to write, because we can - # match consecutive blank lines with /\n+/ instead of something - # contorted like /[ \t]*\n+/ . - text = self._ws_only_line_re.sub("", text) - - if self.safe_mode: - text = self._hash_html_spans(text) - - # Turn block-level HTML blocks into hash entries - text = self._hash_html_blocks(text, raw=True) - - # Strip link definitions, store in hashes. - if "footnotes" in self.extras: - # Must do footnotes first because an unlucky footnote defn - # looks like a link defn: - # [^4]: this "looks like a link defn" - text = self._strip_footnote_definitions(text) - text = self._strip_link_definitions(text) - - text = self._run_block_gamut(text) - - text = self._unescape_special_chars(text) - - if "footnotes" in self.extras: - text = self._add_footnotes(text) - - if self.safe_mode: - text = self._unhash_html_spans(text) - - text += "\n" - return text - - _emacs_oneliner_vars_pat = re.compile(r"-\*-\s*([^\r\n]*?)\s*-\*-", re.UNICODE) - # This regular expression is intended to match blocks like this: - # PREFIX Local Variables: SUFFIX - # PREFIX mode: Tcl SUFFIX - # PREFIX End: SUFFIX - # Some notes: - # - "[ \t]" is used instead of "\s" to specifically exclude newlines - # - "(\r\n|\n|\r)" is used instead of "$" because the sre engine does - # not like anything other than Unix-style line terminators. - _emacs_local_vars_pat = re.compile(r"""^ - (?P(?:[^\r\n|\n|\r])*?) - [\ \t]*Local\ Variables:[\ \t]* - (?P.*?)(?:\r\n|\n|\r) - (?P.*?\1End:) - """, re.IGNORECASE | re.MULTILINE | re.DOTALL | re.VERBOSE) - - def _get_emacs_vars(self, text): - """Return a dictionary of emacs-style local variables. - - Parsing is done loosely according to this spec (and according to - some in-practice deviations from this): - http://www.gnu.org/software/emacs/manual/html_node/emacs/Specifying-File-Variables.html#Specifying-File-Variables - """ - emacs_vars = {} - SIZE = pow(2, 13) # 8kB - - # Search near the start for a '-*-'-style one-liner of variables. - head = text[:SIZE] - if "-*-" in head: - match = self._emacs_oneliner_vars_pat.search(head) - if match: - emacs_vars_str = match.group(1) - assert '\n' not in emacs_vars_str - emacs_var_strs = [s.strip() for s in emacs_vars_str.split(';') - if s.strip()] - if len(emacs_var_strs) == 1 and ':' not in emacs_var_strs[0]: - # While not in the spec, this form is allowed by emacs: - # -*- Tcl -*- - # where the implied "variable" is "mode". This form - # is only allowed if there are no other variables. - emacs_vars["mode"] = emacs_var_strs[0].strip() - else: - for emacs_var_str in emacs_var_strs: - try: - variable, value = emacs_var_str.strip().split(':', 1) - except ValueError: - log.debug("emacs variables error: malformed -*- " - "line: %r", emacs_var_str) - continue - # Lowercase the variable name because Emacs allows "Mode" - # or "mode" or "MoDe", etc. - emacs_vars[variable.lower()] = value.strip() - - tail = text[-SIZE:] - if "Local Variables" in tail: - match = self._emacs_local_vars_pat.search(tail) - if match: - prefix = match.group("prefix") - suffix = match.group("suffix") - lines = match.group("content").splitlines(0) - #print "prefix=%r, suffix=%r, content=%r, lines: %s"\ - # % (prefix, suffix, match.group("content"), lines) - - # Validate the Local Variables block: proper prefix and suffix - # usage. - for i, line in enumerate(lines): - if not line.startswith(prefix): - log.debug("emacs variables error: line '%s' " - "does not use proper prefix '%s'" - % (line, prefix)) - return {} - # Don't validate suffix on last line. Emacs doesn't care, - # neither should we. - if i != len(lines)-1 and not line.endswith(suffix): - log.debug("emacs variables error: line '%s' " - "does not use proper suffix '%s'" - % (line, suffix)) - return {} - - # Parse out one emacs var per line. - continued_for = None - for line in lines[:-1]: # no var on the last line ("PREFIX End:") - if prefix: line = line[len(prefix):] # strip prefix - if suffix: line = line[:-len(suffix)] # strip suffix - line = line.strip() - if continued_for: - variable = continued_for - if line.endswith('\\'): - line = line[:-1].rstrip() - else: - continued_for = None - emacs_vars[variable] += ' ' + line - else: - try: - variable, value = line.split(':', 1) - except ValueError: - log.debug("local variables error: missing colon " - "in local variables entry: '%s'" % line) - continue - # Do NOT lowercase the variable name, because Emacs only - # allows "mode" (and not "Mode", "MoDe", etc.) in this block. - value = value.strip() - if value.endswith('\\'): - value = value[:-1].rstrip() - continued_for = variable - else: - continued_for = None - emacs_vars[variable] = value - - # Unquote values. - for var, val in emacs_vars.items(): - if len(val) > 1 and (val.startswith('"') and val.endswith('"') - or val.startswith('"') and val.endswith('"')): - emacs_vars[var] = val[1:-1] - - return emacs_vars - - # Cribbed from a post by Bart Lateur: - # - _detab_re = re.compile(r'(.*?)\t', re.M) - def _detab_sub(self, match): - g1 = match.group(1) - return g1 + (' ' * (self.tab_width - len(g1) % self.tab_width)) - def _detab(self, text): - r"""Remove (leading?) tabs from a file. - - >>> m = Markdown() - >>> m._detab("\tfoo") - ' foo' - >>> m._detab(" \tfoo") - ' foo' - >>> m._detab("\t foo") - ' foo' - >>> m._detab(" foo") - ' foo' - >>> m._detab(" foo\n\tbar\tblam") - ' foo\n bar blam' - """ - if '\t' not in text: - return text - return self._detab_re.subn(self._detab_sub, text)[0] - - _block_tags_a = 'p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math|ins|del' - _strict_tag_block_re = re.compile(r""" - ( # save in \1 - ^ # start of line (with re.M) - <(%s) # start tag = \2 - \b # word break - (.*\n)*? # any number of lines, minimally matching - # the matching end tag - [ \t]* # trailing spaces/tabs - (?=\n+|\Z) # followed by a newline or end of document - ) - """ % _block_tags_a, - re.X | re.M) - - _block_tags_b = 'p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math' - _liberal_tag_block_re = re.compile(r""" - ( # save in \1 - ^ # start of line (with re.M) - <(%s) # start tag = \2 - \b # word break - (.*\n)*? # any number of lines, minimally matching - .* # the matching end tag - [ \t]* # trailing spaces/tabs - (?=\n+|\Z) # followed by a newline or end of document - ) - """ % _block_tags_b, - re.X | re.M) - - def _hash_html_block_sub(self, match, raw=False): - html = match.group(1) - if raw and self.safe_mode: - html = self._sanitize_html(html) - key = _hash_text(html) - self.html_blocks[key] = html - return "\n\n" + key + "\n\n" - - def _hash_html_blocks(self, text, raw=False): - """Hashify HTML blocks - - We only want to do this for block-level HTML tags, such as headers, - lists, and tables. That's because we still want to wrap

s around - "paragraphs" that are wrapped in non-block-level tags, such as anchors, - phrase emphasis, and spans. The list of tags we're looking for is - hard-coded. - - @param raw {boolean} indicates if these are raw HTML blocks in - the original source. It makes a difference in "safe" mode. - """ - if '<' not in text: - return text - - # Pass `raw` value into our calls to self._hash_html_block_sub. - hash_html_block_sub = _curry(self._hash_html_block_sub, raw=raw) - - # First, look for nested blocks, e.g.: - #

- #
- # tags for inner block must be indented. - #
- #
- # - # The outermost tags must start at the left margin for this to match, and - # the inner nested divs must be indented. - # We need to do this before the next, more liberal match, because the next - # match will start at the first `
` and stop at the first `
`. - text = self._strict_tag_block_re.sub(hash_html_block_sub, text) - - # Now match more liberally, simply from `\n` to `\n` - text = self._liberal_tag_block_re.sub(hash_html_block_sub, text) - - # Special case just for
. It was easier to make a special - # case than to make the other regex more complicated. - if "", start_idx) + 3 - except ValueError, ex: - break - - # Start position for next comment block search. - start = end_idx - - # Validate whitespace before comment. - if start_idx: - # - Up to `tab_width - 1` spaces before start_idx. - for i in range(self.tab_width - 1): - if text[start_idx - 1] != ' ': - break - start_idx -= 1 - if start_idx == 0: - break - # - Must be preceded by 2 newlines or hit the start of - # the document. - if start_idx == 0: - pass - elif start_idx == 1 and text[0] == '\n': - start_idx = 0 # to match minute detail of Markdown.pl regex - elif text[start_idx-2:start_idx] == '\n\n': - pass - else: - break - - # Validate whitespace after comment. - # - Any number of spaces and tabs. - while end_idx < len(text): - if text[end_idx] not in ' \t': - break - end_idx += 1 - # - Must be following by 2 newlines or hit end of text. - if text[end_idx:end_idx+2] not in ('', '\n', '\n\n'): - continue - - # Escape and hash (must match `_hash_html_block_sub`). - html = text[start_idx:end_idx] - if raw and self.safe_mode: - html = self._sanitize_html(html) - key = _hash_text(html) - self.html_blocks[key] = html - text = text[:start_idx] + "\n\n" + key + "\n\n" + text[end_idx:] - - if "xml" in self.extras: - # Treat XML processing instructions and namespaced one-liner - # tags as if they were block HTML tags. E.g., if standalone - # (i.e. are their own paragraph), the following do not get - # wrapped in a

tag: - # - # - # - _xml_oneliner_re = _xml_oneliner_re_from_tab_width(self.tab_width) - text = _xml_oneliner_re.sub(hash_html_block_sub, text) - - return text - - def _strip_link_definitions(self, text): - # Strips link definitions from text, stores the URLs and titles in - # hash references. - less_than_tab = self.tab_width - 1 - - # Link defs are in the form: - # [id]: url "optional title" - _link_def_re = re.compile(r""" - ^[ ]{0,%d}\[(.+)\]: # id = \1 - [ \t]* - \n? # maybe *one* newline - [ \t]* - ? # url = \2 - [ \t]* - (?: - \n? # maybe one newline - [ \t]* - (?<=\s) # lookbehind for whitespace - ['"(] - ([^\n]*) # title = \3 - ['")] - [ \t]* - )? # title is optional - (?:\n+|\Z) - """ % less_than_tab, re.X | re.M | re.U) - return _link_def_re.sub(self._extract_link_def_sub, text) - - def _extract_link_def_sub(self, match): - id, url, title = match.groups() - key = id.lower() # Link IDs are case-insensitive - self.urls[key] = self._encode_amps_and_angles(url) - if title: - self.titles[key] = title.replace('"', '"') - return "" - - def _extract_footnote_def_sub(self, match): - id, text = match.groups() - text = _dedent(text, skip_first_line=not text.startswith('\n')).strip() - normed_id = re.sub(r'\W', '-', id) - # Ensure footnote text ends with a couple newlines (for some - # block gamut matches). - self.footnotes[normed_id] = text + "\n\n" - return "" - - def _strip_footnote_definitions(self, text): - """A footnote definition looks like this: - - [^note-id]: Text of the note. - - May include one or more indented paragraphs. - - Where, - - The 'note-id' can be pretty much anything, though typically it - is the number of the footnote. - - The first paragraph may start on the next line, like so: - - [^note-id]: - Text of the note. - """ - less_than_tab = self.tab_width - 1 - footnote_def_re = re.compile(r''' - ^[ ]{0,%d}\[\^(.+)\]: # id = \1 - [ \t]* - ( # footnote text = \2 - # First line need not start with the spaces. - (?:\s*.*\n+) - (?: - (?:[ ]{%d} | \t) # Subsequent lines must be indented. - .*\n+ - )* - ) - # Lookahead for non-space at line-start, or end of doc. - (?:(?=^[ ]{0,%d}\S)|\Z) - ''' % (less_than_tab, self.tab_width, self.tab_width), - re.X | re.M) - return footnote_def_re.sub(self._extract_footnote_def_sub, text) - - - _hr_res = [ - re.compile(r"^[ ]{0,2}([ ]?\*[ ]?){3,}[ \t]*$", re.M), - re.compile(r"^[ ]{0,2}([ ]?\-[ ]?){3,}[ \t]*$", re.M), - re.compile(r"^[ ]{0,2}([ ]?\_[ ]?){3,}[ \t]*$", re.M), - ] - - def _run_block_gamut(self, text): - # These are all the transformations that form block-level - # tags like paragraphs, headers, and list items. - - text = self._do_headers(text) - - # Do Horizontal Rules: - hr = "\n tags around block-level tags. - text = self._hash_html_blocks(text) - - text = self._form_paragraphs(text) - - return text - - def _pyshell_block_sub(self, match): - lines = match.group(0).splitlines(0) - _dedentlines(lines) - indent = ' ' * self.tab_width - s = ('\n' # separate from possible cuddled paragraph - + indent + ('\n'+indent).join(lines) - + '\n\n') - return s - - def _prepare_pyshell_blocks(self, text): - """Ensure that Python interactive shell sessions are put in - code blocks -- even if not properly indented. - """ - if ">>>" not in text: - return text - - less_than_tab = self.tab_width - 1 - _pyshell_block_re = re.compile(r""" - ^([ ]{0,%d})>>>[ ].*\n # first line - ^(\1.*\S+.*\n)* # any number of subsequent lines - ^\n # ends with a blank line - """ % less_than_tab, re.M | re.X) - - return _pyshell_block_re.sub(self._pyshell_block_sub, text) - - def _run_span_gamut(self, text): - # These are all the transformations that occur *within* block-level - # tags like paragraphs, headers, and list items. - - text = self._do_code_spans(text) - - text = self._escape_special_chars(text) - - # Process anchor and image tags. - text = self._do_links(text) - - # Make links out of things like `` - # Must come after _do_links(), because you can use < and > - # delimiters in inline links like [this](). - text = self._do_auto_links(text) - - if "link-patterns" in self.extras: - text = self._do_link_patterns(text) - - text = self._encode_amps_and_angles(text) - - text = self._do_italics_and_bold(text) - - # Do hard breaks: - text = re.sub(r" {2,}\n", " - | - # auto-link (e.g., ) - <\w+[^>]*> - | - # comment - | - <\?.*?\?> # processing instruction - ) - """, re.X) - - def _escape_special_chars(self, text): - # Python markdown note: the HTML tokenization here differs from - # that in Markdown.pl, hence the behaviour for subtle cases can - # differ (I believe the tokenizer here does a better job because - # it isn't susceptible to unmatched '<' and '>' in HTML tags). - # Note, however, that '>' is not allowed in an auto-link URL - # here. - escaped = [] - is_html_markup = False - for token in self._sorta_html_tokenize_re.split(text): - if is_html_markup: - # Within tags/HTML-comments/auto-links, encode * and _ - # so they don't conflict with their use in Markdown for - # italics and strong. We're replacing each such - # character with its corresponding MD5 checksum value; - # this is likely overkill, but it should prevent us from - # colliding with the escape values by accident. - escaped.append(token.replace('*', g_escape_table['*']) - .replace('_', g_escape_table['_'])) - else: - escaped.append(self._encode_backslash_escapes(token)) - is_html_markup = not is_html_markup - return ''.join(escaped) - - def _hash_html_spans(self, text): - # Used for safe_mode. - - def _is_auto_link(s): - if ':' in s and self._auto_link_re.match(s): - return True - elif '@' in s and self._auto_email_link_re.match(s): - return True - return False - - tokens = [] - is_html_markup = False - for token in self._sorta_html_tokenize_re.split(text): - if is_html_markup and not _is_auto_link(token): - sanitized = self._sanitize_html(token) - key = _hash_text(sanitized) - self.html_spans[key] = sanitized - tokens.append(key) - else: - tokens.append(token) - is_html_markup = not is_html_markup - return ''.join(tokens) - - def _unhash_html_spans(self, text): - for key, sanitized in self.html_spans.items(): - text = text.replace(key, sanitized) - return text - - def _sanitize_html(self, s): - if self.safe_mode == "replace": - return self.html_removed_text - elif self.safe_mode == "escape": - replacements = [ - ('&', '&'), - ('<', '<'), - ('>', '>'), - ] - for before, after in replacements: - s = s.replace(before, after) - return s - else: - raise MarkdownError("invalid value for 'safe_mode': %r (must be " - "'escape' or 'replace')" % self.safe_mode) - - _tail_of_inline_link_re = re.compile(r''' - # Match tail of: [text](/url/) or [text](/url/ "title") - \( # literal paren - [ \t]* - (?P # \1 - <.*?> - | - .*? - ) - [ \t]* - ( # \2 - (['"]) # quote char = \3 - (?P.*?) - \3 # matching quote - )? # title is optional - \) - ''', re.X | re.S) - _tail_of_reference_link_re = re.compile(r''' - # Match tail of: [text][id] - [ ]? # one optional space - (?:\n[ ]*)? # one optional newline followed by spaces - \[ - (?P<id>.*?) - \] - ''', re.X | re.S) - - def _do_links(self, text): - """Turn Markdown link shortcuts into XHTML <a> and <img> tags. - - This is a combination of Markdown.pl's _DoAnchors() and - _DoImages(). They are done together because that simplified the - approach. It was necessary to use a different approach than - Markdown.pl because of the lack of atomic matching support in - Python's regex engine used in $g_nested_brackets. - """ - MAX_LINK_TEXT_SENTINEL = 3000 # markdown2 issue 24 - - # `anchor_allowed_pos` is used to support img links inside - # anchors, but not anchors inside anchors. An anchor's start - # pos must be `>= anchor_allowed_pos`. - anchor_allowed_pos = 0 - - curr_pos = 0 - while True: # Handle the next link. - # The next '[' is the start of: - # - an inline anchor: [text](url "title") - # - a reference anchor: [text][id] - # - an inline img: ![text](url "title") - # - a reference img: ![text][id] - # - a footnote ref: [^id] - # (Only if 'footnotes' extra enabled) - # - a footnote defn: [^id]: ... - # (Only if 'footnotes' extra enabled) These have already - # been stripped in _strip_footnote_definitions() so no - # need to watch for them. - # - a link definition: [id]: url "title" - # These have already been stripped in - # _strip_link_definitions() so no need to watch for them. - # - not markup: [...anything else... - try: - start_idx = text.index('[', curr_pos) - except ValueError: - break - text_length = len(text) - - # Find the matching closing ']'. - # Markdown.pl allows *matching* brackets in link text so we - # will here too. Markdown.pl *doesn't* currently allow - # matching brackets in img alt text -- we'll differ in that - # regard. - bracket_depth = 0 - for p in range(start_idx+1, min(start_idx+MAX_LINK_TEXT_SENTINEL, - text_length)): - ch = text[p] - if ch == ']': - bracket_depth -= 1 - if bracket_depth < 0: - break - elif ch == '[': - bracket_depth += 1 - else: - # Closing bracket not found within sentinel length. - # This isn't markup. - curr_pos = start_idx + 1 - continue - link_text = text[start_idx+1:p] - - # Possibly a footnote ref? - if "footnotes" in self.extras and link_text.startswith("^"): - normed_id = re.sub(r'\W', '-', link_text[1:]) - if normed_id in self.footnotes: - self.footnote_ids.append(normed_id) - result = '<sup class="footnote-ref" id="fnref-%s">' \ - '<a href="#fn-%s">%s</a></sup>' \ - % (normed_id, normed_id, len(self.footnote_ids)) - text = text[:start_idx] + result + text[p+1:] - else: - # This id isn't defined, leave the markup alone. - curr_pos = p+1 - continue - - # Now determine what this is by the remainder. - p += 1 - if p == text_length: - return text - - # Inline anchor or img? - if text[p] == '(': # attempt at perf improvement - match = self._tail_of_inline_link_re.match(text, p) - if match: - # Handle an inline anchor or img. - is_img = start_idx > 0 and text[start_idx-1] == "!" - if is_img: - start_idx -= 1 - - url, title = match.group("url"), match.group("title") - if url and url[0] == '<': - url = url[1:-1] # '<url>' -> 'url' - # We've got to encode these to avoid conflicting - # with italics/bold. - url = url.replace('*', g_escape_table['*']) \ - .replace('_', g_escape_table['_']) - if title: - title_str = ' title="%s"' \ - % title.replace('*', g_escape_table['*']) \ - .replace('_', g_escape_table['_']) \ - .replace('"', '"') - else: - title_str = '' - if is_img: - result = '<img src="%s" alt="%s"%s%s' \ - % (url, link_text.replace('"', '"'), - title_str, self.empty_element_suffix) - curr_pos = start_idx + len(result) - text = text[:start_idx] + result + text[match.end():] - elif start_idx >= anchor_allowed_pos: - result_head = '<a href="%s"%s>' % (url, title_str) - result = '%s%s</a>' % (result_head, link_text) - # <img> allowed from curr_pos on, <a> from - # anchor_allowed_pos on. - curr_pos = start_idx + len(result_head) - anchor_allowed_pos = start_idx + len(result) - text = text[:start_idx] + result + text[match.end():] - else: - # Anchor not allowed here. - curr_pos = start_idx + 1 - continue - - # Reference anchor or img? - else: - match = self._tail_of_reference_link_re.match(text, p) - if match: - # Handle a reference-style anchor or img. - is_img = start_idx > 0 and text[start_idx-1] == "!" - if is_img: - start_idx -= 1 - link_id = match.group("id").lower() - if not link_id: - link_id = link_text.lower() # for links like [this][] - if link_id in self.urls: - url = self.urls[link_id] - # We've got to encode these to avoid conflicting - # with italics/bold. - url = url.replace('*', g_escape_table['*']) \ - .replace('_', g_escape_table['_']) - title = self.titles.get(link_id) - if title: - title = title.replace('*', g_escape_table['*']) \ - .replace('_', g_escape_table['_']) - title_str = ' title="%s"' % title - else: - title_str = '' - if is_img: - result = '<img src="%s" alt="%s"%s%s' \ - % (url, link_text.replace('"', '"'), - title_str, self.empty_element_suffix) - curr_pos = start_idx + len(result) - text = text[:start_idx] + result + text[match.end():] - elif start_idx >= anchor_allowed_pos: - result = '<a href="%s"%s>%s</a>' \ - % (url, title_str, link_text) - result_head = '<a href="%s"%s>' % (url, title_str) - result = '%s%s</a>' % (result_head, link_text) - # <img> allowed from curr_pos on, <a> from - # anchor_allowed_pos on. - curr_pos = start_idx + len(result_head) - anchor_allowed_pos = start_idx + len(result) - text = text[:start_idx] + result + text[match.end():] - else: - # Anchor not allowed here. - curr_pos = start_idx + 1 - else: - # This id isn't defined, leave the markup alone. - curr_pos = match.end() - continue - - # Otherwise, it isn't markup. - curr_pos = start_idx + 1 - - return text - - - _setext_h_re = re.compile(r'^(.+)[ \t]*\n(=+|-+)[ \t]*\n+', re.M) - def _setext_h_sub(self, match): - n = {"=": 1, "-": 2}[match.group(2)[0]] - demote_headers = self.extras.get("demote-headers") - if demote_headers: - n = min(n + demote_headers, 6) - return "<h%d>%s</h%d>\n\n" \ - % (n, self._run_span_gamut(match.group(1)), n) - - _atx_h_re = re.compile(r''' - ^(\#{1,6}) # \1 = string of #'s - [ \t]* - (.+?) # \2 = Header text - [ \t]* - (?<!\\) # ensure not an escaped trailing '#' - \#* # optional closing #'s (not counted) - \n+ - ''', re.X | re.M) - def _atx_h_sub(self, match): - n = len(match.group(1)) - demote_headers = self.extras.get("demote-headers") - if demote_headers: - n = min(n + demote_headers, 6) - return "<h%d>%s</h%d>\n\n" \ - % (n, self._run_span_gamut(match.group(2)), n) - - def _do_headers(self, text): - # Setext-style headers: - # Header 1 - # ======== - # - # Header 2 - # -------- - text = self._setext_h_re.sub(self._setext_h_sub, text) - - # atx-style headers: - # # Header 1 - # ## Header 2 - # ## Header 2 with closing hashes ## - # ... - # ###### Header 6 - text = self._atx_h_re.sub(self._atx_h_sub, text) - - return text - - - _marker_ul_chars = '*+-' - _marker_any = r'(?:[%s]|\d+\.)' % _marker_ul_chars - _marker_ul = '(?:[%s])' % _marker_ul_chars - _marker_ol = r'(?:\d+\.)' - - def _list_sub(self, match): - lst = match.group(1) - lst_type = match.group(3) in self._marker_ul_chars and "ul" or "ol" - result = self._process_list_items(lst) - if self.list_level: - return "<%s>\n%s</%s>\n" % (lst_type, result, lst_type) - else: - return "<%s>\n%s</%s>\n\n" % (lst_type, result, lst_type) - - def _do_lists(self, text): - # Form HTML ordered (numbered) and unordered (bulleted) lists. - - for marker_pat in (self._marker_ul, self._marker_ol): - # Re-usable pattern to match any entire ul or ol list: - less_than_tab = self.tab_width - 1 - whole_list = r''' - ( # \1 = whole list - ( # \2 - [ ]{0,%d} - (%s) # \3 = first list item marker - [ \t]+ - ) - (?:.+?) - ( # \4 - \Z - | - \n{2,} - (?=\S) - (?! # Negative lookahead for another list item marker - [ \t]* - %s[ \t]+ - ) - ) - ) - ''' % (less_than_tab, marker_pat, marker_pat) - - # We use a different prefix before nested lists than top-level lists. - # See extended comment in _process_list_items(). - # - # Note: There's a bit of duplication here. My original implementation - # created a scalar regex pattern as the conditional result of the test on - # $g_list_level, and then only ran the $text =~ s{...}{...}egmx - # substitution once, using the scalar as the pattern. This worked, - # everywhere except when running under MT on my hosting account at Pair - # Networks. There, this caused all rebuilds to be killed by the reaper (or - # perhaps they crashed, but that seems incredibly unlikely given that the - # same script on the same server ran fine *except* under MT. I've spent - # more time trying to figure out why this is happening than I'd like to - # admit. My only guess, backed up by the fact that this workaround works, - # is that Perl optimizes the substition when it can figure out that the - # pattern will never change, and when this optimization isn't on, we run - # afoul of the reaper. Thus, the slightly redundant code to that uses two - # static s/// patterns rather than one conditional pattern. - - if self.list_level: - sub_list_re = re.compile("^"+whole_list, re.X | re.M | re.S) - text = sub_list_re.sub(self._list_sub, text) - else: - list_re = re.compile(r"(?:(?<=\n\n)|\A\n?)"+whole_list, - re.X | re.M | re.S) - text = list_re.sub(self._list_sub, text) - - return text - - _list_item_re = re.compile(r''' - (\n)? # leading line = \1 - (^[ \t]*) # leading whitespace = \2 - (%s) [ \t]+ # list marker = \3 - ((?:.+?) # list item text = \4 - (\n{1,2})) # eols = \5 - (?= \n* (\Z | \2 (%s) [ \t]+)) - ''' % (_marker_any, _marker_any), - re.M | re.X | re.S) - - _last_li_endswith_two_eols = False - def _list_item_sub(self, match): - item = match.group(4) - leading_line = match.group(1) - leading_space = match.group(2) - if leading_line or "\n\n" in item or self._last_li_endswith_two_eols: - item = self._run_block_gamut(self._outdent(item)) - else: - # Recursion for sub-lists: - item = self._do_lists(self._outdent(item)) - if item.endswith('\n'): - item = item[:-1] - item = self._run_span_gamut(item) - self._last_li_endswith_two_eols = (len(match.group(5)) == 2) - return "<li>%s</li>\n" % item - - def _process_list_items(self, list_str): - # Process the contents of a single ordered or unordered list, - # splitting it into individual list items. - - # The $g_list_level global keeps track of when we're inside a list. - # Each time we enter a list, we increment it; when we leave a list, - # we decrement. If it's zero, we're not in a list anymore. - # - # We do this because when we're not inside a list, we want to treat - # something like this: - # - # I recommend upgrading to version - # 8. Oops, now this line is treated - # as a sub-list. - # - # As a single paragraph, despite the fact that the second line starts - # with a digit-period-space sequence. - # - # Whereas when we're inside a list (or sub-list), that line will be - # treated as the start of a sub-list. What a kludge, huh? This is - # an aspect of Markdown's syntax that's hard to parse perfectly - # without resorting to mind-reading. Perhaps the solution is to - # change the syntax rules such that sub-lists must start with a - # starting cardinal number; e.g. "1." or "a.". - self.list_level += 1 - self._last_li_endswith_two_eols = False - list_str = list_str.rstrip('\n') + '\n' - list_str = self._list_item_re.sub(self._list_item_sub, list_str) - self.list_level -= 1 - return list_str - - def _get_pygments_lexer(self, lexer_name): - try: - from pygments import lexers, util - except ImportError: - return None - try: - return lexers.get_lexer_by_name(lexer_name) - except util.ClassNotFound: - return None - - def _color_with_pygments(self, codeblock, lexer, **formatter_opts): - import pygments - import pygments.formatters - - class HtmlCodeFormatter(pygments.formatters.HtmlFormatter): - def _wrap_code(self, inner): - """A function for use in a Pygments Formatter which - wraps in <code> tags. - """ - yield 0, "<code>" - for tup in inner: - yield tup - yield 0, "</code>" - - def wrap(self, source, outfile): - """Return the source with a code, pre, and div.""" - return self._wrap_div(self._wrap_pre(self._wrap_code(source))) - - formatter = HtmlCodeFormatter(cssclass="codehilite", **formatter_opts) - return pygments.highlight(codeblock, lexer, formatter) - - def _code_block_sub(self, match): - codeblock = match.group(1) - codeblock = self._outdent(codeblock) - codeblock = self._detab(codeblock) - codeblock = codeblock.lstrip('\n') # trim leading newlines - codeblock = codeblock.rstrip() # trim trailing whitespace - - if "code-color" in self.extras and codeblock.startswith(":::"): - lexer_name, rest = codeblock.split('\n', 1) - lexer_name = lexer_name[3:].strip() - lexer = self._get_pygments_lexer(lexer_name) - codeblock = rest.lstrip("\n") # Remove lexer declaration line. - if lexer: - formatter_opts = self.extras['code-color'] or {} - colored = self._color_with_pygments(codeblock, lexer, - **formatter_opts) - return "\n\n%s\n\n" % colored - - codeblock = self._encode_code(codeblock) - return "\n\n<pre><code>%s\n</code></pre>\n\n" % codeblock - - def _do_code_blocks(self, text): - """Process Markdown `<pre><code>` blocks.""" - code_block_re = re.compile(r''' - (?:\n\n|\A) - ( # $1 = the code block -- one or more lines, starting with a space/tab - (?: - (?:[ ]{%d} | \t) # Lines must start with a tab or a tab-width of spaces - .*\n+ - )+ - ) - ((?=^[ ]{0,%d}\S)|\Z) # Lookahead for non-space at line-start, or end of doc - ''' % (self.tab_width, self.tab_width), - re.M | re.X) - - return code_block_re.sub(self._code_block_sub, text) - - - # Rules for a code span: - # - backslash escapes are not interpreted in a code span - # - to include one or or a run of more backticks the delimiters must - # be a longer run of backticks - # - cannot start or end a code span with a backtick; pad with a - # space and that space will be removed in the emitted HTML - # See `test/tm-cases/escapes.text` for a number of edge-case - # examples. - _code_span_re = re.compile(r''' - (?<!\\) - (`+) # \1 = Opening run of ` - (?!`) # See Note A test/tm-cases/escapes.text - (.+?) # \2 = The code block - (?<!`) - \1 # Matching closer - (?!`) - ''', re.X | re.S) - - def _code_span_sub(self, match): - c = match.group(2).strip(" \t") - c = self._encode_code(c) - return "<code>%s</code>" % c - - def _do_code_spans(self, text): - # * Backtick quotes are used for <code></code> spans. - # - # * You can use multiple backticks as the delimiters if you want to - # include literal backticks in the code span. So, this input: - # - # Just type ``foo `bar` baz`` at the prompt. - # - # Will translate to: - # - # <p>Just type <code>foo `bar` baz</code> at the prompt.</p> - # - # There's no arbitrary limit to the number of backticks you - # can use as delimters. If you need three consecutive backticks - # in your code, use four for delimiters, etc. - # - # * You can use spaces to get literal backticks at the edges: - # - # ... type `` `bar` `` ... - # - # Turns to: - # - # ... type <code>`bar`</code> ... - return self._code_span_re.sub(self._code_span_sub, text) - - def _encode_code(self, text): - """Encode/escape certain characters inside Markdown code runs. - The point is that in code, these characters are literals, - and lose their special Markdown meanings. - """ - replacements = [ - # Encode all ampersands; HTML entities are not - # entities within a Markdown code span. - ('&', '&'), - # Do the angle bracket song and dance: - ('<', '<'), - ('>', '>'), - # Now, escape characters that are magic in Markdown: - ('*', g_escape_table['*']), - ('_', g_escape_table['_']), - ('{', g_escape_table['{']), - ('}', g_escape_table['}']), - ('[', g_escape_table['[']), - (']', g_escape_table[']']), - ('\\', g_escape_table['\\']), - ] - for before, after in replacements: - text = text.replace(before, after) - return text - - _strong_re = re.compile(r"(\*\*|__)(?=\S)(.+?[*_]*)(?<=\S)\1", re.S) - _em_re = re.compile(r"(\*|_)(?=\S)(.+?)(?<=\S)\1", re.S) - _code_friendly_strong_re = re.compile(r"\*\*(?=\S)(.+?[*_]*)(?<=\S)\*\*", re.S) - _code_friendly_em_re = re.compile(r"\*(?=\S)(.+?)(?<=\S)\*", re.S) - def _do_italics_and_bold(self, text): - # <strong> must go first: - if "code-friendly" in self.extras: - text = self._code_friendly_strong_re.sub(r"<strong>\1</strong>", text) - text = self._code_friendly_em_re.sub(r"<em>\1</em>", text) - else: - text = self._strong_re.sub(r"<strong>\2</strong>", text) - text = self._em_re.sub(r"<em>\2</em>", text) - return text - - - _block_quote_re = re.compile(r''' - ( # Wrap whole match in \1 - ( - ^[ \t]*>[ \t]? # '>' at the start of a line - .+\n # rest of the first line - (.+\n)* # subsequent consecutive lines - \n* # blanks - )+ - ) - ''', re.M | re.X) - _bq_one_level_re = re.compile('^[ \t]*>[ \t]?', re.M); - - _html_pre_block_re = re.compile(r'(\s*<pre>.+?</pre>)', re.S) - def _dedent_two_spaces_sub(self, match): - return re.sub(r'(?m)^ ', '', match.group(1)) - - def _block_quote_sub(self, match): - bq = match.group(1) - bq = self._bq_one_level_re.sub('', bq) # trim one level of quoting - bq = self._ws_only_line_re.sub('', bq) # trim whitespace-only lines - bq = self._run_block_gamut(bq) # recurse - - bq = re.sub('(?m)^', ' ', bq) - # These leading spaces screw with <pre> content, so we need to fix that: - bq = self._html_pre_block_re.sub(self._dedent_two_spaces_sub, bq) - - return "<blockquote>\n%s\n</blockquote>\n\n" % bq - - def _do_block_quotes(self, text): - if '>' not in text: - return text - return self._block_quote_re.sub(self._block_quote_sub, text) - - def _form_paragraphs(self, text): - # Strip leading and trailing lines: - text = text.strip('\n') - - # Wrap <p> tags. - grafs = re.split(r"\n{2,}", text) - for i, graf in enumerate(grafs): - if graf in self.html_blocks: - # Unhashify HTML blocks - grafs[i] = self.html_blocks[graf] - else: - # Wrap <p> tags. - graf = self._run_span_gamut(graf) - grafs[i] = "<p>" + graf.lstrip(" \t") + "</p>" - - return "\n\n".join(grafs) - - def _add_footnotes(self, text): - if self.footnotes: - footer = [ - '<div class="footnotes">', - '<hr' + self.empty_element_suffix, - '<ol>', - ] - for i, id in enumerate(self.footnote_ids): - if i != 0: - footer.append('') - footer.append('<li id="fn-%s">' % id) - footer.append(self._run_block_gamut(self.footnotes[id])) - backlink = ('<a href="#fnref-%s" ' - 'class="footnoteBackLink" ' - 'title="Jump back to footnote %d in the text.">' - '↩</a>' % (id, i+1)) - if footer[-1].endswith("</p>"): - footer[-1] = footer[-1][:-len("</p>")] \ - + ' ' + backlink + "</p>" - else: - footer.append("\n<p>%s</p>" % backlink) - footer.append('</li>') - footer.append('</ol>') - footer.append('</div>') - return text + '\n\n' + '\n'.join(footer) - else: - return text - - # Ampersand-encoding based entirely on Nat Irons's Amputator MT plugin: - # http://bumppo.net/projects/amputator/ - _ampersand_re = re.compile(r'&(?!#?[xX]?(?:[0-9a-fA-F]+|\w+);)') - _naked_lt_re = re.compile(r'<(?![a-z/?\$!])', re.I) - _naked_gt_re = re.compile(r'''(?<![a-z?!/'"-])>''', re.I) - - def _encode_amps_and_angles(self, text): - # Smart processing for ampersands and angle brackets that need - # to be encoded. - text = self._ampersand_re.sub('&', text) - - # Encode naked <'s - text = self._naked_lt_re.sub('<', text) - - # Encode naked >'s - # Note: Other markdown implementations (e.g. Markdown.pl, PHP - # Markdown) don't do this. - text = self._naked_gt_re.sub('>', text) - return text - - def _encode_backslash_escapes(self, text): - for ch, escape in g_escape_table.items(): - text = text.replace("\\"+ch, escape) - return text - - _auto_link_re = re.compile(r'<((https?|ftp):[^\'">\s]+)>', re.I) - def _auto_link_sub(self, match): - g1 = match.group(1) - return '<a href="%s">%s</a>' % (g1, g1) - - _auto_email_link_re = re.compile(r""" - < - (?:mailto:)? - ( - [-.\w]+ - \@ - [-\w]+(\.[-\w]+)*\.[a-z]+ - ) - > - """, re.I | re.X | re.U) - def _auto_email_link_sub(self, match): - return self._encode_email_address( - self._unescape_special_chars(match.group(1))) - - def _do_auto_links(self, text): - text = self._auto_link_re.sub(self._auto_link_sub, text) - text = self._auto_email_link_re.sub(self._auto_email_link_sub, text) - return text - - def _encode_email_address(self, addr): - # Input: an email address, e.g. "foo@example.com" - # - # Output: the email address as a mailto link, with each character - # of the address encoded as either a decimal or hex entity, in - # the hopes of foiling most address harvesting spam bots. E.g.: - # - # <a href="mailto:foo@e - # xample.com">foo - # @example.com</a> - # - # Based on a filter by Matthew Wickline, posted to the BBEdit-Talk - # mailing list: <http://tinyurl.com/yu7ue> - chars = [_xml_encode_email_char_at_random(ch) - for ch in "mailto:" + addr] - # Strip the mailto: from the visible part. - addr = '<a href="%s">%s</a>' \ - % (''.join(chars), ''.join(chars[7:])) - return addr - - def _do_link_patterns(self, text): - """Caveat emptor: there isn't much guarding against link - patterns being formed inside other standard Markdown links, e.g. - inside a [link def][like this]. - - Dev Notes: *Could* consider prefixing regexes with a negative - lookbehind assertion to attempt to guard against this. - """ - link_from_hash = {} - for regex, repl in self.link_patterns: - replacements = [] - for match in regex.finditer(text): - if hasattr(repl, "__call__"): - href = repl(match) - else: - href = match.expand(repl) - replacements.append((match.span(), href)) - for (start, end), href in reversed(replacements): - escaped_href = ( - href.replace('"', '"') # b/c of attr quote - # To avoid markdown <em> and <strong>: - .replace('*', g_escape_table['*']) - .replace('_', g_escape_table['_'])) - link = '<a href="%s">%s</a>' % (escaped_href, text[start:end]) - hash = md5(link).hexdigest() - link_from_hash[hash] = link - text = text[:start] + hash + text[end:] - for hash, link in link_from_hash.items(): - text = text.replace(hash, link) - return text - - def _unescape_special_chars(self, text): - # Swap back in all the special characters we've hidden. - for ch, hash in g_escape_table.items(): - text = text.replace(hash, ch) - return text - - def _outdent(self, text): - # Remove one level of line-leading tabs or spaces - return self._outdent_re.sub('', text) - - -class MarkdownWithExtras(Markdown): - """A markdowner class that enables most extras: - - - footnotes - - code-color (only has effect if 'pygments' Python module on path) - - These are not included: - - pyshell (specific to Python-related documenting) - - code-friendly (because it *disables* part of the syntax) - - link-patterns (because you need to specify some actual - link-patterns anyway) - """ - extras = ["footnotes", "code-color"] - - -#---- internal support functions - -# From http://aspn.activestate.com/ASPN/Cookbook/Python/Recipe/52549 -def _curry(*args, **kwargs): - function, args = args[0], args[1:] - def result(*rest, **kwrest): - combined = kwargs.copy() - combined.update(kwrest) - return function(*args + rest, **combined) - return result - -# Recipe: regex_from_encoded_pattern (1.0) -def _regex_from_encoded_pattern(s): - """'foo' -> re.compile(re.escape('foo')) - '/foo/' -> re.compile('foo') - '/foo/i' -> re.compile('foo', re.I) - """ - if s.startswith('/') and s.rfind('/') != 0: - # Parse it: /PATTERN/FLAGS - idx = s.rfind('/') - pattern, flags_str = s[1:idx], s[idx+1:] - flag_from_char = { - "i": re.IGNORECASE, - "l": re.LOCALE, - "s": re.DOTALL, - "m": re.MULTILINE, - "u": re.UNICODE, - } - flags = 0 - for char in flags_str: - try: - flags |= flag_from_char[char] - except KeyError: - raise ValueError("unsupported regex flag: '%s' in '%s' " - "(must be one of '%s')" - % (char, s, ''.join(flag_from_char.keys()))) - return re.compile(s[1:idx], flags) - else: # not an encoded regex - return re.compile(re.escape(s)) - -# Recipe: dedent (0.1.2) -def _dedentlines(lines, tabsize=8, skip_first_line=False): - """_dedentlines(lines, tabsize=8, skip_first_line=False) -> dedented lines - - "lines" is a list of lines to dedent. - "tabsize" is the tab width to use for indent width calculations. - "skip_first_line" is a boolean indicating if the first line should - be skipped for calculating the indent width and for dedenting. - This is sometimes useful for docstrings and similar. - - Same as dedent() except operates on a sequence of lines. Note: the - lines list is modified **in-place**. - """ - DEBUG = False - if DEBUG: - print "dedent: dedent(..., tabsize=%d, skip_first_line=%r)"\ - % (tabsize, skip_first_line) - indents = [] - margin = None - for i, line in enumerate(lines): - if i == 0 and skip_first_line: continue - indent = 0 - for ch in line: - if ch == ' ': - indent += 1 - elif ch == '\t': - indent += tabsize - (indent % tabsize) - elif ch in '\r\n': - continue # skip all-whitespace lines - else: - break - else: - continue # skip all-whitespace lines - if DEBUG: print "dedent: indent=%d: %r" % (indent, line) - if margin is None: - margin = indent - else: - margin = min(margin, indent) - if DEBUG: print "dedent: margin=%r" % margin - - if margin is not None and margin > 0: - for i, line in enumerate(lines): - if i == 0 and skip_first_line: continue - removed = 0 - for j, ch in enumerate(line): - if ch == ' ': - removed += 1 - elif ch == '\t': - removed += tabsize - (removed % tabsize) - elif ch in '\r\n': - if DEBUG: print "dedent: %r: EOL -> strip up to EOL" % line - lines[i] = lines[i][j:] - break - else: - raise ValueError("unexpected non-whitespace char %r in " - "line %r while removing %d-space margin" - % (ch, line, margin)) - if DEBUG: - print "dedent: %r: %r -> removed %d/%d"\ - % (line, ch, removed, margin) - if removed == margin: - lines[i] = lines[i][j+1:] - break - elif removed > margin: - lines[i] = ' '*(removed-margin) + lines[i][j+1:] - break - else: - if removed: - lines[i] = lines[i][removed:] - return lines - -def _dedent(text, tabsize=8, skip_first_line=False): - """_dedent(text, tabsize=8, skip_first_line=False) -> dedented text - - "text" is the text to dedent. - "tabsize" is the tab width to use for indent width calculations. - "skip_first_line" is a boolean indicating if the first line should - be skipped for calculating the indent width and for dedenting. - This is sometimes useful for docstrings and similar. - - textwrap.dedent(s), but don't expand tabs to spaces - """ - lines = text.splitlines(1) - _dedentlines(lines, tabsize=tabsize, skip_first_line=skip_first_line) - return ''.join(lines) - - -class _memoized(object): - """Decorator that caches a function's return value each time it is called. - If called later with the same arguments, the cached value is returned, and - not re-evaluated. - - http://wiki.python.org/moin/PythonDecoratorLibrary - """ - def __init__(self, func): - self.func = func - self.cache = {} - def __call__(self, *args): - try: - return self.cache[args] - except KeyError: - self.cache[args] = value = self.func(*args) - return value - except TypeError: - # uncachable -- for instance, passing a list as an argument. - # Better to not cache than to blow up entirely. - return self.func(*args) - def __repr__(self): - """Return the function's docstring.""" - return self.func.__doc__ - - -def _xml_oneliner_re_from_tab_width(tab_width): - """Standalone XML processing instruction regex.""" - return re.compile(r""" - (?: - (?<=\n\n) # Starting after a blank line - | # or - \A\n? # the beginning of the doc - ) - ( # save in $1 - [ ]{0,%d} - (?: - <\?\w+\b\s+.*?\?> # XML processing instruction - | - <\w+:\w+\b\s+.*?/> # namespaced single tag - ) - [ \t]* - (?=\n{2,}|\Z) # followed by a blank line or end of document - ) - """ % (tab_width - 1), re.X) -_xml_oneliner_re_from_tab_width = _memoized(_xml_oneliner_re_from_tab_width) - -def _hr_tag_re_from_tab_width(tab_width): - return re.compile(r""" - (?: - (?<=\n\n) # Starting after a blank line - | # or - \A\n? # the beginning of the doc - ) - ( # save in \1 - [ ]{0,%d} - <(hr) # start tag = \2 - \b # word break - ([^<>])*? # - /?> # the matching end tag - [ \t]* - (?=\n{2,}|\Z) # followed by a blank line or end of document - ) - """ % (tab_width - 1), re.X) -_hr_tag_re_from_tab_width = _memoized(_hr_tag_re_from_tab_width) - - -def _xml_encode_email_char_at_random(ch): - r = random() - # Roughly 10% raw, 45% hex, 45% dec. - # '@' *must* be encoded. I [John Gruber] insist. - # Issue 26: '_' must be encoded. - if r > 0.9 and ch not in "@_": - return ch - elif r < 0.45: - # The [1:] is to drop leading '0': 0x63 -> x63 - return '&#%s;' % hex(ord(ch))[1:] - else: - return '&#%s;' % ord(ch) - -def _hash_text(text): - return 'md5:'+md5(text.encode("utf-8")).hexdigest() - - -#---- mainline - -class _NoReflowFormatter(optparse.IndentedHelpFormatter): - """An optparse formatter that does NOT reflow the description.""" - def format_description(self, description): - return description or "" - -def _test(): - import doctest - doctest.testmod() - -def main(argv=None): - if argv is None: - argv = sys.argv - if not logging.root.handlers: - logging.basicConfig() - - usage = "usage: %prog [PATHS...]" - version = "%prog "+__version__ - parser = optparse.OptionParser(prog="markdown2", usage=usage, - version=version, description=cmdln_desc, - formatter=_NoReflowFormatter()) - parser.add_option("-v", "--verbose", dest="log_level", - action="store_const", const=logging.DEBUG, - help="more verbose output") - parser.add_option("--encoding", - help="specify encoding of text content") - parser.add_option("--html4tags", action="store_true", default=False, - help="use HTML 4 style for empty element tags") - parser.add_option("-s", "--safe", metavar="MODE", dest="safe_mode", - help="sanitize literal HTML: 'escape' escapes " - "HTML meta chars, 'replace' replaces with an " - "[HTML_REMOVED] note") - parser.add_option("-x", "--extras", action="append", - help="Turn on specific extra features (not part of " - "the core Markdown spec). Supported values: " - "'code-friendly' disables _/__ for emphasis; " - "'code-color' adds code-block syntax coloring; " - "'link-patterns' adds auto-linking based on patterns; " - "'footnotes' adds the footnotes syntax;" - "'xml' passes one-liner processing instructions and namespaced XML tags;" - "'pyshell' to put unindented Python interactive shell sessions in a <code> block.") - parser.add_option("--use-file-vars", - help="Look for and use Emacs-style 'markdown-extras' " - "file var to turn on extras. See " - "<http://code.google.com/p/python-markdown2/wiki/Extras>.") - parser.add_option("--link-patterns-file", - help="path to a link pattern file") - parser.add_option("--self-test", action="store_true", - help="run internal self-tests (some doctests)") - parser.add_option("--compare", action="store_true", - help="run against Markdown.pl as well (for testing)") - parser.set_defaults(log_level=logging.INFO, compare=False, - encoding="utf-8", safe_mode=None, use_file_vars=False) - opts, paths = parser.parse_args() - log.setLevel(opts.log_level) - - if opts.self_test: - return _test() - - if opts.extras: - extras = {} - for s in opts.extras: - splitter = re.compile("[,;: ]+") - for e in splitter.split(s): - if '=' in e: - ename, earg = e.split('=', 1) - try: - earg = int(earg) - except ValueError: - pass - else: - ename, earg = e, None - extras[ename] = earg - else: - extras = None - - if opts.link_patterns_file: - link_patterns = [] - f = open(opts.link_patterns_file) - try: - for i, line in enumerate(f.readlines()): - if not line.strip(): continue - if line.lstrip().startswith("#"): continue - try: - pat, href = line.rstrip().rsplit(None, 1) - except ValueError: - raise MarkdownError("%s:%d: invalid link pattern line: %r" - % (opts.link_patterns_file, i+1, line)) - link_patterns.append( - (_regex_from_encoded_pattern(pat), href)) - finally: - f.close() - else: - link_patterns = None - - from os.path import join, dirname, abspath, exists - markdown_pl = join(dirname(dirname(abspath(__file__))), "test", - "Markdown.pl") - for path in paths: - if opts.compare: - print "==== Markdown.pl ====" - perl_cmd = 'perl %s "%s"' % (markdown_pl, path) - o = os.popen(perl_cmd) - perl_html = o.read() - o.close() - sys.stdout.write(perl_html) - print "==== markdown2.py ====" - html = markdown_path(path, encoding=opts.encoding, - html4tags=opts.html4tags, - safe_mode=opts.safe_mode, - extras=extras, link_patterns=link_patterns, - use_file_vars=opts.use_file_vars) - sys.stdout.write( - html.encode(sys.stdout.encoding or "utf-8", 'xmlcharrefreplace')) - if opts.compare: - test_dir = join(dirname(dirname(abspath(__file__))), "test") - if exists(join(test_dir, "test_markdown2.py")): - sys.path.insert(0, test_dir) - from test_markdown2 import norm_html_from_html - norm_html = norm_html_from_html(html) - norm_perl_html = norm_html_from_html(perl_html) - else: - norm_html = html - norm_perl_html = perl_html - print "==== match? %r ====" % (norm_perl_html == norm_html) - - -if __name__ == "__main__": - sys.exit( main(sys.argv) ) - diff --git a/lib/pydelicious.py b/lib/pydelicious.py deleted file mode 100644 index 8e45843..0000000 --- a/lib/pydelicious.py +++ /dev/null @@ -1,1045 +0,0 @@ -"""Library to access del.icio.us data via Python. - -An introduction to the project is given in the README. -pydelicious is released under the BSD license. See license.txt for details -and the copyright holders. - -THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND -ANY EXPRESS 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 THE COPYRIGHT OWNER OR 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. - -TODO: - - distribute license, readme docs via setup.py? - - automatic release build? -""" -import sys -import os -import time -import datetime -import locale -import httplib -import urllib2 -from urllib import urlencode, quote_plus -from StringIO import StringIO -from pprint import pformat - -v = sys.version_info -if v[0] >= 2 and v[1] >= 5: - from hashlib import md5 -else: - from md5 import md5 - -try: - from elementtree.ElementTree import parse as parse_xml -except ImportError: - # Python 2.5 and higher - from xml.etree.ElementTree import parse as parse_xml - -try: - import feedparser -except ImportError: - print >>sys.stderr, \ - "Feedparser not available, no RSS parsing." - feedparser = None - - -### Static config - -__version__ = '0.5.3' -__author__ = 'Frank Timmermann <regenkind_at_gmx_dot_de>' - # GP: does not respond to emails -__contributors__ = [ - 'Greg Pinero', - 'Berend van Berkum <berend+pydelicious@dotmpe.com>'] -__url__ = 'http://code.google.com/p/pydelicious/' -# Old URL: 'http://deliciouspython.python-hosting.com/' -__author_email__ = "" -__docformat__ = "restructuredtext en" -__description__ = "pydelicious.py allows you to access the web service of " \ - "del.icio.us via it's API through Python." -__long_description__ = "The goal is to design an easy to use and fully " \ - "functional Python interface to del.icio.us." - -DLCS_OK_MESSAGES = ('done', 'ok') -"Known text values of positive del.icio.us <result/> answers" -DLCS_WAIT_TIME = 4 -"Time to wait between API requests" -DLCS_REQUEST_TIMEOUT = 444 -"Seconds before socket triggers timeout" -#DLCS_API_REALM = 'del.icio.us API' -DLCS_API_HOST = 'api.del.icio.us' -DLCS_API_PATH = 'v1' -DLCS_API = "https://%s/%s" % (DLCS_API_HOST, DLCS_API_PATH) -DLCS_RSS = 'http://del.icio.us/rss/' -DLCS_FEEDS = 'http://feeds.delicious.com/v2/' - -PREFERRED_ENCODING = locale.getpreferredencoding() -# XXX: might need to check sys.platform/encoding combinations here, ie -#if sys.platform == 'darwin' || PREFERRED_ENCODING == 'macroman: -# PREFERRED_ENCODING = 'utf-8' -if not PREFERRED_ENCODING: - PREFERRED_ENCODING = 'iso-8859-1' - -ISO_8601_DATETIME = '%Y-%m-%dT%H:%M:%SZ' - -USER_AGENT = 'pydelicious/%s %s' % (__version__, __url__) - -DEBUG = 0 -if 'DLCS_DEBUG' in os.environ: - DEBUG = int(os.environ['DLCS_DEBUG']) - if DEBUG: - print >>sys.stderr, \ - "Set DEBUG to %i from DLCS_DEBUG env." % DEBUG - -HTTP_PROXY = None -if 'HTTP_PROXY' in os.environ: - HTTP_PROXY = os.environ['HTTP_PROXY'] - if DEBUG: - print >>sys.stderr, \ - "Set HTTP_PROXY to %i from env." % HTTP_PROXY - -### Timeoutsocket hack taken from FeedParser.py - -# timeoutsocket allows feedparser to time out rather than hang forever on ultra- -# slow servers. Python 2.3 now has this functionality available in the standard -# socket library, so under 2.3 you don't need to install anything. But you -# probably should anyway, because the socket module is buggy and timeoutsocket -# is better. -try: - import timeoutsocket # http://www.timo-tasi.org/python/timeoutsocket.py - timeoutsocket.setDefaultSocketTimeout(DLCS_REQUEST_TIMEOUT) -except ImportError: - import socket - if hasattr(socket, 'setdefaulttimeout'): - socket.setdefaulttimeout(DLCS_REQUEST_TIMEOUT) -if DEBUG: print >>sys.stderr, \ - "Set socket timeout to %s seconds" % DLCS_REQUEST_TIMEOUT - - -### Utility classes - -class _Waiter: - """Waiter makes sure a certain amount of time passes between - successive calls of `Waiter()`. - - Some attributes: - :last: time of last call - :wait: the minimum time needed between calls - :waited: the number of calls throttled - - pydelicious.Waiter is an instance created when the module is loaded. - """ - def __init__(self, wait): - self.wait = wait - self.waited = 0 - self.lastcall = 0; - - def __call__(self): - tt = time.time() - wait = self.wait - - timeago = tt - self.lastcall - - if timeago < wait: - wait = wait - timeago - if DEBUG>0: print >>sys.stderr, "Waiting %s seconds." % wait - time.sleep(wait) - self.waited += 1 - self.lastcall = tt + wait - else: - self.lastcall = tt - -Waiter = _Waiter(DLCS_WAIT_TIME) - - -class PyDeliciousException(Exception): - """Standard pydelicious error""" -class PyDeliciousThrottled(Exception): pass -class PyDeliciousUnauthorized(Exception): pass - -class DeliciousError(Exception): - """Raised when the server responds with a negative answer""" - - @staticmethod - def raiseFor(error_string, path, **params): - if error_string == 'item already exists': - raise DeliciousItemExistsError, params['url'] - else: - raise DeliciousError, "%s, while calling <%s?%s>" % (error_string, - path, urlencode(params)) - -class DeliciousItemExistsError(DeliciousError): - """Raised then adding an already existing post.""" - - -class HTTPErrorHandler(urllib2.HTTPDefaultErrorHandler): - - def http_error_401(self, req, fp, code, msg, headers): - raise PyDeliciousUnauthorized, "Check credentials." - - def http_error_503(self, req, fp, code, msg, headers): - # Retry-After? - errmsg = "Try again later." - if 'Retry-After' in headers: - errmsg = "You may try again after %s" % headers['Retry-After'] - raise PyDeliciousThrottled, errmsg - - -### Utility functions - -def dict0(d): - "Removes empty string values from dictionary" - return dict([(k,v) for k,v in d.items() - if v=='' and isinstance(v, basestring)]) - - -def delicious_datetime(str): - """Parse a ISO 8601 formatted string to a Python datetime ... - """ - return datetime.datetime(*time.strptime(str, ISO_8601_DATETIME)[0:6]) - - -def http_request(url, user_agent=USER_AGENT, retry=4, opener=None): - """Retrieve the contents referenced by the URL using urllib2. - - Retries up to four times (default) on exceptions. - """ - request = urllib2.Request(url, headers={'User-Agent':user_agent}) - - if not opener: - opener = urllib2.build_opener() - - # Remember last error - e = None - - # Repeat request on time-out errors - tries = retry; - while tries: - try: - return opener.open(request) - - except urllib2.HTTPError, e: - # reraise unexpected protocol errors as PyDeliciousException - raise PyDeliciousException, "%s" % e - - except urllib2.URLError, e: - # xxx: Ugly check for time-out errors - #if len(e)>0 and 'timed out' in arg[0]: - print >> sys.stderr, "%s, %s tries left." % (e, tries) - Waiter() - tries = tries - 1 - #else: - # tries = None - - # Give up - raise PyDeliciousException, \ - "Unable to retrieve data at '%s', %s" % (url, e) - - -def build_api_opener(host, user, passwd, extra_handlers=() ): - """ - Build a urllib2 style opener with HTTP Basic authorization for one host - and additional error handling. If HTTP_PROXY is set a proxyhandler is also - added. - """ - - global DEBUG - - if DEBUG: httplib.HTTPConnection.debuglevel = 1 - - password_manager = urllib2.HTTPPasswordMgrWithDefaultRealm() - password_manager.add_password(None, host, user, passwd) - auth_handler = urllib2.HTTPBasicAuthHandler(password_manager) - - extra_handlers += ( HTTPErrorHandler(), ) - if HTTP_PROXY: - extra_handlers += ( urllib2.ProxyHandler( {'http': HTTP_PROXY} ), ) - - return urllib2.build_opener(auth_handler, *extra_handlers) - - -def dlcs_api_opener(user, passwd): - "Build an opener for DLCS_API_HOST, see build_api_opener()" - - return build_api_opener(DLCS_API_HOST, user, passwd) - - -def dlcs_api_request(path, params='', user='', passwd='', throttle=True, - opener=None): - """Retrieve/query a path within the del.icio.us API. - - This implements a minimum interval between calls to avoid - throttling. [#]_ Use param 'throttle' to turn this behaviour off. - - .. [#] http://del.icio.us/help/api/ - """ - if throttle: - Waiter() - - if params: - url = "%s/%s?%s" % (DLCS_API, path, urlencode(params)) - else: - url = "%s/%s" % (DLCS_API, path) - - if DEBUG: print >>sys.stderr, \ - "dlcs_api_request: %s" % url - - if not opener: - opener = dlcs_api_opener(user, passwd) - - fl = http_request(url, opener=opener) - - if DEBUG>2: print >>sys.stderr, \ - pformat(fl.info().headers) - - return fl - - -def dlcs_encode_params(params, usercodec=PREFERRED_ENCODING): - """Turn all param values (int, list, bool) into utf8 encoded strings. - """ - - if params: - for key in params.keys(): - if isinstance(params[key], bool): - if params[key]: - params[key] = 'yes' - else: - params[key] = 'no' - - elif isinstance(params[key], int): - params[key] = str(params[key]) - - elif not params[key]: - # strip/ignore empties other than False or 0 - del params[key] - continue - - elif isinstance(params[key], list): - params[key] = " ".join(params[key]) - - elif not isinstance(params[key], unicode): - params[key] = params[key].decode(usercodec) - - assert isinstance(params[key], basestring) - - params = dict([ (k, v.encode('utf8')) - for k, v in params.items() if v]) - - return params - - -def dlcs_parse_xml(data, split_tags=False): - """Parse any del.icio.us XML document and return Python data structure. - - Recognizes all XML document formats as returned by the version 1 API and - translates to a JSON-like data structure (dicts 'n lists). - - Returned instance is always a dictionary. Examples:: - - {'posts': [{'url':'...','hash':'...',},],} - {'tags':['tag1', 'tag2',]} - {'dates': [{'count':'...','date':'...'},], 'tag':'', 'user':'...'} - {'result':(True, "done")} - # etcetera. - """ - # TODO: split_tags is not implemented - - if DEBUG>3: print >>sys.stderr, "dlcs_parse_xml: parsing from ", data - - if not hasattr(data, 'read'): - data = StringIO(data) - - doc = parse_xml(data) - root = doc.getroot() - fmt = root.tag - - # Split up into three cases: Data, Result or Update - if fmt in ('tags', 'posts', 'dates', 'bundles'): - - # Data: expect a list of data elements, 'resources'. - # Use `fmt` (without last 's') to find data elements, elements - # don't have contents, attributes contain all the data we need: - # append to list - elist = [el.attrib for el in doc.findall(fmt[:-1])] - - # Return list in dict, use tagname of rootnode as keyname. - data = {fmt: elist} - - # Root element might have attributes too, append dict. - data.update(root.attrib) - - return data - - elif fmt == 'result': - - # Result: answer to operations - if root.attrib.has_key('code'): - msg = root.attrib['code'] - else: - msg = root.text - - # XXX: Return {'result':(True, msg)} for /known/ O.K. messages, - # use (False, msg) otherwise. Move this to DeliciousAPI? - v = msg in DLCS_OK_MESSAGES - return {fmt: (v, msg)} - - elif fmt == 'update': - - # Update: "time" - return {fmt: { - 'time':time.strptime(root.attrib['time'], ISO_8601_DATETIME) }} - - else: - raise PyDeliciousException, "Unknown XML document format '%s'" % fmt - - -def dlcs_rss_request(tag="", popular=0, user="", url=''): - """Parse a RSS request. - - This requests old (now undocumented?) URL paths that still seem to work. - """ - - tag = quote_plus(tag) - user = quote_plus(user) - - if url != '': - # http://del.icio.us/rss/url/efbfb246d886393d48065551434dab54 - url = DLCS_RSS + 'url/%s' % md5(url).hexdigest() - - elif user != '' and tag != '': - url = DLCS_RSS + '%(user)s/%(tag)s' % {'user':user, 'tag':tag} - - elif user != '' and tag == '': - # http://del.icio.us/rss/delpy - url = DLCS_RSS + '%s' % user - - elif popular == 0 and tag == '': - url = DLCS_RSS - - elif popular == 0 and tag != '': - # http://del.icio.us/rss/tag/apple - # http://del.icio.us/rss/tag/web2.0 - url = DLCS_RSS + "tag/%s" % tag - - elif popular == 1 and tag == '': - url = DLCS_RSS + 'popular/' - - elif popular == 1 and tag != '': - url = DLCS_RSS + 'popular/%s' % tag - - if DEBUG: - print 'dlcs_rss_request', url - - rss = http_request(url).read() - - # assert feedparser, "dlcs_rss_request requires feedparser to be installed." - if not feedparser: - return rss - - rss = feedparser.parse(rss) - - posts = [] - for e in rss.entries: - if e.has_key("links") and e["links"]!=[] and e["links"][0].has_key("href"): - url = e["links"][0]["href"] - elif e.has_key("link"): - url = e["link"] - elif e.has_key("id"): - url = e["id"] - else: - url = "" - if e.has_key("title"): - description = e['title'] - elif e.has_key("title_detail") and e["title_detail"].has_key("title"): - description = e["title_detail"]['value'] - else: - description = '' - try: tags = e['categories'][0][1] - except: - try: tags = e["category"] - except: tags = "" - if e.has_key("modified"): - dt = e['modified'] - else: - dt = "" - if e.has_key("summary"): - extended = e['summary'] - elif e.has_key("summary_detail"): - e['summary_detail']["value"] - else: - extended = "" - if e.has_key("author"): - user = e['author'] - else: - user = "" - # time = dt ist weist auf ein problem hin - # die benennung der variablen ist nicht einheitlich - # api senden und - # xml bekommen sind zwei verschiedene schuhe :( - posts.append({'url':url, 'description':description, 'tags':tags, - 'dt':dt, 'extended':extended, 'user':user}) - return posts - - -delicious_v2_feeds = { - #"Bookmarks from the hotlist" - '': "%(format)s", - #"Recent bookmarks" - 'recent': "%(format)s/recent", - #"Recent bookmarks by tag" - 'tagged': "%(format)s/tag/%(tags)s", - #"Popular bookmarks" - 'popular': "%(format)s/popular", - #"Popular bookmarks by tag" - 'popular_tagged': "%(format)s/popular/%(tag)s", - #"Recent site alerts (as seen in the top-of-page alert bar on the site)" - 'alerts': "%(format)s/alerts", - #"Bookmarks for a specific user" - 'user': "%(format)s/%(username)s", - #"Bookmarks for a specific user by tag(s)" - 'user_tagged': "%(format)s/%(username)s/%(tags)s", - #"Public summary information about a user (as seen in the network badge)" - 'user_info': "%(format)s/userinfo/%(username)s", - #"A list of all public tags for a user" - 'user_tags': "%(format)s/tags/%(username)s", - #"Bookmarks from a user's subscriptions" - 'user_subscription': "%(format)s/subscriptions/%(username)s", - #"Private feed for a user's inbox bookmarks from others" - 'user_inbox': "%(format)s/inbox/%(username)s?private=%(key)s", - #"Bookmarks from members of a user's network" - 'user_network': "%(format)s/network/%(username)s", - #"Bookmarks from members of a user's network by tag" - 'user_network_tagged': "%(format)s/network/%(username)s/%(tags)s", - #"A list of a user's network members" - 'user_network_member': "%(format)s/networkmembers/%(username)s", - #"A list of a user's network fans" - 'user_network_fan': "%(format)s/networkfans/%(username)s", - #"Recent bookmarks for a URL" - 'url': "%(format)s/url/%(urlmd5)s", - #"Summary information about a URL (as seen in the tagometer)" - 'urlinfo': "json/urlinfo/%(urlmd5)s", -} - -def dlcs_feed(name_or_url, url_map=delicious_v2_feeds, count=15, **params): - - """ - Request and parse a feed. See delicious_v2_feeds for available names and - required parameters. Format defaults to json. - """ - -# http://delicious.com/help/feeds -# TODO: plain or fancy - - format = params.setdefault('format', 'json') - if count == 'all': -# TODO: fetch all - print >>sys.stderr, "! Maxcount 100 " - count = 100 - - if name_or_url in url_map: - params['count'] = count - url = DLCS_FEEDS + url_map[name_or_url] % params - - else: - url = name_or_url - - if DEBUG: - print 'dlcs_feed', url - - feed = http_request(url).read() - - if format == 'rss': - if feedparser: - rss = feedparser.parse(feed) - return rss - - else: - return feed - - elif format == 'json': - return feed - - -### Main module class - -class DeliciousAPI: - - """A single-user Python facade to the del.icio.us HTTP API. - - See http://delicious.com/help/api. - - Methods ``request`` and ``request_raw`` represent the core. For all API - paths there are furthermore methods (e.g. posts_add for 'posts/all') with - an explicit declaration of parameters and documentation. - """ - - def __init__(self, user, passwd, codec=PREFERRED_ENCODING, - api_request=dlcs_api_request, xml_parser=dlcs_parse_xml, - build_opener=dlcs_api_opener, encode_params=dlcs_encode_params): - - """Initialize access to the API for ``user`` with ``passwd``. - - ``codec`` sets the encoding of the arguments, which defaults to the - users preferred locale. - - The ``api_request`` and ``xml_parser`` parameters by default point to - functions within this package with standard implementations which - request and parse a resource. See ``dlcs_api_request()`` and - ``dlcs_parse_xml()``. - - Parameter ``build_opener`` is a callable that, provided with the - credentials, should build a urllib2 opener for the delicious API server - with HTTP authentication. See ``dlcs_api_opener()`` for the default - implementation. - - ``encode_params`` finally preprocesses API parameters before - they are passed to ``api_request``. - """ - - assert user != "" - self.user = user - self.passwd = passwd - self.codec = codec - - # Implement communication to server and parsing of respons messages: - assert callable(encode_params) - self._encode_params = encode_params - assert callable(build_opener) - self._opener = build_opener(user, passwd) - assert callable(api_request) - self._api_request = api_request - assert callable(xml_parser) - self._parse_response = xml_parser - - ### Core functionality - - def request(self, path, _raw=False, **params): - """Sends a request message to `path` in the API, and parses the results - from XML. Use with ``_raw=True`` or ``call request_raw()`` directly - to get the filehandler and process the response message manually. - - Calls to some paths will return a `result` message, i.e.:: - - <result code="..." /> - - or:: - - <result>...</result> - - These should all be parsed to ``{'result':(Boolean, MessageString)}``, - this method raises a ``DeliciousError`` on negative `result` answers. - Positive answers are silently accepted and nothing is returned. - - Using ``_raw=True`` bypasses all parsing and never raises - ``DeliciousError``. - - See ``dlcs_parse_xml()`` and ``self.request_raw()``.""" - - if _raw: - # return answer - return self.request_raw(path, **params) - - else: - params = self._encode_params(params, self.codec) - - # get answer and parse - fl = self._api_request(path, params=params, opener=self._opener) - rs = self._parse_response(fl) - - if type(rs) == dict and 'result' in rs: - if not rs['result'][0]: - # Raise an error for negative 'result' answers - errmsg = "" - if len(rs['result'])>0: - errmsg = rs['result'][1] - DeliciousError.raiseFor(errmsg, path, **params) - - else: - # not out-of-the-oridinary result, OK - return - - return rs - - def request_raw(self, path, **params): - """Calls the path in the API, returns the filehandle. Returned file- - like instances have an ``HTTPMessage`` instance with HTTP header - information available. Use ``filehandle.info()`` or refer to the - ``urllib2.openurl`` documentation. - """ - # see `request()` on how the response can be handled - params = self._encode_params(params, self.codec) - return self._api_request(path, params=params, opener=self._opener) - - ### Explicit declarations of API paths, their parameters and docs - - # Tags - def tags_get(self, **kwds): - """Returns a list of tags and the number of times it is used by the - user. - :: - - <tags> - <tag tag="TagName" count="888"> - """ - return self.request("tags/get", **kwds) - - def tags_delete(self, tag, **kwds): - """Delete an existing tag. - - &tag={TAG} - (required) Tag to delete - """ - return self.request('tags/delete', tag=tag, **kwds) - - def tags_rename(self, old, new, **kwds): - """Rename an existing tag with a new tag name. Returns a `result` - message or raises an ``DeliciousError``. See ``self.request()``. - - &old={TAG} - (required) Tag to rename. - &new={TAG} - (required) New tag name. - """ - return self.request("tags/rename", old=old, new=new, **kwds) - - # Posts - def posts_update(self, **kwds): - """Returns the last update time for the user. Use this before calling - `posts_all` to see if the data has changed since the last fetch. - :: - - <update time="CCYY-MM-DDThh:mm:ssZ"> - """ - return self.request("posts/update", **kwds) - - def posts_dates(self, tag="", **kwds): - """Returns a list of dates with the number of posts at each date. - :: - - <dates> - <date date="CCYY-MM-DD" count="888"> - - &tag={TAG} - (optional) Filter by this tag - """ - return self.request("posts/dates", tag=tag, **kwds) - - def posts_get(self, tag="", dt="", url="", hashes=[], meta=True, **kwds): - """Returns posts matching the arguments. If no date or url is given, - most recent date will be used. - :: - - <posts dt="CCYY-MM-DD" tag="..." user="..."> - <post ...> - - &tag={TAG} {TAG} ... {TAG} - (optional) Filter by this/these tag(s). - &dt={CCYY-MM-DDThh:mm:ssZ} - (optional) Filter by this date, defaults to the most recent date on - which bookmarks were saved. - &url={URL} - (optional) Fetch a bookmark for this URL, regardless of date. - &hashes={MD5} {MD5} ... {MD5} - (optional) Fetch multiple bookmarks by one or more URL MD5s - regardless of date. - &meta=yes - (optional) Include change detection signatures on each item in a - 'meta' attribute. Clients wishing to maintain a synchronized local - store of bookmarks should retain the value of this attribute - its - value will change when any significant field of the bookmark - changes. - """ - return self.request("posts/get", tag=tag, dt=dt, url=url, - hashes=hashes, meta=meta, **kwds) - - def posts_recent(self, tag="", count="", **kwds): - """Returns a list of the most recent posts, filtered by argument. - :: - - <posts tag="..." user="..."> - <post ...> - - &tag={TAG} - (optional) Filter by this tag. - &count={1..100} - (optional) Number of items to retrieve (Default:15, Maximum:100). - """ - return self.request("posts/recent", tag=tag, count=count, **kwds) - - def posts_all(self, tag="", start=None, results=None, fromdt=None, - todt=None, meta=True, hashes=False, **kwds): - """Returns all posts. Please use sparingly. Call the `posts_update` - method to see if you need to fetch this at all. - :: - - <posts tag="..." user="..." update="CCYY-MM-DDThh:mm:ssZ"> - <post ...> - - &tag - (optional) Filter by this tag. - &start={#} - (optional) Start returning posts this many results into the set. - &results={#} - (optional) Return this many results. - &fromdt={CCYY-MM-DDThh:mm:ssZ} - (optional) Filter for posts on this date or later - &todt={CCYY-MM-DDThh:mm:ssZ} - (optional) Filter for posts on this date or earlier - &meta=yes - (optional) Include change detection signatures on each item in a - 'meta' attribute. Clients wishing to maintain a synchronized local - store of bookmarks should retain the value of this attribute - its - value will change when any significant field of the bookmark - changes. - &hashes - (optional, exclusive) Do not fetch post details but a posts - manifest with url- and meta-hashes. Other options do not apply. - """ - if hashes: - return self.request("posts/all", hashes=hashes, **kwds) - else: - return self.request("posts/all", tag=tag, fromdt=fromdt, todt=todt, - start=start, results=results, meta=meta, **kwds) - - def posts_add(self, url, description, extended="", tags="", dt="", - replace=False, shared=True, **kwds): - """Add a post to del.icio.us. Returns a `result` message or raises an - ``DeliciousError``. See ``self.request()``. - - &url (required) - the url of the item. - &description (required) - the description of the item. - &extended (optional) - notes for the item. - &tags (optional) - tags for the item (space delimited). - &dt (optional) - datestamp of the item (format "CCYY-MM-DDThh:mm:ssZ"). - Requires a LITERAL "T" and "Z" like in ISO8601 at - http://www.cl.cam.ac.uk/~mgk25/iso-time.html for example: - "1984-09-01T14:21:31Z" - &replace=no (optional) - don't replace post if given url has already - been posted. - &shared=yes (optional) - wether the item is public. - """ - return self.request("posts/add", url=url, description=description, - extended=extended, tags=tags, dt=dt, - replace=replace, shared=shared, **kwds) - - def posts_delete(self, url, **kwds): - """Delete a post from del.icio.us. Returns a `result` message or - raises an ``DeliciousError``. See ``self.request()``. - - &url (required) - the url of the item. - """ - return self.request("posts/delete", url=url, **kwds) - - # Bundles - def bundles_all(self, **kwds): - """Retrieve user bundles from del.icio.us. - :: - - <bundles> - <bundel name="..." tags=..."> - """ - return self.request("tags/bundles/all", **kwds) - - def bundles_set(self, bundle, tags, **kwds): - """Assign a set of tags to a single bundle, wipes away previous - settings for bundle. Returns a `result` messages or raises an - ``DeliciousError``. See ``self.request()``. - - &bundle (required) - the bundle name. - &tags (required) - list of tags. - """ - if type(tags)==list: - tags = " ".join(tags) - return self.request("tags/bundles/set", bundle=bundle, tags=tags, - **kwds) - - def bundles_delete(self, bundle, **kwds): - """Delete a bundle from del.icio.us. Returns a `result` message or - raises an ``DeliciousError``. See ``self.request()``. - - &bundle (required) - the bundle name. - """ - return self.request("tags/bundles/delete", bundle=bundle, **kwds) - - ### Utils - - # Lookup table for del.icio.us url-path to DeliciousAPI method. - paths = { - 'tags/get': 'tags_get', - 'tags/delete': 'tags_delete', - 'tags/rename': 'tags_rename', - 'posts/update': 'posts_update', - 'posts/dates': 'posts_dates', - 'posts/get': 'posts_get', - 'posts/recent': 'posts_recent', - 'posts/all': 'posts_all', - 'posts/add': 'posts_add', - 'posts/delete': 'posts_delete', - 'tags/bundles/all': 'bundles_all', - 'tags/bundles/set': 'bundles_set', - 'tags/bundles/delete': 'bundles_delete', - } - def get_method(self, path): - return getattr(self, self.paths[path]) - - def get_url(self, url): - """Return the del.icio.us url at which the HTML page with posts for - ``url`` can be found. - """ - return "http://del.icio.us/url/?url=%s" % (url,) - - def __repr__(self): - return "DeliciousAPI(%s)" % self.user - - -### Convenience functions on this package - -def apiNew(user, passwd): - "Creates a new DeliciousAPI object, requires user(name) and passwd." - return DeliciousAPI(user=user, passwd=passwd) - -def add(user, passwd, url, description, tags="", extended="", dt=None, - replace=False): - apiNew(user, passwd).posts_add(url=url, description=description, - extended=extended, tags=tags, dt=dt, replace=replace) - -def get(user, passwd, tag="", dt=None, count=0, hashes=[]): - "Returns a list of posts for the user" - posts = apiNew(user, passwd).posts_get( - tag=tag, dt=dt, hashes=hashes)['posts'] - if count: posts = posts[:count] - return posts - -def get_update(user, passwd): - "Returns the last update time for the user." - return apiNew(user, passwd).posts_update()['update']['time'] - -def get_all(user, passwd, tag="", start=0, results=100, fromdt=None, - todt=None): - "Returns a list with all posts. Please use sparingly. See `get_updated`" - return apiNew(user, passwd).posts_all(tag=tag, start=start, - results=results, fromdt=fromdt, todt=todt, meta=True)['posts'] - -def get_tags(user, passwd): - "Returns a list with all tags for user." - return apiNew(user=user, passwd=passwd).tags_get()['tags'] - -def delete(user, passwd, url): - "Delete the URL from the del.icio.us account." - apiNew(user, passwd).posts_delete(url=url) - -def rename_tag(user, passwd, oldtag, newtag): - "Rename the tag for the del.icio.us account." - apiNew(user=user, passwd=passwd).tags_rename(old=oldtag, new=newtag) - - -### RSS functions - -def getrss(tag="", popular=0, url='', user=""): - """Get posts from del.icio.us via parsing RSS. - - tag (opt) sort by tag - popular (opt) look for the popular stuff - user (opt) get the posts by a user, this striks popular - url (opt) get the posts by url - """ - return dlcs_rss_request(tag=tag, popular=popular, user=user, url=url) - -def get_userposts(user): - "parse RSS for user" - return getrss(user=user) - -def get_tagposts(tag): - "parse RSS for tag" - return getrss(tag=tag) - -def get_urlposts(url): - "parse RSS for URL" - return getrss(url=url) - -def get_popular(tag=""): - "parse RSS for popular URLS for tag" - return getrss(tag=tag, popular=1) - - -### JSON feeds -# TODO: untested - -def json_posts(user, count=15, tag=None, raw=True): - """ - user - count=### the number of posts you want to get (default is 15, maximum - is 100) - raw a raw JSON object is returned, instead of an object named - Delicious.posts - """ - url = "http://del.icio.us/feeds/json/" + \ - dlcs_encode_params({0:user})[0] - if tag: url += '/'+dlcs_encode_params({0:tag})[0] - - return dlcs_feed(url, count=count, raw=raw) - - -def json_tags(user, atleast, count, sort='alpha', raw=True, callback=None): - """ - user - atleast=### include only tags for which there are at least ### - number of posts. - count=### include ### tags, counting down from the top. - sort={alpha|count} construct the object with tags in alphabetic order - (alpha), or by count of posts (count). - callback=NAME wrap the object definition in a function call NAME(...), - thus invoking that function when the feed is executed. - raw a pure JSON object is returned, instead of code that - will construct an object named Delicious.tags. - """ - url = 'http://del.icio.us/feeds/json/tags/' + \ - dlcs_encode_params({0:user})[0] - return dlcs_feed(url, atleast=atleast, count=count, sort=sort, raw=raw, - callback=callback) - - -def json_network(user, raw=True, callback=None): - """ - callback=NAME wrap the object definition in a function call NAME(...) - ?raw a raw JSON object is returned, instead of an object named - Delicious.posts - """ - url = 'http://del.icio.us/feeds/json/network/' + \ - dlcs_encode_params({0:user})[0] - return dlcs_feed(url, raw=raw, callback=callback) - - -def json_fans(user, raw=True, callback=None): - """ - callback=NAME wrap the object definition in a function call NAME(...) - ?raw a pure JSON object is returned, instead of an object named - Delicious. - """ - url = 'http://del.icio.us/feeds/json/fans/' + \ - dlcs_encode_params({0:user})[0] - return dlcs_feed(url, raw=raw, callback=callback) - - -### delicious V2 feeds - -def getfeed(name, **params): - return dlcs_feed(name, **params) - diff --git a/lib/strutils.py b/lib/strutils.py deleted file mode 100644 index 368d3d8..0000000 --- a/lib/strutils.py +++ /dev/null @@ -1,50 +0,0 @@ - -# -# String/unicode conversion utils. -# - -def safestr(s): - """ - Safely corerce *anything* to a string. If the object can't be str'd, an - empty string will be returned. - - You can (and I do) use this for really crappy unicode handling, but it's - a bit like killing a mosquito with a bazooka. - """ - if s is None: - return "" - if isinstance(s, unicode): - return s.encode('ascii', 'xmlcharrefreplace') - else: - try: - return str(s) - except: - return "" - -def safeint(s): - """Like safestr(), but always returns an int. Returns 0 on failure.""" - try: - return int(safestr(s)) - except ValueError: - return 0 - - -def convertentity(m): - import htmlentitydefs - """Convert a HTML entity into normal string (ISO-8859-1)""" - if m.group(1)=='#': - try: - return chr(int(m.group(2))) - except ValueError: - return '&#%s;' % m.group(2) - try: - return htmlentitydefs.entitydefs[m.group(2)] - except KeyError: - return '&%s;' % m.group(2) - -def unquotehtml(s): - import re - """Convert a HTML quoted string into normal string (ISO-8859-1). - - Works with &#XX; and with   > etc.""" - return re.sub(r'&(#?)(.+?);',convertentity,s) diff --git a/lib/utils/APIClients.py b/lib/utils/APIClients.py new file mode 100644 index 0000000..24ab97b --- /dev/null +++ b/lib/utils/APIClients.py @@ -0,0 +1,104 @@ +# APIClients for grabbing data from popular web services +# By Scott Gilbertson +# Copyright is lame, take what you want, except for those portions noted + +# Dependencies: +import sys, urllib +import xml.etree.cElementTree as xml_parser + + +DEBUG = 0 + +""" +base class -- handles GoodReads.com, but works for any rss feed, just send an empty string for anything you don't need +""" +class APIClient: + def __init__(self, base_path, api_key): + self.api_key = api_key + self.base_path = base_path + + def __getattr__(self, method): + def method(_self=self, _method=method, **params): + url = "%s%s?%s&" % (self.base_path, self.api_key, urllib.urlencode(params)) + if DEBUG: print url + data = self.fetch(url) + return data + + return method + + def fetch(self, url): + u = urllib.FancyURLopener(None) + usock = u.open(url) + rawdata = usock.read() + if DEBUG: print rawdata + usock.close() + return xml_parser.fromstring(rawdata) + +""" + Extend APIClient to work with the ma.gnolia.com API + (http://wiki.ma.gnolia.com/Ma.gnolia_API) + Adds some error handling as well +""" +class MagnoliaError(Exception): + def __init__(self, code, message): + self.code = code + self.message = message + + def __str__(self): + return 'Magnolia Error %s: %s' % (self.code, self.message) + + +class MagnoliaClient(APIClient): + def __getattr__(self, method): + def method(_self=self, _method=method, **params): + url = "%s%s?%s&api_key=%s" % (self.base_path, _method, urllib.urlencode(params), self.api_key) + if DEBUG: print url + data = APIClient.fetch(self, url) + return data + return method + + +""" + Extend APIClient to work with the Flickr API + (http://www.flickr.com/services/api/) + Adds error handling as well +""" + +class FlickrError(Exception): + def __init__(self, code, message): + self.code = code + self.message = message + + def __str__(self): + return 'Flickr Error %s: %s' % (self.code, self.message) + +class FlickrClient(APIClient): + def __getattr__(self, method): + def method(_self=self, _method=method, **params): + _method = _method.replace("_", ".") + url = "%s?method=%s&%s&api_key=%s" % (self.base_path, _method, urllib.urlencode(params), self.api_key) + if DEBUG: print url + data = APIClient.fetch(self, url) + return data + return method + +class TumblrClient: + def __init__(self, base_path): + self.base_path = base_path + + def __getattr__(self, method): + def method(_self=self, _method=method, **params): + url = "%s" % (self.base_path) + if DEBUG: print url + data = self.fetch(url) + return data + + return method + + def fetch(self, url): + u = urllib.FancyURLopener(None) + usock = u.open(url) + rawdata = usock.read() + if DEBUG: print rawdata + usock.close() + return xml_parser.fromstring(rawdata) diff --git a/lib/utils/GeoClient.py b/lib/utils/GeoClient.py deleted file mode 100644 index d1966ca..0000000 --- a/lib/utils/GeoClient.py +++ /dev/null @@ -1,292 +0,0 @@ -# -*- coding: utf-8 -*- - -"""Python wrapper for geoname web APIs - -created 20/03/2006 By Nicolas Laurance - -This module allows you to access geoname's web APIs, -and get the results programmatically. -Described here: - http://www.geonames.org/export/ - -def postalCodeSearch(postalcode, placename='', country=COUNTRY, maxRows='10', http_proxy=None): -def postalCodeLookupJSON(postalcode, placename='', country=COUNTRY, maxRows='10',gcallback='', http_proxy=None): -def findNearbyPostalCodes(postalcode, placename='', country=COUNTRY, radius='5', maxRows='10',lat=None,lng=None, http_proxy=None): -def postalCodeCountryInfo(http_proxy=None): -def search(placename='', country=COUNTRY, maxRows='10', style='SHORT',lang=LANG, fclass=None, http_proxy=None): -def findNearbyPlaceName(lat,lng, http_proxy=None): - -Sample usage: ->>> import geoname ->>> result=geoname.postalCodeSearch('35580','guichen','fr','10') ->>> result.totalResultsCount.PCDATA -u'1' ->>> result.code[0].lat.PCDATA -u'47.9666667' ->>> result.code[0].lng.PCDATA -u'-1.8' - - - -""" - -__author__ = "Nicolas Laurance (nlaurance@zindep.com)" -__version__ = "2.0" -__cvsversion__ = "$Revision: 2.0 $"[11:-2] -__date__ = "$Date: 2003/06/20 22:40:53 $"[7:-2] -__copyright__ = "Copyright (c) 2006 Nicolas Laurance" -__license__ = "Python" - -import gnosis.xml.objectify as objectify - -import os, sys, urllib, re -try: - import timeoutsocket # http://www.timo-tasi.org/python/timeoutsocket.py - timeoutsocket.setDefaultSocketTimeout(120) -except ImportError: - pass - -HTTP_PROXY = None -DEBUG = 0 -COUNTRY = 'FR' -LANG ='fr' - - -# don't touch the rest of these constants -class GeonameError(Exception): pass - -## administrative functions -def version(): - print """PyGeoname %(__version__)s -%(__copyright__)s -released %(__date__)s -""" % globals() - -def setProxy(http_proxy): - """set HTTP proxy""" - global HTTP_PROXY - HTTP_PROXY = http_proxy - -def getProxy(http_proxy = None): - """get HTTP proxy""" - return http_proxy or HTTP_PROXY - -def getProxies(http_proxy = None): - http_proxy = getProxy(http_proxy) - if http_proxy: - proxies = {"http": http_proxy} - else: - proxies = None - return proxies - -def _contentsOf(dirname, filename): - filename = os.path.join(dirname, filename) - if not os.path.exists(filename): return None - fsock = open(filename) - contents = fsock.read() - fsock.close() - return contents - -def _getScriptDir(): - if __name__ == '__main__': - return os.path.abspath(os.path.dirname(sys.argv[0])) - else: - return os.path.abspath(os.path.dirname(sys.modules[__name__].__file__)) - -class Bag: pass - -_intFields = ('totalResultsCount') -_dateFields = () -_listFields = ('code','geoname','country',) -_floatFields = ('lat','lng','distance') - -def unmarshal(element): - #import pdb;pdb.set_trace() - xml_obj = objectify.XML_Objectify(element) - rc = xml_obj.make_instance() - return rc - -def _do(url, http_proxy): - proxies = getProxies(http_proxy) - u = urllib.FancyURLopener(proxies) - usock = u.open(url) - rawdata = usock.read() - if DEBUG: print rawdata - usock.close() - data = unmarshal(rawdata) - return data - -## main functions - -def _buildfindNearbyPostalCodes(postalcode, placename, country, radius, maxRows ): - placename=urllib.quote(placename) - searchUrl = "http://ws.geonames.org/findNearbyPostalCodes?postalcode=%(postalcode)s&placename=%(placename)s&country=%(country)s&radius=%(radius)s&maxRows=%(maxRows)s" % vars() - return searchUrl - - -def _buildpostalCodeLookupJSON(postalcode,placename,country,maxRows,gcallback): - placename=urllib.quote(placename) - searchUrl = "http://ws.geonames.org/postalCodeLookupJSON?postalcode=%(postalcode)s&placename=%(placename)s&country=%(country)s&maxRows=%(maxRows)s&callback=%(gcallback)s" % vars() - return searchUrl - -def _buildfindNearbyPostalCodesLL(lat,lng,radius,maxRows): - searchUrl = "http://ws.geonames.org/findNearbyPostalCodes?lat=%(lat)s&lng=%(lng)s&radius=%(radius)s&maxRows=%(maxRows)s" % vars() - return searchUrl - -def _buildfindCountrySubdivision(lat,lng): - searchUrl = "http://ws.geonames.org/countrySubdivision?lat=%(lat)s&lng=%(lng)s" % vars() - return searchUrl - -def _buildfindNearbyPlaceName(lat,lng): - searchUrl = "http://ws.geonames.org/findNearbyPlaceName?lat=%(lat)s&lng=%(lng)s" % vars() - return searchUrl - -def _buildpostalCodeSearch(postalcode, placename, country, maxRows ): - placename=urllib.quote(placename) - searchUrl = "http://ws.geonames.org/postalCodeSearch?postalcode=%(postalcode)s&placename=%(placename)s&country=%(country)s&maxRows=%(maxRows)s" % vars() - return searchUrl - -def _buildsearch(placename, country, maxRows,style,lang, fclass): - placename=urllib.quote(placename) - if fclass: - urlfclass='' - for fc in fclass: - urlfclass+=urllib.quote("&fclass=%s" % fc) - searchUrl = "http://ws.geonames.org/search?q=%(placename)s&country=%(country)s&maxRows=%(maxRows)s&lang=%(lang)s&style=%(style)s&fclass=%(fclass)s" % vars() - return searchUrl - -def postalCodeSearch(postalcode, placename='', country=COUNTRY, maxRows='10', http_proxy=None): - """ - http://ws.geonames.org/postalCodeSearch?postalcode=35580&maxRows=10&country=fr - Url : ws.geonames.org/postalCodeSearch? - Parameters : postalcode ,placename,maxRows,country - <geonames> - <totalResultsCount>7</totalResultsCount> - - - <code> - <postalcode>35580</postalcode> - <name>St Senoux</name> - <countryCode>FR</countryCode> - <lat>47.9</lat> - <lng>-1.7833333</lng> - </code> - """ - url = _buildpostalCodeSearch(postalcode,placename,country,maxRows) - if DEBUG: print url - return _do(url,http_proxy) - -def postalCodeLookupJSON(postalcode, placename='', country=COUNTRY, maxRows='10',gcallback='', http_proxy=None): - """ - Webservice Type : REST /JSON - Url : ws.geonames.org/postalCodeLookupJSON? - Parameters : postalcode,country ,maxRows (default = 20),callback - Result : returns a list of places for the given postalcode in JSON format - """ - url = _buildpostalCodeLookupJSON(postalcode,placename,country,maxRows,gcallback) -# print url - proxies = getProxies(http_proxy) - u = urllib.FancyURLopener(proxies) - usock = u.open(url) - rawdata = usock.read() - if DEBUG: print rawdata - usock.close() - return eval(rawdata[:-3]) - -def findNearbyPostalCodes(postalcode, placename='', country=COUNTRY, radius='5', maxRows='10',lat=None,lng=None, http_proxy=None): - """ - Find nearby postal codes / reverse geocoding - This service comes in two flavors. You can either pass the lat/long or a postalcode/placename. - - Webservice Type : REST - Url : ws.geonames.org/findNearbyPostalCodes? - Parameters : - lat,lng, radius (in km), maxRows (default = 5),country (default = all countries) - or - postalcode,country, radius (in Km), maxRows (default = 5) - Result : returns a list of postalcodes and places for the lat/lng query as xml document - Example: - http://ws.geonames.org/findNearbyPostalCodes?postalcode=35580&placename=guichen&country=FR&radius=5 - <geonames> - - - <code> - <postalcode>35580</postalcode> - <name>Guichen</name> - <countryCode>FR</countryCode> - <lat>47.9666667</lat> - <lng>-1.8</lng> - <distance>0.0</distance> - </code> - """ - if lat and lng : - url = _buildfindNearbyPostalCodesLL(lat,lng,radius,maxRows) - else: - url = _buildfindNearbyPostalCodes(postalcode,placename,country,radius,maxRows) - if DEBUG: print url -# import pdb;pdb.set_trace() - return _do(url,http_proxy).code - - -def postalCodeCountryInfo(http_proxy=None): - """ - http://ws.geonames.org/postalCodeCountryInfo? - <country> - <countryCode>FR</countryCode> - <countryName>France</countryName> - <numPostalCodes>39163</numPostalCodes> - <minPostalCode>01000</minPostalCode> - <maxPostalCode>98000</maxPostalCode> - </country> - - """ - return _do("http://ws.geonames.org/postalCodeCountryInfo?",http_proxy).country - -def search(placename='', country=COUNTRY, maxRows='10', style='SHORT',lang=LANG, fclass=None, http_proxy=None): - """ - Url : ws.geonames.org/search? - Parameters : q : place name (urlencoded utf8) - maxRows : maximal number of rows returned (default = 100) - country : iso country code, two characters (default = all countries) - fclass : featureclass(es) (default= all feature classes); this parameter may occur more then once, example: fclass=P&fclass=A - style : SHORT,MEDIUM,LONG (default = MEDIUM), verbosity of returned xml document - lang : ISO 2-letter language code. (default = en), countryName will be returned in the specified language. - - http://ws.geonames.org/search?q=guichen&maxRows=10&style=SHORT&lang=fr&country=fr - <geonames> - <totalResultsCount>3</totalResultsCount> - - - <geoname> - <name>Laill�</name> - <lat>47.9833333</lat> - <lng>-1.7166667</lng> - </geoname> - """ - url = _buildsearch(placename, country, maxRows,style,lang, fclass) - if DEBUG: print url - return _do(url,http_proxy) - -def findNearbyPlaceName(lat,lng, http_proxy=None): - """ - Webservice Type : REST - Url : ws.geonames.org/findNearbyPlaceName? - Parameters : lat,lng - Result : returns the closest populated place for the lat/lng query as xml document - Example: - http://ws.geonames.org/findNearbyPlaceName?lat=47.3&lng=9 - """ - url = _buildfindNearbyPlaceName(lat,lng) - if DEBUG: print url - return _do(url,http_proxy) - -def findCountrySubdivision(lat,lng, http_proxy=None): - """ - Webservice Type : REST - Url : ws.geonames.org/findNearbyPlaceName? - Parameters : lat,lng - Result : returns the closest populated place for the lat/lng query as xml document - Example: - http://ws.geonames.org/findNearbyPlaceName?lat=47.3&lng=9 - """ - url = _buildfindCountrySubdivision(lat,lng) - if DEBUG: print url - return _do(url,http_proxy) - diff --git a/lib/utils/email_multipart.py b/lib/utils/email_multipart.py deleted file mode 100644 index 4c2e154..0000000 --- a/lib/utils/email_multipart.py +++ /dev/null @@ -1,80 +0,0 @@ -# -*- coding: utf-8 -*- - -# Sending html emails in Django -# Report any bugs to esat @t sleytr*net -# Evren Esat Ozkan - - -from feedparser import _sanitizeHTML -from stripogram import html2text - -from django.conf import settings -from django.template import loader, Context - -from email.MIMEMultipart import MIMEMultipart -from email.MIMEText import MIMEText -from email.MIMEImage import MIMEImage -from smtplib import SMTP -import email.Charset - - -charset='utf-8' - - -email.Charset.add_charset( charset, email.Charset.SHORTEST, None, None ) - -def htmlmail(sbj,recip,msg,template='',texttemplate='',textmsg='',images=(), recip_name='',sender=settings.DEFAULT_FROM_EMAIL,sender_name='',charset=charset): - ''' - if you want to use Django template system: - use `msg` and optionally `textmsg` as template context (dict) - and define `template` and optionally `texttemplate` variables. - otherwise msg and textmsg variables are used as html and text message sources. - - if you want to use images in html message, define physical paths and ids in tuples. - (image paths are relative to MEDIA_ROOT) - example: - images=(('email_images/logo.gif','img1'),('email_images/footer.gif','img2')) - and use them in html like this: - <img src="cid:img1"> - ... - <img src="cid:img2"> - ''' - html=render(msg,template) - if texttemplate or textmsg: text=render((textmsg or msg),texttemplate) - else: text= html2text(_sanitizeHTML(html,charset)) - - msgRoot = MIMEMultipart('related') - msgRoot['Subject'] = sbj - msgRoot['From'] = named(sender,sender_name) - msgRoot['To'] = named(recip,recip_name) - msgRoot.preamble = 'This is a multi-part message in MIME format.' - - msgAlternative = MIMEMultipart('alternative') - msgRoot.attach(msgAlternative) - - msgAlternative.attach(MIMEText(text, _charset=charset)) - msgAlternative.attach(MIMEText(html, 'html', _charset=charset)) - - for img in images: - fp = open(img[0], 'rb') - msgImage = MIMEImage(fp.read()) - fp.close() - msgImage.add_header('Content-ID', '<'+img[1]+'>') - msgRoot.attach(msgImage) - - smtp = SMTP() - smtp.connect(settings.EMAIL_HOST) - smtp.login(settings.EMAIL_HOST_USER , settings.EMAIL_HOST_PASSWORD) - smtp.sendmail(sender, recip, msgRoot.as_string()) - smtp.quit() - - -def render(context,template): - if template: - t = loader.get_template(template) - return t.render(Context(context)) - return context - -def named(mail,name): - if name: return '%s <%s>' % (name,mail) - return mail \ No newline at end of file diff --git a/lib/utils/markdown2.py b/lib/utils/markdown2.py new file mode 100755 index 0000000..d72f414 --- /dev/null +++ b/lib/utils/markdown2.py @@ -0,0 +1,1877 @@ +#!/usr/bin/env python +# Copyright (c) 2007-2008 ActiveState Corp. +# License: MIT (http://www.opensource.org/licenses/mit-license.php) + +r"""A fast and complete Python implementation of Markdown. + +[from http://daringfireball.net/projects/markdown/] +> Markdown is a text-to-HTML filter; it translates an easy-to-read / +> easy-to-write structured text format into HTML. Markdown's text +> format is most similar to that of plain text email, and supports +> features such as headers, *emphasis*, code blocks, blockquotes, and +> links. +> +> Markdown's syntax is designed not as a generic markup language, but +> specifically to serve as a front-end to (X)HTML. You can use span-level +> HTML tags anywhere in a Markdown document, and you can use block level +> HTML tags (like <div> and <table> as well). + +Module usage: + + >>> import markdown2 + >>> markdown2.markdown("*boo!*") # or use `html = markdown_path(PATH)` + u'<p><em>boo!</em></p>\n' + + >>> markdowner = Markdown() + >>> markdowner.convert("*boo!*") + u'<p><em>boo!</em></p>\n' + >>> markdowner.convert("**boom!**") + u'<p><strong>boom!</strong></p>\n' + +This implementation of Markdown implements the full "core" syntax plus a +number of extras (e.g., code syntax coloring, footnotes) as described on +<http://code.google.com/p/python-markdown2/wiki/Extras>. +""" + +cmdln_desc = """A fast and complete Python implementation of Markdown, a +text-to-HTML conversion tool for web writers. +""" + +# Dev Notes: +# - There is already a Python markdown processor +# (http://www.freewisdom.org/projects/python-markdown/). +# - Python's regex syntax doesn't have '\z', so I'm using '\Z'. I'm +# not yet sure if there implications with this. Compare 'pydoc sre' +# and 'perldoc perlre'. + +__version_info__ = (1, 0, 1, 13) # first three nums match Markdown.pl +__version__ = '1.0.1.13' +__author__ = "Trent Mick" + +import os +import sys +from pprint import pprint +import re +import logging +try: + from hashlib import md5 +except ImportError: + from md5 import md5 +import optparse +from random import random +import codecs + + + +#---- Python version compat + +if sys.version_info[:2] < (2,4): + from sets import Set as set + def reversed(sequence): + for i in sequence[::-1]: + yield i + def _unicode_decode(s, encoding, errors='xmlcharrefreplace'): + return unicode(s, encoding, errors) +else: + def _unicode_decode(s, encoding, errors='strict'): + return s.decode(encoding, errors) + + +#---- globals + +DEBUG = False +log = logging.getLogger("markdown") + +DEFAULT_TAB_WIDTH = 4 + +# Table of hash values for escaped characters: +def _escape_hash(s): + # Lame attempt to avoid possible collision with someone actually + # using the MD5 hexdigest of one of these chars in there text. + # Other ideas: random.random(), uuid.uuid() + #return md5(s).hexdigest() # Markdown.pl effectively does this. + return 'md5-'+md5(s).hexdigest() +g_escape_table = dict([(ch, _escape_hash(ch)) + for ch in '\\`*_{}[]()>#+-.!']) + + + +#---- exceptions + +class MarkdownError(Exception): + pass + + + +#---- public api + +def markdown_path(path, encoding="utf-8", + html4tags=False, tab_width=DEFAULT_TAB_WIDTH, + safe_mode=None, extras=None, link_patterns=None, + use_file_vars=False): + text = codecs.open(path, 'r', encoding).read() + return Markdown(html4tags=html4tags, tab_width=tab_width, + safe_mode=safe_mode, extras=extras, + link_patterns=link_patterns, + use_file_vars=use_file_vars).convert(text) + +def markdown(text, html4tags=False, tab_width=DEFAULT_TAB_WIDTH, + safe_mode=None, extras=None, link_patterns=None, + use_file_vars=False): + return Markdown(html4tags=html4tags, tab_width=tab_width, + safe_mode=safe_mode, extras=extras, + link_patterns=link_patterns, + use_file_vars=use_file_vars).convert(text) + +class Markdown(object): + # The dict of "extras" to enable in processing -- a mapping of + # extra name to argument for the extra. Most extras do not have an + # argument, in which case the value is None. + # + # This can be set via (a) subclassing and (b) the constructor + # "extras" argument. + extras = None + + urls = None + titles = None + html_blocks = None + html_spans = None + html_removed_text = "[HTML_REMOVED]" # for compat with markdown.py + + # Used to track when we're inside an ordered or unordered list + # (see _ProcessListItems() for details): + list_level = 0 + + _ws_only_line_re = re.compile(r"^[ \t]+$", re.M) + + def __init__(self, html4tags=False, tab_width=4, safe_mode=None, + extras=None, link_patterns=None, use_file_vars=False): + if html4tags: + self.empty_element_suffix = ">" + else: + self.empty_element_suffix = " />" + self.tab_width = tab_width + + # For compatibility with earlier markdown2.py and with + # markdown.py's safe_mode being a boolean, + # safe_mode == True -> "replace" + if safe_mode is True: + self.safe_mode = "replace" + else: + self.safe_mode = safe_mode + + if self.extras is None: + self.extras = {} + elif not isinstance(self.extras, dict): + self.extras = dict([(e, None) for e in self.extras]) + if extras: + if not isinstance(extras, dict): + extras = dict([(e, None) for e in extras]) + self.extras.update(extras) + assert isinstance(self.extras, dict) + self._instance_extras = self.extras.copy() + self.link_patterns = link_patterns + self.use_file_vars = use_file_vars + self._outdent_re = re.compile(r'^(\t|[ ]{1,%d})' % tab_width, re.M) + + def reset(self): + self.urls = {} + self.titles = {} + self.html_blocks = {} + self.html_spans = {} + self.list_level = 0 + self.extras = self._instance_extras.copy() + if "footnotes" in self.extras: + self.footnotes = {} + self.footnote_ids = [] + + def convert(self, text): + """Convert the given text.""" + # Main function. The order in which other subs are called here is + # essential. Link and image substitutions need to happen before + # _EscapeSpecialChars(), so that any *'s or _'s in the <a> + # and <img> tags get encoded. + + # Clear the global hashes. If we don't clear these, you get conflicts + # from other articles when generating a page which contains more than + # one article (e.g. an index page that shows the N most recent + # articles): + self.reset() + + if not isinstance(text, unicode): + #TODO: perhaps shouldn't presume UTF-8 for string input? + text = unicode(text, 'utf-8') + + if self.use_file_vars: + # Look for emacs-style file variable hints. + emacs_vars = self._get_emacs_vars(text) + if "markdown-extras" in emacs_vars: + splitter = re.compile("[ ,]+") + for e in splitter.split(emacs_vars["markdown-extras"]): + if '=' in e: + ename, earg = e.split('=', 1) + try: + earg = int(earg) + except ValueError: + pass + else: + ename, earg = e, None + self.extras[ename] = earg + + # Standardize line endings: + text = re.sub("\r\n|\r", "\n", text) + + # Make sure $text ends with a couple of newlines: + text += "\n\n" + + # Convert all tabs to spaces. + text = self._detab(text) + + # Strip any lines consisting only of spaces and tabs. + # This makes subsequent regexen easier to write, because we can + # match consecutive blank lines with /\n+/ instead of something + # contorted like /[ \t]*\n+/ . + text = self._ws_only_line_re.sub("", text) + + if self.safe_mode: + text = self._hash_html_spans(text) + + # Turn block-level HTML blocks into hash entries + text = self._hash_html_blocks(text, raw=True) + + # Strip link definitions, store in hashes. + if "footnotes" in self.extras: + # Must do footnotes first because an unlucky footnote defn + # looks like a link defn: + # [^4]: this "looks like a link defn" + text = self._strip_footnote_definitions(text) + text = self._strip_link_definitions(text) + + text = self._run_block_gamut(text) + + text = self._unescape_special_chars(text) + + if "footnotes" in self.extras: + text = self._add_footnotes(text) + + if self.safe_mode: + text = self._unhash_html_spans(text) + + text += "\n" + return text + + _emacs_oneliner_vars_pat = re.compile(r"-\*-\s*([^\r\n]*?)\s*-\*-", re.UNICODE) + # This regular expression is intended to match blocks like this: + # PREFIX Local Variables: SUFFIX + # PREFIX mode: Tcl SUFFIX + # PREFIX End: SUFFIX + # Some notes: + # - "[ \t]" is used instead of "\s" to specifically exclude newlines + # - "(\r\n|\n|\r)" is used instead of "$" because the sre engine does + # not like anything other than Unix-style line terminators. + _emacs_local_vars_pat = re.compile(r"""^ + (?P<prefix>(?:[^\r\n|\n|\r])*?) + [\ \t]*Local\ Variables:[\ \t]* + (?P<suffix>.*?)(?:\r\n|\n|\r) + (?P<content>.*?\1End:) + """, re.IGNORECASE | re.MULTILINE | re.DOTALL | re.VERBOSE) + + def _get_emacs_vars(self, text): + """Return a dictionary of emacs-style local variables. + + Parsing is done loosely according to this spec (and according to + some in-practice deviations from this): + http://www.gnu.org/software/emacs/manual/html_node/emacs/Specifying-File-Variables.html#Specifying-File-Variables + """ + emacs_vars = {} + SIZE = pow(2, 13) # 8kB + + # Search near the start for a '-*-'-style one-liner of variables. + head = text[:SIZE] + if "-*-" in head: + match = self._emacs_oneliner_vars_pat.search(head) + if match: + emacs_vars_str = match.group(1) + assert '\n' not in emacs_vars_str + emacs_var_strs = [s.strip() for s in emacs_vars_str.split(';') + if s.strip()] + if len(emacs_var_strs) == 1 and ':' not in emacs_var_strs[0]: + # While not in the spec, this form is allowed by emacs: + # -*- Tcl -*- + # where the implied "variable" is "mode". This form + # is only allowed if there are no other variables. + emacs_vars["mode"] = emacs_var_strs[0].strip() + else: + for emacs_var_str in emacs_var_strs: + try: + variable, value = emacs_var_str.strip().split(':', 1) + except ValueError: + log.debug("emacs variables error: malformed -*- " + "line: %r", emacs_var_str) + continue + # Lowercase the variable name because Emacs allows "Mode" + # or "mode" or "MoDe", etc. + emacs_vars[variable.lower()] = value.strip() + + tail = text[-SIZE:] + if "Local Variables" in tail: + match = self._emacs_local_vars_pat.search(tail) + if match: + prefix = match.group("prefix") + suffix = match.group("suffix") + lines = match.group("content").splitlines(0) + #print "prefix=%r, suffix=%r, content=%r, lines: %s"\ + # % (prefix, suffix, match.group("content"), lines) + + # Validate the Local Variables block: proper prefix and suffix + # usage. + for i, line in enumerate(lines): + if not line.startswith(prefix): + log.debug("emacs variables error: line '%s' " + "does not use proper prefix '%s'" + % (line, prefix)) + return {} + # Don't validate suffix on last line. Emacs doesn't care, + # neither should we. + if i != len(lines)-1 and not line.endswith(suffix): + log.debug("emacs variables error: line '%s' " + "does not use proper suffix '%s'" + % (line, suffix)) + return {} + + # Parse out one emacs var per line. + continued_for = None + for line in lines[:-1]: # no var on the last line ("PREFIX End:") + if prefix: line = line[len(prefix):] # strip prefix + if suffix: line = line[:-len(suffix)] # strip suffix + line = line.strip() + if continued_for: + variable = continued_for + if line.endswith('\\'): + line = line[:-1].rstrip() + else: + continued_for = None + emacs_vars[variable] += ' ' + line + else: + try: + variable, value = line.split(':', 1) + except ValueError: + log.debug("local variables error: missing colon " + "in local variables entry: '%s'" % line) + continue + # Do NOT lowercase the variable name, because Emacs only + # allows "mode" (and not "Mode", "MoDe", etc.) in this block. + value = value.strip() + if value.endswith('\\'): + value = value[:-1].rstrip() + continued_for = variable + else: + continued_for = None + emacs_vars[variable] = value + + # Unquote values. + for var, val in emacs_vars.items(): + if len(val) > 1 and (val.startswith('"') and val.endswith('"') + or val.startswith('"') and val.endswith('"')): + emacs_vars[var] = val[1:-1] + + return emacs_vars + + # Cribbed from a post by Bart Lateur: + # <http://www.nntp.perl.org/group/perl.macperl.anyperl/154> + _detab_re = re.compile(r'(.*?)\t', re.M) + def _detab_sub(self, match): + g1 = match.group(1) + return g1 + (' ' * (self.tab_width - len(g1) % self.tab_width)) + def _detab(self, text): + r"""Remove (leading?) tabs from a file. + + >>> m = Markdown() + >>> m._detab("\tfoo") + ' foo' + >>> m._detab(" \tfoo") + ' foo' + >>> m._detab("\t foo") + ' foo' + >>> m._detab(" foo") + ' foo' + >>> m._detab(" foo\n\tbar\tblam") + ' foo\n bar blam' + """ + if '\t' not in text: + return text + return self._detab_re.subn(self._detab_sub, text)[0] + + _block_tags_a = 'p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math|ins|del' + _strict_tag_block_re = re.compile(r""" + ( # save in \1 + ^ # start of line (with re.M) + <(%s) # start tag = \2 + \b # word break + (.*\n)*? # any number of lines, minimally matching + </\2> # the matching end tag + [ \t]* # trailing spaces/tabs + (?=\n+|\Z) # followed by a newline or end of document + ) + """ % _block_tags_a, + re.X | re.M) + + _block_tags_b = 'p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math' + _liberal_tag_block_re = re.compile(r""" + ( # save in \1 + ^ # start of line (with re.M) + <(%s) # start tag = \2 + \b # word break + (.*\n)*? # any number of lines, minimally matching + .*</\2> # the matching end tag + [ \t]* # trailing spaces/tabs + (?=\n+|\Z) # followed by a newline or end of document + ) + """ % _block_tags_b, + re.X | re.M) + + def _hash_html_block_sub(self, match, raw=False): + html = match.group(1) + if raw and self.safe_mode: + html = self._sanitize_html(html) + key = _hash_text(html) + self.html_blocks[key] = html + return "\n\n" + key + "\n\n" + + def _hash_html_blocks(self, text, raw=False): + """Hashify HTML blocks + + We only want to do this for block-level HTML tags, such as headers, + lists, and tables. That's because we still want to wrap <p>s around + "paragraphs" that are wrapped in non-block-level tags, such as anchors, + phrase emphasis, and spans. The list of tags we're looking for is + hard-coded. + + @param raw {boolean} indicates if these are raw HTML blocks in + the original source. It makes a difference in "safe" mode. + """ + if '<' not in text: + return text + + # Pass `raw` value into our calls to self._hash_html_block_sub. + hash_html_block_sub = _curry(self._hash_html_block_sub, raw=raw) + + # First, look for nested blocks, e.g.: + # <div> + # <div> + # tags for inner block must be indented. + # </div> + # </div> + # + # The outermost tags must start at the left margin for this to match, and + # the inner nested divs must be indented. + # We need to do this before the next, more liberal match, because the next + # match will start at the first `<div>` and stop at the first `</div>`. + text = self._strict_tag_block_re.sub(hash_html_block_sub, text) + + # Now match more liberally, simply from `\n<tag>` to `</tag>\n` + text = self._liberal_tag_block_re.sub(hash_html_block_sub, text) + + # Special case just for <hr />. It was easier to make a special + # case than to make the other regex more complicated. + if "<hr" in text: + _hr_tag_re = _hr_tag_re_from_tab_width(self.tab_width) + text = _hr_tag_re.sub(hash_html_block_sub, text) + + # Special case for standalone HTML comments: + if "<!--" in text: + start = 0 + while True: + # Delimiters for next comment block. + try: + start_idx = text.index("<!--", start) + except ValueError, ex: + break + try: + end_idx = text.index("-->", start_idx) + 3 + except ValueError, ex: + break + + # Start position for next comment block search. + start = end_idx + + # Validate whitespace before comment. + if start_idx: + # - Up to `tab_width - 1` spaces before start_idx. + for i in range(self.tab_width - 1): + if text[start_idx - 1] != ' ': + break + start_idx -= 1 + if start_idx == 0: + break + # - Must be preceded by 2 newlines or hit the start of + # the document. + if start_idx == 0: + pass + elif start_idx == 1 and text[0] == '\n': + start_idx = 0 # to match minute detail of Markdown.pl regex + elif text[start_idx-2:start_idx] == '\n\n': + pass + else: + break + + # Validate whitespace after comment. + # - Any number of spaces and tabs. + while end_idx < len(text): + if text[end_idx] not in ' \t': + break + end_idx += 1 + # - Must be following by 2 newlines or hit end of text. + if text[end_idx:end_idx+2] not in ('', '\n', '\n\n'): + continue + + # Escape and hash (must match `_hash_html_block_sub`). + html = text[start_idx:end_idx] + if raw and self.safe_mode: + html = self._sanitize_html(html) + key = _hash_text(html) + self.html_blocks[key] = html + text = text[:start_idx] + "\n\n" + key + "\n\n" + text[end_idx:] + + if "xml" in self.extras: + # Treat XML processing instructions and namespaced one-liner + # tags as if they were block HTML tags. E.g., if standalone + # (i.e. are their own paragraph), the following do not get + # wrapped in a <p> tag: + # <?foo bar?> + # + # <xi:include xmlns:xi="http://www.w3.org/2001/XInclude" href="chapter_1.md"/> + _xml_oneliner_re = _xml_oneliner_re_from_tab_width(self.tab_width) + text = _xml_oneliner_re.sub(hash_html_block_sub, text) + + return text + + def _strip_link_definitions(self, text): + # Strips link definitions from text, stores the URLs and titles in + # hash references. + less_than_tab = self.tab_width - 1 + + # Link defs are in the form: + # [id]: url "optional title" + _link_def_re = re.compile(r""" + ^[ ]{0,%d}\[(.+)\]: # id = \1 + [ \t]* + \n? # maybe *one* newline + [ \t]* + <?(.+?)>? # url = \2 + [ \t]* + (?: + \n? # maybe one newline + [ \t]* + (?<=\s) # lookbehind for whitespace + ['"(] + ([^\n]*) # title = \3 + ['")] + [ \t]* + )? # title is optional + (?:\n+|\Z) + """ % less_than_tab, re.X | re.M | re.U) + return _link_def_re.sub(self._extract_link_def_sub, text) + + def _extract_link_def_sub(self, match): + id, url, title = match.groups() + key = id.lower() # Link IDs are case-insensitive + self.urls[key] = self._encode_amps_and_angles(url) + if title: + self.titles[key] = title.replace('"', '"') + return "" + + def _extract_footnote_def_sub(self, match): + id, text = match.groups() + text = _dedent(text, skip_first_line=not text.startswith('\n')).strip() + normed_id = re.sub(r'\W', '-', id) + # Ensure footnote text ends with a couple newlines (for some + # block gamut matches). + self.footnotes[normed_id] = text + "\n\n" + return "" + + def _strip_footnote_definitions(self, text): + """A footnote definition looks like this: + + [^note-id]: Text of the note. + + May include one or more indented paragraphs. + + Where, + - The 'note-id' can be pretty much anything, though typically it + is the number of the footnote. + - The first paragraph may start on the next line, like so: + + [^note-id]: + Text of the note. + """ + less_than_tab = self.tab_width - 1 + footnote_def_re = re.compile(r''' + ^[ ]{0,%d}\[\^(.+)\]: # id = \1 + [ \t]* + ( # footnote text = \2 + # First line need not start with the spaces. + (?:\s*.*\n+) + (?: + (?:[ ]{%d} | \t) # Subsequent lines must be indented. + .*\n+ + )* + ) + # Lookahead for non-space at line-start, or end of doc. + (?:(?=^[ ]{0,%d}\S)|\Z) + ''' % (less_than_tab, self.tab_width, self.tab_width), + re.X | re.M) + return footnote_def_re.sub(self._extract_footnote_def_sub, text) + + + _hr_res = [ + re.compile(r"^[ ]{0,2}([ ]?\*[ ]?){3,}[ \t]*$", re.M), + re.compile(r"^[ ]{0,2}([ ]?\-[ ]?){3,}[ \t]*$", re.M), + re.compile(r"^[ ]{0,2}([ ]?\_[ ]?){3,}[ \t]*$", re.M), + ] + + def _run_block_gamut(self, text): + # These are all the transformations that form block-level + # tags like paragraphs, headers, and list items. + + text = self._do_headers(text) + + # Do Horizontal Rules: + hr = "\n<hr"+self.empty_element_suffix+"\n" + for hr_re in self._hr_res: + text = hr_re.sub(hr, text) + + text = self._do_lists(text) + + if "pyshell" in self.extras: + text = self._prepare_pyshell_blocks(text) + + text = self._do_code_blocks(text) + + text = self._do_block_quotes(text) + + # We already ran _HashHTMLBlocks() before, in Markdown(), but that + # was to escape raw HTML in the original Markdown source. This time, + # we're escaping the markup we've just created, so that we don't wrap + # <p> tags around block-level tags. + text = self._hash_html_blocks(text) + + text = self._form_paragraphs(text) + + return text + + def _pyshell_block_sub(self, match): + lines = match.group(0).splitlines(0) + _dedentlines(lines) + indent = ' ' * self.tab_width + s = ('\n' # separate from possible cuddled paragraph + + indent + ('\n'+indent).join(lines) + + '\n\n') + return s + + def _prepare_pyshell_blocks(self, text): + """Ensure that Python interactive shell sessions are put in + code blocks -- even if not properly indented. + """ + if ">>>" not in text: + return text + + less_than_tab = self.tab_width - 1 + _pyshell_block_re = re.compile(r""" + ^([ ]{0,%d})>>>[ ].*\n # first line + ^(\1.*\S+.*\n)* # any number of subsequent lines + ^\n # ends with a blank line + """ % less_than_tab, re.M | re.X) + + return _pyshell_block_re.sub(self._pyshell_block_sub, text) + + def _run_span_gamut(self, text): + # These are all the transformations that occur *within* block-level + # tags like paragraphs, headers, and list items. + + text = self._do_code_spans(text) + + text = self._escape_special_chars(text) + + # Process anchor and image tags. + text = self._do_links(text) + + # Make links out of things like `<http://example.com/>` + # Must come after _do_links(), because you can use < and > + # delimiters in inline links like [this](<url>). + text = self._do_auto_links(text) + + if "link-patterns" in self.extras: + text = self._do_link_patterns(text) + + text = self._encode_amps_and_angles(text) + + text = self._do_italics_and_bold(text) + + # Do hard breaks: + text = re.sub(r" {2,}\n", " <br%s\n" % self.empty_element_suffix, text) + + return text + + # "Sorta" because auto-links are identified as "tag" tokens. + _sorta_html_tokenize_re = re.compile(r""" + ( + # tag + </? + (?:\w+) # tag name + (?:\s+(?:[\w-]+:)?[\w-]+=(?:".*?"|'.*?'))* # attributes + \s*/?> + | + # auto-link (e.g., <http://www.activestate.com/>) + <\w+[^>]*> + | + <!--.*?--> # comment + | + <\?.*?\?> # processing instruction + ) + """, re.X) + + def _escape_special_chars(self, text): + # Python markdown note: the HTML tokenization here differs from + # that in Markdown.pl, hence the behaviour for subtle cases can + # differ (I believe the tokenizer here does a better job because + # it isn't susceptible to unmatched '<' and '>' in HTML tags). + # Note, however, that '>' is not allowed in an auto-link URL + # here. + escaped = [] + is_html_markup = False + for token in self._sorta_html_tokenize_re.split(text): + if is_html_markup: + # Within tags/HTML-comments/auto-links, encode * and _ + # so they don't conflict with their use in Markdown for + # italics and strong. We're replacing each such + # character with its corresponding MD5 checksum value; + # this is likely overkill, but it should prevent us from + # colliding with the escape values by accident. + escaped.append(token.replace('*', g_escape_table['*']) + .replace('_', g_escape_table['_'])) + else: + escaped.append(self._encode_backslash_escapes(token)) + is_html_markup = not is_html_markup + return ''.join(escaped) + + def _hash_html_spans(self, text): + # Used for safe_mode. + + def _is_auto_link(s): + if ':' in s and self._auto_link_re.match(s): + return True + elif '@' in s and self._auto_email_link_re.match(s): + return True + return False + + tokens = [] + is_html_markup = False + for token in self._sorta_html_tokenize_re.split(text): + if is_html_markup and not _is_auto_link(token): + sanitized = self._sanitize_html(token) + key = _hash_text(sanitized) + self.html_spans[key] = sanitized + tokens.append(key) + else: + tokens.append(token) + is_html_markup = not is_html_markup + return ''.join(tokens) + + def _unhash_html_spans(self, text): + for key, sanitized in self.html_spans.items(): + text = text.replace(key, sanitized) + return text + + def _sanitize_html(self, s): + if self.safe_mode == "replace": + return self.html_removed_text + elif self.safe_mode == "escape": + replacements = [ + ('&', '&'), + ('<', '<'), + ('>', '>'), + ] + for before, after in replacements: + s = s.replace(before, after) + return s + else: + raise MarkdownError("invalid value for 'safe_mode': %r (must be " + "'escape' or 'replace')" % self.safe_mode) + + _tail_of_inline_link_re = re.compile(r''' + # Match tail of: [text](/url/) or [text](/url/ "title") + \( # literal paren + [ \t]* + (?P<url> # \1 + <.*?> + | + .*? + ) + [ \t]* + ( # \2 + (['"]) # quote char = \3 + (?P<title>.*?) + \3 # matching quote + )? # title is optional + \) + ''', re.X | re.S) + _tail_of_reference_link_re = re.compile(r''' + # Match tail of: [text][id] + [ ]? # one optional space + (?:\n[ ]*)? # one optional newline followed by spaces + \[ + (?P<id>.*?) + \] + ''', re.X | re.S) + + def _do_links(self, text): + """Turn Markdown link shortcuts into XHTML <a> and <img> tags. + + This is a combination of Markdown.pl's _DoAnchors() and + _DoImages(). They are done together because that simplified the + approach. It was necessary to use a different approach than + Markdown.pl because of the lack of atomic matching support in + Python's regex engine used in $g_nested_brackets. + """ + MAX_LINK_TEXT_SENTINEL = 3000 # markdown2 issue 24 + + # `anchor_allowed_pos` is used to support img links inside + # anchors, but not anchors inside anchors. An anchor's start + # pos must be `>= anchor_allowed_pos`. + anchor_allowed_pos = 0 + + curr_pos = 0 + while True: # Handle the next link. + # The next '[' is the start of: + # - an inline anchor: [text](url "title") + # - a reference anchor: [text][id] + # - an inline img: ![text](url "title") + # - a reference img: ![text][id] + # - a footnote ref: [^id] + # (Only if 'footnotes' extra enabled) + # - a footnote defn: [^id]: ... + # (Only if 'footnotes' extra enabled) These have already + # been stripped in _strip_footnote_definitions() so no + # need to watch for them. + # - a link definition: [id]: url "title" + # These have already been stripped in + # _strip_link_definitions() so no need to watch for them. + # - not markup: [...anything else... + try: + start_idx = text.index('[', curr_pos) + except ValueError: + break + text_length = len(text) + + # Find the matching closing ']'. + # Markdown.pl allows *matching* brackets in link text so we + # will here too. Markdown.pl *doesn't* currently allow + # matching brackets in img alt text -- we'll differ in that + # regard. + bracket_depth = 0 + for p in range(start_idx+1, min(start_idx+MAX_LINK_TEXT_SENTINEL, + text_length)): + ch = text[p] + if ch == ']': + bracket_depth -= 1 + if bracket_depth < 0: + break + elif ch == '[': + bracket_depth += 1 + else: + # Closing bracket not found within sentinel length. + # This isn't markup. + curr_pos = start_idx + 1 + continue + link_text = text[start_idx+1:p] + + # Possibly a footnote ref? + if "footnotes" in self.extras and link_text.startswith("^"): + normed_id = re.sub(r'\W', '-', link_text[1:]) + if normed_id in self.footnotes: + self.footnote_ids.append(normed_id) + result = '<sup class="footnote-ref" id="fnref-%s">' \ + '<a href="#fn-%s">%s</a></sup>' \ + % (normed_id, normed_id, len(self.footnote_ids)) + text = text[:start_idx] + result + text[p+1:] + else: + # This id isn't defined, leave the markup alone. + curr_pos = p+1 + continue + + # Now determine what this is by the remainder. + p += 1 + if p == text_length: + return text + + # Inline anchor or img? + if text[p] == '(': # attempt at perf improvement + match = self._tail_of_inline_link_re.match(text, p) + if match: + # Handle an inline anchor or img. + is_img = start_idx > 0 and text[start_idx-1] == "!" + if is_img: + start_idx -= 1 + + url, title = match.group("url"), match.group("title") + if url and url[0] == '<': + url = url[1:-1] # '<url>' -> 'url' + # We've got to encode these to avoid conflicting + # with italics/bold. + url = url.replace('*', g_escape_table['*']) \ + .replace('_', g_escape_table['_']) + if title: + title_str = ' title="%s"' \ + % title.replace('*', g_escape_table['*']) \ + .replace('_', g_escape_table['_']) \ + .replace('"', '"') + else: + title_str = '' + if is_img: + result = '<img src="%s" alt="%s"%s%s' \ + % (url, link_text.replace('"', '"'), + title_str, self.empty_element_suffix) + curr_pos = start_idx + len(result) + text = text[:start_idx] + result + text[match.end():] + elif start_idx >= anchor_allowed_pos: + result_head = '<a href="%s"%s>' % (url, title_str) + result = '%s%s</a>' % (result_head, link_text) + # <img> allowed from curr_pos on, <a> from + # anchor_allowed_pos on. + curr_pos = start_idx + len(result_head) + anchor_allowed_pos = start_idx + len(result) + text = text[:start_idx] + result + text[match.end():] + else: + # Anchor not allowed here. + curr_pos = start_idx + 1 + continue + + # Reference anchor or img? + else: + match = self._tail_of_reference_link_re.match(text, p) + if match: + # Handle a reference-style anchor or img. + is_img = start_idx > 0 and text[start_idx-1] == "!" + if is_img: + start_idx -= 1 + link_id = match.group("id").lower() + if not link_id: + link_id = link_text.lower() # for links like [this][] + if link_id in self.urls: + url = self.urls[link_id] + # We've got to encode these to avoid conflicting + # with italics/bold. + url = url.replace('*', g_escape_table['*']) \ + .replace('_', g_escape_table['_']) + title = self.titles.get(link_id) + if title: + title = title.replace('*', g_escape_table['*']) \ + .replace('_', g_escape_table['_']) + title_str = ' title="%s"' % title + else: + title_str = '' + if is_img: + result = '<img src="%s" alt="%s"%s%s' \ + % (url, link_text.replace('"', '"'), + title_str, self.empty_element_suffix) + curr_pos = start_idx + len(result) + text = text[:start_idx] + result + text[match.end():] + elif start_idx >= anchor_allowed_pos: + result = '<a href="%s"%s>%s</a>' \ + % (url, title_str, link_text) + result_head = '<a href="%s"%s>' % (url, title_str) + result = '%s%s</a>' % (result_head, link_text) + # <img> allowed from curr_pos on, <a> from + # anchor_allowed_pos on. + curr_pos = start_idx + len(result_head) + anchor_allowed_pos = start_idx + len(result) + text = text[:start_idx] + result + text[match.end():] + else: + # Anchor not allowed here. + curr_pos = start_idx + 1 + else: + # This id isn't defined, leave the markup alone. + curr_pos = match.end() + continue + + # Otherwise, it isn't markup. + curr_pos = start_idx + 1 + + return text + + + _setext_h_re = re.compile(r'^(.+)[ \t]*\n(=+|-+)[ \t]*\n+', re.M) + def _setext_h_sub(self, match): + n = {"=": 1, "-": 2}[match.group(2)[0]] + demote_headers = self.extras.get("demote-headers") + if demote_headers: + n = min(n + demote_headers, 6) + return "<h%d>%s</h%d>\n\n" \ + % (n, self._run_span_gamut(match.group(1)), n) + + _atx_h_re = re.compile(r''' + ^(\#{1,6}) # \1 = string of #'s + [ \t]* + (.+?) # \2 = Header text + [ \t]* + (?<!\\) # ensure not an escaped trailing '#' + \#* # optional closing #'s (not counted) + \n+ + ''', re.X | re.M) + def _atx_h_sub(self, match): + n = len(match.group(1)) + demote_headers = self.extras.get("demote-headers") + if demote_headers: + n = min(n + demote_headers, 6) + return "<h%d>%s</h%d>\n\n" \ + % (n, self._run_span_gamut(match.group(2)), n) + + def _do_headers(self, text): + # Setext-style headers: + # Header 1 + # ======== + # + # Header 2 + # -------- + text = self._setext_h_re.sub(self._setext_h_sub, text) + + # atx-style headers: + # # Header 1 + # ## Header 2 + # ## Header 2 with closing hashes ## + # ... + # ###### Header 6 + text = self._atx_h_re.sub(self._atx_h_sub, text) + + return text + + + _marker_ul_chars = '*+-' + _marker_any = r'(?:[%s]|\d+\.)' % _marker_ul_chars + _marker_ul = '(?:[%s])' % _marker_ul_chars + _marker_ol = r'(?:\d+\.)' + + def _list_sub(self, match): + lst = match.group(1) + lst_type = match.group(3) in self._marker_ul_chars and "ul" or "ol" + result = self._process_list_items(lst) + if self.list_level: + return "<%s>\n%s</%s>\n" % (lst_type, result, lst_type) + else: + return "<%s>\n%s</%s>\n\n" % (lst_type, result, lst_type) + + def _do_lists(self, text): + # Form HTML ordered (numbered) and unordered (bulleted) lists. + + for marker_pat in (self._marker_ul, self._marker_ol): + # Re-usable pattern to match any entire ul or ol list: + less_than_tab = self.tab_width - 1 + whole_list = r''' + ( # \1 = whole list + ( # \2 + [ ]{0,%d} + (%s) # \3 = first list item marker + [ \t]+ + ) + (?:.+?) + ( # \4 + \Z + | + \n{2,} + (?=\S) + (?! # Negative lookahead for another list item marker + [ \t]* + %s[ \t]+ + ) + ) + ) + ''' % (less_than_tab, marker_pat, marker_pat) + + # We use a different prefix before nested lists than top-level lists. + # See extended comment in _process_list_items(). + # + # Note: There's a bit of duplication here. My original implementation + # created a scalar regex pattern as the conditional result of the test on + # $g_list_level, and then only ran the $text =~ s{...}{...}egmx + # substitution once, using the scalar as the pattern. This worked, + # everywhere except when running under MT on my hosting account at Pair + # Networks. There, this caused all rebuilds to be killed by the reaper (or + # perhaps they crashed, but that seems incredibly unlikely given that the + # same script on the same server ran fine *except* under MT. I've spent + # more time trying to figure out why this is happening than I'd like to + # admit. My only guess, backed up by the fact that this workaround works, + # is that Perl optimizes the substition when it can figure out that the + # pattern will never change, and when this optimization isn't on, we run + # afoul of the reaper. Thus, the slightly redundant code to that uses two + # static s/// patterns rather than one conditional pattern. + + if self.list_level: + sub_list_re = re.compile("^"+whole_list, re.X | re.M | re.S) + text = sub_list_re.sub(self._list_sub, text) + else: + list_re = re.compile(r"(?:(?<=\n\n)|\A\n?)"+whole_list, + re.X | re.M | re.S) + text = list_re.sub(self._list_sub, text) + + return text + + _list_item_re = re.compile(r''' + (\n)? # leading line = \1 + (^[ \t]*) # leading whitespace = \2 + (%s) [ \t]+ # list marker = \3 + ((?:.+?) # list item text = \4 + (\n{1,2})) # eols = \5 + (?= \n* (\Z | \2 (%s) [ \t]+)) + ''' % (_marker_any, _marker_any), + re.M | re.X | re.S) + + _last_li_endswith_two_eols = False + def _list_item_sub(self, match): + item = match.group(4) + leading_line = match.group(1) + leading_space = match.group(2) + if leading_line or "\n\n" in item or self._last_li_endswith_two_eols: + item = self._run_block_gamut(self._outdent(item)) + else: + # Recursion for sub-lists: + item = self._do_lists(self._outdent(item)) + if item.endswith('\n'): + item = item[:-1] + item = self._run_span_gamut(item) + self._last_li_endswith_two_eols = (len(match.group(5)) == 2) + return "<li>%s</li>\n" % item + + def _process_list_items(self, list_str): + # Process the contents of a single ordered or unordered list, + # splitting it into individual list items. + + # The $g_list_level global keeps track of when we're inside a list. + # Each time we enter a list, we increment it; when we leave a list, + # we decrement. If it's zero, we're not in a list anymore. + # + # We do this because when we're not inside a list, we want to treat + # something like this: + # + # I recommend upgrading to version + # 8. Oops, now this line is treated + # as a sub-list. + # + # As a single paragraph, despite the fact that the second line starts + # with a digit-period-space sequence. + # + # Whereas when we're inside a list (or sub-list), that line will be + # treated as the start of a sub-list. What a kludge, huh? This is + # an aspect of Markdown's syntax that's hard to parse perfectly + # without resorting to mind-reading. Perhaps the solution is to + # change the syntax rules such that sub-lists must start with a + # starting cardinal number; e.g. "1." or "a.". + self.list_level += 1 + self._last_li_endswith_two_eols = False + list_str = list_str.rstrip('\n') + '\n' + list_str = self._list_item_re.sub(self._list_item_sub, list_str) + self.list_level -= 1 + return list_str + + def _get_pygments_lexer(self, lexer_name): + try: + from pygments import lexers, util + except ImportError: + return None + try: + return lexers.get_lexer_by_name(lexer_name) + except util.ClassNotFound: + return None + + def _color_with_pygments(self, codeblock, lexer, **formatter_opts): + import pygments + import pygments.formatters + + class HtmlCodeFormatter(pygments.formatters.HtmlFormatter): + def _wrap_code(self, inner): + """A function for use in a Pygments Formatter which + wraps in <code> tags. + """ + yield 0, "<code>" + for tup in inner: + yield tup + yield 0, "</code>" + + def wrap(self, source, outfile): + """Return the source with a code, pre, and div.""" + return self._wrap_div(self._wrap_pre(self._wrap_code(source))) + + formatter = HtmlCodeFormatter(cssclass="codehilite", **formatter_opts) + return pygments.highlight(codeblock, lexer, formatter) + + def _code_block_sub(self, match): + codeblock = match.group(1) + codeblock = self._outdent(codeblock) + codeblock = self._detab(codeblock) + codeblock = codeblock.lstrip('\n') # trim leading newlines + codeblock = codeblock.rstrip() # trim trailing whitespace + + if "code-color" in self.extras and codeblock.startswith(":::"): + lexer_name, rest = codeblock.split('\n', 1) + lexer_name = lexer_name[3:].strip() + lexer = self._get_pygments_lexer(lexer_name) + codeblock = rest.lstrip("\n") # Remove lexer declaration line. + if lexer: + formatter_opts = self.extras['code-color'] or {} + colored = self._color_with_pygments(codeblock, lexer, + **formatter_opts) + return "\n\n%s\n\n" % colored + + codeblock = self._encode_code(codeblock) + return "\n\n<pre><code>%s\n</code></pre>\n\n" % codeblock + + def _do_code_blocks(self, text): + """Process Markdown `<pre><code>` blocks.""" + code_block_re = re.compile(r''' + (?:\n\n|\A) + ( # $1 = the code block -- one or more lines, starting with a space/tab + (?: + (?:[ ]{%d} | \t) # Lines must start with a tab or a tab-width of spaces + .*\n+ + )+ + ) + ((?=^[ ]{0,%d}\S)|\Z) # Lookahead for non-space at line-start, or end of doc + ''' % (self.tab_width, self.tab_width), + re.M | re.X) + + return code_block_re.sub(self._code_block_sub, text) + + + # Rules for a code span: + # - backslash escapes are not interpreted in a code span + # - to include one or or a run of more backticks the delimiters must + # be a longer run of backticks + # - cannot start or end a code span with a backtick; pad with a + # space and that space will be removed in the emitted HTML + # See `test/tm-cases/escapes.text` for a number of edge-case + # examples. + _code_span_re = re.compile(r''' + (?<!\\) + (`+) # \1 = Opening run of ` + (?!`) # See Note A test/tm-cases/escapes.text + (.+?) # \2 = The code block + (?<!`) + \1 # Matching closer + (?!`) + ''', re.X | re.S) + + def _code_span_sub(self, match): + c = match.group(2).strip(" \t") + c = self._encode_code(c) + return "<code>%s</code>" % c + + def _do_code_spans(self, text): + # * Backtick quotes are used for <code></code> spans. + # + # * You can use multiple backticks as the delimiters if you want to + # include literal backticks in the code span. So, this input: + # + # Just type ``foo `bar` baz`` at the prompt. + # + # Will translate to: + # + # <p>Just type <code>foo `bar` baz</code> at the prompt.</p> + # + # There's no arbitrary limit to the number of backticks you + # can use as delimters. If you need three consecutive backticks + # in your code, use four for delimiters, etc. + # + # * You can use spaces to get literal backticks at the edges: + # + # ... type `` `bar` `` ... + # + # Turns to: + # + # ... type <code>`bar`</code> ... + return self._code_span_re.sub(self._code_span_sub, text) + + def _encode_code(self, text): + """Encode/escape certain characters inside Markdown code runs. + The point is that in code, these characters are literals, + and lose their special Markdown meanings. + """ + replacements = [ + # Encode all ampersands; HTML entities are not + # entities within a Markdown code span. + ('&', '&'), + # Do the angle bracket song and dance: + ('<', '<'), + ('>', '>'), + # Now, escape characters that are magic in Markdown: + ('*', g_escape_table['*']), + ('_', g_escape_table['_']), + ('{', g_escape_table['{']), + ('}', g_escape_table['}']), + ('[', g_escape_table['[']), + (']', g_escape_table[']']), + ('\\', g_escape_table['\\']), + ] + for before, after in replacements: + text = text.replace(before, after) + return text + + _strong_re = re.compile(r"(\*\*|__)(?=\S)(.+?[*_]*)(?<=\S)\1", re.S) + _em_re = re.compile(r"(\*|_)(?=\S)(.+?)(?<=\S)\1", re.S) + _code_friendly_strong_re = re.compile(r"\*\*(?=\S)(.+?[*_]*)(?<=\S)\*\*", re.S) + _code_friendly_em_re = re.compile(r"\*(?=\S)(.+?)(?<=\S)\*", re.S) + def _do_italics_and_bold(self, text): + # <strong> must go first: + if "code-friendly" in self.extras: + text = self._code_friendly_strong_re.sub(r"<strong>\1</strong>", text) + text = self._code_friendly_em_re.sub(r"<em>\1</em>", text) + else: + text = self._strong_re.sub(r"<strong>\2</strong>", text) + text = self._em_re.sub(r"<em>\2</em>", text) + return text + + + _block_quote_re = re.compile(r''' + ( # Wrap whole match in \1 + ( + ^[ \t]*>[ \t]? # '>' at the start of a line + .+\n # rest of the first line + (.+\n)* # subsequent consecutive lines + \n* # blanks + )+ + ) + ''', re.M | re.X) + _bq_one_level_re = re.compile('^[ \t]*>[ \t]?', re.M); + + _html_pre_block_re = re.compile(r'(\s*<pre>.+?</pre>)', re.S) + def _dedent_two_spaces_sub(self, match): + return re.sub(r'(?m)^ ', '', match.group(1)) + + def _block_quote_sub(self, match): + bq = match.group(1) + bq = self._bq_one_level_re.sub('', bq) # trim one level of quoting + bq = self._ws_only_line_re.sub('', bq) # trim whitespace-only lines + bq = self._run_block_gamut(bq) # recurse + + bq = re.sub('(?m)^', ' ', bq) + # These leading spaces screw with <pre> content, so we need to fix that: + bq = self._html_pre_block_re.sub(self._dedent_two_spaces_sub, bq) + + return "<blockquote>\n%s\n</blockquote>\n\n" % bq + + def _do_block_quotes(self, text): + if '>' not in text: + return text + return self._block_quote_re.sub(self._block_quote_sub, text) + + def _form_paragraphs(self, text): + # Strip leading and trailing lines: + text = text.strip('\n') + + # Wrap <p> tags. + grafs = re.split(r"\n{2,}", text) + for i, graf in enumerate(grafs): + if graf in self.html_blocks: + # Unhashify HTML blocks + grafs[i] = self.html_blocks[graf] + else: + # Wrap <p> tags. + graf = self._run_span_gamut(graf) + grafs[i] = "<p>" + graf.lstrip(" \t") + "</p>" + + return "\n\n".join(grafs) + + def _add_footnotes(self, text): + if self.footnotes: + footer = [ + '<div class="footnotes">', + '<hr' + self.empty_element_suffix, + '<ol>', + ] + for i, id in enumerate(self.footnote_ids): + if i != 0: + footer.append('') + footer.append('<li id="fn-%s">' % id) + footer.append(self._run_block_gamut(self.footnotes[id])) + backlink = ('<a href="#fnref-%s" ' + 'class="footnoteBackLink" ' + 'title="Jump back to footnote %d in the text.">' + '↩</a>' % (id, i+1)) + if footer[-1].endswith("</p>"): + footer[-1] = footer[-1][:-len("</p>")] \ + + ' ' + backlink + "</p>" + else: + footer.append("\n<p>%s</p>" % backlink) + footer.append('</li>') + footer.append('</ol>') + footer.append('</div>') + return text + '\n\n' + '\n'.join(footer) + else: + return text + + # Ampersand-encoding based entirely on Nat Irons's Amputator MT plugin: + # http://bumppo.net/projects/amputator/ + _ampersand_re = re.compile(r'&(?!#?[xX]?(?:[0-9a-fA-F]+|\w+);)') + _naked_lt_re = re.compile(r'<(?![a-z/?\$!])', re.I) + _naked_gt_re = re.compile(r'''(?<![a-z?!/'"-])>''', re.I) + + def _encode_amps_and_angles(self, text): + # Smart processing for ampersands and angle brackets that need + # to be encoded. + text = self._ampersand_re.sub('&', text) + + # Encode naked <'s + text = self._naked_lt_re.sub('<', text) + + # Encode naked >'s + # Note: Other markdown implementations (e.g. Markdown.pl, PHP + # Markdown) don't do this. + text = self._naked_gt_re.sub('>', text) + return text + + def _encode_backslash_escapes(self, text): + for ch, escape in g_escape_table.items(): + text = text.replace("\\"+ch, escape) + return text + + _auto_link_re = re.compile(r'<((https?|ftp):[^\'">\s]+)>', re.I) + def _auto_link_sub(self, match): + g1 = match.group(1) + return '<a href="%s">%s</a>' % (g1, g1) + + _auto_email_link_re = re.compile(r""" + < + (?:mailto:)? + ( + [-.\w]+ + \@ + [-\w]+(\.[-\w]+)*\.[a-z]+ + ) + > + """, re.I | re.X | re.U) + def _auto_email_link_sub(self, match): + return self._encode_email_address( + self._unescape_special_chars(match.group(1))) + + def _do_auto_links(self, text): + text = self._auto_link_re.sub(self._auto_link_sub, text) + text = self._auto_email_link_re.sub(self._auto_email_link_sub, text) + return text + + def _encode_email_address(self, addr): + # Input: an email address, e.g. "foo@example.com" + # + # Output: the email address as a mailto link, with each character + # of the address encoded as either a decimal or hex entity, in + # the hopes of foiling most address harvesting spam bots. E.g.: + # + # <a href="mailto:foo@e + # xample.com">foo + # @example.com</a> + # + # Based on a filter by Matthew Wickline, posted to the BBEdit-Talk + # mailing list: <http://tinyurl.com/yu7ue> + chars = [_xml_encode_email_char_at_random(ch) + for ch in "mailto:" + addr] + # Strip the mailto: from the visible part. + addr = '<a href="%s">%s</a>' \ + % (''.join(chars), ''.join(chars[7:])) + return addr + + def _do_link_patterns(self, text): + """Caveat emptor: there isn't much guarding against link + patterns being formed inside other standard Markdown links, e.g. + inside a [link def][like this]. + + Dev Notes: *Could* consider prefixing regexes with a negative + lookbehind assertion to attempt to guard against this. + """ + link_from_hash = {} + for regex, repl in self.link_patterns: + replacements = [] + for match in regex.finditer(text): + if hasattr(repl, "__call__"): + href = repl(match) + else: + href = match.expand(repl) + replacements.append((match.span(), href)) + for (start, end), href in reversed(replacements): + escaped_href = ( + href.replace('"', '"') # b/c of attr quote + # To avoid markdown <em> and <strong>: + .replace('*', g_escape_table['*']) + .replace('_', g_escape_table['_'])) + link = '<a href="%s">%s</a>' % (escaped_href, text[start:end]) + hash = md5(link).hexdigest() + link_from_hash[hash] = link + text = text[:start] + hash + text[end:] + for hash, link in link_from_hash.items(): + text = text.replace(hash, link) + return text + + def _unescape_special_chars(self, text): + # Swap back in all the special characters we've hidden. + for ch, hash in g_escape_table.items(): + text = text.replace(hash, ch) + return text + + def _outdent(self, text): + # Remove one level of line-leading tabs or spaces + return self._outdent_re.sub('', text) + + +class MarkdownWithExtras(Markdown): + """A markdowner class that enables most extras: + + - footnotes + - code-color (only has effect if 'pygments' Python module on path) + + These are not included: + - pyshell (specific to Python-related documenting) + - code-friendly (because it *disables* part of the syntax) + - link-patterns (because you need to specify some actual + link-patterns anyway) + """ + extras = ["footnotes", "code-color"] + + +#---- internal support functions + +# From http://aspn.activestate.com/ASPN/Cookbook/Python/Recipe/52549 +def _curry(*args, **kwargs): + function, args = args[0], args[1:] + def result(*rest, **kwrest): + combined = kwargs.copy() + combined.update(kwrest) + return function(*args + rest, **combined) + return result + +# Recipe: regex_from_encoded_pattern (1.0) +def _regex_from_encoded_pattern(s): + """'foo' -> re.compile(re.escape('foo')) + '/foo/' -> re.compile('foo') + '/foo/i' -> re.compile('foo', re.I) + """ + if s.startswith('/') and s.rfind('/') != 0: + # Parse it: /PATTERN/FLAGS + idx = s.rfind('/') + pattern, flags_str = s[1:idx], s[idx+1:] + flag_from_char = { + "i": re.IGNORECASE, + "l": re.LOCALE, + "s": re.DOTALL, + "m": re.MULTILINE, + "u": re.UNICODE, + } + flags = 0 + for char in flags_str: + try: + flags |= flag_from_char[char] + except KeyError: + raise ValueError("unsupported regex flag: '%s' in '%s' " + "(must be one of '%s')" + % (char, s, ''.join(flag_from_char.keys()))) + return re.compile(s[1:idx], flags) + else: # not an encoded regex + return re.compile(re.escape(s)) + +# Recipe: dedent (0.1.2) +def _dedentlines(lines, tabsize=8, skip_first_line=False): + """_dedentlines(lines, tabsize=8, skip_first_line=False) -> dedented lines + + "lines" is a list of lines to dedent. + "tabsize" is the tab width to use for indent width calculations. + "skip_first_line" is a boolean indicating if the first line should + be skipped for calculating the indent width and for dedenting. + This is sometimes useful for docstrings and similar. + + Same as dedent() except operates on a sequence of lines. Note: the + lines list is modified **in-place**. + """ + DEBUG = False + if DEBUG: + print "dedent: dedent(..., tabsize=%d, skip_first_line=%r)"\ + % (tabsize, skip_first_line) + indents = [] + margin = None + for i, line in enumerate(lines): + if i == 0 and skip_first_line: continue + indent = 0 + for ch in line: + if ch == ' ': + indent += 1 + elif ch == '\t': + indent += tabsize - (indent % tabsize) + elif ch in '\r\n': + continue # skip all-whitespace lines + else: + break + else: + continue # skip all-whitespace lines + if DEBUG: print "dedent: indent=%d: %r" % (indent, line) + if margin is None: + margin = indent + else: + margin = min(margin, indent) + if DEBUG: print "dedent: margin=%r" % margin + + if margin is not None and margin > 0: + for i, line in enumerate(lines): + if i == 0 and skip_first_line: continue + removed = 0 + for j, ch in enumerate(line): + if ch == ' ': + removed += 1 + elif ch == '\t': + removed += tabsize - (removed % tabsize) + elif ch in '\r\n': + if DEBUG: print "dedent: %r: EOL -> strip up to EOL" % line + lines[i] = lines[i][j:] + break + else: + raise ValueError("unexpected non-whitespace char %r in " + "line %r while removing %d-space margin" + % (ch, line, margin)) + if DEBUG: + print "dedent: %r: %r -> removed %d/%d"\ + % (line, ch, removed, margin) + if removed == margin: + lines[i] = lines[i][j+1:] + break + elif removed > margin: + lines[i] = ' '*(removed-margin) + lines[i][j+1:] + break + else: + if removed: + lines[i] = lines[i][removed:] + return lines + +def _dedent(text, tabsize=8, skip_first_line=False): + """_dedent(text, tabsize=8, skip_first_line=False) -> dedented text + + "text" is the text to dedent. + "tabsize" is the tab width to use for indent width calculations. + "skip_first_line" is a boolean indicating if the first line should + be skipped for calculating the indent width and for dedenting. + This is sometimes useful for docstrings and similar. + + textwrap.dedent(s), but don't expand tabs to spaces + """ + lines = text.splitlines(1) + _dedentlines(lines, tabsize=tabsize, skip_first_line=skip_first_line) + return ''.join(lines) + + +class _memoized(object): + """Decorator that caches a function's return value each time it is called. + If called later with the same arguments, the cached value is returned, and + not re-evaluated. + + http://wiki.python.org/moin/PythonDecoratorLibrary + """ + def __init__(self, func): + self.func = func + self.cache = {} + def __call__(self, *args): + try: + return self.cache[args] + except KeyError: + self.cache[args] = value = self.func(*args) + return value + except TypeError: + # uncachable -- for instance, passing a list as an argument. + # Better to not cache than to blow up entirely. + return self.func(*args) + def __repr__(self): + """Return the function's docstring.""" + return self.func.__doc__ + + +def _xml_oneliner_re_from_tab_width(tab_width): + """Standalone XML processing instruction regex.""" + return re.compile(r""" + (?: + (?<=\n\n) # Starting after a blank line + | # or + \A\n? # the beginning of the doc + ) + ( # save in $1 + [ ]{0,%d} + (?: + <\?\w+\b\s+.*?\?> # XML processing instruction + | + <\w+:\w+\b\s+.*?/> # namespaced single tag + ) + [ \t]* + (?=\n{2,}|\Z) # followed by a blank line or end of document + ) + """ % (tab_width - 1), re.X) +_xml_oneliner_re_from_tab_width = _memoized(_xml_oneliner_re_from_tab_width) + +def _hr_tag_re_from_tab_width(tab_width): + return re.compile(r""" + (?: + (?<=\n\n) # Starting after a blank line + | # or + \A\n? # the beginning of the doc + ) + ( # save in \1 + [ ]{0,%d} + <(hr) # start tag = \2 + \b # word break + ([^<>])*? # + /?> # the matching end tag + [ \t]* + (?=\n{2,}|\Z) # followed by a blank line or end of document + ) + """ % (tab_width - 1), re.X) +_hr_tag_re_from_tab_width = _memoized(_hr_tag_re_from_tab_width) + + +def _xml_encode_email_char_at_random(ch): + r = random() + # Roughly 10% raw, 45% hex, 45% dec. + # '@' *must* be encoded. I [John Gruber] insist. + # Issue 26: '_' must be encoded. + if r > 0.9 and ch not in "@_": + return ch + elif r < 0.45: + # The [1:] is to drop leading '0': 0x63 -> x63 + return '&#%s;' % hex(ord(ch))[1:] + else: + return '&#%s;' % ord(ch) + +def _hash_text(text): + return 'md5:'+md5(text.encode("utf-8")).hexdigest() + + +#---- mainline + +class _NoReflowFormatter(optparse.IndentedHelpFormatter): + """An optparse formatter that does NOT reflow the description.""" + def format_description(self, description): + return description or "" + +def _test(): + import doctest + doctest.testmod() + +def main(argv=None): + if argv is None: + argv = sys.argv + if not logging.root.handlers: + logging.basicConfig() + + usage = "usage: %prog [PATHS...]" + version = "%prog "+__version__ + parser = optparse.OptionParser(prog="markdown2", usage=usage, + version=version, description=cmdln_desc, + formatter=_NoReflowFormatter()) + parser.add_option("-v", "--verbose", dest="log_level", + action="store_const", const=logging.DEBUG, + help="more verbose output") + parser.add_option("--encoding", + help="specify encoding of text content") + parser.add_option("--html4tags", action="store_true", default=False, + help="use HTML 4 style for empty element tags") + parser.add_option("-s", "--safe", metavar="MODE", dest="safe_mode", + help="sanitize literal HTML: 'escape' escapes " + "HTML meta chars, 'replace' replaces with an " + "[HTML_REMOVED] note") + parser.add_option("-x", "--extras", action="append", + help="Turn on specific extra features (not part of " + "the core Markdown spec). Supported values: " + "'code-friendly' disables _/__ for emphasis; " + "'code-color' adds code-block syntax coloring; " + "'link-patterns' adds auto-linking based on patterns; " + "'footnotes' adds the footnotes syntax;" + "'xml' passes one-liner processing instructions and namespaced XML tags;" + "'pyshell' to put unindented Python interactive shell sessions in a <code> block.") + parser.add_option("--use-file-vars", + help="Look for and use Emacs-style 'markdown-extras' " + "file var to turn on extras. See " + "<http://code.google.com/p/python-markdown2/wiki/Extras>.") + parser.add_option("--link-patterns-file", + help="path to a link pattern file") + parser.add_option("--self-test", action="store_true", + help="run internal self-tests (some doctests)") + parser.add_option("--compare", action="store_true", + help="run against Markdown.pl as well (for testing)") + parser.set_defaults(log_level=logging.INFO, compare=False, + encoding="utf-8", safe_mode=None, use_file_vars=False) + opts, paths = parser.parse_args() + log.setLevel(opts.log_level) + + if opts.self_test: + return _test() + + if opts.extras: + extras = {} + for s in opts.extras: + splitter = re.compile("[,;: ]+") + for e in splitter.split(s): + if '=' in e: + ename, earg = e.split('=', 1) + try: + earg = int(earg) + except ValueError: + pass + else: + ename, earg = e, None + extras[ename] = earg + else: + extras = None + + if opts.link_patterns_file: + link_patterns = [] + f = open(opts.link_patterns_file) + try: + for i, line in enumerate(f.readlines()): + if not line.strip(): continue + if line.lstrip().startswith("#"): continue + try: + pat, href = line.rstrip().rsplit(None, 1) + except ValueError: + raise MarkdownError("%s:%d: invalid link pattern line: %r" + % (opts.link_patterns_file, i+1, line)) + link_patterns.append( + (_regex_from_encoded_pattern(pat), href)) + finally: + f.close() + else: + link_patterns = None + + from os.path import join, dirname, abspath, exists + markdown_pl = join(dirname(dirname(abspath(__file__))), "test", + "Markdown.pl") + for path in paths: + if opts.compare: + print "==== Markdown.pl ====" + perl_cmd = 'perl %s "%s"' % (markdown_pl, path) + o = os.popen(perl_cmd) + perl_html = o.read() + o.close() + sys.stdout.write(perl_html) + print "==== markdown2.py ====" + html = markdown_path(path, encoding=opts.encoding, + html4tags=opts.html4tags, + safe_mode=opts.safe_mode, + extras=extras, link_patterns=link_patterns, + use_file_vars=opts.use_file_vars) + sys.stdout.write( + html.encode(sys.stdout.encoding or "utf-8", 'xmlcharrefreplace')) + if opts.compare: + test_dir = join(dirname(dirname(abspath(__file__))), "test") + if exists(join(test_dir, "test_markdown2.py")): + sys.path.insert(0, test_dir) + from test_markdown2 import norm_html_from_html + norm_html = norm_html_from_html(html) + norm_perl_html = norm_html_from_html(perl_html) + else: + norm_html = html + norm_perl_html = perl_html + print "==== match? %r ====" % (norm_perl_html == norm_html) + + +if __name__ == "__main__": + sys.exit( main(sys.argv) ) + diff --git a/lib/utils/pydelicious.py b/lib/utils/pydelicious.py index dd33788..8e45843 100644 --- a/lib/utils/pydelicious.py +++ b/lib/utils/pydelicious.py @@ -1,131 +1,126 @@ """Library to access del.icio.us data via Python. -:examples: - - Using the API class directly: - - >>> a = pydelicious.apiNew('user', 'passwd') - >>> # or: - >>> a = DeliciousAPI('user', 'passwd') - >>> a.tags_get() # Same as: - >>> a.request('tags/get', ) - - Or by calling one of the methods on the module: - - - add(user, passwd, url, description, tags = "", extended = "", dt = "", replace="no") - - get(user, passwd, tag="", dt="", count = 0) - - get_all(user, passwd, tag = "") - - delete(user, passwd, url) - - rename_tag(user, passwd, oldtag, newtag) - - get_tags(user, passwd) - - >>> a = apiNew(user, passwd) - >>> a.posts_add(url="http://my.com/", desciption="my.com", extended="the url is my.moc", tags="my com") - True - >>> len(a.posts_all()) - 1 - >>> get_all(user, passwd) - 1 - - This are short functions for getrss calls. - - >>> rss_ - -def get_userposts(user): -def get_tagposts(tag): -def get_urlposts(url): -def get_popular(tag = ""): - - >>> json_posts() - >>> json_tags() - >>> json_network() - >>> json_fans() - -:License: pydelicious is released under the BSD license. See 'license.txt' - for more informations. - -:todo, bvb: - - Rewriting comments to english. More documentation, examples. - - Added JSON-like return values for XML data (del.icio.us also serves some JSON...) - - better error/exception classes and handling, work in progress. - -:todo: - - Source code SHOULD BE ASCII! - - More tests. - - handling different encodings, what, how? - >>> pydelicious.getrss(tag="t[a]g") - url: http://del.icio.us/rss/tag/t[a]g - - Parse datetimes in XML. - - Test RSS functionality? HTML scraping doesn't work yet? - - API functions need required argument checks. - - interesting functionality in other libraries (ruby, java, perl, etc)? - - what is pydelicious used for? - - license, readme docs via setup.py verdelen? - - automatic releas build - -:done: - * Refactored the API class, much cleaner now and functions dlcs_api_request, dlcs_parse_xml are available for who wants them. +An introduction to the project is given in the README. +pydelicious is released under the BSD license. See license.txt for details +and the copyright holders. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS 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 THE COPYRIGHT OWNER OR 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. + +TODO: + - distribute license, readme docs via setup.py? + - automatic release build? """ import sys import os import time import datetime -import md5, httplib -import urllib, urllib2, time +import locale +import httplib +import urllib2 +from urllib import urlencode, quote_plus from StringIO import StringIO +from pprint import pformat + +v = sys.version_info +if v[0] >= 2 and v[1] >= 5: + from hashlib import md5 +else: + from md5 import md5 try: from elementtree.ElementTree import parse as parse_xml except ImportError: - from xml.etree.ElementTree import parse as parse_xml + # Python 2.5 and higher + from xml.etree.ElementTree import parse as parse_xml -import feedparser +try: + import feedparser +except ImportError: + print >>sys.stderr, \ + "Feedparser not available, no RSS parsing." + feedparser = None ### Static config -__version__ = '0.5.0' -__author__ = 'Frank Timmermann <regenkind_at_gmx_dot_de>' # GP: does not respond to emails +__version__ = '0.5.3' +__author__ = 'Frank Timmermann <regenkind_at_gmx_dot_de>' + # GP: does not respond to emails __contributors__ = [ 'Greg Pinero', 'Berend van Berkum <berend+pydelicious@dotmpe.com>'] __url__ = 'http://code.google.com/p/pydelicious/' -__author_email__ = "" # Old URL: 'http://deliciouspython.python-hosting.com/' - -__description__ = '''pydelicious.py allows you to access the web service of del.icio.us via it's API through python.''' -__long_description__ = '''the goal is to design an easy to use and fully functional python interface to del.icio.us. ''' - -DLCS_OK_MESSAGES = ('done', 'ok') # Known text values of positive del.icio.us <result> answers +__author_email__ = "" +__docformat__ = "restructuredtext en" +__description__ = "pydelicious.py allows you to access the web service of " \ + "del.icio.us via it's API through Python." +__long_description__ = "The goal is to design an easy to use and fully " \ + "functional Python interface to del.icio.us." + +DLCS_OK_MESSAGES = ('done', 'ok') +"Known text values of positive del.icio.us <result/> answers" DLCS_WAIT_TIME = 4 -DLCS_REQUEST_TIMEOUT = 444 # Seconds before socket triggers timeout +"Time to wait between API requests" +DLCS_REQUEST_TIMEOUT = 444 +"Seconds before socket triggers timeout" #DLCS_API_REALM = 'del.icio.us API' -DLCS_API_HOST = 'https://api.del.icio.us' +DLCS_API_HOST = 'api.del.icio.us' DLCS_API_PATH = 'v1' -DLCS_API = "%s/%s" % (DLCS_API_HOST, DLCS_API_PATH) +DLCS_API = "https://%s/%s" % (DLCS_API_HOST, DLCS_API_PATH) DLCS_RSS = 'http://del.icio.us/rss/' +DLCS_FEEDS = 'http://feeds.delicious.com/v2/' + +PREFERRED_ENCODING = locale.getpreferredencoding() +# XXX: might need to check sys.platform/encoding combinations here, ie +#if sys.platform == 'darwin' || PREFERRED_ENCODING == 'macroman: +# PREFERRED_ENCODING = 'utf-8' +if not PREFERRED_ENCODING: + PREFERRED_ENCODING = 'iso-8859-1' ISO_8601_DATETIME = '%Y-%m-%dT%H:%M:%SZ' -USER_AGENT = 'pydelicious.py/%s %s' % (__version__, __url__) +USER_AGENT = 'pydelicious/%s %s' % (__version__, __url__) DEBUG = 0 if 'DLCS_DEBUG' in os.environ: DEBUG = int(os.environ['DLCS_DEBUG']) - - -# Taken from FeedParser.py -# timeoutsocket allows feedparser to time out rather than hang forever on ultra-slow servers. -# Python 2.3 now has this functionality available in the standard socket library, so under -# 2.3 you don't need to install anything. But you probably should anyway, because the socket -# module is buggy and timeoutsocket is better. + if DEBUG: + print >>sys.stderr, \ + "Set DEBUG to %i from DLCS_DEBUG env." % DEBUG + +HTTP_PROXY = None +if 'HTTP_PROXY' in os.environ: + HTTP_PROXY = os.environ['HTTP_PROXY'] + if DEBUG: + print >>sys.stderr, \ + "Set HTTP_PROXY to %i from env." % HTTP_PROXY + +### Timeoutsocket hack taken from FeedParser.py + +# timeoutsocket allows feedparser to time out rather than hang forever on ultra- +# slow servers. Python 2.3 now has this functionality available in the standard +# socket library, so under 2.3 you don't need to install anything. But you +# probably should anyway, because the socket module is buggy and timeoutsocket +# is better. try: import timeoutsocket # http://www.timo-tasi.org/python/timeoutsocket.py timeoutsocket.setDefaultSocketTimeout(DLCS_REQUEST_TIMEOUT) except ImportError: import socket - if hasattr(socket, 'setdefaulttimeout'): socket.setdefaulttimeout(DLCS_REQUEST_TIMEOUT) -if DEBUG: print >>sys.stderr, "Set socket timeout to %s seconds" % DLCS_REQUEST_TIMEOUT + if hasattr(socket, 'setdefaulttimeout'): + socket.setdefaulttimeout(DLCS_REQUEST_TIMEOUT) +if DEBUG: print >>sys.stderr, \ + "Set socket timeout to %s seconds" % DLCS_REQUEST_TIMEOUT ### Utility classes @@ -163,91 +158,64 @@ class _Waiter: Waiter = _Waiter(DLCS_WAIT_TIME) + class PyDeliciousException(Exception): - '''Std. pydelicious error''' - pass + """Standard pydelicious error""" +class PyDeliciousThrottled(Exception): pass +class PyDeliciousUnauthorized(Exception): pass class DeliciousError(Exception): """Raised when the server responds with a negative answer""" + @staticmethod + def raiseFor(error_string, path, **params): + if error_string == 'item already exists': + raise DeliciousItemExistsError, params['url'] + else: + raise DeliciousError, "%s, while calling <%s?%s>" % (error_string, + path, urlencode(params)) -class DefaultErrorHandler(urllib2.HTTPDefaultErrorHandler): - '''xxx, bvb: Where is this used? should it be registered somewhere with urllib2? - - Handles HTTP Error, currently only 503. - ''' - def http_error_503(self, req, fp, code, msg, headers): - raise urllib2.HTTPError(req, code, throttled_message, headers, fp) - +class DeliciousItemExistsError(DeliciousError): + """Raised then adding an already existing post.""" -class post(dict): - """Post object, contains href, description, hash, dt, tags, - extended, user, count(, shared). - xxx, bvb: Not used in DeliciousAPI - """ - def __init__(self, href="", description="", hash="", time="", tag="", extended="", user="", count="", - tags="", url="", dt=""): # tags or tag? - self["href"] = href - if url != "": self["href"] = url - self["description"] = description - self["hash"] = hash - self["dt"] = dt - if time != "": self["dt"] = time - self["tags"] = tags - if tag != "": self["tags"] = tag # tag or tags? # !! tags - self["extended"] = extended - self["user"] = user - self["count"] = count - - def __getattr__(self, name): - try: return self[name] - except: object.__getattribute__(self, name) - - -class posts(list): - def __init__(self, *args): - for i in args: self.append(i) - - def __getattr__(self, attr): - try: return [p[attr] for p in self] - except: object.__getattribute__(self, attr) +class HTTPErrorHandler(urllib2.HTTPDefaultErrorHandler): -### Utility functions + def http_error_401(self, req, fp, code, msg, headers): + raise PyDeliciousUnauthorized, "Check credentials." -def str2uni(s): - # type(in) str or unicode - # type(out) unicode - return ("".join([unichr(ord(i)) for i in s])) + def http_error_503(self, req, fp, code, msg, headers): + # Retry-After? + errmsg = "Try again later." + if 'Retry-After' in headers: + errmsg = "You may try again after %s" % headers['Retry-After'] + raise PyDeliciousThrottled, errmsg -def str2utf8(s): - # type(in) str or unicode - # type(out) str - return ("".join([unichr(ord(i)).encode("utf-8") for i in s])) -def str2quote(s): - return urllib.quote_plus("".join([unichr(ord(i)).encode("utf-8") for i in s])) +### Utility functions def dict0(d): - # Trims empty dict entries - # {'a':'a', 'b':'', 'c': 'c'} => {'a': 'a', 'c': 'c'} - dd = dict() - for i in d: - if d[i] != "": dd[i] = d[i] - return dd + "Removes empty string values from dictionary" + return dict([(k,v) for k,v in d.items() + if v=='' and isinstance(v, basestring)]) + def delicious_datetime(str): """Parse a ISO 8601 formatted string to a Python datetime ... """ return datetime.datetime(*time.strptime(str, ISO_8601_DATETIME)[0:6]) -def http_request(url, user_agent=USER_AGENT, retry=4): + +def http_request(url, user_agent=USER_AGENT, retry=4, opener=None): """Retrieve the contents referenced by the URL using urllib2. Retries up to four times (default) on exceptions. """ request = urllib2.Request(url, headers={'User-Agent':user_agent}) + if not opener: + opener = urllib2.build_opener() + # Remember last error e = None @@ -255,9 +223,10 @@ def http_request(url, user_agent=USER_AGENT, retry=4): tries = retry; while tries: try: - return urllib2.urlopen(request) + return opener.open(request) - except urllib2.HTTPError, e: # protocol errors, + except urllib2.HTTPError, e: + # reraise unexpected protocol errors as PyDeliciousException raise PyDeliciousException, "%s" % e except urllib2.URLError, e: @@ -273,50 +242,99 @@ def http_request(url, user_agent=USER_AGENT, retry=4): raise PyDeliciousException, \ "Unable to retrieve data at '%s', %s" % (url, e) -def http_auth_request(url, host, user, passwd, user_agent=USER_AGENT): - """Call an HTTP server with authorization credentials using urllib2. + +def build_api_opener(host, user, passwd, extra_handlers=() ): """ + Build a urllib2 style opener with HTTP Basic authorization for one host + and additional error handling. If HTTP_PROXY is set a proxyhandler is also + added. + """ + + global DEBUG + if DEBUG: httplib.HTTPConnection.debuglevel = 1 - # Hook up handler/opener to urllib2 password_manager = urllib2.HTTPPasswordMgrWithDefaultRealm() password_manager.add_password(None, host, user, passwd) auth_handler = urllib2.HTTPBasicAuthHandler(password_manager) - opener = urllib2.build_opener(auth_handler) - urllib2.install_opener(opener) - return http_request(url, user_agent) + extra_handlers += ( HTTPErrorHandler(), ) + if HTTP_PROXY: + extra_handlers += ( urllib2.ProxyHandler( {'http': HTTP_PROXY} ), ) + + return urllib2.build_opener(auth_handler, *extra_handlers) + + +def dlcs_api_opener(user, passwd): + "Build an opener for DLCS_API_HOST, see build_api_opener()" -def dlcs_api_request(path, params='', user='', passwd='', throttle=True): + return build_api_opener(DLCS_API_HOST, user, passwd) + + +def dlcs_api_request(path, params='', user='', passwd='', throttle=True, + opener=None): """Retrieve/query a path within the del.icio.us API. This implements a minimum interval between calls to avoid throttling. [#]_ Use param 'throttle' to turn this behaviour off. - todo: back off on 503's (HTTPError, URLError? testing - - Returned XML does not always correspond with given del.icio.us examples - [#]_. - .. [#] http://del.icio.us/help/api/ """ if throttle: Waiter() if params: - # params come as a dict, strip empty entries and urlencode - url = "%s/%s?%s" % (DLCS_API, path, urllib.urlencode(dict0(params))) + url = "%s/%s?%s" % (DLCS_API, path, urlencode(params)) else: url = "%s/%s" % (DLCS_API, path) - if DEBUG: print >>sys.stderr, "dlcs_api_request: %s" % url + if DEBUG: print >>sys.stderr, \ + "dlcs_api_request: %s" % url - try: - return http_auth_request(url, DLCS_API_HOST, user, passwd, USER_AGENT) + if not opener: + opener = dlcs_api_opener(user, passwd) + + fl = http_request(url, opener=opener) + + if DEBUG>2: print >>sys.stderr, \ + pformat(fl.info().headers) + + return fl + + +def dlcs_encode_params(params, usercodec=PREFERRED_ENCODING): + """Turn all param values (int, list, bool) into utf8 encoded strings. + """ + + if params: + for key in params.keys(): + if isinstance(params[key], bool): + if params[key]: + params[key] = 'yes' + else: + params[key] = 'no' + + elif isinstance(params[key], int): + params[key] = str(params[key]) + + elif not params[key]: + # strip/ignore empties other than False or 0 + del params[key] + continue + + elif isinstance(params[key], list): + params[key] = " ".join(params[key]) + + elif not isinstance(params[key], unicode): + params[key] = params[key].decode(usercodec) + + assert isinstance(params[key], basestring) + + params = dict([ (k, v.encode('utf8')) + for k, v in params.items() if v]) + + return params - # bvb: Is this ever raised? When? - except DefaultErrorHandler, e: - print >>sys.stderr, "%s" % e def dlcs_parse_xml(data, split_tags=False): """Parse any del.icio.us XML document and return Python data structure. @@ -332,6 +350,7 @@ def dlcs_parse_xml(data, split_tags=False): {'result':(True, "done")} # etcetera. """ + # TODO: split_tags is not implemented if DEBUG>3: print >>sys.stderr, "dlcs_parse_xml: parsing from ", data @@ -367,55 +386,67 @@ def dlcs_parse_xml(data, split_tags=False): else: msg = root.text - # Return {'result':(True, msg)} for /known/ O.K. messages, - # use (False, msg) otherwise + # XXX: Return {'result':(True, msg)} for /known/ O.K. messages, + # use (False, msg) otherwise. Move this to DeliciousAPI? v = msg in DLCS_OK_MESSAGES return {fmt: (v, msg)} elif fmt == 'update': # Update: "time" - #return {fmt: root.attrib} - return {fmt: {'time':time.strptime(root.attrib['time'], ISO_8601_DATETIME)}} + return {fmt: { + 'time':time.strptime(root.attrib['time'], ISO_8601_DATETIME) }} else: raise PyDeliciousException, "Unknown XML document format '%s'" % fmt -def dlcs_rss_request(tag = "", popular = 0, user = "", url = ''): - """Handle a request for RSS - - todo: translate from German - rss sollte nun wieder funktionieren, aber diese try, except scheisse ist so nicht schoen +def dlcs_rss_request(tag="", popular=0, user="", url=''): + """Parse a RSS request. - rss wird unterschiedlich zusammengesetzt. ich kann noch keinen einheitlichen zusammenhang - zwischen daten (url, desc, ext, usw) und dem feed erkennen. warum k[o]nnen die das nicht einheitlich machen? + This requests old (now undocumented?) URL paths that still seem to work. """ - tag = str2quote(tag) - user = str2quote(user) + + tag = quote_plus(tag) + user = quote_plus(user) + if url != '': # http://del.icio.us/rss/url/efbfb246d886393d48065551434dab54 - url = DLCS_RSS + '''url/%s'''%md5.new(url).hexdigest() + url = DLCS_RSS + 'url/%s' % md5(url).hexdigest() + elif user != '' and tag != '': - url = DLCS_RSS + '''%(user)s/%(tag)s'''%dict(user=user, tag=tag) + url = DLCS_RSS + '%(user)s/%(tag)s' % {'user':user, 'tag':tag} + elif user != '' and tag == '': # http://del.icio.us/rss/delpy - url = DLCS_RSS + '''%s'''%user + url = DLCS_RSS + '%s' % user + elif popular == 0 and tag == '': url = DLCS_RSS + elif popular == 0 and tag != '': # http://del.icio.us/rss/tag/apple # http://del.icio.us/rss/tag/web2.0 - url = DLCS_RSS + "tag/%s"%tag + url = DLCS_RSS + "tag/%s" % tag + elif popular == 1 and tag == '': - url = DLCS_RSS + '''popular/''' + url = DLCS_RSS + 'popular/' + elif popular == 1 and tag != '': - url = DLCS_RSS + '''popular/%s'''%tag + url = DLCS_RSS + 'popular/%s' % tag + + if DEBUG: + print 'dlcs_rss_request', url + rss = http_request(url).read() + + # assert feedparser, "dlcs_rss_request requires feedparser to be installed." + if not feedparser: + return rss + rss = feedparser.parse(rss) - # print rss -# for e in rss.entries: print e;print - l = posts() + + posts = [] for e in rss.entries: if e.has_key("links") and e["links"]!=[] and e["links"][0].has_key("href"): url = e["links"][0]["href"] @@ -453,59 +484,147 @@ def dlcs_rss_request(tag = "", popular = 0, user = "", url = ''): # die benennung der variablen ist nicht einheitlich # api senden und # xml bekommen sind zwei verschiedene schuhe :( - l.append(post(url = url, description = description, tags = tags, dt = dt, extended = extended, user = user)) - return l + posts.append({'url':url, 'description':description, 'tags':tags, + 'dt':dt, 'extended':extended, 'user':user}) + return posts + + +delicious_v2_feeds = { + #"Bookmarks from the hotlist" + '': "%(format)s", + #"Recent bookmarks" + 'recent': "%(format)s/recent", + #"Recent bookmarks by tag" + 'tagged': "%(format)s/tag/%(tags)s", + #"Popular bookmarks" + 'popular': "%(format)s/popular", + #"Popular bookmarks by tag" + 'popular_tagged': "%(format)s/popular/%(tag)s", + #"Recent site alerts (as seen in the top-of-page alert bar on the site)" + 'alerts': "%(format)s/alerts", + #"Bookmarks for a specific user" + 'user': "%(format)s/%(username)s", + #"Bookmarks for a specific user by tag(s)" + 'user_tagged': "%(format)s/%(username)s/%(tags)s", + #"Public summary information about a user (as seen in the network badge)" + 'user_info': "%(format)s/userinfo/%(username)s", + #"A list of all public tags for a user" + 'user_tags': "%(format)s/tags/%(username)s", + #"Bookmarks from a user's subscriptions" + 'user_subscription': "%(format)s/subscriptions/%(username)s", + #"Private feed for a user's inbox bookmarks from others" + 'user_inbox': "%(format)s/inbox/%(username)s?private=%(key)s", + #"Bookmarks from members of a user's network" + 'user_network': "%(format)s/network/%(username)s", + #"Bookmarks from members of a user's network by tag" + 'user_network_tagged': "%(format)s/network/%(username)s/%(tags)s", + #"A list of a user's network members" + 'user_network_member': "%(format)s/networkmembers/%(username)s", + #"A list of a user's network fans" + 'user_network_fan': "%(format)s/networkfans/%(username)s", + #"Recent bookmarks for a URL" + 'url': "%(format)s/url/%(urlmd5)s", + #"Summary information about a URL (as seen in the tagometer)" + 'urlinfo': "json/urlinfo/%(urlmd5)s", +} + +def dlcs_feed(name_or_url, url_map=delicious_v2_feeds, count=15, **params): + + """ + Request and parse a feed. See delicious_v2_feeds for available names and + required parameters. Format defaults to json. + """ + +# http://delicious.com/help/feeds +# TODO: plain or fancy + + format = params.setdefault('format', 'json') + if count == 'all': +# TODO: fetch all + print >>sys.stderr, "! Maxcount 100 " + count = 100 + + if name_or_url in url_map: + params['count'] = count + url = DLCS_FEEDS + url_map[name_or_url] % params + + else: + url = name_or_url + + if DEBUG: + print 'dlcs_feed', url + + feed = http_request(url).read() + + if format == 'rss': + if feedparser: + rss = feedparser.parse(feed) + return rss + + else: + return feed + + elif format == 'json': + return feed ### Main module class class DeliciousAPI: - """Class providing main interace to del.icio.us API. + + """A single-user Python facade to the del.icio.us HTTP API. + + See http://delicious.com/help/api. Methods ``request`` and ``request_raw`` represent the core. For all API paths there are furthermore methods (e.g. posts_add for 'posts/all') with - an explicit declaration of the parameters and documentation. These all call - ``request`` and pass on extra keywords like ``_raw``. + an explicit declaration of parameters and documentation. """ - def __init__(self, user, passwd, codec='iso-8859-1', api_request=dlcs_api_request, xml_parser=dlcs_parse_xml): - """Initialize access to the API with ``user`` and ``passwd``. + def __init__(self, user, passwd, codec=PREFERRED_ENCODING, + api_request=dlcs_api_request, xml_parser=dlcs_parse_xml, + build_opener=dlcs_api_opener, encode_params=dlcs_encode_params): + + """Initialize access to the API for ``user`` with ``passwd``. - ``codec`` sets the encoding of the arguments. + ``codec`` sets the encoding of the arguments, which defaults to the + users preferred locale. The ``api_request`` and ``xml_parser`` parameters by default point to - functions within this package with standard implementations to + functions within this package with standard implementations which request and parse a resource. See ``dlcs_api_request()`` and - ``dlcs_parse_xml()``. Note that ``api_request`` should return a - file-like instance with an HTTPMessage instance under ``info()``, - see ``urllib2.openurl`` for more info. + ``dlcs_parse_xml()``. + + Parameter ``build_opener`` is a callable that, provided with the + credentials, should build a urllib2 opener for the delicious API server + with HTTP authentication. See ``dlcs_api_opener()`` for the default + implementation. + + ``encode_params`` finally preprocesses API parameters before + they are passed to ``api_request``. """ + assert user != "" self.user = user self.passwd = passwd self.codec = codec # Implement communication to server and parsing of respons messages: + assert callable(encode_params) + self._encode_params = encode_params + assert callable(build_opener) + self._opener = build_opener(user, passwd) assert callable(api_request) self._api_request = api_request assert callable(xml_parser) self._parse_response = xml_parser - def _call_server(self, path, **params): - params = dict0(params) - for key in params: - params[key] = params[key].encode(self.codec) - - # see __init__ for _api_request() - return self._api_request(path, params, self.user, self.passwd) - - ### Core functionality def request(self, path, _raw=False, **params): - """Calls a path in the API, parses the answer to a JSON-like structure by - default. Use with ``_raw=True`` or ``call request_raw()`` directly to - get the filehandler and process the response message manually. + """Sends a request message to `path` in the API, and parses the results + from XML. Use with ``_raw=True`` or ``call request_raw()`` directly + to get the filehandler and process the response message manually. Calls to some paths will return a `result` message, i.e.:: @@ -515,46 +634,56 @@ class DeliciousAPI: <result>...</result> - These are all parsed to ``{'result':(Boolean, MessageString)}`` and this - method will raise ``DeliciousError`` on negative `result` answers. Using - ``_raw=True`` bypasses all parsing and will never raise ``DeliciousError``. + These should all be parsed to ``{'result':(Boolean, MessageString)}``, + this method raises a ``DeliciousError`` on negative `result` answers. + Positive answers are silently accepted and nothing is returned. + + Using ``_raw=True`` bypasses all parsing and never raises + ``DeliciousError``. See ``dlcs_parse_xml()`` and ``self.request_raw()``.""" - # method _parse_response is bound in `__init__()`, `_call_server` - # uses `_api_request` also set in `__init__()` if _raw: # return answer return self.request_raw(path, **params) else: + params = self._encode_params(params, self.codec) + # get answer and parse - fl = self._call_server(path, **params) + fl = self._api_request(path, params=params, opener=self._opener) rs = self._parse_response(fl) - # Raise an error for negative 'result' answers - if type(rs) == dict and rs == 'result' and not rs['result'][0]: - errmsg = "" - if len(rs['result'])>0: - errmsg = rs['result'][1:] - raise DeliciousError, errmsg + if type(rs) == dict and 'result' in rs: + if not rs['result'][0]: + # Raise an error for negative 'result' answers + errmsg = "" + if len(rs['result'])>0: + errmsg = rs['result'][1] + DeliciousError.raiseFor(errmsg, path, **params) + + else: + # not out-of-the-oridinary result, OK + return return rs def request_raw(self, path, **params): - """Calls the path in the API, returns the filehandle. Returned - file-like instances have an ``HTTPMessage`` instance with HTTP header + """Calls the path in the API, returns the filehandle. Returned file- + like instances have an ``HTTPMessage`` instance with HTTP header information available. Use ``filehandle.info()`` or refer to the ``urllib2.openurl`` documentation. """ # see `request()` on how the response can be handled - return self._call_server(path, **params) + params = self._encode_params(params, self.codec) + return self._api_request(path, params=params, opener=self._opener) ### Explicit declarations of API paths, their parameters and docs # Tags def tags_get(self, **kwds): - """Returns a list of tags and the number of times it is used by the user. + """Returns a list of tags and the number of times it is used by the + user. :: <tags> @@ -562,14 +691,22 @@ class DeliciousAPI: """ return self.request("tags/get", **kwds) + def tags_delete(self, tag, **kwds): + """Delete an existing tag. + + &tag={TAG} + (required) Tag to delete + """ + return self.request('tags/delete', tag=tag, **kwds) + def tags_rename(self, old, new, **kwds): """Rename an existing tag with a new tag name. Returns a `result` message or raises an ``DeliciousError``. See ``self.request()``. - &old (required) - Tag to rename. - &new (required) - New name. + &old={TAG} + (required) Tag to rename. + &new={TAG} + (required) New tag name. """ return self.request("tags/rename", old=old, new=new, **kwds) @@ -590,12 +727,12 @@ class DeliciousAPI: <dates> <date date="CCYY-MM-DD" count="888"> - &tag (optional). - Filter by this tag. + &tag={TAG} + (optional) Filter by this tag """ return self.request("posts/dates", tag=tag, **kwds) - def posts_get(self, tag="", dt="", url="", **kwds): + def posts_get(self, tag="", dt="", url="", hashes=[], meta=True, **kwds): """Returns posts matching the arguments. If no date or url is given, most recent date will be used. :: @@ -603,14 +740,25 @@ class DeliciousAPI: <posts dt="CCYY-MM-DD" tag="..." user="..."> <post ...> - &tag (optional). - Filter by this tag. - &dt (optional). - Filter by this date (CCYY-MM-DDThh:mm:ssZ). - &url (optional). - Filter by this url. + &tag={TAG} {TAG} ... {TAG} + (optional) Filter by this/these tag(s). + &dt={CCYY-MM-DDThh:mm:ssZ} + (optional) Filter by this date, defaults to the most recent date on + which bookmarks were saved. + &url={URL} + (optional) Fetch a bookmark for this URL, regardless of date. + &hashes={MD5} {MD5} ... {MD5} + (optional) Fetch multiple bookmarks by one or more URL MD5s + regardless of date. + &meta=yes + (optional) Include change detection signatures on each item in a + 'meta' attribute. Clients wishing to maintain a synchronized local + store of bookmarks should retain the value of this attribute - its + value will change when any significant field of the bookmark + changes. """ - return self.request("posts/get", tag=tag, dt=dt, url=url, **kwds) + return self.request("posts/get", tag=tag, dt=dt, url=url, + hashes=hashes, meta=meta, **kwds) def posts_recent(self, tag="", count="", **kwds): """Returns a list of the most recent posts, filtered by argument. @@ -619,14 +767,15 @@ class DeliciousAPI: <posts tag="..." user="..."> <post ...> - &tag (optional). - Filter by this tag. - &count (optional). - Number of items to retrieve (Default:15, Maximum:100). + &tag={TAG} + (optional) Filter by this tag. + &count={1..100} + (optional) Number of items to retrieve (Default:15, Maximum:100). """ return self.request("posts/recent", tag=tag, count=count, **kwds) - def posts_all(self, tag="", **kwds): + def posts_all(self, tag="", start=None, results=None, fromdt=None, + todt=None, meta=True, hashes=False, **kwds): """Returns all posts. Please use sparingly. Call the `posts_update` method to see if you need to fetch this at all. :: @@ -634,13 +783,34 @@ class DeliciousAPI: <posts tag="..." user="..." update="CCYY-MM-DDThh:mm:ssZ"> <post ...> - &tag (optional). - Filter by this tag. + &tag + (optional) Filter by this tag. + &start={#} + (optional) Start returning posts this many results into the set. + &results={#} + (optional) Return this many results. + &fromdt={CCYY-MM-DDThh:mm:ssZ} + (optional) Filter for posts on this date or later + &todt={CCYY-MM-DDThh:mm:ssZ} + (optional) Filter for posts on this date or earlier + &meta=yes + (optional) Include change detection signatures on each item in a + 'meta' attribute. Clients wishing to maintain a synchronized local + store of bookmarks should retain the value of this attribute - its + value will change when any significant field of the bookmark + changes. + &hashes + (optional, exclusive) Do not fetch post details but a posts + manifest with url- and meta-hashes. Other options do not apply. """ - return self.request("posts/all", tag=tag, **kwds) + if hashes: + return self.request("posts/all", hashes=hashes, **kwds) + else: + return self.request("posts/all", tag=tag, fromdt=fromdt, todt=todt, + start=start, results=results, meta=meta, **kwds) def posts_add(self, url, description, extended="", tags="", dt="", - replace="no", shared="yes", **kwds): + replace=False, shared=True, **kwds): """Add a post to del.icio.us. Returns a `result` message or raises an ``DeliciousError``. See ``self.request()``. @@ -654,10 +824,12 @@ class DeliciousAPI: tags for the item (space delimited). &dt (optional) datestamp of the item (format "CCYY-MM-DDThh:mm:ssZ"). - - Requires a LITERAL "T" and "Z" like in ISO8601 at http://www.cl.cam.ac.uk/~mgk25/iso-time.html for example: "1984-09-01T14:21:31Z" - &replace=no (optional) - don't replace post if given url has already been posted. - &shared=no (optional) - make the item private + Requires a LITERAL "T" and "Z" like in ISO8601 at + http://www.cl.cam.ac.uk/~mgk25/iso-time.html for example: + "1984-09-01T14:21:31Z" + &replace=no (optional) - don't replace post if given url has already + been posted. + &shared=yes (optional) - wether the item is public. """ return self.request("posts/add", url=url, description=description, extended=extended, tags=tags, dt=dt, @@ -690,7 +862,7 @@ class DeliciousAPI: &bundle (required) the bundle name. &tags (required) - list of tags (space seperated). + list of tags. """ if type(tags)==list: tags = " ".join(tags) @@ -710,19 +882,22 @@ class DeliciousAPI: # Lookup table for del.icio.us url-path to DeliciousAPI method. paths = { - 'tags/get': tags_get, - 'tags/rename': tags_rename, - 'posts/update': posts_update, - 'posts/dates': posts_dates, - 'posts/get': posts_get, - 'posts/recent': posts_recent, - 'posts/all': posts_all, - 'posts/add': posts_add, - 'posts/delete': posts_delete, - 'tags/bundles/all': bundles_all, - 'tags/bundles/set': bundles_set, - 'tags/bundles/delete': bundles_delete, + 'tags/get': 'tags_get', + 'tags/delete': 'tags_delete', + 'tags/rename': 'tags_rename', + 'posts/update': 'posts_update', + 'posts/dates': 'posts_dates', + 'posts/get': 'posts_get', + 'posts/recent': 'posts_recent', + 'posts/all': 'posts_all', + 'posts/add': 'posts_add', + 'posts/delete': 'posts_delete', + 'tags/bundles/all': 'bundles_all', + 'tags/bundles/set': 'bundles_set', + 'tags/bundles/delete': 'bundles_delete', } + def get_method(self, path): + return getattr(self, self.paths[path]) def get_url(self, url): """Return the del.icio.us url at which the HTML page with posts for @@ -730,41 +905,55 @@ class DeliciousAPI: """ return "http://del.icio.us/url/?url=%s" % (url,) + def __repr__(self): + return "DeliciousAPI(%s)" % self.user + ### Convenience functions on this package def apiNew(user, passwd): - """creates a new DeliciousAPI object. - requires user(name) and passwd - """ + "Creates a new DeliciousAPI object, requires user(name) and passwd." return DeliciousAPI(user=user, passwd=passwd) -def add(user, passwd, url, description, tags="", extended="", dt="", replace="no"): - return apiNew(user, passwd).posts_add(url=url, description=description, extended=extended, tags=tags, dt=dt, replace=replace) +def add(user, passwd, url, description, tags="", extended="", dt=None, + replace=False): + apiNew(user, passwd).posts_add(url=url, description=description, + extended=extended, tags=tags, dt=dt, replace=replace) -def get(user, passwd, tag="", dt="", count = 0): - posts = apiNew(user, passwd).posts_get(tag=tag,dt=dt) - if count != 0: posts = posts[0:count] +def get(user, passwd, tag="", dt=None, count=0, hashes=[]): + "Returns a list of posts for the user" + posts = apiNew(user, passwd).posts_get( + tag=tag, dt=dt, hashes=hashes)['posts'] + if count: posts = posts[:count] return posts -def get_all(user, passwd, tag=""): - return apiNew(user, passwd).posts_all(tag=tag) +def get_update(user, passwd): + "Returns the last update time for the user." + return apiNew(user, passwd).posts_update()['update']['time'] + +def get_all(user, passwd, tag="", start=0, results=100, fromdt=None, + todt=None): + "Returns a list with all posts. Please use sparingly. See `get_updated`" + return apiNew(user, passwd).posts_all(tag=tag, start=start, + results=results, fromdt=fromdt, todt=todt, meta=True)['posts'] + +def get_tags(user, passwd): + "Returns a list with all tags for user." + return apiNew(user=user, passwd=passwd).tags_get()['tags'] def delete(user, passwd, url): - return apiNew(user, passwd).posts_delete(url=url) + "Delete the URL from the del.icio.us account." + apiNew(user, passwd).posts_delete(url=url) def rename_tag(user, passwd, oldtag, newtag): - return apiNew(user=user, passwd=passwd).tags_rename(old=oldtag, new=newtag) + "Rename the tag for the del.icio.us account." + apiNew(user=user, passwd=passwd).tags_rename(old=oldtag, new=newtag) -def get_tags(user, passwd): - return apiNew(user=user, passwd=passwd).tags_get() +### RSS functions -### RSS functions bvb: still working...? def getrss(tag="", popular=0, url='', user=""): - """get posts from del.icio.us via parsing RSS (bvb:or HTML) - - todo: not tested + """Get posts from del.icio.us via parsing RSS. tag (opt) sort by tag popular (opt) look for the popular stuff @@ -774,44 +963,83 @@ def getrss(tag="", popular=0, url='', user=""): return dlcs_rss_request(tag=tag, popular=popular, user=user, url=url) def get_userposts(user): - return getrss(user = user) + "parse RSS for user" + return getrss(user=user) def get_tagposts(tag): - return getrss(tag = tag) + "parse RSS for tag" + return getrss(tag=tag) def get_urlposts(url): - return getrss(url = url) + "parse RSS for URL" + return getrss(url=url) + +def get_popular(tag=""): + "parse RSS for popular URLS for tag" + return getrss(tag=tag, popular=1) -def get_popular(tag = ""): - return getrss(tag = tag, popular = 1) +### JSON feeds +# TODO: untested -### TODO: implement JSON fetching -def json_posts(user, count=15): - """http://del.icio.us/feeds/json/mpe - http://del.icio.us/feeds/json/mpe/art+history - count=### the number of posts you want to get (default is 15, maximum is 100) - raw a raw JSON object is returned, instead of an object named Delicious.posts +def json_posts(user, count=15, tag=None, raw=True): """ + user + count=### the number of posts you want to get (default is 15, maximum + is 100) + raw a raw JSON object is returned, instead of an object named + Delicious.posts + """ + url = "http://del.icio.us/feeds/json/" + \ + dlcs_encode_params({0:user})[0] + if tag: url += '/'+dlcs_encode_params({0:tag})[0] + + return dlcs_feed(url, count=count, raw=raw) -def json_tags(user, atleast, count, sort='alpha'): - """http://del.icio.us/feeds/json/tags/mpe - atleast=### include only tags for which there are at least ### number of posts - count=### include ### tags, counting down from the top - sort={alpha|count} construct the object with tags in alphabetic order (alpha), or by count of posts (count) - callback=NAME wrap the object definition in a function call NAME(...), thus invoking that function when the feed is executed - raw a pure JSON object is returned, instead of code that will construct an object named Delicious.tags + +def json_tags(user, atleast, count, sort='alpha', raw=True, callback=None): + """ + user + atleast=### include only tags for which there are at least ### + number of posts. + count=### include ### tags, counting down from the top. + sort={alpha|count} construct the object with tags in alphabetic order + (alpha), or by count of posts (count). + callback=NAME wrap the object definition in a function call NAME(...), + thus invoking that function when the feed is executed. + raw a pure JSON object is returned, instead of code that + will construct an object named Delicious.tags. """ + url = 'http://del.icio.us/feeds/json/tags/' + \ + dlcs_encode_params({0:user})[0] + return dlcs_feed(url, atleast=atleast, count=count, sort=sort, raw=raw, + callback=callback) -def json_network(user): - """http://del.icio.us/feeds/json/network/mpe + +def json_network(user, raw=True, callback=None): + """ callback=NAME wrap the object definition in a function call NAME(...) - ?raw a raw JSON object is returned, instead of an object named Delicious.posts + ?raw a raw JSON object is returned, instead of an object named + Delicious.posts """ + url = 'http://del.icio.us/feeds/json/network/' + \ + dlcs_encode_params({0:user})[0] + return dlcs_feed(url, raw=raw, callback=callback) -def json_fans(user): - """http://del.icio.us/feeds/json/fans/mpe + +def json_fans(user, raw=True, callback=None): + """ callback=NAME wrap the object definition in a function call NAME(...) - ?raw a pure JSON object is returned, instead of an object named Delicious. + ?raw a pure JSON object is returned, instead of an object named + Delicious. """ + url = 'http://del.icio.us/feeds/json/fans/' + \ + dlcs_encode_params({0:user})[0] + return dlcs_feed(url, raw=raw, callback=callback) + + +### delicious V2 feeds + +def getfeed(name, **params): + return dlcs_feed(name, **params) diff --git a/lib/utils/strutils.py b/lib/utils/strutils.py new file mode 100644 index 0000000..368d3d8 --- /dev/null +++ b/lib/utils/strutils.py @@ -0,0 +1,50 @@ + +# +# String/unicode conversion utils. +# + +def safestr(s): + """ + Safely corerce *anything* to a string. If the object can't be str'd, an + empty string will be returned. + + You can (and I do) use this for really crappy unicode handling, but it's + a bit like killing a mosquito with a bazooka. + """ + if s is None: + return "" + if isinstance(s, unicode): + return s.encode('ascii', 'xmlcharrefreplace') + else: + try: + return str(s) + except: + return "" + +def safeint(s): + """Like safestr(), but always returns an int. Returns 0 on failure.""" + try: + return int(safestr(s)) + except ValueError: + return 0 + + +def convertentity(m): + import htmlentitydefs + """Convert a HTML entity into normal string (ISO-8859-1)""" + if m.group(1)=='#': + try: + return chr(int(m.group(2))) + except ValueError: + return '&#%s;' % m.group(2) + try: + return htmlentitydefs.entitydefs[m.group(2)] + except KeyError: + return '&%s;' % m.group(2) + +def unquotehtml(s): + import re + """Convert a HTML quoted string into normal string (ISO-8859-1). + + Works with &#XX; and with   > etc.""" + return re.sub(r'&(#?)(.+?);',convertentity,s) diff --git a/lib/view_wrapper.py b/lib/view_wrapper.py deleted file mode 100644 index 0e1e492..0000000 --- a/lib/view_wrapper.py +++ /dev/null @@ -1,10 +0,0 @@ -from django.shortcuts import render_to_response -from django.template.context import RequestContext - -def luxagraf_render(request, *args, **kwargs): - """ - Replacement for render_to_response that uses RequestContext and sets an - extra template variable, TEMPLATE_NAME. - """ - kwargs['context_instance'] = RequestContext(request) - return render_to_response(*args, **kwargs) \ No newline at end of file diff --git a/templates/includes/map_entry_list.html b/templates/includes/map_entry_list.html index f011f2a..b3fe171 100644 --- a/templates/includes/map_entry_list.html +++ b/templates/includes/map_entry_list.html @@ -55,11 +55,11 @@ // Add a marker for each project - point_los_angeles = JLngLat(-118.23588250421922, 34.055823874326265); + point_los_angeles = JLngLat(-118.23626874231368, 34.05537054642469); markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_los_angeles = new GMarker(point_los_angeles, markerOptions); map.addOverlay(marker_los_angeles); - marker_los_angeles.info_window_content = '<div class="infowin"><h4>Los Angeles, I'm Yours<\/h4><span class="date blok">May 17, 2010 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbnail/2010/launiontickets.jpg" height="100" alt="Los Angeles, I'm Yours" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Los Angeles is all about the car. Shiny, air\x2Dconditioned comfort, gliding you soundlessly from one place to another without the need to interact with anything in between. But I have discovered that if you abandon the car for the subway and your own two feet, the illusion that L.A. is just a model train set world \x26mdash\x3B tiny, plastic and devoid of any ground beneath the ground \x26mdash\x3B fades and you find yourself, for a time, in a real city. <a href="/2010/may/17/los-angeles-im-yours/">Read it »<\/a><\/p><\/div>' + marker_los_angeles.info_window_content = '<div class="infowin"><h4>Los Angeles, I'm Yours<\/h4><span class="date blok">May 14, 2010 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbnail/2010/launiontickets.jpg" height="100" alt="Los Angeles, I'm Yours" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Los Angeles is all about the car. Shiny, air\u002Dconditioned comfort, gliding you soundlessly from one place to another without the need to interact with anything in between. But I have discovered that if you abandon the car for the subway and your own two feet, the illusion that L.A. is just a model train set world \u0026mdash\u003B tiny, plastic and devoid of any ground beneath the ground \u0026mdash\u003B fades and you find yourself, for a time, in a real city. <a href="/2010/may/14/los-angeles-im-yours/">Read it »<\/a><\/p><\/div>' marker_los_angeles.bindInfoWindowHtml(marker_los_angeles.info_window_content, {maxWidth:400}); GEvent.addListener(marker_los_angeles, "click", function() { map.panTo(point_los_angeles, 2); @@ -70,7 +70,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_therell_be = new GMarker(point_therell_be, markerOptions); map.addOverlay(marker_therell_be); - marker_therell_be.info_window_content = '<div class="infowin"><h4>(There'll Be) Peace in the Valley<\/h4><span class="date blok">April 24, 2010 (Death Valley, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbnail/2010/deathvalley.jpg" height="100" alt="(There'll Be) Peace in the Valley" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Sometimes you ignore the places close to home because, well, there\x27s always next weekend. Which is why I never made it Death Valley in the twenty\x2Dfive years I lived in California. It took being all the way across the country to get me out to Death Valley. Which might explain why I actually got up before dawn just to watch the sunrise at Zabriskie Point. <a href="/2010/apr/24/death-valley/">Read it »<\/a><\/p><\/div>' + marker_therell_be.info_window_content = '<div class="infowin"><h4>(There'll Be) Peace in the Valley<\/h4><span class="date blok">April 24, 2010 (Death Valley, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbnail/2010/deathvalley.jpg" height="100" alt="(There'll Be) Peace in the Valley" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Sometimes you ignore the places close to home because, well, there\u0027s always next weekend. Which is why I never made it Death Valley in the twenty\u002Dfive years I lived in California. It took being all the way across the country to get me out to Death Valley. Which might explain why I actually got up before dawn just to watch the sunrise at Zabriskie Point. <a href="/2010/apr/24/death-valley/">Read it »<\/a><\/p><\/div>' marker_therell_be.bindInfoWindowHtml(marker_therell_be.info_window_content, {maxWidth:400}); GEvent.addListener(marker_therell_be, "click", function() { map.panTo(point_therell_be, 2); @@ -81,7 +81,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_so_far = new GMarker(point_so_far, markerOptions); map.addOverlay(marker_so_far); - marker_so_far.info_window_content = '<div class="infowin"><h4>So Far, I Have Not Found The Science<\/h4><span class="date blok">March 13, 2010 (Okefenokee Swamp, Georgia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2010/okeefenokee.jpg" height="100" alt="So Far, I Have Not Found The Science" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>A canoe trip through the Okefenokee Swamp down in the southern most corner of Georgia. Paddling the strange reddish and incredibly still waters. Begging alligators, aching muscles and the kindly folks of Stintson\x27s Barbecue all getting their due. <a href="/2010/mar/13/so-far-i-have-not-found-science/">Read it »<\/a><\/p><\/div>' + marker_so_far.info_window_content = '<div class="infowin"><h4>So Far, I Have Not Found The Science<\/h4><span class="date blok">March 13, 2010 (Okefenokee Swamp, Georgia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2010/okeefenokee.jpg" height="100" alt="So Far, I Have Not Found The Science" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>A canoe trip through the Okefenokee Swamp down in the southern most corner of Georgia. Paddling the strange reddish and incredibly still waters. Begging alligators, aching muscles and the kindly folks of Stintson\u0027s Barbecue all getting their due. <a href="/2010/mar/13/so-far-i-have-not-found-science/">Read it »<\/a><\/p><\/div>' marker_so_far.bindInfoWindowHtml(marker_so_far.info_window_content, {maxWidth:400}); GEvent.addListener(marker_so_far, "click", function() { map.panTo(point_so_far, 2); @@ -92,7 +92,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_how_to = new GMarker(point_how_to, markerOptions); map.addOverlay(marker_how_to); - marker_how_to.info_window_content = '<div class="infowin"><h4>How to Get Off Your Butt and Travel the World<\/h4><span class="date blok">May 3, 2009 (Athens, Georgia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2010/traveltheworld.jpg" height="100" alt="How to Get Off Your Butt and Travel the World" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>How do you make the leap from cubicle daydreams to life on to the road? You want to travel the world, but, like me, you have a million excuses stopping you. How do overcome the inertia that keeps you trapped in a life that isn\x27t what you want it to be? Here\x27s a few practical tips and how tos designed to motivate you to get off your butt and travel the world. <a href="/2009/may/03/how-to-get-your-butt-and-travel-world/">Read it »<\/a><\/p><\/div>' + marker_how_to.info_window_content = '<div class="infowin"><h4>How to Get Off Your Butt and Travel the World<\/h4><span class="date blok">May 3, 2009 (Athens, Georgia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2010/traveltheworld.jpg" height="100" alt="How to Get Off Your Butt and Travel the World" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>How do you make the leap from cubicle daydreams to life on to the road? You want to travel the world, but, like me, you have a million excuses stopping you. How do overcome the inertia that keeps you trapped in a life that isn\u0027t what you want it to be? Here\u0027s a few practical tips and how tos designed to motivate you to get off your butt and travel the world. <a href="/2009/may/03/how-to-get-your-butt-and-travel-world/">Read it »<\/a><\/p><\/div>' marker_how_to.bindInfoWindowHtml(marker_how_to.info_window_content, {maxWidth:400}); GEvent.addListener(marker_how_to, "click", function() { map.panTo(point_how_to, 2); @@ -103,7 +103,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_no_strangers = new GMarker(point_no_strangers, markerOptions); map.addOverlay(marker_no_strangers); - marker_no_strangers.info_window_content = '<div class="infowin"><h4>No Strangers on a Train<\/h4><span class="date blok">April 13, 2009 (Athens, Georgia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2009/strangersonatrain.jpg" height="100" alt="No Strangers on a Train" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We mythologize trains because they harken back to an age of community travel, a real, tangible community of travelers, not just backpackers, but people from all walks of life, people traveling near and far together in a shared space that isn\x27t locked down like an airplane and isn\x27t isolated like a car\x3B it\x27s a shared travel experience and there are precious few of those left in our world. <a href="/2009/apr/13/strangers-on-a-train/">Read it »<\/a><\/p><\/div>' + marker_no_strangers.info_window_content = '<div class="infowin"><h4>No Strangers on a Train<\/h4><span class="date blok">April 13, 2009 (Athens, Georgia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2009/strangersonatrain.jpg" height="100" alt="No Strangers on a Train" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We mythologize trains because they harken back to an age of community travel, a real, tangible community of travelers, not just backpackers, but people from all walks of life, people traveling near and far together in a shared space that isn\u0027t locked down like an airplane and isn\u0027t isolated like a car\u003B it\u0027s a shared travel experience and there are precious few of those left in our world. <a href="/2009/apr/13/strangers-on-a-train/">Read it »<\/a><\/p><\/div>' marker_no_strangers.bindInfoWindowHtml(marker_no_strangers.info_window_content, {maxWidth:400}); GEvent.addListener(marker_no_strangers, "click", function() { map.panTo(point_no_strangers, 2); @@ -114,7 +114,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_leonardo_da = new GMarker(point_leonardo_da, markerOptions); map.addOverlay(marker_leonardo_da); - marker_leonardo_da.info_window_content = '<div class="infowin"><h4>Leonardo Da Vinci and the Codex on Bunnies<\/h4><span class="date blok">December 9, 2008 (Birmingham, Alabama)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2009/codexofbunnies.jpg" height="100" alt="Leonardo Da Vinci and the Codex on Bunnies" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>A few pages from Leonardo Da Vinci\x27s notebooks make a rare trip outside Italy, to Birmingham, AL, of all places. But the Birmingham Museum of Art is home to far more alarming works of art, works which depict the eventual, inevitable, bunny takeover, after which all the elements of our reality will be replaced by bunnies. Seriously. You heard it here first. <a href="/2008/dec/09/leonardo-da-vinci-and-codex-bunnies/">Read it »<\/a><\/p><\/div>' + marker_leonardo_da.info_window_content = '<div class="infowin"><h4>Leonardo Da Vinci and the Codex on Bunnies<\/h4><span class="date blok">December 9, 2008 (Birmingham, Alabama)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2009/codexofbunnies.jpg" height="100" alt="Leonardo Da Vinci and the Codex on Bunnies" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>A few pages from Leonardo Da Vinci\u0027s notebooks make a rare trip outside Italy, to Birmingham, AL, of all places. But the Birmingham Museum of Art is home to far more alarming works of art, works which depict the eventual, inevitable, bunny takeover, after which all the elements of our reality will be replaced by bunnies. Seriously. You heard it here first. <a href="/2008/dec/09/leonardo-da-vinci-and-codex-bunnies/">Read it »<\/a><\/p><\/div>' marker_leonardo_da.bindInfoWindowHtml(marker_leonardo_da.info_window_content, {maxWidth:400}); GEvent.addListener(marker_leonardo_da, "click", function() { map.panTo(point_leonardo_da, 2); @@ -125,7 +125,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_elkmont_and = new GMarker(point_elkmont_and, markerOptions); map.addOverlay(marker_elkmont_and); - marker_elkmont_and.info_window_content = '<div class="infowin"><h4>Elkmont and the Great Smoky Mountains<\/h4><span class="date blok">October 31, 2008 (Great Smoky Mountains, Tennessee)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/reflectedtrees.jpg" height="100" alt="Elkmont and the Great Smoky Mountains" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Pigeon Forge is Myrtle Beach in the mountains. Redneck weddings cascade straight out of the chapel and into the mini golf reception area. Pigeon Forge is everything that\x27s wrong with America. But we aren\x27t here for Pigeon Forge, it just happens to have a free condo we\x27re staying in. We\x27re here for the mountains. Smoky Mountain National Park is just a few miles up the road. <a href="/2008/oct/31/elkmont-and-great-smoky-mountains/">Read it »<\/a><\/p><\/div>' + marker_elkmont_and.info_window_content = '<div class="infowin"><h4>Elkmont and the Great Smoky Mountains<\/h4><span class="date blok">October 31, 2008 (Great Smoky Mountains, Tennessee)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/reflectedtrees.jpg" height="100" alt="Elkmont and the Great Smoky Mountains" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Pigeon Forge is Myrtle Beach in the mountains. Redneck weddings cascade straight out of the chapel and into the mini golf reception area. Pigeon Forge is everything that\u0027s wrong with America. But we aren\u0027t here for Pigeon Forge, it just happens to have a free condo we\u0027re staying in. We\u0027re here for the mountains. Smoky Mountain National Park is just a few miles up the road. <a href="/2008/oct/31/elkmont-and-great-smoky-mountains/">Read it »<\/a><\/p><\/div>' marker_elkmont_and.bindInfoWindowHtml(marker_elkmont_and.info_window_content, {maxWidth:400}); GEvent.addListener(marker_elkmont_and, "click", function() { map.panTo(point_elkmont_and, 2); @@ -136,7 +136,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_rope_swings = new GMarker(point_rope_swings, markerOptions); map.addOverlay(marker_rope_swings); - marker_rope_swings.info_window_content = '<div class="infowin"><h4>Rope Swings and River Floats<\/h4><span class="date blok">July 27, 2008 (Mountain Cabin, Georgia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/chestateeriver.jpg" height="100" alt="Rope Swings and River Floats" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Two weekends ago we went up to the mountains, just outside of Dahlonega GA, and floated the Chestatee River using inner tubes, various pool toys and one super\x2Dcool inflatable seahorse. Unfortunately, proving one of my travel mottos \x2D\x2D you can never go back \x2D\x2D a return trip proved disastrous. <a href="/2008/jul/27/rope-swings-and-river-floats/">Read it »<\/a><\/p><\/div>' + marker_rope_swings.info_window_content = '<div class="infowin"><h4>Rope Swings and River Floats<\/h4><span class="date blok">July 27, 2008 (Mountain Cabin, Georgia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/chestateeriver.jpg" height="100" alt="Rope Swings and River Floats" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Two weekends ago we went up to the mountains, just outside of Dahlonega GA, and floated the Chestatee River using inner tubes, various pool toys and one super\u002Dcool inflatable seahorse. Unfortunately, proving one of my travel mottos \u002D\u002D you can never go back \u002D\u002D a return trip proved disastrous. <a href="/2008/jul/27/rope-swings-and-river-floats/">Read it »<\/a><\/p><\/div>' marker_rope_swings.bindInfoWindowHtml(marker_rope_swings.info_window_content, {maxWidth:400}); GEvent.addListener(marker_rope_swings, "click", function() { map.panTo(point_rope_swings, 2); @@ -147,7 +147,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_our_days = new GMarker(point_our_days, markerOptions); map.addOverlay(marker_our_days); - marker_our_days.info_window_content = '<div class="infowin"><h4>Our Days Are Becoming Nights<\/h4><span class="date blok">July 6, 2008 (León, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/daysnights.jpg" height="100" alt="Our Days Are Becoming Nights" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>A short thought on the eve of our departure from Nicaragua: Everywhere I go I think, I should live here... I should be able to not just visit places, but in habit them. Of course that isn\x27t possible, which is too bad. <a href="/2008/jul/06/our-days-are-becoming-nights/">Read it »<\/a><\/p><\/div>' + marker_our_days.info_window_content = '<div class="infowin"><h4>Our Days Are Becoming Nights<\/h4><span class="date blok">July 6, 2008 (León, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/daysnights.jpg" height="100" alt="Our Days Are Becoming Nights" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>A short thought on the eve of our departure from Nicaragua: Everywhere I go I think, I should live here... I should be able to not just visit places, but in habit them. Of course that isn\u0027t possible, which is too bad. <a href="/2008/jul/06/our-days-are-becoming-nights/">Read it »<\/a><\/p><\/div>' marker_our_days.bindInfoWindowHtml(marker_our_days.info_window_content, {maxWidth:400}); GEvent.addListener(marker_our_days, "click", function() { map.panTo(point_our_days, 2); @@ -158,7 +158,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_tiny_cities = new GMarker(point_tiny_cities, markerOptions); map.addOverlay(marker_tiny_cities); - marker_tiny_cities.info_window_content = '<div class="infowin"><h4>Tiny Cities Made of Ash<\/h4><span class="date blok">July 3, 2008 (León, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/citiesmadeofash.jpg" height="100" alt="Tiny Cities Made of Ash" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The church bells of Le\x26oacute\x3Bn have become a constant cacophony, not the rhythmic ringing out of the hours or tolling from Mass that the human mind seems to find pleasant, but the atonal banging that only appeals to the young and dumb. But Francisco is entirely unperturbed\x3B He\x27s too fascinated with the tattoo on Corrinne\x27s shoulder to bother with what slowly just becomes yet another sound echoing through Le\x26oacute\x3Bn. <a href="/2008/jul/03/tiny-cities-made-ash/">Read it »<\/a><\/p><\/div>' + marker_tiny_cities.info_window_content = '<div class="infowin"><h4>Tiny Cities Made of Ash<\/h4><span class="date blok">July 3, 2008 (León, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/citiesmadeofash.jpg" height="100" alt="Tiny Cities Made of Ash" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The church bells of Le\u0026oacute\u003Bn have become a constant cacophony, not the rhythmic ringing out of the hours or tolling from Mass that the human mind seems to find pleasant, but the atonal banging that only appeals to the young and dumb. But Francisco is entirely unperturbed\u003B He\u0027s too fascinated with the tattoo on Corrinne\u0027s shoulder to bother with what slowly just becomes yet another sound echoing through Le\u0026oacute\u003Bn. <a href="/2008/jul/03/tiny-cities-made-ash/">Read it »<\/a><\/p><\/div>' marker_tiny_cities.bindInfoWindowHtml(marker_tiny_cities.info_window_content, {maxWidth:400}); GEvent.addListener(marker_tiny_cities, "click", function() { map.panTo(point_tiny_cities, 2); @@ -169,7 +169,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_you_cant = new GMarker(point_you_cant, markerOptions); map.addOverlay(marker_you_cant); - marker_you_cant.info_window_content = '<div class="infowin"><h4>You Can't Go Home Again<\/h4><span class="date blok">June 30, 2008 (Little Corn Island, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/nohomeagain.jpg" height="100" alt="You Can't Go Home Again" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The first time we came to Little Corn Island it was April, the tail end of the dry season. It rained once or twice, but never for more than five minutes and always followed by more sunshine. This time it\x27s the end of June, just well into the wet season, and the island is an entirely different place. <a href="/2008/jun/30/you-cant-go-home-again/">Read it »<\/a><\/p><\/div>' + marker_you_cant.info_window_content = '<div class="infowin"><h4>You Can't Go Home Again<\/h4><span class="date blok">June 30, 2008 (Little Corn Island, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/nohomeagain.jpg" height="100" alt="You Can't Go Home Again" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The first time we came to Little Corn Island it was April, the tail end of the dry season. It rained once or twice, but never for more than five minutes and always followed by more sunshine. This time it\u0027s the end of June, just well into the wet season, and the island is an entirely different place. <a href="/2008/jun/30/you-cant-go-home-again/">Read it »<\/a><\/p><\/div>' marker_you_cant.bindInfoWindowHtml(marker_you_cant.info_window_content, {maxWidth:400}); GEvent.addListener(marker_you_cant, "click", function() { map.panTo(point_you_cant, 2); @@ -180,7 +180,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_returning_again = new GMarker(point_returning_again, markerOptions); map.addOverlay(marker_returning_again); - marker_returning_again.info_window_content = '<div class="infowin"><h4>Returning Again &mdash; Back on Little Corn Island<\/h4><span class="date blok">June 26, 2008 (Little Corn Island, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/littlecornagain.jpg" height="100" alt="Returning Again &mdash; Back on Little Corn Island" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/> Generally speaking, the world seems so huge and so full of amazing destinations that repeating one never struck me as a judicious use of my short allotment of time. But for Little Corn Island I\x27m willing to make an exception and of course, the universe being what it is, our second trip to Little Corn Island has been unpredictable and entirely new. <a href="/2008/jun/26/returning-again-back-little-corn-island/">Read it »<\/a><\/p><\/div>' + marker_returning_again.info_window_content = '<div class="infowin"><h4>Returning Again &mdash; Back on Little Corn Island<\/h4><span class="date blok">June 26, 2008 (Little Corn Island, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/littlecornagain.jpg" height="100" alt="Returning Again &mdash; Back on Little Corn Island" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/> Generally speaking, the world seems so huge and so full of amazing destinations that repeating one never struck me as a judicious use of my short allotment of time. But for Little Corn Island I\u0027m willing to make an exception and of course, the universe being what it is, our second trip to Little Corn Island has been unpredictable and entirely new. <a href="/2008/jun/26/returning-again-back-little-corn-island/">Read it »<\/a><\/p><\/div>' marker_returning_again.bindInfoWindowHtml(marker_returning_again.info_window_content, {maxWidth:400}); GEvent.addListener(marker_returning_again, "click", function() { map.panTo(point_returning_again, 2); @@ -191,7 +191,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_in_love = new GMarker(point_in_love, markerOptions); map.addOverlay(marker_in_love); - marker_in_love.info_window_content = '<div class="infowin"><h4>In Love With a View: Vagabonds, Responsibilty and Living Well<\/h4><span class="date blok">June 7, 2008 (Athens, Georgia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/wrong.jpg" height="100" alt="In Love With a View: Vagabonds, Responsibilty and Living Well" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Why all the vitriol about a seemingly innocuous concept \x2D\x2D that traveling doesn\x27t have to cost a lot of money, isn\x27t all that difficult and hey, you can even go right now? People like us, who feel tied down by responsibility, find the suggestion that we actually aren\x27t tied down patronizing and yes, elitist. <a href="/2008/jun/07/love-with-a-view-vagabonds-responsibilty-living-we/">Read it »<\/a><\/p><\/div>' + marker_in_love.info_window_content = '<div class="infowin"><h4>In Love With a View: Vagabonds, Responsibilty and Living Well<\/h4><span class="date blok">June 7, 2008 (Athens, Georgia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/wrong.jpg" height="100" alt="In Love With a View: Vagabonds, Responsibilty and Living Well" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Why all the vitriol about a seemingly innocuous concept \u002D\u002D that traveling doesn\u0027t have to cost a lot of money, isn\u0027t all that difficult and hey, you can even go right now? People like us, who feel tied down by responsibility, find the suggestion that we actually aren\u0027t tied down patronizing and yes, elitist. <a href="/2008/jun/07/love-with-a-view-vagabonds-responsibilty-living-we/">Read it »<\/a><\/p><\/div>' marker_in_love.bindInfoWindowHtml(marker_in_love.info_window_content, {maxWidth:400}); GEvent.addListener(marker_in_love, "click", function() { map.panTo(point_in_love, 2); @@ -202,7 +202,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_little_island = new GMarker(point_little_island, markerOptions); map.addOverlay(marker_little_island); - marker_little_island.info_window_content = '<div class="infowin"><h4>Little Island in the Sun<\/h4><span class="date blok">April 5, 2008 (Little Corn Island, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/coconutsun.jpg" height="100" alt="Little Island in the Sun" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We arrived on Little Corn Island around sundown and met Ali, whom I at first took to be a tout, but he showed us the way to our guesthouse and, after settling in and getting a feel for the island, I realized that Ali, wasn\x27t a tout, he was just a really nice guy who enjoyed doing favors for tourists, just beware the Yoni beverage he offers. <a href="/2008/apr/05/little-island-sun/">Read it »<\/a><\/p><\/div>' + marker_little_island.info_window_content = '<div class="infowin"><h4>Little Island in the Sun<\/h4><span class="date blok">April 5, 2008 (Little Corn Island, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/coconutsun.jpg" height="100" alt="Little Island in the Sun" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We arrived on Little Corn Island around sundown and met Ali, whom I at first took to be a tout, but he showed us the way to our guesthouse and, after settling in and getting a feel for the island, I realized that Ali, wasn\u0027t a tout, he was just a really nice guy who enjoyed doing favors for tourists, just beware the Yoni beverage he offers. <a href="/2008/apr/05/little-island-sun/">Read it »<\/a><\/p><\/div>' marker_little_island.bindInfoWindowHtml(marker_little_island.info_window_content, {maxWidth:400}); GEvent.addListener(marker_little_island, "click", function() { map.panTo(point_little_island, 2); @@ -213,7 +213,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_return_to = new GMarker(point_return_to, markerOptions); map.addOverlay(marker_return_to); - marker_return_to.info_window_content = '<div class="infowin"><h4>Return to the Sea<\/h4><span class="date blok">April 2, 2008 (San Juan Del Sur, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/sanjuansunset.jpg" height="100" alt="Return to the Sea" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Southwestern Nicaragua is a very small strip of land with Lago Nicaragua to the east and the Pacific Ocean to the west. The main town in the area, Juan Del Sur, is nestled around a well protected harbor with a mediocre strip of sand. For the nice beaches you have to head up or down the coast to one of the many small inlets. <a href="/2008/apr/02/return-sea/">Read it »<\/a><\/p><\/div>' + marker_return_to.info_window_content = '<div class="infowin"><h4>Return to the Sea<\/h4><span class="date blok">April 2, 2008 (San Juan Del Sur, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/sanjuansunset.jpg" height="100" alt="Return to the Sea" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Southwestern Nicaragua is a very small strip of land with Lago Nicaragua to the east and the Pacific Ocean to the west. The main town in the area, Juan Del Sur, is nestled around a well protected harbor with a mediocre strip of sand. For the nice beaches you have to head up or down the coast to one of the many small inlets. <a href="/2008/apr/02/return-sea/">Read it »<\/a><\/p><\/div>' marker_return_to.bindInfoWindowHtml(marker_return_to.info_window_content, {maxWidth:400}); GEvent.addListener(marker_return_to, "click", function() { map.panTo(point_return_to, 2); @@ -224,7 +224,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_ring_the = new GMarker(point_ring_the, markerOptions); map.addOverlay(marker_ring_the); - marker_ring_the.info_window_content = '<div class="infowin"><h4>Ring The Bells<\/h4><span class="date blok">March 30, 2008 (Granada, Nicaragua)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/ringthbells.jpg" height="100" alt="Ring The Bells" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Church, which dates from the 1600s has the the narrowest, steepest, circular concrete staircase that I\x27ve ever encountered. It had a low railing and circled up four stories worth of precipitous dropoffs before you hit solid ground. From the top was a views of Granada\x27s endless sea of mottled pink, orange and brown hues \x2D\x2D terra cotta roof tiles stretching from the shores of Lago Nicaragua all the way back toward the hills. <a href="/2008/mar/30/ring-bells/">Read it »<\/a><\/p><\/div>' + marker_ring_the.info_window_content = '<div class="infowin"><h4>Ring The Bells<\/h4><span class="date blok">March 30, 2008 (Granada, Nicaragua)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/ringthbells.jpg" height="100" alt="Ring The Bells" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Church, which dates from the 1600s has the the narrowest, steepest, circular concrete staircase that I\u0027ve ever encountered. It had a low railing and circled up four stories worth of precipitous dropoffs before you hit solid ground. From the top was a views of Granada\u0027s endless sea of mottled pink, orange and brown hues \u002D\u002D terra cotta roof tiles stretching from the shores of Lago Nicaragua all the way back toward the hills. <a href="/2008/mar/30/ring-bells/">Read it »<\/a><\/p><\/div>' marker_ring_the.bindInfoWindowHtml(marker_ring_the.info_window_content, {maxWidth:400}); GEvent.addListener(marker_ring_the, "click", function() { map.panTo(point_ring_the, 2); @@ -235,7 +235,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_fall = new GMarker(point_fall, markerOptions); map.addOverlay(marker_fall); - marker_fall.info_window_content = '<div class="infowin"><h4>Fall<\/h4><span class="date blok">November 14, 2007 (Athens, Georgia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/fall.jpg" height="100" alt="Fall" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The trees are in full technicolor swing. The land is slowly dying, and not just because it\x27s Fall, we\x27re also in the middle of a prolonged drought and this year the leaves are opting for a James Dean\x2Dstyle, leave\x2Da\x2Dgood\x2Dlooking\x2Dcorpse exit. If you\x27re a leaf and you\x27ve got to go, do it with class. <a href="/2007/nov/14/fall/">Read it »<\/a><\/p><\/div>' + marker_fall.info_window_content = '<div class="infowin"><h4>Fall<\/h4><span class="date blok">November 14, 2007 (Athens, Georgia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/fall.jpg" height="100" alt="Fall" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The trees are in full technicolor swing. The land is slowly dying, and not just because it\u0027s Fall, we\u0027re also in the middle of a prolonged drought and this year the leaves are opting for a James Dean\u002Dstyle, leave\u002Da\u002Dgood\u002Dlooking\u002Dcorpse exit. If you\u0027re a leaf and you\u0027ve got to go, do it with class. <a href="/2007/nov/14/fall/">Read it »<\/a><\/p><\/div>' marker_fall.bindInfoWindowHtml(marker_fall.info_window_content, {maxWidth:400}); GEvent.addListener(marker_fall, "click", function() { map.panTo(point_fall, 2); @@ -246,7 +246,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_on_the = new GMarker(point_on_the, markerOptions); map.addOverlay(marker_on_the); - marker_on_the.info_window_content = '<div class="infowin"><h4>On The Other Ocean<\/h4><span class="date blok">July 23, 2007 (Catalina Island, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/sailing.jpg" height="100" alt="On The Other Ocean" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Consider what would happen if your house were tilted 30 degrees to the left, how this would complicate ordinary activities \x2D\x2D like say walking. Now throw in a bouncing motion that lifts the floor five or six feet up and down in a seesaw\x2Dlike motion on a perpendicular axis to the 30 degree tilt \x2D\x2D things become more like riding a seesaw that\x27s attached to a merry\x2Dgo\x2Dround which is missing a few bolts. That\x27s sailing. <a href="/2007/jul/23/other-ocean/">Read it »<\/a><\/p><\/div>' + marker_on_the.info_window_content = '<div class="infowin"><h4>On The Other Ocean<\/h4><span class="date blok">July 23, 2007 (Catalina Island, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/sailing.jpg" height="100" alt="On The Other Ocean" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Consider what would happen if your house were tilted 30 degrees to the left, how this would complicate ordinary activities \u002D\u002D like say walking. Now throw in a bouncing motion that lifts the floor five or six feet up and down in a seesaw\u002Dlike motion on a perpendicular axis to the 30 degree tilt \u002D\u002D things become more like riding a seesaw that\u0027s attached to a merry\u002Dgo\u002Dround which is missing a few bolts. That\u0027s sailing. <a href="/2007/jul/23/other-ocean/">Read it »<\/a><\/p><\/div>' marker_on_the.bindInfoWindowHtml(marker_on_the.info_window_content, {maxWidth:400}); GEvent.addListener(marker_on_the, "click", function() { map.panTo(point_on_the, 2); @@ -257,7 +257,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_being_there = new GMarker(point_being_there, markerOptions); map.addOverlay(marker_being_there); - marker_being_there.info_window_content = '<div class="infowin"><h4>Being There<\/h4><span class="date blok">June 17, 2007 (Myrtle Beach Airport, South Carolina)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/myrtlebeachcrap.jpg" height="100" alt="Being There" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Myrtle Beach does not exist. Nearly everything in Myrtle Beach is a paltry derivative of some original form. For instance, most of the country has golf courses, in Myrtle Beach there are endless rows of putt\x2Dputt courses, where most towns attempt to draw in big name musical acts for their tourist venues, Myrtle Beach is content with impersonators. <a href="/2007/jun/17/being-there/">Read it »<\/a><\/p><\/div>' + marker_being_there.info_window_content = '<div class="infowin"><h4>Being There<\/h4><span class="date blok">June 17, 2007 (Myrtle Beach Airport, South Carolina)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/myrtlebeachcrap.jpg" height="100" alt="Being There" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Myrtle Beach does not exist. Nearly everything in Myrtle Beach is a paltry derivative of some original form. For instance, most of the country has golf courses, in Myrtle Beach there are endless rows of putt\u002Dputt courses, where most towns attempt to draw in big name musical acts for their tourist venues, Myrtle Beach is content with impersonators. <a href="/2007/jun/17/being-there/">Read it »<\/a><\/p><\/div>' marker_being_there.bindInfoWindowHtml(marker_being_there.info_window_content, {maxWidth:400}); GEvent.addListener(marker_being_there, "click", function() { map.panTo(point_being_there, 2); @@ -268,7 +268,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_sailing_through = new GMarker(point_sailing_through, markerOptions); map.addOverlay(marker_sailing_through); - marker_sailing_through.info_window_content = '<div class="infowin"><h4>Sailing Through<\/h4><span class="date blok">June 14, 2007 (Charleston, South Carolina)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/charlestonships.jpg" height="100" alt="Sailing Through" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The rumors are true. I moved back to the south\x3B Athens GA to be exact. But I hate staying in one place for too long, so after a month or two in Athens I headed up to Charleston to visit a friend. The south is curious place. If you\x27ve never been here I couldn\x27t hope to explain it, but it\x27s not so much a place as an approach. A way of getting somewhere more than anywhere specific. Perhaps even a wrong turn. \x0D\x0A <a href="/2007/jun/14/sailing-through/">Read it »<\/a><\/p><\/div>' + marker_sailing_through.info_window_content = '<div class="infowin"><h4>Sailing Through<\/h4><span class="date blok">June 14, 2007 (Charleston, South Carolina)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/charlestonships.jpg" height="100" alt="Sailing Through" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The rumors are true. I moved back to the south\u003B Athens GA to be exact. But I hate staying in one place for too long, so after a month or two in Athens I headed up to Charleston to visit a friend. The south is curious place. If you\u0027ve never been here I couldn\u0027t hope to explain it, but it\u0027s not so much a place as an approach. A way of getting somewhere more than anywhere specific. Perhaps even a wrong turn. \u000D\u000A <a href="/2007/jun/14/sailing-through/">Read it »<\/a><\/p><\/div>' marker_sailing_through.bindInfoWindowHtml(marker_sailing_through.info_window_content, {maxWidth:400}); GEvent.addListener(marker_sailing_through, "click", function() { map.panTo(point_sailing_through, 2); @@ -279,7 +279,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_goodbye_to = new GMarker(point_goodbye_to, markerOptions); map.addOverlay(marker_goodbye_to); - marker_goodbye_to.info_window_content = '<div class="infowin"><h4>Goodbye to the Mother and the Cove<\/h4><span class="date blok">March 1, 2007 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/lacloud.jpg" height="100" alt="Goodbye to the Mother and the Cove" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\x27s strange how you can plan something, go through all the motions of making it happen without ever really understanding what you\x27re doing. I\x27ve been doing this for the better part of three years now. I realized recently that I have no real idea how I came to be here. \x0D\x0A <a href="/2007/mar/01/goodbye-mother-and-cove/">Read it »<\/a><\/p><\/div>' + marker_goodbye_to.info_window_content = '<div class="infowin"><h4>Goodbye to the Mother and the Cove<\/h4><span class="date blok">March 1, 2007 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/lacloud.jpg" height="100" alt="Goodbye to the Mother and the Cove" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\u0027s strange how you can plan something, go through all the motions of making it happen without ever really understanding what you\u0027re doing. I\u0027ve been doing this for the better part of three years now. I realized recently that I have no real idea how I came to be here. \u000D\u000A <a href="/2007/mar/01/goodbye-mother-and-cove/">Read it »<\/a><\/p><\/div>' marker_goodbye_to.bindInfoWindowHtml(marker_goodbye_to.info_window_content, {maxWidth:400}); GEvent.addListener(marker_goodbye_to, "click", function() { map.panTo(point_goodbye_to, 2); @@ -290,7 +290,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_everything_all = new GMarker(point_everything_all, markerOptions); map.addOverlay(marker_everything_all); - marker_everything_all.info_window_content = '<div class="infowin"><h4>Everything All The Time<\/h4><span class="date blok">February 3, 2007 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/end.jpg" height="100" alt="Everything All The Time" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I don\x27t know if I\x27m just overly paranoid but when I call up memories in the dark hours of the Beaujolais\x2Dsoaked pre\x2Ddawn, I see a collection of mildly amusing, occasionally painful series of embarrassments, misunderstandings and general wrong\x2Dplace, wrong\x2Dtime sort of moments. Which isn\x27t to imply that my life is a British sitcom, just that I\x27m not in a hurry to re\x2Dlive any of it. <a href="/2007/feb/03/everything-all-time/">Read it »<\/a><\/p><\/div>' + marker_everything_all.info_window_content = '<div class="infowin"><h4>Everything All The Time<\/h4><span class="date blok">February 3, 2007 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/end.jpg" height="100" alt="Everything All The Time" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I don\u0027t know if I\u0027m just overly paranoid but when I call up memories in the dark hours of the Beaujolais\u002Dsoaked pre\u002Ddawn, I see a collection of mildly amusing, occasionally painful series of embarrassments, misunderstandings and general wrong\u002Dplace, wrong\u002Dtime sort of moments. Which isn\u0027t to imply that my life is a British sitcom, just that I\u0027m not in a hurry to re\u002Dlive any of it. <a href="/2007/feb/03/everything-all-time/">Read it »<\/a><\/p><\/div>' marker_everything_all.bindInfoWindowHtml(marker_everything_all.info_window_content, {maxWidth:400}); GEvent.addListener(marker_everything_all, "click", function() { map.panTo(point_everything_all, 2); @@ -301,7 +301,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_sun = new GMarker(point_the_sun, markerOptions); map.addOverlay(marker_the_sun); - marker_the_sun.info_window_content = '<div class="infowin"><h4>The Sun Came Up With No Conclusions<\/h4><span class="date blok">January 11, 2007 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/illuminatus.jpg" height="100" alt="The Sun Came Up With No Conclusions" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>\x22And so it is that we, as men, do not exist until we do\x3B and then it is that we play with our world of existent things, and order and disorder them, and so it shall be that non\x2Dexistence shall take us back from existence and that nameless spirituality shall return to Void, like a tired child home from a very wild circus.\x22 \x2D\x2D Robert Anton Wilson and Kerry Thornley. Good luck and Godspeed Mr. Wilson. <a href="/2007/jan/11/sun-came-no-conclusions/">Read it »<\/a><\/p><\/div>' + marker_the_sun.info_window_content = '<div class="infowin"><h4>The Sun Came Up With No Conclusions<\/h4><span class="date blok">January 11, 2007 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/illuminatus.jpg" height="100" alt="The Sun Came Up With No Conclusions" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>\u0022And so it is that we, as men, do not exist until we do\u003B and then it is that we play with our world of existent things, and order and disorder them, and so it shall be that non\u002Dexistence shall take us back from existence and that nameless spirituality shall return to Void, like a tired child home from a very wild circus.\u0022 \u002D\u002D Robert Anton Wilson and Kerry Thornley. Good luck and Godspeed Mr. Wilson. <a href="/2007/jan/11/sun-came-no-conclusions/">Read it »<\/a><\/p><\/div>' marker_the_sun.bindInfoWindowHtml(marker_the_sun.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_sun, "click", function() { map.panTo(point_the_sun, 2); @@ -312,7 +312,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_give_it = new GMarker(point_give_it, markerOptions); map.addOverlay(marker_give_it); - marker_give_it.info_window_content = '<div class="infowin"><h4>Give It Up Or Turnit A Loose<\/h4><span class="date blok">December 25, 2006 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/jamesbrown.jpg" height="100" alt="Give It Up Or Turnit A Loose" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Traveling soul. Soul is not something out there or in you, it\x27s the place where you meet the out there\x3B something very similar to what I think James Brown meant \x26mdash\x3B a mixture of the secular and the spiritual, the profane and the sublime. <a href="/2006/dec/25/give-it-or-turnit-loose/">Read it »<\/a><\/p><\/div>' + marker_give_it.info_window_content = '<div class="infowin"><h4>Give It Up Or Turnit A Loose<\/h4><span class="date blok">December 25, 2006 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/jamesbrown.jpg" height="100" alt="Give It Up Or Turnit A Loose" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Traveling soul. Soul is not something out there or in you, it\u0027s the place where you meet the out there\u003B something very similar to what I think James Brown meant \u0026mdash\u003B a mixture of the secular and the spiritual, the profane and the sublime. <a href="/2006/dec/25/give-it-or-turnit-loose/">Read it »<\/a><\/p><\/div>' marker_give_it.bindInfoWindowHtml(marker_give_it.info_window_content, {maxWidth:400}); GEvent.addListener(marker_give_it, "click", function() { map.panTo(point_give_it, 2); @@ -323,7 +323,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_homeward = new GMarker(point_homeward, markerOptions); map.addOverlay(marker_homeward); - marker_homeward.info_window_content = '<div class="infowin"><h4>Homeward<\/h4><span class="date blok">June 9, 2006 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/trappedmoth.jpg" height="100" alt="Homeward" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>New York, New York. John F Kennedy airport 1 am date unknown, sleepy looking customs guard stamps a passport without hardly looking at, without even checking to see where I had been. A light drizzle is falling outside and the subways extension to the terminal never looked so good. What is it like to be home? I don\x27t know, I\x27ll tell you when I get there. <a href="/2006/jun/09/homeward/">Read it »<\/a><\/p><\/div>' + marker_homeward.info_window_content = '<div class="infowin"><h4>Homeward<\/h4><span class="date blok">June 9, 2006 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/trappedmoth.jpg" height="100" alt="Homeward" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>New York, New York. John F Kennedy airport 1 am date unknown, sleepy looking customs guard stamps a passport without hardly looking at, without even checking to see where I had been. A light drizzle is falling outside and the subways extension to the terminal never looked so good. What is it like to be home? I don\u0027t know, I\u0027ll tell you when I get there. <a href="/2006/jun/09/homeward/">Read it »<\/a><\/p><\/div>' marker_homeward.bindInfoWindowHtml(marker_homeward.info_window_content, {maxWidth:400}); GEvent.addListener(marker_homeward, "click", function() { map.panTo(point_homeward, 2); @@ -334,7 +334,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_cadenza = new GMarker(point_cadenza, markerOptions); map.addOverlay(marker_cadenza); - marker_cadenza.info_window_content = '<div class="infowin"><h4>Cadenza<\/h4><span class="date blok">June 6, 2006 (Paris, France)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/parisglow.jpg" height="100" alt="Cadenza" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Paris \x2D Outside it\x27s raining. Beads of water form on the window in front of me. The glow of the unseen sun is fading behind midnight blue clouds and darkening sky. An old man in a butcher apron selling oysters under an awning smokes a cigarette and watches the mothers and children walking home with bags of groceries. <a href="/2006/jun/06/cadenza/">Read it »<\/a><\/p><\/div>' + marker_cadenza.info_window_content = '<div class="infowin"><h4>Cadenza<\/h4><span class="date blok">June 6, 2006 (Paris, France)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/parisglow.jpg" height="100" alt="Cadenza" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Paris \u002D Outside it\u0027s raining. Beads of water form on the window in front of me. The glow of the unseen sun is fading behind midnight blue clouds and darkening sky. An old man in a butcher apron selling oysters under an awning smokes a cigarette and watches the mothers and children walking home with bags of groceries. <a href="/2006/jun/06/cadenza/">Read it »<\/a><\/p><\/div>' marker_cadenza.bindInfoWindowHtml(marker_cadenza.info_window_content, {maxWidth:400}); GEvent.addListener(marker_cadenza, "click", function() { map.panTo(point_cadenza, 2); @@ -345,7 +345,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_i_dont = new GMarker(point_i_dont, markerOptions); map.addOverlay(marker_i_dont); - marker_i_dont.info_window_content = '<div class="infowin"><h4>I Don't Sleep I Dream<\/h4><span class="date blok">May 28, 2006 (Vienna, Austria)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/freudsoffice.jpg" height="100" alt="I Don't Sleep I Dream" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>How can Freud\x27s former residence in Vienna lack a couch? The closest thing is up against the wall, behind a small writing desk in what was then the waiting room \x26mdash\x3B a small divan where one might stare at the patternless ceiling until the patterns emerge as it were. “Tell me about it,” he began. <a href="/2006/may/28/i-dont-sleep-i-dream/">Read it »<\/a><\/p><\/div>' + marker_i_dont.info_window_content = '<div class="infowin"><h4>I Don't Sleep I Dream<\/h4><span class="date blok">May 28, 2006 (Vienna, Austria)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/freudsoffice.jpg" height="100" alt="I Don't Sleep I Dream" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>How can Freud\u0027s former residence in Vienna lack a couch? The closest thing is up against the wall, behind a small writing desk in what was then the waiting room \u0026mdash\u003B a small divan where one might stare at the patternless ceiling until the patterns emerge as it were. “Tell me about it,” he began. <a href="/2006/may/28/i-dont-sleep-i-dream/">Read it »<\/a><\/p><\/div>' marker_i_dont.bindInfoWindowHtml(marker_i_dont.info_window_content, {maxWidth:400}); GEvent.addListener(marker_i_dont, "click", function() { map.panTo(point_i_dont, 2); @@ -356,7 +356,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_unreflected = new GMarker(point_unreflected, markerOptions); map.addOverlay(marker_unreflected); - marker_unreflected.info_window_content = '<div class="infowin"><h4>Unreflected<\/h4><span class="date blok">May 27, 2006 (Vienna, Austria)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/selfportraitconvex.jpg" height="100" alt="Unreflected" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Kunsthistorisches Museum contains probably the best collection of art outside of France \x26mdash\x3B Rubens, Rembrandt, Vermeer, Raphael, Velazquez, Bruegel and a certain Italian for whom I have a festering personal obsession, which shall be addressed shortly \x26mdash\x3B and what\x27s remarkable about this magnificent assemblage is that the vast majority of it was once the Hapsburg\x27s private collection. <a href="/2006/may/27/unreflected/">Read it »<\/a><\/p><\/div>' + marker_unreflected.info_window_content = '<div class="infowin"><h4>Unreflected<\/h4><span class="date blok">May 27, 2006 (Vienna, Austria)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/selfportraitconvex.jpg" height="100" alt="Unreflected" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Kunsthistorisches Museum contains probably the best collection of art outside of France \u0026mdash\u003B Rubens, Rembrandt, Vermeer, Raphael, Velazquez, Bruegel and a certain Italian for whom I have a festering personal obsession, which shall be addressed shortly \u0026mdash\u003B and what\u0027s remarkable about this magnificent assemblage is that the vast majority of it was once the Hapsburg\u0027s private collection. <a href="/2006/may/27/unreflected/">Read it »<\/a><\/p><\/div>' marker_unreflected.bindInfoWindowHtml(marker_unreflected.info_window_content, {maxWidth:400}); GEvent.addListener(marker_unreflected, "click", function() { map.panTo(point_unreflected, 2); @@ -367,7 +367,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_four_minutes = new GMarker(point_four_minutes, markerOptions); map.addOverlay(marker_four_minutes); - marker_four_minutes.info_window_content = '<div class="infowin"><h4>Four Minutes Thirty-Three Seconds<\/h4><span class="date blok">May 26, 2006 (Prague, Czech Republic)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/wallofnames.jpg" height="100" alt="Four Minutes Thirty-Three Seconds" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Just north of Prague\x27s old town square and east of the River Vltava is Josefov, the old Jewish quarter of Prague. The Pinkas Synagogue in Josefov is an unassuming pale, sand\x2Dcolored building with a slightly sunken entrance. Inside is a small alter and little else. The floor is bare\x3B there are no places for worshipers to sit. The synagogue is little more than walls. And on the walls inscribed in extremely small print are the names of the 77,297 Jewish citizens of Bohemia and Moravia who died in the Holocaust.\x0D\x0A <a href="/2006/may/26/four-minutes-thirty-three-seconds/">Read it »<\/a><\/p><\/div>' + marker_four_minutes.info_window_content = '<div class="infowin"><h4>Four Minutes Thirty-Three Seconds<\/h4><span class="date blok">May 26, 2006 (Prague, Czech Republic)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/wallofnames.jpg" height="100" alt="Four Minutes Thirty-Three Seconds" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Just north of Prague\u0027s old town square and east of the River Vltava is Josefov, the old Jewish quarter of Prague. The Pinkas Synagogue in Josefov is an unassuming pale, sand\u002Dcolored building with a slightly sunken entrance. Inside is a small alter and little else. The floor is bare\u003B there are no places for worshipers to sit. The synagogue is little more than walls. And on the walls inscribed in extremely small print are the names of the 77,297 Jewish citizens of Bohemia and Moravia who died in the Holocaust.\u000D\u000A <a href="/2006/may/26/four-minutes-thirty-three-seconds/">Read it »<\/a><\/p><\/div>' marker_four_minutes.bindInfoWindowHtml(marker_four_minutes.info_window_content, {maxWidth:400}); GEvent.addListener(marker_four_minutes, "click", function() { map.panTo(point_four_minutes, 2); @@ -378,7 +378,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_inside_and = new GMarker(point_inside_and, markerOptions); map.addOverlay(marker_inside_and); - marker_inside_and.info_window_content = '<div class="infowin"><h4>Inside and Out<\/h4><span class="date blok">May 25, 2006 (Cesky Krumlov, Czech Republic)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/krumlovcastleatnight.jpg" height="100" alt="Inside and Out" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Chasing Egon Schiele: The attention to detail that makes the difference between a building and work of art was everywhere in Cesky Krumlov, from the delicate pink and red complements of a fine dovetailed corner, to the white plaster and oak beams of the Egon Schiele museum, which, despite geometric differences, looked not unlike the Globe Theatre in London. <a href="/2006/may/25/inside-and-out/">Read it »<\/a><\/p><\/div>' + marker_inside_and.info_window_content = '<div class="infowin"><h4>Inside and Out<\/h4><span class="date blok">May 25, 2006 (Cesky Krumlov, Czech Republic)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/krumlovcastleatnight.jpg" height="100" alt="Inside and Out" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Chasing Egon Schiele: The attention to detail that makes the difference between a building and work of art was everywhere in Cesky Krumlov, from the delicate pink and red complements of a fine dovetailed corner, to the white plaster and oak beams of the Egon Schiele museum, which, despite geometric differences, looked not unlike the Globe Theatre in London. <a href="/2006/may/25/inside-and-out/">Read it »<\/a><\/p><\/div>' marker_inside_and.bindInfoWindowHtml(marker_inside_and.info_window_content, {maxWidth:400}); GEvent.addListener(marker_inside_and, "click", function() { map.panTo(point_inside_and, 2); @@ -389,7 +389,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_king = new GMarker(point_the_king, markerOptions); map.addOverlay(marker_the_king); - marker_the_king.info_window_content = '<div class="infowin"><h4>The King of Carrot Flowers Part Two<\/h4><span class="date blok">May 22, 2006 (Bled, Slovenia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/sloveniachurch.jpg" height="100" alt="The King of Carrot Flowers Part Two" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>There is a roughly 200km loop of road that leads northwest out of Bled, through a pass in the Julian Alps and then down the other side, twisting and winding back toward Bled by way of craggy canyons, small hamlets and crystalline rivers. We set out sometime after breakfast. <a href="/2006/may/22/king-carrot-flowers-part-two/">Read it »<\/a><\/p><\/div>' + marker_the_king.info_window_content = '<div class="infowin"><h4>The King of Carrot Flowers Part Two<\/h4><span class="date blok">May 22, 2006 (Bled, Slovenia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/sloveniachurch.jpg" height="100" alt="The King of Carrot Flowers Part Two" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>There is a roughly 200km loop of road that leads northwest out of Bled, through a pass in the Julian Alps and then down the other side, twisting and winding back toward Bled by way of craggy canyons, small hamlets and crystalline rivers. We set out sometime after breakfast. <a href="/2006/may/22/king-carrot-flowers-part-two/">Read it »<\/a><\/p><\/div>' marker_the_king.bindInfoWindowHtml(marker_the_king.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_king, "click", function() { map.panTo(point_the_king, 2); @@ -400,7 +400,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_ghost = new GMarker(point_ghost, markerOptions); map.addOverlay(marker_ghost); - marker_ghost.info_window_content = '<div class="infowin"><h4>Ghost<\/h4><span class="date blok">May 19, 2006 (Ljubljana, Slovenia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/trogirnight.jpg" height="100" alt="Ghost" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Like Dubrovnik, Trogir is a walled city of roughly Venetian vintage, but Trogir\x27s wall has largely crumbled away or been removed. Still, it has the gorgeous narrow cobblestone streets, arched doorways and towering forts that give all Dalmatian towns their Rapunzel\x2Dlike fairly tale quality. <a href="/2006/may/19/ghost/">Read it »<\/a><\/p><\/div>' + marker_ghost.info_window_content = '<div class="infowin"><h4>Ghost<\/h4><span class="date blok">May 19, 2006 (Ljubljana, Slovenia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/trogirnight.jpg" height="100" alt="Ghost" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Like Dubrovnik, Trogir is a walled city of roughly Venetian vintage, but Trogir\u0027s wall has largely crumbled away or been removed. Still, it has the gorgeous narrow cobblestone streets, arched doorways and towering forts that give all Dalmatian towns their Rapunzel\u002Dlike fairly tale quality. <a href="/2006/may/19/ghost/">Read it »<\/a><\/p><\/div>' marker_ghost.bindInfoWindowHtml(marker_ghost.info_window_content, {maxWidth:400}); GEvent.addListener(marker_ghost, "click", function() { map.panTo(point_ghost, 2); @@ -411,7 +411,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_feel_good = new GMarker(point_feel_good, markerOptions); map.addOverlay(marker_feel_good); - marker_feel_good.info_window_content = '<div class="infowin"><h4>Feel Good Lost<\/h4><span class="date blok">May 17, 2006 (Dubrovnik, Croatia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/dubrovnik.jpg" height="100" alt="Feel Good Lost" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Dubrovnik, Croatia was heavily shelled during the Bosnian conflict and roughly 65 percent of its buildings were hit, built for the most part you\x27d never know it. Most of the buildings date from about 1468, though some were destroyed in the great earthquake of 1667, still, by and large, the city looks as it did in the fifteenth century. <a href="/2006/may/17/feel-good-lost/">Read it »<\/a><\/p><\/div>' + marker_feel_good.info_window_content = '<div class="infowin"><h4>Feel Good Lost<\/h4><span class="date blok">May 17, 2006 (Dubrovnik, Croatia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/dubrovnik.jpg" height="100" alt="Feel Good Lost" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Dubrovnik, Croatia was heavily shelled during the Bosnian conflict and roughly 65 percent of its buildings were hit, built for the most part you\u0027d never know it. Most of the buildings date from about 1468, though some were destroyed in the great earthquake of 1667, still, by and large, the city looks as it did in the fifteenth century. <a href="/2006/may/17/feel-good-lost/">Read it »<\/a><\/p><\/div>' marker_feel_good.bindInfoWindowHtml(marker_feel_good.info_window_content, {maxWidth:400}); GEvent.addListener(marker_feel_good, "click", function() { map.panTo(point_feel_good, 2); @@ -422,7 +422,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_blue_milk = new GMarker(point_blue_milk, markerOptions); map.addOverlay(marker_blue_milk); - marker_blue_milk.info_window_content = '<div class="infowin"><h4>Blue Milk<\/h4><span class="date blok">May 15, 2006 (Lake Plitvice, Croatia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/plitvice.jpg" height="100" alt="Blue Milk" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\x27s hard to understand, standing on the banks of such crystalline, cerulean lakes, whose dazzling colors come from the mineral rich silt runoff of glaciers, that the largest European conflict since world war two began here, at Like Plitvice Croatia. But indeed this is where the first shots were fired on Easter Sunday in 1991 and the first casualty was a park policeman. <a href="/2006/may/15/blue-milk/">Read it »<\/a><\/p><\/div>' + marker_blue_milk.info_window_content = '<div class="infowin"><h4>Blue Milk<\/h4><span class="date blok">May 15, 2006 (Lake Plitvice, Croatia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/plitvice.jpg" height="100" alt="Blue Milk" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\u0027s hard to understand, standing on the banks of such crystalline, cerulean lakes, whose dazzling colors come from the mineral rich silt runoff of glaciers, that the largest European conflict since world war two began here, at Like Plitvice Croatia. But indeed this is where the first shots were fired on Easter Sunday in 1991 and the first casualty was a park policeman. <a href="/2006/may/15/blue-milk/">Read it »<\/a><\/p><\/div>' marker_blue_milk.bindInfoWindowHtml(marker_blue_milk.info_window_content, {maxWidth:400}); GEvent.addListener(marker_blue_milk, "click", function() { map.panTo(point_blue_milk, 2); @@ -433,7 +433,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_refracted_light = new GMarker(point_refracted_light, markerOptions); map.addOverlay(marker_refracted_light); - marker_refracted_light.info_window_content = '<div class="infowin"><h4>Refracted Light and Grace<\/h4><span class="date blok">May 10, 2006 (Budapest, Hungary)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/castlehillbuda.jpg" height="100" alt="Refracted Light and Grace" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Evening, after dinner, outside on the balcony, smoking cigarettes and contemplating the nightscape of Buda\x27s Castle Hill rising up out of its own golden reflection in the shimmering Danube waters. The drone of car horns in the distance and the electric tram squealing as it pulls out of the station below on the river a boat slowly churns upstream... <a href="/2006/may/10/refracted-light-and-grace/">Read it »<\/a><\/p><\/div>' + marker_refracted_light.info_window_content = '<div class="infowin"><h4>Refracted Light and Grace<\/h4><span class="date blok">May 10, 2006 (Budapest, Hungary)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/castlehillbuda.jpg" height="100" alt="Refracted Light and Grace" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Evening, after dinner, outside on the balcony, smoking cigarettes and contemplating the nightscape of Buda\u0027s Castle Hill rising up out of its own golden reflection in the shimmering Danube waters. The drone of car horns in the distance and the electric tram squealing as it pulls out of the station below on the river a boat slowly churns upstream... <a href="/2006/may/10/refracted-light-and-grace/">Read it »<\/a><\/p><\/div>' marker_refracted_light.bindInfoWindowHtml(marker_refracted_light.info_window_content, {maxWidth:400}); GEvent.addListener(marker_refracted_light, "click", function() { map.panTo(point_refracted_light, 2); @@ -444,7 +444,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_london_calling = new GMarker(point_london_calling, markerOptions); map.addOverlay(marker_london_calling); - marker_london_calling.info_window_content = '<div class="infowin"><h4>London Calling<\/h4><span class="date blok">May 9, 2006 (London, United Kingdom)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/londonthames.jpg" height="100" alt="London Calling" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>London: The British don\x27t want me \x2D\x2D no money, no proof I\x27m leaving and no real reason for coming, good lord, I must be a vagabond, up to no good, surely. Eventually the customs agent relents and lets me in, a favor I repay by nearly burning down one of London\x27s bigger parks. Seriously. <a href="/2006/may/09/london-calling/">Read it »<\/a><\/p><\/div>' + marker_london_calling.info_window_content = '<div class="infowin"><h4>London Calling<\/h4><span class="date blok">May 9, 2006 (London, United Kingdom)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/londonthames.jpg" height="100" alt="London Calling" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>London: The British don\u0027t want me \u002D\u002D no money, no proof I\u0027m leaving and no real reason for coming, good lord, I must be a vagabond, up to no good, surely. Eventually the customs agent relents and lets me in, a favor I repay by nearly burning down one of London\u0027s bigger parks. Seriously. <a href="/2006/may/09/london-calling/">Read it »<\/a><\/p><\/div>' marker_london_calling.bindInfoWindowHtml(marker_london_calling.info_window_content, {maxWidth:400}); GEvent.addListener(marker_london_calling, "click", function() { map.panTo(point_london_calling, 2); @@ -455,7 +455,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_closing_time = new GMarker(point_closing_time, markerOptions); map.addOverlay(marker_closing_time); - marker_closing_time.info_window_content = '<div class="infowin"><h4>Closing Time<\/h4><span class="date blok">April 30, 2006 (Trang, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/thailandtrain.jpg" height="100" alt="Closing Time" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Headed back to Europe: I started to write a bit of reminiscence, trying to remember the highlights of my time in Asia before I return to the west, but about halfway through I kept thinking of a popular Buddhist saying \x26mdash\x3B be here now. Most of these dispatches are written in past tense, but this time I want to simply be here now. This moment, on this train. This is the last time I\x27ll post something from Southeast Asia. <a href="/2006/apr/30/closing-time/">Read it »<\/a><\/p><\/div>' + marker_closing_time.info_window_content = '<div class="infowin"><h4>Closing Time<\/h4><span class="date blok">April 30, 2006 (Trang, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/thailandtrain.jpg" height="100" alt="Closing Time" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Headed back to Europe: I started to write a bit of reminiscence, trying to remember the highlights of my time in Asia before I return to the west, but about halfway through I kept thinking of a popular Buddhist saying \u0026mdash\u003B be here now. Most of these dispatches are written in past tense, but this time I want to simply be here now. This moment, on this train. This is the last time I\u0027ll post something from Southeast Asia. <a href="/2006/apr/30/closing-time/">Read it »<\/a><\/p><\/div>' marker_closing_time.bindInfoWindowHtml(marker_closing_time.info_window_content, {maxWidth:400}); GEvent.addListener(marker_closing_time, "click", function() { map.panTo(point_closing_time, 2); @@ -466,7 +466,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_beginning_of = new GMarker(point_beginning_of, markerOptions); map.addOverlay(marker_beginning_of); - marker_beginning_of.info_window_content = '<div class="infowin"><h4>Beginning of the End<\/h4><span class="date blok">April 21, 2006 (Koh Kradan, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/kokradan.jpg" height="100" alt="Beginning of the End" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I wasn\x27t expecting much from Ko Kradan, but in the end I discovered a slice of Thailand the way it\x27s often describe by wistful hippies who first came here twenty years ago. Tong and Ngu and the rest of the Thais working at Paradise Lost were the nicest people I met in Thailand and Wally was by far the most laid back farang I\x27ve come across. I ended up staying on Ko Kradan for the remainder of my time in the south.\x0D\x0A <a href="/2006/apr/21/beginning-end/">Read it »<\/a><\/p><\/div>' + marker_beginning_of.info_window_content = '<div class="infowin"><h4>Beginning of the End<\/h4><span class="date blok">April 21, 2006 (Koh Kradan, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/kokradan.jpg" height="100" alt="Beginning of the End" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I wasn\u0027t expecting much from Ko Kradan, but in the end I discovered a slice of Thailand the way it\u0027s often describe by wistful hippies who first came here twenty years ago. Tong and Ngu and the rest of the Thais working at Paradise Lost were the nicest people I met in Thailand and Wally was by far the most laid back farang I\u0027ve come across. I ended up staying on Ko Kradan for the remainder of my time in the south.\u000D\u000A <a href="/2006/apr/21/beginning-end/">Read it »<\/a><\/p><\/div>' marker_beginning_of.bindInfoWindowHtml(marker_beginning_of.info_window_content, {maxWidth:400}); GEvent.addListener(marker_beginning_of, "click", function() { map.panTo(point_beginning_of, 2); @@ -477,7 +477,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_going_down = new GMarker(point_going_down, markerOptions); map.addOverlay(marker_going_down); - marker_going_down.info_window_content = '<div class="infowin"><h4>Going Down South<\/h4><span class="date blok">April 10, 2006 (Koh Phi Phi, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/thailandleahkate.jpg" height="100" alt="Going Down South" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Phi Phi Island Resort, where some friends were staying, is nestled on the leeward shore of Koh Phi Phi Island and posts a private beach, beautiful reef, fancy swimming pools and rooms with real sheets. Unheard of. I sauntered in a day early, acted like I owned the place, rented snorkel gear, charged it to a random room number and spent the afternoon on the reef. If only I could have put it on the Underhill\x27s credit card. <a href="/2006/apr/10/going-down-south/">Read it »<\/a><\/p><\/div>' + marker_going_down.info_window_content = '<div class="infowin"><h4>Going Down South<\/h4><span class="date blok">April 10, 2006 (Koh Phi Phi, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/thailandleahkate.jpg" height="100" alt="Going Down South" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Phi Phi Island Resort, where some friends were staying, is nestled on the leeward shore of Koh Phi Phi Island and posts a private beach, beautiful reef, fancy swimming pools and rooms with real sheets. Unheard of. I sauntered in a day early, acted like I owned the place, rented snorkel gear, charged it to a random room number and spent the afternoon on the reef. If only I could have put it on the Underhill\u0027s credit card. <a href="/2006/apr/10/going-down-south/">Read it »<\/a><\/p><\/div>' marker_going_down.bindInfoWindowHtml(marker_going_down.info_window_content, {maxWidth:400}); GEvent.addListener(marker_going_down, "click", function() { map.panTo(point_going_down, 2); @@ -488,7 +488,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_book = new GMarker(point_the_book, markerOptions); map.addOverlay(marker_the_book); - marker_the_book.info_window_content = '<div class="infowin"><h4>The Book of Right On<\/h4><span class="date blok">March 30, 2006 (Sinoukville, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/goodbyes.jpg" height="100" alt="The Book of Right On" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The next day we continued on to Sinoukville which is Cambodia\x27s attempt at a seaside resort. Combining the essential elements of Goa and Thailand, Sinoukville is a pleasant, if somewhat hippy\x2Doriented, travelers haven. We rented Honda Dreams and cruised down the coast to deserted white sand beaches, thatched huts serving noodles and rice, where we watched sunsets and dodged rain storms. <a href="/2006/mar/30/book-right/">Read it »<\/a><\/p><\/div>' + marker_the_book.info_window_content = '<div class="infowin"><h4>The Book of Right On<\/h4><span class="date blok">March 30, 2006 (Sinoukville, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/goodbyes.jpg" height="100" alt="The Book of Right On" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The next day we continued on to Sinoukville which is Cambodia\u0027s attempt at a seaside resort. Combining the essential elements of Goa and Thailand, Sinoukville is a pleasant, if somewhat hippy\u002Doriented, travelers haven. We rented Honda Dreams and cruised down the coast to deserted white sand beaches, thatched huts serving noodles and rice, where we watched sunsets and dodged rain storms. <a href="/2006/mar/30/book-right/">Read it »<\/a><\/p><\/div>' marker_the_book.bindInfoWindowHtml(marker_the_book.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_book, "click", function() { map.panTo(point_the_book, 2); @@ -499,7 +499,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_midnight_in = new GMarker(point_midnight_in, markerOptions); map.addOverlay(marker_midnight_in); - marker_midnight_in.info_window_content = '<div class="infowin"><h4>Midnight in a Perfect World<\/h4><span class="date blok">March 26, 2006 (Death Island, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/deathisland.jpg" height="100" alt="Midnight in a Perfect World" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Death Island, as Rob nicknamed it, was just what I needed. The first day we sat down for lunch and ordered crab\x3B a boy in his underwear proceeded to run out of the kitchen, swam out in the ocean and began unloading crabs from a trap into a bucket. It doesn\x27t get much fresher than that. Throw in a nice beach, some cheap bungalows and you\x27re away. <a href="/2006/mar/26/midnight-perfect-world/">Read it »<\/a><\/p><\/div>' + marker_midnight_in.info_window_content = '<div class="infowin"><h4>Midnight in a Perfect World<\/h4><span class="date blok">March 26, 2006 (Death Island, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/deathisland.jpg" height="100" alt="Midnight in a Perfect World" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Death Island, as Rob nicknamed it, was just what I needed. The first day we sat down for lunch and ordered crab\u003B a boy in his underwear proceeded to run out of the kitchen, swam out in the ocean and began unloading crabs from a trap into a bucket. It doesn\u0027t get much fresher than that. Throw in a nice beach, some cheap bungalows and you\u0027re away. <a href="/2006/mar/26/midnight-perfect-world/">Read it »<\/a><\/p><\/div>' marker_midnight_in.bindInfoWindowHtml(marker_midnight_in.info_window_content, {maxWidth:400}); GEvent.addListener(marker_midnight_in, "click", function() { map.panTo(point_midnight_in, 2); @@ -510,7 +510,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_angkor_wat = new GMarker(point_angkor_wat, markerOptions); map.addOverlay(marker_angkor_wat); - marker_angkor_wat.info_window_content = '<div class="infowin"><h4>Angkor Wat<\/h4><span class="date blok">March 21, 2006 (Angkor Wat, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/angkorwat.jpg" height="100" alt="Angkor Wat" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Roughly half a million people a year visit Angkor Wat. The first evening we decided to see just how tourist\x2Dfilled Angkor was by heading to the most popular sunset temple, Phnom Bakheng, to watch the sunset. And there were a lot of tourists. Thousands of them. And that was just at one temple. Thus was hatched the plan: see Angkor in the heat of the day. Yes it will be hot. Hot hot hot. Fucking hot. But hopefully empty. <a href="/2006/mar/21/angkor-wat/">Read it »<\/a><\/p><\/div>' + marker_angkor_wat.info_window_content = '<div class="infowin"><h4>Angkor Wat<\/h4><span class="date blok">March 21, 2006 (Angkor Wat, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/angkorwat.jpg" height="100" alt="Angkor Wat" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Roughly half a million people a year visit Angkor Wat. The first evening we decided to see just how tourist\u002Dfilled Angkor was by heading to the most popular sunset temple, Phnom Bakheng, to watch the sunset. And there were a lot of tourists. Thousands of them. And that was just at one temple. Thus was hatched the plan: see Angkor in the heat of the day. Yes it will be hot. Hot hot hot. Fucking hot. But hopefully empty. <a href="/2006/mar/21/angkor-wat/">Read it »<\/a><\/p><\/div>' marker_angkor_wat.bindInfoWindowHtml(marker_angkor_wat.info_window_content, {maxWidth:400}); GEvent.addListener(marker_angkor_wat, "click", function() { map.panTo(point_angkor_wat, 2); @@ -521,7 +521,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_wait_til = new GMarker(point_wait_til, markerOptions); map.addOverlay(marker_wait_til); - marker_wait_til.info_window_content = '<div class="infowin"><h4>...Wait 'til it Blows<\/h4><span class="date blok">March 18, 2006 (Seam Reap, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/landmines.jpg" height="100" alt="...Wait 'til it Blows" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>One the things I may have failed to mention thus far in my Cambodia reportage is that this was/is one of the most heavily mined areas in the world. You might think that removing landmines involves sophisticated technology of the sort you see in BBC documentaries on Bosnia, but here in Cambodia landmine removal is most often handled by the technological marvel of southeast Asia \x26mdash\x3B the bamboo stick. <a href="/2006/mar/18/wait-til-it-blows/">Read it »<\/a><\/p><\/div>' + marker_wait_til.info_window_content = '<div class="infowin"><h4>...Wait 'til it Blows<\/h4><span class="date blok">March 18, 2006 (Seam Reap, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/landmines.jpg" height="100" alt="...Wait 'til it Blows" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>One the things I may have failed to mention thus far in my Cambodia reportage is that this was/is one of the most heavily mined areas in the world. You might think that removing landmines involves sophisticated technology of the sort you see in BBC documentaries on Bosnia, but here in Cambodia landmine removal is most often handled by the technological marvel of southeast Asia \u0026mdash\u003B the bamboo stick. <a href="/2006/mar/18/wait-til-it-blows/">Read it »<\/a><\/p><\/div>' marker_wait_til.bindInfoWindowHtml(marker_wait_til.info_window_content, {maxWidth:400}); GEvent.addListener(marker_wait_til, "click", function() { map.panTo(point_wait_til, 2); @@ -532,7 +532,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_beginning_to = new GMarker(point_beginning_to, markerOptions); map.addOverlay(marker_beginning_to); - marker_beginning_to.info_window_content = '<div class="infowin"><h4>Beginning to See the Light<\/h4><span class="date blok">March 16, 2006 (Battambang, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/floatingvillage.jpg" height="100" alt="Beginning to See the Light" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Surprisingly, a floating village is not that different than a village on the land. There are the same stores, the computer repair shop, the grocers, the petrol station, the temple, the dance hall and all the other things that makeup a town. I could even say with some authority that the town is laid out in streets, watery pathways that form nearly perfect lines. <a href="/2006/mar/16/beginning-see-light/">Read it »<\/a><\/p><\/div>' + marker_beginning_to.info_window_content = '<div class="infowin"><h4>Beginning to See the Light<\/h4><span class="date blok">March 16, 2006 (Battambang, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/floatingvillage.jpg" height="100" alt="Beginning to See the Light" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Surprisingly, a floating village is not that different than a village on the land. There are the same stores, the computer repair shop, the grocers, the petrol station, the temple, the dance hall and all the other things that makeup a town. I could even say with some authority that the town is laid out in streets, watery pathways that form nearly perfect lines. <a href="/2006/mar/16/beginning-see-light/">Read it »<\/a><\/p><\/div>' marker_beginning_to.bindInfoWindowHtml(marker_beginning_to.info_window_content, {maxWidth:400}); GEvent.addListener(marker_beginning_to, "click", function() { map.panTo(point_beginning_to, 2); @@ -543,7 +543,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_blood_on = new GMarker(point_blood_on, markerOptions); map.addOverlay(marker_blood_on); - marker_blood_on.info_window_content = '<div class="infowin"><h4>Blood on the Tracks<\/h4><span class="date blok">March 14, 2006 (Phenom Phen, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/killingfields.jpg" height="100" alt="Blood on the Tracks" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>As I mentioned in the last entry I came down with a bit of a fever for a few days. This was accompanied by what we in the group have come to term, for lack of a nicer, but equally descriptive phrase \x26mdash\x3B pissing out the ass. It\x27s not a pretty picture. Nor is it a pleasant experience, and consequently I don\x27t have a real clear recollection of the journey from Ban Lung to Kratie or from Kratie out to Sen Monoron. <a href="/2006/mar/14/blood-tracks/">Read it »<\/a><\/p><\/div>' + marker_blood_on.info_window_content = '<div class="infowin"><h4>Blood on the Tracks<\/h4><span class="date blok">March 14, 2006 (Phenom Phen, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/killingfields.jpg" height="100" alt="Blood on the Tracks" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>As I mentioned in the last entry I came down with a bit of a fever for a few days. This was accompanied by what we in the group have come to term, for lack of a nicer, but equally descriptive phrase \u0026mdash\u003B pissing out the ass. It\u0027s not a pretty picture. Nor is it a pleasant experience, and consequently I don\u0027t have a real clear recollection of the journey from Ban Lung to Kratie or from Kratie out to Sen Monoron. <a href="/2006/mar/14/blood-tracks/">Read it »<\/a><\/p><\/div>' marker_blood_on.bindInfoWindowHtml(marker_blood_on.info_window_content, {maxWidth:400}); GEvent.addListener(marker_blood_on, "click", function() { map.panTo(point_blood_on, 2); @@ -554,7 +554,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_ticket_to = new GMarker(point_ticket_to, markerOptions); map.addOverlay(marker_ticket_to); - marker_ticket_to.info_window_content = '<div class="infowin"><h4>Ticket To Ride<\/h4><span class="date blok">March 7, 2006 (Ban Lung, Cambodia)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/hondadream.jpg" height="100" alt="Ticket To Ride" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I can\x27t see. My eyebrows are orange with dust. I cannot see them, but I know they must be\x3B they were yesterday. Every now and then when her legs clench down on my hips or her fingernails dig into my shoulders, I remember Debi is behind me and I am more or less responsible for not killing both of us. <a href="/2006/mar/07/ticket-ride/">Read it »<\/a><\/p><\/div>' + marker_ticket_to.info_window_content = '<div class="infowin"><h4>Ticket To Ride<\/h4><span class="date blok">March 7, 2006 (Ban Lung, Cambodia)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/hondadream.jpg" height="100" alt="Ticket To Ride" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I can\u0027t see. My eyebrows are orange with dust. I cannot see them, but I know they must be\u003B they were yesterday. Every now and then when her legs clench down on my hips or her fingernails dig into my shoulders, I remember Debi is behind me and I am more or less responsible for not killing both of us. <a href="/2006/mar/07/ticket-ride/">Read it »<\/a><\/p><\/div>' marker_ticket_to.bindInfoWindowHtml(marker_ticket_to.info_window_content, {maxWidth:400}); GEvent.addListener(marker_ticket_to, "click", function() { map.panTo(point_ticket_to, 2); @@ -565,7 +565,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_little_corner = new GMarker(point_little_corner, markerOptions); map.addOverlay(marker_little_corner); - marker_little_corner.info_window_content = '<div class="infowin"><h4>Little Corner of the World<\/h4><span class="date blok">February 28, 2006 (Four Thousand Islands, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/siphondon.jpg" height="100" alt="Little Corner of the World" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\x27s difficult to explain but the further south you go in Laos the more relaxed life becomes. Since life in the north is not exactly high stress, by the time we arrived in the four thousand Islands we had to check our pulse periodically to ensure that time was in fact still moving forward. <a href="/2006/feb/28/little-corner-world/">Read it »<\/a><\/p><\/div>' + marker_little_corner.info_window_content = '<div class="infowin"><h4>Little Corner of the World<\/h4><span class="date blok">February 28, 2006 (Four Thousand Islands, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/siphondon.jpg" height="100" alt="Little Corner of the World" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\u0027s difficult to explain but the further south you go in Laos the more relaxed life becomes. Since life in the north is not exactly high stress, by the time we arrived in the four thousand Islands we had to check our pulse periodically to ensure that time was in fact still moving forward. <a href="/2006/feb/28/little-corner-world/">Read it »<\/a><\/p><\/div>' marker_little_corner.bindInfoWindowHtml(marker_little_corner.info_window_content, {maxWidth:400}); GEvent.addListener(marker_little_corner, "click", function() { map.panTo(point_little_corner, 2); @@ -576,7 +576,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_can8217t_get = new GMarker(point_can8217t_get, markerOptions); map.addOverlay(marker_can8217t_get); - marker_can8217t_get.info_window_content = '<div class="infowin"><h4>Can&#8217;t Get There From Here<\/h4><span class="date blok">February 24, 2006 (Attapeu, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/attapeulight.jpg" height="100" alt="Can&#8217;t Get There From Here" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The most magical light in Laos lives on the Bolevan Plateau. For some reason not many tourists seem to make it out to the Bolevan Plateau, in spite of the fact that the roads are quite good, transport runs regularly, the villages peaceful, even sleepy, little hamlets. In short, the Bolevan Plateau is wonderful, and not the least in part because no one else is there. <a href="/2006/feb/24/cant-get-there-here/">Read it »<\/a><\/p><\/div>' + marker_can8217t_get.info_window_content = '<div class="infowin"><h4>Can&#8217;t Get There From Here<\/h4><span class="date blok">February 24, 2006 (Attapeu, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/attapeulight.jpg" height="100" alt="Can&#8217;t Get There From Here" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The most magical light in Laos lives on the Bolevan Plateau. For some reason not many tourists seem to make it out to the Bolevan Plateau, in spite of the fact that the roads are quite good, transport runs regularly, the villages peaceful, even sleepy, little hamlets. In short, the Bolevan Plateau is wonderful, and not the least in part because no one else is there. <a href="/2006/feb/24/cant-get-there-here/">Read it »<\/a><\/p><\/div>' marker_can8217t_get.bindInfoWindowHtml(marker_can8217t_get.info_window_content, {maxWidth:400}); GEvent.addListener(marker_can8217t_get, "click", function() { map.panTo(point_can8217t_get, 2); @@ -587,7 +587,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_safe_as = new GMarker(point_safe_as, markerOptions); map.addOverlay(marker_safe_as); - marker_safe_as.info_window_content = '<div class="infowin"><h4>Safe as Milk<\/h4><span class="date blok">February 18, 2006 (Sekong, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/usbombs.jpg" height="100" alt="Safe as Milk" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>You would think, if you were the United States and you were illegally and unofficially bombing a foreign country you might not want to stamp \x22US Bomb\x22 on the side of your bombs, and yet there it was all over Laos: \x22US Bomb.\x22 Clearly somebody didn\x27t think things all the way through, especially given that roughly one third of said bombs failed to explode. <a href="/2006/feb/18/safe-milk/">Read it »<\/a><\/p><\/div>' + marker_safe_as.info_window_content = '<div class="infowin"><h4>Safe as Milk<\/h4><span class="date blok">February 18, 2006 (Sekong, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/usbombs.jpg" height="100" alt="Safe as Milk" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>You would think, if you were the United States and you were illegally and unofficially bombing a foreign country you might not want to stamp \u0022US Bomb\u0022 on the side of your bombs, and yet there it was all over Laos: \u0022US Bomb.\u0022 Clearly somebody didn\u0027t think things all the way through, especially given that roughly one third of said bombs failed to explode. <a href="/2006/feb/18/safe-milk/">Read it »<\/a><\/p><\/div>' marker_safe_as.bindInfoWindowHtml(marker_safe_as.info_window_content, {maxWidth:400}); GEvent.addListener(marker_safe_as, "click", function() { map.panTo(point_safe_as, 2); @@ -598,7 +598,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_everyday_the = new GMarker(point_everyday_the, markerOptions); map.addOverlay(marker_everyday_the); - marker_everyday_the.info_window_content = '<div class="infowin"><h4>Everyday the Fourteenth<\/h4><span class="date blok">February 14, 2006 (Champasak, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/hinbunriver.jpg" height="100" alt="Everyday the Fourteenth" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We piled four large bags, four daypacks and five people in a six meter dugout canoe. The boat was powered by the ever\x2Dpresent\x2Din\x2Dsoutheast\x2DAsia long tail motor which is essential a lawnmower engine with a three meter pole extending out of it to which a small propeller is attached \x26mdash\x3B perfect for navigating shallow water. And by shallow I mean sometimes a mere inch between the hull and the riverbed. <a href="/2006/feb/14/everyday-fourteenth/">Read it »<\/a><\/p><\/div>' + marker_everyday_the.info_window_content = '<div class="infowin"><h4>Everyday the Fourteenth<\/h4><span class="date blok">February 14, 2006 (Champasak, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/hinbunriver.jpg" height="100" alt="Everyday the Fourteenth" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We piled four large bags, four daypacks and five people in a six meter dugout canoe. The boat was powered by the ever\u002Dpresent\u002Din\u002Dsoutheast\u002DAsia long tail motor which is essential a lawnmower engine with a three meter pole extending out of it to which a small propeller is attached \u0026mdash\u003B perfect for navigating shallow water. And by shallow I mean sometimes a mere inch between the hull and the riverbed. <a href="/2006/feb/14/everyday-fourteenth/">Read it »<\/a><\/p><\/div>' marker_everyday_the.bindInfoWindowHtml(marker_everyday_the.info_window_content, {maxWidth:400}); GEvent.addListener(marker_everyday_the, "click", function() { map.panTo(point_everyday_the, 2); @@ -609,7 +609,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_water_slides = new GMarker(point_water_slides, markerOptions); map.addOverlay(marker_water_slides); - marker_water_slides.info_window_content = '<div class="infowin"><h4>Water Slides and Spirit Guides<\/h4><span class="date blok">February 10, 2006 (Champasak, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/konglorcave.jpg" height="100" alt="Water Slides and Spirit Guides" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The dramatic black karst limestone mountains ringing Ban Na Hin grew darker as the light faded. I was sitting alone on the back porch of our guesthouse watching the light slowly disappear from the bottoms of the clouds and wondering absently how many pages it would take to explain how I came to be in the tiny town of Ban Na Hin, or if such an explanation even really existed. <a href="/2006/feb/10/water-slides-and-spirit-guides/">Read it »<\/a><\/p><\/div>' + marker_water_slides.info_window_content = '<div class="infowin"><h4>Water Slides and Spirit Guides<\/h4><span class="date blok">February 10, 2006 (Champasak, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/konglorcave.jpg" height="100" alt="Water Slides and Spirit Guides" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The dramatic black karst limestone mountains ringing Ban Na Hin grew darker as the light faded. I was sitting alone on the back porch of our guesthouse watching the light slowly disappear from the bottoms of the clouds and wondering absently how many pages it would take to explain how I came to be in the tiny town of Ban Na Hin, or if such an explanation even really existed. <a href="/2006/feb/10/water-slides-and-spirit-guides/">Read it »<\/a><\/p><\/div>' marker_water_slides.bindInfoWindowHtml(marker_water_slides.info_window_content, {maxWidth:400}); GEvent.addListener(marker_water_slides, "click", function() { map.panTo(point_water_slides, 2); @@ -620,7 +620,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_lovely = new GMarker(point_the_lovely, markerOptions); map.addOverlay(marker_the_lovely); - marker_the_lovely.info_window_content = '<div class="infowin"><h4>The Lovely Universe<\/h4><span class="date blok">February 4, 2006 (Vang Vieng, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/vangveing.jpg" height="100" alt="The Lovely Universe" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I would like to say that I have something memorable to write about Vang Vieng, but the truth is we mostly sat around doing very little, making new friends, drinking a beer around the fire and waiting out the Chinese new year celebrations, which meant none of us could get Cambodian visas until the following Monday. We were forced to relax beside the river for several more days than we intended. Yes friends, traveling is hard, but I do it for you. <a href="/2006/feb/04/lovely-universe/">Read it »<\/a><\/p><\/div>' + marker_the_lovely.info_window_content = '<div class="infowin"><h4>The Lovely Universe<\/h4><span class="date blok">February 4, 2006 (Vang Vieng, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/vangveing.jpg" height="100" alt="The Lovely Universe" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I would like to say that I have something memorable to write about Vang Vieng, but the truth is we mostly sat around doing very little, making new friends, drinking a beer around the fire and waiting out the Chinese new year celebrations, which meant none of us could get Cambodian visas until the following Monday. We were forced to relax beside the river for several more days than we intended. Yes friends, traveling is hard, but I do it for you. <a href="/2006/feb/04/lovely-universe/">Read it »<\/a><\/p><\/div>' marker_the_lovely.bindInfoWindowHtml(marker_the_lovely.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_lovely, "click", function() { map.panTo(point_the_lovely, 2); @@ -631,7 +631,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_i_used = new GMarker(point_i_used, markerOptions); map.addOverlay(marker_i_used); - marker_i_used.info_window_content = '<div class="infowin"><h4>I Used to Fly Like Peter Pan<\/h4><span class="date blok">January 21, 2006 (Luang Nam Tha, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/gibbonexperience.jpg" height="100" alt="I Used to Fly Like Peter Pan" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The next time someone asks you, \x26#8220\x3Bwould you like to live in a tree house and travel five hundred feet above the ground attached to a zip wire?\x26#8221\x3B I highly suggest you say, \x26#8220\x3Byes, where do a I sign up?\x26#8221\x3B If you happen to be in Laos, try the Gibbon Experience. <a href="/2006/jan/21/i-used-fly-peter-pan/">Read it »<\/a><\/p><\/div>' + marker_i_used.info_window_content = '<div class="infowin"><h4>I Used to Fly Like Peter Pan<\/h4><span class="date blok">January 21, 2006 (Luang Nam Tha, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/gibbonexperience.jpg" height="100" alt="I Used to Fly Like Peter Pan" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The next time someone asks you, \u0026#8220\u003Bwould you like to live in a tree house and travel five hundred feet above the ground attached to a zip wire?\u0026#8221\u003B I highly suggest you say, \u0026#8220\u003Byes, where do a I sign up?\u0026#8221\u003B If you happen to be in Laos, try the Gibbon Experience. <a href="/2006/jan/21/i-used-fly-peter-pan/">Read it »<\/a><\/p><\/div>' marker_i_used.bindInfoWindowHtml(marker_i_used.info_window_content, {maxWidth:400}); GEvent.addListener(marker_i_used, "click", function() { map.panTo(point_i_used, 2); @@ -642,7 +642,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_hymn_of = new GMarker(point_hymn_of, markerOptions); map.addOverlay(marker_hymn_of); - marker_hymn_of.info_window_content = '<div class="infowin"><h4>Hymn of the Big Wheel<\/h4><span class="date blok">January 19, 2006 (Luang Prabang, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/bluemilkwaterfall.jpg" height="100" alt="Hymn of the Big Wheel" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Jose Saramago writes in \x3Ccite\x3EThe Year of the Death of Ricardo Reis\x3C/cite\x3E that the gods \x22journey like us in the river of things, differing from us only because we call them gods and sometimes believe in them.\x22 Sitting in the middle of the river listening to the gurgle of water moving over stone and around trees I began to think that perhaps this is the sound of some lost language, a sound capable of creating mountains, valleys, estuaries, isthmuses and all the other forms around us, gurgling and sonorous but without clear meaning, shrouded in turquoise, a mystery through which we can move our sense of wonder intact. <a href="/2006/jan/19/hymn-big-wheel/">Read it »<\/a><\/p><\/div>' + marker_hymn_of.info_window_content = '<div class="infowin"><h4>Hymn of the Big Wheel<\/h4><span class="date blok">January 19, 2006 (Luang Prabang, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/bluemilkwaterfall.jpg" height="100" alt="Hymn of the Big Wheel" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Jose Saramago writes in \u003Ccite\u003EThe Year of the Death of Ricardo Reis\u003C/cite\u003E that the gods \u0022journey like us in the river of things, differing from us only because we call them gods and sometimes believe in them.\u0022 Sitting in the middle of the river listening to the gurgle of water moving over stone and around trees I began to think that perhaps this is the sound of some lost language, a sound capable of creating mountains, valleys, estuaries, isthmuses and all the other forms around us, gurgling and sonorous but without clear meaning, shrouded in turquoise, a mystery through which we can move our sense of wonder intact. <a href="/2006/jan/19/hymn-big-wheel/">Read it »<\/a><\/p><\/div>' marker_hymn_of.bindInfoWindowHtml(marker_hymn_of.info_window_content, {maxWidth:400}); GEvent.addListener(marker_hymn_of, "click", function() { map.panTo(point_hymn_of, 2); @@ -653,7 +653,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_down_the = new GMarker(point_down_the, markerOptions); map.addOverlay(marker_down_the); - marker_down_the.info_window_content = '<div class="infowin"><h4>Down the River<\/h4><span class="date blok">January 17, 2006 (Luang Prabang, Lao (PDR))<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/mekongslowboat.jpg" height="100" alt="Down the River" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Morning in Chiang Khong Thailand revealed itself as a foggy, and not a little mysterious, affair with the far shore of the Mekong, the Laos shore, almost completely hidden in a veil of mist. The first ferry crossed at eight and I was on it, looking to meet up with the slow boat to Luang Prabang. <a href="/2006/jan/17/down-river/">Read it »<\/a><\/p><\/div>' + marker_down_the.info_window_content = '<div class="infowin"><h4>Down the River<\/h4><span class="date blok">January 17, 2006 (Luang Prabang, Lao (PDR))<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/mekongslowboat.jpg" height="100" alt="Down the River" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Morning in Chiang Khong Thailand revealed itself as a foggy, and not a little mysterious, affair with the far shore of the Mekong, the Laos shore, almost completely hidden in a veil of mist. The first ferry crossed at eight and I was on it, looking to meet up with the slow boat to Luang Prabang. <a href="/2006/jan/17/down-river/">Read it »<\/a><\/p><\/div>' marker_down_the.bindInfoWindowHtml(marker_down_the.info_window_content, {maxWidth:400}); GEvent.addListener(marker_down_the, "click", function() { map.panTo(point_down_the, 2); @@ -664,7 +664,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_king = new GMarker(point_the_king, markerOptions); map.addOverlay(marker_the_king); - marker_the_king.info_window_content = '<div class="infowin"><h4>The King of Carrot Flowers<\/h4><span class="date blok">January 17, 2006 (Doi Inthanan National Park, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/maesaorchids.jpg" height="100" alt="The King of Carrot Flowers" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The light outside the windows was still a pre\x2Ddawn inky blue when the freezing cold water hit my back. A cold shower at six thirty in the morning is infinitely more powerful, albeit not at long lasting, as a cup of coffee. After dropping my body temperature a few degrees and having no towel to dry off with, just a dirty shirt and ceaseless ceiling fan, a cup of tea seemed like a good idea so I stopped in at the restaurant downstairs and, after a cup of hot water with some Jasmine leaves swirling at the bottom of it, I climbed on my rental motorbike and set out for Doi Inthanan National Park. <a href="/2006/jan/17/king-carrot-flowers/">Read it »<\/a><\/p><\/div>' + marker_the_king.info_window_content = '<div class="infowin"><h4>The King of Carrot Flowers<\/h4><span class="date blok">January 17, 2006 (Doi Inthanan National Park, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/maesaorchids.jpg" height="100" alt="The King of Carrot Flowers" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The light outside the windows was still a pre\u002Ddawn inky blue when the freezing cold water hit my back. A cold shower at six thirty in the morning is infinitely more powerful, albeit not at long lasting, as a cup of coffee. After dropping my body temperature a few degrees and having no towel to dry off with, just a dirty shirt and ceaseless ceiling fan, a cup of tea seemed like a good idea so I stopped in at the restaurant downstairs and, after a cup of hot water with some Jasmine leaves swirling at the bottom of it, I climbed on my rental motorbike and set out for Doi Inthanan National Park. <a href="/2006/jan/17/king-carrot-flowers/">Read it »<\/a><\/p><\/div>' marker_the_king.bindInfoWindowHtml(marker_the_king.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_king, "click", function() { map.panTo(point_the_king, 2); @@ -675,7 +675,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_you_and = new GMarker(point_you_and, markerOptions); map.addOverlay(marker_you_and); - marker_you_and.info_window_content = '<div class="infowin"><h4>You and I Are Disappearing<\/h4><span class="date blok">January 11, 2006 (Chang Mai, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/changmaiumong.jpg" height="100" alt="You and I Are Disappearing" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The all night bus reached Chiang Mai well past dawn, the city already beginning to stir. I considered trying to nap, but in the end decided to explore the town. What better way to see Buddhist temples than in the dreamy fog of sleeplessness? Chiang Mai has over three hundred wats within the somewhat sprawling city limits, most of them reasonably modern and, in my opinion, not worth visiting. I narrowed the field to three, which I figured was a nice round one percent. \x0D\x0A <a href="/2006/jan/11/you-and-i-are-disappearing/">Read it »<\/a><\/p><\/div>' + marker_you_and.info_window_content = '<div class="infowin"><h4>You and I Are Disappearing<\/h4><span class="date blok">January 11, 2006 (Chang Mai, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/changmaiumong.jpg" height="100" alt="You and I Are Disappearing" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The all night bus reached Chiang Mai well past dawn, the city already beginning to stir. I considered trying to nap, but in the end decided to explore the town. What better way to see Buddhist temples than in the dreamy fog of sleeplessness? Chiang Mai has over three hundred wats within the somewhat sprawling city limits, most of them reasonably modern and, in my opinion, not worth visiting. I narrowed the field to three, which I figured was a nice round one percent. \u000D\u000A <a href="/2006/jan/11/you-and-i-are-disappearing/">Read it »<\/a><\/p><\/div>' marker_you_and.bindInfoWindowHtml(marker_you_and.info_window_content, {maxWidth:400}); GEvent.addListener(marker_you_and, "click", function() { map.panTo(point_you_and, 2); @@ -686,7 +686,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_buddha_on = new GMarker(point_buddha_on, markerOptions); map.addOverlay(marker_buddha_on); - marker_buddha_on.info_window_content = '<div class="infowin"><h4>Buddha on the Bounty<\/h4><span class="date blok">January 5, 2006 (Bangkok, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/jimthompsonhouse.jpg" height="100" alt="Buddha on the Bounty" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The house Jim Thompson left behind in Bangkok is gorgeous, but the real charm is the garden and its orchids. I wandered around the gardens which really aren\x27t that large for some time and then found a bench near a collection of orchids, where I sat for the better part of an hour, occasionally taking a photograph or two, but mostly thinking about how human orchids are. <a href="/2006/jan/05/buddha-bounty/">Read it »<\/a><\/p><\/div>' + marker_buddha_on.info_window_content = '<div class="infowin"><h4>Buddha on the Bounty<\/h4><span class="date blok">January 5, 2006 (Bangkok, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/jimthompsonhouse.jpg" height="100" alt="Buddha on the Bounty" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The house Jim Thompson left behind in Bangkok is gorgeous, but the real charm is the garden and its orchids. I wandered around the gardens which really aren\u0027t that large for some time and then found a bench near a collection of orchids, where I sat for the better part of an hour, occasionally taking a photograph or two, but mostly thinking about how human orchids are. <a href="/2006/jan/05/buddha-bounty/">Read it »<\/a><\/p><\/div>' marker_buddha_on.bindInfoWindowHtml(marker_buddha_on.info_window_content, {maxWidth:400}); GEvent.addListener(marker_buddha_on, "click", function() { map.panTo(point_buddha_on, 2); @@ -697,7 +697,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_brink_of = new GMarker(point_brink_of, markerOptions); map.addOverlay(marker_brink_of); - marker_brink_of.info_window_content = '<div class="infowin"><h4>Brink of the Clouds<\/h4><span class="date blok">January 3, 2006 (Bangkok, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/bangkokfrombaiyoke.jpg" height="100" alt="Brink of the Clouds" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>\x22The city is a cathedral\x22 writes James Salter, \x22its scent is dreams.\x22 Salter may have been referring to New York, but his words ring true in Bangkok. And the best place to feel it at night is on the river or from the top of the Baiyoke Sky Hotel \x26mdash\x3B where a circular, revolving observation deck offers 360\x26deg\x3B views of the Bangkok nightscape. <a href="/2006/jan/03/brink-clouds/">Read it »<\/a><\/p><\/div>' + marker_brink_of.info_window_content = '<div class="infowin"><h4>Brink of the Clouds<\/h4><span class="date blok">January 3, 2006 (Bangkok, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/bangkokfrombaiyoke.jpg" height="100" alt="Brink of the Clouds" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>\u0022The city is a cathedral\u0022 writes James Salter, \u0022its scent is dreams.\u0022 Salter may have been referring to New York, but his words ring true in Bangkok. And the best place to feel it at night is on the river or from the top of the Baiyoke Sky Hotel \u0026mdash\u003B where a circular, revolving observation deck offers 360\u0026deg\u003B views of the Bangkok nightscape. <a href="/2006/jan/03/brink-clouds/">Read it »<\/a><\/p><\/div>' marker_brink_of.bindInfoWindowHtml(marker_brink_of.info_window_content, {maxWidth:400}); GEvent.addListener(marker_brink_of, "click", function() { map.panTo(point_brink_of, 2); @@ -708,7 +708,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_are_you = new GMarker(point_are_you, markerOptions); map.addOverlay(marker_are_you); - marker_are_you.info_window_content = '<div class="infowin"><h4>Are You Amplified to Rock?<\/h4><span class="date blok">January 1, 2006 (Bangkok, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/bangkokriver.jpg" height="100" alt="Are You Amplified to Rock?" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\x27s a new year, are you amplified to rock? Ready, set, go. <a href="/2006/jan/01/are-you-amplified-rock/">Read it »<\/a><\/p><\/div>' + marker_are_you.info_window_content = '<div class="infowin"><h4>Are You Amplified to Rock?<\/h4><span class="date blok">January 1, 2006 (Bangkok, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/bangkokriver.jpg" height="100" alt="Are You Amplified to Rock?" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>It\u0027s a new year, are you amplified to rock? Ready, set, go. <a href="/2006/jan/01/are-you-amplified-rock/">Read it »<\/a><\/p><\/div>' marker_are_you.bindInfoWindowHtml(marker_are_you.info_window_content, {maxWidth:400}); GEvent.addListener(marker_are_you, "click", function() { map.panTo(point_are_you, 2); @@ -719,7 +719,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_merry_christmas = new GMarker(point_merry_christmas, markerOptions); map.addOverlay(marker_merry_christmas); - marker_merry_christmas.info_window_content = '<div class="infowin"><h4>Merry Christmas 2005<\/h4><span class="date blok">December 25, 2005 (Bangkok, Thailand)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/bangkokfort.jpg" height="100" alt="Merry Christmas 2005" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Seasons Greeting from luxagraf. I\x27m in Bangkok, Thailand at the moment. I am taking a short break from traveling to do a little working so I don\x27t have much to report. I\x27ve seen the two big temples down in the Khaosan Rd area, but otherwise I\x27ve been trying to live an ordinary life in Bangkok, if such a thing is possible. <a href="/2005/dec/25/merry-christmas-2005/">Read it »<\/a><\/p><\/div>' + marker_merry_christmas.info_window_content = '<div class="infowin"><h4>Merry Christmas 2005<\/h4><span class="date blok">December 25, 2005 (Bangkok, Thailand)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/bangkokfort.jpg" height="100" alt="Merry Christmas 2005" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Seasons Greeting from luxagraf. I\u0027m in Bangkok, Thailand at the moment. I am taking a short break from traveling to do a little working so I don\u0027t have much to report. I\u0027ve seen the two big temples down in the Khaosan Rd area, but otherwise I\u0027ve been trying to live an ordinary life in Bangkok, if such a thing is possible. <a href="/2005/dec/25/merry-christmas-2005/">Read it »<\/a><\/p><\/div>' marker_merry_christmas.bindInfoWindowHtml(marker_merry_christmas.info_window_content, {maxWidth:400}); GEvent.addListener(marker_merry_christmas, "click", function() { map.panTo(point_merry_christmas, 2); @@ -730,7 +730,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_sunset_over = new GMarker(point_sunset_over, markerOptions); map.addOverlay(marker_sunset_over); - marker_sunset_over.info_window_content = '<div class="infowin"><h4>Sunset Over the Himalayas<\/h4><span class="date blok">December 17, 2005 (Pokhara, Nepal)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/pokharaboat.jpg" height="100" alt="Sunset Over the Himalayas" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>After about forty\x2Dfive minutes of paddling I reached a point where the views of the Annapurna range were, in the words of an Englishman I met in Katmandu, \x22gob smacking gorgeous.\x22 I put down the paddle and moved to the center of the boat where the benches were wider and, using my bag a cushion, lay back against the gunnel and hung my feet over the opposite side so that they just skimmed the surface of the chilly water. <a href="/2005/dec/17/sunset-over-himalayas/">Read it »<\/a><\/p><\/div>' + marker_sunset_over.info_window_content = '<div class="infowin"><h4>Sunset Over the Himalayas<\/h4><span class="date blok">December 17, 2005 (Pokhara, Nepal)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/pokharaboat.jpg" height="100" alt="Sunset Over the Himalayas" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>After about forty\u002Dfive minutes of paddling I reached a point where the views of the Annapurna range were, in the words of an Englishman I met in Katmandu, \u0022gob smacking gorgeous.\u0022 I put down the paddle and moved to the center of the boat where the benches were wider and, using my bag a cushion, lay back against the gunnel and hung my feet over the opposite side so that they just skimmed the surface of the chilly water. <a href="/2005/dec/17/sunset-over-himalayas/">Read it »<\/a><\/p><\/div>' marker_sunset_over.bindInfoWindowHtml(marker_sunset_over.info_window_content, {maxWidth:400}); GEvent.addListener(marker_sunset_over, "click", function() { map.panTo(point_sunset_over, 2); @@ -741,7 +741,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_pashupatinath = new GMarker(point_pashupatinath, markerOptions); map.addOverlay(marker_pashupatinath); - marker_pashupatinath.info_window_content = '<div class="infowin"><h4>Pashupatinath<\/h4><span class="date blok">December 15, 2005 (Pashupatinath, Nepal)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/nepalburninggahts.jpg" height="100" alt="Pashupatinath" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Nestled on a hillside beside the Bagmati River, Pashupatinath is one of the holiest sites in the world for Hindus, second only to Varanasi in India. Pashupatinath consists of a large temple which is open only to Hindus, surrounded by a number of smaller shrines and then down on the banks of the Bagmati are the burning ghats where bodies are cremated. <a href="/2005/dec/15/pashupatinath/">Read it »<\/a><\/p><\/div>' + marker_pashupatinath.info_window_content = '<div class="infowin"><h4>Pashupatinath<\/h4><span class="date blok">December 15, 2005 (Pashupatinath, Nepal)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/nepalburninggahts.jpg" height="100" alt="Pashupatinath" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Nestled on a hillside beside the Bagmati River, Pashupatinath is one of the holiest sites in the world for Hindus, second only to Varanasi in India. Pashupatinath consists of a large temple which is open only to Hindus, surrounded by a number of smaller shrines and then down on the banks of the Bagmati are the burning ghats where bodies are cremated. <a href="/2005/dec/15/pashupatinath/">Read it »<\/a><\/p><\/div>' marker_pashupatinath.bindInfoWindowHtml(marker_pashupatinath.info_window_content, {maxWidth:400}); GEvent.addListener(marker_pashupatinath, "click", function() { map.panTo(point_pashupatinath, 2); @@ -752,7 +752,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_durbar_square = new GMarker(point_durbar_square, markerOptions); map.addOverlay(marker_durbar_square); - marker_durbar_square.info_window_content = '<div class="infowin"><h4>Durbar Square Kathmandu<\/h4><span class="date blok">December 15, 2005 (Kathmandu, Nepal)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/durbarsquare.jpg" height="100" alt="Durbar Square Kathmandu" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>After saturating myself with the streets of Thamel I went on a longer excursion down to Durbar Square to see the various pagodas, temples and the old palace. The palace itself no longer houses the King, but is still used for coronations and ceremonies and Durbar Square is still very much the hub of Katmandu. <a href="/2005/dec/15/durbar-square-kathmandu/">Read it »<\/a><\/p><\/div>' + marker_durbar_square.info_window_content = '<div class="infowin"><h4>Durbar Square Kathmandu<\/h4><span class="date blok">December 15, 2005 (Kathmandu, Nepal)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/durbarsquare.jpg" height="100" alt="Durbar Square Kathmandu" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>After saturating myself with the streets of Thamel I went on a longer excursion down to Durbar Square to see the various pagodas, temples and the old palace. The palace itself no longer houses the King, but is still used for coronations and ceremonies and Durbar Square is still very much the hub of Katmandu. <a href="/2005/dec/15/durbar-square-kathmandu/">Read it »<\/a><\/p><\/div>' marker_durbar_square.bindInfoWindowHtml(marker_durbar_square.info_window_content, {maxWidth:400}); GEvent.addListener(marker_durbar_square, "click", function() { map.panTo(point_durbar_square, 2); @@ -763,7 +763,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_goodbye_india = new GMarker(point_goodbye_india, markerOptions); map.addOverlay(marker_goodbye_india); - marker_goodbye_india.info_window_content = '<div class="infowin"><h4>Goodbye India<\/h4><span class="date blok">December 10, 2005 (Delhi, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/indiadelhi.jpg" height="100" alt="Goodbye India" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I have taken almost 750 photos and traveled nearly 4000 km (2500 miles) in India, the vast majority of it by train. I have seen everything from depressing squalor to majestic palaces and yet I still feel as if I have hardly scratched the surface. I can\x27t think of another and certainly have never been to a country with the kind of geographic and ethnic diversity of India. <a href="/2005/dec/10/goodbye-india/">Read it »<\/a><\/p><\/div>' + marker_goodbye_india.info_window_content = '<div class="infowin"><h4>Goodbye India<\/h4><span class="date blok">December 10, 2005 (Delhi, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/indiadelhi.jpg" height="100" alt="Goodbye India" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I have taken almost 750 photos and traveled nearly 4000 km (2500 miles) in India, the vast majority of it by train. I have seen everything from depressing squalor to majestic palaces and yet I still feel as if I have hardly scratched the surface. I can\u0027t think of another and certainly have never been to a country with the kind of geographic and ethnic diversity of India. <a href="/2005/dec/10/goodbye-india/">Read it »<\/a><\/p><\/div>' marker_goodbye_india.bindInfoWindowHtml(marker_goodbye_india.info_window_content, {maxWidth:400}); GEvent.addListener(marker_goodbye_india, "click", function() { map.panTo(point_goodbye_india, 2); @@ -774,7 +774,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_taj = new GMarker(point_the_taj, markerOptions); map.addOverlay(marker_the_taj); - marker_the_taj.info_window_content = '<div class="infowin"><h4>The Taj Express<\/h4><span class="date blok">December 9, 2005 (Agra, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/tajmahal.jpg" height="100" alt="The Taj Express" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Taj Mahal is one of the Seven Wonders of the World, and, given the level of hype I was fully prepared to be underwhelmed, but I was wrong. I have never in my life seen anything so extravagant, elegant and colossal. The Taj Mahal seems mythically, spiritually, as well as architecturally, to have risen from nowhere, without equal or context. <a href="/2005/dec/09/taj-express/">Read it »<\/a><\/p><\/div>' + marker_the_taj.info_window_content = '<div class="infowin"><h4>The Taj Express<\/h4><span class="date blok">December 9, 2005 (Agra, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/tajmahal.jpg" height="100" alt="The Taj Express" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Taj Mahal is one of the Seven Wonders of the World, and, given the level of hype I was fully prepared to be underwhelmed, but I was wrong. I have never in my life seen anything so extravagant, elegant and colossal. The Taj Mahal seems mythically, spiritually, as well as architecturally, to have risen from nowhere, without equal or context. <a href="/2005/dec/09/taj-express/">Read it »<\/a><\/p><\/div>' marker_the_taj.bindInfoWindowHtml(marker_the_taj.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_taj, "click", function() { map.panTo(point_the_taj, 2); @@ -785,7 +785,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_on_a = new GMarker(point_on_a, markerOptions); map.addOverlay(marker_on_a); - marker_on_a.info_window_content = '<div class="infowin"><h4>On a Camel With No Name<\/h4><span class="date blok">December 5, 2005 (Thar Desert, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/cameltrek.jpg" height="100" alt="On a Camel With No Name" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Thar Desert is a bewitching if stark place. It reminded me of areas of the Great Basin between Las Vegas and St. George, Utah. Twigging mesquite\x2Dlike trees, bluish gray bushes resembling creosote, a very large bush that resembled a Palo Verde tree and grew in impenetrable clumps, and, strangely, only one species of cactus and not a whole lot of them. <a href="/2005/dec/05/camel-no-name/">Read it »<\/a><\/p><\/div>' + marker_on_a.info_window_content = '<div class="infowin"><h4>On a Camel With No Name<\/h4><span class="date blok">December 5, 2005 (Thar Desert, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/cameltrek.jpg" height="100" alt="On a Camel With No Name" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Thar Desert is a bewitching if stark place. It reminded me of areas of the Great Basin between Las Vegas and St. George, Utah. Twigging mesquite\u002Dlike trees, bluish gray bushes resembling creosote, a very large bush that resembled a Palo Verde tree and grew in impenetrable clumps, and, strangely, only one species of cactus and not a whole lot of them. <a href="/2005/dec/05/camel-no-name/">Read it »<\/a><\/p><\/div>' marker_on_a.bindInfoWindowHtml(marker_on_a.info_window_content, {maxWidth:400}); GEvent.addListener(marker_on_a, "click", function() { map.panTo(point_on_a, 2); @@ -796,7 +796,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_majestic = new GMarker(point_the_majestic, markerOptions); map.addOverlay(marker_the_majestic); - marker_the_majestic.info_window_content = '<div class="infowin"><h4>The Majestic Fort<\/h4><span class="date blok">December 2, 2005 (Jodhpur, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/jodhpurfort.jpg" height="100" alt="The Majestic Fort" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The next day I hopped in a rickshaw and headed up to tour Meherangarh, or the Majestic Fort as it\x27s known in English. As its English name indicates, it is indeed perched majestically atop the only hill around, and seems not so much built on a hill as to have naturally risen out the very rocks that form the mesa on which it rests. The outer wall encloses some of the sturdiest and most impressive ramparts I\x27ve seen in India or anywhere else. <a href="/2005/dec/02/majestic-fort/">Read it »<\/a><\/p><\/div>' + marker_the_majestic.info_window_content = '<div class="infowin"><h4>The Majestic Fort<\/h4><span class="date blok">December 2, 2005 (Jodhpur, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/jodhpurfort.jpg" height="100" alt="The Majestic Fort" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The next day I hopped in a rickshaw and headed up to tour Meherangarh, or the Majestic Fort as it\u0027s known in English. As its English name indicates, it is indeed perched majestically atop the only hill around, and seems not so much built on a hill as to have naturally risen out the very rocks that form the mesa on which it rests. The outer wall encloses some of the sturdiest and most impressive ramparts I\u0027ve seen in India or anywhere else. <a href="/2005/dec/02/majestic-fort/">Read it »<\/a><\/p><\/div>' marker_the_majestic.bindInfoWindowHtml(marker_the_majestic.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_majestic, "click", function() { map.panTo(point_the_majestic, 2); @@ -807,7 +807,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_around_udaipur = new GMarker(point_around_udaipur, markerOptions); map.addOverlay(marker_around_udaipur); - marker_around_udaipur.info_window_content = '<div class="infowin"><h4>Around Udaipur<\/h4><span class="date blok">November 30, 2005 (Udiapur, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/shiplogram.jpg" height="100" alt="Around Udaipur" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Just out of Udaipur is a government sponsored \x22artist colony\x22 for various cultures from the five nearby states, Rajasthan, Gujarat, Karnataka, Goa and Madhya Pradesh. On one hand Shilpogram is a wonderful idea on the part of the government, but on the other hand the \x22artists colony\x22 is slightly creepy. Amidst displays of typical tribal life there were artists and craftsmen and women hawking their wares along with dancers and musicians performing traditional songs. The whole thing had the feel of a living museum, or, for the creepy angle \x26mdash\x3B human zoo. <a href="/2005/nov/30/around-udaipur/">Read it »<\/a><\/p><\/div>' + marker_around_udaipur.info_window_content = '<div class="infowin"><h4>Around Udaipur<\/h4><span class="date blok">November 30, 2005 (Udiapur, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/shiplogram.jpg" height="100" alt="Around Udaipur" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Just out of Udaipur is a government sponsored \u0022artist colony\u0022 for various cultures from the five nearby states, Rajasthan, Gujarat, Karnataka, Goa and Madhya Pradesh. On one hand Shilpogram is a wonderful idea on the part of the government, but on the other hand the \u0022artists colony\u0022 is slightly creepy. Amidst displays of typical tribal life there were artists and craftsmen and women hawking their wares along with dancers and musicians performing traditional songs. The whole thing had the feel of a living museum, or, for the creepy angle \u0026mdash\u003B human zoo. <a href="/2005/nov/30/around-udaipur/">Read it »<\/a><\/p><\/div>' marker_around_udaipur.bindInfoWindowHtml(marker_around_udaipur.info_window_content, {maxWidth:400}); GEvent.addListener(marker_around_udaipur, "click", function() { map.panTo(point_around_udaipur, 2); @@ -818,7 +818,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_monsoon = new GMarker(point_the_monsoon, markerOptions); map.addOverlay(marker_the_monsoon); - marker_the_monsoon.info_window_content = '<div class="infowin"><h4>The Monsoon Palace<\/h4><span class="date blok">November 29, 2005 (Udiapur, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/monsoonpalace.jpg" height="100" alt="The Monsoon Palace" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We started out in the early evening quickly leaving behind Udaipur and its increasing urban sprawl. The road to the Monsoon Palace passes through the Sajjan Garh Nature Preserve and there was a sudden and dramatic drop in temperature, but then the road climbed out of the hollow and the temperature jumped back up to comfortable as we began to climb the mountain in a series of hairpin switchbacks. As the sun slowly slunk behind the mountain range to the west the balconies and balustrades of the Monsoon Palace took on an increasingly orange hue. <a href="/2005/nov/29/monsoon-palace/">Read it »<\/a><\/p><\/div>' + marker_the_monsoon.info_window_content = '<div class="infowin"><h4>The Monsoon Palace<\/h4><span class="date blok">November 29, 2005 (Udiapur, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/monsoonpalace.jpg" height="100" alt="The Monsoon Palace" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>We started out in the early evening quickly leaving behind Udaipur and its increasing urban sprawl. The road to the Monsoon Palace passes through the Sajjan Garh Nature Preserve and there was a sudden and dramatic drop in temperature, but then the road climbed out of the hollow and the temperature jumped back up to comfortable as we began to climb the mountain in a series of hairpin switchbacks. As the sun slowly slunk behind the mountain range to the west the balconies and balustrades of the Monsoon Palace took on an increasingly orange hue. <a href="/2005/nov/29/monsoon-palace/">Read it »<\/a><\/p><\/div>' marker_the_monsoon.bindInfoWindowHtml(marker_the_monsoon.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_monsoon, "click", function() { map.panTo(point_the_monsoon, 2); @@ -829,7 +829,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_city = new GMarker(point_the_city, markerOptions); map.addOverlay(marker_the_city); - marker_the_city.info_window_content = '<div class="infowin"><h4>The City Palace<\/h4><span class="date blok">November 28, 2005 (Udiapur, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/citypalaceudaipur.jpg" height="100" alt="The City Palace" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I spent some time sitting in the inner gardens of the City Place, listening to rustling trees and the various guides bringing small groups of western and Indian tourists through the garden. In the center of the hanging gardens was the kings, extremely oversized bath, which reminded me of children\x27s book that I once gave to a friend\x27s daughter\x3B it was a massively oversized and lavishly illustrated book that told the story of a king who refused to get out of the bath and instead made his ministers, advisors, cooks and even his wife conduct business by getting in the bath with him. <a href="/2005/nov/28/city-palace/">Read it »<\/a><\/p><\/div>' + marker_the_city.info_window_content = '<div class="infowin"><h4>The City Palace<\/h4><span class="date blok">November 28, 2005 (Udiapur, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/citypalaceudaipur.jpg" height="100" alt="The City Palace" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I spent some time sitting in the inner gardens of the City Place, listening to rustling trees and the various guides bringing small groups of western and Indian tourists through the garden. In the center of the hanging gardens was the kings, extremely oversized bath, which reminded me of children\u0027s book that I once gave to a friend\u0027s daughter\u003B it was a massively oversized and lavishly illustrated book that told the story of a king who refused to get out of the bath and instead made his ministers, advisors, cooks and even his wife conduct business by getting in the bath with him. <a href="/2005/nov/28/city-palace/">Read it »<\/a><\/p><\/div>' marker_the_city.bindInfoWindowHtml(marker_the_city.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_city, "click", function() { map.panTo(point_the_city, 2); @@ -840,7 +840,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_living_in = new GMarker(point_living_in, markerOptions); map.addOverlay(marker_living_in); - marker_living_in.info_window_content = '<div class="infowin"><h4>Living in Airport Terminals<\/h4><span class="date blok">November 27, 2005 (Ahmedabad, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/ceilingfanindia.jpg" height="100" alt="Living in Airport Terminals" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Airport terminals are fast becoming my favorite part of traveling. When you stop and observe them closely as I have been forced to do on this trip, terminals are actually quite beautiful weird places. Terminals inhabit a unique space in the architecture of humanity, perhaps the strangest of all spaces we have created\x3B a space that is itself only a boundary that delineates the border between what was and what will be without leaving any space at all for what is. <a href="/2005/nov/27/living-airport-terminals/">Read it »<\/a><\/p><\/div>' + marker_living_in.info_window_content = '<div class="infowin"><h4>Living in Airport Terminals<\/h4><span class="date blok">November 27, 2005 (Ahmedabad, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/ceilingfanindia.jpg" height="100" alt="Living in Airport Terminals" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Airport terminals are fast becoming my favorite part of traveling. When you stop and observe them closely as I have been forced to do on this trip, terminals are actually quite beautiful weird places. Terminals inhabit a unique space in the architecture of humanity, perhaps the strangest of all spaces we have created\u003B a space that is itself only a boundary that delineates the border between what was and what will be without leaving any space at all for what is. <a href="/2005/nov/27/living-airport-terminals/">Read it »<\/a><\/p><\/div>' marker_living_in.bindInfoWindowHtml(marker_living_in.info_window_content, {maxWidth:400}); GEvent.addListener(marker_living_in, "click", function() { map.panTo(point_living_in, 2); @@ -851,7 +851,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_anjuna_market = new GMarker(point_anjuna_market, markerOptions); map.addOverlay(marker_anjuna_market); - marker_anjuna_market.info_window_content = '<div class="infowin"><h4>Anjuna Market<\/h4><span class="date blok">November 23, 2005 (Anjuna Beach, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/anjunabeachmarket.jpg" height="100" alt="Anjuna Market" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Earlier today I caught a bus up to the Anjuna Flea Market and can now tell you for certain that old hippies do not die, they simply move to Goa. The flea market was quite a spectacle\x3B riots of color at every turn and more silver jewelry than you could shake a stick at. <a href="/2005/nov/23/anjuna-market/">Read it »<\/a><\/p><\/div>' + marker_anjuna_market.info_window_content = '<div class="infowin"><h4>Anjuna Market<\/h4><span class="date blok">November 23, 2005 (Anjuna Beach, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/anjunabeachmarket.jpg" height="100" alt="Anjuna Market" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Earlier today I caught a bus up to the Anjuna Flea Market and can now tell you for certain that old hippies do not die, they simply move to Goa. The flea market was quite a spectacle\u003B riots of color at every turn and more silver jewelry than you could shake a stick at. <a href="/2005/nov/23/anjuna-market/">Read it »<\/a><\/p><\/div>' marker_anjuna_market.bindInfoWindowHtml(marker_anjuna_market.info_window_content, {maxWidth:400}); GEvent.addListener(marker_anjuna_market, "click", function() { map.panTo(point_anjuna_market, 2); @@ -862,7 +862,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_fish_story = new GMarker(point_fish_story, markerOptions); map.addOverlay(marker_fish_story); - marker_fish_story.info_window_content = '<div class="infowin"><h4>Fish Story<\/h4><span class="date blok">November 19, 2005 (Colva Beach, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/colvabeach.jpg" height="100" alt="Fish Story" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Arabian Sea is warm and the sand sucks at your feet when you walk, schools of tiny fish dart and disappear into each receding wave. In the morning the water is nearly glassy and the beach slopes off so slowly one can walk out at least 200 meters and be only waist deep. <a href="/2005/nov/19/fish-story/">Read it »<\/a><\/p><\/div>' + marker_fish_story.info_window_content = '<div class="infowin"><h4>Fish Story<\/h4><span class="date blok">November 19, 2005 (Colva Beach, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/colvabeach.jpg" height="100" alt="Fish Story" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The Arabian Sea is warm and the sand sucks at your feet when you walk, schools of tiny fish dart and disappear into each receding wave. In the morning the water is nearly glassy and the beach slopes off so slowly one can walk out at least 200 meters and be only waist deep. <a href="/2005/nov/19/fish-story/">Read it »<\/a><\/p><\/div>' marker_fish_story.bindInfoWindowHtml(marker_fish_story.info_window_content, {maxWidth:400}); GEvent.addListener(marker_fish_story, "click", function() { map.panTo(point_fish_story, 2); @@ -873,7 +873,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_backwaters = new GMarker(point_the_backwaters, markerOptions); map.addOverlay(marker_the_backwaters); - marker_the_backwaters.info_window_content = '<div class="infowin"><h4>The Backwaters of Kerala<\/h4><span class="date blok">November 14, 2005 (Kerala Backwaters, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/keralabackwater.jpg" height="100" alt="The Backwaters of Kerala" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The guide showed us Tamarind trees, coconut palms, lemon trees, vanilla vine, plantain trees and countless other shrubs and bushes whose names I have already forgotten. The most fascinating was a plant that produces a fruit something like a miniature mango that contains cyanide and which, as our guide informed us, is cultivated mainly to commit suicide with \x26mdash\x3B as if it was no big deal and everyone is at least occasionally tempted to each the killer mango. <a href="/2005/nov/14/backwaters-kerala/">Read it »<\/a><\/p><\/div>' + marker_the_backwaters.info_window_content = '<div class="infowin"><h4>The Backwaters of Kerala<\/h4><span class="date blok">November 14, 2005 (Kerala Backwaters, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/keralabackwater.jpg" height="100" alt="The Backwaters of Kerala" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The guide showed us Tamarind trees, coconut palms, lemon trees, vanilla vine, plantain trees and countless other shrubs and bushes whose names I have already forgotten. The most fascinating was a plant that produces a fruit something like a miniature mango that contains cyanide and which, as our guide informed us, is cultivated mainly to commit suicide with \u0026mdash\u003B as if it was no big deal and everyone is at least occasionally tempted to each the killer mango. <a href="/2005/nov/14/backwaters-kerala/">Read it »<\/a><\/p><\/div>' marker_the_backwaters.bindInfoWindowHtml(marker_the_backwaters.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_backwaters, "click", function() { map.panTo(point_the_backwaters, 2); @@ -884,7 +884,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_vasco_de = new GMarker(point_vasco_de, markerOptions); map.addOverlay(marker_vasco_de); - marker_vasco_de.info_window_content = '<div class="infowin"><h4>Vasco de Gama Exhumed<\/h4><span class="date blok">November 10, 2005 (Fort Kochi, India)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/fortcochin.jpg" height="100" alt="Vasco de Gama Exhumed" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Fort Cochin is curious collision of cultures \x26mdash\x3B Chinese, India and even Portuguese. Many of the obviously older buildings are of a distinctly Iberian\x2Dstyle \x26mdash\x3B moss covered, adobe\x2Dcolored arches abound. There is graveyard just down the road with a tombstone that bears the name Vasco de Gama, who died and was buried here for fourteen years before being moved to Lisbon (there we go again, more Europeans digging up and moving the dead). <a href="/2005/nov/10/vasco-de-gama-exhumed/">Read it »<\/a><\/p><\/div>' + marker_vasco_de.info_window_content = '<div class="infowin"><h4>Vasco de Gama Exhumed<\/h4><span class="date blok">November 10, 2005 (Fort Kochi, India)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/fortcochin.jpg" height="100" alt="Vasco de Gama Exhumed" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Fort Cochin is curious collision of cultures \u0026mdash\u003B Chinese, India and even Portuguese. Many of the obviously older buildings are of a distinctly Iberian\u002Dstyle \u0026mdash\u003B moss covered, adobe\u002Dcolored arches abound. There is graveyard just down the road with a tombstone that bears the name Vasco de Gama, who died and was buried here for fourteen years before being moved to Lisbon (there we go again, more Europeans digging up and moving the dead). <a href="/2005/nov/10/vasco-de-gama-exhumed/">Read it »<\/a><\/p><\/div>' marker_vasco_de.bindInfoWindowHtml(marker_vasco_de.info_window_content, {maxWidth:400}); GEvent.addListener(marker_vasco_de, "click", function() { map.panTo(point_vasco_de, 2); @@ -895,7 +895,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_riots_iraqi = new GMarker(point_riots_iraqi, markerOptions); map.addOverlay(marker_riots_iraqi); - marker_riots_iraqi.info_window_content = '<div class="infowin"><h4>Riots, Iraqi Restaurants, Goodbye Seine<\/h4><span class="date blok">November 8, 2005 (Paris, France)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/republique.jpg" height="100" alt="Riots, Iraqi Restaurants, Goodbye Seine" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Well it\x27s my last night here in Paris and I\x27ve chosen to return to the best restaurant we\x27ve been to so far, an Iraqi restaurant in a Marais. I am using all my willpower right now to avoid having a political outburst re the quality of Iraqi food versus the intelligence of George Bush etc etc. I\x27m traveling\x3B I don\x27t want to get into politics except to say that my dislike for the current El Presidente was no small factor in my decision to go abroad. <a href="/2005/nov/08/riots-iraqi-restaurants-goodbye-seine/">Read it »<\/a><\/p><\/div>' + marker_riots_iraqi.info_window_content = '<div class="infowin"><h4>Riots, Iraqi Restaurants, Goodbye Seine<\/h4><span class="date blok">November 8, 2005 (Paris, France)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/republique.jpg" height="100" alt="Riots, Iraqi Restaurants, Goodbye Seine" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Well it\u0027s my last night here in Paris and I\u0027ve chosen to return to the best restaurant we\u0027ve been to so far, an Iraqi restaurant in a Marais. I am using all my willpower right now to avoid having a political outburst re the quality of Iraqi food versus the intelligence of George Bush etc etc. I\u0027m traveling\u003B I don\u0027t want to get into politics except to say that my dislike for the current El Presidente was no small factor in my decision to go abroad. <a href="/2005/nov/08/riots-iraqi-restaurants-goodbye-seine/">Read it »<\/a><\/p><\/div>' marker_riots_iraqi.bindInfoWindowHtml(marker_riots_iraqi.info_window_content, {maxWidth:400}); GEvent.addListener(marker_riots_iraqi, "click", function() { map.panTo(point_riots_iraqi, 2); @@ -906,7 +906,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_bury_your = new GMarker(point_bury_your, markerOptions); map.addOverlay(marker_bury_your); - marker_bury_your.info_window_content = '<div class="infowin"><h4>Bury Your Dead<\/h4><span class="date blok">November 6, 2005 (Paris, France)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/pariscatacombs.jpg" height="100" alt="Bury Your Dead" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I would like to say that the catacombs of Paris had some spectacular effect on me seeing that I strolled through human remains, skulls and femurs mainly, \x22decoratively arranged,\x22 but the truth is, after you get over the initial shock of seeing a skull, well, it turns out you can get adjusted to just about anything. Maybe that in and off itself is the scary part. <a href="/2005/nov/06/bury-your-dead/">Read it »<\/a><\/p><\/div>' + marker_bury_your.info_window_content = '<div class="infowin"><h4>Bury Your Dead<\/h4><span class="date blok">November 6, 2005 (Paris, France)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/pariscatacombs.jpg" height="100" alt="Bury Your Dead" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I would like to say that the catacombs of Paris had some spectacular effect on me seeing that I strolled through human remains, skulls and femurs mainly, \u0022decoratively arranged,\u0022 but the truth is, after you get over the initial shock of seeing a skull, well, it turns out you can get adjusted to just about anything. Maybe that in and off itself is the scary part. <a href="/2005/nov/06/bury-your-dead/">Read it »<\/a><\/p><\/div>' marker_bury_your.bindInfoWindowHtml(marker_bury_your.info_window_content, {maxWidth:400}); GEvent.addListener(marker_bury_your, "click", function() { map.panTo(point_bury_your, 2); @@ -917,7 +917,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_houses = new GMarker(point_the_houses, markerOptions); map.addOverlay(marker_the_houses); - marker_the_houses.info_window_content = '<div class="infowin"><h4>The Houses We Live In<\/h4><span class="date blok">November 1, 2005 (Paris, France)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/pariscityscape.jpg" height="100" alt="The Houses We Live In" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I\x27ve been thinking the last couple of days about something Bill\x27s dad said to me before I left. I\x27m paraphrasing here since I don\x27t remember the exact phrasing he used, but something to the effect of \x22people are essentially the same everywhere, they just build their houses differently.\x22 Indeed, Parisian architecture is completely unlike anything in America. Perhaps more than any other single element, architecture reflects culture and the ideas of the people that make up culture. <a href="/2005/nov/01/houses-we-live/">Read it »<\/a><\/p><\/div>' + marker_the_houses.info_window_content = '<div class="infowin"><h4>The Houses We Live In<\/h4><span class="date blok">November 1, 2005 (Paris, France)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/pariscityscape.jpg" height="100" alt="The Houses We Live In" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I\u0027ve been thinking the last couple of days about something Bill\u0027s dad said to me before I left. I\u0027m paraphrasing here since I don\u0027t remember the exact phrasing he used, but something to the effect of \u0022people are essentially the same everywhere, they just build their houses differently.\u0022 Indeed, Parisian architecture is completely unlike anything in America. Perhaps more than any other single element, architecture reflects culture and the ideas of the people that make up culture. <a href="/2005/nov/01/houses-we-live/">Read it »<\/a><\/p><\/div>' marker_the_houses.bindInfoWindowHtml(marker_the_houses.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_houses, "click", function() { map.panTo(point_the_houses, 2); @@ -928,7 +928,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_sainte_chapelle = new GMarker(point_sainte_chapelle, markerOptions); map.addOverlay(marker_sainte_chapelle); - marker_sainte_chapelle.info_window_content = '<div class="infowin"><h4>Sainte Chapelle<\/h4><span class="date blok">October 28, 2005 (Paris, France)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/saintechapelle.jpg" height="100" alt="Sainte Chapelle" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Sainte Chapelle was interesting to see after the modern, conceptual art stuff at the Pompidou, rather than simple stained glass, Sainte Chapelle felt quite conceptual. In a sense the entire Bible (i.e. all history from that perspective) is unfolding simultaneously, quite a so\x2Dcalled post\x2Dmodern idea if you think about it. And yet it was conceived and executed over 800 years ago. Kind of kicks a lot pretentious modern art in its collective ass. <a href="/2005/oct/28/sainte-chapelle/">Read it »<\/a><\/p><\/div>' + marker_sainte_chapelle.info_window_content = '<div class="infowin"><h4>Sainte Chapelle<\/h4><span class="date blok">October 28, 2005 (Paris, France)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/saintechapelle.jpg" height="100" alt="Sainte Chapelle" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Sainte Chapelle was interesting to see after the modern, conceptual art stuff at the Pompidou, rather than simple stained glass, Sainte Chapelle felt quite conceptual. In a sense the entire Bible (i.e. all history from that perspective) is unfolding simultaneously, quite a so\u002Dcalled post\u002Dmodern idea if you think about it. And yet it was conceived and executed over 800 years ago. Kind of kicks a lot pretentious modern art in its collective ass. <a href="/2005/oct/28/sainte-chapelle/">Read it »<\/a><\/p><\/div>' marker_sainte_chapelle.bindInfoWindowHtml(marker_sainte_chapelle.info_window_content, {maxWidth:400}); GEvent.addListener(marker_sainte_chapelle, "click", function() { map.panTo(point_sainte_chapelle, 2); @@ -939,7 +939,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_living_in = new GMarker(point_living_in, markerOptions); map.addOverlay(marker_living_in); - marker_living_in.info_window_content = '<div class="infowin"><h4>Living in a Railway Car<\/h4><span class="date blok">October 24, 2005 (Paris, France)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/letour.jpg" height="100" alt="Living in a Railway Car" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>This French apartment is more like a railway sleeper car than apartment proper. Maybe fifteen feet long and only three feet wide at the ceiling. More like five feet wide at the floor, but, because it\x27s an attic, the outer wall slopes in and you lose two feet by the time you get to the ceiling. It\x27s narrow enough that you can\x27t pass another body when you walk to length of it. <a href="/2005/oct/24/living-railway-car/">Read it »<\/a><\/p><\/div>' + marker_living_in.info_window_content = '<div class="infowin"><h4>Living in a Railway Car<\/h4><span class="date blok">October 24, 2005 (Paris, France)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/letour.jpg" height="100" alt="Living in a Railway Car" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>This French apartment is more like a railway sleeper car than apartment proper. Maybe fifteen feet long and only three feet wide at the ceiling. More like five feet wide at the floor, but, because it\u0027s an attic, the outer wall slopes in and you lose two feet by the time you get to the ceiling. It\u0027s narrow enough that you can\u0027t pass another body when you walk to length of it. <a href="/2005/oct/24/living-railway-car/">Read it »<\/a><\/p><\/div>' marker_living_in.bindInfoWindowHtml(marker_living_in.info_window_content, {maxWidth:400}); GEvent.addListener(marker_living_in, "click", function() { map.panTo(point_living_in, 2); @@ -950,7 +950,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_twenty_more = new GMarker(point_twenty_more, markerOptions); map.addOverlay(marker_twenty_more); - marker_twenty_more.info_window_content = '<div class="infowin"><h4>Twenty More Minutes to Go<\/h4><span class="date blok">October 20, 2005 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/nightswings.jpg" height="100" alt="Twenty More Minutes to Go" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Well it\x27s the night before I leave. I just got done pacing around the driveway of my parents house smoking cigarettes\x26#8230\x3B nervously? Excitedly? Restlessly? A bit of all of those I suppose. I walk across the street, over the drainage ditch and head for the swing set at the park. Right now I\x27m swinging in a park in Costa Mesa California. Tomorrow France. Weird. [Photo to the right, \x3Ca href\x3D\x22http://www.flickr.com/photos/scarin/53961434/\x22\x3Evia Flickr\x3C/a\x3E] <a href="/2005/oct/20/twenty-more-minutes-go/">Read it »<\/a><\/p><\/div>' + marker_twenty_more.info_window_content = '<div class="infowin"><h4>Twenty More Minutes to Go<\/h4><span class="date blok">October 20, 2005 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/nightswings.jpg" height="100" alt="Twenty More Minutes to Go" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Well it\u0027s the night before I leave. I just got done pacing around the driveway of my parents house smoking cigarettes\u0026#8230\u003B nervously? Excitedly? Restlessly? A bit of all of those I suppose. I walk across the street, over the drainage ditch and head for the swing set at the park. Right now I\u0027m swinging in a park in Costa Mesa California. Tomorrow France. Weird. [Photo to the right, \u003Ca href\u003D\u0022http://www.flickr.com/photos/scarin/53961434/\u0022\u003Evia Flickr\u003C/a\u003E] <a href="/2005/oct/20/twenty-more-minutes-go/">Read it »<\/a><\/p><\/div>' marker_twenty_more.bindInfoWindowHtml(marker_twenty_more.info_window_content, {maxWidth:400}); GEvent.addListener(marker_twenty_more, "click", function() { map.panTo(point_twenty_more, 2); @@ -961,7 +961,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_travel_tips = new GMarker(point_travel_tips, markerOptions); map.addOverlay(marker_travel_tips); - marker_travel_tips.info_window_content = '<div class="infowin"><h4>Travel Tips and Resources<\/h4><span class="date blok">October 19, 2005 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/travelgear.jpg" height="100" alt="Travel Tips and Resources" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>An overview of the things you might want to bring on an extended trip, as well as some tips and recommendations on things like visas and vaccinations. The part that was most helpful for me was learning what I \x3Cem\x3Edidn\x27t\x3C/em\x3E need to bring \x26mdash\x3B as it turns out, quite a bit. Nowadays my pack is much smaller and lighter. <a href="/2005/oct/19/tips-and-resources/">Read it »<\/a><\/p><\/div>' + marker_travel_tips.info_window_content = '<div class="infowin"><h4>Travel Tips and Resources<\/h4><span class="date blok">October 19, 2005 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/travelgear.jpg" height="100" alt="Travel Tips and Resources" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>An overview of the things you might want to bring on an extended trip, as well as some tips and recommendations on things like visas and vaccinations. The part that was most helpful for me was learning what I \u003Cem\u003Edidn\u0027t\u003C/em\u003E need to bring \u0026mdash\u003B as it turns out, quite a bit. Nowadays my pack is much smaller and lighter. <a href="/2005/oct/19/tips-and-resources/">Read it »<\/a><\/p><\/div>' marker_travel_tips.bindInfoWindowHtml(marker_travel_tips.info_window_content, {maxWidth:400}); GEvent.addListener(marker_travel_tips, "click", function() { map.panTo(point_travel_tips, 2); @@ -972,7 +972,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_new = new GMarker(point_the_new, markerOptions); map.addOverlay(marker_the_new); - marker_the_new.info_window_content = '<div class="infowin"><h4>The New Luddites<\/h4><span class="date blok">October 8, 2005 (Los Angeles, California)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/books.jpg" height="100" alt="The New Luddites" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>An older, non\x2Dtravel piece about Google\x27s plan to scan all the world\x27s books and Luddite\x2Dlike response from many authors. Let\x27s see, someone wants to make your book easier to find, searchable and indexable and you\x27re opposed to it? You\x27re a fucking idiot. <a href="/2005/oct/08/new-luddites/">Read it »<\/a><\/p><\/div>' + marker_the_new.info_window_content = '<div class="infowin"><h4>The New Luddites<\/h4><span class="date blok">October 8, 2005 (Los Angeles, California)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/books.jpg" height="100" alt="The New Luddites" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>An older, non\u002Dtravel piece about Google\u0027s plan to scan all the world\u0027s books and Luddite\u002Dlike response from many authors. Let\u0027s see, someone wants to make your book easier to find, searchable and indexable and you\u0027re opposed to it? You\u0027re a fucking idiot. <a href="/2005/oct/08/new-luddites/">Read it »<\/a><\/p><\/div>' marker_the_new.bindInfoWindowHtml(marker_the_new.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_new, "click", function() { map.panTo(point_the_new, 2); @@ -983,7 +983,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_one_nation = new GMarker(point_one_nation, markerOptions); map.addOverlay(marker_one_nation); - marker_one_nation.info_window_content = '<div class="infowin"><h4>One Nation Under a Groove<\/h4><span class="date blok">March 25, 2005 (Northampton, Massachusetts)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/ipod.jpg" height="100" alt="One Nation Under a Groove" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The sky is falling! The iPod! It\x27s ruining our culture! Or, uh, maybe it\x27s just like the Walkman, but better. And since, so far as I can tell, the world did not collapse with the introduction of the Walkman and headphones, it probably isn\x27t going to fall apart just because the storage format for our music has changed. [Photo to the right via \x3Ca href\x3D\x22http://flickr.com/photos/rogpool/2960735485/\x22\x3EFlickr\x3C/a\x3E] <a href="/2005/mar/25/one-nation-under-groove/">Read it »<\/a><\/p><\/div>' + marker_one_nation.info_window_content = '<div class="infowin"><h4>One Nation Under a Groove<\/h4><span class="date blok">March 25, 2005 (Northampton, Massachusetts)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/ipod.jpg" height="100" alt="One Nation Under a Groove" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>The sky is falling! The iPod! It\u0027s ruining our culture! Or, uh, maybe it\u0027s just like the Walkman, but better. And since, so far as I can tell, the world did not collapse with the introduction of the Walkman and headphones, it probably isn\u0027t going to fall apart just because the storage format for our music has changed. [Photo to the right via \u003Ca href\u003D\u0022http://flickr.com/photos/rogpool/2960735485/\u0022\u003EFlickr\u003C/a\u003E] <a href="/2005/mar/25/one-nation-under-groove/">Read it »<\/a><\/p><\/div>' marker_one_nation.bindInfoWindowHtml(marker_one_nation.info_window_content, {maxWidth:400}); GEvent.addListener(marker_one_nation, "click", function() { map.panTo(point_one_nation, 2); @@ -994,7 +994,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_farewell_mr = new GMarker(point_farewell_mr, markerOptions); map.addOverlay(marker_farewell_mr); - marker_farewell_mr.info_window_content = '<div class="infowin"><h4>Farewell Mr. Hunter S Thompson<\/h4><span class="date blok">February 24, 2005 (Northampton, Massachusetts)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/thompson.jpg" height="100" alt="Farewell Mr. Hunter S Thompson" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Hunter S. Thompson departs on a journey to the western lands. Thompson\x27s \x3Cem\x3EFear and Loathing in Las Vegas\x3C/em\x3E delivered the penultimate eulogy for the dreams of the 1960\x27s, one that mourned, but also tried to lay the empty idealism to rest. <a href="/2005/feb/24/farewell-mr-hunter-s-thompson/">Read it »<\/a><\/p><\/div>' + marker_farewell_mr.info_window_content = '<div class="infowin"><h4>Farewell Mr. Hunter S Thompson<\/h4><span class="date blok">February 24, 2005 (Northampton, Massachusetts)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/thompson.jpg" height="100" alt="Farewell Mr. Hunter S Thompson" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Hunter S. Thompson departs on a journey to the western lands. Thompson\u0027s \u003Cem\u003EFear and Loathing in Las Vegas\u003C/em\u003E delivered the penultimate eulogy for the dreams of the 1960\u0027s, one that mourned, but also tried to lay the empty idealism to rest. <a href="/2005/feb/24/farewell-mr-hunter-s-thompson/">Read it »<\/a><\/p><\/div>' marker_farewell_mr.bindInfoWindowHtml(marker_farewell_mr.info_window_content, {maxWidth:400}); GEvent.addListener(marker_farewell_mr, "click", function() { map.panTo(point_farewell_mr, 2); @@ -1005,7 +1005,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_the_art = new GMarker(point_the_art, markerOptions); map.addOverlay(marker_the_art); - marker_the_art.info_window_content = '<div class="infowin"><h4>The Art of the Essay<\/h4><span class="date blok">October 10, 2004 (Northampton, Massachusetts)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/essay.jpg" height="100" alt="The Art of the Essay" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I generally ignore internet debates, they never go anywhere, so why bother. But we all have our weak points and when programmer Paul Graham posted what might be the dumbest essay on writing that\x27s ever been written, I just couldn\x27t help myuself. <a href="/2004/oct/10/art-essay/">Read it »<\/a><\/p><\/div>' + marker_the_art.info_window_content = '<div class="infowin"><h4>The Art of the Essay<\/h4><span class="date blok">October 10, 2004 (Northampton, Massachusetts)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/essay.jpg" height="100" alt="The Art of the Essay" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>I generally ignore internet debates, they never go anywhere, so why bother. But we all have our weak points and when programmer Paul Graham posted what might be the dumbest essay on writing that\u0027s ever been written, I just couldn\u0027t help myuself. <a href="/2004/oct/10/art-essay/">Read it »<\/a><\/p><\/div>' marker_the_art.bindInfoWindowHtml(marker_the_art.info_window_content, {maxWidth:400}); GEvent.addListener(marker_the_art, "click", function() { map.panTo(point_the_art, 2); @@ -1016,7 +1016,7 @@ markerOptions = { clickable:true, draggable:false, icon:tinyIcon}; marker_farewell_mr = new GMarker(point_farewell_mr, markerOptions); map.addOverlay(marker_farewell_mr); - marker_farewell_mr.info_window_content = '<div class="infowin"><h4>Farewell Mr. Cash<\/h4><span class="date blok">September 12, 2003 (Northampton, Massachusetts)<\/span><p><img src="http://images.luxagraf.net/post-thumbs/2008/cash.jpg" height="100" alt="Farewell Mr. Cash" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Johnny Cash heads for the western lands. <a href="/2003/sep/12/farewell-mr-cash/">Read it »<\/a><\/p><\/div>' + marker_farewell_mr.info_window_content = '<div class="infowin"><h4>Farewell Mr. Cash<\/h4><span class="date blok">September 12, 2003 (Northampton, Massachusetts)<\/span><p><img src="http://127.0.0.1:8000/images/post-thumbs/2008/cash.jpg" height="100" alt="Farewell Mr. Cash" style="float: left; border: #000 10px solid; margin-right: 8px; margin-bottom: 4px; height: 100px;" \/>Johnny Cash heads for the western lands. <a href="/2003/sep/12/farewell-mr-cash/">Read it »<\/a><\/p><\/div>' marker_farewell_mr.bindInfoWindowHtml(marker_farewell_mr.info_window_content, {maxWidth:400}); GEvent.addListener(marker_farewell_mr, "click", function() { map.panTo(point_farewell_mr, 2); -- cgit v1.2.3-70-g09d2