blob: bf0b2d951c7ab75e05558461b8dd8fef5ca8e39f [file] [log] [blame]
// 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}. */
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 {
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();
}
}
}