1#!/usr/bin/env python 2# 3# Copyright (C) 2016 Google, Inc. 4# 5# Licensed under the Apache License, Version 2.0 (the "License"); 6# you may not use this file except in compliance with the License. 7# You may obtain a copy of the License at: 8# 9# http://www.apache.org/licenses/LICENSE-2.0 10# 11# Unless required by applicable law or agreed to in writing, software 12# distributed under the License is distributed on an "AS IS" BASIS, 13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14# See the License for the specific language governing permissions and 15# limitations under the License. 16 17import collections 18import itertools 19import os 20import re 21import subprocess 22 23# Parsing states: 24# STATE_INITIAL: looking for rpc or function defintion 25# STATE_RPC_DECORATOR: in the middle of a multi-line rpc definition 26# STATE_FUNCTION_DECORATOR: in the middle of a multi-line function definition 27# STATE_COMPLETE: done parsing a function 28STATE_INITIAL = 1 29STATE_RPC_DECORATOR = 2 30STATE_FUNCTION_DEFINITION = 3 31STATE_COMPLETE = 4 32 33# RE to match key=value tuples with matching quoting on value. 34KEY_VAL_RE = re.compile(r''' 35 (?P<key>\w+)\s*=\s* # Key consists of only alphanumerics 36 (?P<quote>["']?) # Optional quote character. 37 (?P<value>.*?) # Value is a non greedy match 38 (?P=quote) # Closing quote equals the first. 39 ($|,) # Entry ends with comma or end of string 40 ''', re.VERBOSE) 41 42# RE to match a function definition and extract out the function name. 43FUNC_RE = re.compile(r'.+\s+(\w+)\s*\(.*') 44 45 46class Function(object): 47 """Represents a RPC-exported function.""" 48 49 def __init__(self, rpc_def, func_def): 50 """Constructs a function object given its RPC and function signature.""" 51 self._function = '' 52 self._signature = '' 53 self._description = '' 54 self._returns = '' 55 56 self._ParseRpcDefinition(rpc_def) 57 self._ParseFunctionDefinition(func_def) 58 59 def _ParseRpcDefinition(self, s): 60 """Parse RPC definition.""" 61 # collapse string concatenation 62 s = s.replace('" + "', '') 63 s = s.strip('()') 64 for m in KEY_VAL_RE.finditer(s): 65 if m.group('key') == 'description': 66 self._description = m.group('value') 67 if m.group('key') == 'returns': 68 self._returns = m.group('value') 69 70 def _ParseFunctionDefinition(self, s): 71 """Parse function definition.""" 72 # Remove some keywords we don't care about. 73 s = s.replace('public ', '') 74 s = s.replace('synchronized ', '') 75 # Remove any throw specifications. 76 s = re.sub('\s+throws.*', '', s) 77 s = s.strip('{') 78 # Remove all the RPC parameter annotations. 79 s = s.replace('@RpcOptional ', '') 80 s = s.replace('@RpcOptional() ', '') 81 s = re.sub('@RpcParameter\s*\(.+?\)\s+', '', s) 82 s = re.sub('@RpcDefault\s*\(.+?\)\s+', '', s) 83 m = FUNC_RE.match(s) 84 if m: 85 self._function = m.group(1) 86 self._signature = s.strip() 87 88 @property 89 def function(self): 90 return self._function 91 92 @property 93 def signature(self): 94 return self._signature 95 96 @property 97 def description(self): 98 return self._description 99 100 @property 101 def returns(self): 102 return self._returns 103 104 105class DocGenerator(object): 106 """Documentation genereator.""" 107 108 def __init__(self, basepath): 109 """Construct based on all the *Facade.java files in the given basepath.""" 110 self._functions = collections.defaultdict(list) 111 112 for path, dirs, files in os.walk(basepath): 113 for f in files: 114 if f.endswith('Facade.java'): 115 self._Parse(os.path.join(path, f)) 116 117 def _Parse(self, filename): 118 """Parser state machine for a single file.""" 119 state = STATE_INITIAL 120 self._current_rpc = '' 121 self._current_function = '' 122 123 with open(filename, 'r') as f: 124 for line in f.readlines(): 125 line = line.strip() 126 if state == STATE_INITIAL: 127 state = self._ParseLineInitial(line) 128 elif state == STATE_RPC_DECORATOR: 129 state = self._ParseLineRpcDecorator(line) 130 elif state == STATE_FUNCTION_DEFINITION: 131 state = self._ParseLineFunctionDefinition(line) 132 133 if state == STATE_COMPLETE: 134 self._EmitFunction(filename) 135 state = STATE_INITIAL 136 137 def _ParseLineInitial(self, line): 138 """Parse a line while in STATE_INITIAL.""" 139 if line.startswith('@Rpc('): 140 self._current_rpc = line[4:] 141 if not line.endswith(')'): 142 # Multi-line RPC definition 143 return STATE_RPC_DECORATOR 144 elif line.startswith('public'): 145 self._current_function = line 146 if not line.endswith('{'): 147 # Multi-line function definition 148 return STATE_FUNCTION_DEFINITION 149 else: 150 return STATE_COMPLETE 151 return STATE_INITIAL 152 153 def _ParseLineRpcDecorator(self, line): 154 """Parse a line while in STATE_RPC_DECORATOR.""" 155 self._current_rpc += ' ' + line 156 if line.endswith(')'): 157 # Done with RPC definition 158 return STATE_INITIAL 159 else: 160 # Multi-line RPC definition 161 return STATE_RPC_DECORATOR 162 163 def _ParseLineFunctionDefinition(self, line): 164 """Parse a line while in STATE_FUNCTION_DEFINITION.""" 165 self._current_function += ' ' + line 166 if line.endswith('{'): 167 # Done with function definition 168 return STATE_COMPLETE 169 else: 170 # Multi-line function definition 171 return STATE_FUNCTION_DEFINITION 172 173 def _EmitFunction(self, filename): 174 """Store a function definition from the current parse state.""" 175 if self._current_rpc and self._current_function: 176 module = os.path.basename(filename)[0:-5] 177 f = Function(self._current_rpc, self._current_function) 178 if f.function: 179 self._functions[module].append(f) 180 181 self._current_rpc = None 182 self._current_function = None 183 184 def WriteOutput(self, filename): 185 git_rev = None 186 try: 187 git_rev = subprocess.check_output('git rev-parse HEAD', 188 shell=True).strip() 189 except subprocess.CalledProcessError as e: 190 # Getting the commit ID is optional; we continue if we cannot get it 191 pass 192 193 with open(filename, 'w') as f: 194 if git_rev: 195 f.write('Generated at commit `%s`\n\n' % git_rev) 196 # Write table of contents 197 for module in sorted(self._functions.keys()): 198 f.write('**%s**\n\n' % module) 199 for func in self._functions[module]: 200 f.write(' * [%s](#%s)\n' % 201 (func.function, func.function.lower())) 202 f.write('\n') 203 204 f.write('# Method descriptions\n\n') 205 for func in itertools.chain.from_iterable( 206 self._functions.values()): 207 f.write('## %s\n\n' % func.function) 208 f.write('```\n') 209 f.write('%s\n\n' % func.signature) 210 f.write('%s\n' % func.description) 211 if func.returns: 212 if func.returns.lower().startswith('return'): 213 f.write('\n%s\n' % func.returns) 214 else: 215 f.write('\nReturns %s\n' % func.returns) 216 f.write('```\n\n') 217 218# Main 219basepath = os.path.abspath(os.path.join(os.path.dirname( 220 os.path.realpath(__file__)), '..')) 221g = DocGenerator(basepath) 222g.WriteOutput(os.path.join(basepath, 'Docs/ApiReference.md')) 223