Source code for thoth.storages.graph.models_performance

#!/usr/bin/env python3
# thoth-storages
# Copyright(C) 2019, 2020 Fridolin Pokorny
#
# This program is free software: you can redistribute it and / or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.

"""Performance indicators (models) used in Thoth with their schemas."""

from functools import partial
import logging

from typing import Dict, Any

from sqlalchemy import Column
from sqlalchemy import Integer
from sqlalchemy import ForeignKey
from sqlalchemy import String
from sqlalchemy import Float
from sqlalchemy.orm import relationship
from sqlalchemy.orm import Session

from .models_base import BaseExtension
from .models_base import Base

_LOGGER = logging.getLogger(__name__)


[docs]class PerformanceIndicatorBase: """A base class for implementing performance indicators.""" # Component used for the performance indicator. component = Column(String(256), nullable=False) # Origin from where the performance indicator was obtained. In case of Git repo, # it holds Git repo URL, in case of URL it holds URL to the script. origin = Column(String(256), nullable=False) # Reference of the script, in case of Git repo it holds commit SHA, in case of URL it carries # SHA256 of the script which was used to test the performance with (performance indicator script). version = Column(String(256), nullable=False) # This one is used later on in queries in adviser, all the relevant performance indicators should # respect this property and place results in there. overall_score = Column(Float, nullable=True) # The actual exit code of the performance indicator. exit_code = Column(Integer, nullable=False) # Process statistics: # https://docs.python.org/3/library/resource.html#resource.getrusage ru_utime = Column(Float, nullable=False) ru_stime = Column(Float, nullable=False) ru_maxrss = Column(Integer, nullable=False) ru_ixrss = Column(Integer, nullable=False) ru_idrss = Column(Integer, nullable=False) ru_isrss = Column(Integer, nullable=False) ru_minflt = Column(Integer, nullable=False) ru_majflt = Column(Integer, nullable=False) ru_nswap = Column(Integer, nullable=False) ru_inblock = Column(Integer, nullable=False) ru_oublock = Column(Integer, nullable=False) ru_msgsnd = Column(Integer, nullable=False) ru_msgrcv = Column(Integer, nullable=False) ru_nsignals = Column(Integer, nullable=False) ru_nvcsw = Column(Integer, nullable=False) ru_nivcsw = Column(Integer, nullable=False)
[docs] @classmethod def create_from_report( cls, session: Session, inspection_specification: Dict[str, Any], inspection_result: Dict[str, Any], inspection_run_id: int, ) -> "PerformanceIndicatorBase": """Create performance indicator record together with related observed performance edge based on inspection.""" # Place core parts of the base class into the model. overall_score = inspection_result.get("overall_score") if overall_score is None: _LOGGER.warning("No overall score detected in performance indicator %r", overall_score) partial_model = partial( cls.get_or_create, session, inspection_run_id=inspection_run_id, component=inspection_result.get("component"), origin=inspection_specification["script"], version=inspection_result.get("version") or inspection_result["script_sha256"], overall_score=overall_score, exit_code=inspection_result.get("exit_code"), ru_utime=inspection_result.get("usage", {}).get("ru_utime"), ru_stime=inspection_result.get("usage", {}).get("ru_stime"), ru_maxrss=inspection_result.get("usage", {}).get("ru_maxrss"), ru_ixrss=inspection_result.get("usage", {}).get("ru_ixrss"), ru_idrss=inspection_result.get("usage", {}).get("ru_idrss"), ru_isrss=inspection_result.get("usage", {}).get("ru_isrss"), ru_minflt=inspection_result.get("usage", {}).get("ru_minflt"), ru_majflt=inspection_result.get("usage", {}).get("ru_majflt"), ru_nswap=inspection_result.get("usage", {}).get("ru_nswap"), ru_inblock=inspection_result.get("usage", {}).get("ru_inblock"), ru_oublock=inspection_result.get("usage", {}).get("ru_oublock"), ru_msgsnd=inspection_result.get("usage", {}).get("ru_msgsnd"), ru_msgrcv=inspection_result.get("usage", {}).get("ru_msgrcv"), ru_nsignals=inspection_result.get("usage", {}).get("ru_nsignals"), ru_nvcsw=inspection_result.get("usage", {}).get("ru_nvcsw"), ru_nivcsw=inspection_result.get("usage", {}).get("ru_nivcsw"), ) return cls.from_report(inspection_result, partial_model)
[docs] @classmethod def from_report(cls, inspection_result: Dict[str, Any], partial_model: type(partial)) -> "PerformanceIndicatorBase": """Create model from the inspection report respecting parameters and result reported by the indicator.""" kwargs = {} for parameter, parameter_value in inspection_result["@parameters"].items(): kwargs[parameter] = parameter_value for result_name, result_value in inspection_result["@result"].items(): if result_name in kwargs: raise ValueError("Collision in result name and parameter name") kwargs[result_name] = result_value return partial_model(**kwargs)
[docs]class PiMatmul(Base, BaseExtension, PerformanceIndicatorBase): """A class for representing a matrix multiplication micro-performance test.""" __tablename__ = "pi_matmul" id = Column(Integer, primary_key=True, autoincrement=True) inspection_run_id = Column(Integer, ForeignKey("inspection_run.id"), nullable=False) inspection_run = relationship("InspectionRun", back_populates="matmul_perf_indicators") # Device used during performance indicator run - CPU/GPU/TPU/... device = Column(String(256), nullable=False) # Size of the matrix tested. matrix_size = Column(Integer, nullable=False) # Type of item in the matrix. dtype = Column(String(256), nullable=False) # Number of repetitions of matrix multiplication performed. reps = Column(Integer, nullable=False) # Elapsed seconds. elapsed = Column(Float, nullable=False) # Final rate gflops/s. rate = Column(Float, nullable=False)
[docs]class PiConv1D(Base, BaseExtension, PerformanceIndicatorBase): """A class for representing a conv1D micro-performance test.""" __tablename__ = "pi_conv1d" id = Column(Integer, primary_key=True, autoincrement=True) inspection_run_id = Column(Integer, ForeignKey("inspection_run.id"), nullable=False) inspection_run = relationship("InspectionRun", back_populates="conv1d_perf_indicators") # Device used during performance indicator run - CPU/GPU/TPU/... device = Column(String(256), nullable=False) # Type of item in the tensor. dtype = Column(String(256), nullable=False) # Number of repetitions of conv2d performed. reps = Column(Integer, nullable=False) # Data format NHWC Channel_last or NCHW Channel_first data_format = Column(String(256), nullable=False) # INPUT TENSOR batch = Column(Integer, nullable=False) input_width = Column(Integer, nullable=False) input_channels = Column(Integer, nullable=False) # FILTER filter_width = Column(Integer, nullable=False) output_channels = Column(Integer, nullable=False) # Stride, the speed by which the filter moves across the image strides = Column(Integer, nullable=False) # Padding padding = Column(Integer, nullable=False) # Elapsed seconds. elapsed = Column(Float, nullable=False) # Final rate gflops/s. rate = Column(Float, nullable=False)
[docs]class PiConv2D(Base, BaseExtension, PerformanceIndicatorBase): """A class for representing a conv2D micro-performance test.""" __tablename__ = "pi_conv2d" id = Column(Integer, primary_key=True, autoincrement=True) inspection_run_id = Column(Integer, ForeignKey("inspection_run.id"), nullable=False) inspection_run = relationship("InspectionRun", back_populates="conv2d_perf_indicators") # Device used during performance indicator run - CPU/GPU/TPU/... device = Column(String(256), nullable=False) # Type of item in the tensor. dtype = Column(String(256), nullable=False) # Number of repetitions of conv2d performed. reps = Column(Integer, nullable=False) # Data format NHWC Channel_last or NCHW Channel_first data_format = Column(String(256), nullable=False) # INPUT TENSOR batch = Column(Integer, nullable=False) input_height = Column(Integer, nullable=False) input_width = Column(Integer, nullable=False) input_channels = Column(Integer, nullable=False) # FILTER filter_height = Column(Integer, nullable=False) filter_width = Column(Integer, nullable=False) output_channels = Column(Integer, nullable=False) # Stride, the speed by which the filter moves across the image strides = Column(Integer, nullable=False) # Padding padding = Column(Integer, nullable=False) # Elapsed seconds. elapsed = Column(Float, nullable=False) # Final rate gflops/s. rate = Column(Float, nullable=False)
[docs]class PiPyBench(Base, BaseExtension, PerformanceIndicatorBase): """A class for representing Pybench results for Python Interpreter.""" __tablename__ = "pi_pybench" id = Column(Integer, primary_key=True, autoincrement=True) inspection_run_id = Column(Integer, ForeignKey("inspection_run.id"), nullable=False) inspection_run = relationship("InspectionRun", back_populates="pybench_perf_indicators") # Number of rounds used for each function. rounds = Column(Integer, nullable=False) # Average results for each function built_in_function_calls_average = Column(Float, nullable=False) built_in_method_lookup_average = Column(Float, nullable=False) compare_floats_average = Column(Float, nullable=False) compare_floats_integers_average = Column(Float, nullable=False) compare_integers_average = Column(Float, nullable=False) compare_interned_strings_average = Column(Float, nullable=False) compare_longs_average = Column(Float, nullable=False) compare_strings_average = Column(Float, nullable=False) compare_unicode_average = Column(Float, nullable=False) concat_strings_average = Column(Float, nullable=False) concat_unicode_average = Column(Float, nullable=False) create_instances_average = Column(Float, nullable=False) create_new_instances_average = Column(Float, nullable=False) create_strings_with_concat_average = Column(Float, nullable=False) create_unicode_with_concat_average = Column(Float, nullable=False) dict_creation_average = Column(Float, nullable=False) dict_with_float_keys_average = Column(Float, nullable=False) dict_with_integer_keys_average = Column(Float, nullable=False) dict_with_string_keys_average = Column(Float, nullable=False) for_loops_average = Column(Float, nullable=False) if_then_else_average = Column(Float, nullable=False) list_slicing_average = Column(Float, nullable=False) nested_for_loops_average = Column(Float, nullable=False) normal_class_attribute_average = Column(Float, nullable=False) normal_instance_attribute_average = Column(Float, nullable=False) python_function_calls_average = Column(Float, nullable=False) python_method_calls_average = Column(Float, nullable=False) recursion_average = Column(Float, nullable=False) second_import_average = Column(Float, nullable=False) second_package_import_average = Column(Float, nullable=False) second_submodule_import_average = Column(Float, nullable=False) simple_complex_arithmetic_average = Column(Float, nullable=False) simple_dict_manipulation_average = Column(Float, nullable=False) simple_float_arithmetic_average = Column(Float, nullable=False) simple_int_float_arithmetic_average = Column(Float, nullable=False) simple_integer_arithmetic_average = Column(Float, nullable=False) simple_list_manipulation_average = Column(Float, nullable=False) simple_long_arithmetic_average = Column(Float, nullable=False) small_lists_average = Column(Float, nullable=False) small_tuples_average = Column(Float, nullable=False) special_class_attribute_average = Column(Float, nullable=False) special_instance_attribute_average = Column(Float, nullable=False) string_mappings_average = Column(Float, nullable=False) string_predicates_average = Column(Float, nullable=False) string_slicing_average = Column(Float, nullable=False) try_except_average = Column(Float, nullable=False) try_raise_except_average = Column(Float, nullable=False) tuple_slicing_average = Column(Float, nullable=False) unicode_mappings_average = Column(Float, nullable=False) unicode_predicates_average = Column(Float, nullable=False) unicode_properties_average = Column(Float, nullable=False) unicode_slicing_average = Column(Float, nullable=False) # Total average results totals_average = Column(Float, nullable=False)
PERFORMANCE_MODELS_ML_FRAMEWORKS = frozenset((PiMatmul, PiConv1D, PiConv2D)) ALL_PERFORMANCE_MODELS = frozenset((PiMatmul, PiConv1D, PiConv2D, PiPyBench)) PERFORMANCE_MODEL_BY_NAME = {model_class.__name__: model_class for model_class in ALL_PERFORMANCE_MODELS}