Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(cli): support manual signing on build command #7769

Open
wants to merge 23 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
2306058
Adding xcode-signing-style build flag
theproducer Nov 15, 2024
4383b67
Adding singing cert and provisioning profile to cli
theproducer Nov 19, 2024
bc63823
Merge branch 'main' into feat/xcode-signing
theproducer Nov 19, 2024
e96c4f1
updating command line description
theproducer Nov 19, 2024
23300df
fmt
theproducer Nov 19, 2024
bc80d75
Merge branch 'main' into feat/xcode-signing
theproducer Nov 20, 2024
8e50ecb
Making xcodeSigningType optional
theproducer Dec 4, 2024
ab7e9b9
Check for empty signing cert and profile during manual builds
theproducer Dec 4, 2024
e40dd06
Merge branch 'main' into feat/xcode-signing
theproducer Dec 4, 2024
4159a23
Adding iOS build options to cap config
theproducer Dec 12, 2024
936f033
Wiring cap config ios build options into build command
theproducer Dec 12, 2024
b87cd77
Merge branch 'main' into feat/xcode-signing
theproducer Dec 16, 2024
400e01f
Merge branch 'main' into feat/xcode-signing
jcesarmobile Dec 18, 2024
069ab09
Merge branch 'main' into feat/xcode-signing
jcesarmobile Dec 19, 2024
79f401f
Merge branch 'main' into feat/xcode-signing
theproducer Jan 14, 2025
af2f2eb
Adding support for setting export method
theproducer Jan 15, 2025
e2347fe
deleting iOS buildOptions from generated capacitor.config.json
theproducer Jan 15, 2025
e438586
Adding exportMethod to capConfig
theproducer Jan 15, 2025
5858e67
Askings for team id
theproducer Jan 16, 2025
6b28c45
fixing manual build and archive configuration
theproducer Jan 16, 2025
3f648b8
Manual signing should not required for setting the team id
theproducer Jan 16, 2025
a9418ba
fmt
theproducer Jan 16, 2025
a209f6e
Changing xocdeExportMethod default to ‘app-store-connect’
theproducer Jan 20, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 7 additions & 1 deletion cli/src/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -267,7 +267,12 @@ async function loadIOSConfig(rootDir: string, extConfig: ExternalConfig): Promis
const podPath = lazy(() => determineGemfileOrCocoapodPath(rootDir, platformDirAbs, nativeProjectDirAbs));
const webDirAbs = lazy(() => determineIOSWebDirAbs(nativeProjectDirAbs, nativeTargetDirAbs, nativeXcodeProjDirAbs));
const cordovaPluginsDir = 'capacitor-cordova-ios-plugins';

const buildOptions = {
xcodeExportMethod: extConfig.ios?.buildOptions?.exportMethod,
xcodeSigningStyle: extConfig.ios?.buildOptions?.signingStyle,
signingCertificate: extConfig.ios?.buildOptions?.signingCertificate,
provisioningProfile: extConfig.ios?.buildOptions?.provisioningProfile,
};
return {
name,
minVersion: '14.0',
Expand All @@ -287,6 +292,7 @@ async function loadIOSConfig(rootDir: string, extConfig: ExternalConfig): Promis
webDir: lazy(async () => relative(platformDirAbs, await webDirAbs)),
webDirAbs,
podPath,
buildOptions,
};
}

Expand Down
29 changes: 29 additions & 0 deletions cli/src/declarations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -478,6 +478,35 @@ export interface CapacitorConfig {
* @default true
*/
initialFocus?: boolean;

buildOptions?: {
/**
* The signing style to use when building the app for distribution.
*
* @since 7.0.0
* @default 'automatic'
*/
signingStyle?: 'automatic' | 'manual';
/**
* The method used by xcodebuild to export the archive
*
* @since 7.0.0
* @default 'debugging'
*/
exportMethod?: string;
/**
* A certificate name, SHA-1 hash, or automatic selector to use for signing for iOS builds.
*
* @since 7.0.0
*/
signingCertificate?: string;
/**
* A provisioning profile name or UUID for iOS builds.
*
* @since 7.0.0
*/
provisioningProfile?: string;
};
};

server?: {
Expand Down
17 changes: 17 additions & 0 deletions cli/src/definitions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,16 @@ export interface AndroidConfig extends PlatformConfig {
};
}

export enum XcodeExportMethod {
AppStoreConnect = 'app-store-connect',
ReleaseTesting = 'release-testing',
Enterprise = 'enterprise',
Debugging = 'debugging',
DeveloperID = 'developer-id',
MacApplication = 'mac-application',
Validation = 'validation',
}

export interface IOSConfig extends PlatformConfig {
readonly cordovaPluginsDir: string;
readonly cordovaPluginsDirAbs: string;
Expand All @@ -117,6 +127,13 @@ export interface IOSConfig extends PlatformConfig {
readonly nativeXcodeProjDirAbs: string;
readonly nativeXcodeWorkspaceDir: Promise<string>;
readonly nativeXcodeWorkspaceDirAbs: Promise<string>;
readonly buildOptions: {
teamId?: string;
exportMethod?: XcodeExportMethod;
xcodeSigningStyle?: 'automatic' | 'manual';
signingCertificate?: string;
provisioningProfile?: string;
};
}

export type WebConfig = PlatformConfig;
Expand Down
45 changes: 45 additions & 0 deletions cli/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,41 @@ export function runProgram(config: Config): void {
'jarsigner',
]),
)
.addOption(
new Option('--xcode-team-id <xcodeTeamID>', 'The Developer team to use for building and exporting the archive'),
)
.addOption(
new Option(
'--xcode-export-method <xcodeExportMethod>',
'Describes how xcodebuild should export the archive (default: app-store-connect)',
).choices([
'app-store-connect',

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These look correct, verified via xcodebuild -help, method section

'release-testing',
'enterprise',
'debugging',
'developer-id',
'mac-application',
'validation',
]),
)
.addOption(
new Option(
'--xcode-signing-style <xcodeSigningStyle>',
'The iOS signing style to use when building the app for distribution (default: automatic)',
).choices(['automatic', 'manual']),
)
.addOption(
new Option(
'--xcode-signing-certificate <xcodeSigningCertificate>',
'A certificate name, SHA-1 hash, or automatic selector to use for signing for iOS builds',
),
)
.addOption(
new Option(
'--xcode-provisioning-profile <xcodeProvisioningProfile>',
'A provisioning profile name or UUID for iOS builds',
),
)
.action(
wrapAction(
telemetryAction(
Expand All @@ -163,6 +198,11 @@ export function runProgram(config: Config): void {
androidreleasetype,
signingType,
configuration,
xcodeTeamId,
xcodeExportMethod,
xcodeSigningStyle,
xcodeSigningCertificate,
xcodeProvisioningProfile,
},
) => {
const { buildCommand } = await import('./tasks/build');
Expand All @@ -176,6 +216,11 @@ export function runProgram(config: Config): void {
androidreleasetype,
signingtype: signingType,
configuration,
xcodeTeamId,
xcodeExportMethod,
xcodeSigningType: xcodeSigningStyle,
xcodeSigningCertificate,
xcodeProvisioningProfile,
});
},
),
Expand Down
103 changes: 64 additions & 39 deletions cli/src/ios/build.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ import { basename, join } from 'path';
import { rimraf } from 'rimraf';

import { runTask } from '../common';
import type { Config } from '../definitions';
import { XcodeExportMethod, type Config } from '../definitions';
import { logSuccess } from '../log';
import type { BuildCommandOptions } from '../tasks/build';
import { type BuildCommandOptions } from '../tasks/build';
import { checkPackageManager } from '../util/spm';
import { runCommand } from '../util/subprocess';

Expand All @@ -25,59 +25,84 @@ export async function buildiOS(config: Config, buildOptions: BuildCommandOptions
projectName = basename(await config.ios.nativeXcodeProjDirAbs);
}

if (
buildOptions.xcodeSigningType == 'manual' &&
(!buildOptions.xcodeSigningCertificate || !buildOptions.xcodeProvisioningProfile)
) {
throw 'Manually signed Xcode builds require a signing certificate and provisioning profile.';
}

const buildArgs = [
typeOfBuild,
projectName,
'-scheme',
`${theScheme}`,
'-destination',
`generic/platform=iOS`,
'-archivePath',
`${theScheme}.xcarchive`,
'archive',
];

if (buildOptions.xcodeTeamId) {
buildArgs.push(`DEVELOPMENT_TEAM=${buildOptions.xcodeTeamId}`);
}

if (buildOptions.xcodeSigningType == 'manual') {
buildArgs.push(`PROVISIONING_PROFILE_SPECIFIER=${buildOptions.xcodeProvisioningProfile}`);
}

await runTask('Building xArchive', async () =>
runCommand(
'xcodebuild',
[
typeOfBuild,
projectName,
'-scheme',
`${theScheme}`,
'-destination',
`generic/platform=iOS`,
'-archivePath',
`${theScheme}.xcarchive`,
'archive',
],
{
cwd: config.ios.nativeProjectDirAbs,
},
),
runCommand('xcodebuild', buildArgs, {
cwd: config.ios.nativeProjectDirAbs,
}),
);

const manualSigningContents = `<key>provisioningProfiles</key>
<dict>
<key>${config.app.appId}</key>
<string>${buildOptions.xcodeProvisioningProfile ?? ''}</string>
</dict>
<key>signingCertificate</key>
<string>${buildOptions.xcodeSigningCertificate ?? ''}</string>`;
Comment on lines +63 to +67
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will these be OK as empty strings, or would it fail the build?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, that block is only added if signing style is manual, and if that's chosen, presumably, the developer would be supplying a provisioning profile and/or signing certificate. If what they supply is wrong (or if it's empty), then it won't work.

I believe the signing certificate can be optional, so I may push up a change to remove that key and string if the value is empty.


const archivePlistContents = `<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>method</key>
<string>app-store-connect</string>
<string>${buildOptions.xcodeExportMethod ?? XcodeExportMethod.Debugging}</string>
<key>signingStyle</key>
<string>${buildOptions.xcodeSigningType}</string>
${buildOptions.xcodeSigningType == 'manual' ? manualSigningContents : ''}
</dict>
</plist>`;

const archivePlistPath = join(`${config.ios.nativeProjectDirAbs}`, 'archive.plist');

writeFileSync(archivePlistPath, archivePlistContents);

const archiveArgs = [
'archive',
'-archivePath',
`${theScheme}.xcarchive`,
'-exportArchive',
'-exportOptionsPlist',
'archive.plist',
'-exportPath',
'output',
'-configuration',
buildOptions.configuration,
];

if (buildOptions.xcodeSigningType == 'automatic') {
archiveArgs.push('-allowProvisioningUpdates');
}

await runTask('Building IPA', async () =>
runCommand(
'xcodebuild',
[
'archive',
'-archivePath',
`${theScheme}.xcarchive`,
'-exportArchive',
'-exportOptionsPlist',
'archive.plist',
'-exportPath',
'output',
'-allowProvisioningUpdates',
'-configuration',
buildOptions.configuration,
],
{
cwd: config.ios.nativeProjectDirAbs,
},
),
runCommand('xcodebuild', archiveArgs, {
cwd: config.ios.nativeProjectDirAbs,
}),
);

await runTask('Cleaning up', async () => {
Expand Down
12 changes: 12 additions & 0 deletions cli/src/tasks/build.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { buildAndroid } from '../android/build';
import { selectPlatforms, promptForPlatform } from '../common';
import type { Config } from '../definitions';
import { XcodeExportMethod } from '../definitions';
import { fatal, isFatal } from '../errors';
import { buildiOS } from '../ios/build';

Expand All @@ -14,6 +15,11 @@ export interface BuildCommandOptions {
androidreleasetype?: 'AAB' | 'APK';
signingtype?: 'apksigner' | 'jarsigner';
configuration: string;
xcodeTeamId?: string;
xcodeExportMethod?: XcodeExportMethod;
xcodeSigningType?: 'automatic' | 'manual';
xcodeSigningCertificate?: string;
xcodeProvisioningProfile?: string;
}

export async function buildCommand(
Expand Down Expand Up @@ -42,6 +48,12 @@ export async function buildCommand(
androidreleasetype: buildOptions.androidreleasetype || config.android.buildOptions.releaseType || 'AAB',
signingtype: buildOptions.signingtype || config.android.buildOptions.signingType || 'jarsigner',
configuration: buildOptions.configuration || 'Release',
xcodeTeamId: buildOptions.xcodeTeamId || config.ios.buildOptions.teamId,
xcodeExportMethod:
buildOptions.xcodeExportMethod || config.ios.buildOptions.exportMethod || XcodeExportMethod.AppStoreConnect,
xcodeSigningType: buildOptions.xcodeSigningType || config.ios.buildOptions.xcodeSigningStyle || 'automatic',
xcodeSigningCertificate: buildOptions.xcodeSigningCertificate || config.ios.buildOptions.signingCertificate,
xcodeProvisioningProfile: buildOptions.xcodeProvisioningProfile || config.ios.buildOptions.provisioningProfile,
};

try {
Expand Down
1 change: 1 addition & 0 deletions cli/src/tasks/copy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ async function copyCapacitorConfig(config: Config, nativeAbsDir: string) {

await runTask(`Creating ${c.strong(nativeConfigFile)} in ${nativeRelDir}`, async () => {
delete (config.app.extConfig.android as any)?.buildOptions;
delete (config.app.extConfig.ios as any)?.buildOptions;
await writeJSON(nativeConfigFilePath, config.app.extConfig, {
spaces: '\t',
});
Expand Down
Loading