OLD | NEW |
| (Empty) |
1 # Copyright 2013 The Chromium Authors. All rights reserved. | |
2 # Use of this source code is governed by a BSD-style license that can be | |
3 # found in the LICENSE file. | |
4 import logging | |
5 import optparse | |
6 import os | |
7 import pkgutil | |
8 import pydoc | |
9 import re | |
10 import sys | |
11 | |
12 import telemetry | |
13 from telemetry.core import util | |
14 | |
15 telemetry_dir = util.GetTelemetryDir() | |
16 docs_dir = os.path.join(telemetry_dir, 'docs', 'pydoc') | |
17 | |
18 def RemoveAllDocs(): | |
19 for dirname, _, filenames in os.walk(docs_dir): | |
20 for filename in filenames: | |
21 os.remove(os.path.join(dirname, filename)) | |
22 | |
23 def GenerateHTMLForModule(module): | |
24 html = pydoc.html.page(pydoc.describe(module), | |
25 pydoc.html.document(module, module.__name__)) | |
26 | |
27 # pydoc writes out html with links in a variety of funky ways. We need | |
28 # to fix them up. | |
29 assert not telemetry_dir.endswith(os.sep) | |
30 links = re.findall('(<a href="(.+?)">(.+?)</a>)', html) | |
31 for link_match in links: | |
32 link, href, link_text = link_match | |
33 if not href.startswith('file:'): | |
34 continue | |
35 | |
36 new_href = href.replace('file:', '') | |
37 new_href = new_href.replace(telemetry_dir, '..') | |
38 new_href = new_href.replace(os.sep, '/') | |
39 | |
40 new_link_text = link_text.replace(telemetry_dir + os.sep, '') | |
41 | |
42 new_link = '<a href="%s">%s</a>' % (new_href, new_link_text) | |
43 html = html.replace(link, new_link) | |
44 | |
45 # pydoc writes out html with absolute path file links. This is not suitable | |
46 # for checked in documentation. So, fix up the HTML after it is generated. | |
47 #html = re.sub('href="file:%s' % telemetry_dir, 'href="..', html) | |
48 #html = re.sub(telemetry_dir + os.sep, '', html) | |
49 return html | |
50 | |
51 def WriteHTMLForModule(module): | |
52 page = GenerateHTMLForModule(module) | |
53 path = os.path.join(docs_dir, '%s.html' % module.__name__) | |
54 with open(path, 'w') as f: | |
55 sys.stderr.write('Wrote %s\n' % os.path.relpath(path)) | |
56 f.write(page) | |
57 | |
58 def GetAllModulesToDocument(module): | |
59 modules = [module] | |
60 for _, modname, _ in pkgutil.walk_packages( | |
61 module.__path__, module.__name__ + '.'): | |
62 if modname.endswith('_unittest'): | |
63 logging.debug("skipping %s due to being a unittest", modname) | |
64 continue | |
65 | |
66 module = __import__(modname, fromlist=[""]) | |
67 name, _ = os.path.splitext(module.__file__) | |
68 if not os.path.exists(name + '.py'): | |
69 logging.info("skipping %s due to being an orphan .pyc", module.__file__) | |
70 continue | |
71 | |
72 modules.append(module) | |
73 return modules | |
74 | |
75 class AlreadyDocumentedModule(object): | |
76 def __init__(self, filename): | |
77 self.filename = filename | |
78 | |
79 @property | |
80 def name(self): | |
81 basename = os.path.basename(self.filename) | |
82 return os.path.splitext(basename)[0] | |
83 | |
84 @property | |
85 def contents(self): | |
86 with open(self.filename, 'r') as f: | |
87 return f.read() | |
88 | |
89 def GetAlreadyDocumentedModules(): | |
90 modules = [] | |
91 for dirname, _, filenames in os.walk(docs_dir): | |
92 for filename in filenames: | |
93 path = os.path.join(dirname, filename) | |
94 modules.append(AlreadyDocumentedModule(path)) | |
95 return modules | |
96 | |
97 | |
98 def IsUpdateDocsNeeded(): | |
99 already_documented_modules = GetAlreadyDocumentedModules() | |
100 already_documented_modules_by_name = dict( | |
101 (module.name, module) for module in already_documented_modules) | |
102 current_modules = GetAllModulesToDocument(telemetry) | |
103 | |
104 # Quick check: if the names of modules has changed, we definitely need | |
105 # an update. | |
106 already_documented_module_names = set( | |
107 m.name for m in already_documented_modules) | |
108 | |
109 current_module_names = set([m.__name__ for m in current_modules]) | |
110 | |
111 if current_module_names != already_documented_module_names: | |
112 return True | |
113 | |
114 # Generate the new docs and compare aganist the old. If changed, then a | |
115 # an update is needed. | |
116 for current_module in current_modules: | |
117 already_documented_module = already_documented_modules_by_name[ | |
118 current_module.__name__] | |
119 current_html = GenerateHTMLForModule(current_module) | |
120 if current_html != already_documented_module.contents: | |
121 return True | |
122 | |
123 return False | |
124 | |
125 def Main(args): | |
126 parser = optparse.OptionParser() | |
127 parser.add_option( | |
128 '-v', '--verbose', action='count', dest='verbosity', | |
129 help='Increase verbosity level (repeat as needed)') | |
130 options, args = parser.parse_args(args) | |
131 if options.verbosity >= 2: | |
132 logging.getLogger().setLevel(logging.DEBUG) | |
133 elif options.verbosity: | |
134 logging.getLogger().setLevel(logging.INFO) | |
135 else: | |
136 logging.getLogger().setLevel(logging.WARNING) | |
137 | |
138 assert os.path.isdir(docs_dir), '%s does not exist' % docs_dir | |
139 | |
140 RemoveAllDocs() | |
141 | |
142 old_cwd = os.getcwd() | |
143 try: | |
144 os.chdir(telemetry_dir) | |
145 for module in GetAllModulesToDocument(telemetry): | |
146 WriteHTMLForModule(module) | |
147 finally: | |
148 os.chdir(old_cwd) | |
OLD | NEW |