1# Copyright 2019 - The Android Open Source Project 2# 3# Licensed under the Apache License, Version 2.0 (the "License"); 4# you may not use this file except in compliance with the License. 5# You may obtain a copy of the License at 6# 7# http://www.apache.org/licenses/LICENSE-2.0 8# 9# Unless required by applicable law or agreed to in writing, software 10# distributed under the License is distributed on an "AS IS" BASIS, 11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12# See the License for the specific language governing permissions and 13# limitations under the License. 14 15############################################################################### 16# This script adds a HTML snippet to the generated reference docs located at 17# developer.android.com/reference. The snippet renders HTML that adds links to 18# toggle between the Java and Kotlin versions of the page. 19############################################################################### 20 21import getopt 22import os 23import sys 24 25 26# GLOBAL FLAGS 27 28global stubs 29global java_stubs, kotlin_stubs 30global work, verbose, show_solo, max_stubs 31global java_source_abs_path 32global kotlin_source_abs_path 33 34verbose = False # set True to list all files as they are stubbed (--verbose) 35work = False # set True to insert stubs, False to do a dry run for stats (--work) 36show_solo = False # set True to list files that only appear in one language, rather than both (--solo) 37max_stubs = 0 # set positive to create a limited number of stubs (--max 12) 38 39 40# You must run the script from the refodcs reference/ root directory 41 42java_ref_root = os.getcwd() 43kotlin_ref_root = os.path.join(java_ref_root, "kotlin") 44root = os.path.split(java_ref_root)[1] 45if root != "reference": 46 print("You must cd to the refocs reference/ root directory") 47 sys.exit() 48 49 50# This method inserts the language switcher into the two top-level Android 51# Platform pages: packages.html and classes.html 52# For both Java and Kotlin 53def insert_platform_summaries(): 54 global stubs 55 global java_stubs, kotlin_stubs 56 global verbose, work, show_solo 57 global java_source_abs_path 58 global kotlin_source_abs_path 59 60 stubs = 0 61 java_stubs = 0 62 kotlin_stubs = 0 63 64 java_source_abs_path = java_ref_root 65 kotlin_source_abs_path = kotlin_ref_root 66 insert_stub(os.path.join(java_ref_root, "packages.html"), True, True) 67 insert_stub(os.path.join(kotlin_ref_root, "packages.html"), False, True) 68 69 insert_stub(os.path.join(java_ref_root, "classes.html"), True, True) 70 insert_stub(os.path.join(kotlin_ref_root, "classes.html"), False, True) 71 72# This method uses switcher2, which assumes the refdocs stay in their current 73# assymetrical dirs (ref/android and ref/kotlin/android) 74# And just puts the switcher in the existing docs 75def insert_stub(doc, java, both): 76 global stubs 77 global java_stubs, kotlin_stubs 78 global verbose, work, show_solo 79 global java_source_abs_path 80 global kotlin_source_abs_path 81 82 stubs = stubs+1 83 84 if verbose: 85 print("File: ", stubs, doc) 86 else: 87 fn = os.path.split(doc) 88 print("File: ", stubs, fn[1], end="\r") 89 90 if (java): 91 java_stubs = java_stubs + 1 92 else: 93 kotlin_stubs = kotlin_stubs + 1 94 95 96 if (work): 97 if (java): 98 file_path = doc[len(java_ref_root)+1:] 99 stub = doc.replace(java_source_abs_path, kotlin_source_abs_path) 100 if (both): 101 slug1 = "sed -i 's/<\/h1>/{}/' {}".format("<\/h1>\\n{% setvar page_path %}_page_path_{% endsetvar %}\\n{% setvar can_switch %}1{% endsetvar %}\\n{% include \"reference\/_java_switcher2.md\" %}",doc) 102 else: 103 slug1 = "sed -i 's/<\/h1>/{}/' {}".format("<\/h1>\\n{% include \"reference\/_java_switcher2.md\" %}",doc) 104 else: 105 file_path = doc[len(kotlin_ref_root)+1:] 106 stub = doc.replace(kotlin_source_abs_path, java_source_abs_path) 107 if (both): 108 slug1 = "sed -i 's/<\/h1>/{}/' {}".format("<\/h1>\\n{% setvar page_path %}_page_path_{% endsetvar %}\\n{% setvar can_switch %}1{% endsetvar %}\\n{% include \"reference\/_kotlin_switcher2.md\" %}",doc) 109 else: 110 slug1 = "sed -i 's/<\/h1>/{}/' {}".format("<\/h1>\\n{% include \"reference\/_kotlin_switcher2.md\" %}",doc) 111 112 os.system(slug1) 113 if (both): 114 page_path_slug = "sed -i 's/_page_path_/{}/' {}".format(file_path.replace("/","\/"),doc) 115 os.system(page_path_slug) 116 117 118def scan_files(stem): 119 global work, verbose, show_solo, max_stubs 120 global stubs 121 global java_stubs, kotlin_stubs 122 global java_source_abs_path 123 global kotlin_source_abs_path 124 125 java_source_abs_path = os.path.join(java_ref_root, stem) 126 kotlin_source_abs_path = os.path.join(kotlin_ref_root, stem) 127 128 # Pass 1 129 # Loop over java content, create stubs for java, 130 # and for corresponding Kotlin (when it exsits) 131 132 # solo is java-only classes 133 # both is java+kotlin 134 stubs = 0 135 java_stubs = 0 136 kotlin_stubs = 0 137 solo = 0 138 both = 0 139 140 print("*** PASS1 (Java) ***") 141 maxed_out = False 142 for root, dirs, files in os.walk(java_source_abs_path): 143 if maxed_out: 144 break; 145 for file_ in files: 146 ext = os.path.splitext(file_) 147 ext = ext[1] 148 if not ext: 149 # this catches package-lists with no extension 150 print("***", os.path.join(root, file_)) 151 elif ext != ".html": 152 # filter out png, yaml, etc 153 continue 154 else: 155 # we have java content 156 doc = os.path.join(root, file_) 157 158 159 160 # look for matching kotlin file 161 kotlinsource = doc.replace(java_source_abs_path, kotlin_source_abs_path) 162 if os.path.isfile(kotlinsource): 163 # corresponding kotlin content exists 164 insert_stub(doc, True, True) 165 insert_stub(kotlinsource, False, True) 166 both = both+1 167 else: 168 # no kotlin content 169 if (show_solo): 170 print("solo: ", doc) 171 insert_stub(doc, True, False) 172 solo = solo+1 173 174 if max_stubs>0 and stubs>=max_stubs: 175 print() 176 print("max java stubs: ", max_stubs) 177 maxed_out = True; 178 break 179 180 print("Java+Kotlin:", both, "Only Java:", solo) 181 print() 182 183 184 # PASS 2 185 # Loop over kotlin content, create stubs for Kotlin-only APIs 186 print("*** PASS2 (Kotlin) ***") 187 solo = 0 188 both = 0 189 maxed_out = False 190 stubs = 0 191 for root, dirs, files in os.walk(kotlin_source_abs_path): 192 if maxed_out: 193 break; 194 for file_ in files: 195 ext = os.path.splitext (file_) 196 ext = ext[1] 197 if not ext: 198 # this catches package-lists with no extension 199 print("***", os.path.join(root, file_)) 200 elif ext != ".html": 201 # filter out png, yaml, etc 202 continue 203 else: 204 # we have kotlin content 205 doc = os.path.join(root, file_) 206 javadoc = doc.replace(kotlin_source_abs_path, java_source_abs_path) 207 file_name = os.path.splitext(file_)[0] 208 file_path = doc[len(kotlin_source_abs_path)+1:] 209 include_path = os.path.join("/reference/_kotlin", file_path) 210 211 if os.path.isfile(javadoc): 212 # corresponding java content exists 213 # so we already created the kotlin stub file 214 # nothing to do 215 both = both+1 216 else: 217 # no java content 218 # create the kotlin stub file 219 if (show_solo): 220 print("solo: ", doc) 221 insert_stub(doc , False, False) 222 solo = solo+1 223 224 if (max_stubs>0 and stubs>=max_stubs): 225 print() 226 print("max koltin stubs: ", max_stubs) 227 maxed_out = True; 228 break 229 230 231 print("Java+Kotlin:", both, "Only Kotlin:", solo) 232 print() 233 print("Java: ", java_stubs, " Kotlin: ", kotlin_stubs, "Total: ", java_stubs + kotlin_stubs) 234 235 236def main(argv): 237 238 global work, verbose, show_solo, max_stubs 239 global java_source_abs_path 240 global kotlin_source_abs_path 241 stem = "" 242 243 try: 244 opts, args = getopt.getopt(argv,"",["work","verbose","solo","max="]) 245 except getopt.GetoptError: 246 print('USAGE: switcher --work --verbose --solo --max=<max_stubs> platform|androidx|support|chrome') 247 sys.exit(2) 248 249 for opt, arg in opts: 250 if opt == '--work': 251 work = True 252 elif opt == "--verbose": 253 print("verbose") 254 verbose = True 255 elif opt == "--solo": 256 print("verbose") 257 show_solo = True 258 elif opt == "--max": 259 max_stubs = int(arg) 260 print("max ", max_stubs) 261 262 if len(args)>0: 263 source = args[0] 264 if source == "platform": 265 stem = "android" 266 print() 267 print("*** PLATFORM PAGES ***") 268 print("======================") 269 270 # Insert the switchers at the top level first 271 insert_platform_summaries() 272 273 elif source == "androidx": 274 stem = "androidx" 275 print() 276 print("*** ANDROIDX SUPPORT LIBRARY PAGES ***") 277 print("======================================") 278 279 elif source == "support": 280 stem = "android/support/v4/media" 281 print() 282 print("*** ANDROIDX SUPPORT LIBRARY PAGES ***") 283 print("======================================") 284 285 elif source == "chrome": 286 stem = "org/chromium/support_lib_boundary" 287 print() 288 print("*** ANDROIDX CHROMIUM PAGES ***") 289 print("===============================") 290 291 if (len(stem)>0): 292 scan_files(stem) 293 print(" *** DONE ***") 294 else: 295 print('You must specify one of: platform|androidx|support|chrome') 296 297 298 299if __name__ == "__main__": 300 main(sys.argv[1:]) 301 302