blob: 583f33954ed8e619e23e592166381db1698254d4 [file] [log] [blame]
#!/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 runpy
import sys
from typing import Any
from typing import Dict
from typing import List
from typing import Optional
from typing import Union
# 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))
from python.generators.trace_processor_table.public import Column
from python.generators.trace_processor_table.public import ColumnDoc
from python.generators.trace_processor_table.public import Table
import python.generators.trace_processor_table.util as util
def gen_json_for_column(table: Table, col: Column,
doc: Union[ColumnDoc, str]) -> Optional[Dict[str, Any]]:
"""Generates the JSON documentation for a column in a table."""
# id and type columns should be skipped if the table specifies so.
is_skippable_col = col._is_auto_added_id or col._is_auto_added_type
if table.tabledoc.skip_id_and_type and is_skippable_col:
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(doc, ColumnDoc):
comment = doc.doc
if doc.joinable:
join_table, join_type = doc.joinable.split('.')
else:
join_table, join_type = None, None
elif isinstance(doc, str):
comment = doc
join_table, join_type = None, None
else:
raise Exception('Unknown column documentation type')
parsed_type = util.parse_type(table, col.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_auto_added_id:
id_table_name = util.public_sql_name_for_table(parsed_type.id_table)
ref_class_name = parsed_type.id_table.class_name
# We shouldn't really specify the join tables when it's a simple id join.
assert join_table is None
assert join_type is None
join_table = id_table_name
join_type = "id"
return {
'name': col.name,
'type': docs_type,
'comment': comment,
'optional': parsed_type.is_optional,
'refTableCppName': ref_class_name,
'joinTable': join_table,
'joinCol': join_type,
}
def main():
parser = argparse.ArgumentParser()
parser.add_argument('--out', required=True)
parser.add_argument('inputs', nargs='*')
args = parser.parse_args()
tables: List[Table] = []
for in_path in args.inputs:
tables.extend(runpy.run_path(in_path)['ALL_TABLES'])
for table in tables:
util.normalize_table_columns(table)
table_docs = []
for table in tables:
doc = table.tabledoc
cols = (
gen_json_for_column(table, c, doc.columns[c.name])
for c in table.columns
if c._is_self_column)
table_docs.append({
'name': util.public_sql_name_for_table(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())