"""Convenience methods for plugin_api.""" load("@rules_java//java:defs.bzl", "java_import") # The current indirect ij_product mapping (eg. "intellij-latest") INDIRECT_IJ_PRODUCTS = { # Indirect ij_product mapping for internal Blaze Plugin "intellij-latest": "intellij-2022.3", "intellij-latest-mac": "intellij-2022.3-mac", "intellij-beta": "intellij-2022.3", "intellij-under-dev": "intellij-2022.3", "intellij-ue-latest": "intellij-ue-2022.3", "intellij-ue-latest-mac": "intellij-ue-2022.3-mac", "intellij-ue-beta": "intellij-ue-2022.3", "intellij-ue-under-dev": "intellij-ue-2022.3", "android-studio-latest": "android-studio-2023.1", "android-studio-latest-mac": "android-studio-2023.1-mac", "android-studio-beta": "android-studio-2023.1", "android-studio-beta-mac": "android-studio-2023.1-mac", "android-studio-canary": "android-studio-2023.2", "android-studio-canary-mac": "android-studio-2023.2-mac", "clion-latest": "clion-2022.3", "clion-latest-mac": "clion-2022.3-mac", "clion-beta": "clion-2022.3", "clion-under-dev": "clion-2022.3", # Indirect ij_product mapping for Bazel Plugin OSS # The old names for -oss-oldest-stable and -oss-latest-stable were # -oss-stable and -oss-beta respectively. "intellij-oss-oldest-stable": "intellij-2023.3", "intellij-oss-latest-stable": "intellij-2024.1", "intellij-oss-under-dev": "intellij-2024.1", "intellij-ue-oss-oldest-stable": "intellij-ue-2023.3", "intellij-ue-oss-latest-stable": "intellij-ue-2024.1", "intellij-ue-oss-under-dev": "intellij-ue-2024.1", "android-studio-oss-oldest-stable": "android-studio-2023.1", "android-studio-oss-latest-stable": "android-studio-2023.2", "android-studio-oss-under-dev": "android-studio-2023.2", "clion-oss-oldest-stable": "clion-2023.3", "clion-oss-latest-stable": "clion-2024.1", "clion-oss-under-dev": "clion-2024.1", # Indirect ij_product mapping for Cloud Code Plugin OSS "intellij-cc-oldest-stable": "intellij-2022.3", "intellij-cc-latest-stable": "intellij-2022.3", "intellij-cc-under-dev": "intellij-2022.3", "intellij-ue-cc-oldest-stable": "intellij-ue-2022.3", "intellij-ue-cc-latest-stable": "intellij-ue-2022.3", "intellij-ue-cc-under-dev": "intellij-ue-2022.3", } (CHANNEL_STABLE, CHANNEL_BETA, CHANNEL_CANARY, CHANNEL_FREEFORM) = ("stable", "beta", "canary", "freeform") INDIRECT_PRODUCT_CHANNELS = { # Channel mapping for internal Blaze Plugin "intellij-latest": CHANNEL_STABLE, "intellij-latest-mac": CHANNEL_STABLE, "intellij-beta": CHANNEL_BETA, "intellij-under-dev": CHANNEL_CANARY, "intellij-ue-latest": CHANNEL_STABLE, "intellij-ue-latest-mac": CHANNEL_STABLE, "intellij-ue-beta": CHANNEL_BETA, "intellij-ue-under-dev": CHANNEL_CANARY, "android-studio-latest": CHANNEL_STABLE, "android-studio-latest-mac": CHANNEL_STABLE, "android-studio-beta": CHANNEL_BETA, "android-studio-beta-mac": CHANNEL_BETA, "android-studio-canary": CHANNEL_CANARY, "android-studio-canary-mac": CHANNEL_CANARY, "clion-latest": CHANNEL_STABLE, "clion-latest-mac": CHANNEL_STABLE, "clion-beta": CHANNEL_BETA, "clion-under-dev": CHANNEL_CANARY, # Channel mapping for Bazel Plugin OSS "intellij-oss-oldest-stable": CHANNEL_STABLE, "intellij-oss-latest-stable": CHANNEL_STABLE, "intellij-oss-under-dev": CHANNEL_CANARY, "intellij-ue-oss-oldest-stable": CHANNEL_STABLE, "intellij-ue-oss-latest-stable": CHANNEL_STABLE, "intellij-ue-oss-under-dev": CHANNEL_CANARY, "android-studio-oss-oldest-stable": CHANNEL_STABLE, "android-studio-oss-latest-stable": CHANNEL_STABLE, "android-studio-oss-under-dev": CHANNEL_CANARY, "clion-oss-oldest-stable": CHANNEL_STABLE, "clion-oss-latest-stable": CHANNEL_STABLE, "clion-oss-under-dev": CHANNEL_CANARY, # Channel mapping for Cloud Code Plugin OSS "intellij-cc-oldest-stable": CHANNEL_STABLE, "intellij-cc-latest-stable": CHANNEL_STABLE, "intellij-cc-under-dev": CHANNEL_CANARY, "intellij-ue-cc-oldest-stable": CHANNEL_STABLE, "intellij-ue-cc-latest-stable": CHANNEL_STABLE, "intellij-ue-cc-under-dev": CHANNEL_CANARY, } def _check_channel_map(): if INDIRECT_PRODUCT_CHANNELS.keys() != INDIRECT_IJ_PRODUCTS.keys(): fail("Key mismatch between INDIRECT_PRODUCT_CHANNELS and INDIRECT_IJ_PRODUCTS: missing: %s extra: %s" % ( [k for k in INDIRECT_IJ_PRODUCTS.keys() if k not in INDIRECT_PRODUCT_CHANNELS.keys()], [k for k in INDIRECT_PRODUCT_CHANNELS.keys() if k not in INDIRECT_IJ_PRODUCTS.keys()], )) unexpected = [ (k, v) for k, v in INDIRECT_PRODUCT_CHANNELS.items() if v not in [CHANNEL_STABLE, CHANNEL_BETA, CHANNEL_CANARY] ] if unexpected: fail("Unexpected values in INDIRECT_PRODUCT_CHANNELS: %s" % unexpected) DIRECT_IJ_PRODUCTS = { "intellij-2021.3": struct( ide = "intellij", directory = "intellij_ce_2021_3", ), "intellij-2021.3-mac": struct( ide = "intellij", directory = "intellij_ce_2021_3", ), "intellij-2022.1": struct( ide = "intellij", directory = "intellij_ce_2022_1", ), "intellij-2022.1-mac": struct( ide = "intellij", directory = "intellij_ce_2022_1", ), "intellij-2022.2": struct( ide = "intellij", directory = "intellij_ce_2022_2", ), "intellij-2022.2-mac": struct( ide = "intellij", directory = "intellij_ce_2022_2", ), "intellij-2022.3": struct( ide = "intellij", directory = "intellij_ce_2022_3", ), "intellij-2022.3-mac": struct( ide = "intellij", directory = "intellij_ce_2022_3", ), "intellij-2023.1": struct( ide = "intellij", directory = "intellij_ce_2023_1", ), "intellij-2023.1-mac": struct( ide = "intellij", directory = "intellij_ce_2023_1", ), "intellij-2023.2": struct( ide = "intellij", directory = "intellij_ce_2023_2", ), "intellij-2023.2-mac": struct( ide = "intellij", directory = "intellij_ce_2023_2", ), "intellij-2023.3": struct( ide = "intellij", directory = "intellij_ce_2023_3", ), "intellij-2023.3-mac": struct( ide = "intellij", directory = "intellij_ce_2023_3", ), "intellij-2024.1": struct( ide = "intellij", directory = "intellij_ce_2024_1", ), "intellij-2024.1-mac": struct( ide = "intellij", directory = "intellij_ce_2024_1", ), "intellij-ue-2021.3": struct( ide = "intellij-ue", directory = "intellij_ue_2021_3", ), "intellij-ue-2021.3-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2021_3", ), "intellij-ue-2022.1": struct( ide = "intellij-ue", directory = "intellij_ue_2022_1", ), "intellij-ue-2022.1-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2022_1", ), "intellij-ue-2022.2": struct( ide = "intellij-ue", directory = "intellij_ue_2022_2", ), "intellij-ue-2022.2-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2022_2", ), "intellij-ue-2022.3": struct( ide = "intellij-ue", directory = "intellij_ue_2022_3", ), "intellij-ue-2022.3-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2022_3", ), "intellij-ue-2023.1": struct( ide = "intellij-ue", directory = "intellij_ue_2023_1", ), "intellij-ue-2023.1-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2023_1", ), "intellij-ue-2023.2": struct( ide = "intellij-ue", directory = "intellij_ue_2023_2", ), "intellij-ue-2023.2-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2023_2", ), "intellij-ue-2023.3": struct( ide = "intellij-ue", directory = "intellij_ue_2023_3", ), "intellij-ue-2023.3-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2023_3", ), "intellij-ue-2024.1": struct( ide = "intellij-ue", directory = "intellij_ue_2024_1", ), "intellij-ue-2024.1-mac": struct( ide = "intellij-ue", directory = "intellij_ue_2024_1", ), "android-studio-2022.3": struct( ide = "android-studio", archive = "android_studio_with_blaze_2022_3", oss_workspace = "android_studio_2022_3", ), "android-studio-2023.1": struct( ide = "android-studio", archive = "android_studio_with_blaze_2023_1", oss_workspace = "android_studio_2023_1", ), "android-studio-2023.2": struct( ide = "android-studio", archive = "android_studio_with_blaze_2023_2", oss_workspace = "android_studio_2023_2", ), "clion-2021.3": struct( ide = "clion", directory = "clion_2021_3", ), "clion-2021.3-mac": struct( ide = "clion", directory = "clion_2021_3", ), "clion-2022.1": struct( ide = "clion", directory = "clion_2022_1", ), "clion-2022.1-mac": struct( ide = "clion", directory = "clion_2022_1", ), "clion-2022.2": struct( ide = "clion", directory = "clion_2022_2", ), "clion-2022.2-mac": struct( ide = "clion", directory = "clion_2022_2", ), "clion-2022.3": struct( ide = "clion", directory = "clion_2022_3", ), "clion-2022.3-mac": struct( ide = "clion", directory = "clion_2022_3", ), "clion-2023.1": struct( ide = "clion", directory = "clion_2023_1", ), "clion-2023.1-mac": struct( ide = "clion", directory = "clion_2023_1", ), "clion-2023.2": struct( ide = "clion", directory = "clion_2023_2", ), "clion-2023.2-mac": struct( ide = "clion", directory = "clion_2023_2", ), "clion-2023.3": struct( ide = "clion", directory = "clion_2023_3", ), "clion-2023.3-mac": struct( ide = "clion", directory = "clion_2023_3", ), "clion-2024.1": struct( ide = "clion", directory = "clion_2024_1", ), "clion-2024.1-mac": struct( ide = "clion", directory = "clion_2024_1", ), } def select_for_plugin_api(params): """Selects for a plugin_api. Args: params: A dict with ij_product -> value. You may only include direct ij_products here, not indirects (eg. intellij-latest). Returns: A select statement on all plugin_apis. Unless you include a "default", a non-matched plugin_api will result in an error. Example: java_library( name = "foo", srcs = select_for_plugin_api({ "intellij-2016.3.1": [...my intellij 2016.3 sources ....], "intellij-2012.2.4": [...my intellij 2016.2 sources ...], }), ) """ for indirect_ij_product in INDIRECT_IJ_PRODUCTS: if indirect_ij_product in params: error_message = "".join([ "Do not select on indirect ij_product %s. " % indirect_ij_product, "Instead, select on an exact ij_product.", ]) fail(error_message) return _do_select_for_plugin_api(params) def _do_select_for_plugin_api(params): """A version of select_for_plugin_api which accepts indirect products.""" if not params: fail("Empty select_for_plugin_api") expanded_params = dict(**params) # Expand all indirect plugin_apis to point to their # corresponding direct plugin_api. # # {"intellij-2016.3.1": "foo"} -> # {"intellij-2016.3.1": "foo", "intellij-latest": "foo"} fallback_value = None for indirect_ij_product, resolved_plugin_api in INDIRECT_IJ_PRODUCTS.items(): if resolved_plugin_api in params: expanded_params[indirect_ij_product] = params[resolved_plugin_api] if not fallback_value: fallback_value = params[resolved_plugin_api] if indirect_ij_product in params: expanded_params[resolved_plugin_api] = params[indirect_ij_product] # Map the shorthand ij_products to full config_setting targets. # This makes it more convenient so the user doesn't have to # fully specify the path to the plugin_apis select_params = dict() for ij_product, value in expanded_params.items(): if ij_product == "default": select_params["//conditions:default"] = value else: select_params["//intellij_platform_sdk:" + ij_product] = value return select( select_params, no_match_error = "define an intellij product version, e.g. --define=ij_product=intellij-latest", ) def select_for_ide(intellij = None, intellij_ue = None, android_studio = None, clion = None, default = []): """Selects for the supported IDEs. Args: intellij: Files to use for IntelliJ. If None, will use default. intellij_ue: Files to use for IntelliJ UE. If None, will use value chosen for 'intellij'. android_studio: Files to use for Android Studio. If None will use default. clion: Files to use for CLion. If None will use default. default: Files to use for any IDEs not passed. Returns: A select statement on all plugin_apis to lists of files, sorted into IDEs. Example: java_library( name = "foo", srcs = select_for_ide( clion = [":cpp_only_sources"], default = [":java_only_sources"], ), ) """ intellij = intellij if intellij != None else default intellij_ue = intellij_ue if intellij_ue != None else intellij android_studio = android_studio if android_studio != None else default clion = clion if clion != None else default ide_to_value = { "intellij": intellij, "intellij-ue": intellij_ue, "android-studio": android_studio, "clion": clion, } # Map (direct ij_product) -> corresponding ide value params = dict() for ij_product, value in DIRECT_IJ_PRODUCTS.items(): params[ij_product] = ide_to_value[value.ide] params["default"] = default return select_for_plugin_api(params) def _plugin_api_directory(value): if hasattr(value, "oss_workspace"): directory = value.oss_workspace else: directory = value.directory return "@" + directory + "//" def select_from_plugin_api_directory(intellij, android_studio, clion, intellij_ue = None): """Internal convenience method to generate select statement from the IDE's plugin_api directories. Args: intellij: the items that IntelliJ product to use. android_studio: the items that android studio product to use. clion: the items that clion product to use. intellij_ue: the items that intellij ue product to use. Returns: a select statement map from DIRECT_IJ_PRODUCTS to items that they should use. """ ide_to_value = { "intellij": intellij, "intellij-ue": intellij_ue if intellij_ue else intellij, "android-studio": android_studio, "clion": clion, } # Map (direct ij_product) -> corresponding product directory params = dict() for ij_product, value in DIRECT_IJ_PRODUCTS.items(): params[ij_product] = [_plugin_api_directory(value) + item for item in ide_to_value[value.ide]] # No ij_product == intellij-latest params["default"] = params[INDIRECT_IJ_PRODUCTS["intellij-latest"]] return select_for_plugin_api(params) def select_from_plugin_api_version_directory(params): """Selects for a plugin_api direct version based on its directory. Args: params: A dict with ij_product -> value. You may only include direct ij_products here, not indirects (eg. intellij-latest). Returns: A select statement on all plugin_apis. Unless you include a "default", a non-matched plugin_api will result in an error. """ for indirect_ij_product in INDIRECT_IJ_PRODUCTS: if indirect_ij_product in params: error_message = "".join([ "Do not select on indirect ij_product %s. " % indirect_ij_product, "Instead, select on an exact ij_product.", ]) fail(error_message) # Map (direct ij_product) -> corresponding value relative to product directory for ij_product, value in params.items(): if ij_product != "default": params[ij_product] = [_plugin_api_directory(DIRECT_IJ_PRODUCTS[ij_product]) + item for item in value] return _do_select_for_plugin_api(params) def get_versions_to_build(product): """"Returns a set of unique product version aliases to test and build during regular release process. For each product, we care about four versions aliases to build and release to JetBrains repository; -latest, -beta, -oss-oldest-stable and oss-latest-stable. However, some of these aliases can point to the same IDE version and this can lead to conflicts if we attempt to blindly build and upload the four versions. This function is used to return only the aliases that point to different IDE versions of the given product. Args: product: name of the product; android-studio, clion, intellij-ue Returns: A space separated list of product version aliases to build, the values can be oss-oldest-stable, oss-latest-stable, internal-stable and internal-beta. """ aliases_to_build = [] plugin_api_versions = [] for alias in ["oss-oldest-stable", "latest", "oss-latest-stable", "beta"]: indirect_ij_product = product + "-" + alias if indirect_ij_product not in INDIRECT_IJ_PRODUCTS: fail( "Product-version alias %s not found." % indirect_ij_product, "Invalid product: %s only android-studio, clion and intellij-ue are accepted." % product, ) version = INDIRECT_IJ_PRODUCTS[indirect_ij_product] if version not in plugin_api_versions: plugin_api_versions.append(version) if alias == "latest": aliases_to_build.append("internal-stable") elif alias == "beta": aliases_to_build.append("internal-beta") else: aliases_to_build.append(alias) return " ".join(aliases_to_build) def get_unique_supported_oss_ide_versions(product): """"Returns the unique supported IDE versions for the given product in the OSS Bazel plugin Args: product: name of the product; android-studio, clion, intellij-ue Returns: A space separated list of the aliases of the unique IDE versions for the OSS Bazel plugin. """ supported_versions = [] unique_aliases = [] for alias in ["oss-oldest-stable", "oss-latest-stable"]: indirect_ij_product = product + "-" + alias if indirect_ij_product not in INDIRECT_IJ_PRODUCTS: fail( "Product-version alias %s not found." % indirect_ij_product, "Invalid product: %s, only android-studio, clion and intellij-ue are accepted." % product, ) ver = INDIRECT_IJ_PRODUCTS[indirect_ij_product] if ver not in supported_versions: supported_versions.append(ver) unique_aliases.append(alias) return " ".join(unique_aliases) def combine_visibilities(*args): """ Concatenates the given lists of visibilities and returns the combined list. If one of the given elements is //visibility:public then return //visibility:public If one of the lists is None, skip it. If the result list is empty, then return None. Args: *args: the list of visibilities lists to combine Returns: the concatenated visibility targets list """ res = [] for arg in args: if arg: for visibility in arg: if visibility == "//visibility:public": return ["//visibility:public"] res.append(visibility) if res == []: return None return res def no_mockito_extensions(name, jars, **kwargs): """Removes mockito extensions from jars. Args: name: Name of the resulting java_import target. jars: List of jars from which to remove mockito extensions. **kwargs: Arbitrary attributes for the java_import target. """ output_jars = [] for input_jar in jars: output_jar_name = name + "_" + input_jar.replace("/", "_") output_jar = name + "/" + input_jar native.genrule( name = output_jar_name, srcs = [input_jar], outs = [output_jar], cmd = """ cp "$<" "$@" chmod u+w "$@" zip -d "$@" mockito-extensions/* """, ) output_jars.append(output_jar_name) java_import( name = name, jars = output_jars, **kwargs ) # Since 2022.3, JVM 17 is required to start IntelliJ # https://blog.jetbrains.com/platform/2022/08/intellij-project-migrates-to-java-17/ def java_version_flags(): java11 = ["-source", "11", "-target", "11"] java17 = ["-source", "17", "-target", "17"] return select_for_plugin_api({ "intellij-2021.3": java11, "intellij-2021.3-mac": java11, "intellij-2022.1": java11, "intellij-2022.1-mac": java11, "intellij-2022.2": java11, "intellij-2022.2-mac": java11, "intellij-ue-2021.3": java11, "intellij-ue-2021.3-mac": java11, "intellij-ue-2022.1": java11, "intellij-ue-2022.1-mac": java11, "intellij-ue-2022.2": java11, "intellij-ue-2022.2-mac": java11, "android-studio-2022.2": java11, "clion-2021.3": java11, "clion-2021.3-mac": java11, "clion-2022.1": java11, "clion-2022.1-mac": java11, "clion-2022.2": java11, "clion-2022.2-mac": java11, "default": java17, }) def select_for_channel(channel_map): """Returns a select based on the IDE channel (stable, beta, canary). Args: channel_map: a dict with keys "stable", "beta" and "canary". The rest of targets will be considered "freeform" Returns: A select that will select values from channel_map based on the build config. """ _check_channel_map() if channel_map.keys() != [CHANNEL_STABLE, CHANNEL_BETA, CHANNEL_CANARY, CHANNEL_FREEFORM]: fail("channel_map must contain exactly %s, %s and %s" % (CHANNEL_STABLE, CHANNEL_BETA, CHANNEL_CANARY, CHANNEL_FREEFORM)) select_map = { ("//intellij_platform_sdk:%s" % indirect_product): channel_map[channel] for indirect_product, channel in INDIRECT_PRODUCT_CHANNELS.items() } # We reverse INDIRECT_IJ_PRODUCTS.items() here to that the inverse map contains # the first occurrence of any value that is duplicated, not the last: inverse_ij_products = {v: k for k, v in reversed(INDIRECT_IJ_PRODUCTS.items())} # Add directly specified IDE versions which some builds use: select_map.update( { ("//intellij_platform_sdk:%s" % direct_product): channel_map[INDIRECT_PRODUCT_CHANNELS[indirect_product]] for direct_product, indirect_product in inverse_ij_products.items() }, ) # Some IDE versions are not in a channel, but users would still like to build and test them: select_map.update( { ("//intellij_platform_sdk:%s" % direct_product): channel_map[CHANNEL_FREEFORM] for direct_product in DIRECT_IJ_PRODUCTS.keys() if direct_product not in inverse_ij_products }, ) select_map.update({"//conditions:default": channel_map[CHANNEL_STABLE]}) return select(select_map)