# Copyright Kevin Deldycke <kevin@deldycke.com> and contributors.## 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 2# 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, write to the Free Software# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA."""Operations on a mix of Groups and Platforms."""from__future__importannotationsfromitertoolsimportcombinationsfromtypingimportTYPE_CHECKING,FrozenSet,Iterablefromboltons.iterutilsimportunique,unique_iterfrom.groupimportGroupfrom.group_dataimportALL_GROUPS,ALL_PLATFORMSfrom.platformimportPlatformifTYPE_CHECKING:from.import_TNestedReferencesALL_PLATFORM_IDS:FrozenSet[str]=frozenset((p.idforpinALL_PLATFORMS.platforms))"""Set of all recognized platform IDs."""ALL_GROUP_IDS:FrozenSet[str]=frozenset((p.idforpinALL_GROUPS))"""Set of all recognized group IDs."""ALL_IDS:FrozenSet[str]=ALL_PLATFORM_IDS|ALL_GROUP_IDS"""Set of all recognized platform and group IDs."""
[docs]defplatforms_from_ids(*platform_ids:str)->tuple[Platform]:"""Returns a deduplicated tuple of platforms matching the provided IDs. IDs are case-insensitive, and can refer to any platforms or groups. Matching groups will be expanded to the platforms they contain. ..tip:: If you want to reduce the returned set and removes as much overlaps as possible, you can use the ``extra_platforms.reduce()`` function on the results. """ids=unique((s.lower()forsinplatform_ids))unrecognized_ids=set(ids)-ALL_IDSifunrecognized_ids:raiseValueError("Unrecognized group or platform IDs: "+", ".join(sorted(unrecognized_ids)))platforms=[]forplatform_idinids:ifplatform_idinALL_PLATFORM_IDS:platforms.append(ALL_PLATFORMS[platform_id])else:groups=groups_from_ids(platform_id)assertlen(groups)==1platforms.extend(groups[0].platforms)returntuple(unique_iter(platforms))
[docs]defgroups_from_ids(*group_ids:str)->tuple[Group]:"""Returns a deduplicated tuple of groups matching the provided IDs. IDs are case-insensitive. ..tip:: If you want to reduce the returned set and removes as much overlaps as possible, you can use the ``extra_platforms.reduce()`` function on the results. """ids=unique((s.lower()forsingroup_ids))unrecognized_ids=set(ids)-ALL_GROUP_IDSifunrecognized_ids:raiseValueError("Unrecognized group IDs: "+", ".join(sorted(unrecognized_ids)))groups=[]forgroup_idinids:forgroupinALL_GROUPS:ifgroup.id==group_id:groups.append(group)returntuple(unique_iter(groups))
[docs]defreduce(items:_TNestedReferences,target_pool:Iterable[Group|Platform]|None=None)->frozenset[Group|Platform]:"""Reduce a collection of platforms to a minimal set. Returns a deduplicated set of ``Group`` and ``Platform`` that covers the same exact platforms as the original input, but group as much platforms as possible, to reduce the number of items. Only the groups defined in the ``target_pool`` are considered for the reduction. If no reference pool is provided, use all known groups. .. hint:: Maybe this could be solved with some `Euler diagram <https://en.wikipedia.org/wiki/Euler_diagram>`_ algorithms, like those implemented in `eule <https://github.com/trouchet/eule>`_. This is being discussed upstream at `trouchet/eule#120 <https://github.com/trouchet/eule/issues/120>`_. .. todo:: Should we rename or alias this method to `collapse()`? Cannot decide if it is more descriptive or not... """# Collect all platforms.platforms=frozenset(Group._extract_platforms(items))# List all groups overlapping the set of input platforms.iftarget_poolisNone:target_pool=ALL_GROUPSoverlapping_groups=frozenset(gforgintarget_poolifisinstance(g,Group)andg.issubset(platforms))# Test all combination of groups to find the smallest set of groups + platforms.min_items=0results:list[frozenset[Group|Platform]]=[]# Serialize group sets for deterministic lookups. Sort them by platform count.groups=tuple(sorted(overlapping_groups,key=len,reverse=True))forsubset_sizeinrange(1,len(groups)+1):# If we already have a solution that involves less items than the current# subset of groups we're going to evaluates, there is no point in continuing.ifmin_itemsandsubset_size>min_items:breakforgroup_subsetincombinations(groups,subset_size):# If any group overlaps another, there is no point in exploring this subset.ifnotall(g[0].isdisjoint(g[1])forgincombinations(group_subset,2)):continue# Remove all platforms covered by the groups.ungrouped_platforms=set(platforms.copy())ungrouped_platforms.difference_update(*group_subset)# Merge the groups and the remaining platforms.reduction=frozenset(ungrouped_platforms.union(group_subset))reduction_size=len(reduction)# Reset the results if we have a new solution that is better than the# previous ones.ifnotresultsorreduction_size<min_items:results=[reduction]min_items=reduction_size# If the solution is as good as the previous one, add it to the results.elifreduction_size==min_items:results.append(reduction)iflen(results)>1:msg=f"Multiple solutions found: {results}"raiseRuntimeError(msg)# If no reduced solution was found, return the original platforms.ifnotresults:returnplatformsreturnresults.pop()