| #!/usr/bin/env python3 |
| # Copyright (C) 2022 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| import argparse |
| import json |
| import os |
| import sys |
| from typing import Any |
| from typing import Dict |
| from typing import Optional |
| |
| # Allow importing of root-relative modules. |
| ROOT_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) |
| sys.path.append(os.path.join(ROOT_DIR)) |
| |
| #pylint: disable=wrong-import-position |
| from python.generators.trace_processor_table.public import ColumnDoc |
| from python.generators.trace_processor_table.public import ColumnFlag |
| import python.generators.trace_processor_table.util as util |
| from python.generators.trace_processor_table.util import ParsedTable |
| from python.generators.trace_processor_table.util import ParsedColumn |
| #pylint: enable=wrong-import-position |
| |
| |
| def gen_json_for_column(table: ParsedTable, |
| col: ParsedColumn) -> Optional[Dict[str, Any]]: |
| """Generates the JSON documentation for a column in a table.""" |
| assert table.table.tabledoc |
| |
| # id and type columns should be skipped if the table specifies so. |
| is_skippable_col = col.is_implicit_id or col.is_implicit_type |
| if table.table.tabledoc.skip_id_and_type and is_skippable_col: |
| return None |
| |
| # Ignore hidden columns in the documentation. |
| if ColumnFlag.HIDDEN in col.column.flags: |
| return None |
| |
| # Our default assumption is the documentation for a column is a plain string |
| # so just make the comment for the column equal to that. |
| |
| if isinstance(col.doc, ColumnDoc): |
| comment = col.doc.doc |
| if col.doc.joinable: |
| join_table, join_col = col.doc.joinable.split('.') |
| else: |
| join_table, join_col = None, None |
| elif isinstance(col.doc, str): |
| comment = col.doc |
| join_table, join_col = None, None |
| else: |
| raise Exception('Unknown column documentation type ' |
| f'{table.table.class_name}::{col.column.name}') |
| |
| parsed_type = util.parse_type_with_cols(table.table, |
| [c.column for c in table.columns], |
| col.column.type) |
| docs_type = parsed_type.cpp_type |
| if docs_type == 'StringPool::Id': |
| docs_type = 'string' |
| |
| ref_class_name = None |
| if parsed_type.id_table and not col.is_implicit_id: |
| id_table_name = util.public_sql_name(parsed_type.id_table) |
| ref_class_name = parsed_type.id_table.class_name |
| |
| if not join_table and not join_col: |
| join_table = id_table_name |
| join_col = "id" |
| |
| return { |
| 'name': col.column.name, |
| 'type': docs_type, |
| 'comment': comment, |
| 'optional': parsed_type.is_optional, |
| 'refTableCppName': ref_class_name, |
| 'joinTable': join_table, |
| 'joinCol': join_col, |
| } |
| |
| |
| def main(): |
| parser = argparse.ArgumentParser() |
| parser.add_argument('--out', required=True) |
| parser.add_argument('inputs', nargs='*') |
| parser.add_argument('--relative-input-dir') |
| args = parser.parse_args() |
| |
| def get_relin_path(in_path: str): |
| if not args.relative_input_dir: |
| return in_path |
| return os.path.relpath(in_path, args.relative_input_dir) |
| |
| modules = [ |
| os.path.splitext(get_relin_path(i).replace('/', '.'))[0] |
| for i in args.inputs |
| ] |
| table_docs = [] |
| for parsed in util.parse_tables_from_modules(modules): |
| table = parsed.table |
| |
| # If there is no non-intrinsic alias for the table, don't |
| # include the table in the docs. |
| name = util.public_sql_name(table) |
| if name.startswith('__intrinsic_') or name.startswith('experimental_'): |
| continue |
| |
| doc = table.tabledoc |
| assert doc |
| cols = ( |
| gen_json_for_column(parsed, c) |
| for c in parsed.columns |
| if not c.is_ancestor) |
| table_docs.append({ |
| 'name': name, |
| 'cppClassName': table.class_name, |
| 'defMacro': table.class_name, |
| 'comment': '\n'.join(l.strip() for l in doc.doc.splitlines()), |
| 'parent': None, |
| 'parentDefName': table.parent.class_name if table.parent else '', |
| 'tablegroup': doc.group, |
| 'cols': [c for c in cols if c] |
| }) |
| |
| with open(args.out, 'w') as out: |
| json.dump(table_docs, out, indent=2) |
| out.write('\n') |
| |
| |
| if __name__ == '__main__': |
| exit(main()) |