blob: 8526d49b8ebe40dcc0cf6a50f76443f07a46e9c9 [file] [log] [blame]
# Copyright 2019 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Objects for describing template code to be generated from structured.xml."""
import hashlib
import os
import re
import struct
from model import _EVENT_TYPE
from model import _PROJECT_TYPE
from model import _METRIC_TYPE
def sanitize_name(name):
return re.sub('[^0-9a-zA-Z_]', '_', name)
def camel_to_snake(name):
pat = '((?<=[a-z0-9])[A-Z]|(?!^)[A-Z](?=[a-z]))'
return re.sub(pat, r'_\1', name).lower()
def HashName(name):
# This must match the hash function in base/metrics/metric_hashes.cc
# >Q: 8 bytes, big endian.
return struct.unpack('>Q', hashlib.md5(name).digest()[:8])[0]
class FileInfo(object):
def __init__(self, relpath, basename):
self.dir_path = relpath
self.guard_path = sanitize_name(os.path.join(relpath, basename)).upper()
class ProjectInfo(object):
def __init__(self, project_obj):
self.name = sanitize_name(project_obj['name'])
self.namespace = camel_to_snake(self.name)
self.name_hash = HashName(self.name)
id_type = project_obj['id']['text']
if id_type == 'uma':
self.id_type = 'kUmaId'
elif id_type == 'per-project':
self.id_type = 'kProjectId'
elif id_type == 'none':
self.id_type = 'kUnidentified'
else:
raise Exception(
"Structured metrics project '{}' has invalid id field '{}'".format(
self.name, id_type))
class EventInfo(object):
def __init__(self, event_obj):
self.raw_name = event_obj['name']
self.name = sanitize_name(event_obj['name'])
self.name_hash = HashName(event_obj['name'])
class MetricInfo(object):
def __init__(self, json_obj):
self.raw_name = json_obj['name']
self.name = sanitize_name(json_obj['name'])
self.hash = HashName(json_obj['name'])
if json_obj['kind'] == 'hashed-string':
self.type = 'std::string&'
self.setter = 'AddStringMetric'
elif json_obj['kind'] == 'int':
self.type = 'int'
self.setter = 'AddIntMetric'
else:
raise Exception("Unexpected metric kind: " + json_obj['kind'])
class Template(object):
"""Template for producing code from structured.xml."""
def __init__(self, basename, file_template, project_template, event_template,
metric_template):
self.basename = basename
self.file_template = file_template
self.project_template = project_template
self.event_template = event_template
self.metric_template = metric_template
def _StampMetricCode(self, file_info, event_info, metric):
"""Stamp a metric by creating name hash constant based on the metric name,
and a setter method."""
return self.metric_template.format(
file=file_info,
event=event_info,
metric=MetricInfo(metric))
def _StampEventCode(self, file_info, project_info, event):
"""Stamp an event class by creating a skeleton of the class based on the
event name, and then stamping code for each metric within it."""
event_info = EventInfo(event)
metric_code = ''.join(
self._StampMetricCode(file_info, event_info, metric)
for metric in event[_METRIC_TYPE.tag])
return self.event_template.format(file=file_info,
project=project_info,
event=event_info,
metric_code=metric_code)
def _StampProjectCode(self, file_info, project):
"""Stamp a project by stamping classes for all constituent events."""
project_info = ProjectInfo(project)
event_code = ''.join(
self._StampEventCode(file_info, project_info, event)
for event in project[_EVENT_TYPE.tag])
return self.project_template.format(file=file_info,
project=project_info,
event_code=event_code)
def _StampFileCode(self, relpath, data):
"""Stamp a file by creating a class for each event within each project, and
a list of all project name hashes."""
file_info = FileInfo(relpath, self.basename)
project_code = [
self._StampProjectCode(file_info, project)
for project in data[_PROJECT_TYPE.tag]
]
project_code = ''.join(project_code)
project_names = {project['name'] for project in data[_PROJECT_TYPE.tag]}
project_name_hashes = [
'UINT64_C(%s)' % HashName(name) for name in sorted(list(project_names))
]
project_name_hashes = '{' + ', '.join(project_name_hashes) + '}'
return self.file_template.format(file=file_info,
project_code=project_code,
project_name_hashes=project_name_hashes)
def WriteFile(self, outdir, relpath, data):
"""Generates code and writes it to a file.
Args:
relpath: The path to the file in the source tree.
rootdir: The root of the path the file should be written to.
data: The parsed structured.xml data.
"""
output = open(os.path.join(outdir, self.basename), 'w')
output.write(self._StampFileCode(relpath, data))
output.close()