blob: 8f8eb7877f4938d029fa760604b611aca36f2866 [file] [log] [blame]
#!/usr/bin/env python3
import argparse
import json
import sys
r"""
Validates the PEL message registry JSON, which includes checking it against
a JSON schema using the jsonschema module as well as doing some extra checks
that can't be encoded in the schema.
"""
def check_duplicate_names(registry_json):
r"""
Check that there aren't any message registry entries with the same
'Name' field. There may be a use case for this in the future, but there
isn't right now.
registry_json: The message registry JSON
"""
names = []
for entry in registry_json['PELs']:
if entry['Name'] in names:
sys.exit("Found multiple uses of error {}".format(entry['Name']))
else:
names.append(entry['Name'])
def check_duplicate_reason_codes(registry_json):
r"""
Check that there aren't any message registry entries with the same
'ReasonCode' field.
registry_json: The message registry JSON
"""
reasonCodes = []
for entry in registry_json['PELs']:
if entry['SRC']['ReasonCode'] in reasonCodes:
sys.exit("Found duplicate SRC reason code {}".format(
entry['SRC']['ReasonCode']))
else:
reasonCodes.append(entry['SRC']['ReasonCode'])
def check_component_id(registry_json):
r"""
Check that the upper byte of the ComponentID field matches the upper byte
of the ReasonCode field, but not on "11" type SRCs where they aren't
supposed to match.
registry_json: The message registry JSON
"""
for entry in registry_json['PELs']:
# Don't check on "11" SRCs as those reason codes aren't supposed to
# match the component ID.
if entry['SRC'].get('Type', '') == "11":
continue
if 'ComponentID' in entry:
id = int(entry['ComponentID'], 16)
reason_code = int(entry['SRC']['ReasonCode'], 16)
if (id & 0xFF00) != (reason_code & 0xFF00):
sys.exit("Found mismatching component ID {} vs reason "
"code {} for error {}".format(
entry['ComponentID'],
entry['SRC']['ReasonCode'],
entry['Name']))
def check_message_args(registry_json):
r"""
Check that if the Message field uses the '%' style placeholders that there
are that many entries in the MessageArgSources field. Also checks that
the MessageArgSources field is present but only if there are placeholders.
registry_json: The message registry JSON
"""
for entry in registry_json['PELs']:
num_placeholders = entry['Documentation']['Message'].count('%')
if num_placeholders == 0:
continue
if 'MessageArgSources' not in entry['Documentation']:
sys.exit("Missing MessageArgSources property for error {}".
format(entry['Name']))
if num_placeholders != \
len(entry['Documentation']['MessageArgSources']):
sys.exit("Different number of placeholders found in "
"Message vs MessageArgSources for error {}".
format(entry['Name']))
def validate_schema(registry, schema):
r"""
Validates the passed in JSON against the passed in schema JSON
registry: Path of the file containing the registry JSON
schema: Path of the file containing the schema JSON
Use None to skip the pure schema validation
"""
with open(registry) as registry_handle:
registry_json = json.load(registry_handle)
if schema:
import jsonschema
with open(schema) as schema_handle:
schema_json = json.load(schema_handle)
try:
jsonschema.validate(registry_json, schema_json)
except jsonschema.ValidationError as e:
print(e)
sys.exit("Schema validation failed")
check_duplicate_names(registry_json)
check_duplicate_reason_codes(registry_json)
check_component_id(registry_json)
check_message_args(registry_json)
if __name__ == '__main__':
parser = argparse.ArgumentParser(
description='PEL message registry processor')
parser.add_argument('-v', '--validate', action='store_true',
dest='validate',
help='Validate the JSON using the schema')
parser.add_argument('-s', '--schema-file', dest='schema_file',
help='The message registry JSON schema file')
parser.add_argument('-r', '--registry-file', dest='registry_file',
help='The message registry JSON file')
parser.add_argument('-k', '--skip-schema-validation', action='store_true',
dest='skip_schema',
help='Skip running schema validation. '
'Only do the extra checks.')
args = parser.parse_args()
if args.validate:
if not args.schema_file:
sys.exit("Schema file required")
if not args.registry_file:
sys.exit("Registry file required")
schema = args.schema_file
if args.skip_schema:
schema = None
validate_schema(args.registry_file, schema)