# This file is part of NIT ( http://www.nitlanguage.org ). # # Copyright 2012 Jean Privat # Copyright 2014 Alexandre Terrasa # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # Helpers for various statistics tools. module metrics_base import model_utils import csv import counter redef class ToolContext # --all var opt_all = new OptionBool("Compute all metrics", "--all") # --mmodules var opt_mmodules = new OptionBool("Compute metrics about mmodules", "--mmodules") # --mclassses var opt_mclasses = new OptionBool("Compute metrics about mclasses", "--mclasses") # --inheritance var opt_inheritance = new OptionBool("Compute metrics about inheritance usage", "--inheritance") # --genericity var opt_refinement = new OptionBool("Compute metrics about refinement usage", "--refinement") # --self var opt_self = new OptionBool("Compute metrics about the usage of explicit and implicit self", "--self") # --ast var opt_ast = new OptionBool("Compute metrics about the usage of nodes and identifiers in the AST", "--ast") # --nullables var opt_nullables = new OptionBool("Compute metrics on nullables send", "--nullables") # --static-types var opt_static_types = new OptionBool("Compute explicit static types metrics", "--static-types") # --tables var opt_tables = new OptionBool("Compute tables metrics", "--tables") # --rta var opt_rta = new OptionBool("Compute RTA metrics", "--rta") # --generate-csv var opt_generate_csv = new OptionBool("Generate CVS format metrics", "--generate-csv") # --generate_hyperdoc var opt_generate_hyperdoc = new OptionBool("Generate Hyperdoc", "--generate_hyperdoc") # --poset var opt_poset = new OptionBool("Complete metrics on posets", "--poset") var opt_dir = new OptionString("Directory where some statistics files are generated", "-d", "--dir") var output_dir: String = "." redef init do super self.option_context.add_option(opt_all) self.option_context.add_option(opt_mmodules) self.option_context.add_option(opt_mclasses) self.option_context.add_option(opt_inheritance) self.option_context.add_option(opt_refinement) self.option_context.add_option(opt_self) self.option_context.add_option(opt_ast) self.option_context.add_option(opt_nullables) self.option_context.add_option(opt_static_types) self.option_context.add_option(opt_tables) self.option_context.add_option(opt_rta) self.option_context.add_option(opt_generate_csv) self.option_context.add_option(opt_generate_hyperdoc) self.option_context.add_option(opt_poset) self.option_context.add_option(opt_dir) end redef fun process_options do super var val = self.opt_dir.value if val != null then val = val.simplify_path val.mkdir self.output_dir = val end end end redef class Model # List of modules in std lib # FIXME this is quite ugly, find a dynamic way... fun std_modules: Set[String] do if self.std_modules_cache == null then self.std_modules_cache = new HashSet[String] self.std_modules_cache.add("collection") self.std_modules_cache.add("abstract_collection") self.std_modules_cache.add("array") self.std_modules_cache.add("hash_collection") self.std_modules_cache.add("list") self.std_modules_cache.add("range") self.std_modules_cache.add("sorter") self.std_modules_cache.add("environ") self.std_modules_cache.add("exec") self.std_modules_cache.add("file") self.std_modules_cache.add("gc") self.std_modules_cache.add("hash") self.std_modules_cache.add("kernel") self.std_modules_cache.add("math") self.std_modules_cache.add("standard") self.std_modules_cache.add("stream") self.std_modules_cache.add("string") self.std_modules_cache.add("string_search") self.std_modules_cache.add("time") end return self.std_modules_cache.as(not null) end private var std_modules_cache: nullable Set[String] end redef class MClass fun is_user_defined: Bool do return self.intro_mmodule.is_user_defined end end redef class MModule fun is_user_defined: Bool do return not self.model.std_modules.has(self.name) end end # A Metric is used to collect data about things # # The concept is reified here for a better organization and documentation interface Metric fun name: String is abstract fun desc: String is abstract # clear all results for this metric fun clear is abstract end # A Metric that collects integer data # # Used to count things class IntMetric[E: Object] super Metric var values = new Counter[E] redef fun clear do values.clear # Return the couple with the highest value fun max: Couple[E, Int] do assert not values.is_empty var elem = values.max.as(not null) var value = values[elem] return new Couple[E, Int](elem, value) end # Return the couple with the lowest value fun min: Couple[E, Int] do assert not values.is_empty var elem = values.min.as(not null) var value = values[elem] return new Couple[E, Int](elem, value) end # Values average fun avg: Float do return values.avg end # A Metric that collects float datas # # Used sor summarization class FloatMetric[E: Object] super Metric var values: Map[E, Float] = new HashMap[E, Float] redef fun clear do values.clear # Return the couple with the highest value fun max: Couple[E, Float] do assert not values.is_empty var max: nullable Float = null var elem: nullable E = null for e, v in values do if max == null or v > max then max = v elem = e end end return new Couple[E, Float](elem.as(not null), max.as(not null)) end # Return the couple with the lowest value fun min: Couple[E, Float] do assert not values.is_empty var min: nullable Float = null var elem: nullable E = null for e, v in values do if min == null or v < min then min = v elem = e end end return new Couple[E, Float](elem.as(not null), min.as(not null)) end # Values average fun avg: Float do if values.is_empty then return 0.0 var sum = 0.0 for value in values.values do sum += value end return sum / values.length.to_f end end # A MetricSet is a metric holder # # It purpose is to be extended with a metric collect service class MetricSet type METRIC: Metric # Metrics to compute var metrics: Map[String, METRIC] = new HashMap[String, METRIC] # Add a metric to the set fun register(metrics: METRIC...) do for metric in metrics do self.metrics[metric.name] = metric # Clear all results for all metrics fun clear do for metric in metrics.values do metric.clear end