| # -*- coding: utf-8 -*- |
| # Copyright 2021 The Chromium Authors |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| """Model of a structured metrics description xml file. |
| |
| This marshals an XML string into a Model, and validates that the XML is |
| semantically correct. The model can also be used to create a canonically |
| formatted version XML. |
| """ |
| |
| import xml.etree.ElementTree as ET |
| import textwrap as tw |
| import model_util as util |
| import re |
| |
| # Default key rotation period if not explicitly specified in the XML. |
| DEFAULT_KEY_ROTATION_PERIOD = 90 |
| |
| # Project name for event sequencing. |
| # |
| # This project name should be consistent with the name in structured.xml as well |
| # as the server. |
| EVENT_SEQUENCE_PROJECT_NAME = 'CrOSEvents' |
| |
| |
| def wrap(text, indent): |
| wrapper = tw.TextWrapper(width=80, |
| initial_indent=indent, |
| subsequent_indent=indent) |
| return wrapper.fill(tw.dedent(text)) |
| |
| |
| # TODO(crbug.com/1148168): This can be removed and replaced with textwrap.indent |
| # once this is run under python3. |
| def indent(text, prefix): |
| return '\n'.join(prefix + line if line else '' for line in text.split('\n')) |
| |
| |
| class Model: |
| """Represents all projects in the structured.xml file. |
| |
| A Model is initialized with an XML string representing the top-level of |
| the structured.xml file. This file is built from three building blocks: |
| metrics, events, and projects. These have the following attributes. |
| |
| METRIC |
| - summary |
| - data type |
| |
| EVENT |
| - summary |
| - one or more metrics |
| |
| PROJECT |
| - summary |
| - id specifier |
| - one or more owners |
| - one or more events |
| |
| The following is an example input XML. |
| |
| <structured-metrics> |
| <project name="MyProject"> |
| <owner>owner@chromium.org</owner> |
| <id>none</id> |
| <scope>profile</scope> |
| <summary> My project. </summary> |
| |
| <event name="MyEvent"> |
| <summary> My event. </summary> |
| <metric name="MyMetric" type="int"> |
| <summary> My metric. </summary> |
| </metric> |
| </event> |
| </project> |
| </structured-metrics> |
| |
| Calling str(model) will return a canonically formatted XML string. |
| """ |
| |
| OWNER_REGEX = r'^.+@(chromium\.org|google\.com)$' |
| NAME_REGEX = r'^[A-Za-z0-9_.]+$' |
| TYPE_REGEX = r'^(hmac-string|raw-string|int|double)$' |
| ID_REGEX = r'^(none|per-project|uma)$' |
| SCOPE_REGEX = r'^(profile|device)$' |
| KEY_REGEX = r'^[0-9]+$' |
| CROS_EVENTS_REGEX = r'(?i)(true|false|)$' |
| |
| def __init__(self, xml_string): |
| elem = ET.fromstring(xml_string) |
| util.check_attributes(elem, set()) |
| util.check_children(elem, {'project'}) |
| util.check_child_names_unique(elem, 'project') |
| |
| projects = util.get_compound_children(elem, 'project') |
| self.projects = [Project(p) for p in projects] |
| |
| def __repr__(self): |
| projects = '\n\n'.join(str(p) for p in self.projects) |
| |
| result = tw.dedent("""\ |
| <structured-metrics> |
| |
| {projects} |
| |
| </structured-metrics>""") |
| return result.format(projects=projects) |
| |
| |
| class Project: |
| """Represents a single structured metrics project. |
| |
| A Project is initialized with an XML node representing one project, eg: |
| |
| <project name="MyProject" cros_events="true"> |
| <owner>owner@chromium.org</owner> |
| <id>none</id> |
| <scope>project</scope> |
| <key-rotation>60</key-rotation> |
| <summary> My project. </summary> |
| |
| <event name="MyEvent"> |
| <summary> My event. </summary> |
| <metric name="MyMetric" type="int"> |
| <summary> My metric. </summary> |
| </metric> |
| </event> |
| </project> |
| |
| Calling str(project) will return a canonically formatted XML string. |
| """ |
| |
| def __init__(self, elem): |
| util.check_attributes(elem, {'name'}, {'cros_events'}) |
| util.check_children(elem, {'id', 'scope', 'summary', 'owner', 'event'}) |
| util.check_child_names_unique(elem, 'event') |
| |
| self.name = util.get_attr(elem, 'name', Model.NAME_REGEX) |
| self.id = util.get_text_child(elem, 'id', Model.ID_REGEX) |
| self.scope = util.get_text_child(elem, 'scope', Model.SCOPE_REGEX) |
| self.summary = util.get_text_child(elem, 'summary') |
| self.owners = util.get_text_children(elem, 'owner', Model.OWNER_REGEX) |
| |
| self.key_rotation_period = DEFAULT_KEY_ROTATION_PERIOD |
| cros_events_attr = util.get_optional_attr(elem, 'cros_events', |
| Model.CROS_EVENTS_REGEX) |
| if cros_events_attr: |
| self.is_event_sequence_project = cros_events_attr.lower() == 'true' |
| else: |
| self.is_event_sequence_project = False |
| |
| # Check if key-rotation is specified. If so, then change the |
| # key_rotation_period. |
| if elem.find('key-rotation') is not None: |
| self.key_rotation_period = util.get_text_child(elem, 'key-rotation', |
| Model.KEY_REGEX) |
| |
| self.events = [ |
| Event(e, self) for e in util.get_compound_children(elem, 'event') |
| ] |
| |
| def __repr__(self): |
| events = '\n\n'.join(str(e) for e in self.events) |
| events = indent(events, ' ') |
| summary = wrap(self.summary, indent=' ') |
| owners = '\n'.join(' <owner>{}</owner>'.format(o) for o in self.owners) |
| if self.is_event_sequence_project: |
| cros_events_attr = ' cros_events="true"' |
| else: |
| cros_events_attr = '' |
| |
| result = tw.dedent("""\ |
| <project name="{name}"{cros_events_attr}> |
| {owners} |
| <id>{id}</id> |
| <scope>{scope}</scope> |
| <key-rotation>{key_rotation}</key-rotation> |
| <summary> |
| {summary} |
| </summary> |
| |
| {events} |
| </project>""") |
| return result.format(name=self.name, |
| cros_events_attr=cros_events_attr, |
| owners=owners, |
| id=self.id, |
| scope=self.scope, |
| summary=summary, |
| key_rotation=self.key_rotation_period, |
| events=events) |
| |
| |
| class Event: |
| """Represents a single structured metrics event. |
| |
| An Event is initialized with an XML node representing one event, eg: |
| |
| <event name="MyEvent"> |
| <summary> My event. </summary> |
| <metric name="MyMetric" type="int"> |
| <summary> My metric. </summary> |
| </metric> |
| </event> |
| |
| Calling str(event) will return a canonically formatted XML string. |
| """ |
| |
| def __init__(self, elem, project): |
| util.check_attributes(elem, {'name'}) |
| |
| if project.is_event_sequence_project: |
| expected_children = {'summary'} |
| else: |
| expected_children = {'summary', 'metric'} |
| |
| util.check_children(elem, expected_children) |
| |
| util.check_child_names_unique(elem, 'metric') |
| |
| self.name = util.get_attr(elem, 'name', Model.NAME_REGEX) |
| self.summary = util.get_text_child(elem, 'summary') |
| self.metrics = [ |
| Metric(m, project) for m in util.get_compound_children( |
| elem, 'metric', project.is_event_sequence_project) |
| ] |
| |
| def __repr__(self): |
| metrics = '\n'.join(str(m) for m in self.metrics) |
| metrics = indent(metrics, ' ') |
| summary = wrap(self.summary, indent=' ') |
| result = tw.dedent("""\ |
| <event name="{name}"> |
| <summary> |
| {summary} |
| </summary> |
| {metrics} |
| </event>""") |
| return result.format(name=self.name, summary=summary, metrics=metrics) |
| |
| |
| class Metric: |
| """Represents a single metric. |
| |
| A Metric is initialized with an XML node representing one metric, eg: |
| |
| <metric name="MyMetric" type="int"> |
| <summary> My metric. </summary> |
| </metric> |
| |
| Calling str(metric) will return a canonically formatted XML string. |
| """ |
| |
| def __init__(self, elem, project): |
| util.check_attributes(elem, {'name', 'type'}) |
| util.check_children(elem, {'summary'}) |
| |
| self.name = util.get_attr(elem, 'name', Model.NAME_REGEX) |
| self.type = util.get_attr(elem, 'type', Model.TYPE_REGEX) |
| self.summary = util.get_text_child(elem, 'summary') |
| |
| if self.type == 'raw-string' and ( |
| project.id != 'none' and project.name != EVENT_SEQUENCE_PROJECT_NAME): |
| util.error( |
| elem, 'raw-string metrics must be in a project with id type ' |
| "'none' or project name '{}', but {} has id type '{}'".format( |
| EVENT_SEQUENCE_PROJECT_NAME, project.name, project.id)) |
| |
| def __repr__(self): |
| summary = wrap(self.summary, indent=' ') |
| result = tw.dedent("""\ |
| <metric name="{name}" type="{type}"> |
| <summary> |
| {summary} |
| </summary> |
| </metric>""") |
| return result.format(name=self.name, type=self.type, summary=summary) |