File: tasks.py

package info (click to toggle)
chromium 139.0.7258.127-1
  • links: PTS, VCS
  • area: main
  • in suites:
  • size: 6,122,068 kB
  • sloc: cpp: 35,100,771; ansic: 7,163,530; javascript: 4,103,002; python: 1,436,920; asm: 946,517; xml: 746,709; pascal: 187,653; perl: 88,691; sh: 88,436; objc: 79,953; sql: 51,488; cs: 44,583; fortran: 24,137; makefile: 22,147; tcl: 15,277; php: 13,980; yacc: 8,984; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (126 lines) | stat: -rw-r--r-- 3,808 bytes parent folder | download | duplicates (5)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
# Copyright 2024 The Chromium Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Datatypes for locally storing Swarming task data."""

import collections
import functools
from typing import Generator


class BotStats:
  """Stores the task stats for a single bot for a particular mixin."""

  def __init__(self):
    self._frozen = False
    self._total_tasks = 0
    self._failed_tasks = 0
    self._per_suite_total_tasks = collections.defaultdict(int)
    self._per_suite_failed_tasks = collections.defaultdict(int)

  def Freeze(self) -> None:
    if self._frozen:
      return
    self._frozen = True

  # Accessors

  @property
  def total_tasks(self) -> int:
    assert self._frozen
    return self._total_tasks

  @property
  def failed_tasks(self) -> int:
    assert self._frozen
    return self._failed_tasks

  @functools.cached_property
  def overall_failure_rate(self) -> float:
    assert self._frozen
    return float(self._failed_tasks) / self._total_tasks

  def GetTotalTasksForSuite(self, test_suite: str) -> int:
    assert self._frozen
    return self._per_suite_total_tasks[test_suite]

  def GetFailedTasksForSuite(self, test_suite: str) -> int:
    assert self._frozen
    return self._per_suite_failed_tasks[test_suite]

  # Mutators

  def AddStatsForSuite(self, test_suite: str, total_tasks: int,
                       failed_tasks: int) -> None:
    assert not self._frozen
    if total_tasks <= 0:
      raise ValueError('total_tasks must be positive')
    if failed_tasks < 0:
      raise ValueError('failed_tasks must be non-negative')
    if failed_tasks > total_tasks:
      raise ValueError('total_tasks must be >= failed_tasks')
    if test_suite in self._per_suite_total_tasks:
      raise ValueError(
          f'Stats for test suite {test_suite} were already provided - queries '
          f'should only return one row for each mixin/bot/test_suite '
          f'combination')
    self._total_tasks += total_tasks
    self._failed_tasks += failed_tasks
    self._per_suite_total_tasks[test_suite] = total_tasks
    self._per_suite_failed_tasks[test_suite] = failed_tasks


class MixinStats:
  """Stores the task stats for a single mixin."""

  def __init__(self):
    self._frozen = False
    self._total_tasks = 0
    self._failed_tasks = 0
    self._bots = collections.defaultdict(BotStats)
    self._cached_overall_failure_rates: list[float] | None = None

  def Freeze(self) -> None:
    if self._frozen:
      return
    self._frozen = True
    for bot in self._bots.values():
      bot.Freeze()

  # Accessors

  @property
  def total_tasks(self):
    assert self._frozen
    return self._total_tasks

  @property
  def failed_tasks(self):
    assert self._frozen
    return self._failed_tasks

  def IterBots(self) -> Generator[tuple[str, 'BotStats'], None, None]:
    assert self._frozen
    for bot_id, stats in self._bots.items():
      yield bot_id, stats

  def GetOverallFailureRates(self) -> list[float]:
    assert self._frozen
    if self._cached_overall_failure_rates is None:
      self._cached_overall_failure_rates = []
      for _, stats in self._bots.items():
        self._cached_overall_failure_rates.append(stats.overall_failure_rate)
    return self._cached_overall_failure_rates

  # Mutators

  def AddStatsForBotAndSuite(self, bot_id: str, test_suite: str,
                             total_tasks: int, failed_tasks: int) -> None:
    assert not self._frozen
    if total_tasks <= 0:
      raise ValueError('total_tasks must be positive')
    if failed_tasks < 0:
      raise ValueError('failed_tasks must be non-negative')
    self._total_tasks += total_tasks
    self._failed_tasks += failed_tasks
    self._bots[bot_id].AddStatsForSuite(test_suite, total_tasks, failed_tasks)