| 
									
										
										
										
											2017-10-07 14:57:04 +02:00
										 |  |  | #!/usr/bin/env python | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-28 13:11:42 +03:00
										 |  |  | import fnmatch | 
					
						
							| 
									
										
										
										
											2017-09-15 11:55:23 -05:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | import sys | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import math | 
					
						
							|  |  |  | import platform | 
					
						
							|  |  |  | import xml.etree.ElementTree as ET | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                                    Config                                    # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | flags = { | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     "c": platform.platform() != "Windows",  # Disable by default on windows, since we use ANSI escape codes | 
					
						
							|  |  |  |     "b": False, | 
					
						
							|  |  |  |     "g": False, | 
					
						
							|  |  |  |     "s": False, | 
					
						
							|  |  |  |     "u": False, | 
					
						
							|  |  |  |     "h": False, | 
					
						
							|  |  |  |     "p": False, | 
					
						
							|  |  |  |     "o": True, | 
					
						
							|  |  |  |     "i": False, | 
					
						
							|  |  |  |     "a": True, | 
					
						
							|  |  |  |     "e": False, | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | } | 
					
						
							|  |  |  | flag_descriptions = { | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     "c": "Toggle colors when outputting.", | 
					
						
							|  |  |  |     "b": "Toggle showing only not fully described classes.", | 
					
						
							|  |  |  |     "g": "Toggle showing only completed classes.", | 
					
						
							|  |  |  |     "s": "Toggle showing comments about the status.", | 
					
						
							|  |  |  |     "u": "Toggle URLs to docs.", | 
					
						
							|  |  |  |     "h": "Show help and exit.", | 
					
						
							|  |  |  |     "p": "Toggle showing percentage as well as counts.", | 
					
						
							|  |  |  |     "o": "Toggle overall column.", | 
					
						
							|  |  |  |     "i": "Toggle collapse of class items columns.", | 
					
						
							|  |  |  |     "a": "Toggle showing all items.", | 
					
						
							|  |  |  |     "e": "Toggle hiding empty items.", | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | } | 
					
						
							|  |  |  | long_flags = { | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     "colors": "c", | 
					
						
							|  |  |  |     "use-colors": "c", | 
					
						
							|  |  |  |     "bad": "b", | 
					
						
							|  |  |  |     "only-bad": "b", | 
					
						
							|  |  |  |     "good": "g", | 
					
						
							|  |  |  |     "only-good": "g", | 
					
						
							|  |  |  |     "comments": "s", | 
					
						
							|  |  |  |     "status": "s", | 
					
						
							|  |  |  |     "urls": "u", | 
					
						
							|  |  |  |     "gen-url": "u", | 
					
						
							|  |  |  |     "help": "h", | 
					
						
							|  |  |  |     "percent": "p", | 
					
						
							|  |  |  |     "use-percentages": "p", | 
					
						
							|  |  |  |     "overall": "o", | 
					
						
							|  |  |  |     "use-overall": "o", | 
					
						
							|  |  |  |     "items": "i", | 
					
						
							|  |  |  |     "collapse": "i", | 
					
						
							|  |  |  |     "all": "a", | 
					
						
							|  |  |  |     "empty": "e", | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | table_columns = [ | 
					
						
							|  |  |  |     "name", | 
					
						
							|  |  |  |     "brief_description", | 
					
						
							|  |  |  |     "description", | 
					
						
							|  |  |  |     "methods", | 
					
						
							|  |  |  |     "constants", | 
					
						
							|  |  |  |     "members", | 
					
						
							|  |  |  |     "signals", | 
					
						
							|  |  |  |     "theme_items", | 
					
						
							|  |  |  | ] | 
					
						
							|  |  |  | table_column_names = ["Name", "Brief Desc.", "Desc.", "Methods", "Constants", "Members", "Signals", "Theme Items"] | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | colors = { | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     "name": [36],  # cyan | 
					
						
							|  |  |  |     "part_big_problem": [4, 31],  # underline, red | 
					
						
							|  |  |  |     "part_problem": [31],  # red | 
					
						
							|  |  |  |     "part_mostly_good": [33],  # yellow | 
					
						
							|  |  |  |     "part_good": [32],  # green | 
					
						
							|  |  |  |     "url": [4, 34],  # underline, blue | 
					
						
							|  |  |  |     "section": [1, 4],  # bold, underline | 
					
						
							|  |  |  |     "state_off": [36],  # cyan | 
					
						
							|  |  |  |     "state_on": [1, 35],  # bold, magenta/plum | 
					
						
							|  |  |  |     "bold": [1],  # bold | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | } | 
					
						
							|  |  |  | overall_progress_description_weigth = 10 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                                    Utils                                     # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | def validate_tag(elem, tag): | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     if elem.tag != tag: | 
					
						
							|  |  |  |         print('Tag mismatch, expected "' + tag + '", got ' + elem.tag) | 
					
						
							|  |  |  |         sys.exit(255) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 19:05:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | def color(color, string): | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     if flags["c"] and terminal_supports_color(): | 
					
						
							|  |  |  |         color_format = "" | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         for code in colors[color]: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             color_format += "\033[" + str(code) + "m" | 
					
						
							|  |  |  |         return color_format + string + "\033[0m" | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     else: | 
					
						
							|  |  |  |         return string | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | ansi_escape = re.compile(r"\x1b[^m]*m") | 
					
						
							| 
									
										
										
										
											2016-10-30 19:05:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | def nonescape_len(s): | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     return len(ansi_escape.sub("", s)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-01 21:00:32 +02:00
										 |  |  | def terminal_supports_color(): | 
					
						
							|  |  |  |     p = sys.platform | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     supported_platform = p != "Pocket PC" and (p != "win32" or "ANSICON" in os.environ) | 
					
						
							| 
									
										
										
										
											2017-10-01 21:00:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     is_a_tty = hasattr(sys.stdout, "isatty") and sys.stdout.isatty() | 
					
						
							| 
									
										
										
										
											2017-10-01 21:00:32 +02:00
										 |  |  |     if not supported_platform or not is_a_tty: | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  |     return True | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | ################################################################################ | 
					
						
							|  |  |  | #                                   Classes                                    # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 19:05:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | class ClassStatusProgress: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:57:40 +01:00
										 |  |  |     def __init__(self, described=0, total=0): | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         self.described = described | 
					
						
							|  |  |  |         self.total = total | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __add__(self, other): | 
					
						
							|  |  |  |         return ClassStatusProgress(self.described + other.described, self.total + other.total) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def increment(self, described): | 
					
						
							|  |  |  |         if described: | 
					
						
							|  |  |  |             self.described += 1 | 
					
						
							|  |  |  |         self.total += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_ok(self): | 
					
						
							|  |  |  |         return self.described >= self.total | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def to_configured_colored_string(self): | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         if flags["p"]: | 
					
						
							|  |  |  |             return self.to_colored_string("{percent}% ({has}/{total})", "{pad_percent}{pad_described}{s}{pad_total}") | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         else: | 
					
						
							|  |  |  |             return self.to_colored_string() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     def to_colored_string(self, format="{has}/{total}", pad_format="{pad_described}{s}{pad_total}"): | 
					
						
							| 
									
										
										
										
											2017-10-01 21:00:32 +02:00
										 |  |  |         ratio = float(self.described) / float(self.total) if self.total != 0 else 1 | 
					
						
							|  |  |  |         percent = int(round(100 * ratio)) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:57:40 +01:00
										 |  |  |         s = format.format(has=str(self.described), total=str(self.total), percent=str(percent)) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         if self.described >= self.total: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             s = color("part_good", s) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:57:40 +01:00
										 |  |  |         elif self.described >= self.total / 4 * 3: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             s = color("part_mostly_good", s) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         elif self.described > 0: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             s = color("part_problem", s) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             s = color("part_big_problem", s) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         pad_size = max(len(str(self.described)), len(str(self.total))) | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         pad_described = "".ljust(pad_size - len(str(self.described))) | 
					
						
							|  |  |  |         pad_percent = "".ljust(3 - len(str(percent))) | 
					
						
							|  |  |  |         pad_total = "".ljust(pad_size - len(str(self.total))) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:57:40 +01:00
										 |  |  |         return pad_format.format(pad_described=pad_described, pad_total=pad_total, pad_percent=pad_percent, s=s) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ClassStatus: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     def __init__(self, name=""): | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.has_brief_description = True | 
					
						
							|  |  |  |         self.has_description = True | 
					
						
							|  |  |  |         self.progresses = { | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             "methods": ClassStatusProgress(), | 
					
						
							|  |  |  |             "constants": ClassStatusProgress(), | 
					
						
							|  |  |  |             "members": ClassStatusProgress(), | 
					
						
							|  |  |  |             "theme_items": ClassStatusProgress(), | 
					
						
							|  |  |  |             "signals": ClassStatusProgress(), | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     def __add__(self, other): | 
					
						
							|  |  |  |         new_status = ClassStatus() | 
					
						
							|  |  |  |         new_status.name = self.name | 
					
						
							|  |  |  |         new_status.has_brief_description = self.has_brief_description and other.has_brief_description | 
					
						
							|  |  |  |         new_status.has_description = self.has_description and other.has_description | 
					
						
							|  |  |  |         for k in self.progresses: | 
					
						
							|  |  |  |             new_status.progresses[k] = self.progresses[k] + other.progresses[k] | 
					
						
							|  |  |  |         return new_status | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     def is_ok(self): | 
					
						
							|  |  |  |         ok = True | 
					
						
							|  |  |  |         ok = ok and self.has_brief_description | 
					
						
							|  |  |  |         ok = ok and self.has_description | 
					
						
							|  |  |  |         for k in self.progresses: | 
					
						
							|  |  |  |             ok = ok and self.progresses[k].is_ok() | 
					
						
							|  |  |  |         return ok | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-03 16:28:51 +02:00
										 |  |  |     def is_empty(self): | 
					
						
							|  |  |  |         sum = 0 | 
					
						
							|  |  |  |         for k in self.progresses: | 
					
						
							|  |  |  |             if self.progresses[k].is_ok(): | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             sum += self.progresses[k].total | 
					
						
							|  |  |  |         return sum < 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     def make_output(self): | 
					
						
							|  |  |  |         output = {} | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         output["name"] = color("name", self.name) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         ok_string = color("part_good", "OK") | 
					
						
							|  |  |  |         missing_string = color("part_big_problem", "MISSING") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         output["brief_description"] = ok_string if self.has_brief_description else missing_string | 
					
						
							|  |  |  |         output["description"] = ok_string if self.has_description else missing_string | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         description_progress = ClassStatusProgress( | 
					
						
							|  |  |  |             (self.has_brief_description + self.has_description) * overall_progress_description_weigth, | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             2 * overall_progress_description_weigth, | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         ) | 
					
						
							|  |  |  |         items_progress = ClassStatusProgress() | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         for k in ["methods", "constants", "members", "signals", "theme_items"]: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |             items_progress += self.progresses[k] | 
					
						
							|  |  |  |             output[k] = self.progresses[k].to_configured_colored_string() | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         output["items"] = items_progress.to_configured_colored_string() | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         output["overall"] = (description_progress + items_progress).to_colored_string( | 
					
						
							|  |  |  |             color("bold", "{percent}%"), "{pad_percent}{s}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         if self.name.startswith("Total"): | 
					
						
							|  |  |  |             output["url"] = color("url", "https://docs.godotengine.org/en/latest/classes/") | 
					
						
							|  |  |  |             if flags["s"]: | 
					
						
							|  |  |  |                 output["comment"] = color("part_good", "ALL OK") | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             output["url"] = color( | 
					
						
							|  |  |  |                 "url", "https://docs.godotengine.org/en/latest/classes/class_{name}.html".format(name=self.name.lower()) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             if flags["s"] and not flags["g"] and self.is_ok(): | 
					
						
							|  |  |  |                 output["comment"] = color("part_good", "ALL OK") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         return output | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-01 21:00:32 +02:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     def generate_for_class(c): | 
					
						
							|  |  |  |         status = ClassStatus() | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         status.name = c.attrib["name"] | 
					
						
							| 
									
										
										
										
											2017-09-15 11:55:23 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         for tag in list(c): | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             if tag.tag == "brief_description": | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |                 status.has_brief_description = len(tag.text.strip()) > 0 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             elif tag.tag == "description": | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |                 status.has_description = len(tag.text.strip()) > 0 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             elif tag.tag in ["methods", "signals"]: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |                 for sub_tag in list(tag): | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |                     descr = sub_tag.find("description") | 
					
						
							| 
									
										
										
										
											2017-11-24 18:25:56 +01:00
										 |  |  |                     status.progresses[tag.tag].increment(len(descr.text.strip()) > 0) | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             elif tag.tag in ["constants", "members", "theme_items"]: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |                 for sub_tag in list(tag): | 
					
						
							| 
									
										
										
										
											2019-10-08 10:02:05 -03:00
										 |  |  |                     if not sub_tag.text is None: | 
					
						
							|  |  |  |                         status.progresses[tag.tag].increment(len(sub_tag.text.strip()) > 0) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             elif tag.tag in ["tutorials"]: | 
					
						
							| 
									
										
										
										
											2017-09-16 13:47:44 +03:00
										 |  |  |                 pass  # Ignore those tags for now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             elif tag.tag in ["theme_items"]: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:57:40 +01:00
										 |  |  |                 pass  # Ignore those tags, since they seem to lack description at all | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 print(tag.tag, tag.attrib) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         return status | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                                  Arguments                                   # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | input_file_list = [] | 
					
						
							|  |  |  | input_class_list = [] | 
					
						
							| 
									
										
										
										
											2017-09-15 11:55:23 -05:00
										 |  |  | merged_file = "" | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | for arg in sys.argv[1:]: | 
					
						
							| 
									
										
										
										
											2017-10-07 14:57:04 +02:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         if arg.startswith("--"): | 
					
						
							| 
									
										
										
										
											2017-10-07 14:57:04 +02:00
										 |  |  |             flags[long_flags[arg[2:]]] = not flags[long_flags[arg[2:]]] | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         elif arg.startswith("-"): | 
					
						
							| 
									
										
										
										
											2017-10-07 14:57:04 +02:00
										 |  |  |             for f in arg[1:]: | 
					
						
							|  |  |  |                 flags[f] = not flags[f] | 
					
						
							|  |  |  |         elif os.path.isdir(arg): | 
					
						
							|  |  |  |             for f in os.listdir(arg): | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |                 if f.endswith(".xml"): | 
					
						
							|  |  |  |                     input_file_list.append(os.path.join(arg, f)) | 
					
						
							| 
									
										
										
										
											2017-10-07 14:57:04 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             input_class_list.append(arg) | 
					
						
							|  |  |  |     except KeyError: | 
					
						
							|  |  |  |         print("Unknown command line flag: " + arg) | 
					
						
							|  |  |  |         sys.exit(1) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if flags["i"]: | 
					
						
							|  |  |  |     for r in ["methods", "constants", "members", "signals", "theme_items"]: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         index = table_columns.index(r) | 
					
						
							|  |  |  |         del table_column_names[index] | 
					
						
							|  |  |  |         del table_columns[index] | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     table_column_names.append("Items") | 
					
						
							|  |  |  |     table_columns.append("items") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if flags["o"] == (not flags["i"]): | 
					
						
							|  |  |  |     table_column_names.append(color("bold", "Overall")) | 
					
						
							|  |  |  |     table_columns.append("overall") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if flags["u"]: | 
					
						
							|  |  |  |     table_column_names.append("Docs URL") | 
					
						
							|  |  |  |     table_columns.append("url") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                                     Help                                     # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if len(input_file_list) < 1 or flags["h"]: | 
					
						
							|  |  |  |     if not flags["h"]: | 
					
						
							|  |  |  |         print(color("section", "Invalid usage") + ": Please specify a classes directory") | 
					
						
							|  |  |  |     print(color("section", "Usage") + ": doc_status.py [flags] <classes_dir> [class names]") | 
					
						
							|  |  |  |     print("\t< and > signify required parameters, while [ and ] signify optional parameters.") | 
					
						
							|  |  |  |     print(color("section", "Available flags") + ":") | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     possible_synonym_list = list(long_flags) | 
					
						
							|  |  |  |     possible_synonym_list.sort() | 
					
						
							|  |  |  |     flag_list = list(flags) | 
					
						
							|  |  |  |     flag_list.sort() | 
					
						
							|  |  |  |     for flag in flag_list: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |         synonyms = [color("name", "-" + flag)] | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         for synonym in possible_synonym_list: | 
					
						
							|  |  |  |             if long_flags[synonym] == flag: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |                 synonyms.append(color("name", "--" + synonym)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         print( | 
					
						
							|  |  |  |             ( | 
					
						
							|  |  |  |                 "{synonyms} (Currently " | 
					
						
							|  |  |  |                 + color("state_" + ("on" if flags[flag] else "off"), "{value}") | 
					
						
							|  |  |  |                 + ")\n\t{description}" | 
					
						
							|  |  |  |             ).format( | 
					
						
							|  |  |  |                 synonyms=", ".join(synonyms), | 
					
						
							|  |  |  |                 value=("on" if flags[flag] else "off"), | 
					
						
							|  |  |  |                 description=flag_descriptions[flag], | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     sys.exit(0) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                               Parse class list                               # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class_names = [] | 
					
						
							|  |  |  | classes = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | for file in input_file_list: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     tree = ET.parse(file) | 
					
						
							|  |  |  |     doc = tree.getroot() | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     if "version" not in doc.attrib: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         print('Version missing from "doc"') | 
					
						
							|  |  |  |         sys.exit(255) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     version = doc.attrib["version"] | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     if doc.attrib["name"] in class_names: | 
					
						
							| 
									
										
										
										
											2017-09-15 11:55:23 -05:00
										 |  |  |         continue | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     class_names.append(doc.attrib["name"]) | 
					
						
							|  |  |  |     classes[doc.attrib["name"]] = doc | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | class_names.sort() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if len(input_class_list) < 1: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     input_class_list = ["*"] | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-28 13:11:42 +03:00
										 |  |  | filtered_classes = set() | 
					
						
							|  |  |  | for pattern in input_class_list: | 
					
						
							|  |  |  |     filtered_classes |= set(fnmatch.filter(class_names, pattern)) | 
					
						
							|  |  |  | filtered_classes = list(filtered_classes) | 
					
						
							|  |  |  | filtered_classes.sort() | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                               Make output table                              # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | table = [table_column_names] | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | table_row_chars = "| - " | 
					
						
							|  |  |  | table_column_chars = "|" | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | total_status = ClassStatus("Total") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-28 13:11:42 +03:00
										 |  |  | for cn in filtered_classes: | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     c = classes[cn] | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     validate_tag(c, "class") | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     status = ClassStatus.generate_for_class(c) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-29 22:27:59 +03:00
										 |  |  |     total_status = total_status + status | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     if (flags["b"] and status.is_ok()) or (flags["g"] and not status.is_ok()) or (not flags["a"]): | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         continue | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     if flags["e"] and status.is_empty(): | 
					
						
							| 
									
										
										
										
											2017-10-03 16:28:51 +02:00
										 |  |  |         continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     out = status.make_output() | 
					
						
							|  |  |  |     row = [] | 
					
						
							|  |  |  |     for column in table_columns: | 
					
						
							|  |  |  |         if column in out: | 
					
						
							|  |  |  |             row.append(out[column]) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             row.append("") | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     if "comment" in out and out["comment"] != "": | 
					
						
							|  |  |  |         row.append(out["comment"]) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     table.append(row) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | #                              Print output table                              # | 
					
						
							|  |  |  | ################################################################################ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if len(table) == 1 and flags["a"]: | 
					
						
							|  |  |  |     print(color("part_big_problem", "No classes suitable for printing!")) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     sys.exit(0) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if len(table) > 2 or not flags["a"]: | 
					
						
							|  |  |  |     total_status.name = "Total = {0}".format(len(table) - 1) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     out = total_status.make_output() | 
					
						
							|  |  |  |     row = [] | 
					
						
							|  |  |  |     for column in table_columns: | 
					
						
							|  |  |  |         if column in out: | 
					
						
							|  |  |  |             row.append(out[column]) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             row.append("") | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     table.append(row) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if flags["a"]: | 
					
						
							| 
									
										
										
										
											2020-02-01 00:19:24 +01:00
										 |  |  |     # Duplicate the headers at the bottom of the table so they can be viewed | 
					
						
							|  |  |  |     # without having to scroll back to the top. | 
					
						
							|  |  |  |     table.append(table_column_names) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | table_column_sizes = [] | 
					
						
							|  |  |  | for row in table: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     for cell_i, cell in enumerate(row): | 
					
						
							|  |  |  |         if cell_i >= len(table_column_sizes): | 
					
						
							|  |  |  |             table_column_sizes.append(0) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         table_column_sizes[cell_i] = max(nonescape_len(cell), table_column_sizes[cell_i]) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | divider_string = table_row_chars[0] | 
					
						
							|  |  |  | for cell_i in range(len(table[0])): | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |     divider_string += ( | 
					
						
							|  |  |  |         table_row_chars[1] + table_row_chars[2] * (table_column_sizes[cell_i]) + table_row_chars[1] + table_row_chars[0] | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | print(divider_string) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | for row_i, row in enumerate(table): | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     row_string = table_column_chars | 
					
						
							|  |  |  |     for cell_i, cell in enumerate(row): | 
					
						
							|  |  |  |         padding_needed = table_column_sizes[cell_i] - nonescape_len(cell) + 2 | 
					
						
							|  |  |  |         if cell_i == 0: | 
					
						
							| 
									
										
										
										
											2017-09-16 13:47:44 +03:00
										 |  |  |             row_string += table_row_chars[3] + cell + table_row_chars[3] * (padding_needed - 1) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  |             row_string += ( | 
					
						
							|  |  |  |                 table_row_chars[3] * int(math.floor(float(padding_needed) / 2)) | 
					
						
							|  |  |  |                 + cell | 
					
						
							|  |  |  |                 + table_row_chars[3] * int(math.ceil(float(padding_needed) / 2)) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         row_string += table_column_chars | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |     print(row_string) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-01 00:19:24 +01:00
										 |  |  |     # Account for the possible double header (if the `a` flag is enabled). | 
					
						
							|  |  |  |     # No need to have a condition for the flag, as this will behave correctly | 
					
						
							|  |  |  |     # if the flag is disabled. | 
					
						
							|  |  |  |     if row_i == 0 or row_i == len(table) - 3 or row_i == len(table) - 2: | 
					
						
							| 
									
										
										
										
											2016-10-30 18:44:57 +01:00
										 |  |  |         print(divider_string) | 
					
						
							| 
									
										
										
										
											2016-04-22 18:32:15 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | print(divider_string) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:28:32 +02:00
										 |  |  | if total_status.is_ok() and not flags["g"]: | 
					
						
							|  |  |  |     print("All listed classes are " + color("part_good", "OK") + "!") |