-
-
Save damian-rzeszot/0b23ad87e5ab5d52aa15c095cbf43c59 to your computer and use it in GitHub Desktop.
alias plistbuddy=/usr/libexec/PlistBuddy | |
alias codesign=/usr/bin/codesign | |
# | |
# Bundle identifier | |
# | |
set_plist_bundle_identifier() { | |
local bundle_identifier="$1" | |
local plist_file="$2" | |
plistbuddy \ | |
-c "set :CFBundleIdentifier $bundle_identifier" \ | |
"$plist_file" | |
} | |
set_appex_bundle_identifier() { | |
local appex_target_name="$1" | |
local bundle_identifier_suffix="$2" | |
local bundle_identifier="$PRODUCT_BUNDLE_IDENTIFIER.$bundle_identifier_suffix" | |
local plist_file="$BUILT_PRODUCTS_DIR/$FULL_PRODUCT_NAME/$BUNDLE_PLUGINS_FOLDER_PATH/$appex_target_name.appex/Info.plist" | |
set_plist_bundle_identifier "$bundle_identifier" "$plist_file" | |
} | |
# | |
# Bundle version | |
# | |
set_plist_bundle_version() { | |
local bundle_version="$1" | |
local plist_file="$2" | |
plistbuddy \ | |
-c "set :CFBundleShortVersionString $bundle_version" \ | |
"$plist_file" | |
} | |
get_plsit_bundle_version() { | |
local plist_file="$1" | |
plistbuddy \ | |
-c "Print :CFBundleShortVersionString" \ | |
"$plist_file" | |
} | |
get_app_bundle_version() { | |
local plist_file="$BUILT_PRODUCTS_DIR/$INFOPLIST_PATH" | |
get_plsit_bundle_version "$plist_file" | |
} | |
set_appex_bundle_version() { | |
local appex_target_name="$1" | |
local bundle_version="$2" | |
local plist_file="$BUILT_PRODUCTS_DIR/$FULL_PRODUCT_NAME/$BUNDLE_PLUGINS_FOLDER_PATH/$appex_target_name.appex/Info.plist" | |
set_plist_bundle_version "$bundle_version" "$plist_file" | |
} | |
# | |
# Bundle build | |
# | |
set_plist_bundle_build() { | |
local bundle_build="$1" | |
local plist_file="$2" | |
plistbuddy \ | |
-c "set :CFBundleVersion $bundle_build" \ | |
"$plist_file" | |
} | |
get_plist_bundle_build() { | |
local plist_file="$1" | |
plistbuddy \ | |
-c "Print :CFBundleVersion" \ | |
"$plist_file" | |
} | |
set_appex_bundle_build() { | |
local appex_target_name="$1" | |
local bundle_version="$2" | |
local plist_file="$BUILT_PRODUCTS_DIR/$FULL_PRODUCT_NAME/$BUNDLE_PLUGINS_FOLDER_PATH/$appex_target_name.appex/Info.plist" | |
set_plist_bundle_build "$bundle_version" "$plist_file" | |
} | |
get_app_bundle_build() { | |
local plist_file="$BUILT_PRODUCTS_DIR/$INFOPLIST_PATH" | |
get_plist_bundle_build "$plist_file" | |
} | |
# | |
# Code signing | |
# | |
prepare_entitlements_file() { | |
local appex_target_name="$1" | |
local bundle_identifier_suffix="$2" | |
local output_file="$3" | |
local original_entitlements="$CONFIGURATION_TEMP_DIR/$appex_target_name.build/$appex_target_name.appex.xcent" | |
local bundle_identifier="$DEVELOPMENT_TEAM.$PRODUCT_BUNDLE_IDENTIFIER.$bundle_identifier_suffix" | |
cp "$original_entitlements" "$output_file" | |
if [[ $CONFIGURATION == "Release" ]] | |
then | |
plistbuddy \ | |
-c "set :application-identifier $bundle_identifier" \ | |
"$output_file" | |
plistbuddy \ | |
-c "set :keychain-access-groups:0 $bundle_identifier" \ | |
"$output_file" | |
fi | |
} | |
copy_provisioning_profile() { | |
local appex_target_name="$1" | |
local provision_source="$2" | |
local provision_destination="$BUILT_PRODUCTS_DIR/$FULL_PRODUCT_NAME/$BUNDLE_PLUGINS_FOLDER_PATH/$appex_target_name.appex/$EMBEDDED_PROFILE_NAME" | |
cp "$provision_source" "$provision_destination" | |
} | |
resign_appex() { | |
local appex_target_name="$1" | |
local entitlements_file="$2" | |
codesign \ | |
--force \ | |
--sign "$EXPANDED_CODE_SIGN_IDENTITY" \ | |
--entitlements "$entitlements_file" \ | |
--timestamp=none \ | |
"$BUILT_PRODUCTS_DIR/$FULL_PRODUCT_NAME/$BUNDLE_PLUGINS_FOLDER_PATH/$appex_target_name.appex" | |
} | |
# | |
# | |
# | |
# | |
# | |
set_appex_bundle_identifier \ | |
"NotificationService" \ | |
"notification-service" | |
set_appex_bundle_version \ | |
"NotificationService" \ | |
`get_app_bundle_version` | |
set_appex_bundle_build \ | |
"NotificationService" \ | |
`get_app_bundle_build` | |
# Be careful if using `keychain-access-groups` entitlement | |
prepare_entitlements_file \ | |
"NotificationService" \ | |
"notification-service" \ | |
"$DERIVED_SOURCES_DIR/NotificationService-Entitlements.plist" | |
copy_provisioning_profile \ | |
"NotificationService" \ | |
"$SOURCE_ROOT/../.github/appstore/$TARGET_NAME/profiles/notification-service.mobileprovision" | |
resign_appex \ | |
"NotificationService" \ | |
"$DERIVED_SOURCES_DIR/NotificationService-Entitlements.plist" |
Let's say I have targets:
- AppStaging (com.example.staging)
- App1 (com.example.app1)
- App2 (com.example.app2)
- NotificationContent (com.example.staging.notification-content)
On daily basis I develop AppStaging. The differences between AppStaging and AppX are: Info.plist, Assets and configuration.json file.
AppStaging and AppX has dependency target to NotificationContent.
For Debug
configuration I use automatic signing (I don't care what's going on there if it's working).
For Release
proper provisioning profiles and other signing settings are set (for all targets):
CODE_SIGN_IDENTITY="iOS Distribution"
CODE_SIGN_STYLE="Manual"
PROVISIONING_PROFILE_SPECIFIER="<uuid>"
There is also a build step Override appex
after Embed App Extensions
where I run the script.
And magic happens here. If I build AppStaging basically everything works as expected (overriding will set exact the same value as were set before). But if I build AppX target:
- NotificationContent (com.example.staging.notification-content) is built and signed
- AppX (e.g. com.example.app1) target is built
- NotificationContent appex is embedded into app bundle
// It's clear there is a conflict here - but there is aOverride appex
step - Appex in the app bundle is update (bundle id, version) and signed
- The app is validated ✅
Thank you for answering my questions. I really Appreciate it.
-
AppX means App1 & App2? Or appex (app extension)
-
You said:
For Release proper provisioning profiles and other signing settings are set (for all targets)
Does 'all targets' mean all non-app extension targets or it includes the app extension targets as well?
I mean an app extension needs its own distinct provisioning profile. If you are to have an appex with a bundleId of: com.example.app2. notification-content
then where are you giving its provisioning profile? ...Before the scripts are ran, that bundleId doesn't exist. Hence you can't tell Xcode of it. RIght?
I mean how do set the PROVISIONING_PROFILE_SPECIFIER
for com.example.app2. notification-content
?
- What's the
Override appex
step? You already mentioned the following three:
override_bundle_identifier "NotificationService" "$PRODUCT_BUNDLE_IDENTIFIER.notification-service"
override_app_version "NotificationService"
override_code_signing "NotificationService"
Is that something else? Or you're just naming all 3 steps override appex?
@prohoney you're right, there is an issue with provisioning profile while uploading to App Store. I updated the script. Within this I was able to upload 2 apps to TestFlight.
For one who originally did it wrong you figured it all out super quick. How do you do it?!
This whole build settings stuff is new to me. Took me a few rounds of reading to understand it. Also you should post more frequently on Stack Overflow or blog about stuff.
-
But if I understand correct, the
"$SOURCE_ROOT/../.github/appstore/$TARGET_NAME/profiles/notification-service.mobileprovision
is a predetermined path to the target's the provisioning profile. You just need to to figure out where to put it. You do that that in yourcopy_provisioning_profile
func. Right? -
I like how you pulled in
$DEVELOPMENT_TEAM
into localbundle_identifier="$DEVELOPMENT_TEAM.$PRODUCT_BUNDLE_IDENTIFIER.$bundle_identifier_suffix"
. That helps you to sign with different teams. Right? -
My 'Notification Service Extension' for which I use to download images, doesn't have any entitlements, but I understand the need to modify the bundleId inside the entitlement's plist. Can I ask why yours needed certain entitlements? Is it because of keychain sharing?
-
That's right.
$SOURCE_ROOT/.../notification-service.mobileprovision
is there, because I use GitHub Actions for deploying the build to App Store. I didn't want to dig in target's build settings via e.g.xcodeproj
. That's fair enough solution. -
Yeap. I don't like hardcoding things in scripts.
$DEVELOPMENT_TEAM
is provided by Xcode. -
Xcode generates entitlements. The generated one for the notification service extension contains original [bundle] ids, so it has to be overridden.
Thank you. I also wanted to suggest that you can use plutil
instead of PlistBuddy
. Based on what an Apple Engineer told me plutil
is actively updated, PlistBuddy
seems to not be updated
Also was there a reason you did alias codesign=/usr/bin/codesign
? Couldn't you just used codesign
without doing that? It's already in the bin
directory`
Sorry. One more thing what's the difference between .entitlements
file and .xcent
file?
plutil
- good point 👍alias codesign
- true, it's not needed 👍.xcent
- I see no difference. Probably somebody decided to usexc
prefixes there, as inxcconfig
,xcproject
, etc. Looks like legacy naming.
Can see my answer here and the steps I've written. The steps it has are not complete as yours. But I just don't get how you get yours is working given that you're updating the bundleId after 'embed App Extension' step. Based on my understanding that step should fail.
FYI this fails when I'm using a physical device. I haven't tried archiving yet...
Specifically the point I'm referencing from my answer:
Similarly the appex can't be embedded if the bundleId isn't prefixed with the parent app's bundleId.
i.e. I think bundleId should change before the 'embed App Extension' step.
Or is it that you're also updating the bundleId of the parent app in some other script before changing stuff for the appex ie you're able to embed the Appex's into the main app, but then later change the bundle Id for both the main app and appex?
Basically are you doing this?
- embed apppex
- update parent app bundleId and stuff
- update appex bundleId and stuff
Hi, I'm joining very late but I'm struggling to understand something.
I know that this needs to be after what it used to be "Embed App Extensions" which now I think is called "Embed Foundation Extensions"
But when I generate my (desired) only extension, this is only shown at Build Phases of the default target but not in the other ones. I'm trying to understand if that does not matter? If i build my app with target #2, and the script is between Build phases of target #1, still works? Thank you :)
🤔 How are you setting appropriate provisioning profiles in build settings? If you were able to switch over in build settings then wouldn't you have been able to switch the bundleId and code-singing too? ie no need for bundleId and code-signing related scripts...
I know how to switch over with different configurations but you and I both have different targets...