All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.sonar.plugins.python.api.PythonVersionUtils Maven / Gradle / Ivy

The newest version!
/*
 * SonarQube Python Plugin
 * Copyright (C) 2011-2024 SonarSource SA
 * mailto:info AT sonarsource DOT com
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the Sonar Source-Available License Version 1, as published by SonarSource SA.
 *
 * 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 Sonar Source-Available License for more details.
 *
 * You should have received a copy of the Sonar Source-Available License
 * along with this program; if not, see https://sonarsource.com/license/ssal/
 */
package org.sonar.plugins.python.api;

import java.util.EnumSet;
import java.util.Map;
import java.util.Set;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import static org.sonar.plugins.python.api.PythonVersionUtils.Version.V_310;
import static org.sonar.plugins.python.api.PythonVersionUtils.Version.V_311;
import static org.sonar.plugins.python.api.PythonVersionUtils.Version.V_312;
import static org.sonar.plugins.python.api.PythonVersionUtils.Version.V_313;
import static org.sonar.plugins.python.api.PythonVersionUtils.Version.V_38;
import static org.sonar.plugins.python.api.PythonVersionUtils.Version.V_39;

public class PythonVersionUtils {

  public enum Version {
    V_38(3, 8, "38"),
    V_39(3, 9, "39"),
    V_310(3, 10, "310"),
    V_311(3, 11, "311"),
    V_312(3, 12, "312"),
    V_313(3, 13, "313");

    private final int major;
    private final int minor;
    private final String serializedValue;

    Version(int major, int minor, String serializedValue) {
      this.major = major;
      this.minor = minor;
      this.serializedValue = serializedValue;
    }

    public int major() {
      return major;
    }

    public int minor() {
      return minor;
    }

    public String serializedValue() {
      return serializedValue;
    }

    public int compare(int major, int minor) {
      if (major() == major) {
        return Integer.compare(minor(), minor);
      }
      return Integer.compare(major(), major);
    }

    @Override
    public String toString() {
      return major + "." + minor;
    }
  }

  private static final Version MIN_SUPPORTED_VERSION = V_38;
  public static final Version MAX_SUPPORTED_VERSION = V_313;

  /**
   * Note that versions between 3 and 3.8 are currently mapped to 3.8 because
   * we don't take into account those version during typeshed symbols serialization
   */
  private static final Map STRING_VERSION_MAP = Map.ofEntries(
    Map.entry("3.0", MIN_SUPPORTED_VERSION),
    Map.entry("3.1", MIN_SUPPORTED_VERSION),
    Map.entry("3.2", MIN_SUPPORTED_VERSION),
    Map.entry("3.3", MIN_SUPPORTED_VERSION),
    Map.entry("3.4", MIN_SUPPORTED_VERSION),
    Map.entry("3.5", MIN_SUPPORTED_VERSION),
    Map.entry("3.6", MIN_SUPPORTED_VERSION),
    Map.entry("3.7", MIN_SUPPORTED_VERSION),
    Map.entry("3.8", V_38),
    Map.entry("3.9", V_39),
    Map.entry("3.10", V_310),
    Map.entry("3.11", V_311),
    Map.entry("3.12", V_312),
    Map.entry("3.13", V_313));
  private static final Logger LOG = LoggerFactory.getLogger(PythonVersionUtils.class);
  public static final String PYTHON_VERSION_KEY = "sonar.python.version";

  private PythonVersionUtils() {
  }

  public static Set fromString(String propertyValue) {
    return fromStringArray(propertyValue.split(","));
  }

  public static Set fromStringArray(String[] versions) {
    if (versions.length == 0) {
      return allVersions();
    }
    Set pythonVersions = EnumSet.noneOf(Version.class);
    for (String versionValue : versions) {
      versionValue = versionValue.trim();
      if ("3".equals(versionValue)) {
        // Only 3.x stubs are supported
        return allVersions();
      }
      Version version = STRING_VERSION_MAP.get(versionValue);
      if (version != null) {
        pythonVersions.add(version);
      } else {
        boolean isGuessSuccessful = guessPythonVersion(pythonVersions, versionValue);
        if (!isGuessSuccessful) {
          return allVersions();
        }
      }
    }
    return pythonVersions;
  }

  public static Set allVersions() {
    return EnumSet.allOf(Version.class);
  }

  private static boolean guessPythonVersion(Set pythonVersions, String versionValue) {
    String[] version = versionValue.split("\\.");
    try {
      int major = Integer.parseInt(version[0]);
      int minor = version.length > 1 ? Integer.parseInt(version[1]) : 0;
      Version guessedVersion = STRING_VERSION_MAP.get(major + "." + minor);
      if (guessedVersion != null) {
        pythonVersions.add(guessedVersion);
        logWarningGuessVersion(versionValue, guessedVersion);
        return true;
      }
      if (major < 3) {
        logWarningPython2(versionValue);
        return false;
      }
      if (MIN_SUPPORTED_VERSION.compare(major, minor) > 0) {
        pythonVersions.add(MIN_SUPPORTED_VERSION);
        logWarningGuessVersion(versionValue, MIN_SUPPORTED_VERSION);
      } else if (MAX_SUPPORTED_VERSION.compare(major, minor) < 0) {
        pythonVersions.add(MAX_SUPPORTED_VERSION);
        logWarningGuessVersion(versionValue, MAX_SUPPORTED_VERSION);
      } else {
        logErrorMessage(versionValue);
        return false;
      }
    } catch (NumberFormatException nfe) {
      logErrorMessage(versionValue);
      return false;
    }
    return true;
  }

  public static boolean areSourcePythonVersionsGreaterOrEqualThan(Set sourcePythonVersions, Version required) {
    // All versions must be greater than or equal to the required version.
    return !sourcePythonVersions.isEmpty() && sourcePythonVersions.stream()
      .allMatch(version -> version.compare(required.major(), required.minor()) >= 0);
  }


  private static void logErrorMessage(String propertyValue) {
    LOG.warn(
      "Error while parsing value of parameter '{}' ({}). Versions must be specified as MAJOR_VERSION.MINOR_VERSION (e.g. \"3.7, 3.8\")",
      PYTHON_VERSION_KEY,
      propertyValue);
  }

  private static void logWarningGuessVersion(String propertyValue, Version guessedVersion) {
    LOG.warn("No explicit support for version {}. Python version has been set to {}.", propertyValue, guessedVersion);
  }

  private static void logWarningPython2(String propertyValue) {
    LOG.warn("No explicit support for version {}. Support for Python versions prior to 3 is deprecated.", propertyValue);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy