2019-09-11 21:17:02 +03:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
import argparse
|
2019-10-18 03:23:53 +03:00
|
|
|
import json
|
2019-09-11 21:17:02 +03:00
|
|
|
import os
|
2019-10-18 14:59:24 +03:00
|
|
|
import pickle
|
2019-10-22 19:42:38 +03:00
|
|
|
import shelve
|
2019-09-11 21:17:02 +03:00
|
|
|
import subprocess
|
|
|
|
import tarfile
|
2019-09-18 14:10:22 +03:00
|
|
|
from datetime import datetime
|
2019-09-11 21:17:02 +03:00
|
|
|
from logging import INFO, basicConfig, getLogger
|
|
|
|
|
2019-09-18 14:10:22 +03:00
|
|
|
import dateutil.parser
|
|
|
|
from dateutil.relativedelta import relativedelta
|
2019-09-19 02:32:01 +03:00
|
|
|
from tqdm import tqdm
|
2019-09-11 21:17:02 +03:00
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
from bugbug import commit_features, db, repository, test_scheduling
|
2019-10-18 15:33:53 +03:00
|
|
|
from bugbug.utils import ExpQueue, download_check_etag, zstd_compress, zstd_decompress
|
2019-09-11 21:17:02 +03:00
|
|
|
|
|
|
|
basicConfig(level=INFO)
|
|
|
|
logger = getLogger(__name__)
|
|
|
|
|
2019-09-24 23:42:44 +03:00
|
|
|
JOBS_TO_CONSIDER = ("test-", "build-")
|
2019-11-05 14:31:04 +03:00
|
|
|
JOBS_TO_IGNORE = ("build-docker-image-",)
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-11-07 14:16:52 +03:00
|
|
|
ADR_CACHE_DB = "data/adr_cache.tar"
|
|
|
|
db.register(
|
|
|
|
ADR_CACHE_DB,
|
2019-11-09 00:13:10 +03:00
|
|
|
"https://community-tc.services.mozilla.com/api/index/v1/task/project.relman.bugbug.data_test_scheduling_history_push_data.latest/artifacts/public/adr_cache.tar.zst",
|
2019-11-15 02:16:22 +03:00
|
|
|
2,
|
2019-11-07 14:16:52 +03:00
|
|
|
support_files=[],
|
|
|
|
)
|
2019-11-09 00:13:10 +03:00
|
|
|
PUSH_DATA_URL = "https://community-tc.services.mozilla.com/api/index/v1/task/project.relman.bugbug.data_test_scheduling_history_push_data.latest/artifacts/public/push_data.json.zst"
|
2019-09-11 21:17:02 +03:00
|
|
|
|
2019-09-24 23:32:16 +03:00
|
|
|
TRAINING_MONTHS = 6
|
|
|
|
|
2019-09-11 21:17:02 +03:00
|
|
|
|
2019-11-15 01:04:57 +03:00
|
|
|
def filter_tasks(tasks, all_tasks):
|
2019-11-10 21:57:05 +03:00
|
|
|
return tuple(
|
|
|
|
task
|
|
|
|
for task in tasks
|
2019-11-15 01:04:57 +03:00
|
|
|
if task in all_tasks
|
|
|
|
and any(task.startswith(j) for j in JOBS_TO_CONSIDER)
|
2019-11-10 21:57:05 +03:00
|
|
|
and not any(task.startswith(j) for j in JOBS_TO_IGNORE)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-09-11 21:17:02 +03:00
|
|
|
class Retriever(object):
|
2019-10-18 15:08:08 +03:00
|
|
|
def __init__(self):
|
2019-09-18 14:10:22 +03:00
|
|
|
os.makedirs("data", exist_ok=True)
|
2019-09-11 21:17:02 +03:00
|
|
|
|
2019-10-18 15:08:08 +03:00
|
|
|
def retrieve_push_data(self):
|
2019-09-11 21:17:02 +03:00
|
|
|
# Download previous cache.
|
2019-11-07 14:16:52 +03:00
|
|
|
cache_path = os.path.splitext(ADR_CACHE_DB)[0]
|
|
|
|
if not db.is_old_version(ADR_CACHE_DB):
|
|
|
|
db.download(ADR_CACHE_DB)
|
|
|
|
if os.path.exists(ADR_CACHE_DB):
|
|
|
|
with tarfile.open(ADR_CACHE_DB, "r") as tar:
|
2019-09-18 14:10:22 +03:00
|
|
|
tar.extractall()
|
2019-11-07 14:16:52 +03:00
|
|
|
assert os.path.exists(cache_path), "Decompressed adr cache exists"
|
2019-09-11 21:17:02 +03:00
|
|
|
|
|
|
|
# Setup adr cache configuration.
|
2019-09-12 13:35:04 +03:00
|
|
|
os.makedirs(os.path.expanduser("~/.config/adr"), exist_ok=True)
|
2019-09-11 21:17:02 +03:00
|
|
|
with open(os.path.expanduser("~/.config/adr/config.toml"), "w") as f:
|
|
|
|
f.write(
|
|
|
|
f"""[adr.cache.stores]
|
2019-11-07 14:16:52 +03:00
|
|
|
file = {{ driver = "file", path = "{os.path.abspath(cache_path)}" }}
|
2019-10-09 17:01:02 +03:00
|
|
|
"""
|
2019-09-11 21:17:02 +03:00
|
|
|
)
|
|
|
|
|
2019-09-24 23:32:16 +03:00
|
|
|
# We'll use the past TRAINING_MONTHS months only for training the model,
|
|
|
|
# but we use 3 months more than that to calculate the failure statistics.
|
2019-09-11 21:17:02 +03:00
|
|
|
subprocess.run(
|
|
|
|
[
|
|
|
|
"run-adr",
|
|
|
|
"ahal/ci-recipes",
|
|
|
|
"recipe",
|
|
|
|
"-o",
|
2019-09-18 14:10:22 +03:00
|
|
|
os.path.abspath("push_data.json"),
|
2019-09-11 21:17:02 +03:00
|
|
|
"-f",
|
|
|
|
"json",
|
|
|
|
"push_data",
|
|
|
|
"--",
|
|
|
|
"--from",
|
2019-09-24 23:32:16 +03:00
|
|
|
f"today-{TRAINING_MONTHS + 3}month",
|
2019-09-11 21:17:02 +03:00
|
|
|
"--to",
|
2019-11-14 21:56:04 +03:00
|
|
|
"today-3day",
|
2019-09-11 21:17:02 +03:00
|
|
|
"--branch",
|
|
|
|
"autoland",
|
|
|
|
],
|
|
|
|
check=True,
|
2019-09-13 11:09:33 +03:00
|
|
|
stdout=subprocess.DEVNULL, # Redirect to /dev/null, as the logs are too big otherwise.
|
2019-09-11 21:17:02 +03:00
|
|
|
)
|
|
|
|
|
2019-11-07 14:16:52 +03:00
|
|
|
with tarfile.open(ADR_CACHE_DB, "w") as tar:
|
|
|
|
tar.add(cache_path)
|
|
|
|
zstd_compress(ADR_CACHE_DB)
|
2019-10-18 15:08:08 +03:00
|
|
|
|
2019-10-18 15:33:53 +03:00
|
|
|
zstd_compress("push_data.json")
|
|
|
|
|
2019-10-18 15:08:08 +03:00
|
|
|
def generate_test_scheduling_history(self):
|
2019-10-18 15:33:53 +03:00
|
|
|
if not os.path.exists("push_data.json"):
|
|
|
|
download_check_etag(PUSH_DATA_URL, "push_data.json.zst")
|
|
|
|
zstd_decompress("push_data.json")
|
|
|
|
assert os.path.exists(
|
|
|
|
"push_data.json"
|
|
|
|
), "Decompressed push data file exists"
|
|
|
|
|
2019-10-18 15:08:08 +03:00
|
|
|
# Get the commits DB.
|
|
|
|
if db.is_old_version(repository.COMMITS_DB) or not db.exists(
|
|
|
|
repository.COMMITS_DB
|
|
|
|
):
|
|
|
|
db.download(repository.COMMITS_DB, force=True)
|
|
|
|
|
2019-09-24 23:32:16 +03:00
|
|
|
HISTORY_DATE_START = datetime.now() - relativedelta(months=TRAINING_MONTHS)
|
2019-09-18 14:10:22 +03:00
|
|
|
|
|
|
|
HISTORICAL_TIMESPAN = 56
|
|
|
|
|
2019-10-18 14:59:24 +03:00
|
|
|
if not db.is_old_version(test_scheduling.TEST_SCHEDULING_DB):
|
|
|
|
db.download(test_scheduling.TEST_SCHEDULING_DB, support_files_too=True)
|
|
|
|
|
|
|
|
for test_data in test_scheduling.get_test_scheduling_history():
|
|
|
|
pass
|
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
last_node = test_data["revs"][0]
|
2019-10-18 14:59:24 +03:00
|
|
|
else:
|
|
|
|
last_node = None
|
|
|
|
|
2019-10-22 19:42:38 +03:00
|
|
|
past_failures = shelve.open(
|
|
|
|
"data/past_failures.shelve",
|
|
|
|
protocol=pickle.HIGHEST_PROTOCOL,
|
|
|
|
writeback=True,
|
|
|
|
)
|
2019-10-09 17:02:58 +03:00
|
|
|
|
2019-10-22 19:42:38 +03:00
|
|
|
push_num = past_failures["push_num"] if "push_num" in past_failures else 0
|
2019-10-09 17:02:58 +03:00
|
|
|
|
2019-10-22 19:42:38 +03:00
|
|
|
def get_and_update_past_failures(type_, task, items, push_num, is_regression):
|
2019-10-09 17:02:58 +03:00
|
|
|
values_total = []
|
|
|
|
values_prev_7 = []
|
|
|
|
values_prev_14 = []
|
|
|
|
values_prev_28 = []
|
|
|
|
values_prev_56 = []
|
|
|
|
|
2019-10-23 14:28:15 +03:00
|
|
|
key = f"{type_}${task}$"
|
2019-10-22 19:42:38 +03:00
|
|
|
|
2019-10-09 17:02:58 +03:00
|
|
|
for item in items:
|
2019-10-23 14:28:15 +03:00
|
|
|
full_key = key + item
|
2019-10-22 19:42:38 +03:00
|
|
|
|
|
|
|
if full_key not in past_failures:
|
|
|
|
cur = past_failures[full_key] = ExpQueue(
|
2019-10-09 17:02:58 +03:00
|
|
|
push_num, HISTORICAL_TIMESPAN + 1, 0
|
|
|
|
)
|
2019-10-22 19:38:43 +03:00
|
|
|
else:
|
2019-10-22 19:42:38 +03:00
|
|
|
cur = past_failures[full_key]
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-10-22 19:38:43 +03:00
|
|
|
value = cur[push_num]
|
2019-10-09 17:02:58 +03:00
|
|
|
|
|
|
|
values_total.append(value)
|
2019-10-22 19:38:43 +03:00
|
|
|
values_prev_7.append(value - cur[push_num - 7])
|
|
|
|
values_prev_14.append(value - cur[push_num - 14])
|
|
|
|
values_prev_28.append(value - cur[push_num - 28])
|
|
|
|
values_prev_56.append(value - cur[push_num - 56])
|
2019-10-09 17:02:58 +03:00
|
|
|
|
|
|
|
if is_regression:
|
2019-10-22 19:38:43 +03:00
|
|
|
cur[push_num] = value + 1
|
2019-10-09 17:02:58 +03:00
|
|
|
|
|
|
|
return (
|
|
|
|
sum(values_total),
|
|
|
|
sum(values_prev_7),
|
|
|
|
sum(values_prev_14),
|
|
|
|
sum(values_prev_28),
|
|
|
|
sum(values_prev_56),
|
|
|
|
)
|
2019-09-18 14:10:22 +03:00
|
|
|
|
|
|
|
def generate_data():
|
2019-10-18 14:59:24 +03:00
|
|
|
nonlocal push_num
|
2019-09-18 14:10:22 +03:00
|
|
|
saved_nodes = set()
|
2019-11-10 21:57:05 +03:00
|
|
|
skipped_no_commits = 0
|
|
|
|
skipped_too_big_commits = 0
|
|
|
|
skipped_no_tasks = 0
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-10-18 14:59:24 +03:00
|
|
|
# We can start once we get to the last revision we added in the previous run.
|
|
|
|
can_start = True if last_node is None else False
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
commit_map = {}
|
|
|
|
for commit_data in tqdm(repository.get_commits()):
|
|
|
|
if not can_start:
|
|
|
|
if last_node == commit_data["node"]:
|
|
|
|
can_start = True
|
2019-10-22 19:42:38 +03:00
|
|
|
|
2019-10-18 14:59:24 +03:00
|
|
|
continue
|
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
commit_map[commit_data["node"]] = commit_data
|
|
|
|
|
|
|
|
with open("push_data.json", "r") as f:
|
|
|
|
push_data = json.load(f)[1:]
|
|
|
|
|
|
|
|
logger.info(f"push data nodes: {len(push_data)}")
|
|
|
|
|
2019-11-15 01:04:57 +03:00
|
|
|
# In the last 28 pushes, we definitely run all possible tasks.
|
|
|
|
all_tasks_set = set(
|
|
|
|
sum((push_tasks for _, push_tasks, _, _ in push_data[-28:]), [])
|
|
|
|
)
|
|
|
|
logger.info(f"{len(all_tasks_set)} tasks run in the last 28 pushes")
|
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
# We can start once we get to the last revision we added in the previous run.
|
|
|
|
can_start = True if last_node is None else False
|
|
|
|
|
|
|
|
for i in tqdm(range(len(push_data))):
|
|
|
|
(
|
|
|
|
revisions,
|
2019-11-15 01:04:57 +03:00
|
|
|
push_tasks,
|
2019-11-10 21:57:05 +03:00
|
|
|
possible_regressions,
|
|
|
|
likely_regressions,
|
|
|
|
) = push_data.pop(0)
|
|
|
|
|
2019-10-18 14:59:24 +03:00
|
|
|
if not can_start:
|
2019-11-10 21:57:05 +03:00
|
|
|
if last_node == revisions[0]:
|
|
|
|
can_start = True
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
push_num += 1
|
|
|
|
|
|
|
|
# XXX: Some commits are skipped in the repository mining, e.g. merges and backouts. Maybe we should not skip them.
|
|
|
|
commits = tuple(
|
|
|
|
commit_map.pop(revision)
|
|
|
|
for revision in revisions
|
|
|
|
if revision in commit_map
|
|
|
|
)
|
|
|
|
if len(commits) == 0:
|
|
|
|
skipped_no_commits += 1
|
2019-10-18 14:59:24 +03:00
|
|
|
continue
|
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
merged_commits = commit_features.merge_commits(commits)
|
|
|
|
|
|
|
|
# XXX: For now, skip commits which are too large.
|
|
|
|
# In the future we can either:
|
|
|
|
# - Improve shelve perf and go back to consider all files;
|
|
|
|
# - Consider only files which appear with a given frequency, like the "files" feature in commit_features;
|
|
|
|
# - Keep a limit of number of files.
|
|
|
|
if len(merged_commits["files"]) > 50:
|
|
|
|
skipped_too_big_commits += 1
|
2019-09-18 14:10:22 +03:00
|
|
|
continue
|
|
|
|
|
2019-11-15 01:04:57 +03:00
|
|
|
tasks = filter_tasks(push_tasks, all_tasks_set)
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
if len(tasks) == 0:
|
|
|
|
skipped_no_tasks += 1
|
|
|
|
continue
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
# Sync DB every 500 pushes, so we cleanup the shelve cache (we'd run OOM otherwise!).
|
|
|
|
if i % 500 == 0:
|
|
|
|
past_failures.sync()
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-11-10 21:57:05 +03:00
|
|
|
pushdate = dateutil.parser.parse(merged_commits["pushdate"])
|
2019-11-05 14:31:04 +03:00
|
|
|
|
2019-11-15 01:04:57 +03:00
|
|
|
for task in push_tasks:
|
2019-10-09 17:02:58 +03:00
|
|
|
is_regression = (
|
2019-11-10 21:57:05 +03:00
|
|
|
task in possible_regressions or task in likely_regressions
|
2019-10-09 17:02:58 +03:00
|
|
|
)
|
|
|
|
|
2019-10-29 20:00:36 +03:00
|
|
|
(
|
|
|
|
total_failures,
|
|
|
|
past_7_pushes_failures,
|
|
|
|
past_14_pushes_failures,
|
|
|
|
past_28_pushes_failures,
|
|
|
|
past_56_pushes_failures,
|
|
|
|
) = get_and_update_past_failures(
|
2019-10-09 17:02:58 +03:00
|
|
|
"all", task, ["all"], push_num, is_regression
|
|
|
|
)
|
|
|
|
|
2019-10-29 20:00:36 +03:00
|
|
|
(
|
|
|
|
total_types_failures,
|
|
|
|
past_7_pushes_types_failures,
|
|
|
|
past_14_pushes_types_failures,
|
|
|
|
past_28_pushes_types_failures,
|
|
|
|
past_56_pushes_types_failures,
|
|
|
|
) = get_and_update_past_failures(
|
2019-11-10 21:57:05 +03:00
|
|
|
"type", task, merged_commits["types"], push_num, is_regression
|
2019-09-18 14:10:22 +03:00
|
|
|
)
|
2019-10-09 17:02:58 +03:00
|
|
|
|
2019-10-29 20:00:36 +03:00
|
|
|
(
|
|
|
|
total_files_failures,
|
|
|
|
past_7_pushes_files_failures,
|
|
|
|
past_14_pushes_files_failures,
|
|
|
|
past_28_pushes_files_failures,
|
|
|
|
past_56_pushes_files_failures,
|
|
|
|
) = get_and_update_past_failures(
|
2019-11-10 21:57:05 +03:00
|
|
|
"file", task, merged_commits["files"], push_num, is_regression
|
2019-09-18 14:10:22 +03:00
|
|
|
)
|
2019-10-09 17:02:58 +03:00
|
|
|
|
2019-10-29 20:00:36 +03:00
|
|
|
(
|
|
|
|
total_directories_failures,
|
|
|
|
past_7_pushes_directories_failures,
|
|
|
|
past_14_pushes_directories_failures,
|
|
|
|
past_28_pushes_directories_failures,
|
|
|
|
past_56_pushes_directories_failures,
|
|
|
|
) = get_and_update_past_failures(
|
2019-10-09 17:02:58 +03:00
|
|
|
"directory",
|
|
|
|
task,
|
2019-11-10 21:57:05 +03:00
|
|
|
merged_commits["directories"],
|
2019-10-09 17:02:58 +03:00
|
|
|
push_num,
|
|
|
|
is_regression,
|
2019-09-18 14:10:22 +03:00
|
|
|
)
|
2019-10-09 17:02:58 +03:00
|
|
|
|
2019-10-29 20:00:36 +03:00
|
|
|
(
|
|
|
|
total_components_failures,
|
|
|
|
past_7_pushes_components_failures,
|
|
|
|
past_14_pushes_components_failures,
|
|
|
|
past_28_pushes_components_failures,
|
|
|
|
past_56_pushes_components_failures,
|
|
|
|
) = get_and_update_past_failures(
|
2019-10-09 17:02:58 +03:00
|
|
|
"component",
|
|
|
|
task,
|
2019-11-10 21:57:05 +03:00
|
|
|
merged_commits["components"],
|
2019-10-09 17:02:58 +03:00
|
|
|
push_num,
|
|
|
|
is_regression,
|
2019-09-18 14:10:22 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
if pushdate > HISTORY_DATE_START:
|
2019-11-10 21:57:05 +03:00
|
|
|
saved_nodes.add(i)
|
2019-09-18 14:10:22 +03:00
|
|
|
|
|
|
|
yield {
|
2019-11-10 21:57:05 +03:00
|
|
|
"revs": revisions,
|
2019-09-18 14:10:22 +03:00
|
|
|
"name": task,
|
|
|
|
"failures": total_failures,
|
|
|
|
"failures_past_7_pushes": past_7_pushes_failures,
|
|
|
|
"failures_past_14_pushes": past_14_pushes_failures,
|
|
|
|
"failures_past_28_pushes": past_28_pushes_failures,
|
|
|
|
"failures_past_56_pushes": past_56_pushes_failures,
|
2019-10-09 17:02:58 +03:00
|
|
|
"failures_in_types": total_types_failures,
|
|
|
|
"failures_past_7_pushes_in_types": past_7_pushes_types_failures,
|
|
|
|
"failures_past_14_pushes_in_types": past_14_pushes_types_failures,
|
|
|
|
"failures_past_28_pushes_in_types": past_28_pushes_types_failures,
|
|
|
|
"failures_past_56_pushes_in_types": past_56_pushes_types_failures,
|
|
|
|
"failures_in_files": total_files_failures,
|
|
|
|
"failures_past_7_pushes_in_files": past_7_pushes_files_failures,
|
|
|
|
"failures_past_14_pushes_in_files": past_14_pushes_files_failures,
|
|
|
|
"failures_past_28_pushes_in_files": past_28_pushes_files_failures,
|
|
|
|
"failures_past_56_pushes_in_files": past_56_pushes_files_failures,
|
|
|
|
"failures_in_directories": total_directories_failures,
|
|
|
|
"failures_past_7_pushes_in_directories": past_7_pushes_directories_failures,
|
|
|
|
"failures_past_14_pushes_in_directories": past_14_pushes_directories_failures,
|
|
|
|
"failures_past_28_pushes_in_directories": past_28_pushes_directories_failures,
|
|
|
|
"failures_past_56_pushes_in_directories": past_56_pushes_directories_failures,
|
|
|
|
"failures_in_components": total_components_failures,
|
|
|
|
"failures_past_7_pushes_in_components": past_7_pushes_components_failures,
|
|
|
|
"failures_past_14_pushes_in_components": past_14_pushes_components_failures,
|
|
|
|
"failures_past_28_pushes_in_components": past_28_pushes_components_failures,
|
|
|
|
"failures_past_56_pushes_in_components": past_56_pushes_components_failures,
|
2019-11-10 21:57:05 +03:00
|
|
|
"is_possible_regression": task in possible_regressions,
|
|
|
|
"is_likely_regression": task in likely_regressions,
|
2019-09-18 14:10:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
logger.info(f"saved push data nodes: {len(saved_nodes)}")
|
2019-11-10 21:57:05 +03:00
|
|
|
logger.info(f"skipped {skipped_no_commits} (no commits in our DB)")
|
|
|
|
logger.info(f"skipped {skipped_too_big_commits} (too big commits)")
|
|
|
|
logger.info(f"skipped {skipped_no_tasks} (no interesting tasks)")
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-10-18 14:59:24 +03:00
|
|
|
db.append(test_scheduling.TEST_SCHEDULING_DB, generate_data())
|
2019-09-18 14:10:22 +03:00
|
|
|
|
2019-09-19 12:21:21 +03:00
|
|
|
zstd_compress(test_scheduling.TEST_SCHEDULING_DB)
|
2019-09-11 21:17:02 +03:00
|
|
|
|
2019-10-22 19:42:38 +03:00
|
|
|
past_failures["push_num"] = push_num
|
|
|
|
past_failures.close()
|
2019-10-23 14:01:42 +03:00
|
|
|
zstd_compress("data/past_failures.shelve")
|
2019-10-18 14:59:24 +03:00
|
|
|
|
2019-09-11 21:17:02 +03:00
|
|
|
|
|
|
|
def main():
|
|
|
|
description = "Retrieve and extract the test scheduling history from ActiveData"
|
|
|
|
parser = argparse.ArgumentParser(description=description)
|
|
|
|
|
2019-10-18 15:33:53 +03:00
|
|
|
parser.add_argument(
|
|
|
|
"op", help="Which operation to perform.", choices=["retrieve", "generate"]
|
|
|
|
)
|
|
|
|
|
|
|
|
args = parser.parse_args()
|
2019-09-11 21:17:02 +03:00
|
|
|
|
|
|
|
retriever = Retriever()
|
2019-10-18 15:33:53 +03:00
|
|
|
if args.op == "retrieve":
|
|
|
|
retriever.retrieve_push_data()
|
|
|
|
elif args.op == "generate":
|
|
|
|
retriever.generate_test_scheduling_history()
|
2019-09-11 21:17:02 +03:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|