summaryrefslogtreecommitdiffstats
path: root/chrome/common/extensions/docs/server2/document_renderer.py
blob: 6a033752a925fea7dac64505ead54444f7776001 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
# Copyright 2013 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.

import logging
import os
from document_parser import ParseDocument
from platform_util import ExtractPlatformFromURL
from third_party.json_schema_compiler.model import UnixName


class DocumentRenderer(object):
  '''Performs document-level rendering such as the title, references,
  and table of contents: pulling that data out of the document, then
  replacing the $(title), $(ref:...) and $(table_of_contents) tokens with them.

  This can be thought of as a parallel to TemplateRenderer; while
  TemplateRenderer is responsible for interpreting templates and rendering files
  within the template engine, DocumentRenderer is responsible for interpreting
  higher-level document concepts like the title and TOC, then performing string
  replacement for them. The syntax for this replacement is $(...) where ... is
  the concept. Currently title and table_of_contents are supported.
  '''

  def __init__(self, table_of_contents_renderer, platform_bundle):
    self._table_of_contents_renderer = table_of_contents_renderer
    self._platform_bundle = platform_bundle

  def _RenderLinks(self, document, path):
    ''' Replaces all $(ref:...) references in |document| with html links.

        References have two forms:

          $(ref:api.node) - Replaces the reference with a link to node on the
                            API page. The title is set to the name of the node.

          $(ref:api.node Title) - Same as the previous form, but title is set
                                  to "Title".
    '''
    START_REF = '$(ref:'
    END_REF = ')'
    MAX_REF_LENGTH = 256

    new_document = []

    # Keeps track of position within |document|
    cursor_index = 0
    start_ref_index = document.find(START_REF)

    while start_ref_index != -1:
      end_ref_index = document.find(END_REF, start_ref_index)

      if (end_ref_index == -1 or
          end_ref_index - start_ref_index > MAX_REF_LENGTH):
        end_ref_index = document.find(' ', start_ref_index)
        logging.error('%s:%s has no terminating ) at line %s' % (
            path,
            document[start_ref_index:end_ref_index],
            document.count('\n', 0, end_ref_index)))

        new_document.append(document[cursor_index:end_ref_index + 1])
      else:
        ref = document[start_ref_index:end_ref_index]
        ref_parts = ref[len(START_REF):].split(None, 1)

        # Guess the api name from the html name, replacing '_' with '.' (e.g.
        # if the page is app_window.html, guess the api name is app.window)
        api_name = os.path.splitext(os.path.basename(path))[0].replace('_', '.')
        title = ref_parts[0] if len(ref_parts) == 1 else ref_parts[1]

        platform = ExtractPlatformFromURL(path)
        if platform is None:
          logging.error('Cannot resolve reference without a platform.')
          continue
        ref_dict = self._platform_bundle.GetReferenceResolver(
            platform).SafeGetLink(ref_parts[0],
                                  namespace=api_name,
                                  title=title,
                                  path=path)

        new_document.append(document[cursor_index:start_ref_index])
        new_document.append('<a href=%s>%s</a>' % (ref_dict['href'],
                                                   ref_dict['text']))

      cursor_index = end_ref_index + 1
      start_ref_index = document.find(START_REF, cursor_index)

    new_document.append(document[cursor_index:])

    return ''.join(new_document)

  def Render(self, document, path, render_title=False):
    # Render links first so that parsing and later replacements aren't
    # affected by $(ref...) substitutions
    document = self._RenderLinks(document, path)

    parsed_document = ParseDocument(document, expect_title=render_title)
    toc_text, toc_warnings = self._table_of_contents_renderer.Render(
        parsed_document.sections)

    # Only 1 title and 1 table of contents substitution allowed; in the common
    # case, save necessarily running over the entire file.
    if parsed_document.title:
      document = document.replace('$(title)', parsed_document.title, 1)
    return (document.replace('$(table_of_contents)', toc_text, 1),
            parsed_document.warnings + toc_warnings)