// Copyright 2016 The Bazel Authors. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package com.google.devtools.build.lib.rules.objc; import static com.google.devtools.build.lib.packages.ImplicitOutputsFunction.fromTemplates; import static com.google.devtools.build.lib.rules.objc.ObjcProvider.FLAG; import static com.google.devtools.build.lib.rules.objc.ObjcProvider.Flag.HAS_WATCH2_EXTENSION; import static com.google.devtools.build.lib.rules.objc.ObjcProvider.MERGE_ZIP; import static com.google.devtools.build.lib.rules.objc.ObjcRuleClasses.WatchApplicationBundleRule.WATCH_APP_NAME_ATTR; import com.google.common.base.Joiner; import com.google.common.collect.HashMultiset; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Multiset; import com.google.devtools.build.lib.actions.Artifact; import com.google.devtools.build.lib.analysis.ConfiguredTarget; import com.google.devtools.build.lib.analysis.RuleConfiguredTarget.Mode; import com.google.devtools.build.lib.analysis.RuleConfiguredTargetBuilder; import com.google.devtools.build.lib.analysis.RuleContext; import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; import com.google.devtools.build.lib.packages.ImplicitOutputsFunction.SafeImplicitOutputsFunction; import com.google.devtools.build.lib.rules.RuleConfiguredTargetFactory; import com.google.devtools.build.lib.rules.apple.AppleConfiguration; import com.google.devtools.build.lib.rules.apple.Platform; import com.google.devtools.build.lib.rules.apple.Platform.PlatformType; import com.google.devtools.build.lib.rules.objc.WatchUtils.WatchOSVersion; import com.google.devtools.build.lib.rules.test.InstrumentedFilesCollector; import com.google.devtools.build.lib.rules.test.InstrumentedFilesProvider; import com.google.devtools.build.lib.syntax.Type; /** * Implementation for {@code apple_watch2_extension}. * * @deprecated The native bundling rules have been deprecated. This class will be removed in the * future. */ @Deprecated public class AppleWatch2Extension implements RuleConfiguredTargetFactory { /** Template for the containing application folder. */ public static final SafeImplicitOutputsFunction APP_NAME_IPA = fromTemplates("%{app_name}.ipa"); @Override public ConfiguredTarget create(RuleContext ruleContext) throws InterruptedException, RuleErrorException { ruleContext.ruleWarning( "This rule is deprecated. Please use the new Apple build rules " + "(https://github.com/bazelbuild/rules_apple) to build Apple targets."); validateAttributesAndConfiguration(ruleContext); ObjcProvider.Builder exposedObjcProviderBuilder = new ObjcProvider.Builder(); NestedSetBuilder<Artifact> filesToBuild = NestedSetBuilder.stableOrder(); // 1. Build watch extension bundle. createWatchExtensionBundle(ruleContext, filesToBuild, exposedObjcProviderBuilder); // 2. Build watch application bundle, which will contain the extension bundle. createWatchApplicationBundle( ruleContext, watchExtensionIpaArtifact(ruleContext), filesToBuild); RuleConfiguredTargetBuilder targetBuilder = ObjcRuleClasses.ruleConfiguredTarget(ruleContext, filesToBuild.build()) .addProvider( InstrumentedFilesProvider.class, InstrumentedFilesCollector.forward(ruleContext, "binary")); // 3. Add final watch application artifacts to the ObjcProvider, for bundling the watch // application bundle into the final iOS application IPA depending on this rule. exposedObjcProviderBuilder.add(MERGE_ZIP, ruleContext.getImplicitOutputArtifact(APP_NAME_IPA)); WatchUtils.registerActionsToAddWatchSupport( ruleContext, exposedObjcProviderBuilder, WatchOSVersion.OS2); exposedObjcProviderBuilder.add(FLAG, HAS_WATCH2_EXTENSION); targetBuilder.addProvider(ObjcProvider.class, exposedObjcProviderBuilder.build()); return targetBuilder.build(); } /** * Registers actions to create the watch extension bundle. * * @param ruleContext rule context in which to create the bundle */ private void createWatchExtensionBundle(RuleContext ruleContext, NestedSetBuilder<Artifact> filesToBuild, ObjcProvider.Builder exposedObjcProviderBuilder) throws InterruptedException { new Watch2ExtensionSupport( ruleContext, ObjcRuleClasses.intermediateArtifacts(ruleContext), watchExtensionBundleName(ruleContext)) .createBundle(watchExtensionIpaArtifact(ruleContext), filesToBuild, exposedObjcProviderBuilder); } /** * Registers actions to create the watch application bundle. This will contain the watch extension * bundle. The output artifacts are {@link #APP_NAME_IPA} (which is an implicit output of this * rule), and artifacts which are added to {@code exposedObjcProviderBuilder} for consumption by * depending targets. * * @param ruleContext rule context in which to create the bundle * @param extensionIpa the artifact representing the final extension bundle ipa * @param filesToBuild the list to contain the files to be built for this bundle */ private void createWatchApplicationBundle( RuleContext ruleContext, Artifact extensionIpa, NestedSetBuilder<Artifact> filesToBuild) throws InterruptedException { new WatchApplicationSupport( ruleContext, WatchOSVersion.OS2, // TODO(cparsons): Remove dependency attributes from WatchApplicationSupport, // as this is redundant with other attributes. ImmutableSet.<Attribute>of(), new IntermediateArtifacts(ruleContext, "", watchApplicationBundleName(ruleContext)), watchApplicationBundleName(ruleContext), watchApplicationIpaArtifact(ruleContext), watchApplicationBundleName(ruleContext)) .createBundle(ImmutableList.of(extensionIpa), filesToBuild); } /** Returns the {@Artifact} containing final watch application bundle. */ private Artifact watchApplicationIpaArtifact(RuleContext ruleContext) throws InterruptedException { return ruleContext.getImplicitOutputArtifact(APP_NAME_IPA); } /** Returns the {@Artifact} containing final watch extension bundle. */ private Artifact watchExtensionIpaArtifact(RuleContext ruleContext) throws InterruptedException { return ruleContext.getImplicitOutputArtifact(ReleaseBundlingSupport.IPA); } private String watchApplicationBundleName(RuleContext ruleContext) { return ruleContext.attributes().get(WATCH_APP_NAME_ATTR, Type.STRING); } private String watchExtensionBundleName(RuleContext ruleContext) { return ruleContext.getLabel().getName(); } private void validateAttributesAndConfiguration(RuleContext ruleContext) throws RuleErrorException { boolean hasError = false; Multiset<Artifact> appResources = HashMultiset.create(); appResources.addAll(ruleContext.getPrerequisiteArtifacts("app_resources", Mode.TARGET).list()); appResources.addAll(ruleContext.getPrerequisiteArtifacts("app_strings", Mode.TARGET).list()); for (Multiset.Entry<Artifact> entry : appResources.entrySet()) { if (entry.getCount() > 1) { ruleContext.ruleError( "The same file was included multiple times in this rule: " + entry.getElement().getRootRelativePathString()); hasError = true; } } Multiset<Artifact> extResources = HashMultiset.create(); extResources.addAll(ruleContext.getPrerequisiteArtifacts("ext_resources", Mode.TARGET).list()); extResources.addAll(ruleContext.getPrerequisiteArtifacts("ext_strings", Mode.TARGET).list()); for (Multiset.Entry<Artifact> entry : extResources.entrySet()) { if (entry.getCount() > 1) { ruleContext.ruleError( "The same file was included multiple times in this rule: " + entry.getElement().getRootRelativePathString()); hasError = true; } } AppleConfiguration appleConfiguration = ruleContext.getFragment(AppleConfiguration.class); Platform watchPlatform = appleConfiguration.getMultiArchPlatform(PlatformType.WATCHOS); Platform iosPlatform = appleConfiguration.getMultiArchPlatform(PlatformType.IOS); if (watchPlatform.isDevice() != iosPlatform.isDevice()) { hasError = true; if (watchPlatform.isDevice()) { ruleContext.ruleError( String.format( "Building a watch extension for watch device architectures [%s] " + "requires a device ios architecture. Found [%s] instead.", Joiner.on(",").join(appleConfiguration.getMultiArchitectures(PlatformType.WATCHOS)), Joiner.on(",").join(appleConfiguration.getMultiArchitectures(PlatformType.IOS)))); } else { ruleContext.ruleError( String.format( "Building a watch extension for ios device architectures [%s] " + "requires a device watch architecture. Found [%s] instead.", Joiner.on(",").join(appleConfiguration.getMultiArchitectures(PlatformType.IOS)), Joiner.on(",") .join(appleConfiguration.getMultiArchitectures(PlatformType.WATCHOS)))); } ruleContext.ruleError( "For building watch extension, there may only be a watch device " + "architecture if and only if there is an ios device architecture"); } if (hasError) { throw new RuleErrorException(); } } }