Source code for pypdevsbbl.tracers.FootprintTracer

#  Copyright 2020 Modelling, Simulation and Design Lab (MSDL) at
#  McGill University and the University of Antwerp (http://msdl.cs.mcgill.ca/)
#
#  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.

"""This file contains a tracer that can can gather the execution cost of a simulation.

The execution cost is either measured at the input port(s) or the output port(s) of a
block. For readability, choose one and keep it consistent whenever possible.
"""

from pypdevsbbl.tracers import BaseTracer, superTrace


[docs]class FootprintTracer(BaseTracer): """Gathers cost information of the simulation. The provided cost can be whatever is required in the current simulation. It can be the carbon footprint, the pricetag or even the amount of electricity. For this reason, multiple :class:`FootprintTracer` may be registered to the simulation. Give each :class:`FootprintTracer` a name to correctly associate the traced cost with the real-life consequences. Args: uid: A unique identifier for distributed simulation. Will be set automatically. server: The server that runs the distributed simulation. Will be set automatically. filename (str): The name of the file to write the results to. When ``None``, all results will be written to the standard output. name (str): The name of the :class:`FootprintTracer`. Used to differentiate the results of different instances. mapping (dict): A mapping from a port of a building block to a number or a function that returns a number. This number represents the cost for each item that is inputted or outputted on that port. The function takes the item as an argument. E.g. an object with a weight of 60 kilograms might be more expensive when compared to an object that weighs 10 kilograms. full (bool): When ``True``, output the full trace. Otherwise, only output the results. cpi (bool): Whether or not to print the cost per item at the end. These costs will be sorted with the largest cost first. Defaults to ``False``. """ def __init__(self, uid, server, filename, name, mapping, full=True, cpi=False): super(FootprintTracer, self).__init__(uid, server, filename) self.name = name self.mapping = mapping self.footprint = {} self.items = {} self.full = full self.cpi = cpi # Make sure all mappings are formatted in the same way for port in self.mapping: if not callable(self.mapping[port]): val = self.mapping[port] self.mapping[port] = lambda x, v=val: v self.inports = [x for x in self.mapping.keys() if x.is_input] self.inDEVS = [x.host_DEVS for x in self.inports] self.outports = [x for x in self.mapping.keys() if not x.is_input] self.outDEVS = [x.host_DEVS for x in self.outports]
[docs] def startTracer(self, recover): super(FootprintTracer, self).startTracer(recover) self.file.write("FOOTPRINT OF '%s':\n" % self.name) self.file.write("================" + ("=" * len(self.name)) + "\n")
[docs] def stopTracer(self): if self.full: self.file.write("\n") self.file.write("\tTOTAL FOOTPRINT COST: %s\n" % str(self.total())) if self.cpi: self.file.write("\tCOST PER ITEM:\n") for it, val in sorted(self.items.items(), key=lambda x: x[1], reverse=True): self.file.write("\t\t%s: %f\n" % (str(it), self.items[it])) self.file.write("\n") super(FootprintTracer, self).stopTracer()
[docs] def traceExternal(self, aDEVS): """Traces all items arriving on input ports.""" dvs = aDEVS.model if dvs in self.inDEVS: for port in aDEVS.my_input: if port in self.inports: for x in aDEVS.my_input[port]: lst = self.footprint.setdefault(port, 0) v = self.mapping[port](x) self.footprint[port] += v self.items.setdefault(x, 0) self.items[x] += v if self.full: superTrace(self.server, self.uid, aDEVS, ["\t[INP]", port.getPortFullName(), "(", "%.5f" % x, ") :", lst, "to", "%.5f" % self.footprint[port], "\n"])
[docs] def traceInternal(self, aDEVS): """Traces all items arriving on output ports.""" dvs = aDEVS.model if dvs in self.outDEVS: for port in aDEVS.my_output: if port in self.outports: for x in aDEVS.my_output[port]: lst = self.footprint.setdefault(port, 0) v = self.mapping[port](x) self.footprint[port] += v self.items.setdefault(x, 0) self.items[x] += v if self.full: superTrace(self.server, self.uid, aDEVS, ["\t[OUT]", port.getPortFullName(), "(", "%.5f" % x, ") :", lst, "to", "%.5f" % self.footprint[port], "\n"])
[docs] def total(self): """Obtain the total footprint of the simulation.""" return sum(self.footprint.values())