File: docusage.py

package info (click to toggle)
sqlkit 0.9.5-1
  • links: PTS, VCS
  • area: main
  • in suites: wheezy
  • size: 8,184 kB
  • sloc: python: 17,477; sql: 166; makefile: 95; xml: 23; sh: 11
file content (122 lines) | stat: -rw-r--r-- 3,488 bytes parent folder | download | duplicates (2)
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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
# -*- coding: utf-8 -*-
'''
This module provides the ``includesh`` directive. It is implemented by
:class:`IncludeShellDirective` class.
'''
import os
import re
import types
import tempfile

# Import required docutils modules
from sphinx.util.compat import Directive

from docutils import io, nodes, statemachine, utils
from docutils.parsers.rst import directives
from docutils.parsers.rst.directives.misc import Include
# from docutils.parsers.rst.directives.tables import ListTable

import sphinx


#---------------------------------------------------------#
class IncludePyProgramDirective(Include):
  '''
  IncludeShellDirective implements the directive. The class
  is registered as a directive in :func:`rusty.includesh.setup`
  '''
  required_arguments = 1
  optional_arguments = 0
  has_content = None
  node_class = None
  option_spec = {
      'verbatim': directives.flag,
      'docstring': directives.unchanged,
  }

  #---------------------------------------------------------#
  def run(self):
    '''
    Called automatically by the docutils.
    '''

    # Take the current inclusion file, read it and save in
    # temporary location where the actual ``Include`` directive
    # can read it

    # This part is taken from docutils
    if not self.state.document.settings.file_insertion_enabled:
      raise self.warning('"%s" directive disabled.' % self.name)

    source = self.state_machine.input_lines.source(
      self.lineno - self.state_machine.input_offset - 1)

    source_dir = os.path.dirname(os.path.abspath(source))
    path = directives.path(self.arguments[0])

    if path.startswith('<') and path.endswith('>'):
      path = os.path.join(self.standard_include_path, path[1:-1])

    path = os.path.normpath(os.path.join(source_dir, path))
    path = utils.relative_path(None, path)

    try:
      self.state.document.settings.record_dependencies.add(path)
      include_file = open(path, 'r+b')
    except IOError, error:
      raise self.severe('Problems with "%s" directive path:\n%s: %s.'
                        % (self.name, error.__class__.__name__, error))

    try:
      include_text = include_file.read()
    except UnicodeError, error:
      raise self.severe(
        'Problem with "%s" directive:\n%s: %s'
        % (self.name, error.__class__.__name__, error))

    if not 'docstring' in self.options:
      self.options['docstring'] = '__doc__'
      
    self.arguments[0] = self.get_docstring_file(path)

    return Include.run(self)

  def get_docstring_file(self, path):
    p = Program(path)
    tfd, target_path = tempfile.mkstemp(prefix='sd-')
    f = open(target_path, 'wb')
    text = p.docstring
    
    if 'verbatim' in self.options:
      f.write('::\n')
      pat = re.compile('^', re.MULTILINE)
      text = re.sub(pat, "  ", text )      

    f.write("%s\n" % text)
    return target_path


def setup(app):
  '''
  Extension setup, called by Sphinx
  '''
  app.add_config_value('templates_map', {}, 'html')
  # Sphinx 5 support
  if '5' in sphinx.__version__.split('.'):
    app.add_directive('docusage', IncludePyProgramDirective, 0, (0,0,0))
  else:
    app.add_directive('docusage', IncludePyProgramDirective)

class Program(object):

    DOCSTRING = re.compile(
        r'''
        """(?P<docstring>.*?)"""
        ''',
        re.VERBOSE|re.DOTALL
        )
    def __init__(self, filename):

        self.text = open(filename).read()
        self.docstring = self.DOCSTRING.search(self.text).group('docstring')