Source code for pm4py.objects.petri_net.utils.consumption_matrix

'''
    PM4Py – A Process Mining Library for Python
Copyright (C) 2024 Process Intelligence Solutions UG (haftungsbeschränkt)

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or 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 Affero General Public License for more details.

You should have received a copy of the GNU Affero General Public License
along with this program.  If not, see this software project's root or
visit <https://www.gnu.org/licenses/>.

Website: https://processintelligence.solutions
Contact: info@processintelligence.solutions
'''
"""
Implements the consumption matrix as explained in the following paper:

van Dongen, Boudewijn F. "Efficiently computing alignments." International Conference on Business Process Management.
Springer, Cham, 2018.
https://link.springer.com/chapter/10.1007/978-3-319-98648-7_12
"""

from typing import Dict

import numpy as np

from pm4py.objects.petri_net.obj import PetriNet


[docs] class ConsumptionMatrix(object): def __init__(self, net: PetriNet): """ Constructor Parameters -------------- net Petri net """ self.__place_indices = {} self.__transition_indices = {} self.__C = None places = sorted([x for x in net.places], key=lambda x: (x.name, id(x))) transitions = sorted( [x for x in net.transitions], key=lambda x: (x.name, id(x)) ) for p in places: self.__place_indices[p] = len(self.__place_indices) for t in transitions: self.__transition_indices[t] = len(self.__transition_indices) self.__compute_C_matrix(net) def __compute_C_matrix(self, net: PetriNet): """ Builds the C matrix Parameters --------------- net Petri net """ inv_indices = {y: x for x, y in self.__transition_indices.items()} inv_indices = [inv_indices[i] for i in range(len(inv_indices))] self.__C = [ [0 for i in range(len(self.__transition_indices))] for j in range(len(self.__place_indices)) ] for p in net.places: outgoing_trans = [a.target for a in p.out_arcs] self.__C[self.__place_indices[p]] = [ -1 if inv_indices[i] in outgoing_trans else 0 for i in range(len(inv_indices)) ] def __get_transition_indices(self) -> Dict[PetriNet.Transition, int]: """ Gets the transitions in the order in which they have been inserted in the consumption matrix Returns ------------- trans_indices Dictionary associating to each transition an incremental number """ return self.__transition_indices def __get_place_indices(self) -> Dict[PetriNet.Place, int]: """ Gets the places in the order in which they have been inserted in the consumption matrix Returns ------------- place_indices Dictionary associating to each place an incremental number """ return self.__place_indices def __get_c_matrix(self) -> np.ndarray: """ Gets the Numpy representation of the consumption matrix Returns ------------- C C matrix """ return self.__C places = property(__get_place_indices) transitions = property(__get_transition_indices) c_matrix = property(__get_c_matrix)
[docs] def construct(net: PetriNet) -> ConsumptionMatrix: """ Construct a consumption matrix given a Petri net Parameters ---------------- net Petri net Returns --------------- cons_mat Consumption matrix object """ return ConsumptionMatrix(net)