Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Analyzer - Generate baseline given results from multiple nodes. #575

Merged
merged 19 commits into from
Nov 22, 2023
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
47 changes: 47 additions & 0 deletions docs/cli.md
Original file line number Diff line number Diff line change
Expand Up @@ -275,6 +275,53 @@ Run result summary and output the results in html format:
sb result summary --data-file outputs/results-summary.jsonl --rule-file rule.yaml --output-file-format html
```

### `sb result generate-baseline`

Generate the baseline file automatically from multiple machines results according to rules defined in rule file.

```bash title="SB CLI"
sb result summary --data-file
yukirora marked this conversation as resolved.
Show resolved Hide resolved
--summary-rule-file
[--diagnosis-rule-file]
[--baseline-file]
[--decimal-place-value]
[--output-dir]
```

#### Required arguments

| Name | Description |
|-----------------------------|-----------------------------------------|
| `--data-file` `-d` | Path to raw data file. |
| `--summary-rule-file` `-sr` | Path to summary rule file. |

#### Optional arguments

| Name | Default | Description |
|-------------------------------|---------|-----------------------------------------------------------------------------|
| `--diagnosis-rule-file` `-dr` | `None` | Path to diagnosis rule file. Default: None. |
| `--baseline-file` `-b` | `None` | Path to previous baseline file. Default: None. |
| `--decimal-place-value` | 2 | Number of valid decimal places to show in output. Default: 2. |
| `--output-dir` | `None` | Path to output directory, outputs/{datetime} will be used if not specified. |

#### Global arguments

| Name | Default | Description |
|---------------|---------|--------------------|
| `--help` `-h` | N/A | Show help message. |

#### Examples

Run result generate-baseline to generate baseline.json file:
```bash title="SB CLI"
sb result generate-baseline --data-file outputs/results-summary.jsonl --summary-rule-file summary-rule.yaml --diagnosis-rule-file diagnosis-rule.yaml
```

Run result generate-baseline and merge with previous baseline:
```bash title="SB CLI"
sb result generate-baseline --data-file outputs/results-summary.jsonl --summary-rule-file summary-rule.yaml --diagnosis-rule-file diagnosis-rule.yaml --baseline-file previous-baseline.json
```

### `sb run`

Run the SuperBench benchmarks distributedly.
Expand Down
49 changes: 49 additions & 0 deletions docs/user-tutorial/baseline-generation.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
---
id: baseline-generation
yukirora marked this conversation as resolved.
Show resolved Hide resolved
---

# Baseline Generation

## Introduction

This tool is to generate a baseline json file based on the raw benchmark results of multiple machines.

## Usage

1. [Install SuperBench](../getting-started/installation.mdx) on the local machine.

2. Prepare the raw data and rule files on the local machine.

3. Generate the baseline file automatically using `sb result generate-baseline` command. The detailed command can be found from [SuperBench CLI](../cli.md).

```bash
sb result generate-baseline --data-file ./results-summary.jsonl --summary-rule-file ./summary-rule.yaml --diagnosis-rule-file ./diagnosis-rule.yaml --output-dir ${output-dir}
```

4. Find the output result file named 'baseline.json' under ${output_dir}.

## Input

The input includes 4 files:

- **Raw Data**: jsonl file including multiple nodes' results automatically generated by SuperBench runner.

:::tip Tips
Raw data file can be found at ${output-dir}/results-summary.jsonl after each successful run.
:::

- **Summary Rule File**: It uses YAML format and defines how to generate the result summary including how to classify the metrics and what statistical methods (P50, mean, etc.) are applied.

- **Diagnosis Rule File(optional)**: It uses YAML format and includes each metrics' rules to filter defective machines for diagnosis, and will not filter machines if not specified.

- **Previous Baseline File(optional)**: It is baseline file in json format that got from previous run and plan to merge into the latest baseline.

### Rule File

**Summary Rule File** is the same with rule file defined in [Result Summary](./result-summary.md)

**Diagnosis Rule File** is the same with rule file defined in [Data Diagnosis](./data-diagnosis.md)

## Output

The baseline file (baseline.json) from multiple machines will be generated under ${output_dir}.
3 changes: 2 additions & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -160,11 +160,12 @@ def run(self):
'matplotlib>=3.0.0',
'natsort>=7.1.1',
'networkx>=2.5',
'numpy>=1.19.2',
'numpy>=1.20.3',
'omegaconf==2.0.6',
'openpyxl>=3.0.7',
'pandas>=1.1.5',
'pssh @ git+https://github.com/lilydjwg/[email protected]',
'python-dateutil>=2.8.2',
'pyyaml>=5.3',
'requests>=2.27.1',
'seaborn>=0.11.2',
Expand Down
6 changes: 5 additions & 1 deletion superbench/analyzer/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,5 +8,9 @@
from superbench.analyzer.diagnosis_rule_op import RuleOp, DiagnosisRuleType
from superbench.analyzer.summary_op import SummaryOp, SummaryType
from superbench.analyzer.result_summary import ResultSummary
from superbench.analyzer.baseline_generation import BaselineGeneration

__all__ = ['DataDiagnosis', 'DiagnosisRuleType', 'RuleOp', 'RuleBase', 'SummaryOp', 'SummaryType', 'ResultSummary']
__all__ = [
'DataDiagnosis', 'DiagnosisRuleType', 'RuleOp', 'RuleBase', 'SummaryOp', 'SummaryType', 'ResultSummary',
'BaselineGeneration'
]
193 changes: 193 additions & 0 deletions superbench/analyzer/baseline_generation.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
# Copyright (c) Microsoft Corporation.
# Licensed under the MIT license.

"""A module for baseline generation."""

from copy import deepcopy
import json
import re

from joblib import Parallel, delayed
import pandas as pd

from superbench.common.utils import logger
from superbench.analyzer import file_handler
from superbench.analyzer import data_analysis
from superbench.analyzer import DataDiagnosis
from superbench.analyzer import ResultSummary
from superbench.analyzer.diagnosis_rule_op import RuleOp, DiagnosisRuleType


class BaselineGeneration(DataDiagnosis):
"""The class to generate baseline for raw data."""
def fix_threshold_outlier_detection(self, data_series, single_metric_with_baseline, metric, rule_op):
"""Fix threshold outlier detection algorithm.

Step 0: Put all data in the collection.
Step 1: Regenerate the collection.
Calculate the average number in the collection as the baseline.
Remove all data which cannot pass the fix threshold based on the new baseline.
Step 2: If no data has been removed from Step 1, go to Step 3; otherwise, go to Step 1.
Step 3: Use the baseline and fix threshold for Outlier Detection.

Args:
data_series (pd.Series): data of the metric.
single_metric_with_baseline (dict): baseline of the single metric in 'metrics' in 2-layer dict format.
metric (str): the name of the metric to execute the algorithm.
rule_op (function): diagnosis rule op function.

Returns:
tuple: the baseline of the metric, normal data of the metric.
"""
if single_metric_with_baseline['metrics'][metric] \
is not None and single_metric_with_baseline['metrics'][metric] != -1:
return single_metric_with_baseline['metrics'][metric]
tmp_single_metric_with_baseline = deepcopy(single_metric_with_baseline)
tmp_single_metric_with_baseline['metrics'] = {}
clean = False
while clean is False:
clean = True
baseline_val = data_series.mean()
for val in data_series.index:
tmp_single_metric_with_baseline['metrics'][metric] = baseline_val
if baseline_val == 0:
break
data_row = pd.Series([data_series[val]], index=[metric])
details = []
categories = set()
summary_data_row = pd.Series(index=[metric], dtype=float)
violated_num = rule_op(data_row, tmp_single_metric_with_baseline, summary_data_row, details, categories)
if violated_num:
data_series = data_series.drop(val)
clean = False
baseline = tmp_single_metric_with_baseline['metrics'][metric]
return baseline, data_series

def get_aggregate_data(self, raw_data_file, summary_rule_file):
r"""Aggregate raw data according to the summary rule file.

If the metric is aggregated by rank (:\d+), remove the rank info to generate the metric name and aggregate data.
If the metric is aggregated by regex pattern, aggregate the data and copy to all metrics matches this pattern.

Args:
raw_data_file (str): the file name of the raw data file.
summary_rule_file (str): the file name of the summary rule file.

Returns:
DataFrame: aggregated data
"""
self.rs = ResultSummary()
rules = self.rs._preprocess(raw_data_file, summary_rule_file)
# parse rules for result summary
if not self.rs._parse_rules(rules):
return
aggregated_df = pd.DataFrame()
for rule in self.rs._sb_rules:
single_metric_rule = self.rs._sb_rules[rule]
metrics = list(single_metric_rule['metrics'].keys())
data_df_of_rule = self.rs._raw_data_df[metrics]
if self.rs._sb_rules[rule]['aggregate']:
# if aggregate is True, aggregate in ranks
if self.rs._sb_rules[rule]['aggregate'] is True:
data_df_of_rule = data_analysis.aggregate(data_df_of_rule)
# if aggregate is not empty and is a pattern in regex, aggregate according to pattern
else:
pattern = self.rs._sb_rules[rule]['aggregate']
data_df_of_rule_with_short_name = data_analysis.aggregate(data_df_of_rule, pattern)
data_df_of_rule = pd.DataFrame(columns=metrics)
# restore the columns of data_fd to full metric names
for metric in metrics:
short = ''
match = re.search(pattern, metric)
if match:
metric_in_list = list(metric)
for i in range(1, len(match.groups()) + 1):
metric_in_list[match.start(i):match.end(i)] = '*'
short = ''.join(metric_in_list)
data_df_of_rule[metric] = data_df_of_rule_with_short_name[short]
aggregated_df = pd.concat([aggregated_df, data_df_of_rule], axis=1)
return aggregated_df

def generate_baseline(self, algo, aggregated_df, diagnosis_rule_file, baseline):
"""Generate the baseline in json format.

Args:
algo (str): the algorithm to generate the baseline.
aggregated_df (DataFrame): aggregated data.
diagnosis_rule_file (str): the file name of the diagnosis rules which used in fix_threshold algorithm.
baseline (dict): existing baseline of some metrics.

Returns:
dict: baseline of metrics defined in diagnosis_rule_files for fix_threshold algorithm or
defined in rule_summary_files for mean.
"""
# re-organize metrics by benchmark names
self._benchmark_metrics_dict = self._get_metrics_by_benchmarks(list(self._raw_data_df.columns))
if algo == 'mean':
mean_df = self._raw_data_df.mean()
for metric in self._raw_data_df.columns:
if metric in baseline:
return baseline[metric]
baseline[metric] = mean_df[metric]
elif algo == 'fix_threshold':
# read diagnosis rules
rules = file_handler.read_rules(diagnosis_rule_file)
if not self._parse_rules_and_baseline(rules, baseline):
return baseline
else:
for rule in self._sb_rules:
single_metric_rule = self._sb_rules[rule]
metrics = list(single_metric_rule['metrics'].keys())
function_name = self._sb_rules[rule]['function']
rule_op = RuleOp.get_rule_func(DiagnosisRuleType(function_name))
outputs = Parallel(n_jobs=-1)(
delayed(self.fix_threshold_outlier_detection)
(aggregated_df[metric], single_metric_rule, metric, rule_op) for metric in metrics
)
for index, out in enumerate(outputs):
baseline[metrics[index]] = out[0]
aggregated_df[metrics[index]] = out[1]
return baseline

def run(
self, raw_data_file, summary_rule_file, diagnosis_rule_file, pre_baseline_file, algorithm, output_dir, digit=2
):
"""Export baseline to json file.

Args:
raw_data_file (str): Path to raw data jsonl file.
summary_rule_file (str): the file name of the summary rule file.
diagnosis_rule_file (str): the file name of the diagnosis rules which used in fix_threshold algorithm.
pre_baseline_file (str): the file name of the previous baseline file.
algorithm (str): the algorithm to generate the baseline.
output_dir (str): the directory to save the baseline file.
digit (int): the number of digits after the decimal point.
"""
try:
# aggregate results from different devices
self._raw_data_df = self.get_aggregate_data(raw_data_file, summary_rule_file)
# read existing baseline
baseline = {}
if pre_baseline_file:
baseline = file_handler.read_baseline(pre_baseline_file)
# generate baseline accordint to rules in diagnosis and fix threshold outlier detection method
baseline = self.generate_baseline(algorithm, self._raw_data_df, diagnosis_rule_file, baseline)
for metric in baseline:
val = baseline[metric]
if metric in self._raw_data_df:
if isinstance(self._raw_data_df[metric].iloc[0], float):
baseline[metric] = f'%.{digit}g' % val if abs(val) < 1 else f'%.{digit}f' % val
elif isinstance(self._raw_data_df[metric].iloc[0], int):
baseline[metric] = int(val)
else:
try:
baseline[metric] = float(val)
except Exception as e:
logger.error('Analyzer: {} baseline is not numeric, msg: {}'.format(metric, str(e)))
baseline = json.dumps(baseline, indent=2, sort_keys=True)
baseline = re.sub(r': \"(\d+.?\d*)\"', r': \1', baseline)
with open(output_dir + '/baseline.json', mode='w') as f:
yukirora marked this conversation as resolved.
Show resolved Hide resolved
f.write(baseline)

except Exception as e:
logger.error('Analyzer: generate baseline failed, msg: {}'.format(str(e)))
2 changes: 1 addition & 1 deletion superbench/analyzer/file_handler.py
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ def read_raw_data(raw_data_path):
raw_data_df = raw_data_df.rename(raw_data_df['node'])
raw_data_df = raw_data_df.drop(columns=['node'])
except Exception as e:
logger.log_and_raise(exception=IOError, msg='Analyzer: invalid raw data fomat - {}'.format(str(e)))
logger.log_and_raise(exception=IOError, msg='Analyzer: invalid raw data format - {}'.format(str(e)))
return raw_data_df


Expand Down
13 changes: 13 additions & 0 deletions superbench/cli/_commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ def load_command_table(self, args):
with CommandGroup(self, 'result', 'superbench.cli._result_handler#{}') as g:
g.command('diagnosis', 'diagnosis_command_handler')
g.command('summary', 'summary_command_handler')
g.command('generate-baseline', 'generate_baseline_command_handler')
return super().load_command_table(args)

def load_arguments(self, command):
Expand Down Expand Up @@ -74,6 +75,18 @@ def load_arguments(self, command):
with ArgumentsContext(self, 'result') as ac:
ac.argument('raw_data_file', options_list=('--data-file', '-d'), type=str, help='Path to raw data file.')
ac.argument('rule_file', options_list=('--rule-file', '-r'), type=str, help='Path to rule file.')
ac.argument(
'summary_rule_file',
options_list=('--summary-rule-file', '-sr'),
type=str,
help='Path to summary rule file.'
)
ac.argument(
'diagnosis_rule_file',
options_list=('--diagnosis-rule-file', '-dr'),
type=str,
help='Path to diagnosis rule file.'
)
ac.argument(
'baseline_file', options_list=('--baseline-file', '-b'), type=str, help='Path to baseline file.'
)
Expand Down
21 changes: 21 additions & 0 deletions superbench/cli/_help.py
Original file line number Diff line number Diff line change
Expand Up @@ -183,6 +183,27 @@
--output-file-format html
""".format(cli_name=CLI_NAME)

helps['result generate-baseline'] = """
type: command
short-summary: >
Generate the baseline of benchmarking results from jsonline file
according to rules defined in rule file.
examples:
- name: run result generate-baseline to generate baseline.json file
text: >
{cli_name} result generate-baseline
--data-file outputs/results-summary.jsonl
--summary-rule-file summary-rule.yaml
--diagnosis-rule-file diagnosis-rule.yaml
- name: run result generate-baseline and merge with previous baseline
text: >
{cli_name} result generate-baseline
--data-file outputs/results-summary.jsonl
--summary-rule-file summary-rule.yaml
--diagnosis-rule-file diagnosis-rule.yaml
--baseline-file previous-baseline.json
""".format(cli_name=CLI_NAME)


class SuperBenchCLIHelp(CLIHelp):
"""SuperBench CLI help loader."""
Expand Down
Loading
Loading