/* * This file is part of onkostar-plugin-dnpm * * Copyright (c) 2025 the original author or authors. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ package dev.dnpm.oshelper.atc.services; import dev.dnpm.oshelper.atc.AgentCode; import org.jspecify.annotations.NullMarked; import java.util.Comparator; import java.util.List; import java.util.Objects; import java.util.function.Function; import java.util.stream.Collectors; import static java.util.stream.Collectors.groupingBy; /** * Common interface for agent code services * * @author Paul-Christian Volkmer * @since 2.0.0 */ public interface AgentCodeService { /** * Filters the provided list of AgentCodes to include only the latest available versions * for each unique code name and, if required, code. * If the input list is empty or null, an empty list will be returned. * * @param agentCodes A list of {@link AgentCode} objects to be filtered. Each object in the list * should contain information about the agent's code, name, system, and version. * @param classifier A function that returns a string value to be used as a key for grouping agent codes. * @return A list of {@link AgentCode} objects representing the latest available versions of unique agent codes. * If no valid inputs are provided, an empty list is returned. */ @NullMarked static List latestAgentCodeVersions(List agentCodes, Function classifier) { return latestAvailable(agentCodes, classifier); } /** * Function that extracts the name of an {@link AgentCode} object. * This function is used as a classifier for grouping agent codes by their name. */ Function byCodeOnly = AgentCode::getName; /** * Function that extracts the name and code of an {@link AgentCode} object. * This function is used as a classifier for grouping agent codes by their name and code. */ Function byNameAndCode = (agentCode -> String.format("%s_%s", agentCode.getCode(), agentCode.getName()) ); @NullMarked private static List latestAvailable( List agentCodes, Function classifier ) { return agentCodes.stream() .collect(groupingBy(classifier)).values().stream() .map(list -> list.stream() .sorted(Comparator.comparing(AgentCode::getCode)) .max((ac1, ac2) -> { if (null == ac1.getVersion()) { return -1; } else if (null == ac2.getVersion()) { return 1; } return ac1.getVersion().compareTo(ac2.getVersion()); }) .orElse(null) ) .filter(Objects::nonNull) .distinct() .collect(Collectors.toList()); } /** * Queries source for agents with name and code starting with query string. * If size is zero, all available results will be returned. * * @param query The query string * @param size Maximal amount of responses * @return A list with agent codes */ List findAgentCodes(String query, int size); }