1#!/usr/bin/env python3 2# 3# Copyright (C) 2017 The Android Open Source Project 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 17"""Outputs HTML based on an input JSON file. 18 19Outputs HTML tables suitable for inclusion in the Android documentation that 20reflect the crypto algorithm support shown in the provided data file. 21""" 22 23import argparse 24import operator 25 26import crypto_docs 27 28 29find_by_name = crypto_docs.find_by_name 30 31 32def sort_by_name(seq): 33 return sorted(seq, key=lambda x: x['name']) 34 35 36def has_notes(category): 37 for algorithm in category['algorithms']: 38 if 'note' in algorithm: 39 return True 40 return False 41 42 43# Prevents the given value from being word-wrapped. This is mainly to ensure that 44# long identifiers with hyphens, like OAEPwithSHA-1andMGF1Padding, don't get word-wrapped 45# at the hyphen. 46def nowrap(value): 47 return '<span style="white-space: nowrap">%s</span>' % value 48 49 50def main(): 51 parser = argparse.ArgumentParser(description='Output algorithm support HTML tables') 52 parser.add_argument('--for_javadoc', 53 action='store_true', 54 help='If specified, format for inclusion in class documentation') 55 parser.add_argument('--category', 56 action='append', 57 help='The category to display, may be specified multiple times') 58 parser.add_argument('file', 59 help='The JSON file to use for data') 60 args = parser.parse_args() 61 62 output = [] 63 data = crypto_docs.load_json(args.file) 64 categories = sort_by_name(data['categories']) 65 output.append('<h2 id="SupportedAlgorithms">Supported Algorithms</h2>') 66 output.append('') 67 output.append('<ul>') 68 for category in categories: 69 if not category['name'].endswith('.Enabled'): 70 output.append(' <li><a href="#Supported{name}">' 71 '<code>{name}</code></a></li>'.format(**category)) 72 output.append('</ul>') 73 for category in categories: 74 if args.category and category['name'] not in args.category: 75 continue 76 show_notes = has_notes(category) 77 if category['name'].endswith('.Enabled'): 78 # These are handled in the "Supported" section below 79 continue 80 if category['name'] == 'Cipher': 81 # We display ciphers in a four-column table to conserve space and 82 # so that it's more comprehensible. To do this, we have to 83 # collapse all our ciphers into "equivalence classes" of a sort. 84 85 # First, collect the relevant data for each algorithm into a tuple. 86 # The mode and padding are in lists because we are going to collapse 87 # multiple tuples with those in later steps. 88 algorithms = sort_by_name(category['algorithms']) 89 tuples = [] 90 for algorithm in algorithms: 91 name, mode, padding = algorithm['name'].split('/') 92 tuples.append(( 93 name, 94 [mode], 95 [padding], 96 algorithm['supported_api_levels'], 97 'deprecated' in algorithm and algorithm['deprecated'], 98 algorithm.get('note', ''))) 99 # Sort the tuples by all items except padding, then collapse 100 # items with all non-padding values the same (which will always be 101 # neighboring items) into a single item. 102 tuples.sort(key=operator.itemgetter(0, 1, 3, 4)) 103 i = 0 104 while i < len(tuples) - 1: 105 if (tuples[i][0] == tuples[i+1][0] 106 and tuples[i][1] == tuples[i+1][1] 107 and tuples[i][3] == tuples[i+1][3] 108 and tuples[i][4] == tuples[i+1][4] 109 and tuples[i][5] == tuples[i+1][5]): 110 tuples[i][2].extend(tuples[i+1][2]) 111 del tuples[i+1] 112 else: 113 i += 1 114 # Do the same thing as above, but with modes. 115 tuples.sort(key=operator.itemgetter(0, 2, 3, 4)) 116 i = 0 117 while i < len(tuples) - 1: 118 if (tuples[i][0] == tuples[i+1][0] 119 and tuples[i][2] == tuples[i+1][2] 120 and tuples[i][3] == tuples[i+1][3] 121 and tuples[i][4] == tuples[i+1][4] 122 and tuples[i][5] == tuples[i+1][5]): 123 tuples[i][1].extend(tuples[i+1][1]) 124 del tuples[i+1] 125 else: 126 i += 1 127 # Display the table with rowspans for those entries where all the 128 # items have the same algorithm, mode, etc 129 output.append('<h3 id="Supported{name}">{name}</h3>'.format(**category)) 130 output.append('<table>') 131 output.append(' <thead>') 132 output.append(' <tr>') 133 output.append(' <th>Algorithm</th>') 134 output.append(' <th>Modes</th>') 135 output.append(' <th>Paddings</th>') 136 output.append(' <th>Supported API Levels</th>') 137 if show_notes: 138 output.append(' <th>Notes</th>') 139 output.append(' </tr>') 140 output.append(' </thead>') 141 output.append(' <tbody>') 142 tuples.sort(key=operator.itemgetter(0, 4, 1, 2, 3)) 143 i = 0 144 cur_deprecated = None 145 cur_algorithm = None 146 cur_mode = None 147 while i < len(tuples): 148 row = tuples[i] 149 if row[4] != cur_deprecated: 150 cur_deprecated = row[4] 151 cur_note = row[5] 152 cur_algorithm = None 153 cur_mode = None 154 if cur_deprecated: 155 output.append(' <tr class="deprecated">') 156 else: 157 output.append(' <tr>') 158 if row[0] != cur_algorithm: 159 cur_algorithm = row[0] 160 cur_mode = None 161 j = i + 1 162 while (j < len(tuples) 163 and tuples[j][4] == cur_deprecated 164 and tuples[j][5] == cur_note 165 and tuples[j][0] == cur_algorithm): 166 j += 1 167 rowspan = j - i 168 if rowspan > 1: 169 output.append(' <td rowspan="%d">%s</td>' % (rowspan, nowrap(cur_algorithm))) 170 else: 171 output.append(' <td>%s</td>' % nowrap(cur_algorithm)) 172 if row[1] != cur_mode: 173 cur_mode = row[1] 174 j = i + 1 175 while (j < len(tuples) 176 and tuples[j][4] == cur_deprecated 177 and tuples[j][5] == cur_note 178 and tuples[j][0] == cur_algorithm 179 and tuples[j][1] == cur_mode): 180 j += 1 181 rowspan = j - i 182 modestring = '<br>'.join([nowrap(x) for x in cur_mode]) 183 if rowspan > 1: 184 output.append(' <td rowspan="%d">%s</td>' % (rowspan, modestring)) 185 else: 186 output.append(' <td>%s</td>' % modestring) 187 output.append(' <td>%s</td>' % '<br>'.join([nowrap(x) for x in row[2]])) 188 output.append(' <td>%s</td>' % nowrap(row[3])) 189 if show_notes: 190 output.append(' <td>%s</td>' % row[5]) 191 output.append(' </tr>') 192 i += 1 193 output.append(' </tbody>') 194 output.append('</table>') 195 elif category['name'].endswith('.Supported'): 196 # Some categories come with a "Supported" and "Enabled" list, and we 197 # group those together in one table for display. Every entry that's enabled 198 # must be supported, so we can just look up the enabled version for each 199 # supported item 200 basename = category['name'][:-len('.Supported')] 201 supported = sort_by_name(category['algorithms']) 202 enabled = sort_by_name(find_by_name(categories, basename + '.Enabled')['algorithms']) 203 output.append('<h3 id="Supported{0}">{0}</h3>'.format(basename)) 204 output.append('<table>') 205 output.append(' <thead>') 206 output.append(' <tr>') 207 output.append(' <th>Algorithm</th>') 208 output.append(' <th>Supported API Levels</th>') 209 output.append(' <th>Enabled By Default</th>') 210 if show_notes: 211 output.append(' <th>Notes</th>') 212 output.append(' </tr>') 213 output.append(' </thead>') 214 output.append(' <tbody>') 215 for algorithm in supported: 216 if 'deprecated' in algorithm and algorithm['deprecated']: 217 output.append(' <tr class="deprecated">') 218 else: 219 output.append(' <tr>') 220 output.append(' <td>%s</td>' % nowrap(algorithm['name'])) 221 output.append(' <td>%s</td>' % nowrap(algorithm['supported_api_levels'])) 222 enabled_alg = find_by_name(enabled, algorithm['name']) 223 if enabled_alg is None: 224 output.append(' <td></td>') 225 else: 226 output.append(' <td>%s</td>' % nowrap(enabled_alg['supported_api_levels'])) 227 if show_notes: 228 if 'note' in algorithm: 229 output.append(' <td>%s</td>' % algorithm['note']) 230 else: 231 output.append(' <td></td>') 232 output.append(' </tr>') 233 output.append(' </tbody>') 234 output.append('</table>') 235 else: 236 output.append('<h3 id="Supported{name}">{name}</h3>'.format(**category)) 237 output.append('<table>') 238 output.append(' <thead>') 239 output.append(' <tr>') 240 output.append(' <th>Algorithm</th>') 241 output.append(' <th>Supported API Levels</th>') 242 if show_notes: 243 output.append(' <th>Notes</th>') 244 output.append(' </tr>') 245 output.append(' </thead>') 246 output.append(' <tbody>') 247 algorithms = sort_by_name(category['algorithms']) 248 for algorithm in algorithms: 249 if 'deprecated' in algorithm and algorithm['deprecated']: 250 output.append(' <tr class="deprecated">') 251 else: 252 output.append(' <tr>') 253 output.append(' <td>%s</td>' % nowrap(algorithm['name'])) 254 output.append(' <td>%s</td>' % nowrap(algorithm['supported_api_levels'])) 255 if show_notes: 256 if 'note' in algorithm: 257 output.append(' <td>%s</td>' % algorithm['note']) 258 else: 259 output.append(' <td></td>') 260 output.append(' </tr>') 261 output.append(' </tbody>') 262 output.append('</table>') 263 if args.for_javadoc: 264 for i in range(len(output)): 265 output[i] = ' * ' + output[i] 266 print('\n'.join(output)) 267 268 269if __name__ == '__main__': 270 main() 271