blob: 04339bfa029cc09f6c6d835a1caf0cc3b84cb8de [file] [log] [blame] [edit]
#!/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
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': util.public_sql_name(table),
'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())