/* * Copyright 2015-present Facebook, Inc. * * 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.facebook.buck.apple; import com.dd.plist.NSArray; import com.dd.plist.NSData; import com.dd.plist.NSDate; import com.dd.plist.NSDictionary; import com.dd.plist.NSObject; import com.dd.plist.NSString; import com.dd.plist.PropertyListParser; import com.facebook.buck.model.Pair; import com.facebook.buck.rules.RuleKeyAppendable; import com.facebook.buck.rules.RuleKeyObjectSink; import com.facebook.buck.util.ProcessExecutor; import com.facebook.buck.util.ProcessExecutorParams; import com.facebook.buck.util.immutables.BuckStyleImmutable; import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableSet; import com.google.common.hash.HashCode; import com.google.common.hash.HashFunction; import com.google.common.hash.Hashing; import java.io.IOException; import java.nio.file.Path; import java.util.Date; import java.util.EnumSet; import java.util.Optional; import java.util.Set; import java.util.regex.Matcher; import java.util.regex.Pattern; import org.immutables.value.Value; /** Metadata contained in a provisioning profile (.mobileprovision). */ @Value.Immutable @BuckStyleImmutable abstract class AbstractProvisioningProfileMetadata implements RuleKeyAppendable { private static final Pattern BUNDLE_ID_PATTERN = Pattern.compile("^([A-Z0-9]{10,10})\\.(.+)$"); /** * Returns a (prefix, identifier) pair for which the profile is valid. * * <p>e.g. (ABCDE12345, com.example.TestApp) or (ABCDE12345, *) */ public abstract Pair<String, String> getAppID(); public abstract Date getExpirationDate(); public abstract String getUUID(); public abstract Path getProfilePath(); /** The set of platforms the profile is valid for. */ @Value.Default public ImmutableList<String> getPlatforms() { return ImmutableList.of(ApplePlatform.IPHONEOS.getProvisioningProfileName().get()); } /** Key/value pairs of the "Entitlements" dictionary in the embedded plist. */ public abstract ImmutableMap<String, NSObject> getEntitlements(); /** SHA1 hashes of the certificates in the "DeveloperCertificates" section. */ public abstract ImmutableSet<HashCode> getDeveloperCertificateFingerprints(); /** * Takes a application identifier and splits it into prefix and bundle ID. * * <p>Prefix is always a ten-character alphanumeric sequence. Bundle ID may be a fully-qualified * name or a wildcard ending in *. */ public static Pair<String, String> splitAppID(String appID) { Matcher matcher = BUNDLE_ID_PATTERN.matcher(appID); if (matcher.find()) { String prefix = matcher.group(1); String bundleID = matcher.group(2); return new Pair<>(prefix, bundleID); } else { throw new IllegalArgumentException("Malformed app ID: " + appID); } } /** * Takes an ImmutableMap representing an entitlements file, returns the application prefix if it * can be inferred from keys in the entitlement. Otherwise, it returns empty. */ public static Optional<String> prefixFromEntitlements( ImmutableMap<String, NSObject> entitlements) { try { NSArray keychainAccessGroups = ((NSArray) entitlements.get("keychain-access-groups")); Preconditions.checkNotNull(keychainAccessGroups); String appID = keychainAccessGroups.objectAtIndex(0).toString(); return Optional.of(splitAppID(appID).getFirst()); } catch (RuntimeException e) { return Optional.empty(); } } public static ProvisioningProfileMetadata fromProvisioningProfilePath( ProcessExecutor executor, ImmutableList<String> readCommand, Path profilePath) throws IOException, InterruptedException { Set<ProcessExecutor.Option> options = EnumSet.of(ProcessExecutor.Option.EXPECTING_STD_OUT); // Extract the XML from its signed message wrapper. ProcessExecutorParams processExecutorParams = ProcessExecutorParams.builder() .addAllCommand(readCommand) .addCommand(profilePath.toString()) .build(); ProcessExecutor.Result result; result = executor.launchAndExecute( processExecutorParams, options, /* stdin */ Optional.empty(), /* timeOutMs */ Optional.empty(), /* timeOutHandler */ Optional.empty()); if (result.getExitCode() != 0) { throw new IOException( result.getMessageForResult("Invalid provisioning profile: " + profilePath)); } try { NSDictionary plist = (NSDictionary) PropertyListParser.parse(result.getStdout().get().getBytes()); Date expirationDate = ((NSDate) plist.get("ExpirationDate")).getDate(); String uuid = ((NSString) plist.get("UUID")).getContent(); ImmutableSet.Builder<HashCode> certificateFingerprints = ImmutableSet.builder(); NSArray certificates = (NSArray) plist.get("DeveloperCertificates"); HashFunction hasher = Hashing.sha1(); if (certificates != null) { for (NSObject item : certificates.getArray()) { certificateFingerprints.add(hasher.hashBytes(((NSData) item).bytes())); } } ImmutableMap.Builder<String, NSObject> builder = ImmutableMap.builder(); NSDictionary entitlements = ((NSDictionary) plist.get("Entitlements")); for (String key : entitlements.keySet()) { builder = builder.put(key, entitlements.objectForKey(key)); } String appID = entitlements.get("application-identifier").toString(); ProvisioningProfileMetadata.Builder provisioningProfileMetadata = ProvisioningProfileMetadata.builder(); if (plist.get("Platform") != null) { for (Object platform : (Object[]) plist.get("Platform").toJavaObject()) { provisioningProfileMetadata.addPlatforms((String) platform); } } return provisioningProfileMetadata .setAppID(ProvisioningProfileMetadata.splitAppID(appID)) .setExpirationDate(expirationDate) .setUUID(uuid) .setProfilePath(profilePath) .setEntitlements(builder.build()) .setDeveloperCertificateFingerprints(certificateFingerprints.build()) .build(); } catch (Exception e) { throw new IllegalArgumentException("Malformed embedded plist: " + e); } } @Override public void appendToRuleKey(RuleKeyObjectSink sink) { sink.setReflectively("provisioning-profile-uuid", getUUID()); } public ImmutableMap<String, NSObject> getMergeableEntitlements() { final ImmutableSet<String> excludedKeys = ImmutableSet.of( "com.apple.developer.restricted-resource-mode", "inter-app-audio", "com.apple.developer.icloud-container-development-container-identifiers", "com.apple.developer.homekit", "com.apple.developer.healthkit", "com.apple.developer.in-app-payments", "com.apple.developer.maps", "com.apple.external-accessory.wireless-configuration"); ImmutableMap<String, NSObject> allEntitlements = getEntitlements(); ImmutableMap.Builder<String, NSObject> filteredEntitlementsBuilder = ImmutableMap.builder(); for (String key : allEntitlements.keySet()) { if (!excludedKeys.contains(key)) { filteredEntitlementsBuilder.put(key, allEntitlements.get(key)); } } return filteredEntitlementsBuilder.build(); } }