diff --git a/.github/workflows/automatic_pull_request_review.yml b/.github/workflows/automatic_pull_request_review.yml index 8bcdb9dd..8b5b107c 100644 --- a/.github/workflows/automatic_pull_request_review.yml +++ b/.github/workflows/automatic_pull_request_review.yml @@ -7,7 +7,7 @@ on: jobs: review_pull_request: name: Pull request review by Danger - runs-on: macOS-latest + runs-on: macOS-12 steps: - name: Cancel Previous Runs uses: styfle/cancel-workflow-action@0.5.0 @@ -42,12 +42,12 @@ jobs: run: bundle exec pod install - name: Build and Test - run: bundle exec fastlane build_and_test + run: bundle exec fastlane buildAndTest env: CI: true - name: Clean up previous code coverage report - run: bundle exec fastlane clean_up_xcov + run: bundle exec fastlane cleanUpOutput - name: Review pull request by Danger env: diff --git a/.github/workflows/deploy_Firebase.yml b/.github/workflows/deploy_Firebase.yml index 1c712ff3..49c4cf19 100644 --- a/.github/workflows/deploy_Firebase.yml +++ b/.github/workflows/deploy_Firebase.yml @@ -24,7 +24,7 @@ jobs: uses: norio-nomura/action-swiftlint@3.1.0 with: args: --strict - + build: name: Build runs-on: macOS-latest @@ -48,7 +48,7 @@ jobs: run: | yarn global add firebase-tools echo "$(yarn global bin)" >> $GITHUB_PATH - + - name: Bundle install # if: steps.bundleCache.outputs.cache-hit != 'true' run: bundle install @@ -67,15 +67,15 @@ jobs: shell: bash - name: Build and Test - run: bundle exec fastlane build_and_test + run: bundle exec fastlane buildAndTest - name: Match Ad-hoc - run: bundle exec fastlane sync_adhoc_signing + run: bundle exec fastlane syncAdHocStagingCodeSigning env: MATCH_PASSWORD: ${{ secrets.MATCH_PASS }} - name: Build App and Distribute to Firebase - run: bundle exec fastlane build_and_upload_staging_app + run: bundle exec fastlane buildStagingAndUploadToFirebase env: FIREBASE_TOKEN: ${{ secrets.FIREBASE_TOKEN }} diff --git a/.github/workflows/deploy_Release_Firebase.yml b/.github/workflows/deploy_Release_Firebase.yml index fcb6ff75..c63c0333 100644 --- a/.github/workflows/deploy_Release_Firebase.yml +++ b/.github/workflows/deploy_Release_Firebase.yml @@ -24,7 +24,7 @@ jobs: uses: norio-nomura/action-swiftlint@3.1.0 with: args: --strict - + build: name: Build runs-on: macOS-latest @@ -43,7 +43,7 @@ jobs: uses: webfactory/ssh-agent@v0.5.4 with: ssh-private-key: ${{ secrets.SSH_PRIVATE_KEY }} - + - name: Bundle install run: bundle install @@ -61,15 +61,15 @@ jobs: shell: bash - name: Build and Test - run: bundle exec fastlane build_and_test + run: bundle exec fastlane buildAndTest - name: Match Ad-hoc - run: bundle exec fastlane sync_adhoc_production_signing + run: bundle exec fastlane syncAdHocProductionCodeSigning env: MATCH_PASSWORD: ${{ secrets.MATCH_PASS }} - name: Build Production App and Distribute to Firebase - run: bundle exec fastlane build_and_upload_production_app + run: bundle exec fastlane buildProductionAndUploadToFirebase env: FIREBASE_TOKEN: ${{ secrets.FIREBASE_TOKEN }} diff --git a/.github/workflows/deploy_AppStore.yml b/.github/workflows/deploy_app_store.yml similarity index 91% rename from .github/workflows/deploy_AppStore.yml rename to .github/workflows/deploy_app_store.yml index a4e33d7c..0277e1fc 100644 --- a/.github/workflows/deploy_AppStore.yml +++ b/.github/workflows/deploy_app_store.yml @@ -24,7 +24,7 @@ jobs: uses: norio-nomura/action-swiftlint@3.1.0 with: args: --strict - + build: name: Build runs-on: macOS-latest @@ -44,7 +44,7 @@ jobs: uses: webfactory/ssh-agent@v0.5.4 with: ssh-private-key: ${{ secrets.SSH_PRIVATE_KEY }} - + - name: Bundle install run: bundle install @@ -62,15 +62,15 @@ jobs: shell: bash - name: Build and Test - run: bundle exec fastlane build_and_test + run: bundle exec fastlane buildAndTest - name: Match AppStore - run: bundle exec fastlane sync_appstore_signing + run: bundle exec fastlane syncAppStoreCodeSigning env: MATCH_PASSWORD: ${{ secrets.MATCH_PASS }} - name: Build App and Distribute to AppStore - run: bundle exec fastlane build_and_upload_appstore_app + run: bundle exec fastlane buildAndUploadToAppStore env: APPSTORE_CONNECT_API_KEY: ${{ secrets.APPSTORE_CONNECT_API_KEY }} diff --git a/.github/workflows/draft-a-new-release.yml b/.github/workflows/draft_a_new_release.yml similarity index 100% rename from .github/workflows/draft-a-new-release.yml rename to .github/workflows/draft_a_new_release.yml diff --git a/.github/workflows/publish_docs_to_wiki.yml b/.github/workflows/publish_docs_to_wiki.yml index a8cadfbe..dc71b021 100644 --- a/.github/workflows/publish_docs_to_wiki.yml +++ b/.github/workflows/publish_docs_to_wiki.yml @@ -16,4 +16,4 @@ jobs: USER_NAME: team-nimblehq USER_EMAIL: dev@nimblehq.co secrets: - USER_TOKEN: ${{ secrets.NIMBLE_DEV_TOKEN }} + USER_TOKEN: ${{ secrets.NIMBLE_DEV_TOKEN }} diff --git a/.github/workflows/test_install_script.yml b/.github/workflows/test_install_script.yml index de2bfeaa..115a66d3 100644 --- a/.github/workflows/test_install_script.yml +++ b/.github/workflows/test_install_script.yml @@ -7,7 +7,7 @@ on: jobs: Test: name: Test - runs-on: macOS-latest + runs-on: macOS-12 steps: - name: Cancel Previous Runs uses: styfle/cancel-workflow-action@0.5.0 @@ -25,6 +25,6 @@ jobs: run: sh make.sh --bundle-id co.nimblehq.template --bundle-id-staging co.nimblehq.template.staging --project-name TemplateApp - name: Build and Test - run: bundle exec fastlane build_and_test + run: bundle exec fastlane buildAndTest env: CI: true diff --git a/.github/workflows/test_upload_build_to_test_flight.yml b/.github/workflows/test_upload_build_to_test_flight.yml index 3d767cde..bbaa2043 100644 --- a/.github/workflows/test_upload_build_to_test_flight.yml +++ b/.github/workflows/test_upload_build_to_test_flight.yml @@ -11,7 +11,7 @@ name: Test Upload Build to TestFlight on: pull_request -jobs: +jobs: build: name: Build runs-on: macOS-12 @@ -30,7 +30,7 @@ jobs: uses: webfactory/ssh-agent@v0.5.4 with: ssh-private-key: ${{ secrets.SSH_PRIVATE_KEY }} - + - name: Bundle install run: bundle install @@ -42,7 +42,7 @@ jobs: key: ${{ runner.os }}-pods-${{ hashFiles('**/Podfile.lock') }} restore-keys: | ${{ runner.os }}-pods- - + - name: Start Install Script for Template App run: sh make.sh --bundle-id co.nimblehq.ios.templates --bundle-id-staging co.nimblehq.ios.templates.staging --project-name TemplateApp @@ -54,16 +54,18 @@ jobs: ISSUER_ID: ${{ secrets.ISSUER_ID }} - name: Set Up Test Project for App Store - run: bundle exec fastlane set_up_test_project + run: bundle exec fastlane setUpTestProject - name: Update Provision Profile - run: bundle exec fastlane update_provision_settings + run: bundle exec fastlane updateProvisionSettings env: MATCH_PASSWORD: ${{ secrets.MATCH_PASS }} - + - name: Build App and Distribute to AppStore - run: bundle exec fastlane build_and_upload_appstore_app + run: bundle exec fastlane buildAndUploadToAppStore env: APPSTORE_CONNECT_API_KEY: ${{ secrets.APPSTORE_CONNECT_API_KEY }} + API_KEY_ID: ${{ secrets.API_KEY_ID }} + ISSUER_ID: ${{ secrets.ISSUER_ID }} SKIP_FIREBASE_DSYM: "true" BUMP_APP_STORE_BUILD_NUMBER: "true" diff --git a/.gitignore b/.gitignore index ba1271a7..d6e4bf5e 100644 --- a/.gitignore +++ b/.gitignore @@ -64,12 +64,13 @@ fastlane/Preview.html fastlane/screenshots/**/*.png fastlane/test_output fastlane/xcov_output +fastlane/FastlaneRunner # Sourcery *.generated.swift -# Derived Data folder -DerivedData +# Output folder (Derived Data, Build, ipa, etc) +Output # Tuist .tuist-generated diff --git a/Dangerfile b/Dangerfile index 85449fe6..5ccf3dff 100644 --- a/Dangerfile +++ b/Dangerfile @@ -23,7 +23,10 @@ swiftlint.lint_files( additional_swiftlint_args: '--strict' ) -xcresultPath = "#{Constants.TEST_OUTPUT_DIRECTORY_PATH}/#{Constants.TESTS_SCHEME}.xcresult" +xcresultPath = "./fastlane/test_output/{PROJECT_NAME} Staging.xcresult" +scheme = "'{PROJECT_NAME} Staging'" +workspace = "./{PROJECT_NAME}.xcworkspace" +xcovOutputDirectory = "./fastlane/xcov_output" # Xcode summary changed_files = (git.modified_files - git.deleted_files) + git.added_files @@ -40,9 +43,9 @@ xcode_summary.report xcresultPath # Xcov xcov.report( - scheme: Constants.TESTS_SCHEME, - workspace: Constants.WORKSPACE_PATH, - output_directory: Constants.XCOV_OUTPUT_DIRECTORY_PATH, + scheme: scheme, + workspace: workspace, + output_directory: xcovOutputDirectory, xccov_file_direct_path: xcresultPath, only_project_targets: true, markdown_report: true, diff --git a/Podfile b/Podfile index 0c06ba70..fb6ca9ed 100644 --- a/Podfile +++ b/Podfile @@ -53,7 +53,6 @@ post_install do |installer| target.build_configurations.each do |config| config.build_settings.delete 'IPHONEOS_DEPLOYMENT_TARGET' config.build_settings['ENABLE_BITCODE'] = 'NO' - # Fix some pods creating bundle, asking for signing if target.respond_to?(:product_type) and target.product_type == "com.apple.product-type.bundle" target.build_configurations.each do |config| config.build_settings['CODE_SIGNING_ALLOWED'] = 'NO' diff --git a/bitrise.yml b/bitrise.yml index 5ad914fc..ebe78c74 100644 --- a/bitrise.yml +++ b/bitrise.yml @@ -41,7 +41,7 @@ workflows: - fastlane@3: title: Build and upload Production app to App Store inputs: - - lane: build_and_upload_appstore_app + - lane: buildAndUploadToAppStoreLane - deploy-to-bitrise-io: inputs: - deploy_path: $BUILD_PATH @@ -86,7 +86,7 @@ workflows: - fastlane@3: title: Build and Upload Production App inputs: - - lane: build_and_upload_production_app + - lane: buildProductionAndUploadToFirebaseLane - deploy-to-bitrise-io: inputs: - deploy_path: $BUILD_PATH @@ -131,7 +131,7 @@ workflows: - fastlane@3: title: Build and Upload Staging App inputs: - - lane: build_and_upload_staging_app + - lane: buildStagingAndUploadToFirebaseLane - deploy-to-bitrise-io: inputs: - deploy_path: $BUILD_PATH @@ -167,10 +167,10 @@ workflows: - cocoapods-install@2: {} - fastlane@3: inputs: - - lane: build_and_test + - lane: buildAndTestLane - fastlane@3: inputs: - - lane: clean_up_xcov + - lane: cleanUpOutputLane - danger@2: inputs: - github_api_token: "$DANGER_GITHUB_API_TOKEN" diff --git a/fastlane/.env.example b/fastlane/.env.example deleted file mode 100644 index 7be6bf7b..00000000 --- a/fastlane/.env.example +++ /dev/null @@ -1,4 +0,0 @@ -FIREBASE_CLI_TOKEN = -KEYCHAIN_PASSWORD = -APPSTORE_CONNECT_API_KEY = -MANUAL_VERSION = diff --git a/fastlane/Appfile.swift b/fastlane/Appfile.swift new file mode 100644 index 00000000..a576f5c3 --- /dev/null +++ b/fastlane/Appfile.swift @@ -0,0 +1,7 @@ +var appIdentifier: String { return "[[APP_IDENTIFIER]]" } // The bundle identifier of your app +var appleID: String { return "[[APPLE_ID]]" } // Your Apple Developer Portal username + + + +// For more information about the Appfile, see: +// https://docs.fastlane.tools/advanced/#appfile diff --git a/fastlane/Constants/Constant.swift b/fastlane/Constants/Constant.swift new file mode 100644 index 00000000..006207f3 --- /dev/null +++ b/fastlane/Constants/Constant.swift @@ -0,0 +1,139 @@ +// +// Constant.swift +// FastlaneRunner +// +// Created by Su Ho on 22/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +import Foundation + +enum Constant { + + // MARK: - App Store + + static let testFlightTesterGroups = ["<#group1#>", "<#group2#>"] + + // MARK: - Firebase + + static let stagingFirebaseAppId = "<#stagingFirebaseAppId#>" + static let productionFirebaseAppId = "<#productionFirebaseAppId#>" + static let firebaseTesterGroups = "<#group1#>, <#group2#>" + + // MARK: - Match + + static let userName = "<#userName#>" + static let teamId = "<#teamId#>" + static let keychainName = "github_action_keychain" + static let matchURL = "git@github.com:{organization}/{repo}.git" + + // MARK: - Path + + static let outputPath = "./Output" + static let buildPath = "\(outputPath)/Build" + static let derivedDataPath = "\(outputPath)/DerivedData" + static let projectPath: String = "./\(projectName).xcodeproj" + static let testOutputDirectoryPath = "./fastlane/test_output" + static let infoPlistPath = "\(projectName)/Configurations/Plists/Info.plist" + + // MARK: Platform + + static var platform: PlatformType { + if EnvironmentParser.bool(key: "BITRISE_IO") { + return .bitrise + } else if EnvironmentParser.bool(key: "GITHUB_ACTIONS") { + return .gitHubActions + } + return .unknown + } + + // MARK: - Project + + static let stagingBundleId = "{BUNDLE_ID_STAGING}" + static let productionBundleId = "{BUNDLE_ID_PRODUCTION}" + static let projectName = "{PROJECT_NAME}" + + // MARK: - Symbol + + static let uploadSymbolsBinaryPath: String = "./Pods/FirebaseCrashlytics/upload-symbols" + static let dSYMSuffix: String = ".dSYM.zip" + + // MARK: - Build and Version + + static let manualVersion: String = "" + + // MARK: - Device + + static let devices = ["iPhone 12 Pro Max"] + + // MARK: - Test + + static let testTarget: String = "\(projectName)Tests" + static let uiTestTarget: String = "\(projectName)UITests" +} + +extension Constant { + + enum Environment: String { + + case staging = "Staging" + case production = "" + + var productName: String { "\(Constant.projectName) \(rawValue)".trimmed } + + var scheme: String { "\(Constant.projectName) \(rawValue)".trimmed } + + var bundleId: String { + switch self { + case .staging: return Constant.stagingBundleId + case .production: return Constant.productionBundleId + } + } + + var firebaseAppId: String { + switch self { + case .staging: return Constant.stagingFirebaseAppId + case .production: return Constant.productionFirebaseAppId + } + } + + var gspPath: String { + let infoName = "GoogleService-Info.plist" + let googleServiceFolder = "./\(Constant.projectName)/Configurations/Plists/GoogleService" + switch self { + case .staging: return "\(googleServiceFolder)/Staging/\(infoName)" + case .production: return "\(googleServiceFolder)/Production/\(infoName)" + } + } + + var dsymPath: String { + let outputDirectoryURL = URL(fileURLWithPath: Constant.outputPath) + return outputDirectoryURL.appendingPathComponent(productName + ".app" + Constant.dSYMSuffix).relativePath + } + } + + enum BuildType: String { + + case adHoc = "ad-hoc" + case appStore = "app-store" + + var value: String { return rawValue } + + var method: String { + switch self { + case .adHoc: return "AdHoc" + case .appStore: return "AppStore" + } + } + } + + enum PlatformType { + + case gitHubActions, bitrise, unknown + } +} + +extension String { + + fileprivate var trimmed: String { trimmingCharacters(in: .whitespacesAndNewlines) } +} diff --git a/fastlane/Constants/Constants.rb b/fastlane/Constants/Constants.rb deleted file mode 100644 index 195ef1d6..00000000 --- a/fastlane/Constants/Constants.rb +++ /dev/null @@ -1,124 +0,0 @@ -# frozen_string_literal: true - -class Constants - ################# - #### PROJECT #### - ################# - - # Workspace path - def self.WORKSPACE_PATH - './{PROJECT_NAME}.xcworkspace' - end - - # Project path - def self.PROJECT_PATH - './{PROJECT_NAME}.xcodeproj' - end - - # bundle ID for Staging app - def self.BUNDLE_ID_STAGING - '{BUNDLE_ID_STAGING}' - end - - # bundle ID for Production app - def self.BUNDLE_ID_PRODUCTION - '{BUNDLE_ID_PRODUCTION}' - end - - ################# - #### BUILDING ### - ################# - - # a derived data path - def self.DERIVED_DATA_PATH - './DerivedData' - end - - # a build path - def self.BUILD_PATH - './Build' - end - - ################# - #### TESTING #### - ################# - - # a device name - def self.DEVICE - ENV.fetch('DEVICE', 'iPhone 12 Pro Max') - end - - # a scheme name for testing - def self.TESTS_SCHEME - '{PROJECT_NAME} Staging' - end - - # a target name for tests - def self.TESTS_TARGET - '{PROJECT_NAME}Tests' - end - - # a target name for UI tests - def self.UI_TESTS_TARGET - '{PROJECT_NAME}UITests' - end - - # xcov output directory path - def self.XCOV_OUTPUT_DIRECTORY_PATH - './fastlane/xcov_output' - end - - # test output directory path - def self.TEST_OUTPUT_DIRECTORY_PATH - './fastlane/test_output' - end - - ################# - #### KEYCHAIN #### - ################# - - # Keychain name - def self.KEYCHAIN_NAME - 'github_action_keychain' - end - - # a scheme name for unit testing - def self.KEYCHAIN_PASSWORD - 'password' - end - - ################# - ### ARCHIVING ### - ################# - - # an staging environment scheme name - def self.SCHEME_NAME_STAGING - '{PROJECT_NAME} Staging' - end - - # a Production environment scheme name - def self.SCHEME_NAME_PRODUCTION - '{PROJECT_NAME}' - end - - # an staging product name - def self.PRODUCT_NAME_STAGING - '{PROJECT_NAME} Staging' - end - - # a staging TestFlight product name - def self.PRODUCT_NAME_STAGING_TEST_FLIGHT - '{PROJECT_NAME} TestFlight' - end - - # a Production product name - def self.PRODUCT_NAME_PRODUCTION - '{PROJECT_NAME}' - end - - # a main target name - def self.MAIN_TARGET_NAME - '{PROJECT_NAME}' - end - -end diff --git a/fastlane/Constants/DeliverableConstants.rb b/fastlane/Constants/DeliverableConstants.rb deleted file mode 100644 index a6532a81..00000000 --- a/fastlane/Constants/DeliverableConstants.rb +++ /dev/null @@ -1,67 +0,0 @@ -# frozen_string_literal: true - -class DeliverableConstants - - ################## - #### FIREBASE #### - ################## - - # a gsp files directory - def self.GSP_DIRECTORY - './' - end - - # a gsp file name for staging - def self.GSP_STAGING - './{PROJECT_NAME}/Configurations/Plists/GoogleService/Staging/GoogleService-Info.plist' - end - - # a gsp file name for production - def self.GSP_PRODUCTION - './{PROJECT_NAME}/Configurations/Plists/GoogleService/Production/GoogleService-Info.plist' - end - - # The path to the upload-symbols file of the Fabric app - def self.BINARY_PATH - './Pods/FirebaseCrashlytics/upload-symbols' - end - - # a firebase app ID for Staging - def self.FIREBASE_APP_ID_STAGING - '<#PROJECT_FIREBASE_APP_ID#>' - end - - # a firebase app ID for Production - def self.FIREBASE_APP_ID_PRODUCTION - '<#PROJECT_FIREBASE_APP_ID#>' - end - - # Firebase Tester group name, seperate by comma(,) string - def self.FIREBASE_TESTER_GROUPS - "nimble-dev" - end - - ################## - ### DEV PORTAL ### - ################## - - # Apple ID for Apple Developer Portal - def self.DEV_PORTAL_APPLE_ID - '<#dev@example.com#>' - end - - ##################### - ### App Store API ### - ##################### - - # App Store Connect API Key ID - def self.APP_STORE_KEY_ID - '<#API_KEY_ID#>' - end - - # App Store Connect API Issuer ID - def self.APP_STORE_ISSUER_ID - '<#ISSUER_ID#>' - end - -end diff --git a/fastlane/Constants/Environments.rb b/fastlane/Constants/Environments.rb deleted file mode 100644 index a5a487ce..00000000 --- a/fastlane/Constants/Environments.rb +++ /dev/null @@ -1,38 +0,0 @@ -class Environments - - def self.FIREBASE_CLI_TOKEN - ENV['FIREBASE_CLI_TOKEN'] - end - - def self.KEYCHAIN_PASSWORD - ENV['KEYCHAIN_PASSWORD'] - end - - def self.CI - ENV['CI'] - end - - def self.APPSTORE_CONNECT_API_KEY - ENV['APPSTORE_CONNECT_API_KEY'] - end - - def self.MANUAL_VERSION - ENV['MANUAL_VERSION'] - end - - def self.SKIP_FIREBASE_DSYM - ENV['SKIP_FIREBASE_DSYM'] - end - - def self.BUMP_APP_STORE_BUILD_NUMBER - ENV['BUMP_APP_STORE_BUILD_NUMBER'] - end - - def self.GITHUB_ACTIONS - ENV['GITHUB_ACTIONS'] - end - - def self.BITRISE_IO - ENV['BITRISE_IO'] - end -end diff --git a/fastlane/Constants/Secret.swift b/fastlane/Constants/Secret.swift new file mode 100644 index 00000000..370d094d --- /dev/null +++ b/fastlane/Constants/Secret.swift @@ -0,0 +1,23 @@ +// +// Secret.swift +// FastlaneRunner +// +// Created by Su Ho on 22/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +// This file will be added manually from CI platform (Github Actions / Bitrise / ...) +enum Secret { + + static let keychainPassword = EnvironmentParser.string(key: "KEYCHAIN_PASSWORD") + + static let firebaseCLIToken = EnvironmentParser.string(key: "FIREBASE_CLI_TOKEN") + + static let appstoreConnectAPIKey = EnvironmentParser.string(key: "APPSTORE_CONNECT_API_KEY") + + static let appStoreKeyIdKey = EnvironmentParser.string(key: "API_KEY_ID") + + static let appStoreIssuerIdKey = EnvironmentParser.string(key: "ISSUER_ID") + + static let bumpAppStoreBuildNumber = EnvironmentParser.bool(key: "BUMP_APP_STORE_BUILD_NUMBER") +} diff --git a/fastlane/Fastfile b/fastlane/Fastfile deleted file mode 100644 index 8e1e07ef..00000000 --- a/fastlane/Fastfile +++ /dev/null @@ -1,266 +0,0 @@ -# frozen_string_literal: true - -require './Constants/Constants' -require './Constants/Environments' -require './Constants/DeliverableConstants' -require './Managers/VersioningManager' -require './Managers/BuildManager' -require './Managers/SymbolManager' -require './Managers/DistributionManager' -require './Managers/MatchManager' -require './Managers/TestManager' -require './Managers/EnvironmentManager' - -test_manager = TestManager.new( - fastlane: self, - device: Constants.DEVICE, - output_directory: Constants.TEST_OUTPUT_DIRECTORY_PATH -) - -versioning_manager = VersioningManager.new( - fastlane: self, - project_path: Constants.PROJECT_PATH, - main_target_name: Constants.MAIN_TARGET_NAME -) - -builder = BuildManager.new(fastlane: self) - -distribution_manager = DistributionManager.new( - fastlane: self, - build_path: Constants.BUILD_PATH, - firebase_token: Environments.FIREBASE_CLI_TOKEN -) - -match_manager = MatchManager.new( - fastlane: self, - keychain_name: Constants.KEYCHAIN_NAME, - keychain_password: Constants.KEYCHAIN_PASSWORD, - is_ci: Environments.CI, - username: DeliverableConstants.DEV_PORTAL_APPLE_ID -) - -environment_manager = EnvironmentManager.new( - fastlane: self, - is_github_actions: Environments.GITHUB_ACTIONS, - is_bitrise: Environments.BITRISE_IO, - build_path: Constants.BUILD_PATH -) - -before_all do - ensure_bundle_exec -end - -default_platform(:ios) - -platform :ios do - - # Testing - - desc 'Build and Test' - lane :build_and_test do - test_manager.build_and_test( - scheme: Constants.TESTS_SCHEME, - targets: [Constants.TESTS_TARGET, Constants.UI_TESTS_TARGET] - ) - end - - # Code Sign - - desc 'Sync Development match signing' - lane :sync_development_signing do - match_manager.sync_development_signing( - app_identifier: [ - Constants.BUNDLE_ID_STAGING, - Constants.BUNDLE_ID_PRODUCTION - ] - ) - end - - desc 'Sync Adhoc match signing' - lane :sync_adhoc_signing do - match_manager.sync_adhoc_signing(app_identifier: [Constants.BUNDLE_ID_STAGING]) - end - - desc 'Sync Adhoc Production match signing' - lane :sync_adhoc_production_signing do - match_manager.sync_adhoc_signing(app_identifier: [Constants.BUNDLE_ID_PRODUCTION]) - end - - desc 'Sync AppStore match signing' - lane :sync_appstore_signing do - match_manager.sync_app_store_signing(app_identifier: [Constants.BUNDLE_ID_PRODUCTION]) - end - - desc 'Register new devices' - lane :register_new_device do - device_name = prompt(text: 'Enter the device name: ') - device_udid = prompt(text: 'Enter the device UDID: ') - device_hash = {} - device_hash[device_name] = device_udid - register_devices( - devices: device_hash, - username: DeliverableConstants.DEV_PORTAL_APPLE_ID - ) - - match_manager.sync_development_signing(app_identifier: nil, force: true) - match_manager.sync_adhoc_signing(app_identifier: nil, force: true) - end - - # Firebase - - desc 'Build and upload staging app to Firebase' - lane :build_and_upload_staging_app do - set_app_version - bump_build - - builder.build_ad_hoc( - Constants.SCHEME_NAME_STAGING, - Constants.PRODUCT_NAME_STAGING, - Constants.BUNDLE_ID_STAGING - ) - upload_build_to_firebase( - product_name: Constants.PRODUCT_NAME_PRODUCTION, - firebase_app_id: DeliverableConstants.FIREBASE_APP_ID_STAGING, - tester_groups: DeliverableConstants.FIREBASE_TESTER_GROUPS - ) - upload_build_dsym_to_firebase( - product_name: Constants.PRODUCT_NAME_STAGING, - gsp_name: DeliverableConstants.GSP_STAGING - ) - environment_manager.save_build_context_to_ci(version_number: versioning_manager.version_number) - end - - desc 'Build and upload Production app to Firebase' - lane :build_and_upload_production_app do - set_app_version - bump_build - builder.build_ad_hoc( - Constants.SCHEME_NAME_PRODUCTION, - Constants.PRODUCT_NAME_PRODUCTION, - Constants.BUNDLE_ID_PRODUCTION - ) - upload_build_to_firebase( - product_name: Constants.PRODUCT_NAME_PRODUCTION, - firebase_app_id: DeliverableConstants.FIREBASE_APP_ID_PRODUCTION, - tester_groups: DeliverableConstants.FIREBASE_TESTER_GROUPS - ) - upload_build_dsym_to_firebase( - product_name: Constants.PRODUCT_NAME_PRODUCTION, - gsp_name: DeliverableConstants.GSP_PRODUCTION - ) - environment_manager.save_build_context_to_ci(version_number: versioning_manager.version_number) - end - - desc 'upload develop build to Firebase app distribution' - private_lane :upload_build_to_firebase do |options| - release_notes = "#{options[:product_name]} #{versioning_manager.version_and_build_number}" - distribution_manager.upload_to_firebase( - product_name: options[:product_name], - firebase_app_id: options[:firebase_app_id], - tester_groups: options[:tester_groups], - notes: release_notes - ) - end - - desc 'upload build dsym to Firebase' - private_lane :upload_build_dsym_to_firebase do |options| - symbol_manager = SymbolManager.new( - fastlane: self, - version: versioning_manager.version_number, - build_number: versioning_manager.build_number, - build_path: Constants.BUILD_PATH, - gsp_directory: DeliverableConstants.GSP_DIRECTORY, - binary_path: DeliverableConstants.BINARY_PATH - ) - - symbol_manager.upload_built_symbol_to_firebase( - product_name: options[:product_name], - gsp_name: options[:gsp_name] - ) - end - - # AppStore - - desc 'Build and upload Production app to App Store' - lane :build_and_upload_appstore_app do - set_app_version - set_connect_api_key if is_ci - if (Environments.BUMP_APP_STORE_BUILD_NUMBER || '') == 'true' - bump_appstore_build(app_identifier: Constants.BUNDLE_ID_PRODUCTION) - else - bump_build - end - builder.build_app_store( - Constants.SCHEME_NAME_PRODUCTION, - Constants.PRODUCT_NAME_PRODUCTION, - Constants.BUNDLE_ID_PRODUCTION - ) - upload_build_to_appstore - if (Environments.SKIP_FIREBASE_DSYM || '') == '' - upload_build_dsym_to_firebase( - product_name: Constants.PRODUCT_NAME_PRODUCTION, - gsp_name: DeliverableConstants.GSP_PRODUCTION - ) - end - environment_manager.save_build_context_to_ci(version_number: versioning_manager.version_number) - end - - desc 'upload develop build to App Store' - private_lane :upload_build_to_appstore do - distribution_manager.upload_to_app_store_connect( - product_name: Constants.PRODUCT_NAME_PRODUCTION, - bundle_identifier: Constants.BUNDLE_ID_PRODUCTION - ) - end - - desc 'set App Store Connect API Key' - private_lane :set_connect_api_key do - app_store_connect_api_key( - key_id: DeliverableConstants.APP_STORE_KEY_ID, - issuer_id: DeliverableConstants.APP_STORE_ISSUER_ID, - key_content: Environments.APPSTORE_CONNECT_API_KEY || '', - is_key_content_base64: true - ) - end - - # Private helper lanes - - desc 'check if any specific version number in build environment' - private_lane :set_app_version do - # Set up env var MANUAL_VERSION if we need to override the version number - if (Environments.MANUAL_VERSION || '') != '' - increment_version_number( - version_number: Environments.MANUAL_VERSION - ) - end - end - - desc 'set build number with number of commits' - private_lane :bump_build do - increment_build_number( - build_number: number_of_commits, - xcodeproj: Constants.PROJECT_PATH - ) - end - - desc 'set build number with App Store latest build' - private_lane :bump_appstore_build do |options| - increment_build_number({ - build_number: latest_testflight_build_number( - app_identifier: options[:app_identifier] - ) + 1 - }) - end - - # Utilities - - desc 'Clean up derived data' - lane :clean_up do - clear_derived_data(derived_data_path: Constants.DERIVED_DATA_PATH) - end - - desc 'Clean up xcov output' - lane :clean_up_xcov do - clear_derived_data(derived_data_path: Constants.XCOV_OUTPUT_DIRECTORY_PATH) - end -end diff --git a/fastlane/Fastfile.swift b/fastlane/Fastfile.swift new file mode 100644 index 00000000..86b58fcd --- /dev/null +++ b/fastlane/Fastfile.swift @@ -0,0 +1,212 @@ +// This file contains the fastlane.tools configuration +// You can find the documentation at https://docs.fastlane.tools +// +// For a list of all available actions, check out +// +// https://docs.fastlane.tools/actions +// + +import Foundation + +class Fastfile: LaneFile { + + // MARK: - Code signing + + func syncDevelopmentCodeSigningLane() { + desc("Sync the Development match signing for the Staging build") + Match.syncCodeSigning( + type: .development, + appIdentifier: [Constant.stagingBundleId] + ) + } + + func syncAdHocStagingCodeSigningLane() { + desc("Sync the Ad Hoc match signing for the Staging build") + Match.syncCodeSigning( + type: .adHoc, + appIdentifier: [Constant.stagingBundleId] + ) + } + + func syncAdHocProductionCodeSigningLane() { + desc("Sync the Ad Hoc match signing for the Production build") + Match.syncCodeSigning( + type: .adHoc, + appIdentifier: [Constant.productionBundleId] + ) + } + + func syncAppStoreCodeSigningLane() { + desc("Sync the App Store match signing for the Production build") + Match.syncCodeSigning( + type: .appStore, + appIdentifier: [Constant.productionBundleId] + ) + } + + // MARK: - Build + + func buildAdHocStagingLane() { + desc("Build ad-hoc staging") + Build.adHoc(environment: .staging) + } + + func buildAdHocProductionLane() { + desc("Build ad-hoc production") + Build.adHoc(environment: .production) + } + + func buildAppStoreLane() { + desc("Build app store") + Build.appStore() + } + + // MARK: - Upload builds to Firebase and AppStore + + func buildStagingAndUploadToFirebaseLane() { + desc("Build Staging app and upload to Firebase") + + setAppVersion() + bumpBuild() + + buildAdHocStagingLane() + + // TODO: - Make release notes + Distribution.uploadToFirebase(environment: .staging, releaseNotes: "") + + Symbol.uploadToCrashlytics(environment: .staging) + + Build.saveBuildContextToCI() + } + + func buildProductionAndUploadToFirebaseLane() { + desc("Build Staging app and upload to Firebase") + + setAppVersion() + bumpBuild() + + buildAdHocProductionLane() + + // TODO: - Make release notes + Distribution.uploadToFirebase(environment: .production, releaseNotes: "") + + Symbol.uploadToCrashlytics(environment: .production) + + Build.saveBuildContextToCI() + } + + func buildAndUploadToAppStoreLane() { + desc("Build Production app and upload to App Store") + + setAppVersion() + AppStoreAuthentication.connectAPIKey() + if Secret.bumpAppStoreBuildNumber { + bumpAppstoreBuild() + } else { + bumpBuild() + } + + buildAppStoreLane() + + Distribution.uploadToAppStore() + + Symbol.uploadToCrashlytics(environment: .production) + + Build.saveBuildContextToCI() + } + + func buildAndUploadToTestFlightLane() { + desc("Build Production app and upload to TestFlight") + + setAppVersion() + bumpBuild() + + buildAppStoreLane() + + AppStoreAuthentication.connectAPIKey() + Distribution.uploadToTestFlight() + + Symbol.uploadToCrashlytics(environment: .production) + + Build.saveBuildContextToCI() + } + + // MARK: - Test + + func buildAndTestLane() { + desc("Build and Test project") + Test.buildAndTest( + environment: .staging, + targets: [Constant.testTarget, Constant.uiTestTarget], + devices: Constant.devices + ) + } + + func updateProvisionSettingsLane() { + desc("Update Provision Profile") + syncAppStoreCodeSigningLane() + updateCodeSigningSettings( + path: Constant.projectPath, + useAutomaticSigning: .userDefined(false), + teamId: .userDefined(EnvironmentParser.string(key: "sigh_\(Constant.productionBundleId)_appstore_team-id")), + codeSignIdentity: .userDefined("iPhone Distribution"), + profileName: .userDefined(EnvironmentParser.string(key: "sigh_\(Constant.productionBundleId)_appstore_profile-name")) + ) + } + + func setUpTestProjectLane() { + desc("Disable Exempt Encryption") + Test.disableExemptEncryption() + } + + // MARK: - Register device + + func registerNewDeviceLane() { + let deviceName = prompt(text: "Enter the device name:") + let deviceUDID = prompt(text: "Enter the device UDID:") + + registerDevice( + name: deviceName, + udid: deviceUDID, + teamId: .userDefined(Constant.teamId) + ) + + Match.syncCodeSigning(type: .development, appIdentifier: [], isForce: true) + Match.syncCodeSigning(type: .adHoc, appIdentifier: [], isForce: true) + } + + // MARK: - Utilities + + func cleanUpOutputLane() { + desc("Clean up Output") + clearDerivedData(derivedDataPath: Constant.outputPath) + } + + // MARK: - Private Helper + + private func setAppVersion() { + desc("Check if any specific version number in build environment") + guard !Constant.manualVersion.isEmpty else { return } + incrementVersionNumber( + versionNumber: .userDefined(Constant.manualVersion) + ) + } + + private func bumpBuild(buildNumber: Int = numberOfCommits()) { + desc("Set build number with number of commits") + incrementBuildNumber( + buildNumber: .userDefined(String(buildNumber)), + xcodeproj: .userDefined(Constant.projectPath) + ) + } + + private func bumpAppstoreBuild() { + desc("Set build number with App Store latest build") + let theLatestBuildNumber = latestTestflightBuildNumber( + appIdentifier: Constant.productionBundleId + ) + 1 + incrementBuildNumber( + buildNumber: .userDefined("\(theLatestBuildNumber)") + ) + } +} diff --git a/fastlane/Gymfile b/fastlane/Gymfile deleted file mode 100644 index aa2d9d85..00000000 --- a/fastlane/Gymfile +++ /dev/null @@ -1,6 +0,0 @@ -# frozen_string_literal: true - -clean(true) -output_directory(Constants.BUILD_PATH) # .ipa -build_path(Constants.BUILD_PATH) # .xcarchive is stored -derived_data_path(Constants.DERIVED_DATA_PATH) # .app diff --git a/fastlane/Helpers/AppStoreAuthentication.swift b/fastlane/Helpers/AppStoreAuthentication.swift new file mode 100644 index 00000000..8eaded06 --- /dev/null +++ b/fastlane/Helpers/AppStoreAuthentication.swift @@ -0,0 +1,21 @@ +// +// AppStoreAuthentication.swift +// FastlaneRunner +// +// Created by Su T. on 28/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +import Foundation + +enum AppStoreAuthentication { + + static func connectAPIKey() { + appStoreConnectApiKey( + keyId: Secret.appStoreKeyIdKey, + issuerId: Secret.appStoreIssuerIdKey, + keyContent: .userDefined(Secret.appstoreConnectAPIKey), + isKeyContentBase64: .userDefined(true) // Check if the AppStore Connect API Key is base64 + ) + } +} diff --git a/fastlane/Helpers/Build.swift b/fastlane/Helpers/Build.swift new file mode 100644 index 00000000..e187f5b9 --- /dev/null +++ b/fastlane/Helpers/Build.swift @@ -0,0 +1,63 @@ +// +// Build.swift +// FastlaneRunner +// +// Created by Su T. on 23/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +enum Build { + + // MARK: Build the application + + static func adHoc(environment: Constant.Environment) { + build(environment: environment, type: .adHoc) + } + + static func appStore() { + build(environment: .production, type: .appStore) + } + + // MARK: Save the build context + + static func saveBuildContextToCI() { + switch Constant.platform { + case .gitHubActions: + let ipaPath = laneContext()["IPA_OUTPUT_PATH"] + let dsymPath = laneContext()["DSYM_OUTPUT_PATH"] + let buildNumber = laneContext()["BUILD_NUMBER"] + + sh(command: "echo IPA_OUTPUT_PATH=\(ipaPath ?? "") >> $GITHUB_ENV") + sh(command: "echo DSYM_OUTPUT_PATH=\(dsymPath ?? "") >> $GITHUB_ENV") + sh(command: "echo BUILD_NUMBER=\(buildNumber ?? "") >> $GITHUB_ENV") + sh(command: "echo VERSION_NUMBER=\(Version.versionNumber) >> $GITHUB_ENV") + case .bitrise: + sh(command: "envman add --key BUILD_PATH --value '\(Constant.outputPath)'") + default: break + } + } + + // MARK: Private + + static private func build( + environment: Constant.Environment, + type: Constant.BuildType + ) { + buildApp( + scheme: .userDefined(environment.scheme), + clean: .userDefined(true), + outputDirectory: Constant.outputPath, + outputName: .userDefined(environment.productName), + includeSymbols: .userDefined(true), + exportMethod: .userDefined(type.value), + exportOptions: .userDefined([ + "provisioningProfiles": [ + environment.bundleId: "match \(type.method) \(environment.bundleId)" + ] + ]), + buildPath: .userDefined(Constant.buildPath), + derivedDataPath: .userDefined(Constant.derivedDataPath), + xcodebuildFormatter: "xcpretty" // Default `xcbeautify` will never work + ) + } +} diff --git a/fastlane/Helpers/Distribution.swift b/fastlane/Helpers/Distribution.swift new file mode 100644 index 00000000..9361dd4c --- /dev/null +++ b/fastlane/Helpers/Distribution.swift @@ -0,0 +1,62 @@ +// +// Distribution.swift +// FastlaneRunner +// +// Created by Su T. on 25/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +import Foundation + +enum Distribution { + + static func uploadToFirebase( + environment: Constant.Environment, + groups: String = Constant.firebaseTesterGroups, + releaseNotes: String + ) { + let ipaPath = makeIPAPath(environment: environment) + firebaseAppDistribution( + ipaPath: .userDefined(ipaPath), + app: .userDefined(environment.firebaseAppId), + groups: .userDefined(groups), + releaseNotes: .userDefined(releaseNotes), + firebaseCliToken: .userDefined(Secret.firebaseCLIToken), + debug: .userDefined(true) + ) + } + + static func uploadToAppStore(environment: Constant.Environment = .production) { + let ipaPath = makeIPAPath(environment: environment) + appstore( + appIdentifier: .userDefined(environment.bundleId), + ipa: .userDefined(ipaPath), + skipScreenshots: .userDefined(true), + skipMetadata: .userDefined(true), + force: .userDefined(true), + runPrecheckBeforeSubmit: .userDefined(false) + ) + } + + static func uploadToTestFlight( + environment: Constant.Environment = .production, + changeLog: String = "", + betaAppReviewInfo: [String: Any] = [:], + groups: [String] = Constant.testFlightTesterGroups + ) { + let ipaPath = makeIPAPath(environment: environment) + testflight( + appIdentifier: .userDefined(environment.bundleId), + ipa: .userDefined(ipaPath), + demoAccountRequired: .userDefined(betaAppReviewInfo.isEmpty), + betaAppReviewInfo: .userDefined(betaAppReviewInfo), + changelog: .userDefined(changeLog), + skipWaitingForBuildProcessing: .userDefined(true), + groups: .userDefined(groups) + ) + } + + private static func makeIPAPath(environment: Constant.Environment) -> String { + "\(Constant.outputPath)/\(environment.productName).ipa" + } +} diff --git a/fastlane/Helpers/EnvironmentParser.swift b/fastlane/Helpers/EnvironmentParser.swift new file mode 100644 index 00000000..cd94154e --- /dev/null +++ b/fastlane/Helpers/EnvironmentParser.swift @@ -0,0 +1,20 @@ +// +// EnvironmentParser.swift +// FastlaneRunner +// +// Created by Su T. on 09/11/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +import Foundation + +enum EnvironmentParser { + + static func bool(key: String) -> Bool { + string(key: key) == "true" + } + + static func string(key: String) -> String { + environmentVariable(get: .userDefined(key)) + } +} diff --git a/fastlane/Helpers/Keychain.swift b/fastlane/Helpers/Keychain.swift new file mode 100644 index 00000000..4482c788 --- /dev/null +++ b/fastlane/Helpers/Keychain.swift @@ -0,0 +1,20 @@ +// +// Keychain.swift +// FastlaneRunner +// +// Created by Su Ho on 22/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +enum Keychain { + + static func create() { + createKeychain( + name: .userDefined(Constant.keychainName), + password: Secret.keychainPassword, + defaultKeychain: .userDefined(true), + unlock: .userDefined(true), + timeout: 3600 + ) + } +} diff --git a/fastlane/Helpers/Match.swift b/fastlane/Helpers/Match.swift new file mode 100644 index 00000000..f19aab1e --- /dev/null +++ b/fastlane/Helpers/Match.swift @@ -0,0 +1,49 @@ +// +// Match.swift +// FastlaneRunner +// +// Created by Su Ho on 22/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +enum Match { + + static func syncCodeSigning(type: MatchType, appIdentifier: [String], isForce: Bool = false) { + if isCi() { + Keychain.create() + match( + type: type.value, + readonly: .userDefined(!isForce), + appIdentifier: appIdentifier, + username: .userDefined(Constant.userName), + teamId: .userDefined(Constant.teamId), + gitUrl: Constant.matchURL, + keychainName: Constant.keychainName, + keychainPassword: .userDefined(Secret.keychainPassword), + force: .userDefined(isForce) + ) + } else { + match( + type: type.value, + readonly: .userDefined(!isForce), + appIdentifier: appIdentifier, + username: .userDefined(Constant.userName), + teamId: .userDefined(Constant.teamId), + gitUrl: Constant.matchURL, + force: .userDefined(isForce) + ) + } + } +} + +extension Match { + + enum MatchType: String { + + case development + case adHoc = "adhoc" + case appStore = "appstore" + + var value: String { return rawValue } + } +} diff --git a/fastlane/Helpers/Symbol.swift b/fastlane/Helpers/Symbol.swift new file mode 100644 index 00000000..8d0e318c --- /dev/null +++ b/fastlane/Helpers/Symbol.swift @@ -0,0 +1,25 @@ +// +// Symbol.swift +// FastlaneRunner +// +// Created by Su Ho on 27/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +import Foundation + +enum Symbol { + + static func uploadToCrashlytics(environment: Constant.Environment) { + guard FileManager.default.fileExists(atPath: environment.dsymPath) else { + return log(message: "Can't find the dSYM file") + } + uploadSymbolsToCrashlytics( + dsymPath: environment.dsymPath, + gspPath: .userDefined(environment.gspPath), + appId: .userDefined(environment.firebaseAppId), + binaryPath: .userDefined(Constant.uploadSymbolsBinaryPath), + debug: true // We sometimes has issues with dSYM files, so I enabled this flag. + ) + } +} diff --git a/fastlane/Helpers/Test.swift b/fastlane/Helpers/Test.swift new file mode 100644 index 00000000..46e91908 --- /dev/null +++ b/fastlane/Helpers/Test.swift @@ -0,0 +1,35 @@ +// +// Test.swift +// FastlaneRunner +// +// Created by Khanh on 27/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +enum Test { + + static func buildAndTest( + environment: Constant.Environment, + targets: [String], + devices: [String] + ) { + scan( + scheme: .userDefined(environment.scheme), + devices: .userDefined(devices), + onlyTesting: targets, + codeCoverage: .userDefined(true), + outputDirectory: Constant.testOutputDirectoryPath, + xcodebuildFormatter: "xcpretty", + resultBundle: .userDefined(true), + failBuild: .userDefined(false) + ) + } + + static func disableExemptEncryption() { + setInfoPlistValue( + key: "ITSAppUsesNonExemptEncryption", + value: "false", + path: "\(Constant.projectName)/Configurations/Plists/Info.plist" + ) + } +} diff --git a/fastlane/Helpers/Version.swift b/fastlane/Helpers/Version.swift new file mode 100644 index 00000000..b52def0f --- /dev/null +++ b/fastlane/Helpers/Version.swift @@ -0,0 +1,49 @@ +// +// Version.swift +// FastlaneRunner +// +// Created by Khanh on 26/09/2022. +// Copyright © 2022 Nimble. All rights reserved. +// + +enum Version { + + // MARK: - Getting + + static var versionNumber: String { + get { + FastlaneRunner.getVersionNumber( + xcodeproj: .userDefined(Constant.projectPath), + target: .userDefined(Constant.projectName) + ) + } + + set { + incrementVersionNumber( + versionNumber: .userDefined(newValue), + xcodeproj: .userDefined(Constant.projectPath) + ) + } + } + + static var buildNumber: String { + get { + FastlaneRunner.getBuildNumber(xcodeproj: .userDefined(Constant.projectPath)) + } + + set { + incrementBuildNumber( + buildNumber: .userDefined(newValue), + xcodeproj: .userDefined(Constant.projectPath) + ) + } + } + + static func getVersionAndBuildNumber() -> String { + "\(versionNumber) (Build: \(buildNumber)" + } + + static var releaseTag: String { + "release/\(versionNumber)" + } +} diff --git a/fastlane/Managers/BuildManager.rb b/fastlane/Managers/BuildManager.rb deleted file mode 100644 index 9654abe4..00000000 --- a/fastlane/Managers/BuildManager.rb +++ /dev/null @@ -1,34 +0,0 @@ -# frozen_string_literal: true - -class BuildManager - def initialize(fastlane:) - @fastlane = fastlane - end - - def build_ad_hoc(scheme, product_name, bundle_identifier) - @fastlane.gym( - scheme: scheme, - export_method: 'ad-hoc', - export_options: { - provisioningProfiles: { - bundle_identifier => "match AdHoc #{bundle_identifier}" - } - }, - output_name: product_name, - disable_xcpretty: true - ) - end - - def build_app_store(scheme, product_name, bundle_identifier) - @fastlane.gym( - scheme: scheme, - export_method: 'app-store', - export_options: { - provisioningProfiles: { - bundle_identifier => "match AppStore #{bundle_identifier}" - } - }, - output_name: product_name - ) - end -end diff --git a/fastlane/Managers/DistributionManager.rb b/fastlane/Managers/DistributionManager.rb deleted file mode 100644 index fdedc74a..00000000 --- a/fastlane/Managers/DistributionManager.rb +++ /dev/null @@ -1,42 +0,0 @@ -# frozen_string_literal: true - -class DistributionManager - def initialize(fastlane:, build_path:, firebase_token:) - @fastlane = fastlane - @build_path = build_path - @firebase_token = firebase_token - end - - def upload_to_firebase(product_name:, firebase_app_id:, notes:, tester_groups:) - ipa_path = "#{@build_path}/#{product_name}.ipa" - @fastlane.firebase_app_distribution( - app: firebase_app_id, - ipa_path: ipa_path, - groups: tester_groups, - firebase_cli_token: @firebase_token, - release_notes: notes - ) - end - - def upload_to_app_store_connect(product_name:, bundle_identifier:) - @fastlane.deliver( - ipa: "#{@build_path}/#{product_name}.ipa", - app_identifier: bundle_identifier, - force: true, - skip_metadata: true, - skip_screenshots: true, - run_precheck_before_submit: false - ) - end - - def upload_to_testflight(product_name:, bundle_identifier:, notes:, tester_groups:, app_review_info:) - @fastlane.pilot( - ipa: "#{@build_path}/#{product_name}.ipa", - app_identifier: bundle_identifier, - groups: tester_groups, - changelog: notes, - beta_app_review_info: app_review_info, - demo_account_required: !app_review_info.nil? - ) - end -end diff --git a/fastlane/Managers/MatchManager.rb b/fastlane/Managers/MatchManager.rb deleted file mode 100644 index a7cafd67..00000000 --- a/fastlane/Managers/MatchManager.rb +++ /dev/null @@ -1,92 +0,0 @@ -# frozen_string_literal: true - -class MatchManager - def initialize( - fastlane:, - keychain_name:, - keychain_password:, - is_ci:, - username: - ) - @fastlane = fastlane - @keychain_name = keychain_name - @keychain_password = keychain_password - @is_ci = is_ci - @username = username - end - - def sync_development_signing(app_identifier:, force: false) - if @is_ci - create_ci_keychain - @fastlane.match( - type: 'development', - keychain_name: @keychain_name, - keychain_password: @keychain_password, - app_identifier: app_identifier, - readonly: !force, - force: force - ) - else - @fastlane.match( - type: 'development', - app_identifier: app_identifier, - readonly: !force, - force: force, - username: @username - ) - end - end - - def sync_adhoc_signing(app_identifier:, force: false) - if @is_ci - create_ci_keychain - @fastlane.match( - type: 'adhoc', - keychain_name: @keychain_name, - keychain_password: @keychain_password, - app_identifier: app_identifier, - readonly: !force, - force: force - ) - else - @fastlane.match( - type: 'adhoc', - app_identifier: app_identifier, - readonly: !force, - force: force, - username: @username - ) - end - end - - def sync_app_store_signing(app_identifier:) - if @is_ci - create_ci_keychain - @fastlane.match( - type: 'appstore', - keychain_name: @keychain_name, - keychain_password: @keychain_password, - app_identifier: app_identifier, - readonly: true - ) - else - @fastlane.match( - type: 'appstore', - app_identifier: app_identifier, - readonly: true, - username: @username - ) - end - end - - def create_ci_keychain - @fastlane.create_keychain( - name: @keychain_name, - password: @keychain_password, - default_keychain: true, - unlock: true, - timeout: 3600, - lock_when_sleeps: false - ) - end -end diff --git a/fastlane/Managers/SymbolManager.rb b/fastlane/Managers/SymbolManager.rb deleted file mode 100644 index bf96c5a0..00000000 --- a/fastlane/Managers/SymbolManager.rb +++ /dev/null @@ -1,32 +0,0 @@ -# frozen_string_literal: true - -class SymbolManager - def initialize(fastlane:, version:, build_number:, build_path:, gsp_directory:, binary_path:) - @fastlane = fastlane - @version = version - @build_number = build_number - @build_path = build_path - @gsp_directory = gsp_directory - @binary_path = binary_path - end - - def download_processed_dsym_then_upload_to_firebase(bundle_identifier:, gsp_name:) - @fastlane.download_dsyms( - app_identifier: bundle_identifier, - version: @version, - build_number: @build_number, - wait_for_dsym_processing: true - ) - @fastlane.upload_symbols_to_crashlytics( - gsp_path: "#{@gsp_directory}/#{gsp_name}" - ) - end - - def upload_built_symbol_to_firebase(product_name:, gsp_name:) - @fastlane.upload_symbols_to_crashlytics( - dsym_path: "#{@build_path}/#{product_name}.app.dSYM.zip", - gsp_path: "#{@gsp_directory}/#{gsp_name}", - binary_path: "#{@binary_path}" - ) - end -end diff --git a/fastlane/Managers/TestManager.rb b/fastlane/Managers/TestManager.rb deleted file mode 100644 index 229d4138..00000000 --- a/fastlane/Managers/TestManager.rb +++ /dev/null @@ -1,21 +0,0 @@ -# frozen_string_literal: true - -class TestManager - def initialize(fastlane:, device:, output_directory:) - @fastlane = fastlane - @device = device - @output_directory = output_directory - end - - def build_and_test(scheme:, targets:) - @fastlane.scan( - scheme: scheme, - device: @device, - output_directory: @output_directory, - code_coverage: true, - result_bundle: true, - only_testing: targets, - fail_build: false - ) - end -end diff --git a/fastlane/Managers/VersioningManager.rb b/fastlane/Managers/VersioningManager.rb deleted file mode 100644 index 9b4124cd..00000000 --- a/fastlane/Managers/VersioningManager.rb +++ /dev/null @@ -1,46 +0,0 @@ -# frozen_string_literal: true - -class VersioningManager - def initialize( - fastlane:, - project_path:, - main_target_name: - ) - @fastlane = fastlane - @project_path = project_path - @main_target_name = main_target_name - end - - def build_number=(build_number) - @fastlane.increment_build_number( - build_number: build_number, - xcodeproj: @project_path - ) - end - - def build_number - @fastlane.get_build_number(xcodeproj: @project_path) - end - - def version_number - @fastlane.get_version_number( - xcodeproj: @project_path, - target: @main_target_name - ) - end - - def version_number=(version_number) - @fastlane.increment_version_number( - version_number: version_number, - xcodeproj: @project_path - ) - end - - def version_and_build_number - "#{version_number} (Build: #{build_number})" - end - - def release_tag - "release/#{version_number}/#{@fastlane.number_of_commits}" - end -end diff --git a/fastlane/Matchfile b/fastlane/Matchfile deleted file mode 100644 index 399f7e45..00000000 --- a/fastlane/Matchfile +++ /dev/null @@ -1,3 +0,0 @@ -git_url("git@github.com:{organization}/{repo}.git") - -storage_mode("git") diff --git a/fastlane/Tests/Fastfile b/fastlane/Tests/Fastfile deleted file mode 100644 index e5cd14b3..00000000 --- a/fastlane/Tests/Fastfile +++ /dev/null @@ -1,41 +0,0 @@ -# frozen_string_literal: true - -require './Constants/Constants' -require './Constants/Environments' -require './Managers/MatchManager' - -match_manager = MatchManager.new( - fastlane: self, - keychain_name: Constants.KEYCHAIN_NAME, - keychain_password: Constants.KEYCHAIN_PASSWORD, - is_ci: Environments.CI, - username: '' -) - -private_lane :template_app_icon do - appicon( - appicon_image_file: 'fastlane/Tests/appicon.png', - appicon_devices: %i[iphone ios_marketing], - appicon_path: '{PROJECT_NAME}/Resources/Assets/Assets.xcassets' - ) -end - -private_lane :disable_exempt_encryption do - set_info_plist_value(path: "{PROJECT_NAME}/Configurations/Plists/Info.plist", key: "ITSAppUsesNonExemptEncryption", value: false) -end - -lane :set_up_test_project do - template_app_icon - disable_exempt_encryption -end - -desc 'Sync AppStore match signing' -lane :update_provision_settings do - match_manager.sync_app_store_signing(app_identifier: [Constants.BUNDLE_ID_PRODUCTION]) - update_code_signing_settings( - use_automatic_signing: false, - team_id: ENV["sigh_#{Constants.BUNDLE_ID_PRODUCTION}_appstore_team-id"], - profile_name: ENV["sigh_#{Constants.BUNDLE_ID_PRODUCTION}_appstore_profile-name"], - code_sign_identity: 'iPhone Distribution' - ) -end diff --git a/fastlane/Tests/appicon.png b/fastlane/Tests/appicon.png deleted file mode 100644 index 649cfec8..00000000 Binary files a/fastlane/Tests/appicon.png and /dev/null differ diff --git a/fastlane/swift/Actions.swift b/fastlane/swift/Actions.swift new file mode 100644 index 00000000..521d0e4d --- /dev/null +++ b/fastlane/swift/Actions.swift @@ -0,0 +1 @@ +import Foundation \ No newline at end of file diff --git a/fastlane/swift/Appfile.swift b/fastlane/swift/Appfile.swift new file mode 100644 index 00000000..633b1b93 --- /dev/null +++ b/fastlane/swift/Appfile.swift @@ -0,0 +1,15 @@ +// Appfile.swift +// Copyright (c) 2022 FastlaneTools + +var appIdentifier: String { return "" } // The bundle identifier of your app +var appleID: String { return "" } // Your Apple Developer Portal username + +var teamID: String { return "" } // Developer Portal Team ID +var itcTeam: String? { return nil } // App Store Connect Team ID (may be nil if no team) + +// you can even provide different app identifiers, Apple IDs and team names per lane: +// More information: https://docs.fastlane.tools/advanced/#appfile + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.1] diff --git a/fastlane/swift/ArgumentProcessor.swift b/fastlane/swift/ArgumentProcessor.swift new file mode 100644 index 00000000..8ec1cff4 --- /dev/null +++ b/fastlane/swift/ArgumentProcessor.swift @@ -0,0 +1,89 @@ +// ArgumentProcessor.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +struct ArgumentProcessor { + let args: [RunnerArgument] + let currentLane: String + let commandTimeout: Int + let port: UInt32 + + init(args: [String]) { + // Dump the first arg which is the program name + let fastlaneArgs = stride(from: 1, to: args.count - 1, by: 2).map { + RunnerArgument(name: args[$0], value: args[$0 + 1]) + } + self.args = fastlaneArgs + + let fastlaneArgsMinusLanes = fastlaneArgs.filter { arg in + arg.name.lowercased() != "lane" + } + + let potentialLogMode = fastlaneArgsMinusLanes.filter { arg in + arg.name.lowercased() == "logmode" + } + + port = UInt32(fastlaneArgsMinusLanes.first(where: { $0.name == "swiftServerPort" })?.value ?? "") ?? 2000 + + // Configure logMode since we might need to use it before we finish parsing + if let logModeArg = potentialLogMode.first { + let logModeString = logModeArg.value + Logger.logMode = Logger.LogMode(logMode: logModeString) + } + + let lanes = self.args.filter { arg in + arg.name.lowercased() == "lane" + } + verbose(message: lanes.description) + + guard lanes.count == 1 else { + let message = "You must have exactly one lane specified as an arg, here's what I got: \(lanes)" + log(message: message) + fatalError(message) + } + + let lane = lanes.first! + currentLane = lane.value + + // User might have configured a timeout for the socket connection + let potentialTimeout = fastlaneArgsMinusLanes.filter { arg in + arg.name.lowercased() == "timeoutseconds" + } + + if let logModeArg = potentialLogMode.first { + let logModeString = logModeArg.value + Logger.logMode = Logger.LogMode(logMode: logModeString) + } + + if let timeoutArg = potentialTimeout.first { + let timeoutString = timeoutArg.value + commandTimeout = (timeoutString as NSString).integerValue + } else { + commandTimeout = SocketClient.defaultCommandTimeoutSeconds + } + } + + func laneParameters() -> [String: String] { + let laneParametersArgs = args.filter { arg in + let lowercasedName = arg.name.lowercased() + return lowercasedName != "timeoutseconds" && lowercasedName != "lane" && lowercasedName != "logmode" + } + var laneParameters = [String: String]() + for arg in laneParametersArgs { + laneParameters[arg.name] = arg.value + } + return laneParameters + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/Atomic.swift b/fastlane/swift/Atomic.swift new file mode 100644 index 00000000..45d11357 --- /dev/null +++ b/fastlane/swift/Atomic.swift @@ -0,0 +1,150 @@ +// Atomic.swift +// Copyright (c) 2022 FastlaneTools + +import Foundation + +protocol DictionaryProtocol: class { + associatedtype Key: Hashable + associatedtype Value + + subscript(_: Key) -> Value? { get set } + + @discardableResult + func removeValue(forKey key: Key) -> Value? + + func get(_ key: Key) -> Value? + func set(_ key: Key, value: Value?) +} + +extension DictionaryProtocol { + subscript(_ key: Key) -> Value? { + get { + get(key) + } + set { + set(key, value: newValue) + } + } +} + +protocol LockProtocol: DictionaryProtocol { + associatedtype Lock + + var _lock: Lock { get set } + + func lock() + func unlock() +} + +protocol AnyLock {} + +extension UnsafeMutablePointer: AnyLock { + @available(macOS, deprecated: 10.12) + static func make() -> Self where Pointee == OSSpinLock { + let spin = UnsafeMutablePointer.allocate(capacity: 1) + spin.initialize(to: OS_SPINLOCK_INIT) + return spin + } + + @available(macOS, introduced: 10.12) + static func make() -> Self where Pointee == os_unfair_lock { + let unfairLock = UnsafeMutablePointer.allocate(capacity: 1) + unfairLock.initialize(to: os_unfair_lock()) + return unfairLock + } + + @available(macOS, deprecated: 10.12) + static func lock(_ lock: Self) where Pointee == OSSpinLock { + OSSpinLockLock(lock) + } + + @available(macOS, deprecated: 10.12) + static func unlock(_ lock: Self) where Pointee == OSSpinLock { + OSSpinLockUnlock(lock) + } + + @available(macOS, introduced: 10.12) + static func lock(_ lock: Self) where Pointee == os_unfair_lock { + os_unfair_lock_lock(lock) + } + + @available(macOS, introduced: 10.12) + static func unlock(_ lock: Self) where Pointee == os_unfair_lock { + os_unfair_lock_unlock(lock) + } +} + +// MARK: - Classes + +class AtomicDictionary: LockProtocol { + typealias Lock = AnyLock + + var _lock: Lock + + private var storage: [Key: Value] = [:] + + init(_ lock: Lock) { + _lock = lock + } + + @discardableResult + func removeValue(forKey key: Key) -> Value? { + lock() + defer { unlock() } + return storage.removeValue(forKey: key) + } + + func get(_ key: Key) -> Value? { + lock() + defer { unlock() } + return storage[key] + } + + func set(_ key: Key, value: Value?) { + lock() + defer { unlock() } + storage[key] = value + } + + func lock() { + fatalError() + } + + func unlock() { + fatalError() + } +} + +@available(macOS, introduced: 10.12) +final class UnfairAtomicDictionary: AtomicDictionary { + typealias Lock = UnsafeMutablePointer + + init() { + super.init(Lock.make()) + } + + override func lock() { + Lock.lock(_lock as! Lock) + } + + override func unlock() { + Lock.unlock(_lock as! Lock) + } +} + +@available(macOS, deprecated: 10.12) +final class OSSPinAtomicDictionary: AtomicDictionary { + typealias Lock = UnsafeMutablePointer + + init() { + super.init(Lock.make()) + } + + override func lock() { + Lock.lock(_lock as! Lock) + } + + override func unlock() { + Lock.unlock(_lock as! Lock) + } +} diff --git a/fastlane/swift/ControlCommand.swift b/fastlane/swift/ControlCommand.swift new file mode 100644 index 00000000..65706eb2 --- /dev/null +++ b/fastlane/swift/ControlCommand.swift @@ -0,0 +1,74 @@ +// ControlCommand.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +struct ControlCommand: RubyCommandable { + static let commandKey = "command" + var type: CommandType { return .control } + + enum ShutdownCommandType { + static let userMessageKey: String = "userMessage" + + enum CancelReason { + static let reasonKey: String = "reason" + case clientError + case serverError + + var reasonText: String { + switch self { + case .clientError: + return "clientError" + case .serverError: + return "serverError" + } + } + } + + case done + case cancel(cancelReason: CancelReason) + + var token: String { + switch self { + case .done: + return "done" + case .cancel: + return "cancelFastlaneRun" + } + } + } + + let message: String? + let id: String = UUID().uuidString + let shutdownCommandType: ShutdownCommandType + var commandJson: String { + var jsonDictionary: [String: Any] = [ControlCommand.commandKey: shutdownCommandType.token] + + if let message = message { + jsonDictionary[ShutdownCommandType.userMessageKey] = message + } + if case let .cancel(reason) = shutdownCommandType { + jsonDictionary[ShutdownCommandType.CancelReason.reasonKey] = reason.reasonText + } + + let jsonData = try! JSONSerialization.data(withJSONObject: jsonDictionary, options: []) + let jsonString = String(data: jsonData, encoding: .utf8)! + return jsonString + } + + init(commandType: ShutdownCommandType, message: String? = nil) { + shutdownCommandType = commandType + self.message = message + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/Deliverfile.swift b/fastlane/swift/Deliverfile.swift new file mode 100644 index 00000000..241ef61b --- /dev/null +++ b/fastlane/swift/Deliverfile.swift @@ -0,0 +1,20 @@ +// Deliverfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `deliver` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Deliverfile: DeliverfileProtocol { + // If you want to enable `deliver`, run `fastlane deliver init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/DeliverfileProtocol.swift b/fastlane/swift/DeliverfileProtocol.swift new file mode 100644 index 00000000..02c7aebf --- /dev/null +++ b/fastlane/swift/DeliverfileProtocol.swift @@ -0,0 +1,267 @@ +// DeliverfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol DeliverfileProtocol: AnyObject { + /// Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + var apiKeyPath: String? { get } + + /// Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + var apiKey: [String: Any]? { get } + + /// Your Apple ID Username + var username: String? { get } + + /// The bundle identifier of your app + var appIdentifier: String? { get } + + /// The version that should be edited or created + var appVersion: String? { get } + + /// Path to your ipa file + var ipa: String? { get } + + /// Path to your pkg file + var pkg: String? { get } + + /// If set the given build number (already uploaded to iTC) will be used instead of the current built one + var buildNumber: String? { get } + + /// The platform to use (optional) + var platform: String { get } + + /// Modify live metadata, this option disables ipa upload and screenshot upload + var editLive: Bool { get } + + /// Force usage of live version rather than edit version + var useLiveVersion: Bool { get } + + /// Path to the folder containing the metadata files + var metadataPath: String? { get } + + /// Path to the folder containing the screenshots + var screenshotsPath: String? { get } + + /// Skip uploading an ipa or pkg to App Store Connect + var skipBinaryUpload: Bool { get } + + /// Don't upload the screenshots + var skipScreenshots: Bool { get } + + /// Don't upload the metadata (e.g. title, description). This will still upload screenshots + var skipMetadata: Bool { get } + + /// Don’t create or update the app version that is being prepared for submission + var skipAppVersionUpdate: Bool { get } + + /// Skip verification of HTML preview file + var force: Bool { get } + + /// Clear all previously uploaded screenshots before uploading the new ones + var overwriteScreenshots: Bool { get } + + /// Sync screenshots with local ones. This is currently beta option so set true to 'FASTLANE_ENABLE_BETA_DELIVER_SYNC_SCREENSHOTS' environment variable as well + var syncScreenshots: Bool { get } + + /// Submit the new version for Review after uploading everything + var submitForReview: Bool { get } + + /// Verifies archive with App Store Connect without uploading + var verifyOnly: Bool { get } + + /// Rejects the previously submitted build if it's in a state where it's possible + var rejectIfPossible: Bool { get } + + /// Should the app be automatically released once it's approved? (Can not be used together with `auto_release_date`) + var automaticRelease: Bool? { get } + + /// Date in milliseconds for automatically releasing on pending approval (Can not be used together with `automatic_release`) + var autoReleaseDate: Int? { get } + + /// Enable the phased release feature of iTC + var phasedRelease: Bool { get } + + /// Reset the summary rating when you release a new version of the application + var resetRatings: Bool { get } + + /// The price tier of this application + var priceTier: Int? { get } + + /// Path to the app rating's config + var appRatingConfigPath: String? { get } + + /// Extra information for the submission (e.g. compliance specifications, IDFA settings) + var submissionInformation: [String: Any]? { get } + + /// The ID of your App Store Connect team if you're in multiple teams + var teamId: String? { get } + + /// The name of your App Store Connect team if you're in multiple teams + var teamName: String? { get } + + /// The short ID of your Developer Portal team, if you're in multiple teams. Different from your iTC team ID! + var devPortalTeamId: String? { get } + + /// The name of your Developer Portal team if you're in multiple teams + var devPortalTeamName: String? { get } + + /// The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + var itcProvider: String? { get } + + /// Run precheck before submitting to app review + var runPrecheckBeforeSubmit: Bool { get } + + /// The default precheck rule level unless otherwise configured + var precheckDefaultRuleLevel: String { get } + + /// **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - An array of localized metadata items to upload individually by language so that errors can be identified. E.g. ['name', 'keywords', 'description']. Note: slow + var individualMetadataItems: [String]? { get } + + /// **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the app icon + var appIcon: String? { get } + + /// **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the Apple Watch app icon + var appleWatchAppIcon: String? { get } + + /// Metadata: The copyright notice + var copyright: String? { get } + + /// Metadata: The english name of the primary category (e.g. `Business`, `Books`) + var primaryCategory: String? { get } + + /// Metadata: The english name of the secondary category (e.g. `Business`, `Books`) + var secondaryCategory: String? { get } + + /// Metadata: The english name of the primary first sub category (e.g. `Educational`, `Puzzle`) + var primaryFirstSubCategory: String? { get } + + /// Metadata: The english name of the primary second sub category (e.g. `Educational`, `Puzzle`) + var primarySecondSubCategory: String? { get } + + /// Metadata: The english name of the secondary first sub category (e.g. `Educational`, `Puzzle`) + var secondaryFirstSubCategory: String? { get } + + /// Metadata: The english name of the secondary second sub category (e.g. `Educational`, `Puzzle`) + var secondarySecondSubCategory: String? { get } + + /// **DEPRECATED!** This is no longer used by App Store Connect - Metadata: A hash containing the trade representative contact information + var tradeRepresentativeContactInformation: [String: Any]? { get } + + /// Metadata: A hash containing the review information + var appReviewInformation: [String: Any]? { get } + + /// Metadata: Path to the app review attachment file + var appReviewAttachmentFile: String? { get } + + /// Metadata: The localised app description + var description: [String: Any]? { get } + + /// Metadata: The localised app name + var name: [String: Any]? { get } + + /// Metadata: The localised app subtitle + var subtitle: [String: Any]? { get } + + /// Metadata: An array of localised keywords + var keywords: [String: Any]? { get } + + /// Metadata: An array of localised promotional texts + var promotionalText: [String: Any]? { get } + + /// Metadata: Localised release notes for this version + var releaseNotes: [String: Any]? { get } + + /// Metadata: Localised privacy url + var privacyUrl: [String: Any]? { get } + + /// Metadata: Localised Apple TV privacy policy text + var appleTvPrivacyPolicy: [String: Any]? { get } + + /// Metadata: Localised support url + var supportUrl: [String: Any]? { get } + + /// Metadata: Localised marketing url + var marketingUrl: [String: Any]? { get } + + /// Metadata: List of languages to activate + var languages: [String]? { get } + + /// Ignore errors when invalid languages are found in metadata and screenshot directories + var ignoreLanguageDirectoryValidation: Bool { get } + + /// Should precheck check in-app purchases? + var precheckIncludeInAppPurchases: Bool { get } + + /// The (spaceship) app ID of the app you want to use/modify + var app: Int? { get } +} + +public extension DeliverfileProtocol { + var apiKeyPath: String? { return nil } + var apiKey: [String: Any]? { return nil } + var username: String? { return nil } + var appIdentifier: String? { return nil } + var appVersion: String? { return nil } + var ipa: String? { return nil } + var pkg: String? { return nil } + var buildNumber: String? { return nil } + var platform: String { return "ios" } + var editLive: Bool { return false } + var useLiveVersion: Bool { return false } + var metadataPath: String? { return nil } + var screenshotsPath: String? { return nil } + var skipBinaryUpload: Bool { return false } + var skipScreenshots: Bool { return false } + var skipMetadata: Bool { return false } + var skipAppVersionUpdate: Bool { return false } + var force: Bool { return false } + var overwriteScreenshots: Bool { return false } + var syncScreenshots: Bool { return false } + var submitForReview: Bool { return false } + var verifyOnly: Bool { return false } + var rejectIfPossible: Bool { return false } + var automaticRelease: Bool? { return nil } + var autoReleaseDate: Int? { return nil } + var phasedRelease: Bool { return false } + var resetRatings: Bool { return false } + var priceTier: Int? { return nil } + var appRatingConfigPath: String? { return nil } + var submissionInformation: [String: Any]? { return nil } + var teamId: String? { return nil } + var teamName: String? { return nil } + var devPortalTeamId: String? { return nil } + var devPortalTeamName: String? { return nil } + var itcProvider: String? { return nil } + var runPrecheckBeforeSubmit: Bool { return true } + var precheckDefaultRuleLevel: String { return "warn" } + var individualMetadataItems: [String]? { return nil } + var appIcon: String? { return nil } + var appleWatchAppIcon: String? { return nil } + var copyright: String? { return nil } + var primaryCategory: String? { return nil } + var secondaryCategory: String? { return nil } + var primaryFirstSubCategory: String? { return nil } + var primarySecondSubCategory: String? { return nil } + var secondaryFirstSubCategory: String? { return nil } + var secondarySecondSubCategory: String? { return nil } + var tradeRepresentativeContactInformation: [String: Any]? { return nil } + var appReviewInformation: [String: Any]? { return nil } + var appReviewAttachmentFile: String? { return nil } + var description: [String: Any]? { return nil } + var name: [String: Any]? { return nil } + var subtitle: [String: Any]? { return nil } + var keywords: [String: Any]? { return nil } + var promotionalText: [String: Any]? { return nil } + var releaseNotes: [String: Any]? { return nil } + var privacyUrl: [String: Any]? { return nil } + var appleTvPrivacyPolicy: [String: Any]? { return nil } + var supportUrl: [String: Any]? { return nil } + var marketingUrl: [String: Any]? { return nil } + var languages: [String]? { return nil } + var ignoreLanguageDirectoryValidation: Bool { return false } + var precheckIncludeInAppPurchases: Bool { return true } + var app: Int? { return nil } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.112] diff --git a/fastlane/swift/Fastfile.swift b/fastlane/swift/Fastfile.swift new file mode 100644 index 00000000..7393b6a8 --- /dev/null +++ b/fastlane/swift/Fastfile.swift @@ -0,0 +1,16 @@ +// This class is automatically included in FastlaneRunner during build +// If you have a custom Fastfile.swift, this file will be replaced by it +// Don't modify this file unless you are familiar with how fastlane's swift code generation works +// *** This file will be overwritten or replaced during build time *** + +import Foundation + +open class Fastfile: LaneFile { + override public init() { + super.init() + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.1] diff --git a/fastlane/swift/Fastlane.swift b/fastlane/swift/Fastlane.swift new file mode 100644 index 00000000..61a75c79 --- /dev/null +++ b/fastlane/swift/Fastlane.swift @@ -0,0 +1,13552 @@ +// Fastlane.swift +// Copyright (c) 2022 FastlaneTools + +import Foundation +/** + Run ADB Actions + + - parameters: + - serial: Android serial of the device to use for this command + - command: All commands you want to pass to the adb command, e.g. `kill-server` + - adbPath: The path to your `adb` binary (can be left blank if the ANDROID_SDK_ROOT, ANDROID_HOME or ANDROID_SDK environment variable is set) + + - returns: The output of the adb command + + see adb --help for more details + */ +@discardableResult public func adb(serial: String = "", + command: OptionalConfigValue = .fastlaneDefault(nil), + adbPath: String = "adb") -> String +{ + let serialArg = RubyCommand.Argument(name: "serial", value: serial, type: nil) + let commandArg = command.asRubyArgument(name: "command", type: nil) + let adbPathArg = RubyCommand.Argument(name: "adb_path", value: adbPath, type: nil) + let array: [RubyCommand.Argument?] = [serialArg, + commandArg, + adbPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "adb", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Get an array of Connected android device serials + + - parameter adbPath: The path to your `adb` binary (can be left blank if the ANDROID_SDK_ROOT environment variable is set) + + - returns: Returns an array of all currently connected android devices. Example: [] + + Fetches device list via adb, e.g. run an adb command on all connected devices. + */ +public func adbDevices(adbPath: String = "adb") { + let adbPathArg = RubyCommand.Argument(name: "adb_path", value: adbPath, type: nil) + let array: [RubyCommand.Argument?] = [adbPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "adb_devices", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Modify the default list of supported platforms + + - parameter platforms: The optional extra platforms to support + */ +public func addExtraPlatforms(platforms: [String] = []) { + let platformsArg = RubyCommand.Argument(name: "platforms", value: platforms, type: nil) + let array: [RubyCommand.Argument?] = [platformsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "add_extra_platforms", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will add an annotated git tag to the current branch + + - parameters: + - tag: Define your own tag text. This will replace all other parameters + - grouping: Is used to keep your tags organised under one 'folder' + - includesLane: Whether the current lane should be included in the tag and message composition, e.g. '//' + - prefix: Anything you want to put in front of the version number (e.g. 'v') + - postfix: Anything you want to put at the end of the version number (e.g. '-RC1') + - buildNumber: The build number. Defaults to the result of increment_build_number if you're using it + - message: The tag message. Defaults to the tag's name + - commit: The commit or object where the tag will be set. Defaults to the current HEAD + - force: Force adding the tag + - sign: Make a GPG-signed tag, using the default e-mail address's key + + This will automatically tag your build with the following format: `//`, where:| + | + >- `grouping` is just to keep your tags organised under one 'folder', defaults to 'builds'| + - `lane` is the name of the current fastlane lane, if chosen to be included via 'includes_lane' option, which defaults to 'true'| + - `prefix` is anything you want to stick in front of the version number, e.g. 'v'| + - `postfix` is anything you want to stick at the end of the version number, e.g. '-RC1'| + - `build_number` is the build number, which defaults to the value emitted by the `increment_build_number` action| + >| + For example, for build 1234 in the 'appstore' lane, it will tag the commit with `builds/appstore/1234`. + */ +public func addGitTag(tag: OptionalConfigValue = .fastlaneDefault(nil), + grouping: String = "builds", + includesLane: OptionalConfigValue = .fastlaneDefault(true), + prefix: String = "", + postfix: String = "", + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + message: OptionalConfigValue = .fastlaneDefault(nil), + commit: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + sign: OptionalConfigValue = .fastlaneDefault(false)) +{ + let tagArg = tag.asRubyArgument(name: "tag", type: nil) + let groupingArg = RubyCommand.Argument(name: "grouping", value: grouping, type: nil) + let includesLaneArg = includesLane.asRubyArgument(name: "includes_lane", type: nil) + let prefixArg = RubyCommand.Argument(name: "prefix", value: prefix, type: nil) + let postfixArg = RubyCommand.Argument(name: "postfix", value: postfix, type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let messageArg = message.asRubyArgument(name: "message", type: nil) + let commitArg = commit.asRubyArgument(name: "commit", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let signArg = sign.asRubyArgument(name: "sign", type: nil) + let array: [RubyCommand.Argument?] = [tagArg, + groupingArg, + includesLaneArg, + prefixArg, + postfixArg, + buildNumberArg, + messageArg, + commitArg, + forceArg, + signArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "add_git_tag", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Returns the current build_number of either live or edit version + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - initialBuildNumber: sets the build number to given value if no build is in current train + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - teamId: The ID of your App Store Connect team if you're in multiple teams + - live: Query the live version (ready-for-sale) + - version: The version number whose latest build number we want + - platform: The platform to use (optional) + - teamName: The name of your App Store Connect team if you're in multiple teams + + Returns the current build number of either the live or testflight version - it is useful for getting the build_number of the current or ready-for-sale app version, and it also works on non-live testflight version. + If you need to handle more build-trains please see `latest_testflight_build_number`. + */ +public func appStoreBuildNumber(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + initialBuildNumber: String, + appIdentifier: String, + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + live: OptionalConfigValue = .fastlaneDefault(true), + version: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + teamName: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let initialBuildNumberArg = RubyCommand.Argument(name: "initial_build_number", value: initialBuildNumber, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let liveArg = live.asRubyArgument(name: "live", type: nil) + let versionArg = version.asRubyArgument(name: "version", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + initialBuildNumberArg, + appIdentifierArg, + usernameArg, + teamIdArg, + liveArg, + versionArg, + platformArg, + teamNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "app_store_build_number", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Load the App Store Connect API token to use in other fastlane tools and actions + + - parameters: + - keyId: The key ID + - issuerId: The issuer ID + - keyFilepath: The path to the key p8 file + - keyContent: The content of the key p8 file + - isKeyContentBase64: Whether :key_content is Base64 encoded or not + - duration: The token session duration + - inHouse: Is App Store or Enterprise (in house) team? App Store Connect API cannot determine this on its own (yet) + - setSpaceshipToken: Authorizes all Spaceship::ConnectAPI requests by automatically setting Spaceship::ConnectAPI.token + + Load the App Store Connect API token to use in other fastlane tools and actions + */ +public func appStoreConnectApiKey(keyId: String, + issuerId: String, + keyFilepath: OptionalConfigValue = .fastlaneDefault(nil), + keyContent: OptionalConfigValue = .fastlaneDefault(nil), + isKeyContentBase64: OptionalConfigValue = .fastlaneDefault(false), + duration: Int = 500, + inHouse: OptionalConfigValue = .fastlaneDefault(false), + setSpaceshipToken: OptionalConfigValue = .fastlaneDefault(true)) +{ + let keyIdArg = RubyCommand.Argument(name: "key_id", value: keyId, type: nil) + let issuerIdArg = RubyCommand.Argument(name: "issuer_id", value: issuerId, type: nil) + let keyFilepathArg = keyFilepath.asRubyArgument(name: "key_filepath", type: nil) + let keyContentArg = keyContent.asRubyArgument(name: "key_content", type: nil) + let isKeyContentBase64Arg = isKeyContentBase64.asRubyArgument(name: "is_key_content_base64", type: nil) + let durationArg = RubyCommand.Argument(name: "duration", value: duration, type: nil) + let inHouseArg = inHouse.asRubyArgument(name: "in_house", type: nil) + let setSpaceshipTokenArg = setSpaceshipToken.asRubyArgument(name: "set_spaceship_token", type: nil) + let array: [RubyCommand.Argument?] = [keyIdArg, + issuerIdArg, + keyFilepathArg, + keyContentArg, + isKeyContentBase64Arg, + durationArg, + inHouseArg, + setSpaceshipTokenArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "app_store_connect_api_key", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload your app to [Appaloosa Store](https://www.appaloosa-store.com/) + + - parameters: + - binary: Binary path. Optional for ipa if you use the `ipa` or `xcodebuild` action + - apiToken: Your API token + - storeId: Your Store id + - groupIds: Your app is limited to special users? Give us the group ids + - screenshots: Add some screenshots application to your store or hit [enter] + - locale: Select the folder locale for your screenshots + - device: Select the device format for your screenshots + - description: Your app description + - changelog: Your app changelog + + Appaloosa is a private mobile application store. This action offers a quick deployment on the platform. + You can create an account, push to your existing account, or manage your user groups. + We accept iOS and Android applications. + */ +public func appaloosa(binary: String, + apiToken: String, + storeId: String, + groupIds: String = "", + screenshots: String, + locale: String = "en-US", + device: OptionalConfigValue = .fastlaneDefault(nil), + description: OptionalConfigValue = .fastlaneDefault(nil), + changelog: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let binaryArg = RubyCommand.Argument(name: "binary", value: binary, type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let storeIdArg = RubyCommand.Argument(name: "store_id", value: storeId, type: nil) + let groupIdsArg = RubyCommand.Argument(name: "group_ids", value: groupIds, type: nil) + let screenshotsArg = RubyCommand.Argument(name: "screenshots", value: screenshots, type: nil) + let localeArg = RubyCommand.Argument(name: "locale", value: locale, type: nil) + let deviceArg = device.asRubyArgument(name: "device", type: nil) + let descriptionArg = description.asRubyArgument(name: "description", type: nil) + let changelogArg = changelog.asRubyArgument(name: "changelog", type: nil) + let array: [RubyCommand.Argument?] = [binaryArg, + apiTokenArg, + storeIdArg, + groupIdsArg, + screenshotsArg, + localeArg, + deviceArg, + descriptionArg, + changelogArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "appaloosa", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload your app to [Appetize.io](https://appetize.io/) to stream it in browser + + - parameters: + - apiHost: Appetize API host + - apiToken: Appetize.io API Token + - url: URL from which the ipa file can be fetched. Alternative to :path + - platform: Platform. Either `ios` or `android` + - path: Path to zipped build on the local filesystem. Either this or `url` must be specified + - publicKey: If not provided, a new app will be created. If provided, the existing build will be overwritten + - note: Notes you wish to add to the uploaded app + - timeout: The number of seconds to wait until automatically ending the session due to user inactivity. Must be 30, 60, 90, 120, 180, 300, 600, 1800, 3600 or 7200. Default is 120 + + If you provide a `public_key`, this will overwrite an existing application. If you want to have this build as a new app version, you shouldn't provide this value. + + To integrate appetize into your GitHub workflow check out the [device_grid guide](https://github.com/fastlane/fastlane/blob/master/fastlane/lib/fastlane/actions/device_grid/README.md). + */ +public func appetize(apiHost: String = "api.appetize.io", + apiToken: String, + url: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + path: OptionalConfigValue = .fastlaneDefault(nil), + publicKey: OptionalConfigValue = .fastlaneDefault(nil), + note: OptionalConfigValue = .fastlaneDefault(nil), + timeout: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiHostArg = RubyCommand.Argument(name: "api_host", value: apiHost, type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let urlArg = url.asRubyArgument(name: "url", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let pathArg = path.asRubyArgument(name: "path", type: nil) + let publicKeyArg = publicKey.asRubyArgument(name: "public_key", type: nil) + let noteArg = note.asRubyArgument(name: "note", type: nil) + let timeoutArg = timeout.asRubyArgument(name: "timeout", type: nil) + let array: [RubyCommand.Argument?] = [apiHostArg, + apiTokenArg, + urlArg, + platformArg, + pathArg, + publicKeyArg, + noteArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "appetize", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generate an URL for appetize simulator + + - parameters: + - publicKey: Public key of the app you wish to update + - baseUrl: Base URL of Appetize service + - device: Device type: iphone4s, iphone5s, iphone6, iphone6plus, ipadair, iphone6s, iphone6splus, ipadair2, nexus5, nexus7 or nexus9 + - scale: Scale of the simulator + - orientation: Device orientation + - language: Device language in ISO 639-1 language code, e.g. 'de' + - color: Color of the device + - launchUrl: Specify a deep link to open when your app is launched + - osVersion: The operating system version on which to run your app, e.g. 10.3, 8.0 + - params: Specify params value to be passed to Appetize + - proxy: Specify a HTTP proxy to be passed to Appetize + + - returns: The URL to preview the iPhone app + + Check out the [device_grid guide](https://github.com/fastlane/fastlane/blob/master/fastlane/lib/fastlane/actions/device_grid/README.md) for more information + */ +public func appetizeViewingUrlGenerator(publicKey: String, + baseUrl: String = "https://appetize.io/embed", + device: String = "iphone5s", + scale: OptionalConfigValue = .fastlaneDefault(nil), + orientation: String = "portrait", + language: OptionalConfigValue = .fastlaneDefault(nil), + color: String = "black", + launchUrl: OptionalConfigValue = .fastlaneDefault(nil), + osVersion: OptionalConfigValue = .fastlaneDefault(nil), + params: OptionalConfigValue = .fastlaneDefault(nil), + proxy: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let publicKeyArg = RubyCommand.Argument(name: "public_key", value: publicKey, type: nil) + let baseUrlArg = RubyCommand.Argument(name: "base_url", value: baseUrl, type: nil) + let deviceArg = RubyCommand.Argument(name: "device", value: device, type: nil) + let scaleArg = scale.asRubyArgument(name: "scale", type: nil) + let orientationArg = RubyCommand.Argument(name: "orientation", value: orientation, type: nil) + let languageArg = language.asRubyArgument(name: "language", type: nil) + let colorArg = RubyCommand.Argument(name: "color", value: color, type: nil) + let launchUrlArg = launchUrl.asRubyArgument(name: "launch_url", type: nil) + let osVersionArg = osVersion.asRubyArgument(name: "os_version", type: nil) + let paramsArg = params.asRubyArgument(name: "params", type: nil) + let proxyArg = proxy.asRubyArgument(name: "proxy", type: nil) + let array: [RubyCommand.Argument?] = [publicKeyArg, + baseUrlArg, + deviceArg, + scaleArg, + orientationArg, + languageArg, + colorArg, + launchUrlArg, + osVersionArg, + paramsArg, + proxyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "appetize_viewing_url_generator", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Run UI test by Appium with RSpec + + - parameters: + - platform: Appium platform name + - specPath: Path to Appium spec directory + - appPath: Path to Appium target app file + - invokeAppiumServer: Use local Appium server with invoke automatically + - host: Hostname of Appium server + - port: HTTP port of Appium server + - appiumPath: Path to Appium executable + - caps: Hash of caps for Appium::Driver + - appiumLib: Hash of appium_lib for Appium::Driver + */ +public func appium(platform: String, + specPath: String, + appPath: String, + invokeAppiumServer: OptionalConfigValue = .fastlaneDefault(true), + host: String = "0.0.0.0", + port: Int = 4723, + appiumPath: OptionalConfigValue = .fastlaneDefault(nil), + caps: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appiumLib: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil)) +{ + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let specPathArg = RubyCommand.Argument(name: "spec_path", value: specPath, type: nil) + let appPathArg = RubyCommand.Argument(name: "app_path", value: appPath, type: nil) + let invokeAppiumServerArg = invokeAppiumServer.asRubyArgument(name: "invoke_appium_server", type: nil) + let hostArg = RubyCommand.Argument(name: "host", value: host, type: nil) + let portArg = RubyCommand.Argument(name: "port", value: port, type: nil) + let appiumPathArg = appiumPath.asRubyArgument(name: "appium_path", type: nil) + let capsArg = caps.asRubyArgument(name: "caps", type: nil) + let appiumLibArg = appiumLib.asRubyArgument(name: "appium_lib", type: nil) + let array: [RubyCommand.Argument?] = [platformArg, + specPathArg, + appPathArg, + invokeAppiumServerArg, + hostArg, + portArg, + appiumPathArg, + capsArg, + appiumLibArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "appium", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generate Apple-like source code documentation from the source code + + - parameters: + - input: Path(s) to source file directories or individual source files. Accepts a single path or an array of paths + - output: Output path + - templates: Template files path + - docsetInstallPath: DocSet installation path + - include: Include static doc(s) at path + - ignore: Ignore given path + - excludeOutput: Exclude given path from output + - indexDesc: File including main index description + - projectName: Project name + - projectVersion: Project version + - projectCompany: Project company + - companyId: Company UTI (i.e. reverse DNS name) + - createHtml: Create HTML + - createDocset: Create documentation set + - installDocset: Install documentation set to Xcode + - publishDocset: Prepare DocSet for publishing + - noCreateDocset: Create HTML and skip creating a DocSet + - htmlAnchors: The html anchor format to use in DocSet HTML + - cleanOutput: Remove contents of output path before starting + - docsetBundleId: DocSet bundle identifier + - docsetBundleName: DocSet bundle name + - docsetDesc: DocSet description + - docsetCopyright: DocSet copyright message + - docsetFeedName: DocSet feed name + - docsetFeedUrl: DocSet feed URL + - docsetFeedFormats: DocSet feed formats. Separated by a comma [atom,xml] + - docsetPackageUrl: DocSet package (.xar) URL + - docsetFallbackUrl: DocSet fallback URL + - docsetPublisherId: DocSet publisher identifier + - docsetPublisherName: DocSet publisher name + - docsetMinXcodeVersion: DocSet min. Xcode version + - docsetPlatformFamily: DocSet platform family + - docsetCertIssuer: DocSet certificate issuer + - docsetCertSigner: DocSet certificate signer + - docsetBundleFilename: DocSet bundle filename + - docsetAtomFilename: DocSet atom feed filename + - docsetXmlFilename: DocSet xml feed filename + - docsetPackageFilename: DocSet package (.xar,.tgz) filename + - options: Documentation generation options + - crossrefFormat: Cross reference template regex + - exitThreshold: Exit code threshold below which 0 is returned + - docsSectionTitle: Title of the documentation section (defaults to "Programming Guides" + - warnings: Documentation generation warnings + - logformat: Log format [0-3] + - verbose: Log verbosity level [0-6,xcode] + + Runs `appledoc [OPTIONS] ` for the project + */ +public func appledoc(input: [String], + output: OptionalConfigValue = .fastlaneDefault(nil), + templates: OptionalConfigValue = .fastlaneDefault(nil), + docsetInstallPath: OptionalConfigValue = .fastlaneDefault(nil), + include: OptionalConfigValue = .fastlaneDefault(nil), + ignore: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + excludeOutput: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + indexDesc: OptionalConfigValue = .fastlaneDefault(nil), + projectName: String, + projectVersion: OptionalConfigValue = .fastlaneDefault(nil), + projectCompany: String, + companyId: OptionalConfigValue = .fastlaneDefault(nil), + createHtml: OptionalConfigValue = .fastlaneDefault(false), + createDocset: OptionalConfigValue = .fastlaneDefault(false), + installDocset: OptionalConfigValue = .fastlaneDefault(false), + publishDocset: OptionalConfigValue = .fastlaneDefault(false), + noCreateDocset: OptionalConfigValue = .fastlaneDefault(false), + htmlAnchors: OptionalConfigValue = .fastlaneDefault(nil), + cleanOutput: OptionalConfigValue = .fastlaneDefault(false), + docsetBundleId: OptionalConfigValue = .fastlaneDefault(nil), + docsetBundleName: OptionalConfigValue = .fastlaneDefault(nil), + docsetDesc: OptionalConfigValue = .fastlaneDefault(nil), + docsetCopyright: OptionalConfigValue = .fastlaneDefault(nil), + docsetFeedName: OptionalConfigValue = .fastlaneDefault(nil), + docsetFeedUrl: OptionalConfigValue = .fastlaneDefault(nil), + docsetFeedFormats: OptionalConfigValue = .fastlaneDefault(nil), + docsetPackageUrl: OptionalConfigValue = .fastlaneDefault(nil), + docsetFallbackUrl: OptionalConfigValue = .fastlaneDefault(nil), + docsetPublisherId: OptionalConfigValue = .fastlaneDefault(nil), + docsetPublisherName: OptionalConfigValue = .fastlaneDefault(nil), + docsetMinXcodeVersion: OptionalConfigValue = .fastlaneDefault(nil), + docsetPlatformFamily: OptionalConfigValue = .fastlaneDefault(nil), + docsetCertIssuer: OptionalConfigValue = .fastlaneDefault(nil), + docsetCertSigner: OptionalConfigValue = .fastlaneDefault(nil), + docsetBundleFilename: OptionalConfigValue = .fastlaneDefault(nil), + docsetAtomFilename: OptionalConfigValue = .fastlaneDefault(nil), + docsetXmlFilename: OptionalConfigValue = .fastlaneDefault(nil), + docsetPackageFilename: OptionalConfigValue = .fastlaneDefault(nil), + options: OptionalConfigValue = .fastlaneDefault(nil), + crossrefFormat: OptionalConfigValue = .fastlaneDefault(nil), + exitThreshold: Int = 2, + docsSectionTitle: OptionalConfigValue = .fastlaneDefault(nil), + warnings: OptionalConfigValue = .fastlaneDefault(nil), + logformat: OptionalConfigValue = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let inputArg = RubyCommand.Argument(name: "input", value: input, type: nil) + let outputArg = output.asRubyArgument(name: "output", type: nil) + let templatesArg = templates.asRubyArgument(name: "templates", type: nil) + let docsetInstallPathArg = docsetInstallPath.asRubyArgument(name: "docset_install_path", type: nil) + let includeArg = include.asRubyArgument(name: "include", type: nil) + let ignoreArg = ignore.asRubyArgument(name: "ignore", type: nil) + let excludeOutputArg = excludeOutput.asRubyArgument(name: "exclude_output", type: nil) + let indexDescArg = indexDesc.asRubyArgument(name: "index_desc", type: nil) + let projectNameArg = RubyCommand.Argument(name: "project_name", value: projectName, type: nil) + let projectVersionArg = projectVersion.asRubyArgument(name: "project_version", type: nil) + let projectCompanyArg = RubyCommand.Argument(name: "project_company", value: projectCompany, type: nil) + let companyIdArg = companyId.asRubyArgument(name: "company_id", type: nil) + let createHtmlArg = createHtml.asRubyArgument(name: "create_html", type: nil) + let createDocsetArg = createDocset.asRubyArgument(name: "create_docset", type: nil) + let installDocsetArg = installDocset.asRubyArgument(name: "install_docset", type: nil) + let publishDocsetArg = publishDocset.asRubyArgument(name: "publish_docset", type: nil) + let noCreateDocsetArg = noCreateDocset.asRubyArgument(name: "no_create_docset", type: nil) + let htmlAnchorsArg = htmlAnchors.asRubyArgument(name: "html_anchors", type: nil) + let cleanOutputArg = cleanOutput.asRubyArgument(name: "clean_output", type: nil) + let docsetBundleIdArg = docsetBundleId.asRubyArgument(name: "docset_bundle_id", type: nil) + let docsetBundleNameArg = docsetBundleName.asRubyArgument(name: "docset_bundle_name", type: nil) + let docsetDescArg = docsetDesc.asRubyArgument(name: "docset_desc", type: nil) + let docsetCopyrightArg = docsetCopyright.asRubyArgument(name: "docset_copyright", type: nil) + let docsetFeedNameArg = docsetFeedName.asRubyArgument(name: "docset_feed_name", type: nil) + let docsetFeedUrlArg = docsetFeedUrl.asRubyArgument(name: "docset_feed_url", type: nil) + let docsetFeedFormatsArg = docsetFeedFormats.asRubyArgument(name: "docset_feed_formats", type: nil) + let docsetPackageUrlArg = docsetPackageUrl.asRubyArgument(name: "docset_package_url", type: nil) + let docsetFallbackUrlArg = docsetFallbackUrl.asRubyArgument(name: "docset_fallback_url", type: nil) + let docsetPublisherIdArg = docsetPublisherId.asRubyArgument(name: "docset_publisher_id", type: nil) + let docsetPublisherNameArg = docsetPublisherName.asRubyArgument(name: "docset_publisher_name", type: nil) + let docsetMinXcodeVersionArg = docsetMinXcodeVersion.asRubyArgument(name: "docset_min_xcode_version", type: nil) + let docsetPlatformFamilyArg = docsetPlatformFamily.asRubyArgument(name: "docset_platform_family", type: nil) + let docsetCertIssuerArg = docsetCertIssuer.asRubyArgument(name: "docset_cert_issuer", type: nil) + let docsetCertSignerArg = docsetCertSigner.asRubyArgument(name: "docset_cert_signer", type: nil) + let docsetBundleFilenameArg = docsetBundleFilename.asRubyArgument(name: "docset_bundle_filename", type: nil) + let docsetAtomFilenameArg = docsetAtomFilename.asRubyArgument(name: "docset_atom_filename", type: nil) + let docsetXmlFilenameArg = docsetXmlFilename.asRubyArgument(name: "docset_xml_filename", type: nil) + let docsetPackageFilenameArg = docsetPackageFilename.asRubyArgument(name: "docset_package_filename", type: nil) + let optionsArg = options.asRubyArgument(name: "options", type: nil) + let crossrefFormatArg = crossrefFormat.asRubyArgument(name: "crossref_format", type: nil) + let exitThresholdArg = RubyCommand.Argument(name: "exit_threshold", value: exitThreshold, type: nil) + let docsSectionTitleArg = docsSectionTitle.asRubyArgument(name: "docs_section_title", type: nil) + let warningsArg = warnings.asRubyArgument(name: "warnings", type: nil) + let logformatArg = logformat.asRubyArgument(name: "logformat", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let array: [RubyCommand.Argument?] = [inputArg, + outputArg, + templatesArg, + docsetInstallPathArg, + includeArg, + ignoreArg, + excludeOutputArg, + indexDescArg, + projectNameArg, + projectVersionArg, + projectCompanyArg, + companyIdArg, + createHtmlArg, + createDocsetArg, + installDocsetArg, + publishDocsetArg, + noCreateDocsetArg, + htmlAnchorsArg, + cleanOutputArg, + docsetBundleIdArg, + docsetBundleNameArg, + docsetDescArg, + docsetCopyrightArg, + docsetFeedNameArg, + docsetFeedUrlArg, + docsetFeedFormatsArg, + docsetPackageUrlArg, + docsetFallbackUrlArg, + docsetPublisherIdArg, + docsetPublisherNameArg, + docsetMinXcodeVersionArg, + docsetPlatformFamilyArg, + docsetCertIssuerArg, + docsetCertSignerArg, + docsetBundleFilenameArg, + docsetAtomFilenameArg, + docsetXmlFilenameArg, + docsetPackageFilenameArg, + optionsArg, + crossrefFormatArg, + exitThresholdArg, + docsSectionTitleArg, + warningsArg, + logformatArg, + verboseArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "appledoc", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `upload_to_app_store` action + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - appIdentifier: The bundle identifier of your app + - appVersion: The version that should be edited or created + - ipa: Path to your ipa file + - pkg: Path to your pkg file + - buildNumber: If set the given build number (already uploaded to iTC) will be used instead of the current built one + - platform: The platform to use (optional) + - editLive: Modify live metadata, this option disables ipa upload and screenshot upload + - useLiveVersion: Force usage of live version rather than edit version + - metadataPath: Path to the folder containing the metadata files + - screenshotsPath: Path to the folder containing the screenshots + - skipBinaryUpload: Skip uploading an ipa or pkg to App Store Connect + - skipScreenshots: Don't upload the screenshots + - skipMetadata: Don't upload the metadata (e.g. title, description). This will still upload screenshots + - skipAppVersionUpdate: Don’t create or update the app version that is being prepared for submission + - force: Skip verification of HTML preview file + - overwriteScreenshots: Clear all previously uploaded screenshots before uploading the new ones + - syncScreenshots: Sync screenshots with local ones. This is currently beta option so set true to 'FASTLANE_ENABLE_BETA_DELIVER_SYNC_SCREENSHOTS' environment variable as well + - submitForReview: Submit the new version for Review after uploading everything + - verifyOnly: Verifies archive with App Store Connect without uploading + - rejectIfPossible: Rejects the previously submitted build if it's in a state where it's possible + - automaticRelease: Should the app be automatically released once it's approved? (Can not be used together with `auto_release_date`) + - autoReleaseDate: Date in milliseconds for automatically releasing on pending approval (Can not be used together with `automatic_release`) + - phasedRelease: Enable the phased release feature of iTC + - resetRatings: Reset the summary rating when you release a new version of the application + - priceTier: The price tier of this application + - appRatingConfigPath: Path to the app rating's config + - submissionInformation: Extra information for the submission (e.g. compliance specifications, IDFA settings) + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - devPortalTeamId: The short ID of your Developer Portal team, if you're in multiple teams. Different from your iTC team ID! + - devPortalTeamName: The name of your Developer Portal team if you're in multiple teams + - itcProvider: The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + - runPrecheckBeforeSubmit: Run precheck before submitting to app review + - precheckDefaultRuleLevel: The default precheck rule level unless otherwise configured + - individualMetadataItems: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - An array of localized metadata items to upload individually by language so that errors can be identified. E.g. ['name', 'keywords', 'description']. Note: slow + - appIcon: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the app icon + - appleWatchAppIcon: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the Apple Watch app icon + - copyright: Metadata: The copyright notice + - primaryCategory: Metadata: The english name of the primary category (e.g. `Business`, `Books`) + - secondaryCategory: Metadata: The english name of the secondary category (e.g. `Business`, `Books`) + - primaryFirstSubCategory: Metadata: The english name of the primary first sub category (e.g. `Educational`, `Puzzle`) + - primarySecondSubCategory: Metadata: The english name of the primary second sub category (e.g. `Educational`, `Puzzle`) + - secondaryFirstSubCategory: Metadata: The english name of the secondary first sub category (e.g. `Educational`, `Puzzle`) + - secondarySecondSubCategory: Metadata: The english name of the secondary second sub category (e.g. `Educational`, `Puzzle`) + - tradeRepresentativeContactInformation: **DEPRECATED!** This is no longer used by App Store Connect - Metadata: A hash containing the trade representative contact information + - appReviewInformation: Metadata: A hash containing the review information + - appReviewAttachmentFile: Metadata: Path to the app review attachment file + - description: Metadata: The localised app description + - name: Metadata: The localised app name + - subtitle: Metadata: The localised app subtitle + - keywords: Metadata: An array of localised keywords + - promotionalText: Metadata: An array of localised promotional texts + - releaseNotes: Metadata: Localised release notes for this version + - privacyUrl: Metadata: Localised privacy url + - appleTvPrivacyPolicy: Metadata: Localised Apple TV privacy policy text + - supportUrl: Metadata: Localised support url + - marketingUrl: Metadata: Localised marketing url + - languages: Metadata: List of languages to activate + - ignoreLanguageDirectoryValidation: Ignore errors when invalid languages are found in metadata and screenshot directories + - precheckIncludeInAppPurchases: Should precheck check in-app purchases? + - app: The (spaceship) app ID of the app you want to use/modify + + Using _upload_to_app_store_ after _build_app_ and _capture_screenshots_ will automatically upload the latest ipa and screenshots with no other configuration. + + If you don't want to verify an HTML preview for App Store builds, use the `:force` option. + This is useful when running _fastlane_ on your Continuous Integration server: + `_upload_to_app_store_(force: true)` + If your account is on multiple teams and you need to tell the `iTMSTransporter` which 'provider' to use, you can set the `:itc_provider` option to pass this info. + */ +public func appstore(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil), + pkg: OptionalConfigValue = .fastlaneDefault(nil), + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + editLive: OptionalConfigValue = .fastlaneDefault(false), + useLiveVersion: OptionalConfigValue = .fastlaneDefault(false), + metadataPath: OptionalConfigValue = .fastlaneDefault(nil), + screenshotsPath: OptionalConfigValue = .fastlaneDefault(nil), + skipBinaryUpload: OptionalConfigValue = .fastlaneDefault(false), + skipScreenshots: OptionalConfigValue = .fastlaneDefault(false), + skipMetadata: OptionalConfigValue = .fastlaneDefault(false), + skipAppVersionUpdate: OptionalConfigValue = .fastlaneDefault(false), + force: OptionalConfigValue = .fastlaneDefault(false), + overwriteScreenshots: OptionalConfigValue = .fastlaneDefault(false), + syncScreenshots: OptionalConfigValue = .fastlaneDefault(false), + submitForReview: OptionalConfigValue = .fastlaneDefault(false), + verifyOnly: OptionalConfigValue = .fastlaneDefault(false), + rejectIfPossible: OptionalConfigValue = .fastlaneDefault(false), + automaticRelease: OptionalConfigValue = .fastlaneDefault(nil), + autoReleaseDate: OptionalConfigValue = .fastlaneDefault(nil), + phasedRelease: OptionalConfigValue = .fastlaneDefault(false), + resetRatings: OptionalConfigValue = .fastlaneDefault(false), + priceTier: OptionalConfigValue = .fastlaneDefault(nil), + appRatingConfigPath: OptionalConfigValue = .fastlaneDefault(nil), + submissionInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamId: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamName: OptionalConfigValue = .fastlaneDefault(nil), + itcProvider: OptionalConfigValue = .fastlaneDefault(nil), + runPrecheckBeforeSubmit: OptionalConfigValue = .fastlaneDefault(true), + precheckDefaultRuleLevel: String = "warn", + individualMetadataItems: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + appIcon: OptionalConfigValue = .fastlaneDefault(nil), + appleWatchAppIcon: OptionalConfigValue = .fastlaneDefault(nil), + copyright: OptionalConfigValue = .fastlaneDefault(nil), + primaryCategory: OptionalConfigValue = .fastlaneDefault(nil), + secondaryCategory: OptionalConfigValue = .fastlaneDefault(nil), + primaryFirstSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + primarySecondSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + secondaryFirstSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + secondarySecondSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + tradeRepresentativeContactInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appReviewInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appReviewAttachmentFile: OptionalConfigValue = .fastlaneDefault(nil), + description: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + name: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + subtitle: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + keywords: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + promotionalText: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + releaseNotes: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + privacyUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appleTvPrivacyPolicy: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + supportUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + marketingUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + languages: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + ignoreLanguageDirectoryValidation: OptionalConfigValue = .fastlaneDefault(false), + precheckIncludeInAppPurchases: OptionalConfigValue = .fastlaneDefault(true), + app: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let pkgArg = pkg.asRubyArgument(name: "pkg", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let editLiveArg = editLive.asRubyArgument(name: "edit_live", type: nil) + let useLiveVersionArg = useLiveVersion.asRubyArgument(name: "use_live_version", type: nil) + let metadataPathArg = metadataPath.asRubyArgument(name: "metadata_path", type: nil) + let screenshotsPathArg = screenshotsPath.asRubyArgument(name: "screenshots_path", type: nil) + let skipBinaryUploadArg = skipBinaryUpload.asRubyArgument(name: "skip_binary_upload", type: nil) + let skipScreenshotsArg = skipScreenshots.asRubyArgument(name: "skip_screenshots", type: nil) + let skipMetadataArg = skipMetadata.asRubyArgument(name: "skip_metadata", type: nil) + let skipAppVersionUpdateArg = skipAppVersionUpdate.asRubyArgument(name: "skip_app_version_update", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let overwriteScreenshotsArg = overwriteScreenshots.asRubyArgument(name: "overwrite_screenshots", type: nil) + let syncScreenshotsArg = syncScreenshots.asRubyArgument(name: "sync_screenshots", type: nil) + let submitForReviewArg = submitForReview.asRubyArgument(name: "submit_for_review", type: nil) + let verifyOnlyArg = verifyOnly.asRubyArgument(name: "verify_only", type: nil) + let rejectIfPossibleArg = rejectIfPossible.asRubyArgument(name: "reject_if_possible", type: nil) + let automaticReleaseArg = automaticRelease.asRubyArgument(name: "automatic_release", type: nil) + let autoReleaseDateArg = autoReleaseDate.asRubyArgument(name: "auto_release_date", type: nil) + let phasedReleaseArg = phasedRelease.asRubyArgument(name: "phased_release", type: nil) + let resetRatingsArg = resetRatings.asRubyArgument(name: "reset_ratings", type: nil) + let priceTierArg = priceTier.asRubyArgument(name: "price_tier", type: nil) + let appRatingConfigPathArg = appRatingConfigPath.asRubyArgument(name: "app_rating_config_path", type: nil) + let submissionInformationArg = submissionInformation.asRubyArgument(name: "submission_information", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let devPortalTeamIdArg = devPortalTeamId.asRubyArgument(name: "dev_portal_team_id", type: nil) + let devPortalTeamNameArg = devPortalTeamName.asRubyArgument(name: "dev_portal_team_name", type: nil) + let itcProviderArg = itcProvider.asRubyArgument(name: "itc_provider", type: nil) + let runPrecheckBeforeSubmitArg = runPrecheckBeforeSubmit.asRubyArgument(name: "run_precheck_before_submit", type: nil) + let precheckDefaultRuleLevelArg = RubyCommand.Argument(name: "precheck_default_rule_level", value: precheckDefaultRuleLevel, type: nil) + let individualMetadataItemsArg = individualMetadataItems.asRubyArgument(name: "individual_metadata_items", type: nil) + let appIconArg = appIcon.asRubyArgument(name: "app_icon", type: nil) + let appleWatchAppIconArg = appleWatchAppIcon.asRubyArgument(name: "apple_watch_app_icon", type: nil) + let copyrightArg = copyright.asRubyArgument(name: "copyright", type: nil) + let primaryCategoryArg = primaryCategory.asRubyArgument(name: "primary_category", type: nil) + let secondaryCategoryArg = secondaryCategory.asRubyArgument(name: "secondary_category", type: nil) + let primaryFirstSubCategoryArg = primaryFirstSubCategory.asRubyArgument(name: "primary_first_sub_category", type: nil) + let primarySecondSubCategoryArg = primarySecondSubCategory.asRubyArgument(name: "primary_second_sub_category", type: nil) + let secondaryFirstSubCategoryArg = secondaryFirstSubCategory.asRubyArgument(name: "secondary_first_sub_category", type: nil) + let secondarySecondSubCategoryArg = secondarySecondSubCategory.asRubyArgument(name: "secondary_second_sub_category", type: nil) + let tradeRepresentativeContactInformationArg = tradeRepresentativeContactInformation.asRubyArgument(name: "trade_representative_contact_information", type: nil) + let appReviewInformationArg = appReviewInformation.asRubyArgument(name: "app_review_information", type: nil) + let appReviewAttachmentFileArg = appReviewAttachmentFile.asRubyArgument(name: "app_review_attachment_file", type: nil) + let descriptionArg = description.asRubyArgument(name: "description", type: nil) + let nameArg = name.asRubyArgument(name: "name", type: nil) + let subtitleArg = subtitle.asRubyArgument(name: "subtitle", type: nil) + let keywordsArg = keywords.asRubyArgument(name: "keywords", type: nil) + let promotionalTextArg = promotionalText.asRubyArgument(name: "promotional_text", type: nil) + let releaseNotesArg = releaseNotes.asRubyArgument(name: "release_notes", type: nil) + let privacyUrlArg = privacyUrl.asRubyArgument(name: "privacy_url", type: nil) + let appleTvPrivacyPolicyArg = appleTvPrivacyPolicy.asRubyArgument(name: "apple_tv_privacy_policy", type: nil) + let supportUrlArg = supportUrl.asRubyArgument(name: "support_url", type: nil) + let marketingUrlArg = marketingUrl.asRubyArgument(name: "marketing_url", type: nil) + let languagesArg = languages.asRubyArgument(name: "languages", type: nil) + let ignoreLanguageDirectoryValidationArg = ignoreLanguageDirectoryValidation.asRubyArgument(name: "ignore_language_directory_validation", type: nil) + let precheckIncludeInAppPurchasesArg = precheckIncludeInAppPurchases.asRubyArgument(name: "precheck_include_in_app_purchases", type: nil) + let appArg = app.asRubyArgument(name: "app", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + appVersionArg, + ipaArg, + pkgArg, + buildNumberArg, + platformArg, + editLiveArg, + useLiveVersionArg, + metadataPathArg, + screenshotsPathArg, + skipBinaryUploadArg, + skipScreenshotsArg, + skipMetadataArg, + skipAppVersionUpdateArg, + forceArg, + overwriteScreenshotsArg, + syncScreenshotsArg, + submitForReviewArg, + verifyOnlyArg, + rejectIfPossibleArg, + automaticReleaseArg, + autoReleaseDateArg, + phasedReleaseArg, + resetRatingsArg, + priceTierArg, + appRatingConfigPathArg, + submissionInformationArg, + teamIdArg, + teamNameArg, + devPortalTeamIdArg, + devPortalTeamNameArg, + itcProviderArg, + runPrecheckBeforeSubmitArg, + precheckDefaultRuleLevelArg, + individualMetadataItemsArg, + appIconArg, + appleWatchAppIconArg, + copyrightArg, + primaryCategoryArg, + secondaryCategoryArg, + primaryFirstSubCategoryArg, + primarySecondSubCategoryArg, + secondaryFirstSubCategoryArg, + secondarySecondSubCategoryArg, + tradeRepresentativeContactInformationArg, + appReviewInformationArg, + appReviewAttachmentFileArg, + descriptionArg, + nameArg, + subtitleArg, + keywordsArg, + promotionalTextArg, + releaseNotesArg, + privacyUrlArg, + appleTvPrivacyPolicyArg, + supportUrlArg, + marketingUrlArg, + languagesArg, + ignoreLanguageDirectoryValidationArg, + precheckIncludeInAppPurchasesArg, + appArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "appstore", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload dSYM file to [Apteligent (Crittercism)](http://www.apteligent.com/) + + - parameters: + - dsym: dSYM.zip file to upload to Apteligent + - appId: Apteligent App ID key e.g. 569f5c87cb99e10e00c7xxxx + - apiKey: Apteligent App API key e.g. IXPQIi8yCbHaLliqzRoo065tH0lxxxxx + */ +public func apteligent(dsym: OptionalConfigValue = .fastlaneDefault(nil), + appId: String, + apiKey: String) +{ + let dsymArg = dsym.asRubyArgument(name: "dsym", type: nil) + let appIdArg = RubyCommand.Argument(name: "app_id", value: appId, type: nil) + let apiKeyArg = RubyCommand.Argument(name: "api_key", value: apiKey, type: nil) + let array: [RubyCommand.Argument?] = [dsymArg, + appIdArg, + apiKeyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "apteligent", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action uploads an artifact to artifactory + + - parameters: + - file: File to be uploaded to artifactory + - repo: Artifactory repo to put the file in + - repoPath: Path to deploy within the repo, including filename + - endpoint: Artifactory endpoint + - username: Artifactory username + - password: Artifactory password + - apiKey: Artifactory API key + - properties: Artifact properties hash + - sslPemFile: Location of pem file to use for ssl verification + - sslVerify: Verify SSL + - proxyUsername: Proxy username + - proxyPassword: Proxy password + - proxyAddress: Proxy address + - proxyPort: Proxy port + - readTimeout: Read timeout + + Connect to the artifactory server using either a username/password or an api_key + */ +public func artifactory(file: String, + repo: String, + repoPath: String, + endpoint: String, + username: OptionalConfigValue = .fastlaneDefault(nil), + password: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue = .fastlaneDefault(nil), + properties: [String: Any] = [:], + sslPemFile: OptionalConfigValue = .fastlaneDefault(nil), + sslVerify: OptionalConfigValue = .fastlaneDefault(true), + proxyUsername: OptionalConfigValue = .fastlaneDefault(nil), + proxyPassword: OptionalConfigValue = .fastlaneDefault(nil), + proxyAddress: OptionalConfigValue = .fastlaneDefault(nil), + proxyPort: OptionalConfigValue = .fastlaneDefault(nil), + readTimeout: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let fileArg = RubyCommand.Argument(name: "file", value: file, type: nil) + let repoArg = RubyCommand.Argument(name: "repo", value: repo, type: nil) + let repoPathArg = RubyCommand.Argument(name: "repo_path", value: repoPath, type: nil) + let endpointArg = RubyCommand.Argument(name: "endpoint", value: endpoint, type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let passwordArg = password.asRubyArgument(name: "password", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let propertiesArg = RubyCommand.Argument(name: "properties", value: properties, type: nil) + let sslPemFileArg = sslPemFile.asRubyArgument(name: "ssl_pem_file", type: nil) + let sslVerifyArg = sslVerify.asRubyArgument(name: "ssl_verify", type: nil) + let proxyUsernameArg = proxyUsername.asRubyArgument(name: "proxy_username", type: nil) + let proxyPasswordArg = proxyPassword.asRubyArgument(name: "proxy_password", type: nil) + let proxyAddressArg = proxyAddress.asRubyArgument(name: "proxy_address", type: nil) + let proxyPortArg = proxyPort.asRubyArgument(name: "proxy_port", type: nil) + let readTimeoutArg = readTimeout.asRubyArgument(name: "read_timeout", type: nil) + let array: [RubyCommand.Argument?] = [fileArg, + repoArg, + repoPathArg, + endpointArg, + usernameArg, + passwordArg, + apiKeyArg, + propertiesArg, + sslPemFileArg, + sslVerifyArg, + proxyUsernameArg, + proxyPasswordArg, + proxyAddressArg, + proxyPortArg, + readTimeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "artifactory", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Configures Xcode's Codesigning options + + - parameters: + - path: Path to your Xcode project + - useAutomaticSigning: Defines if project should use automatic signing + - teamId: Team ID, is used when upgrading project + - targets: Specify targets you want to toggle the signing mech. (default to all targets) + - codeSignIdentity: Code signing identity type (iPhone Developer, iPhone Distribution) + - profileName: Provisioning profile name to use for code signing + - profileUuid: Provisioning profile UUID to use for code signing + - bundleIdentifier: Application Product Bundle Identifier + + - returns: The current status (boolean) of codesigning after modification + + Configures Xcode's Codesigning options of all targets in the project + */ +public func automaticCodeSigning(path: String, + useAutomaticSigning: OptionalConfigValue = .fastlaneDefault(false), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + targets: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + codeSignIdentity: OptionalConfigValue = .fastlaneDefault(nil), + profileName: OptionalConfigValue = .fastlaneDefault(nil), + profileUuid: OptionalConfigValue = .fastlaneDefault(nil), + bundleIdentifier: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let useAutomaticSigningArg = useAutomaticSigning.asRubyArgument(name: "use_automatic_signing", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let targetsArg = targets.asRubyArgument(name: "targets", type: nil) + let codeSignIdentityArg = codeSignIdentity.asRubyArgument(name: "code_sign_identity", type: nil) + let profileNameArg = profileName.asRubyArgument(name: "profile_name", type: nil) + let profileUuidArg = profileUuid.asRubyArgument(name: "profile_uuid", type: nil) + let bundleIdentifierArg = bundleIdentifier.asRubyArgument(name: "bundle_identifier", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + useAutomaticSigningArg, + teamIdArg, + targetsArg, + codeSignIdentityArg, + profileNameArg, + profileUuidArg, + bundleIdentifierArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "automatic_code_signing", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action backs up your file to "[path].back" + + - parameter path: Path to the file you want to backup + */ +public func backupFile(path: String) { + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "backup_file", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Save your [zipped] xcarchive elsewhere from default path + + - parameters: + - xcarchive: Path to your xcarchive file. Optional if you use the `xcodebuild` action + - destination: Where your archive will be placed + - zip: Enable compression of the archive + - zipFilename: Filename of the compressed archive. Will be appended by `.xcarchive.zip`. Default value is the output xcarchive filename + - versioned: Create a versioned (date and app version) subfolder where to put the archive + */ +public func backupXcarchive(xcarchive: String, + destination: String, + zip: OptionalConfigValue = .fastlaneDefault(true), + zipFilename: OptionalConfigValue = .fastlaneDefault(nil), + versioned: OptionalConfigValue = .fastlaneDefault(true)) +{ + let xcarchiveArg = RubyCommand.Argument(name: "xcarchive", value: xcarchive, type: nil) + let destinationArg = RubyCommand.Argument(name: "destination", value: destination, type: nil) + let zipArg = zip.asRubyArgument(name: "zip", type: nil) + let zipFilenameArg = zipFilename.asRubyArgument(name: "zip_filename", type: nil) + let versionedArg = versioned.asRubyArgument(name: "versioned", type: nil) + let array: [RubyCommand.Argument?] = [xcarchiveArg, + destinationArg, + zipArg, + zipFilenameArg, + versionedArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "backup_xcarchive", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Automatically add a badge to your app icon + + - parameters: + - dark: Adds a dark flavored badge on top of your icon + - custom: Add your custom overlay/badge image + - noBadge: Hides the beta badge + - shield: Add a shield to your app icon from shields.io + - alpha: Adds and alpha badge instead of the default beta one + - path: Sets the root path to look for AppIcons + - shieldIoTimeout: Set custom duration for the timeout of the shields.io request in seconds + - glob: Glob pattern for finding image files + - alphaChannel: Keeps/adds an alpha channel to the icon (useful for android icons) + - shieldGravity: Position of shield on icon. Default: North - Choices include: NorthWest, North, NorthEast, West, Center, East, SouthWest, South, SouthEast + - shieldNoResize: Shield image will no longer be resized to aspect fill the full icon. Instead it will only be shrunk to not exceed the icon graphic + + Please use the [badge plugin](https://github.com/HazAT/fastlane-plugin-badge) instead. + This action will add a light/dark badge onto your app icon. + You can also provide your custom badge/overlay or add a shield for more customization. + More info: [https://github.com/HazAT/badge](https://github.com/HazAT/badge) + **Note**: If you want to reset the badge back to default, you can use `sh 'git checkout -- /Assets.xcassets/'`. + */ +public func badge(dark: OptionalConfigValue = .fastlaneDefault(nil), + custom: OptionalConfigValue = .fastlaneDefault(nil), + noBadge: OptionalConfigValue = .fastlaneDefault(nil), + shield: OptionalConfigValue = .fastlaneDefault(nil), + alpha: OptionalConfigValue = .fastlaneDefault(nil), + path: String = ".", + shieldIoTimeout: OptionalConfigValue = .fastlaneDefault(nil), + glob: OptionalConfigValue = .fastlaneDefault(nil), + alphaChannel: OptionalConfigValue = .fastlaneDefault(nil), + shieldGravity: OptionalConfigValue = .fastlaneDefault(nil), + shieldNoResize: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let darkArg = dark.asRubyArgument(name: "dark", type: nil) + let customArg = custom.asRubyArgument(name: "custom", type: nil) + let noBadgeArg = noBadge.asRubyArgument(name: "no_badge", type: nil) + let shieldArg = shield.asRubyArgument(name: "shield", type: nil) + let alphaArg = alpha.asRubyArgument(name: "alpha", type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let shieldIoTimeoutArg = shieldIoTimeout.asRubyArgument(name: "shield_io_timeout", type: nil) + let globArg = glob.asRubyArgument(name: "glob", type: nil) + let alphaChannelArg = alphaChannel.asRubyArgument(name: "alpha_channel", type: nil) + let shieldGravityArg = shieldGravity.asRubyArgument(name: "shield_gravity", type: nil) + let shieldNoResizeArg = shieldNoResize.asRubyArgument(name: "shield_no_resize", type: nil) + let array: [RubyCommand.Argument?] = [darkArg, + customArg, + noBadgeArg, + shieldArg, + alphaArg, + pathArg, + shieldIoTimeoutArg, + globArg, + alphaChannelArg, + shieldGravityArg, + shieldNoResizeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "badge", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generate and upload an ipa file to appetize.io + + - parameters: + - xcodebuild: Parameters that are passed to the xcodebuild action + - scheme: The scheme to build. Can also be passed using the `xcodebuild` parameter + - apiToken: Appetize.io API Token + - publicKey: If not provided, a new app will be created. If provided, the existing build will be overwritten + - note: Notes you wish to add to the uploaded app + - timeout: The number of seconds to wait until automatically ending the session due to user inactivity. Must be 30, 60, 90, 120, 180, 300, 600, 1800, 3600 or 7200. Default is 120 + + This should be called from danger. + More information in the [device_grid guide](https://github.com/fastlane/fastlane/blob/master/fastlane/lib/fastlane/actions/device_grid/README.md). + */ +public func buildAndUploadToAppetize(xcodebuild: [String: Any] = [:], + scheme: OptionalConfigValue = .fastlaneDefault(nil), + apiToken: String, + publicKey: OptionalConfigValue = .fastlaneDefault(nil), + note: OptionalConfigValue = .fastlaneDefault(nil), + timeout: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let xcodebuildArg = RubyCommand.Argument(name: "xcodebuild", value: xcodebuild, type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let publicKeyArg = publicKey.asRubyArgument(name: "public_key", type: nil) + let noteArg = note.asRubyArgument(name: "note", type: nil) + let timeoutArg = timeout.asRubyArgument(name: "timeout", type: nil) + let array: [RubyCommand.Argument?] = [xcodebuildArg, + schemeArg, + apiTokenArg, + publicKeyArg, + noteArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "build_and_upload_to_appetize", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `gradle` action + + - parameters: + - task: The gradle task you want to execute, e.g. `assemble`, `bundle` or `test`. For tasks such as `assembleMyFlavorRelease` you should use gradle(task: 'assemble', flavor: 'Myflavor', build_type: 'Release') + - flavor: The flavor that you want the task for, e.g. `MyFlavor`. If you are running the `assemble` task in a multi-flavor project, and you rely on Actions.lane_context[SharedValues::GRADLE_APK_OUTPUT_PATH] then you must specify a flavor here or else this value will be undefined + - buildType: The build type that you want the task for, e.g. `Release`. Useful for some tasks such as `assemble` + - tasks: The multiple gradle tasks that you want to execute, e.g. `[assembleDebug, bundleDebug]` + - flags: All parameter flags you want to pass to the gradle command, e.g. `--exitcode --xml file.xml` + - projectDir: The root directory of the gradle project + - gradlePath: The path to your `gradlew`. If you specify a relative path, it is assumed to be relative to the `project_dir` + - properties: Gradle properties to be exposed to the gradle script + - systemProperties: Gradle system properties to be exposed to the gradle script + - serial: Android serial, which device should be used for this command + - printCommand: Control whether the generated Gradle command is printed as output before running it (true/false) + - printCommandOutput: Control whether the output produced by given Gradle command is printed while running (true/false) + + - returns: The output of running the gradle task + + Run `./gradlew tasks` to get a list of all available gradle tasks for your project + */ +public func buildAndroidApp(task: OptionalConfigValue = .fastlaneDefault(nil), + flavor: OptionalConfigValue = .fastlaneDefault(nil), + buildType: OptionalConfigValue = .fastlaneDefault(nil), + tasks: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + flags: OptionalConfigValue = .fastlaneDefault(nil), + projectDir: String = ".", + gradlePath: OptionalConfigValue = .fastlaneDefault(nil), + properties: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + systemProperties: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + serial: String = "", + printCommand: OptionalConfigValue = .fastlaneDefault(true), + printCommandOutput: OptionalConfigValue = .fastlaneDefault(true)) +{ + let taskArg = task.asRubyArgument(name: "task", type: nil) + let flavorArg = flavor.asRubyArgument(name: "flavor", type: nil) + let buildTypeArg = buildType.asRubyArgument(name: "build_type", type: nil) + let tasksArg = tasks.asRubyArgument(name: "tasks", type: nil) + let flagsArg = flags.asRubyArgument(name: "flags", type: nil) + let projectDirArg = RubyCommand.Argument(name: "project_dir", value: projectDir, type: nil) + let gradlePathArg = gradlePath.asRubyArgument(name: "gradle_path", type: nil) + let propertiesArg = properties.asRubyArgument(name: "properties", type: nil) + let systemPropertiesArg = systemProperties.asRubyArgument(name: "system_properties", type: nil) + let serialArg = RubyCommand.Argument(name: "serial", value: serial, type: nil) + let printCommandArg = printCommand.asRubyArgument(name: "print_command", type: nil) + let printCommandOutputArg = printCommandOutput.asRubyArgument(name: "print_command_output", type: nil) + let array: [RubyCommand.Argument?] = [taskArg, + flavorArg, + buildTypeArg, + tasksArg, + flagsArg, + projectDirArg, + gradlePathArg, + propertiesArg, + systemPropertiesArg, + serialArg, + printCommandArg, + printCommandOutputArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "build_android_app", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Easily build and sign your app (via _gym_) + + - parameters: + - workspace: Path to the workspace file + - project: Path to the project file + - scheme: The project's scheme. Make sure it's marked as `Shared` + - clean: Should the project be cleaned before building it? + - outputDirectory: The directory in which the ipa file should be stored in + - outputName: The name of the resulting ipa file + - configuration: The configuration to use when building the app. Defaults to 'Release' + - silent: Hide all information that's not necessary while building + - codesigningIdentity: The name of the code signing identity to use. It has to match the name exactly. e.g. 'iPhone Distribution: SunApps GmbH' + - skipPackageIpa: Should we skip packaging the ipa? + - skipPackagePkg: Should we skip packaging the pkg? + - includeSymbols: Should the ipa file include symbols? + - includeBitcode: Should the ipa file include bitcode? + - exportMethod: Method used to export the archive. Valid values are: app-store, validation, ad-hoc, package, enterprise, development, developer-id and mac-application + - exportOptions: Path to an export options plist or a hash with export options. Use 'xcodebuild -help' to print the full set of available options + - exportXcargs: Pass additional arguments to xcodebuild for the package phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - skipBuildArchive: Export ipa from previously built xcarchive. Uses archive_path as source + - skipArchive: After building, don't archive, effectively not including -archivePath param + - skipCodesigning: Build without codesigning + - catalystPlatform: Platform to build when using a Catalyst enabled app. Valid values are: ios, macos + - installerCertName: Full name of 3rd Party Mac Developer Installer or Developer ID Installer certificate. Example: `3rd Party Mac Developer Installer: Your Company (ABC1234XWYZ)` + - buildPath: The directory in which the archive should be stored in + - archivePath: The path to the created archive + - derivedDataPath: The directory where built products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - resultBundlePath: Path to the result bundle directory to create. Ignored if `result_bundle` if false + - buildlogPath: The directory where to store the build log + - sdk: The SDK that should be used for building the application + - toolchain: The toolchain that should be used for building the application (e.g. com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a) + - destination: Use a custom destination for building the app + - exportTeamId: Optional: Sometimes you need to specify a team id when exporting the ipa file + - xcargs: Pass additional arguments to xcodebuild for the build phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - disableXcpretty: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Disable xcpretty formatting of build output + - xcprettyTestFormat: Use the test (RSpec style) format for build output + - xcprettyFormatter: A custom xcpretty formatter to use + - xcprettyReportJunit: Have xcpretty create a JUnit-style XML report at the provided path + - xcprettyReportHtml: Have xcpretty create a simple HTML report at the provided path + - xcprettyReportJson: Have xcpretty create a JSON compilation database at the provided path + - xcprettyUtf: Have xcpretty use unicode encoding when reporting builds + - analyzeBuildTime: Analyze the project build time and store the output in 'culprits.txt' file + - skipProfileDetection: Do not try to build a profile mapping from the xcodeproj. Match or a manually provided mapping should be used + - xcodebuildCommand: Allows for override of the default `xcodebuild` command + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - useSystemScm: Lets xcodebuild use system's scm configuration + + - returns: The absolute path to the generated ipa file + + More information: https://fastlane.tools/gym + */ +@discardableResult public func buildApp(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + clean: OptionalConfigValue = .fastlaneDefault(false), + outputDirectory: String = ".", + outputName: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + silent: OptionalConfigValue = .fastlaneDefault(false), + codesigningIdentity: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageIpa: OptionalConfigValue = .fastlaneDefault(false), + skipPackagePkg: OptionalConfigValue = .fastlaneDefault(false), + includeSymbols: OptionalConfigValue = .fastlaneDefault(nil), + includeBitcode: OptionalConfigValue = .fastlaneDefault(nil), + exportMethod: OptionalConfigValue = .fastlaneDefault(nil), + exportOptions: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + exportXcargs: OptionalConfigValue = .fastlaneDefault(nil), + skipBuildArchive: OptionalConfigValue = .fastlaneDefault(nil), + skipArchive: OptionalConfigValue = .fastlaneDefault(nil), + skipCodesigning: OptionalConfigValue = .fastlaneDefault(nil), + catalystPlatform: OptionalConfigValue = .fastlaneDefault(nil), + installerCertName: OptionalConfigValue = .fastlaneDefault(nil), + buildPath: OptionalConfigValue = .fastlaneDefault(nil), + archivePath: OptionalConfigValue = .fastlaneDefault(nil), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + resultBundle: OptionalConfigValue = .fastlaneDefault(false), + resultBundlePath: OptionalConfigValue = .fastlaneDefault(nil), + buildlogPath: String = "~/Library/Logs/gym", + sdk: OptionalConfigValue = .fastlaneDefault(nil), + toolchain: OptionalConfigValue = .fastlaneDefault(nil), + destination: OptionalConfigValue = .fastlaneDefault(nil), + exportTeamId: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(nil), + xcodebuildFormatter: String = "xcbeautify", + disableXcpretty: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyTestFormat: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyFormatter: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportJunit: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportHtml: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportJson: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyUtf: OptionalConfigValue = .fastlaneDefault(nil), + analyzeBuildTime: OptionalConfigValue = .fastlaneDefault(nil), + skipProfileDetection: OptionalConfigValue = .fastlaneDefault(false), + xcodebuildCommand: String = "xcodebuild", + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(false), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(false), + useSystemScm: OptionalConfigValue = .fastlaneDefault(false)) -> String +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputNameArg = outputName.asRubyArgument(name: "output_name", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let codesigningIdentityArg = codesigningIdentity.asRubyArgument(name: "codesigning_identity", type: nil) + let skipPackageIpaArg = skipPackageIpa.asRubyArgument(name: "skip_package_ipa", type: nil) + let skipPackagePkgArg = skipPackagePkg.asRubyArgument(name: "skip_package_pkg", type: nil) + let includeSymbolsArg = includeSymbols.asRubyArgument(name: "include_symbols", type: nil) + let includeBitcodeArg = includeBitcode.asRubyArgument(name: "include_bitcode", type: nil) + let exportMethodArg = exportMethod.asRubyArgument(name: "export_method", type: nil) + let exportOptionsArg = exportOptions.asRubyArgument(name: "export_options", type: nil) + let exportXcargsArg = exportXcargs.asRubyArgument(name: "export_xcargs", type: nil) + let skipBuildArchiveArg = skipBuildArchive.asRubyArgument(name: "skip_build_archive", type: nil) + let skipArchiveArg = skipArchive.asRubyArgument(name: "skip_archive", type: nil) + let skipCodesigningArg = skipCodesigning.asRubyArgument(name: "skip_codesigning", type: nil) + let catalystPlatformArg = catalystPlatform.asRubyArgument(name: "catalyst_platform", type: nil) + let installerCertNameArg = installerCertName.asRubyArgument(name: "installer_cert_name", type: nil) + let buildPathArg = buildPath.asRubyArgument(name: "build_path", type: nil) + let archivePathArg = archivePath.asRubyArgument(name: "archive_path", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let resultBundlePathArg = resultBundlePath.asRubyArgument(name: "result_bundle_path", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let toolchainArg = toolchain.asRubyArgument(name: "toolchain", type: nil) + let destinationArg = destination.asRubyArgument(name: "destination", type: nil) + let exportTeamIdArg = exportTeamId.asRubyArgument(name: "export_team_id", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let xcprettyTestFormatArg = xcprettyTestFormat.asRubyArgument(name: "xcpretty_test_format", type: nil) + let xcprettyFormatterArg = xcprettyFormatter.asRubyArgument(name: "xcpretty_formatter", type: nil) + let xcprettyReportJunitArg = xcprettyReportJunit.asRubyArgument(name: "xcpretty_report_junit", type: nil) + let xcprettyReportHtmlArg = xcprettyReportHtml.asRubyArgument(name: "xcpretty_report_html", type: nil) + let xcprettyReportJsonArg = xcprettyReportJson.asRubyArgument(name: "xcpretty_report_json", type: nil) + let xcprettyUtfArg = xcprettyUtf.asRubyArgument(name: "xcpretty_utf", type: nil) + let analyzeBuildTimeArg = analyzeBuildTime.asRubyArgument(name: "analyze_build_time", type: nil) + let skipProfileDetectionArg = skipProfileDetection.asRubyArgument(name: "skip_profile_detection", type: nil) + let xcodebuildCommandArg = RubyCommand.Argument(name: "xcodebuild_command", value: xcodebuildCommand, type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + schemeArg, + cleanArg, + outputDirectoryArg, + outputNameArg, + configurationArg, + silentArg, + codesigningIdentityArg, + skipPackageIpaArg, + skipPackagePkgArg, + includeSymbolsArg, + includeBitcodeArg, + exportMethodArg, + exportOptionsArg, + exportXcargsArg, + skipBuildArchiveArg, + skipArchiveArg, + skipCodesigningArg, + catalystPlatformArg, + installerCertNameArg, + buildPathArg, + archivePathArg, + derivedDataPathArg, + resultBundleArg, + resultBundlePathArg, + buildlogPathArg, + sdkArg, + toolchainArg, + destinationArg, + exportTeamIdArg, + xcargsArg, + xcconfigArg, + suppressXcodeOutputArg, + xcodebuildFormatterArg, + disableXcprettyArg, + xcprettyTestFormatArg, + xcprettyFormatterArg, + xcprettyReportJunitArg, + xcprettyReportHtmlArg, + xcprettyReportJsonArg, + xcprettyUtfArg, + analyzeBuildTimeArg, + skipProfileDetectionArg, + xcodebuildCommandArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "build_app", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Alias for the `build_app` action but only for iOS + + - parameters: + - workspace: Path to the workspace file + - project: Path to the project file + - scheme: The project's scheme. Make sure it's marked as `Shared` + - clean: Should the project be cleaned before building it? + - outputDirectory: The directory in which the ipa file should be stored in + - outputName: The name of the resulting ipa file + - configuration: The configuration to use when building the app. Defaults to 'Release' + - silent: Hide all information that's not necessary while building + - codesigningIdentity: The name of the code signing identity to use. It has to match the name exactly. e.g. 'iPhone Distribution: SunApps GmbH' + - skipPackageIpa: Should we skip packaging the ipa? + - includeSymbols: Should the ipa file include symbols? + - includeBitcode: Should the ipa file include bitcode? + - exportMethod: Method used to export the archive. Valid values are: app-store, validation, ad-hoc, package, enterprise, development, developer-id and mac-application + - exportOptions: Path to an export options plist or a hash with export options. Use 'xcodebuild -help' to print the full set of available options + - exportXcargs: Pass additional arguments to xcodebuild for the package phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - skipBuildArchive: Export ipa from previously built xcarchive. Uses archive_path as source + - skipArchive: After building, don't archive, effectively not including -archivePath param + - skipCodesigning: Build without codesigning + - buildPath: The directory in which the archive should be stored in + - archivePath: The path to the created archive + - derivedDataPath: The directory where built products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - resultBundlePath: Path to the result bundle directory to create. Ignored if `result_bundle` if false + - buildlogPath: The directory where to store the build log + - sdk: The SDK that should be used for building the application + - toolchain: The toolchain that should be used for building the application (e.g. com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a) + - destination: Use a custom destination for building the app + - exportTeamId: Optional: Sometimes you need to specify a team id when exporting the ipa file + - xcargs: Pass additional arguments to xcodebuild for the build phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - disableXcpretty: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Disable xcpretty formatting of build output + - xcprettyTestFormat: Use the test (RSpec style) format for build output + - xcprettyFormatter: A custom xcpretty formatter to use + - xcprettyReportJunit: Have xcpretty create a JUnit-style XML report at the provided path + - xcprettyReportHtml: Have xcpretty create a simple HTML report at the provided path + - xcprettyReportJson: Have xcpretty create a JSON compilation database at the provided path + - xcprettyUtf: Have xcpretty use unicode encoding when reporting builds + - analyzeBuildTime: Analyze the project build time and store the output in 'culprits.txt' file + - skipProfileDetection: Do not try to build a profile mapping from the xcodeproj. Match or a manually provided mapping should be used + - xcodebuildCommand: Allows for override of the default `xcodebuild` command + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - useSystemScm: Lets xcodebuild use system's scm configuration + + - returns: The absolute path to the generated ipa file + + More information: https://fastlane.tools/gym + */ +@discardableResult public func buildIosApp(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + clean: OptionalConfigValue = .fastlaneDefault(false), + outputDirectory: String = ".", + outputName: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + silent: OptionalConfigValue = .fastlaneDefault(false), + codesigningIdentity: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageIpa: OptionalConfigValue = .fastlaneDefault(false), + includeSymbols: OptionalConfigValue = .fastlaneDefault(nil), + includeBitcode: OptionalConfigValue = .fastlaneDefault(nil), + exportMethod: OptionalConfigValue = .fastlaneDefault(nil), + exportOptions: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + exportXcargs: OptionalConfigValue = .fastlaneDefault(nil), + skipBuildArchive: OptionalConfigValue = .fastlaneDefault(nil), + skipArchive: OptionalConfigValue = .fastlaneDefault(nil), + skipCodesigning: OptionalConfigValue = .fastlaneDefault(nil), + buildPath: OptionalConfigValue = .fastlaneDefault(nil), + archivePath: OptionalConfigValue = .fastlaneDefault(nil), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + resultBundle: OptionalConfigValue = .fastlaneDefault(false), + resultBundlePath: OptionalConfigValue = .fastlaneDefault(nil), + buildlogPath: String = "~/Library/Logs/gym", + sdk: OptionalConfigValue = .fastlaneDefault(nil), + toolchain: OptionalConfigValue = .fastlaneDefault(nil), + destination: OptionalConfigValue = .fastlaneDefault(nil), + exportTeamId: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(nil), + xcodebuildFormatter: String = "xcbeautify", + disableXcpretty: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyTestFormat: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyFormatter: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportJunit: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportHtml: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportJson: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyUtf: OptionalConfigValue = .fastlaneDefault(nil), + analyzeBuildTime: OptionalConfigValue = .fastlaneDefault(nil), + skipProfileDetection: OptionalConfigValue = .fastlaneDefault(false), + xcodebuildCommand: String = "xcodebuild", + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(false), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(false), + useSystemScm: OptionalConfigValue = .fastlaneDefault(false)) -> String +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputNameArg = outputName.asRubyArgument(name: "output_name", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let codesigningIdentityArg = codesigningIdentity.asRubyArgument(name: "codesigning_identity", type: nil) + let skipPackageIpaArg = skipPackageIpa.asRubyArgument(name: "skip_package_ipa", type: nil) + let includeSymbolsArg = includeSymbols.asRubyArgument(name: "include_symbols", type: nil) + let includeBitcodeArg = includeBitcode.asRubyArgument(name: "include_bitcode", type: nil) + let exportMethodArg = exportMethod.asRubyArgument(name: "export_method", type: nil) + let exportOptionsArg = exportOptions.asRubyArgument(name: "export_options", type: nil) + let exportXcargsArg = exportXcargs.asRubyArgument(name: "export_xcargs", type: nil) + let skipBuildArchiveArg = skipBuildArchive.asRubyArgument(name: "skip_build_archive", type: nil) + let skipArchiveArg = skipArchive.asRubyArgument(name: "skip_archive", type: nil) + let skipCodesigningArg = skipCodesigning.asRubyArgument(name: "skip_codesigning", type: nil) + let buildPathArg = buildPath.asRubyArgument(name: "build_path", type: nil) + let archivePathArg = archivePath.asRubyArgument(name: "archive_path", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let resultBundlePathArg = resultBundlePath.asRubyArgument(name: "result_bundle_path", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let toolchainArg = toolchain.asRubyArgument(name: "toolchain", type: nil) + let destinationArg = destination.asRubyArgument(name: "destination", type: nil) + let exportTeamIdArg = exportTeamId.asRubyArgument(name: "export_team_id", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let xcprettyTestFormatArg = xcprettyTestFormat.asRubyArgument(name: "xcpretty_test_format", type: nil) + let xcprettyFormatterArg = xcprettyFormatter.asRubyArgument(name: "xcpretty_formatter", type: nil) + let xcprettyReportJunitArg = xcprettyReportJunit.asRubyArgument(name: "xcpretty_report_junit", type: nil) + let xcprettyReportHtmlArg = xcprettyReportHtml.asRubyArgument(name: "xcpretty_report_html", type: nil) + let xcprettyReportJsonArg = xcprettyReportJson.asRubyArgument(name: "xcpretty_report_json", type: nil) + let xcprettyUtfArg = xcprettyUtf.asRubyArgument(name: "xcpretty_utf", type: nil) + let analyzeBuildTimeArg = analyzeBuildTime.asRubyArgument(name: "analyze_build_time", type: nil) + let skipProfileDetectionArg = skipProfileDetection.asRubyArgument(name: "skip_profile_detection", type: nil) + let xcodebuildCommandArg = RubyCommand.Argument(name: "xcodebuild_command", value: xcodebuildCommand, type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + schemeArg, + cleanArg, + outputDirectoryArg, + outputNameArg, + configurationArg, + silentArg, + codesigningIdentityArg, + skipPackageIpaArg, + includeSymbolsArg, + includeBitcodeArg, + exportMethodArg, + exportOptionsArg, + exportXcargsArg, + skipBuildArchiveArg, + skipArchiveArg, + skipCodesigningArg, + buildPathArg, + archivePathArg, + derivedDataPathArg, + resultBundleArg, + resultBundlePathArg, + buildlogPathArg, + sdkArg, + toolchainArg, + destinationArg, + exportTeamIdArg, + xcargsArg, + xcconfigArg, + suppressXcodeOutputArg, + xcodebuildFormatterArg, + disableXcprettyArg, + xcprettyTestFormatArg, + xcprettyFormatterArg, + xcprettyReportJunitArg, + xcprettyReportHtmlArg, + xcprettyReportJsonArg, + xcprettyUtfArg, + analyzeBuildTimeArg, + skipProfileDetectionArg, + xcodebuildCommandArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "build_ios_app", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Alias for the `build_app` action but only for macOS + + - parameters: + - workspace: Path to the workspace file + - project: Path to the project file + - scheme: The project's scheme. Make sure it's marked as `Shared` + - clean: Should the project be cleaned before building it? + - outputDirectory: The directory in which the ipa file should be stored in + - outputName: The name of the resulting ipa file + - configuration: The configuration to use when building the app. Defaults to 'Release' + - silent: Hide all information that's not necessary while building + - codesigningIdentity: The name of the code signing identity to use. It has to match the name exactly. e.g. 'iPhone Distribution: SunApps GmbH' + - skipPackagePkg: Should we skip packaging the pkg? + - includeSymbols: Should the ipa file include symbols? + - includeBitcode: Should the ipa file include bitcode? + - exportMethod: Method used to export the archive. Valid values are: app-store, validation, ad-hoc, package, enterprise, development, developer-id and mac-application + - exportOptions: Path to an export options plist or a hash with export options. Use 'xcodebuild -help' to print the full set of available options + - exportXcargs: Pass additional arguments to xcodebuild for the package phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - skipBuildArchive: Export ipa from previously built xcarchive. Uses archive_path as source + - skipArchive: After building, don't archive, effectively not including -archivePath param + - skipCodesigning: Build without codesigning + - installerCertName: Full name of 3rd Party Mac Developer Installer or Developer ID Installer certificate. Example: `3rd Party Mac Developer Installer: Your Company (ABC1234XWYZ)` + - buildPath: The directory in which the archive should be stored in + - archivePath: The path to the created archive + - derivedDataPath: The directory where built products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - resultBundlePath: Path to the result bundle directory to create. Ignored if `result_bundle` if false + - buildlogPath: The directory where to store the build log + - sdk: The SDK that should be used for building the application + - toolchain: The toolchain that should be used for building the application (e.g. com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a) + - destination: Use a custom destination for building the app + - exportTeamId: Optional: Sometimes you need to specify a team id when exporting the ipa file + - xcargs: Pass additional arguments to xcodebuild for the build phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - disableXcpretty: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Disable xcpretty formatting of build output + - xcprettyTestFormat: Use the test (RSpec style) format for build output + - xcprettyFormatter: A custom xcpretty formatter to use + - xcprettyReportJunit: Have xcpretty create a JUnit-style XML report at the provided path + - xcprettyReportHtml: Have xcpretty create a simple HTML report at the provided path + - xcprettyReportJson: Have xcpretty create a JSON compilation database at the provided path + - xcprettyUtf: Have xcpretty use unicode encoding when reporting builds + - analyzeBuildTime: Analyze the project build time and store the output in 'culprits.txt' file + - skipProfileDetection: Do not try to build a profile mapping from the xcodeproj. Match or a manually provided mapping should be used + - xcodebuildCommand: Allows for override of the default `xcodebuild` command + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - useSystemScm: Lets xcodebuild use system's scm configuration + + - returns: The absolute path to the generated ipa file + + More information: https://fastlane.tools/gym + */ +@discardableResult public func buildMacApp(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + clean: OptionalConfigValue = .fastlaneDefault(false), + outputDirectory: String = ".", + outputName: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + silent: OptionalConfigValue = .fastlaneDefault(false), + codesigningIdentity: OptionalConfigValue = .fastlaneDefault(nil), + skipPackagePkg: OptionalConfigValue = .fastlaneDefault(false), + includeSymbols: OptionalConfigValue = .fastlaneDefault(nil), + includeBitcode: OptionalConfigValue = .fastlaneDefault(nil), + exportMethod: OptionalConfigValue = .fastlaneDefault(nil), + exportOptions: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + exportXcargs: OptionalConfigValue = .fastlaneDefault(nil), + skipBuildArchive: OptionalConfigValue = .fastlaneDefault(nil), + skipArchive: OptionalConfigValue = .fastlaneDefault(nil), + skipCodesigning: OptionalConfigValue = .fastlaneDefault(nil), + installerCertName: OptionalConfigValue = .fastlaneDefault(nil), + buildPath: OptionalConfigValue = .fastlaneDefault(nil), + archivePath: OptionalConfigValue = .fastlaneDefault(nil), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + resultBundle: OptionalConfigValue = .fastlaneDefault(false), + resultBundlePath: OptionalConfigValue = .fastlaneDefault(nil), + buildlogPath: String = "~/Library/Logs/gym", + sdk: OptionalConfigValue = .fastlaneDefault(nil), + toolchain: OptionalConfigValue = .fastlaneDefault(nil), + destination: OptionalConfigValue = .fastlaneDefault(nil), + exportTeamId: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(nil), + xcodebuildFormatter: String = "xcbeautify", + disableXcpretty: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyTestFormat: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyFormatter: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportJunit: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportHtml: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyReportJson: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyUtf: OptionalConfigValue = .fastlaneDefault(nil), + analyzeBuildTime: OptionalConfigValue = .fastlaneDefault(nil), + skipProfileDetection: OptionalConfigValue = .fastlaneDefault(false), + xcodebuildCommand: String = "xcodebuild", + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(false), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(false), + useSystemScm: OptionalConfigValue = .fastlaneDefault(false)) -> String +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputNameArg = outputName.asRubyArgument(name: "output_name", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let codesigningIdentityArg = codesigningIdentity.asRubyArgument(name: "codesigning_identity", type: nil) + let skipPackagePkgArg = skipPackagePkg.asRubyArgument(name: "skip_package_pkg", type: nil) + let includeSymbolsArg = includeSymbols.asRubyArgument(name: "include_symbols", type: nil) + let includeBitcodeArg = includeBitcode.asRubyArgument(name: "include_bitcode", type: nil) + let exportMethodArg = exportMethod.asRubyArgument(name: "export_method", type: nil) + let exportOptionsArg = exportOptions.asRubyArgument(name: "export_options", type: nil) + let exportXcargsArg = exportXcargs.asRubyArgument(name: "export_xcargs", type: nil) + let skipBuildArchiveArg = skipBuildArchive.asRubyArgument(name: "skip_build_archive", type: nil) + let skipArchiveArg = skipArchive.asRubyArgument(name: "skip_archive", type: nil) + let skipCodesigningArg = skipCodesigning.asRubyArgument(name: "skip_codesigning", type: nil) + let installerCertNameArg = installerCertName.asRubyArgument(name: "installer_cert_name", type: nil) + let buildPathArg = buildPath.asRubyArgument(name: "build_path", type: nil) + let archivePathArg = archivePath.asRubyArgument(name: "archive_path", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let resultBundlePathArg = resultBundlePath.asRubyArgument(name: "result_bundle_path", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let toolchainArg = toolchain.asRubyArgument(name: "toolchain", type: nil) + let destinationArg = destination.asRubyArgument(name: "destination", type: nil) + let exportTeamIdArg = exportTeamId.asRubyArgument(name: "export_team_id", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let xcprettyTestFormatArg = xcprettyTestFormat.asRubyArgument(name: "xcpretty_test_format", type: nil) + let xcprettyFormatterArg = xcprettyFormatter.asRubyArgument(name: "xcpretty_formatter", type: nil) + let xcprettyReportJunitArg = xcprettyReportJunit.asRubyArgument(name: "xcpretty_report_junit", type: nil) + let xcprettyReportHtmlArg = xcprettyReportHtml.asRubyArgument(name: "xcpretty_report_html", type: nil) + let xcprettyReportJsonArg = xcprettyReportJson.asRubyArgument(name: "xcpretty_report_json", type: nil) + let xcprettyUtfArg = xcprettyUtf.asRubyArgument(name: "xcpretty_utf", type: nil) + let analyzeBuildTimeArg = analyzeBuildTime.asRubyArgument(name: "analyze_build_time", type: nil) + let skipProfileDetectionArg = skipProfileDetection.asRubyArgument(name: "skip_profile_detection", type: nil) + let xcodebuildCommandArg = RubyCommand.Argument(name: "xcodebuild_command", value: xcodebuildCommand, type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + schemeArg, + cleanArg, + outputDirectoryArg, + outputNameArg, + configurationArg, + silentArg, + codesigningIdentityArg, + skipPackagePkgArg, + includeSymbolsArg, + includeBitcodeArg, + exportMethodArg, + exportOptionsArg, + exportXcargsArg, + skipBuildArchiveArg, + skipArchiveArg, + skipCodesigningArg, + installerCertNameArg, + buildPathArg, + archivePathArg, + derivedDataPathArg, + resultBundleArg, + resultBundlePathArg, + buildlogPathArg, + sdkArg, + toolchainArg, + destinationArg, + exportTeamIdArg, + xcargsArg, + xcconfigArg, + suppressXcodeOutputArg, + xcodebuildFormatterArg, + disableXcprettyArg, + xcprettyTestFormatArg, + xcprettyFormatterArg, + xcprettyReportJunitArg, + xcprettyReportHtmlArg, + xcprettyReportJsonArg, + xcprettyUtfArg, + analyzeBuildTimeArg, + skipProfileDetectionArg, + xcodebuildCommandArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "build_mac_app", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + This action runs `bundle install` (if available) + + - parameters: + - binstubs: Generate bin stubs for bundled gems to ./bin + - clean: Run bundle clean automatically after install + - fullIndex: Use the rubygems modern index instead of the API endpoint + - gemfile: Use the specified gemfile instead of Gemfile + - jobs: Install gems using parallel workers + - local: Do not attempt to fetch gems remotely and use the gem cache instead + - deployment: Install using defaults tuned for deployment and CI environments + - noCache: Don't update the existing gem cache + - noPrune: Don't remove stale gems from the cache + - path: Specify a different path than the system default ($BUNDLE_PATH or $GEM_HOME). Bundler will remember this value for future installs on this machine + - system: Install to the system location ($BUNDLE_PATH or $GEM_HOME) even if the bundle was previously installed somewhere else for this application + - quiet: Only output warnings and errors + - retry: Retry network and git requests that have failed + - shebang: Specify a different shebang executable name than the default (usually 'ruby') + - standalone: Make a bundle that can work without the Bundler runtime + - trustPolicy: Sets level of security when dealing with signed gems. Accepts `LowSecurity`, `MediumSecurity` and `HighSecurity` as values + - without: Exclude gems that are part of the specified named group + - with: Include gems that are part of the specified named group + - frozen: Don't allow the Gemfile.lock to be updated after install + - redownload: Force download every gem, even if the required versions are already available locally + */ +public func bundleInstall(binstubs: OptionalConfigValue = .fastlaneDefault(nil), + clean: OptionalConfigValue = .fastlaneDefault(false), + fullIndex: OptionalConfigValue = .fastlaneDefault(false), + gemfile: OptionalConfigValue = .fastlaneDefault(nil), + jobs: OptionalConfigValue = .fastlaneDefault(nil), + local: OptionalConfigValue = .fastlaneDefault(false), + deployment: OptionalConfigValue = .fastlaneDefault(false), + noCache: OptionalConfigValue = .fastlaneDefault(false), + noPrune: OptionalConfigValue = .fastlaneDefault(false), + path: OptionalConfigValue = .fastlaneDefault(nil), + system: OptionalConfigValue = .fastlaneDefault(false), + quiet: OptionalConfigValue = .fastlaneDefault(false), + retry: OptionalConfigValue = .fastlaneDefault(nil), + shebang: OptionalConfigValue = .fastlaneDefault(nil), + standalone: OptionalConfigValue = .fastlaneDefault(nil), + trustPolicy: OptionalConfigValue = .fastlaneDefault(nil), + without: OptionalConfigValue = .fastlaneDefault(nil), + with: OptionalConfigValue = .fastlaneDefault(nil), + frozen: OptionalConfigValue = .fastlaneDefault(false), + redownload: OptionalConfigValue = .fastlaneDefault(false)) +{ + let binstubsArg = binstubs.asRubyArgument(name: "binstubs", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let fullIndexArg = fullIndex.asRubyArgument(name: "full_index", type: nil) + let gemfileArg = gemfile.asRubyArgument(name: "gemfile", type: nil) + let jobsArg = jobs.asRubyArgument(name: "jobs", type: nil) + let localArg = local.asRubyArgument(name: "local", type: nil) + let deploymentArg = deployment.asRubyArgument(name: "deployment", type: nil) + let noCacheArg = noCache.asRubyArgument(name: "no_cache", type: nil) + let noPruneArg = noPrune.asRubyArgument(name: "no_prune", type: nil) + let pathArg = path.asRubyArgument(name: "path", type: nil) + let systemArg = system.asRubyArgument(name: "system", type: nil) + let quietArg = quiet.asRubyArgument(name: "quiet", type: nil) + let retryArg = retry.asRubyArgument(name: "retry", type: nil) + let shebangArg = shebang.asRubyArgument(name: "shebang", type: nil) + let standaloneArg = standalone.asRubyArgument(name: "standalone", type: nil) + let trustPolicyArg = trustPolicy.asRubyArgument(name: "trust_policy", type: nil) + let withoutArg = without.asRubyArgument(name: "without", type: nil) + let withArg = with.asRubyArgument(name: "with", type: nil) + let frozenArg = frozen.asRubyArgument(name: "frozen", type: nil) + let redownloadArg = redownload.asRubyArgument(name: "redownload", type: nil) + let array: [RubyCommand.Argument?] = [binstubsArg, + cleanArg, + fullIndexArg, + gemfileArg, + jobsArg, + localArg, + deploymentArg, + noCacheArg, + noPruneArg, + pathArg, + systemArg, + quietArg, + retryArg, + shebangArg, + standaloneArg, + trustPolicyArg, + withoutArg, + withArg, + frozenArg, + redownloadArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "bundle_install", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Automated localized screenshots of your Android app (via _screengrab_) + + - parameters: + - androidHome: Path to the root of your Android SDK installation, e.g. ~/tools/android-sdk-macosx + - buildToolsVersion: **DEPRECATED!** The Android build tools version to use, e.g. '23.0.2' + - locales: A list of locales which should be used + - clearPreviousScreenshots: Enabling this option will automatically clear previously generated screenshots before running screengrab + - outputDirectory: The directory where to store the screenshots + - skipOpenSummary: Don't open the summary after running _screengrab_ + - appPackageName: The package name of the app under test (e.g. com.yourcompany.yourapp) + - testsPackageName: The package name of the tests bundle (e.g. com.yourcompany.yourapp.test) + - useTestsInPackages: Only run tests in these Java packages + - useTestsInClasses: Only run tests in these Java classes + - launchArguments: Additional launch arguments + - testInstrumentationRunner: The fully qualified class name of your test instrumentation runner + - endingLocale: **DEPRECATED!** Return the device to this locale after running tests + - useAdbRoot: **DEPRECATED!** Restarts the adb daemon using `adb root` to allow access to screenshots directories on device. Use if getting 'Permission denied' errors + - appApkPath: The path to the APK for the app under test + - testsApkPath: The path to the APK for the tests bundle + - specificDevice: Use the device or emulator with the given serial number or qualifier + - deviceType: Type of device used for screenshots. Matches Google Play Types (phone, sevenInch, tenInch, tv, wear) + - exitOnTestFailure: Whether or not to exit Screengrab on test failure. Exiting on failure will not copy screenshots to local machine nor open screenshots summary + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - useTimestampSuffix: Add timestamp suffix to screenshot filename + - adbHost: Configure the host used by adb to connect, allows running on remote devices farm + */ +public func captureAndroidScreenshots(androidHome: OptionalConfigValue = .fastlaneDefault(nil), + buildToolsVersion: OptionalConfigValue = .fastlaneDefault(nil), + locales: [String] = ["en-US"], + clearPreviousScreenshots: OptionalConfigValue = .fastlaneDefault(false), + outputDirectory: String = "fastlane/metadata/android", + skipOpenSummary: OptionalConfigValue = .fastlaneDefault(false), + appPackageName: String, + testsPackageName: OptionalConfigValue = .fastlaneDefault(nil), + useTestsInPackages: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + useTestsInClasses: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + launchArguments: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + testInstrumentationRunner: String = "androidx.test.runner.AndroidJUnitRunner", + endingLocale: String = "en-US", + useAdbRoot: OptionalConfigValue = .fastlaneDefault(false), + appApkPath: OptionalConfigValue = .fastlaneDefault(nil), + testsApkPath: OptionalConfigValue = .fastlaneDefault(nil), + specificDevice: OptionalConfigValue = .fastlaneDefault(nil), + deviceType: String = "phone", + exitOnTestFailure: OptionalConfigValue = .fastlaneDefault(true), + reinstallApp: OptionalConfigValue = .fastlaneDefault(false), + useTimestampSuffix: OptionalConfigValue = .fastlaneDefault(true), + adbHost: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let androidHomeArg = androidHome.asRubyArgument(name: "android_home", type: nil) + let buildToolsVersionArg = buildToolsVersion.asRubyArgument(name: "build_tools_version", type: nil) + let localesArg = RubyCommand.Argument(name: "locales", value: locales, type: nil) + let clearPreviousScreenshotsArg = clearPreviousScreenshots.asRubyArgument(name: "clear_previous_screenshots", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let skipOpenSummaryArg = skipOpenSummary.asRubyArgument(name: "skip_open_summary", type: nil) + let appPackageNameArg = RubyCommand.Argument(name: "app_package_name", value: appPackageName, type: nil) + let testsPackageNameArg = testsPackageName.asRubyArgument(name: "tests_package_name", type: nil) + let useTestsInPackagesArg = useTestsInPackages.asRubyArgument(name: "use_tests_in_packages", type: nil) + let useTestsInClassesArg = useTestsInClasses.asRubyArgument(name: "use_tests_in_classes", type: nil) + let launchArgumentsArg = launchArguments.asRubyArgument(name: "launch_arguments", type: nil) + let testInstrumentationRunnerArg = RubyCommand.Argument(name: "test_instrumentation_runner", value: testInstrumentationRunner, type: nil) + let endingLocaleArg = RubyCommand.Argument(name: "ending_locale", value: endingLocale, type: nil) + let useAdbRootArg = useAdbRoot.asRubyArgument(name: "use_adb_root", type: nil) + let appApkPathArg = appApkPath.asRubyArgument(name: "app_apk_path", type: nil) + let testsApkPathArg = testsApkPath.asRubyArgument(name: "tests_apk_path", type: nil) + let specificDeviceArg = specificDevice.asRubyArgument(name: "specific_device", type: nil) + let deviceTypeArg = RubyCommand.Argument(name: "device_type", value: deviceType, type: nil) + let exitOnTestFailureArg = exitOnTestFailure.asRubyArgument(name: "exit_on_test_failure", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let useTimestampSuffixArg = useTimestampSuffix.asRubyArgument(name: "use_timestamp_suffix", type: nil) + let adbHostArg = adbHost.asRubyArgument(name: "adb_host", type: nil) + let array: [RubyCommand.Argument?] = [androidHomeArg, + buildToolsVersionArg, + localesArg, + clearPreviousScreenshotsArg, + outputDirectoryArg, + skipOpenSummaryArg, + appPackageNameArg, + testsPackageNameArg, + useTestsInPackagesArg, + useTestsInClassesArg, + launchArgumentsArg, + testInstrumentationRunnerArg, + endingLocaleArg, + useAdbRootArg, + appApkPathArg, + testsApkPathArg, + specificDeviceArg, + deviceTypeArg, + exitOnTestFailureArg, + reinstallAppArg, + useTimestampSuffixArg, + adbHostArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "capture_android_screenshots", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generate new localized screenshots on multiple devices (via _snapshot_) + + - parameters: + - workspace: Path the workspace file + - project: Path the project file + - xcargs: Pass additional arguments to xcodebuild for the test phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - devices: A list of devices you want to take the screenshots from + - languages: A list of languages which should be used + - launchArguments: A list of launch arguments which should be used + - outputDirectory: The directory where to store the screenshots + - outputSimulatorLogs: If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + - iosVersion: By default, the latest version should be used automatically. If you want to change it, do it here + - skipOpenSummary: Don't open the HTML summary after running _snapshot_ + - skipHelperVersionCheck: Do not check for most recent SnapshotHelper code + - clearPreviousScreenshots: Enabling this option will automatically clear previously generated screenshots before running snapshot + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - eraseSimulator: Enabling this option will automatically erase the simulator before running the application + - headless: Enabling this option will prevent displaying the simulator window + - overrideStatusBar: Enabling this option will automatically override the status bar to show 9:41 AM, full battery, and full reception (Adjust 'SNAPSHOT_SIMULATOR_WAIT_FOR_BOOT_TIMEOUT' environment variable if override status bar is not working. Might be because simulator is not fully booted. Defaults to 10 seconds) + - overrideStatusBarArguments: Fully customize the status bar by setting each option here. Requires `override_status_bar` to be set to `true`. See `xcrun simctl status_bar --help` + - localizeSimulator: Enabling this option will configure the Simulator's system language + - darkMode: Enabling this option will configure the Simulator to be in dark mode (false for light, true for dark) + - appIdentifier: The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + - addPhotos: A list of photos that should be added to the simulator before running the application + - addVideos: A list of videos that should be added to the simulator before running the application + - htmlTemplate: A path to screenshots.html template + - buildlogPath: The directory where to store the build log + - clean: Should the project be cleaned before building it? + - testWithoutBuilding: Test without building, requires a derived data path + - configuration: The configuration to use when building the app. Defaults to 'Release' + - sdk: The SDK that should be used for building the application + - scheme: The scheme you want to use, this must be the scheme for the UI Tests + - numberOfRetries: The number of times a test can fail before snapshot should stop retrying + - stopAfterFirstError: Should snapshot stop immediately after the tests completely failed on one device? + - derivedDataPath: The directory where build products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - testTargetName: The name of the target you want to test (if you desire to override the Target Application from Xcode) + - namespaceLogFiles: Separate the log files per device and per language + - concurrentSimulators: Take snapshots on multiple simulators concurrently. Note: This option is only applicable when running against Xcode 9 + - disableSlideToType: Disable the simulator from showing the 'Slide to type' prompt + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - testplan: The testplan associated with the scheme that should be used for testing + - onlyTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to run + - skipTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to skip + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - xcprettyArgs: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Additional xcpretty arguments + - disableXcpretty: Disable xcpretty formatting of build + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - useSystemScm: Lets xcodebuild use system's scm configuration + */ +public func captureIosScreenshots(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + devices: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + languages: [String] = ["en-US"], + launchArguments: [String] = [""], + outputDirectory: String = "screenshots", + outputSimulatorLogs: OptionalConfigValue = .fastlaneDefault(false), + iosVersion: OptionalConfigValue = .fastlaneDefault(nil), + skipOpenSummary: OptionalConfigValue = .fastlaneDefault(false), + skipHelperVersionCheck: OptionalConfigValue = .fastlaneDefault(false), + clearPreviousScreenshots: OptionalConfigValue = .fastlaneDefault(false), + reinstallApp: OptionalConfigValue = .fastlaneDefault(false), + eraseSimulator: OptionalConfigValue = .fastlaneDefault(false), + headless: OptionalConfigValue = .fastlaneDefault(true), + overrideStatusBar: OptionalConfigValue = .fastlaneDefault(false), + overrideStatusBarArguments: OptionalConfigValue = .fastlaneDefault(nil), + localizeSimulator: OptionalConfigValue = .fastlaneDefault(false), + darkMode: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + addPhotos: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + addVideos: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + htmlTemplate: OptionalConfigValue = .fastlaneDefault(nil), + buildlogPath: String = "~/Library/Logs/snapshot", + clean: OptionalConfigValue = .fastlaneDefault(false), + testWithoutBuilding: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + sdk: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + numberOfRetries: Int = 1, + stopAfterFirstError: OptionalConfigValue = .fastlaneDefault(false), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + resultBundle: OptionalConfigValue = .fastlaneDefault(false), + testTargetName: OptionalConfigValue = .fastlaneDefault(nil), + namespaceLogFiles: Any? = nil, + concurrentSimulators: OptionalConfigValue = .fastlaneDefault(true), + disableSlideToType: OptionalConfigValue = .fastlaneDefault(false), + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(false), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(false), + testplan: OptionalConfigValue = .fastlaneDefault(nil), + onlyTesting: Any? = nil, + skipTesting: Any? = nil, + xcodebuildFormatter: String = "xcbeautify", + xcprettyArgs: OptionalConfigValue = .fastlaneDefault(nil), + disableXcpretty: OptionalConfigValue = .fastlaneDefault(nil), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(nil), + useSystemScm: OptionalConfigValue = .fastlaneDefault(false)) +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let devicesArg = devices.asRubyArgument(name: "devices", type: nil) + let languagesArg = RubyCommand.Argument(name: "languages", value: languages, type: nil) + let launchArgumentsArg = RubyCommand.Argument(name: "launch_arguments", value: launchArguments, type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputSimulatorLogsArg = outputSimulatorLogs.asRubyArgument(name: "output_simulator_logs", type: nil) + let iosVersionArg = iosVersion.asRubyArgument(name: "ios_version", type: nil) + let skipOpenSummaryArg = skipOpenSummary.asRubyArgument(name: "skip_open_summary", type: nil) + let skipHelperVersionCheckArg = skipHelperVersionCheck.asRubyArgument(name: "skip_helper_version_check", type: nil) + let clearPreviousScreenshotsArg = clearPreviousScreenshots.asRubyArgument(name: "clear_previous_screenshots", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let eraseSimulatorArg = eraseSimulator.asRubyArgument(name: "erase_simulator", type: nil) + let headlessArg = headless.asRubyArgument(name: "headless", type: nil) + let overrideStatusBarArg = overrideStatusBar.asRubyArgument(name: "override_status_bar", type: nil) + let overrideStatusBarArgumentsArg = overrideStatusBarArguments.asRubyArgument(name: "override_status_bar_arguments", type: nil) + let localizeSimulatorArg = localizeSimulator.asRubyArgument(name: "localize_simulator", type: nil) + let darkModeArg = darkMode.asRubyArgument(name: "dark_mode", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let addPhotosArg = addPhotos.asRubyArgument(name: "add_photos", type: nil) + let addVideosArg = addVideos.asRubyArgument(name: "add_videos", type: nil) + let htmlTemplateArg = htmlTemplate.asRubyArgument(name: "html_template", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let testWithoutBuildingArg = testWithoutBuilding.asRubyArgument(name: "test_without_building", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let numberOfRetriesArg = RubyCommand.Argument(name: "number_of_retries", value: numberOfRetries, type: nil) + let stopAfterFirstErrorArg = stopAfterFirstError.asRubyArgument(name: "stop_after_first_error", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let testTargetNameArg = testTargetName.asRubyArgument(name: "test_target_name", type: nil) + let namespaceLogFilesArg = RubyCommand.Argument(name: "namespace_log_files", value: namespaceLogFiles, type: nil) + let concurrentSimulatorsArg = concurrentSimulators.asRubyArgument(name: "concurrent_simulators", type: nil) + let disableSlideToTypeArg = disableSlideToType.asRubyArgument(name: "disable_slide_to_type", type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let testplanArg = testplan.asRubyArgument(name: "testplan", type: nil) + let onlyTestingArg = RubyCommand.Argument(name: "only_testing", value: onlyTesting, type: nil) + let skipTestingArg = RubyCommand.Argument(name: "skip_testing", value: skipTesting, type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let xcprettyArgsArg = xcprettyArgs.asRubyArgument(name: "xcpretty_args", type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + xcargsArg, + xcconfigArg, + devicesArg, + languagesArg, + launchArgumentsArg, + outputDirectoryArg, + outputSimulatorLogsArg, + iosVersionArg, + skipOpenSummaryArg, + skipHelperVersionCheckArg, + clearPreviousScreenshotsArg, + reinstallAppArg, + eraseSimulatorArg, + headlessArg, + overrideStatusBarArg, + overrideStatusBarArgumentsArg, + localizeSimulatorArg, + darkModeArg, + appIdentifierArg, + addPhotosArg, + addVideosArg, + htmlTemplateArg, + buildlogPathArg, + cleanArg, + testWithoutBuildingArg, + configurationArg, + sdkArg, + schemeArg, + numberOfRetriesArg, + stopAfterFirstErrorArg, + derivedDataPathArg, + resultBundleArg, + testTargetNameArg, + namespaceLogFilesArg, + concurrentSimulatorsArg, + disableSlideToTypeArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + testplanArg, + onlyTestingArg, + skipTestingArg, + xcodebuildFormatterArg, + xcprettyArgsArg, + disableXcprettyArg, + suppressXcodeOutputArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "capture_ios_screenshots", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `capture_ios_screenshots` action + + - parameters: + - workspace: Path the workspace file + - project: Path the project file + - xcargs: Pass additional arguments to xcodebuild for the test phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - devices: A list of devices you want to take the screenshots from + - languages: A list of languages which should be used + - launchArguments: A list of launch arguments which should be used + - outputDirectory: The directory where to store the screenshots + - outputSimulatorLogs: If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + - iosVersion: By default, the latest version should be used automatically. If you want to change it, do it here + - skipOpenSummary: Don't open the HTML summary after running _snapshot_ + - skipHelperVersionCheck: Do not check for most recent SnapshotHelper code + - clearPreviousScreenshots: Enabling this option will automatically clear previously generated screenshots before running snapshot + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - eraseSimulator: Enabling this option will automatically erase the simulator before running the application + - headless: Enabling this option will prevent displaying the simulator window + - overrideStatusBar: Enabling this option will automatically override the status bar to show 9:41 AM, full battery, and full reception (Adjust 'SNAPSHOT_SIMULATOR_WAIT_FOR_BOOT_TIMEOUT' environment variable if override status bar is not working. Might be because simulator is not fully booted. Defaults to 10 seconds) + - overrideStatusBarArguments: Fully customize the status bar by setting each option here. Requires `override_status_bar` to be set to `true`. See `xcrun simctl status_bar --help` + - localizeSimulator: Enabling this option will configure the Simulator's system language + - darkMode: Enabling this option will configure the Simulator to be in dark mode (false for light, true for dark) + - appIdentifier: The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + - addPhotos: A list of photos that should be added to the simulator before running the application + - addVideos: A list of videos that should be added to the simulator before running the application + - htmlTemplate: A path to screenshots.html template + - buildlogPath: The directory where to store the build log + - clean: Should the project be cleaned before building it? + - testWithoutBuilding: Test without building, requires a derived data path + - configuration: The configuration to use when building the app. Defaults to 'Release' + - sdk: The SDK that should be used for building the application + - scheme: The scheme you want to use, this must be the scheme for the UI Tests + - numberOfRetries: The number of times a test can fail before snapshot should stop retrying + - stopAfterFirstError: Should snapshot stop immediately after the tests completely failed on one device? + - derivedDataPath: The directory where build products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - testTargetName: The name of the target you want to test (if you desire to override the Target Application from Xcode) + - namespaceLogFiles: Separate the log files per device and per language + - concurrentSimulators: Take snapshots on multiple simulators concurrently. Note: This option is only applicable when running against Xcode 9 + - disableSlideToType: Disable the simulator from showing the 'Slide to type' prompt + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - testplan: The testplan associated with the scheme that should be used for testing + - onlyTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to run + - skipTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to skip + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - xcprettyArgs: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Additional xcpretty arguments + - disableXcpretty: Disable xcpretty formatting of build + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - useSystemScm: Lets xcodebuild use system's scm configuration + */ +public func captureScreenshots(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + devices: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + languages: [String] = ["en-US"], + launchArguments: [String] = [""], + outputDirectory: String = "screenshots", + outputSimulatorLogs: OptionalConfigValue = .fastlaneDefault(false), + iosVersion: OptionalConfigValue = .fastlaneDefault(nil), + skipOpenSummary: OptionalConfigValue = .fastlaneDefault(false), + skipHelperVersionCheck: OptionalConfigValue = .fastlaneDefault(false), + clearPreviousScreenshots: OptionalConfigValue = .fastlaneDefault(false), + reinstallApp: OptionalConfigValue = .fastlaneDefault(false), + eraseSimulator: OptionalConfigValue = .fastlaneDefault(false), + headless: OptionalConfigValue = .fastlaneDefault(true), + overrideStatusBar: OptionalConfigValue = .fastlaneDefault(false), + overrideStatusBarArguments: OptionalConfigValue = .fastlaneDefault(nil), + localizeSimulator: OptionalConfigValue = .fastlaneDefault(false), + darkMode: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + addPhotos: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + addVideos: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + htmlTemplate: OptionalConfigValue = .fastlaneDefault(nil), + buildlogPath: String = "~/Library/Logs/snapshot", + clean: OptionalConfigValue = .fastlaneDefault(false), + testWithoutBuilding: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + sdk: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + numberOfRetries: Int = 1, + stopAfterFirstError: OptionalConfigValue = .fastlaneDefault(false), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + resultBundle: OptionalConfigValue = .fastlaneDefault(false), + testTargetName: OptionalConfigValue = .fastlaneDefault(nil), + namespaceLogFiles: Any? = nil, + concurrentSimulators: OptionalConfigValue = .fastlaneDefault(true), + disableSlideToType: OptionalConfigValue = .fastlaneDefault(false), + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(false), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(false), + testplan: OptionalConfigValue = .fastlaneDefault(nil), + onlyTesting: Any? = nil, + skipTesting: Any? = nil, + xcodebuildFormatter: String = "xcbeautify", + xcprettyArgs: OptionalConfigValue = .fastlaneDefault(nil), + disableXcpretty: OptionalConfigValue = .fastlaneDefault(nil), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(nil), + useSystemScm: OptionalConfigValue = .fastlaneDefault(false)) +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let devicesArg = devices.asRubyArgument(name: "devices", type: nil) + let languagesArg = RubyCommand.Argument(name: "languages", value: languages, type: nil) + let launchArgumentsArg = RubyCommand.Argument(name: "launch_arguments", value: launchArguments, type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputSimulatorLogsArg = outputSimulatorLogs.asRubyArgument(name: "output_simulator_logs", type: nil) + let iosVersionArg = iosVersion.asRubyArgument(name: "ios_version", type: nil) + let skipOpenSummaryArg = skipOpenSummary.asRubyArgument(name: "skip_open_summary", type: nil) + let skipHelperVersionCheckArg = skipHelperVersionCheck.asRubyArgument(name: "skip_helper_version_check", type: nil) + let clearPreviousScreenshotsArg = clearPreviousScreenshots.asRubyArgument(name: "clear_previous_screenshots", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let eraseSimulatorArg = eraseSimulator.asRubyArgument(name: "erase_simulator", type: nil) + let headlessArg = headless.asRubyArgument(name: "headless", type: nil) + let overrideStatusBarArg = overrideStatusBar.asRubyArgument(name: "override_status_bar", type: nil) + let overrideStatusBarArgumentsArg = overrideStatusBarArguments.asRubyArgument(name: "override_status_bar_arguments", type: nil) + let localizeSimulatorArg = localizeSimulator.asRubyArgument(name: "localize_simulator", type: nil) + let darkModeArg = darkMode.asRubyArgument(name: "dark_mode", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let addPhotosArg = addPhotos.asRubyArgument(name: "add_photos", type: nil) + let addVideosArg = addVideos.asRubyArgument(name: "add_videos", type: nil) + let htmlTemplateArg = htmlTemplate.asRubyArgument(name: "html_template", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let testWithoutBuildingArg = testWithoutBuilding.asRubyArgument(name: "test_without_building", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let numberOfRetriesArg = RubyCommand.Argument(name: "number_of_retries", value: numberOfRetries, type: nil) + let stopAfterFirstErrorArg = stopAfterFirstError.asRubyArgument(name: "stop_after_first_error", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let testTargetNameArg = testTargetName.asRubyArgument(name: "test_target_name", type: nil) + let namespaceLogFilesArg = RubyCommand.Argument(name: "namespace_log_files", value: namespaceLogFiles, type: nil) + let concurrentSimulatorsArg = concurrentSimulators.asRubyArgument(name: "concurrent_simulators", type: nil) + let disableSlideToTypeArg = disableSlideToType.asRubyArgument(name: "disable_slide_to_type", type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let testplanArg = testplan.asRubyArgument(name: "testplan", type: nil) + let onlyTestingArg = RubyCommand.Argument(name: "only_testing", value: onlyTesting, type: nil) + let skipTestingArg = RubyCommand.Argument(name: "skip_testing", value: skipTesting, type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let xcprettyArgsArg = xcprettyArgs.asRubyArgument(name: "xcpretty_args", type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + xcargsArg, + xcconfigArg, + devicesArg, + languagesArg, + launchArgumentsArg, + outputDirectoryArg, + outputSimulatorLogsArg, + iosVersionArg, + skipOpenSummaryArg, + skipHelperVersionCheckArg, + clearPreviousScreenshotsArg, + reinstallAppArg, + eraseSimulatorArg, + headlessArg, + overrideStatusBarArg, + overrideStatusBarArgumentsArg, + localizeSimulatorArg, + darkModeArg, + appIdentifierArg, + addPhotosArg, + addVideosArg, + htmlTemplateArg, + buildlogPathArg, + cleanArg, + testWithoutBuildingArg, + configurationArg, + sdkArg, + schemeArg, + numberOfRetriesArg, + stopAfterFirstErrorArg, + derivedDataPathArg, + resultBundleArg, + testTargetNameArg, + namespaceLogFilesArg, + concurrentSimulatorsArg, + disableSlideToTypeArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + testplanArg, + onlyTestingArg, + skipTestingArg, + xcodebuildFormatterArg, + xcprettyArgsArg, + disableXcprettyArg, + suppressXcodeOutputArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "capture_screenshots", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs `carthage` for your project + + - parameters: + - command: Carthage command (one of: build, bootstrap, update, archive) + - dependencies: Carthage dependencies to update, build or bootstrap + - useSsh: Use SSH for downloading GitHub repositories + - useSubmodules: Add dependencies as Git submodules + - useNetrc: Use .netrc for downloading frameworks + - useBinaries: Check out dependency repositories even when prebuilt frameworks exist + - noCheckout: When bootstrapping Carthage do not checkout + - noBuild: When bootstrapping Carthage do not build + - noSkipCurrent: Don't skip building the Carthage project (in addition to its dependencies) + - derivedData: Use derived data folder at path + - verbose: Print xcodebuild output inline + - platform: Define which platform to build for + - cacheBuilds: By default Carthage will rebuild a dependency regardless of whether it's the same resolved version as before. Passing the --cache-builds will cause carthage to avoid rebuilding a dependency if it can + - frameworks: Framework name or names to archive, could be applied only along with the archive command + - output: Output name for the archive, could be applied only along with the archive command. Use following format *.framework.zip + - configuration: Define which build configuration to use when building + - toolchain: Define which xcodebuild toolchain to use when building + - projectDirectory: Define the directory containing the Carthage project + - newResolver: Use new resolver when resolving dependency graph + - logPath: Path to the xcode build output + - useXcframeworks: Create xcframework bundles instead of one framework per platform (requires Xcode 12+) + - archive: Archive built frameworks from the current project + - executable: Path to the `carthage` executable on your machine + */ +public func carthage(command: String = "bootstrap", + dependencies: [String] = [], + useSsh: OptionalConfigValue = .fastlaneDefault(nil), + useSubmodules: OptionalConfigValue = .fastlaneDefault(nil), + useNetrc: OptionalConfigValue = .fastlaneDefault(nil), + useBinaries: OptionalConfigValue = .fastlaneDefault(nil), + noCheckout: OptionalConfigValue = .fastlaneDefault(nil), + noBuild: OptionalConfigValue = .fastlaneDefault(nil), + noSkipCurrent: OptionalConfigValue = .fastlaneDefault(nil), + derivedData: OptionalConfigValue = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(nil), + platform: OptionalConfigValue = .fastlaneDefault(nil), + cacheBuilds: OptionalConfigValue = .fastlaneDefault(false), + frameworks: [String] = [], + output: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + toolchain: OptionalConfigValue = .fastlaneDefault(nil), + projectDirectory: OptionalConfigValue = .fastlaneDefault(nil), + newResolver: OptionalConfigValue = .fastlaneDefault(nil), + logPath: OptionalConfigValue = .fastlaneDefault(nil), + useXcframeworks: OptionalConfigValue = .fastlaneDefault(false), + archive: OptionalConfigValue = .fastlaneDefault(false), + executable: String = "carthage") +{ + let commandArg = RubyCommand.Argument(name: "command", value: command, type: nil) + let dependenciesArg = RubyCommand.Argument(name: "dependencies", value: dependencies, type: nil) + let useSshArg = useSsh.asRubyArgument(name: "use_ssh", type: nil) + let useSubmodulesArg = useSubmodules.asRubyArgument(name: "use_submodules", type: nil) + let useNetrcArg = useNetrc.asRubyArgument(name: "use_netrc", type: nil) + let useBinariesArg = useBinaries.asRubyArgument(name: "use_binaries", type: nil) + let noCheckoutArg = noCheckout.asRubyArgument(name: "no_checkout", type: nil) + let noBuildArg = noBuild.asRubyArgument(name: "no_build", type: nil) + let noSkipCurrentArg = noSkipCurrent.asRubyArgument(name: "no_skip_current", type: nil) + let derivedDataArg = derivedData.asRubyArgument(name: "derived_data", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let platformArg = platform.asRubyArgument(name: "platform", type: nil) + let cacheBuildsArg = cacheBuilds.asRubyArgument(name: "cache_builds", type: nil) + let frameworksArg = RubyCommand.Argument(name: "frameworks", value: frameworks, type: nil) + let outputArg = output.asRubyArgument(name: "output", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let toolchainArg = toolchain.asRubyArgument(name: "toolchain", type: nil) + let projectDirectoryArg = projectDirectory.asRubyArgument(name: "project_directory", type: nil) + let newResolverArg = newResolver.asRubyArgument(name: "new_resolver", type: nil) + let logPathArg = logPath.asRubyArgument(name: "log_path", type: nil) + let useXcframeworksArg = useXcframeworks.asRubyArgument(name: "use_xcframeworks", type: nil) + let archiveArg = archive.asRubyArgument(name: "archive", type: nil) + let executableArg = RubyCommand.Argument(name: "executable", value: executable, type: nil) + let array: [RubyCommand.Argument?] = [commandArg, + dependenciesArg, + useSshArg, + useSubmodulesArg, + useNetrcArg, + useBinariesArg, + noCheckoutArg, + noBuildArg, + noSkipCurrentArg, + derivedDataArg, + verboseArg, + platformArg, + cacheBuildsArg, + frameworksArg, + outputArg, + configurationArg, + toolchainArg, + projectDirectoryArg, + newResolverArg, + logPathArg, + useXcframeworksArg, + archiveArg, + executableArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "carthage", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `get_certificates` action + + - parameters: + - development: Create a development certificate instead of a distribution one + - type: Create specific certificate type (takes precedence over :development) + - force: Create a certificate even if an existing certificate exists + - generateAppleCerts: Create a certificate type for Xcode 11 and later (Apple Development or Apple Distribution) + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - filename: The filename of certificate to store + - outputPath: The path to a directory in which all certificates and private keys should be stored + - keychainPath: Path to a custom keychain + - keychainPassword: This might be required the first time you access certificates on a new mac. For the login/default keychain this is your macOS account password + - skipSetPartitionList: Skips setting the partition list (which can sometimes take a long time). Setting the partition list is usually needed to prevent Xcode from prompting to allow a cert to be used for signing + - platform: Set the provisioning profile's platform (ios, macos, tvos) + + **Important**: It is recommended to use [match](https://docs.fastlane.tools/actions/match/) according to the [codesigning.guide](https://codesigning.guide) for generating and maintaining your certificates. Use _cert_ directly only if you want full control over what's going on and know more about codesigning. + Use this action to download the latest code signing identity. + */ +public func cert(development: OptionalConfigValue = .fastlaneDefault(false), + type: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + generateAppleCerts: OptionalConfigValue = .fastlaneDefault(true), + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + filename: OptionalConfigValue = .fastlaneDefault(nil), + outputPath: String = ".", + keychainPath: String, + keychainPassword: OptionalConfigValue = .fastlaneDefault(nil), + skipSetPartitionList: OptionalConfigValue = .fastlaneDefault(false), + platform: String = "ios") +{ + let developmentArg = development.asRubyArgument(name: "development", type: nil) + let typeArg = type.asRubyArgument(name: "type", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let generateAppleCertsArg = generateAppleCerts.asRubyArgument(name: "generate_apple_certs", type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let filenameArg = filename.asRubyArgument(name: "filename", type: nil) + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let keychainPathArg = RubyCommand.Argument(name: "keychain_path", value: keychainPath, type: nil) + let keychainPasswordArg = keychainPassword.asRubyArgument(name: "keychain_password", type: nil) + let skipSetPartitionListArg = skipSetPartitionList.asRubyArgument(name: "skip_set_partition_list", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let array: [RubyCommand.Argument?] = [developmentArg, + typeArg, + forceArg, + generateAppleCertsArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + filenameArg, + outputPathArg, + keychainPathArg, + keychainPasswordArg, + skipSetPartitionListArg, + platformArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "cert", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Collect git commit messages into a changelog + + - parameters: + - between: Array containing two Git revision values between which to collect messages, you mustn't use it with :commits_count key at the same time + - commitsCount: Number of commits to include in changelog, you mustn't use it with :between key at the same time + - path: Path of the git repository + - pretty: The format applied to each commit while generating the collected value + - dateFormat: The date format applied to each commit while generating the collected value + - ancestryPath: Whether or not to use ancestry-path param + - tagMatchPattern: A glob(7) pattern to match against when finding the last git tag + - matchLightweightTag: Whether or not to match a lightweight tag when searching for the last one + - quiet: Whether or not to disable changelog output + - includeMerges: **DEPRECATED!** Use `:merge_commit_filtering` instead - Whether or not to include any commits that are merges + - mergeCommitFiltering: Controls inclusion of merge commits when collecting the changelog. Valid values: 'include_merges', 'exclude_merges', 'only_include_merges' + + - returns: Returns a String containing your formatted git commits + + By default, messages will be collected back to the last tag, but the range can be controlled + */ +@discardableResult public func changelogFromGitCommits(between: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + commitsCount: OptionalConfigValue = .fastlaneDefault(nil), + path: String = "./", + pretty: String = "%B", + dateFormat: OptionalConfigValue = .fastlaneDefault(nil), + ancestryPath: OptionalConfigValue = .fastlaneDefault(false), + tagMatchPattern: OptionalConfigValue = .fastlaneDefault(nil), + matchLightweightTag: OptionalConfigValue = .fastlaneDefault(true), + quiet: OptionalConfigValue = .fastlaneDefault(false), + includeMerges: OptionalConfigValue = .fastlaneDefault(nil), + mergeCommitFiltering: String = "include_merges") -> String +{ + let betweenArg = between.asRubyArgument(name: "between", type: nil) + let commitsCountArg = commitsCount.asRubyArgument(name: "commits_count", type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let prettyArg = RubyCommand.Argument(name: "pretty", value: pretty, type: nil) + let dateFormatArg = dateFormat.asRubyArgument(name: "date_format", type: nil) + let ancestryPathArg = ancestryPath.asRubyArgument(name: "ancestry_path", type: nil) + let tagMatchPatternArg = tagMatchPattern.asRubyArgument(name: "tag_match_pattern", type: nil) + let matchLightweightTagArg = matchLightweightTag.asRubyArgument(name: "match_lightweight_tag", type: nil) + let quietArg = quiet.asRubyArgument(name: "quiet", type: nil) + let includeMergesArg = includeMerges.asRubyArgument(name: "include_merges", type: nil) + let mergeCommitFilteringArg = RubyCommand.Argument(name: "merge_commit_filtering", value: mergeCommitFiltering, type: nil) + let array: [RubyCommand.Argument?] = [betweenArg, + commitsCountArg, + pathArg, + prettyArg, + dateFormatArg, + ancestryPathArg, + tagMatchPatternArg, + matchLightweightTagArg, + quietArg, + includeMergesArg, + mergeCommitFilteringArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "changelog_from_git_commits", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Send a success/error message to [ChatWork](https://go.chatwork.com/) + + - parameters: + - apiToken: ChatWork API Token + - message: The message to post on ChatWork + - roomid: The room ID + - success: Was this build successful? (true/false) + + Information on how to obtain an API token: [http://developer.chatwork.com/ja/authenticate.html](http://developer.chatwork.com/ja/authenticate.html) + */ +public func chatwork(apiToken: String, + message: String, + roomid: Int, + success: OptionalConfigValue = .fastlaneDefault(true)) +{ + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let roomidArg = RubyCommand.Argument(name: "roomid", value: roomid, type: nil) + let successArg = success.asRubyArgument(name: "success", type: nil) + let array: [RubyCommand.Argument?] = [apiTokenArg, + messageArg, + roomidArg, + successArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "chatwork", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Check your app's metadata before you submit your app to review (via _precheck_) + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - platform: The platform to use (optional) + - defaultRuleLevel: The default rule level unless otherwise configured + - includeInAppPurchases: Should check in-app purchases? + - useLive: Should force check live app? + - negativeAppleSentiment: mentioning  in a way that could be considered negative + - placeholderText: using placeholder text (e.g.:"lorem ipsum", "text here", etc...) + - otherPlatforms: mentioning other platforms, like Android or Blackberry + - futureFunctionality: mentioning features or content that is not currently available in your app + - testWords: using text indicating this release is a test + - curseWords: including words that might be considered objectionable + - freeStuffInIap: using text indicating that your IAP is free + - customText: mentioning any of the user-specified words passed to custom_text(data: [words]) + - copyrightDate: using a copyright date that is any different from this current year, or missing a date + - unreachableUrls: unreachable URLs in app metadata + + - returns: true if precheck passes, else, false + + More information: https://fastlane.tools/precheck + */ +@discardableResult public func checkAppStoreMetadata(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appIdentifier: String, + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + defaultRuleLevel: String = "error", + includeInAppPurchases: OptionalConfigValue = .fastlaneDefault(true), + useLive: OptionalConfigValue = .fastlaneDefault(false), + negativeAppleSentiment: Any? = nil, + placeholderText: Any? = nil, + otherPlatforms: Any? = nil, + futureFunctionality: Any? = nil, + testWords: Any? = nil, + curseWords: Any? = nil, + freeStuffInIap: Any? = nil, + customText: Any? = nil, + copyrightDate: Any? = nil, + unreachableUrls: Any? = nil) -> Bool +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let defaultRuleLevelArg = RubyCommand.Argument(name: "default_rule_level", value: defaultRuleLevel, type: nil) + let includeInAppPurchasesArg = includeInAppPurchases.asRubyArgument(name: "include_in_app_purchases", type: nil) + let useLiveArg = useLive.asRubyArgument(name: "use_live", type: nil) + let negativeAppleSentimentArg = RubyCommand.Argument(name: "negative_apple_sentiment", value: negativeAppleSentiment, type: nil) + let placeholderTextArg = RubyCommand.Argument(name: "placeholder_text", value: placeholderText, type: nil) + let otherPlatformsArg = RubyCommand.Argument(name: "other_platforms", value: otherPlatforms, type: nil) + let futureFunctionalityArg = RubyCommand.Argument(name: "future_functionality", value: futureFunctionality, type: nil) + let testWordsArg = RubyCommand.Argument(name: "test_words", value: testWords, type: nil) + let curseWordsArg = RubyCommand.Argument(name: "curse_words", value: curseWords, type: nil) + let freeStuffInIapArg = RubyCommand.Argument(name: "free_stuff_in_iap", value: freeStuffInIap, type: nil) + let customTextArg = RubyCommand.Argument(name: "custom_text", value: customText, type: nil) + let copyrightDateArg = RubyCommand.Argument(name: "copyright_date", value: copyrightDate, type: nil) + let unreachableUrlsArg = RubyCommand.Argument(name: "unreachable_urls", value: unreachableUrls, type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + appIdentifierArg, + usernameArg, + teamIdArg, + teamNameArg, + platformArg, + defaultRuleLevelArg, + includeInAppPurchasesArg, + useLiveArg, + negativeAppleSentimentArg, + placeholderTextArg, + otherPlatformsArg, + futureFunctionalityArg, + testWordsArg, + curseWordsArg, + freeStuffInIapArg, + customTextArg, + copyrightDateArg, + unreachableUrlsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "check_app_store_metadata", className: nil, args: args) + return parseBool(fromString: runner.executeCommand(command)) +} + +/** + Deletes files created as result of running gym, cert, sigh or download_dsyms + + - parameter excludePattern: Exclude all files from clearing that match the given Regex pattern: e.g. '.*.mobileprovision' + + This action deletes the files that get created in your repo as a result of running the _gym_ and _sigh_ commands. It doesn't delete the `fastlane/report.xml` though, this is probably more suited for the .gitignore. + + Useful if you quickly want to send out a test build by dropping down to the command line and typing something like `fastlane beta`, without leaving your repo in a messy state afterwards. + */ +public func cleanBuildArtifacts(excludePattern: OptionalConfigValue = .fastlaneDefault(nil)) { + let excludePatternArg = excludePattern.asRubyArgument(name: "exclude_pattern", type: nil) + let array: [RubyCommand.Argument?] = [excludePatternArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "clean_build_artifacts", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Remove the cache for pods + + - parameters: + - name: Pod name to be removed from cache + - noAnsi: Show output without ANSI codes + - verbose: Show more debugging information + - silent: Show nothing + - allowRoot: Allows CocoaPods to run as root + */ +public func cleanCocoapodsCache(name: OptionalConfigValue = .fastlaneDefault(nil), + noAnsi: OptionalConfigValue = .fastlaneDefault(false), + verbose: OptionalConfigValue = .fastlaneDefault(false), + silent: OptionalConfigValue = .fastlaneDefault(false), + allowRoot: OptionalConfigValue = .fastlaneDefault(false)) +{ + let nameArg = name.asRubyArgument(name: "name", type: nil) + let noAnsiArg = noAnsi.asRubyArgument(name: "no_ansi", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let allowRootArg = allowRoot.asRubyArgument(name: "allow_root", type: nil) + let array: [RubyCommand.Argument?] = [nameArg, + noAnsiArg, + verboseArg, + silentArg, + allowRootArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "clean_cocoapods_cache", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Deletes the Xcode Derived Data + + - parameter derivedDataPath: Custom path for derivedData + + Deletes the Derived Data from path set on Xcode or a supplied path + */ +public func clearDerivedData(derivedDataPath: String = "~/Library/Developer/Xcode/DerivedData") { + let derivedDataPathArg = RubyCommand.Argument(name: "derived_data_path", value: derivedDataPath, type: nil) + let array: [RubyCommand.Argument?] = [derivedDataPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "clear_derived_data", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Copies a given string into the clipboard. Works only on macOS + + - parameter value: The string that should be copied into the clipboard + */ +public func clipboard(value: String) { + let valueArg = RubyCommand.Argument(name: "value", value: value, type: nil) + let array: [RubyCommand.Argument?] = [valueArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "clipboard", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generates a Code Count that can be read by Jenkins (xml format) + + - parameters: + - binaryPath: Where the cloc binary lives on your system (full path including 'cloc') + - excludeDir: Comma separated list of directories to exclude + - outputDirectory: Where to put the generated report file + - sourceDirectory: Where to look for the source code (relative to the project root folder) + - xml: Should we generate an XML File (if false, it will generate a plain text file)? + + This action will run cloc to generate a SLOC report that the Jenkins SLOCCount plugin can read. + See [https://wiki.jenkins-ci.org/display/JENKINS/SLOCCount+Plugin](https://wiki.jenkins-ci.org/display/JENKINS/SLOCCount+Plugin) and [https://github.com/AlDanial/cloc](https://github.com/AlDanial/cloc) for more information. + */ +public func cloc(binaryPath: String = "/usr/local/bin/cloc", + excludeDir: OptionalConfigValue = .fastlaneDefault(nil), + outputDirectory: String = "build", + sourceDirectory: String = "", + xml: OptionalConfigValue = .fastlaneDefault(true)) +{ + let binaryPathArg = RubyCommand.Argument(name: "binary_path", value: binaryPath, type: nil) + let excludeDirArg = excludeDir.asRubyArgument(name: "exclude_dir", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let sourceDirectoryArg = RubyCommand.Argument(name: "source_directory", value: sourceDirectory, type: nil) + let xmlArg = xml.asRubyArgument(name: "xml", type: nil) + let array: [RubyCommand.Argument?] = [binaryPathArg, + excludeDirArg, + outputDirectoryArg, + sourceDirectoryArg, + xmlArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "cloc", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Print a Club Mate in your build output + */ +public func clubmate() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "clubmate", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs `pod install` for the project + + - parameters: + - repoUpdate: Add `--repo-update` flag to `pod install` command + - cleanInstall: Execute a full pod installation ignoring the content of the project cache + - silent: Execute command without logging output + - verbose: Show more debugging information + - ansi: Show output with ANSI codes + - useBundleExec: Use bundle exec when there is a Gemfile presented + - podfile: Explicitly specify the path to the Cocoapods' Podfile. You can either set it to the Podfile's path or to the folder containing the Podfile file + - errorCallback: A callback invoked with the command output if there is a non-zero exit status + - tryRepoUpdateOnError: Retry with --repo-update if action was finished with error + - deployment: Disallow any changes to the Podfile or the Podfile.lock during installation + - allowRoot: Allows CocoaPods to run as root + - clean: **DEPRECATED!** (Option renamed as clean_install) Remove SCM directories + - integrate: **DEPRECATED!** (Option removed from cocoapods) Integrate the Pods libraries into the Xcode project(s) + + If you use [CocoaPods](http://cocoapods.org) you can use the `cocoapods` integration to run `pod install` before building your app. + */ +public func cocoapods(repoUpdate: OptionalConfigValue = .fastlaneDefault(false), + cleanInstall: OptionalConfigValue = .fastlaneDefault(false), + silent: OptionalConfigValue = .fastlaneDefault(false), + verbose: OptionalConfigValue = .fastlaneDefault(false), + ansi: OptionalConfigValue = .fastlaneDefault(true), + useBundleExec: OptionalConfigValue = .fastlaneDefault(true), + podfile: OptionalConfigValue = .fastlaneDefault(nil), + errorCallback: ((String) -> Void)? = nil, + tryRepoUpdateOnError: OptionalConfigValue = .fastlaneDefault(false), + deployment: OptionalConfigValue = .fastlaneDefault(false), + allowRoot: OptionalConfigValue = .fastlaneDefault(false), + clean: OptionalConfigValue = .fastlaneDefault(true), + integrate: OptionalConfigValue = .fastlaneDefault(true)) +{ + let repoUpdateArg = repoUpdate.asRubyArgument(name: "repo_update", type: nil) + let cleanInstallArg = cleanInstall.asRubyArgument(name: "clean_install", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let ansiArg = ansi.asRubyArgument(name: "ansi", type: nil) + let useBundleExecArg = useBundleExec.asRubyArgument(name: "use_bundle_exec", type: nil) + let podfileArg = podfile.asRubyArgument(name: "podfile", type: nil) + let errorCallbackArg = RubyCommand.Argument(name: "error_callback", value: errorCallback, type: .stringClosure) + let tryRepoUpdateOnErrorArg = tryRepoUpdateOnError.asRubyArgument(name: "try_repo_update_on_error", type: nil) + let deploymentArg = deployment.asRubyArgument(name: "deployment", type: nil) + let allowRootArg = allowRoot.asRubyArgument(name: "allow_root", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let integrateArg = integrate.asRubyArgument(name: "integrate", type: nil) + let array: [RubyCommand.Argument?] = [repoUpdateArg, + cleanInstallArg, + silentArg, + verboseArg, + ansiArg, + useBundleExecArg, + podfileArg, + errorCallbackArg, + tryRepoUpdateOnErrorArg, + deploymentArg, + allowRootArg, + cleanArg, + integrateArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "cocoapods", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will commit a file directly on GitHub via the API + + - parameters: + - repositoryName: The path to your repo, e.g. 'fastlane/fastlane' + - serverUrl: The server url. e.g. 'https://your.internal.github.host/api/v3' (Default: 'https://api.github.com') + - apiToken: Personal API Token for GitHub - generate one at https://github.com/settings/tokens + - apiBearer: Use a Bearer authorization token. Usually generated by Github Apps, e.g. GitHub Actions GITHUB_TOKEN environment variable + - branch: The branch that the file should be committed on (default: master) + - path: The relative path to your file from project root e.g. assets/my_app.xcarchive + - message: The commit message. Defaults to the file name + - secure: Optionally disable secure requests (ssl_verify_peer) + + - returns: A hash containing all relevant information for this commit + Access things like 'html_url', 'sha', 'message' + + Commits a file directly to GitHub. You must provide your GitHub Personal token (get one from [https://github.com/settings/tokens/new](https://github.com/settings/tokens/new)), the repository name and the relative file path from the root git project. + Out parameters provide the commit sha created, which can be used for later usage for examples such as releases, the direct download link and the full response JSON. + Documentation: [https://developer.github.com/v3/repos/contents/#create-a-file](https://developer.github.com/v3/repos/contents/#create-a-file). + */ +@discardableResult public func commitGithubFile(repositoryName: String, + serverUrl: String = "https://api.github.com", + apiToken: OptionalConfigValue = .fastlaneDefault(nil), + apiBearer: OptionalConfigValue = .fastlaneDefault(nil), + branch: String = "master", + path: String, + message: OptionalConfigValue = .fastlaneDefault(nil), + secure: OptionalConfigValue = .fastlaneDefault(true)) -> [String: String] +{ + let repositoryNameArg = RubyCommand.Argument(name: "repository_name", value: repositoryName, type: nil) + let serverUrlArg = RubyCommand.Argument(name: "server_url", value: serverUrl, type: nil) + let apiTokenArg = apiToken.asRubyArgument(name: "api_token", type: nil) + let apiBearerArg = apiBearer.asRubyArgument(name: "api_bearer", type: nil) + let branchArg = RubyCommand.Argument(name: "branch", value: branch, type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let messageArg = message.asRubyArgument(name: "message", type: nil) + let secureArg = secure.asRubyArgument(name: "secure", type: nil) + let array: [RubyCommand.Argument?] = [repositoryNameArg, + serverUrlArg, + apiTokenArg, + apiBearerArg, + branchArg, + pathArg, + messageArg, + secureArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "commit_github_file", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Creates a 'Version Bump' commit. Run after `increment_build_number` + + - parameters: + - message: The commit message when committing the version bump + - xcodeproj: The path to your project file (Not the workspace). If you have only one, this is optional + - force: Forces the commit, even if other files than the ones containing the version number have been modified + - settings: Include Settings.bundle/Root.plist with version bump + - ignore: A regular expression used to filter matched plist files to be modified + - include: A list of extra files to be included in the version bump (string array or comma-separated string) + - noVerify: Whether or not to use --no-verify + + This action will create a 'Version Bump' commit in your repo. Useful in conjunction with `increment_build_number`. + It checks the repo to make sure that only the relevant files have changed. These are the files that `increment_build_number` (`agvtool`) touches:| + | + >- All `.plist` files| + - The `.xcodeproj/project.pbxproj` file| + >| + Then commits those files to the repo. + Customize the message with the `:message` option. It defaults to 'Version Bump'. + If you have other uncommitted changes in your repo, this action will fail. If you started off in a clean repo, and used the _ipa_ and or _sigh_ actions, then you can use the [clean_build_artifacts](https://docs.fastlane.tools/actions/clean_build_artifacts/) action to clean those temporary files up before running this action. + */ +public func commitVersionBump(message: OptionalConfigValue = .fastlaneDefault(nil), + xcodeproj: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + settings: OptionalConfigValue = .fastlaneDefault(false), + ignore: OptionalConfigValue = .fastlaneDefault(nil), + include: [String] = [], + noVerify: OptionalConfigValue = .fastlaneDefault(false)) +{ + let messageArg = message.asRubyArgument(name: "message", type: nil) + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let settingsArg = settings.asRubyArgument(name: "settings", type: nil) + let ignoreArg = ignore.asRubyArgument(name: "ignore", type: nil) + let includeArg = RubyCommand.Argument(name: "include", value: include, type: nil) + let noVerifyArg = noVerify.asRubyArgument(name: "no_verify", type: nil) + let array: [RubyCommand.Argument?] = [messageArg, + xcodeprojArg, + forceArg, + settingsArg, + ignoreArg, + includeArg, + noVerifyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "commit_version_bump", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Copy and save your build artifacts (useful when you use reset_git_repo) + + - parameters: + - keepOriginal: Set this to false if you want move, rather than copy, the found artifacts + - targetPath: The directory in which you want your artifacts placed + - artifacts: An array of file patterns of the files/folders you want to preserve + - failOnMissing: Fail when a source file isn't found + + This action copies artifacts to a target directory. It's useful if you have a CI that will pick up these artifacts and attach them to the build. Useful e.g. for storing your `.ipa`s, `.dSYM.zip`s, `.mobileprovision`s, `.cert`s. + Make sure your `:target_path` is ignored from git, and if you use `reset_git_repo`, make sure the artifacts are added to the exclude list. + */ +public func copyArtifacts(keepOriginal: OptionalConfigValue = .fastlaneDefault(true), + targetPath: String = "artifacts", + artifacts: [String] = [], + failOnMissing: OptionalConfigValue = .fastlaneDefault(false)) +{ + let keepOriginalArg = keepOriginal.asRubyArgument(name: "keep_original", type: nil) + let targetPathArg = RubyCommand.Argument(name: "target_path", value: targetPath, type: nil) + let artifactsArg = RubyCommand.Argument(name: "artifacts", value: artifacts, type: nil) + let failOnMissingArg = failOnMissing.asRubyArgument(name: "fail_on_missing", type: nil) + let array: [RubyCommand.Argument?] = [keepOriginalArg, + targetPathArg, + artifactsArg, + failOnMissingArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "copy_artifacts", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Create Managed Google Play Apps + + - parameters: + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - developerAccountId: The ID of your Google Play Console account. Can be obtained from the URL when you log in (`https://play.google.com/apps/publish/?account=...` or when you 'Obtain private app publishing rights' (https://developers.google.com/android/work/play/custom-app-api/get-started#retrieve_the_developer_account_id) + - apk: Path to the APK file to upload + - appTitle: App Title + - language: Default app language (e.g. 'en_US') + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - timeout: Timeout for read, open, and send (in seconds) + + Create new apps on Managed Google Play. + */ +public func createAppOnManagedPlayStore(jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + developerAccountId: String, + apk: String, + appTitle: String, + language: String = "en_US", + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 300) +{ + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let developerAccountIdArg = RubyCommand.Argument(name: "developer_account_id", value: developerAccountId, type: nil) + let apkArg = RubyCommand.Argument(name: "apk", value: apk, type: nil) + let appTitleArg = RubyCommand.Argument(name: "app_title", value: appTitle, type: nil) + let languageArg = RubyCommand.Argument(name: "language", value: language, type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [jsonKeyArg, + jsonKeyDataArg, + developerAccountIdArg, + apkArg, + appTitleArg, + languageArg, + rootUrlArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "create_app_on_managed_play_store", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Creates the given application on iTC and the Dev Portal (via _produce_) + + - parameters: + - username: Your Apple ID Username + - appIdentifier: App Identifier (Bundle ID, e.g. com.krausefx.app) + - bundleIdentifierSuffix: App Identifier Suffix (Ignored if App Identifier does not end with .*) + - appName: App Name + - appVersion: Initial version number (e.g. '1.0') + - sku: SKU Number (e.g. '1234') + - platform: The platform to use (optional) + - platforms: The platforms to use (optional) + - language: Primary Language (e.g. 'en-US', 'fr-FR') + - companyName: The name of your company. It's used to set company name on App Store Connect team's app pages. Only required if it's the first app you create + - skipItc: Skip the creation of the app on App Store Connect + - itcUsers: Array of App Store Connect users. If provided, you can limit access to this newly created app for users with the App Manager, Developer, Marketer or Sales roles + - enabledFeatures: **DEPRECATED!** Please use `enable_services` instead - Array with Spaceship App Services + - enableServices: Array with Spaceship App Services (e.g. access_wifi: (on|off), app_attest: (on|off), app_group: (on|off), apple_pay: (on|off), associated_domains: (on|off), auto_fill_credential: (on|off), class_kit: (on|off), icloud: (legacy|cloudkit), custom_network_protocol: (on|off), data_protection: (complete|unlessopen|untilfirstauth), extended_virtual_address_space: (on|off), family_controls: (on|off), file_provider_testing_mode: (on|off), fonts: (on|off), game_center: (ios|mac), health_kit: (on|off), hls_interstitial_preview: (on|off), home_kit: (on|off), hotspot: (on|off), in_app_purchase: (on|off), inter_app_audio: (on|off), low_latency_hls: (on|off), managed_associated_domains: (on|off), maps: (on|off), multipath: (on|off), network_extension: (on|off), nfc_tag_reading: (on|off), personal_vpn: (on|off), passbook: (on|off), push_notification: (on|off), sign_in_with_apple: (on), siri_kit: (on|off), system_extension: (on|off), user_management: (on|off), vpn_configuration: (on|off), wallet: (on|off), wireless_accessory: (on|off), car_play_audio_app: (on|off), car_play_messaging_app: (on|off), car_play_navigation_app: (on|off), car_play_voip_calling_app: (on|off), critical_alerts: (on|off), hotspot_helper: (on|off), driver_kit: (on|off), driver_kit_endpoint_security: (on|off), driver_kit_family_hid_device: (on|off), driver_kit_family_networking: (on|off), driver_kit_family_serial: (on|off), driver_kit_hid_event_service: (on|off), driver_kit_transport_hid: (on|off), multitasking_camera_access: (on|off), sf_universal_link_api: (on|off), vp9_decoder: (on|off), music_kit: (on|off), shazam_kit: (on|off), communication_notifications: (on|off), group_activities: (on|off), health_kit_estimate_recalibration: (on|off), time_sensitive_notifications: (on|off)) + - skipDevcenter: Skip the creation of the app on the Apple Developer Portal + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - itcTeamId: The ID of your App Store Connect team if you're in multiple teams + - itcTeamName: The name of your App Store Connect team if you're in multiple teams + + Create new apps on App Store Connect and Apple Developer Portal via _produce_. + If the app already exists, `create_app_online` will not do anything. + For more information about _produce_, visit its documentation page: [https://docs.fastlane.tools/actions/produce/](https://docs.fastlane.tools/actions/produce/). + */ +public func createAppOnline(username: String, + appIdentifier: String, + bundleIdentifierSuffix: OptionalConfigValue = .fastlaneDefault(nil), + appName: String, + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + sku: String, + platform: String = "ios", + platforms: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + language: String = "English", + companyName: OptionalConfigValue = .fastlaneDefault(nil), + skipItc: OptionalConfigValue = .fastlaneDefault(false), + itcUsers: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + enabledFeatures: [String: Any] = [:], + enableServices: [String: Any] = [:], + skipDevcenter: OptionalConfigValue = .fastlaneDefault(false), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + itcTeamId: Any? = nil, + itcTeamName: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let bundleIdentifierSuffixArg = bundleIdentifierSuffix.asRubyArgument(name: "bundle_identifier_suffix", type: nil) + let appNameArg = RubyCommand.Argument(name: "app_name", value: appName, type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let skuArg = RubyCommand.Argument(name: "sku", value: sku, type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let platformsArg = platforms.asRubyArgument(name: "platforms", type: nil) + let languageArg = RubyCommand.Argument(name: "language", value: language, type: nil) + let companyNameArg = companyName.asRubyArgument(name: "company_name", type: nil) + let skipItcArg = skipItc.asRubyArgument(name: "skip_itc", type: nil) + let itcUsersArg = itcUsers.asRubyArgument(name: "itc_users", type: nil) + let enabledFeaturesArg = RubyCommand.Argument(name: "enabled_features", value: enabledFeatures, type: nil) + let enableServicesArg = RubyCommand.Argument(name: "enable_services", value: enableServices, type: nil) + let skipDevcenterArg = skipDevcenter.asRubyArgument(name: "skip_devcenter", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let itcTeamIdArg = RubyCommand.Argument(name: "itc_team_id", value: itcTeamId, type: nil) + let itcTeamNameArg = itcTeamName.asRubyArgument(name: "itc_team_name", type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + appIdentifierArg, + bundleIdentifierSuffixArg, + appNameArg, + appVersionArg, + skuArg, + platformArg, + platformsArg, + languageArg, + companyNameArg, + skipItcArg, + itcUsersArg, + enabledFeaturesArg, + enableServicesArg, + skipDevcenterArg, + teamIdArg, + teamNameArg, + itcTeamIdArg, + itcTeamNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "create_app_online", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Create a new Keychain + + - parameters: + - name: Keychain name + - path: Path to keychain + - password: Password for the keychain + - defaultKeychain: Should the newly created Keychain be the new system default keychain + - unlock: Unlock keychain after create + - timeout: timeout interval in seconds. Set `0` if you want to specify "no time-out" + - lockWhenSleeps: Lock keychain when the system sleeps + - lockAfterTimeout: Lock keychain after timeout interval + - addToSearchList: Add keychain to search list + - requireCreate: Fail the action if the Keychain already exists + */ +public func createKeychain(name: OptionalConfigValue = .fastlaneDefault(nil), + path: OptionalConfigValue = .fastlaneDefault(nil), + password: String, + defaultKeychain: OptionalConfigValue = .fastlaneDefault(false), + unlock: OptionalConfigValue = .fastlaneDefault(false), + timeout: Int = 300, + lockWhenSleeps: OptionalConfigValue = .fastlaneDefault(false), + lockAfterTimeout: OptionalConfigValue = .fastlaneDefault(false), + addToSearchList: OptionalConfigValue = .fastlaneDefault(true), + requireCreate: OptionalConfigValue = .fastlaneDefault(false)) +{ + let nameArg = name.asRubyArgument(name: "name", type: nil) + let pathArg = path.asRubyArgument(name: "path", type: nil) + let passwordArg = RubyCommand.Argument(name: "password", value: password, type: nil) + let defaultKeychainArg = defaultKeychain.asRubyArgument(name: "default_keychain", type: nil) + let unlockArg = unlock.asRubyArgument(name: "unlock", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let lockWhenSleepsArg = lockWhenSleeps.asRubyArgument(name: "lock_when_sleeps", type: nil) + let lockAfterTimeoutArg = lockAfterTimeout.asRubyArgument(name: "lock_after_timeout", type: nil) + let addToSearchListArg = addToSearchList.asRubyArgument(name: "add_to_search_list", type: nil) + let requireCreateArg = requireCreate.asRubyArgument(name: "require_create", type: nil) + let array: [RubyCommand.Argument?] = [nameArg, + pathArg, + passwordArg, + defaultKeychainArg, + unlockArg, + timeoutArg, + lockWhenSleepsArg, + lockAfterTimeoutArg, + addToSearchListArg, + requireCreateArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "create_keychain", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will create a new pull request on GitHub + + - parameters: + - apiToken: Personal API Token for GitHub - generate one at https://github.com/settings/tokens + - apiBearer: Use a Bearer authorization token. Usually generated by Github Apps, e.g. GitHub Actions GITHUB_TOKEN environment variable + - repo: The name of the repository you want to submit the pull request to + - title: The title of the pull request + - body: The contents of the pull request + - draft: Indicates whether the pull request is a draft + - labels: The labels for the pull request + - milestone: The milestone ID (Integer) for the pull request + - head: The name of the branch where your changes are implemented (defaults to the current branch name) + - base: The name of the branch you want your changes pulled into (defaults to `master`) + - apiUrl: The URL of GitHub API - used when the Enterprise (default to `https://api.github.com`) + - assignees: The assignees for the pull request + - reviewers: The reviewers (slug) for the pull request + - teamReviewers: The team reviewers (slug) for the pull request + + - returns: The pull request URL when successful + */ +public func createPullRequest(apiToken: OptionalConfigValue = .fastlaneDefault(nil), + apiBearer: OptionalConfigValue = .fastlaneDefault(nil), + repo: String, + title: String, + body: OptionalConfigValue = .fastlaneDefault(nil), + draft: OptionalConfigValue = .fastlaneDefault(nil), + labels: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + milestone: OptionalConfigValue = .fastlaneDefault(nil), + head: String = "master", + base: String = "master", + apiUrl: String = "https://api.github.com", + assignees: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + reviewers: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + teamReviewers: OptionalConfigValue<[String]?> = .fastlaneDefault(nil)) +{ + let apiTokenArg = apiToken.asRubyArgument(name: "api_token", type: nil) + let apiBearerArg = apiBearer.asRubyArgument(name: "api_bearer", type: nil) + let repoArg = RubyCommand.Argument(name: "repo", value: repo, type: nil) + let titleArg = RubyCommand.Argument(name: "title", value: title, type: nil) + let bodyArg = body.asRubyArgument(name: "body", type: nil) + let draftArg = draft.asRubyArgument(name: "draft", type: nil) + let labelsArg = labels.asRubyArgument(name: "labels", type: nil) + let milestoneArg = milestone.asRubyArgument(name: "milestone", type: nil) + let headArg = RubyCommand.Argument(name: "head", value: head, type: nil) + let baseArg = RubyCommand.Argument(name: "base", value: base, type: nil) + let apiUrlArg = RubyCommand.Argument(name: "api_url", value: apiUrl, type: nil) + let assigneesArg = assignees.asRubyArgument(name: "assignees", type: nil) + let reviewersArg = reviewers.asRubyArgument(name: "reviewers", type: nil) + let teamReviewersArg = teamReviewers.asRubyArgument(name: "team_reviewers", type: nil) + let array: [RubyCommand.Argument?] = [apiTokenArg, + apiBearerArg, + repoArg, + titleArg, + bodyArg, + draftArg, + labelsArg, + milestoneArg, + headArg, + baseArg, + apiUrlArg, + assigneesArg, + reviewersArg, + teamReviewersArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "create_pull_request", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Package multiple build configs of a library/framework into a single xcframework + + - parameters: + - frameworks: Frameworks (without dSYMs) to add to the target xcframework + - frameworksWithDsyms: Frameworks (with dSYMs) to add to the target xcframework + - libraries: Libraries (without headers or dSYMs) to add to the target xcframework + - librariesWithHeadersOrDsyms: Libraries (with headers or dSYMs) to add to the target xcframework + - output: The path to write the xcframework to + - allowInternalDistribution: Specifies that the created xcframework contains information not suitable for public distribution + + Utility for packaging multiple build configurations of a given library + or framework into a single xcframework. + + If you want to package several frameworks just provide one of: + + * An array containing the list of frameworks using the :frameworks parameter + (if they have no associated dSYMs): + ['FrameworkA.framework', 'FrameworkB.framework'] + + * A hash containing the list of frameworks with their dSYMs using the + :frameworks_with_dsyms parameter: + { + 'FrameworkA.framework' => {}, + 'FrameworkB.framework' => { dsyms: 'FrameworkB.framework.dSYM' } + } + + If you want to package several libraries just provide one of: + + * An array containing the list of libraries using the :libraries parameter + (if they have no associated headers or dSYMs): + ['LibraryA.so', 'LibraryB.so'] + + * A hash containing the list of libraries with their headers and dSYMs + using the :libraries_with_headers_or_dsyms parameter: + { + 'LibraryA.so' => { dsyms: 'libraryA.so.dSYM' }, + 'LibraryB.so' => { headers: 'headers' } + } + + Finally specify the location of the xcframework to be generated using the :output + parameter. + + */ +public func createXcframework(frameworks: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + frameworksWithDsyms: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + libraries: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + librariesWithHeadersOrDsyms: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + output: String, + allowInternalDistribution: OptionalConfigValue = .fastlaneDefault(false)) +{ + let frameworksArg = frameworks.asRubyArgument(name: "frameworks", type: nil) + let frameworksWithDsymsArg = frameworksWithDsyms.asRubyArgument(name: "frameworks_with_dsyms", type: nil) + let librariesArg = libraries.asRubyArgument(name: "libraries", type: nil) + let librariesWithHeadersOrDsymsArg = librariesWithHeadersOrDsyms.asRubyArgument(name: "libraries_with_headers_or_dsyms", type: nil) + let outputArg = RubyCommand.Argument(name: "output", value: output, type: nil) + let allowInternalDistributionArg = allowInternalDistribution.asRubyArgument(name: "allow_internal_distribution", type: nil) + let array: [RubyCommand.Argument?] = [frameworksArg, + frameworksWithDsymsArg, + librariesArg, + librariesWithHeadersOrDsymsArg, + outputArg, + allowInternalDistributionArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "create_xcframework", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs `danger` for the project + + - parameters: + - useBundleExec: Use bundle exec when there is a Gemfile presented + - verbose: Show more debugging information + - dangerId: The identifier of this Danger instance + - dangerfile: The location of your Dangerfile + - githubApiToken: GitHub API token for danger + - githubEnterpriseHost: GitHub host URL for GitHub Enterprise + - githubEnterpriseApiBaseUrl: GitHub API base URL for GitHub Enterprise + - failOnErrors: Should always fail the build process, defaults to false + - newComment: Makes Danger post a new comment instead of editing its previous one + - removePreviousComments: Makes Danger remove all previous comment and create a new one in the end of the list + - base: A branch/tag/commit to use as the base of the diff. [master|dev|stable] + - head: A branch/tag/commit to use as the head. [master|dev|stable] + - pr: Run danger on a specific pull request. e.g. "https://github.com/danger/danger/pull/518" + - failIfNoPr: Fail Danger execution if no PR is found + + Formalize your Pull Request etiquette. + More information: [https://github.com/danger/danger](https://github.com/danger/danger). + */ +public func danger(useBundleExec: OptionalConfigValue = .fastlaneDefault(true), + verbose: OptionalConfigValue = .fastlaneDefault(false), + dangerId: OptionalConfigValue = .fastlaneDefault(nil), + dangerfile: OptionalConfigValue = .fastlaneDefault(nil), + githubApiToken: OptionalConfigValue = .fastlaneDefault(nil), + githubEnterpriseHost: OptionalConfigValue = .fastlaneDefault(nil), + githubEnterpriseApiBaseUrl: OptionalConfigValue = .fastlaneDefault(nil), + failOnErrors: OptionalConfigValue = .fastlaneDefault(false), + newComment: OptionalConfigValue = .fastlaneDefault(false), + removePreviousComments: OptionalConfigValue = .fastlaneDefault(false), + base: OptionalConfigValue = .fastlaneDefault(nil), + head: OptionalConfigValue = .fastlaneDefault(nil), + pr: OptionalConfigValue = .fastlaneDefault(nil), + failIfNoPr: OptionalConfigValue = .fastlaneDefault(false)) +{ + let useBundleExecArg = useBundleExec.asRubyArgument(name: "use_bundle_exec", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let dangerIdArg = dangerId.asRubyArgument(name: "danger_id", type: nil) + let dangerfileArg = dangerfile.asRubyArgument(name: "dangerfile", type: nil) + let githubApiTokenArg = githubApiToken.asRubyArgument(name: "github_api_token", type: nil) + let githubEnterpriseHostArg = githubEnterpriseHost.asRubyArgument(name: "github_enterprise_host", type: nil) + let githubEnterpriseApiBaseUrlArg = githubEnterpriseApiBaseUrl.asRubyArgument(name: "github_enterprise_api_base_url", type: nil) + let failOnErrorsArg = failOnErrors.asRubyArgument(name: "fail_on_errors", type: nil) + let newCommentArg = newComment.asRubyArgument(name: "new_comment", type: nil) + let removePreviousCommentsArg = removePreviousComments.asRubyArgument(name: "remove_previous_comments", type: nil) + let baseArg = base.asRubyArgument(name: "base", type: nil) + let headArg = head.asRubyArgument(name: "head", type: nil) + let prArg = pr.asRubyArgument(name: "pr", type: nil) + let failIfNoPrArg = failIfNoPr.asRubyArgument(name: "fail_if_no_pr", type: nil) + let array: [RubyCommand.Argument?] = [useBundleExecArg, + verboseArg, + dangerIdArg, + dangerfileArg, + githubApiTokenArg, + githubEnterpriseHostArg, + githubEnterpriseApiBaseUrlArg, + failOnErrorsArg, + newCommentArg, + removePreviousCommentsArg, + baseArg, + headArg, + prArg, + failIfNoPrArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "danger", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Print out an overview of the lane context values + */ +public func debug() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "debug", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Defines a default platform to not have to specify the platform + */ +public func defaultPlatform() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "default_platform", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Delete keychains and remove them from the search list + + - parameters: + - name: Keychain name + - keychainPath: Keychain path + + Keychains can be deleted after being created with `create_keychain` + */ +public func deleteKeychain(name: OptionalConfigValue = .fastlaneDefault(nil), + keychainPath: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let nameArg = name.asRubyArgument(name: "name", type: nil) + let keychainPathArg = keychainPath.asRubyArgument(name: "keychain_path", type: nil) + let array: [RubyCommand.Argument?] = [nameArg, + keychainPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "delete_keychain", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `upload_to_app_store` action + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - appIdentifier: The bundle identifier of your app + - appVersion: The version that should be edited or created + - ipa: Path to your ipa file + - pkg: Path to your pkg file + - buildNumber: If set the given build number (already uploaded to iTC) will be used instead of the current built one + - platform: The platform to use (optional) + - editLive: Modify live metadata, this option disables ipa upload and screenshot upload + - useLiveVersion: Force usage of live version rather than edit version + - metadataPath: Path to the folder containing the metadata files + - screenshotsPath: Path to the folder containing the screenshots + - skipBinaryUpload: Skip uploading an ipa or pkg to App Store Connect + - skipScreenshots: Don't upload the screenshots + - skipMetadata: Don't upload the metadata (e.g. title, description). This will still upload screenshots + - skipAppVersionUpdate: Don’t create or update the app version that is being prepared for submission + - force: Skip verification of HTML preview file + - overwriteScreenshots: Clear all previously uploaded screenshots before uploading the new ones + - syncScreenshots: Sync screenshots with local ones. This is currently beta option so set true to 'FASTLANE_ENABLE_BETA_DELIVER_SYNC_SCREENSHOTS' environment variable as well + - submitForReview: Submit the new version for Review after uploading everything + - verifyOnly: Verifies archive with App Store Connect without uploading + - rejectIfPossible: Rejects the previously submitted build if it's in a state where it's possible + - automaticRelease: Should the app be automatically released once it's approved? (Can not be used together with `auto_release_date`) + - autoReleaseDate: Date in milliseconds for automatically releasing on pending approval (Can not be used together with `automatic_release`) + - phasedRelease: Enable the phased release feature of iTC + - resetRatings: Reset the summary rating when you release a new version of the application + - priceTier: The price tier of this application + - appRatingConfigPath: Path to the app rating's config + - submissionInformation: Extra information for the submission (e.g. compliance specifications, IDFA settings) + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - devPortalTeamId: The short ID of your Developer Portal team, if you're in multiple teams. Different from your iTC team ID! + - devPortalTeamName: The name of your Developer Portal team if you're in multiple teams + - itcProvider: The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + - runPrecheckBeforeSubmit: Run precheck before submitting to app review + - precheckDefaultRuleLevel: The default precheck rule level unless otherwise configured + - individualMetadataItems: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - An array of localized metadata items to upload individually by language so that errors can be identified. E.g. ['name', 'keywords', 'description']. Note: slow + - appIcon: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the app icon + - appleWatchAppIcon: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the Apple Watch app icon + - copyright: Metadata: The copyright notice + - primaryCategory: Metadata: The english name of the primary category (e.g. `Business`, `Books`) + - secondaryCategory: Metadata: The english name of the secondary category (e.g. `Business`, `Books`) + - primaryFirstSubCategory: Metadata: The english name of the primary first sub category (e.g. `Educational`, `Puzzle`) + - primarySecondSubCategory: Metadata: The english name of the primary second sub category (e.g. `Educational`, `Puzzle`) + - secondaryFirstSubCategory: Metadata: The english name of the secondary first sub category (e.g. `Educational`, `Puzzle`) + - secondarySecondSubCategory: Metadata: The english name of the secondary second sub category (e.g. `Educational`, `Puzzle`) + - tradeRepresentativeContactInformation: **DEPRECATED!** This is no longer used by App Store Connect - Metadata: A hash containing the trade representative contact information + - appReviewInformation: Metadata: A hash containing the review information + - appReviewAttachmentFile: Metadata: Path to the app review attachment file + - description: Metadata: The localised app description + - name: Metadata: The localised app name + - subtitle: Metadata: The localised app subtitle + - keywords: Metadata: An array of localised keywords + - promotionalText: Metadata: An array of localised promotional texts + - releaseNotes: Metadata: Localised release notes for this version + - privacyUrl: Metadata: Localised privacy url + - appleTvPrivacyPolicy: Metadata: Localised Apple TV privacy policy text + - supportUrl: Metadata: Localised support url + - marketingUrl: Metadata: Localised marketing url + - languages: Metadata: List of languages to activate + - ignoreLanguageDirectoryValidation: Ignore errors when invalid languages are found in metadata and screenshot directories + - precheckIncludeInAppPurchases: Should precheck check in-app purchases? + - app: The (spaceship) app ID of the app you want to use/modify + + Using _upload_to_app_store_ after _build_app_ and _capture_screenshots_ will automatically upload the latest ipa and screenshots with no other configuration. + + If you don't want to verify an HTML preview for App Store builds, use the `:force` option. + This is useful when running _fastlane_ on your Continuous Integration server: + `_upload_to_app_store_(force: true)` + If your account is on multiple teams and you need to tell the `iTMSTransporter` which 'provider' to use, you can set the `:itc_provider` option to pass this info. + */ +public func deliver(apiKeyPath: OptionalConfigValue = .fastlaneDefault(deliverfile.apiKeyPath), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.apiKey), + username: OptionalConfigValue = .fastlaneDefault(deliverfile.username), + appIdentifier: OptionalConfigValue = .fastlaneDefault(deliverfile.appIdentifier), + appVersion: OptionalConfigValue = .fastlaneDefault(deliverfile.appVersion), + ipa: OptionalConfigValue = .fastlaneDefault(deliverfile.ipa), + pkg: OptionalConfigValue = .fastlaneDefault(deliverfile.pkg), + buildNumber: OptionalConfigValue = .fastlaneDefault(deliverfile.buildNumber), + platform: String = deliverfile.platform, + editLive: OptionalConfigValue = .fastlaneDefault(deliverfile.editLive), + useLiveVersion: OptionalConfigValue = .fastlaneDefault(deliverfile.useLiveVersion), + metadataPath: OptionalConfigValue = .fastlaneDefault(deliverfile.metadataPath), + screenshotsPath: OptionalConfigValue = .fastlaneDefault(deliverfile.screenshotsPath), + skipBinaryUpload: OptionalConfigValue = .fastlaneDefault(deliverfile.skipBinaryUpload), + skipScreenshots: OptionalConfigValue = .fastlaneDefault(deliverfile.skipScreenshots), + skipMetadata: OptionalConfigValue = .fastlaneDefault(deliverfile.skipMetadata), + skipAppVersionUpdate: OptionalConfigValue = .fastlaneDefault(deliverfile.skipAppVersionUpdate), + force: OptionalConfigValue = .fastlaneDefault(deliverfile.force), + overwriteScreenshots: OptionalConfigValue = .fastlaneDefault(deliverfile.overwriteScreenshots), + syncScreenshots: OptionalConfigValue = .fastlaneDefault(deliverfile.syncScreenshots), + submitForReview: OptionalConfigValue = .fastlaneDefault(deliverfile.submitForReview), + verifyOnly: OptionalConfigValue = .fastlaneDefault(deliverfile.verifyOnly), + rejectIfPossible: OptionalConfigValue = .fastlaneDefault(deliverfile.rejectIfPossible), + automaticRelease: OptionalConfigValue = .fastlaneDefault(deliverfile.automaticRelease), + autoReleaseDate: OptionalConfigValue = .fastlaneDefault(deliverfile.autoReleaseDate), + phasedRelease: OptionalConfigValue = .fastlaneDefault(deliverfile.phasedRelease), + resetRatings: OptionalConfigValue = .fastlaneDefault(deliverfile.resetRatings), + priceTier: OptionalConfigValue = .fastlaneDefault(deliverfile.priceTier), + appRatingConfigPath: OptionalConfigValue = .fastlaneDefault(deliverfile.appRatingConfigPath), + submissionInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.submissionInformation), + teamId: OptionalConfigValue = .fastlaneDefault(deliverfile.teamId), + teamName: OptionalConfigValue = .fastlaneDefault(deliverfile.teamName), + devPortalTeamId: OptionalConfigValue = .fastlaneDefault(deliverfile.devPortalTeamId), + devPortalTeamName: OptionalConfigValue = .fastlaneDefault(deliverfile.devPortalTeamName), + itcProvider: OptionalConfigValue = .fastlaneDefault(deliverfile.itcProvider), + runPrecheckBeforeSubmit: OptionalConfigValue = .fastlaneDefault(deliverfile.runPrecheckBeforeSubmit), + precheckDefaultRuleLevel: Any = deliverfile.precheckDefaultRuleLevel, + individualMetadataItems: OptionalConfigValue<[String]?> = .fastlaneDefault(deliverfile.individualMetadataItems), + appIcon: OptionalConfigValue = .fastlaneDefault(deliverfile.appIcon), + appleWatchAppIcon: OptionalConfigValue = .fastlaneDefault(deliverfile.appleWatchAppIcon), + copyright: OptionalConfigValue = .fastlaneDefault(deliverfile.copyright), + primaryCategory: OptionalConfigValue = .fastlaneDefault(deliverfile.primaryCategory), + secondaryCategory: OptionalConfigValue = .fastlaneDefault(deliverfile.secondaryCategory), + primaryFirstSubCategory: OptionalConfigValue = .fastlaneDefault(deliverfile.primaryFirstSubCategory), + primarySecondSubCategory: OptionalConfigValue = .fastlaneDefault(deliverfile.primarySecondSubCategory), + secondaryFirstSubCategory: OptionalConfigValue = .fastlaneDefault(deliverfile.secondaryFirstSubCategory), + secondarySecondSubCategory: OptionalConfigValue = .fastlaneDefault(deliverfile.secondarySecondSubCategory), + tradeRepresentativeContactInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.tradeRepresentativeContactInformation), + appReviewInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.appReviewInformation), + appReviewAttachmentFile: OptionalConfigValue = .fastlaneDefault(deliverfile.appReviewAttachmentFile), + description: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.description), + name: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.name), + subtitle: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.subtitle), + keywords: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.keywords), + promotionalText: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.promotionalText), + releaseNotes: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.releaseNotes), + privacyUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.privacyUrl), + appleTvPrivacyPolicy: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.appleTvPrivacyPolicy), + supportUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.supportUrl), + marketingUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(deliverfile.marketingUrl), + languages: OptionalConfigValue<[String]?> = .fastlaneDefault(deliverfile.languages), + ignoreLanguageDirectoryValidation: OptionalConfigValue = .fastlaneDefault(deliverfile.ignoreLanguageDirectoryValidation), + precheckIncludeInAppPurchases: OptionalConfigValue = .fastlaneDefault(deliverfile.precheckIncludeInAppPurchases), + app: OptionalConfigValue = .fastlaneDefault(deliverfile.app)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let pkgArg = pkg.asRubyArgument(name: "pkg", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let editLiveArg = editLive.asRubyArgument(name: "edit_live", type: nil) + let useLiveVersionArg = useLiveVersion.asRubyArgument(name: "use_live_version", type: nil) + let metadataPathArg = metadataPath.asRubyArgument(name: "metadata_path", type: nil) + let screenshotsPathArg = screenshotsPath.asRubyArgument(name: "screenshots_path", type: nil) + let skipBinaryUploadArg = skipBinaryUpload.asRubyArgument(name: "skip_binary_upload", type: nil) + let skipScreenshotsArg = skipScreenshots.asRubyArgument(name: "skip_screenshots", type: nil) + let skipMetadataArg = skipMetadata.asRubyArgument(name: "skip_metadata", type: nil) + let skipAppVersionUpdateArg = skipAppVersionUpdate.asRubyArgument(name: "skip_app_version_update", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let overwriteScreenshotsArg = overwriteScreenshots.asRubyArgument(name: "overwrite_screenshots", type: nil) + let syncScreenshotsArg = syncScreenshots.asRubyArgument(name: "sync_screenshots", type: nil) + let submitForReviewArg = submitForReview.asRubyArgument(name: "submit_for_review", type: nil) + let verifyOnlyArg = verifyOnly.asRubyArgument(name: "verify_only", type: nil) + let rejectIfPossibleArg = rejectIfPossible.asRubyArgument(name: "reject_if_possible", type: nil) + let automaticReleaseArg = automaticRelease.asRubyArgument(name: "automatic_release", type: nil) + let autoReleaseDateArg = autoReleaseDate.asRubyArgument(name: "auto_release_date", type: nil) + let phasedReleaseArg = phasedRelease.asRubyArgument(name: "phased_release", type: nil) + let resetRatingsArg = resetRatings.asRubyArgument(name: "reset_ratings", type: nil) + let priceTierArg = priceTier.asRubyArgument(name: "price_tier", type: nil) + let appRatingConfigPathArg = appRatingConfigPath.asRubyArgument(name: "app_rating_config_path", type: nil) + let submissionInformationArg = submissionInformation.asRubyArgument(name: "submission_information", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let devPortalTeamIdArg = devPortalTeamId.asRubyArgument(name: "dev_portal_team_id", type: nil) + let devPortalTeamNameArg = devPortalTeamName.asRubyArgument(name: "dev_portal_team_name", type: nil) + let itcProviderArg = itcProvider.asRubyArgument(name: "itc_provider", type: nil) + let runPrecheckBeforeSubmitArg = runPrecheckBeforeSubmit.asRubyArgument(name: "run_precheck_before_submit", type: nil) + let precheckDefaultRuleLevelArg = RubyCommand.Argument(name: "precheck_default_rule_level", value: precheckDefaultRuleLevel, type: nil) + let individualMetadataItemsArg = individualMetadataItems.asRubyArgument(name: "individual_metadata_items", type: nil) + let appIconArg = appIcon.asRubyArgument(name: "app_icon", type: nil) + let appleWatchAppIconArg = appleWatchAppIcon.asRubyArgument(name: "apple_watch_app_icon", type: nil) + let copyrightArg = copyright.asRubyArgument(name: "copyright", type: nil) + let primaryCategoryArg = primaryCategory.asRubyArgument(name: "primary_category", type: nil) + let secondaryCategoryArg = secondaryCategory.asRubyArgument(name: "secondary_category", type: nil) + let primaryFirstSubCategoryArg = primaryFirstSubCategory.asRubyArgument(name: "primary_first_sub_category", type: nil) + let primarySecondSubCategoryArg = primarySecondSubCategory.asRubyArgument(name: "primary_second_sub_category", type: nil) + let secondaryFirstSubCategoryArg = secondaryFirstSubCategory.asRubyArgument(name: "secondary_first_sub_category", type: nil) + let secondarySecondSubCategoryArg = secondarySecondSubCategory.asRubyArgument(name: "secondary_second_sub_category", type: nil) + let tradeRepresentativeContactInformationArg = tradeRepresentativeContactInformation.asRubyArgument(name: "trade_representative_contact_information", type: nil) + let appReviewInformationArg = appReviewInformation.asRubyArgument(name: "app_review_information", type: nil) + let appReviewAttachmentFileArg = appReviewAttachmentFile.asRubyArgument(name: "app_review_attachment_file", type: nil) + let descriptionArg = description.asRubyArgument(name: "description", type: nil) + let nameArg = name.asRubyArgument(name: "name", type: nil) + let subtitleArg = subtitle.asRubyArgument(name: "subtitle", type: nil) + let keywordsArg = keywords.asRubyArgument(name: "keywords", type: nil) + let promotionalTextArg = promotionalText.asRubyArgument(name: "promotional_text", type: nil) + let releaseNotesArg = releaseNotes.asRubyArgument(name: "release_notes", type: nil) + let privacyUrlArg = privacyUrl.asRubyArgument(name: "privacy_url", type: nil) + let appleTvPrivacyPolicyArg = appleTvPrivacyPolicy.asRubyArgument(name: "apple_tv_privacy_policy", type: nil) + let supportUrlArg = supportUrl.asRubyArgument(name: "support_url", type: nil) + let marketingUrlArg = marketingUrl.asRubyArgument(name: "marketing_url", type: nil) + let languagesArg = languages.asRubyArgument(name: "languages", type: nil) + let ignoreLanguageDirectoryValidationArg = ignoreLanguageDirectoryValidation.asRubyArgument(name: "ignore_language_directory_validation", type: nil) + let precheckIncludeInAppPurchasesArg = precheckIncludeInAppPurchases.asRubyArgument(name: "precheck_include_in_app_purchases", type: nil) + let appArg = app.asRubyArgument(name: "app", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + appVersionArg, + ipaArg, + pkgArg, + buildNumberArg, + platformArg, + editLiveArg, + useLiveVersionArg, + metadataPathArg, + screenshotsPathArg, + skipBinaryUploadArg, + skipScreenshotsArg, + skipMetadataArg, + skipAppVersionUpdateArg, + forceArg, + overwriteScreenshotsArg, + syncScreenshotsArg, + submitForReviewArg, + verifyOnlyArg, + rejectIfPossibleArg, + automaticReleaseArg, + autoReleaseDateArg, + phasedReleaseArg, + resetRatingsArg, + priceTierArg, + appRatingConfigPathArg, + submissionInformationArg, + teamIdArg, + teamNameArg, + devPortalTeamIdArg, + devPortalTeamNameArg, + itcProviderArg, + runPrecheckBeforeSubmitArg, + precheckDefaultRuleLevelArg, + individualMetadataItemsArg, + appIconArg, + appleWatchAppIconArg, + copyrightArg, + primaryCategoryArg, + secondaryCategoryArg, + primaryFirstSubCategoryArg, + primarySecondSubCategoryArg, + secondaryFirstSubCategoryArg, + secondarySecondSubCategoryArg, + tradeRepresentativeContactInformationArg, + appReviewInformationArg, + appReviewAttachmentFileArg, + descriptionArg, + nameArg, + subtitleArg, + keywordsArg, + promotionalTextArg, + releaseNotesArg, + privacyUrlArg, + appleTvPrivacyPolicyArg, + supportUrlArg, + marketingUrlArg, + languagesArg, + ignoreLanguageDirectoryValidationArg, + precheckIncludeInAppPurchasesArg, + appArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "deliver", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload a new build to [DeployGate](https://deploygate.com/) + + - parameters: + - apiToken: Deploygate API Token + - user: Target username or organization name + - ipa: Path to your IPA file. Optional if you use the _gym_ or _xcodebuild_ action + - apk: Path to your APK file + - message: Release Notes + - distributionKey: Target Distribution Key + - releaseNote: Release note for distribution page + - disableNotify: Disables Push notification emails + - distributionName: Target Distribution Name + + You can retrieve your username and API token on [your settings page](https://deploygate.com/settings). + More information about the available options can be found in the [DeployGate Push API document](https://deploygate.com/docs/api). + */ +public func deploygate(apiToken: String, + user: String, + ipa: OptionalConfigValue = .fastlaneDefault(nil), + apk: OptionalConfigValue = .fastlaneDefault(nil), + message: String = "No changelog provided", + distributionKey: OptionalConfigValue = .fastlaneDefault(nil), + releaseNote: OptionalConfigValue = .fastlaneDefault(nil), + disableNotify: OptionalConfigValue = .fastlaneDefault(false), + distributionName: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let userArg = RubyCommand.Argument(name: "user", value: user, type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let apkArg = apk.asRubyArgument(name: "apk", type: nil) + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let distributionKeyArg = distributionKey.asRubyArgument(name: "distribution_key", type: nil) + let releaseNoteArg = releaseNote.asRubyArgument(name: "release_note", type: nil) + let disableNotifyArg = disableNotify.asRubyArgument(name: "disable_notify", type: nil) + let distributionNameArg = distributionName.asRubyArgument(name: "distribution_name", type: nil) + let array: [RubyCommand.Argument?] = [apiTokenArg, + userArg, + ipaArg, + apkArg, + messageArg, + distributionKeyArg, + releaseNoteArg, + disableNotifyArg, + distributionNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "deploygate", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Reads in production secrets set in a dotgpg file and puts them in ENV + + - parameter dotgpgFile: Path to your gpg file + + More information about dotgpg can be found at [https://github.com/ConradIrwin/dotgpg](https://github.com/ConradIrwin/dotgpg). + */ +public func dotgpgEnvironment(dotgpgFile: String) { + let dotgpgFileArg = RubyCommand.Argument(name: "dotgpg_file", value: dotgpgFile, type: nil) + let array: [RubyCommand.Argument?] = [dotgpgFileArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "dotgpg_environment", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Download a file from a remote server (e.g. JSON file) + + - parameter url: The URL that should be downloaded + + Specify the URL to download and get the content as a return value. + Automatically parses JSON into a Ruby data structure. + For more advanced networking code, use the Ruby functions instead: [http://docs.ruby-lang.org/en/2.0.0/Net/HTTP.html](http://docs.ruby-lang.org/en/2.0.0/Net/HTTP.html). + */ +public func download(url: String) { + let urlArg = RubyCommand.Argument(name: "url", value: url, type: nil) + let array: [RubyCommand.Argument?] = [urlArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "download", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Download App Privacy Details from an app in App Store Connect + + - parameters: + - username: Your Apple ID Username for App Store Connect + - appIdentifier: The bundle identifier of your app + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - outputJsonPath: Path to the app usage data JSON file generated by interactive questions + + Download App Privacy Details from an app in App Store Connect. For more detail information, view https://docs.fastlane.tools/uploading-app-privacy-details + */ +public func downloadAppPrivacyDetailsFromAppStore(username: String, + appIdentifier: String, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + outputJsonPath: String = "./fastlane/app_privacy_details.json") +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let outputJsonPathArg = RubyCommand.Argument(name: "output_json_path", value: outputJsonPath, type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + appIdentifierArg, + teamIdArg, + teamNameArg, + outputJsonPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "download_app_privacy_details_from_app_store", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Download dSYM files from App Store Connect for Bitcode apps + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#use-return-value-and-pass-in-as-an-option) + - username: Your Apple ID Username for App Store Connect + - appIdentifier: The bundle identifier of your app + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - platform: The app platform for dSYMs you wish to download (ios, appletvos) + - version: The app version for dSYMs you wish to download, pass in 'latest' to download only the latest build's dSYMs or 'live' to download only the live version dSYMs + - buildNumber: The app build_number for dSYMs you wish to download + - minVersion: The minimum app version for dSYMs you wish to download + - afterUploadedDate: The uploaded date after which you wish to download dSYMs + - outputDirectory: Where to save the download dSYMs, defaults to the current path + - waitForDsymProcessing: Wait for dSYMs to process + - waitTimeout: Number of seconds to wait for dSYMs to process + + This action downloads dSYM files from App Store Connect after the ipa gets re-compiled by Apple. Useful if you have Bitcode enabled.| + | + ```ruby| + lane :refresh_dsyms do| + download_dsyms # Download dSYM files from iTC| + upload_symbols_to_crashlytics # Upload them to Crashlytics| + clean_build_artifacts # Delete the local dSYM files| + end| + ```| + >| + */ +public func downloadDsyms(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: String, + appIdentifier: String, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + version: OptionalConfigValue = .fastlaneDefault(nil), + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + minVersion: OptionalConfigValue = .fastlaneDefault(nil), + afterUploadedDate: OptionalConfigValue = .fastlaneDefault(nil), + outputDirectory: OptionalConfigValue = .fastlaneDefault(nil), + waitForDsymProcessing: OptionalConfigValue = .fastlaneDefault(false), + waitTimeout: Int = 300) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let versionArg = version.asRubyArgument(name: "version", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let minVersionArg = minVersion.asRubyArgument(name: "min_version", type: nil) + let afterUploadedDateArg = afterUploadedDate.asRubyArgument(name: "after_uploaded_date", type: nil) + let outputDirectoryArg = outputDirectory.asRubyArgument(name: "output_directory", type: nil) + let waitForDsymProcessingArg = waitForDsymProcessing.asRubyArgument(name: "wait_for_dsym_processing", type: nil) + let waitTimeoutArg = RubyCommand.Argument(name: "wait_timeout", value: waitTimeout, type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + teamIdArg, + teamNameArg, + platformArg, + versionArg, + buildNumberArg, + minVersionArg, + afterUploadedDateArg, + outputDirectoryArg, + waitForDsymProcessingArg, + waitTimeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "download_dsyms", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Download metadata and binaries from Google Play (via _supply_) + + - parameters: + - packageName: The package name of the application to use + - versionName: Version name (used when uploading new apks/aabs) - defaults to 'versionName' in build.gradle or AndroidManifest.xml + - track: The track of the application to use. The default available tracks are: production, beta, alpha, internal + - metadataPath: Path to the directory containing the metadata files + - key: **DEPRECATED!** Use `--json_key` instead - The p12 File used to authenticate with Google + - issuer: **DEPRECATED!** Use `--json_key` instead - The issuer of the p12 file (email address of the service account) + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - timeout: Timeout for read, open, and send (in seconds) + + More information: https://docs.fastlane.tools/actions/download_from_play_store/ + */ +public func downloadFromPlayStore(packageName: String, + versionName: OptionalConfigValue = .fastlaneDefault(nil), + track: String = "production", + metadataPath: OptionalConfigValue = .fastlaneDefault(nil), + key: OptionalConfigValue = .fastlaneDefault(nil), + issuer: OptionalConfigValue = .fastlaneDefault(nil), + jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 300) +{ + let packageNameArg = RubyCommand.Argument(name: "package_name", value: packageName, type: nil) + let versionNameArg = versionName.asRubyArgument(name: "version_name", type: nil) + let trackArg = RubyCommand.Argument(name: "track", value: track, type: nil) + let metadataPathArg = metadataPath.asRubyArgument(name: "metadata_path", type: nil) + let keyArg = key.asRubyArgument(name: "key", type: nil) + let issuerArg = issuer.asRubyArgument(name: "issuer", type: nil) + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [packageNameArg, + versionNameArg, + trackArg, + metadataPathArg, + keyArg, + issuerArg, + jsonKeyArg, + jsonKeyDataArg, + rootUrlArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "download_from_play_store", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Creates a zipped dSYM in the project root from the .xcarchive + + - parameters: + - archivePath: Path to your xcarchive file. Optional if you use the `xcodebuild` action + - dsymPath: Path for generated dsym. Optional, default is your apps root directory + - all: Whether or not all dSYM files are to be included. Optional, default is false in which only your app dSYM is included + + You can manually specify the path to the xcarchive (not needed if you use `xcodebuild`/`xcarchive` to build your archive) + */ +public func dsymZip(archivePath: OptionalConfigValue = .fastlaneDefault(nil), + dsymPath: OptionalConfigValue = .fastlaneDefault(nil), + all: OptionalConfigValue = .fastlaneDefault(false)) +{ + let archivePathArg = archivePath.asRubyArgument(name: "archive_path", type: nil) + let dsymPathArg = dsymPath.asRubyArgument(name: "dsym_path", type: nil) + let allArg = all.asRubyArgument(name: "all", type: nil) + let array: [RubyCommand.Argument?] = [archivePathArg, + dsymPathArg, + allArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "dsym_zip", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `puts` action + + - parameter message: Message to be printed out + */ +public func echo(message: OptionalConfigValue = .fastlaneDefault(nil)) { + let messageArg = message.asRubyArgument(name: "message", type: nil) + let array: [RubyCommand.Argument?] = [messageArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "echo", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Raises an exception if not using `bundle exec` to run fastlane + + This action will check if you are using `bundle exec` to run fastlane. + You can put it into `before_all` to make sure that fastlane is ran using the `bundle exec fastlane` command. + */ +public func ensureBundleExec() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "ensure_bundle_exec", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Raises an exception if the specified env vars are not set + + - parameter envVars: The environment variables names that should be checked + + This action will check if some environment variables are set. + */ +public func ensureEnvVars(envVars: [String]) { + let envVarsArg = RubyCommand.Argument(name: "env_vars", value: envVars, type: nil) + let array: [RubyCommand.Argument?] = [envVarsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ensure_env_vars", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Raises an exception if not on a specific git branch + + - parameter branch: The branch that should be checked for. String that can be either the full name of the branch or a regex e.g. `^feature/.*$` to match + + This action will check if your git repo is checked out to a specific branch. + You may only want to make releases from a specific branch, so `ensure_git_branch` will stop a lane if it was accidentally executed on an incorrect branch. + */ +public func ensureGitBranch(branch: String = "master") { + let branchArg = RubyCommand.Argument(name: "branch", value: branch, type: nil) + let array: [RubyCommand.Argument?] = [branchArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ensure_git_branch", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Raises an exception if there are uncommitted git changes + + - parameters: + - showUncommittedChanges: The flag whether to show uncommitted changes if the repo is dirty + - showDiff: The flag whether to show the git diff if the repo is dirty + - ignored: The flag whether to ignore file the git status if the repo is dirty + + A sanity check to make sure you are working in a repo that is clean. + Especially useful to put at the beginning of your Fastfile in the `before_all` block, if some of your other actions will touch your filesystem, do things to your git repo, or just as a general reminder to save your work. + Also needed as a prerequisite for some other actions like `reset_git_repo`. + */ +public func ensureGitStatusClean(showUncommittedChanges: OptionalConfigValue = .fastlaneDefault(false), + showDiff: OptionalConfigValue = .fastlaneDefault(false), + ignored: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let showUncommittedChangesArg = showUncommittedChanges.asRubyArgument(name: "show_uncommitted_changes", type: nil) + let showDiffArg = showDiff.asRubyArgument(name: "show_diff", type: nil) + let ignoredArg = ignored.asRubyArgument(name: "ignored", type: nil) + let array: [RubyCommand.Argument?] = [showUncommittedChangesArg, + showDiffArg, + ignoredArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ensure_git_status_clean", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Ensures the given text is nowhere in the code base + + - parameters: + - text: The text that must not be in the code base + - path: The directory containing all the source files + - extension: The extension that should be searched for + - extensions: An array of file extensions that should be searched for + - exclude: Exclude a certain pattern from the search + - excludeDirs: An array of dirs that should not be included in the search + + You don't want any debug code to slip into production. + This can be used to check if there is any debug code still in your codebase or if you have things like `// TO DO` or similar. + */ +public func ensureNoDebugCode(text: String, + path: String = ".", + extension: OptionalConfigValue = .fastlaneDefault(nil), + extensions: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + exclude: OptionalConfigValue = .fastlaneDefault(nil), + excludeDirs: OptionalConfigValue<[String]?> = .fastlaneDefault(nil)) +{ + let textArg = RubyCommand.Argument(name: "text", value: text, type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let extensionArg = `extension`.asRubyArgument(name: "extension", type: nil) + let extensionsArg = extensions.asRubyArgument(name: "extensions", type: nil) + let excludeArg = exclude.asRubyArgument(name: "exclude", type: nil) + let excludeDirsArg = excludeDirs.asRubyArgument(name: "exclude_dirs", type: nil) + let array: [RubyCommand.Argument?] = [textArg, + pathArg, + extensionArg, + extensionsArg, + excludeArg, + excludeDirsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ensure_no_debug_code", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Ensure the right version of Xcode is used + + - parameters: + - version: Xcode version to verify that is selected + - strict: Should the version be verified strictly (all 3 version numbers), or matching only the given version numbers (i.e. `11.3` == `11.3.x`) + + If building your app requires a specific version of Xcode, you can invoke this command before using gym. + For example, to ensure that a beta version of Xcode is not accidentally selected to build, which would make uploading to TestFlight fail. + You can either manually provide a specific version using `version:` or you make use of the `.xcode-version` file. + Using the `strict` parameter, you can either verify the full set of version numbers strictly (i.e. `11.3.1`) or only a subset of them (i.e. `11.3` or `11`). + */ +public func ensureXcodeVersion(version: OptionalConfigValue = .fastlaneDefault(nil), + strict: OptionalConfigValue = .fastlaneDefault(true)) +{ + let versionArg = version.asRubyArgument(name: "version", type: nil) + let strictArg = strict.asRubyArgument(name: "strict", type: nil) + let array: [RubyCommand.Argument?] = [versionArg, + strictArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ensure_xcode_version", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Sets/gets env vars for Fastlane.swift. Don't use in ruby, use `ENV[key] = val` + + - parameters: + - set: Set the environment variables named + - get: Get the environment variable named + - remove: Remove the environment variable named + */ +@discardableResult public func environmentVariable(set: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + get: OptionalConfigValue = .fastlaneDefault(nil), + remove: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let setArg = set.asRubyArgument(name: "set", type: nil) + let getArg = get.asRubyArgument(name: "get", type: nil) + let removeArg = remove.asRubyArgument(name: "remove", type: nil) + let array: [RubyCommand.Argument?] = [setArg, + getArg, + removeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "environment_variable", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Allows to Generate output files based on ERB templates + + - parameters: + - template: ERB Template File + - destination: Destination file + - placeholders: Placeholders given as a hash + - trimMode: Trim mode applied to the ERB + + Renders an ERB template with `:placeholders` given as a hash via parameter. + If no `:destination` is set, it returns the rendered template as string. + */ +public func erb(template: String, + destination: OptionalConfigValue = .fastlaneDefault(nil), + placeholders: [String: Any] = [:], + trimMode: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let templateArg = RubyCommand.Argument(name: "template", value: template, type: nil) + let destinationArg = destination.asRubyArgument(name: "destination", type: nil) + let placeholdersArg = RubyCommand.Argument(name: "placeholders", value: placeholders, type: nil) + let trimModeArg = trimMode.asRubyArgument(name: "trim_mode", type: nil) + let array: [RubyCommand.Argument?] = [templateArg, + destinationArg, + placeholdersArg, + trimModeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "erb", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `min_fastlane_version` action + + Add this to your `Fastfile` to require a certain version of _fastlane_. + Use it if you use an action that just recently came out and you need it. + */ +public func fastlaneVersion() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "fastlane_version", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Send a message to a [Flock](https://flock.com/) group + + - parameters: + - message: Message text + - token: Token for the Flock incoming webhook + - baseUrl: Base URL of the Flock incoming message webhook + + To obtain the token, create a new [incoming message webhook](https://dev.flock.co/wiki/display/FlockAPI/Incoming+Webhooks) in your Flock admin panel. + */ +public func flock(message: String, + token: String, + baseUrl: String = "https://api.flock.co/hooks/sendMessage") +{ + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let tokenArg = RubyCommand.Argument(name: "token", value: token, type: nil) + let baseUrlArg = RubyCommand.Argument(name: "base_url", value: baseUrl, type: nil) + let array: [RubyCommand.Argument?] = [messageArg, + tokenArg, + baseUrlArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "flock", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Adds device frames around all screenshots (via _frameit_) + + - parameters: + - white: Use white device frames + - silver: Use white device frames. Alias for :white + - roseGold: Use rose gold device frames. Alias for :rose_gold + - gold: Use gold device frames. Alias for :gold + - forceDeviceType: Forces a given device type, useful for Mac screenshots, as their sizes vary + - useLegacyIphone5s: Use iPhone 5s instead of iPhone SE frames + - useLegacyIphone6s: Use iPhone 6s frames instead of iPhone 7 frames + - useLegacyIphone7: Use iPhone 7 frames instead of iPhone 8 frames + - useLegacyIphonex: Use iPhone X instead of iPhone XS frames + - useLegacyIphonexr: Use iPhone XR instead of iPhone 11 frames + - useLegacyIphonexs: Use iPhone XS instead of iPhone 11 Pro frames + - useLegacyIphonexsmax: Use iPhone XS Max instead of iPhone 11 Pro Max frames + - forceOrientationBlock: [Advanced] A block to customize your screenshots' device orientation + - debugMode: Output debug information in framed screenshots + - resume: Resume frameit instead of reprocessing all screenshots + - usePlatform: Choose a platform, the valid options are IOS, ANDROID and ANY (default is either general platform defined in the fastfile or IOS to ensure backward compatibility) + - path: The path to the directory containing the screenshots + + Uses [frameit](https://docs.fastlane.tools/actions/frameit/) to prepare perfect screenshots for the App Store, your website, QA or emails. + You can add background and titles to the framed screenshots as well. + */ +public func frameScreenshots(white: OptionalConfigValue = .fastlaneDefault(nil), + silver: OptionalConfigValue = .fastlaneDefault(nil), + roseGold: OptionalConfigValue = .fastlaneDefault(nil), + gold: OptionalConfigValue = .fastlaneDefault(nil), + forceDeviceType: OptionalConfigValue = .fastlaneDefault(nil), + useLegacyIphone5s: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphone6s: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphone7: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonex: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonexr: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonexs: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonexsmax: OptionalConfigValue = .fastlaneDefault(false), + forceOrientationBlock: ((String) -> Void)? = nil, + debugMode: OptionalConfigValue = .fastlaneDefault(false), + resume: OptionalConfigValue = .fastlaneDefault(false), + usePlatform: String = "IOS", + path: String = "./") +{ + let whiteArg = white.asRubyArgument(name: "white", type: nil) + let silverArg = silver.asRubyArgument(name: "silver", type: nil) + let roseGoldArg = roseGold.asRubyArgument(name: "rose_gold", type: nil) + let goldArg = gold.asRubyArgument(name: "gold", type: nil) + let forceDeviceTypeArg = forceDeviceType.asRubyArgument(name: "force_device_type", type: nil) + let useLegacyIphone5sArg = useLegacyIphone5s.asRubyArgument(name: "use_legacy_iphone5s", type: nil) + let useLegacyIphone6sArg = useLegacyIphone6s.asRubyArgument(name: "use_legacy_iphone6s", type: nil) + let useLegacyIphone7Arg = useLegacyIphone7.asRubyArgument(name: "use_legacy_iphone7", type: nil) + let useLegacyIphonexArg = useLegacyIphonex.asRubyArgument(name: "use_legacy_iphonex", type: nil) + let useLegacyIphonexrArg = useLegacyIphonexr.asRubyArgument(name: "use_legacy_iphonexr", type: nil) + let useLegacyIphonexsArg = useLegacyIphonexs.asRubyArgument(name: "use_legacy_iphonexs", type: nil) + let useLegacyIphonexsmaxArg = useLegacyIphonexsmax.asRubyArgument(name: "use_legacy_iphonexsmax", type: nil) + let forceOrientationBlockArg = RubyCommand.Argument(name: "force_orientation_block", value: forceOrientationBlock, type: .stringClosure) + let debugModeArg = debugMode.asRubyArgument(name: "debug_mode", type: nil) + let resumeArg = resume.asRubyArgument(name: "resume", type: nil) + let usePlatformArg = RubyCommand.Argument(name: "use_platform", value: usePlatform, type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [whiteArg, + silverArg, + roseGoldArg, + goldArg, + forceDeviceTypeArg, + useLegacyIphone5sArg, + useLegacyIphone6sArg, + useLegacyIphone7Arg, + useLegacyIphonexArg, + useLegacyIphonexrArg, + useLegacyIphonexsArg, + useLegacyIphonexsmaxArg, + forceOrientationBlockArg, + debugModeArg, + resumeArg, + usePlatformArg, + pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "frame_screenshots", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `frame_screenshots` action + + - parameters: + - white: Use white device frames + - silver: Use white device frames. Alias for :white + - roseGold: Use rose gold device frames. Alias for :rose_gold + - gold: Use gold device frames. Alias for :gold + - forceDeviceType: Forces a given device type, useful for Mac screenshots, as their sizes vary + - useLegacyIphone5s: Use iPhone 5s instead of iPhone SE frames + - useLegacyIphone6s: Use iPhone 6s frames instead of iPhone 7 frames + - useLegacyIphone7: Use iPhone 7 frames instead of iPhone 8 frames + - useLegacyIphonex: Use iPhone X instead of iPhone XS frames + - useLegacyIphonexr: Use iPhone XR instead of iPhone 11 frames + - useLegacyIphonexs: Use iPhone XS instead of iPhone 11 Pro frames + - useLegacyIphonexsmax: Use iPhone XS Max instead of iPhone 11 Pro Max frames + - forceOrientationBlock: [Advanced] A block to customize your screenshots' device orientation + - debugMode: Output debug information in framed screenshots + - resume: Resume frameit instead of reprocessing all screenshots + - usePlatform: Choose a platform, the valid options are IOS, ANDROID and ANY (default is either general platform defined in the fastfile or IOS to ensure backward compatibility) + - path: The path to the directory containing the screenshots + + Uses [frameit](https://docs.fastlane.tools/actions/frameit/) to prepare perfect screenshots for the App Store, your website, QA or emails. + You can add background and titles to the framed screenshots as well. + */ +public func frameit(white: OptionalConfigValue = .fastlaneDefault(nil), + silver: OptionalConfigValue = .fastlaneDefault(nil), + roseGold: OptionalConfigValue = .fastlaneDefault(nil), + gold: OptionalConfigValue = .fastlaneDefault(nil), + forceDeviceType: OptionalConfigValue = .fastlaneDefault(nil), + useLegacyIphone5s: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphone6s: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphone7: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonex: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonexr: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonexs: OptionalConfigValue = .fastlaneDefault(false), + useLegacyIphonexsmax: OptionalConfigValue = .fastlaneDefault(false), + forceOrientationBlock: ((String) -> Void)? = nil, + debugMode: OptionalConfigValue = .fastlaneDefault(false), + resume: OptionalConfigValue = .fastlaneDefault(false), + usePlatform: String = "IOS", + path: String = "./") +{ + let whiteArg = white.asRubyArgument(name: "white", type: nil) + let silverArg = silver.asRubyArgument(name: "silver", type: nil) + let roseGoldArg = roseGold.asRubyArgument(name: "rose_gold", type: nil) + let goldArg = gold.asRubyArgument(name: "gold", type: nil) + let forceDeviceTypeArg = forceDeviceType.asRubyArgument(name: "force_device_type", type: nil) + let useLegacyIphone5sArg = useLegacyIphone5s.asRubyArgument(name: "use_legacy_iphone5s", type: nil) + let useLegacyIphone6sArg = useLegacyIphone6s.asRubyArgument(name: "use_legacy_iphone6s", type: nil) + let useLegacyIphone7Arg = useLegacyIphone7.asRubyArgument(name: "use_legacy_iphone7", type: nil) + let useLegacyIphonexArg = useLegacyIphonex.asRubyArgument(name: "use_legacy_iphonex", type: nil) + let useLegacyIphonexrArg = useLegacyIphonexr.asRubyArgument(name: "use_legacy_iphonexr", type: nil) + let useLegacyIphonexsArg = useLegacyIphonexs.asRubyArgument(name: "use_legacy_iphonexs", type: nil) + let useLegacyIphonexsmaxArg = useLegacyIphonexsmax.asRubyArgument(name: "use_legacy_iphonexsmax", type: nil) + let forceOrientationBlockArg = RubyCommand.Argument(name: "force_orientation_block", value: forceOrientationBlock, type: .stringClosure) + let debugModeArg = debugMode.asRubyArgument(name: "debug_mode", type: nil) + let resumeArg = resume.asRubyArgument(name: "resume", type: nil) + let usePlatformArg = RubyCommand.Argument(name: "use_platform", value: usePlatform, type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [whiteArg, + silverArg, + roseGoldArg, + goldArg, + forceDeviceTypeArg, + useLegacyIphone5sArg, + useLegacyIphone6sArg, + useLegacyIphone7Arg, + useLegacyIphonexArg, + useLegacyIphonexrArg, + useLegacyIphonexsArg, + useLegacyIphonexsmaxArg, + forceOrientationBlockArg, + debugModeArg, + resumeArg, + usePlatformArg, + pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "frameit", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs test coverage reports for your Xcode project + + Generate summarized code coverage reports using [gcovr](http://gcovr.com/) + */ +public func gcovr() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "gcovr", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Get the build number of your project + + - parameters: + - xcodeproj: optional, you must specify the path to your main Xcode project if it is not in the project root directory + - hideErrorWhenVersioningDisabled: Used during `fastlane init` to hide the error message + + This action will return the current build number set on your project. + You first have to set up your Xcode project, if you haven't done it already: [https://developer.apple.com/library/ios/qa/qa1827/_index.html](https://developer.apple.com/library/ios/qa/qa1827/_index.html). + */ +@discardableResult public func getBuildNumber(xcodeproj: OptionalConfigValue = .fastlaneDefault(nil), + hideErrorWhenVersioningDisabled: OptionalConfigValue = .fastlaneDefault(false)) -> String +{ + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let hideErrorWhenVersioningDisabledArg = hideErrorWhenVersioningDisabled.asRubyArgument(name: "hide_error_when_versioning_disabled", type: nil) + let array: [RubyCommand.Argument?] = [xcodeprojArg, + hideErrorWhenVersioningDisabledArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_build_number", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Get the build number from the current repository + + - parameter useHgRevisionNumber: Use hg revision number instead of hash (ignored for non-hg repos) + + - returns: The build number from the current repository + + This action will get the **build number** according to what the SCM HEAD reports. + Currently supported SCMs are svn (uses root revision), git-svn (uses svn revision), git (uses short hash) and mercurial (uses short hash or revision number). + There is an option, `:use_hg_revision_number`, which allows to use mercurial revision number instead of hash. + */ +public func getBuildNumberRepository(useHgRevisionNumber: OptionalConfigValue = .fastlaneDefault(false)) { + let useHgRevisionNumberArg = useHgRevisionNumber.asRubyArgument(name: "use_hg_revision_number", type: nil) + let array: [RubyCommand.Argument?] = [useHgRevisionNumberArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_build_number_repository", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Create new iOS code signing certificates (via _cert_) + + - parameters: + - development: Create a development certificate instead of a distribution one + - type: Create specific certificate type (takes precedence over :development) + - force: Create a certificate even if an existing certificate exists + - generateAppleCerts: Create a certificate type for Xcode 11 and later (Apple Development or Apple Distribution) + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - filename: The filename of certificate to store + - outputPath: The path to a directory in which all certificates and private keys should be stored + - keychainPath: Path to a custom keychain + - keychainPassword: This might be required the first time you access certificates on a new mac. For the login/default keychain this is your macOS account password + - skipSetPartitionList: Skips setting the partition list (which can sometimes take a long time). Setting the partition list is usually needed to prevent Xcode from prompting to allow a cert to be used for signing + - platform: Set the provisioning profile's platform (ios, macos, tvos) + + **Important**: It is recommended to use [match](https://docs.fastlane.tools/actions/match/) according to the [codesigning.guide](https://codesigning.guide) for generating and maintaining your certificates. Use _cert_ directly only if you want full control over what's going on and know more about codesigning. + Use this action to download the latest code signing identity. + */ +public func getCertificates(development: OptionalConfigValue = .fastlaneDefault(false), + type: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + generateAppleCerts: OptionalConfigValue = .fastlaneDefault(true), + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + filename: OptionalConfigValue = .fastlaneDefault(nil), + outputPath: String = ".", + keychainPath: String, + keychainPassword: OptionalConfigValue = .fastlaneDefault(nil), + skipSetPartitionList: OptionalConfigValue = .fastlaneDefault(false), + platform: String = "ios") +{ + let developmentArg = development.asRubyArgument(name: "development", type: nil) + let typeArg = type.asRubyArgument(name: "type", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let generateAppleCertsArg = generateAppleCerts.asRubyArgument(name: "generate_apple_certs", type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let filenameArg = filename.asRubyArgument(name: "filename", type: nil) + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let keychainPathArg = RubyCommand.Argument(name: "keychain_path", value: keychainPath, type: nil) + let keychainPasswordArg = keychainPassword.asRubyArgument(name: "keychain_password", type: nil) + let skipSetPartitionListArg = skipSetPartitionList.asRubyArgument(name: "skip_set_partition_list", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let array: [RubyCommand.Argument?] = [developmentArg, + typeArg, + forceArg, + generateAppleCertsArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + filenameArg, + outputPathArg, + keychainPathArg, + keychainPasswordArg, + skipSetPartitionListArg, + platformArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_certificates", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will verify if a given release version is available on GitHub + + - parameters: + - url: The path to your repo, e.g. 'KrauseFx/fastlane' + - serverUrl: The server url. e.g. 'https://your.github.server/api/v3' (Default: 'https://api.github.com') + - version: The version tag of the release to check + - apiToken: GitHub Personal Token (required for private repositories) + - apiBearer: Use a Bearer authorization token. Usually generated by Github Apps, e.g. GitHub Actions GITHUB_TOKEN environment variable + + This will return all information about a release. For example:| + | + ```no-highlight| + {| + "url"=>"https://api.github.com/repos/KrauseFx/fastlane/releases/1537713",| + "assets_url"=>"https://api.github.com/repos/KrauseFx/fastlane/releases/1537713/assets",| + "upload_url"=>"https://uploads.github.com/repos/KrauseFx/fastlane/releases/1537713/assets{?name}",| + "html_url"=>"https://github.com/fastlane/fastlane/releases/tag/1.8.0",| + "id"=>1537713,| + "tag_name"=>"1.8.0",| + "target_commitish"=>"master",| + "name"=>"1.8.0 Switch Lanes & Pass Parameters",| + "draft"=>false,| + "author"=>| + {"login"=>"KrauseFx",| + "id"=>869950,| + "avatar_url"=>"https://avatars.githubusercontent.com/u/869950?v=3",| + "gravatar_id"=>"",| + "url"=>"https://api.github.com/users/KrauseFx",| + "html_url"=>"https://github.com/fastlane",| + "followers_url"=>"https://api.github.com/users/KrauseFx/followers",| + "following_url"=>"https://api.github.com/users/KrauseFx/following{/other_user}",| + "gists_url"=>"https://api.github.com/users/KrauseFx/gists{/gist_id}",| + "starred_url"=>"https://api.github.com/users/KrauseFx/starred{/owner}{/repo}",| + "subscriptions_url"=>"https://api.github.com/users/KrauseFx/subscriptions",| + "organizations_url"=>"https://api.github.com/users/KrauseFx/orgs",| + "repos_url"=>"https://api.github.com/users/KrauseFx/repos",| + "events_url"=>"https://api.github.com/users/KrauseFx/events{/privacy}",| + "received_events_url"=>"https://api.github.com/users/KrauseFx/received_events",| + "type"=>"User",| + "site_admin"=>false},| + "prerelease"=>false,| + "created_at"=>"2015-07-14T23:33:01Z",| + "published_at"=>"2015-07-14T23:44:10Z",| + "assets"=>[],| + "tarball_url"=>"https://api.github.com/repos/KrauseFx/fastlane/tarball/1.8.0",| + "zipball_url"=>"https://api.github.com/repos/KrauseFx/fastlane/zipball/1.8.0",| + "body"=> ...Markdown...| + "This is one of the biggest updates of _fastlane_ yet"| + }| + ```| + >| + */ +public func getGithubRelease(url: String, + serverUrl: String = "https://api.github.com", + version: String, + apiToken: OptionalConfigValue = .fastlaneDefault(nil), + apiBearer: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let urlArg = RubyCommand.Argument(name: "url", value: url, type: nil) + let serverUrlArg = RubyCommand.Argument(name: "server_url", value: serverUrl, type: nil) + let versionArg = RubyCommand.Argument(name: "version", value: version, type: nil) + let apiTokenArg = apiToken.asRubyArgument(name: "api_token", type: nil) + let apiBearerArg = apiBearer.asRubyArgument(name: "api_bearer", type: nil) + let array: [RubyCommand.Argument?] = [urlArg, + serverUrlArg, + versionArg, + apiTokenArg, + apiBearerArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_github_release", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Returns value from Info.plist of your project as native Ruby data structures + + - parameters: + - key: Name of parameter + - path: Path to plist file you want to read + + Get a value from a plist file, which can be used to fetch the app identifier and more information about your app + */ +@discardableResult public func getInfoPlistValue(key: String, + path: String) -> String +{ + let keyArg = RubyCommand.Argument(name: "key", value: key, type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [keyArg, + pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_info_plist_value", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Returns a value from Info.plist inside a .ipa file + + - parameters: + - key: Name of parameter + - ipa: Path to IPA + + - returns: Returns the value in the .ipa's Info.plist corresponding to the passed in Key + + This is useful for introspecting Info.plist files for `.ipa` files that have already been built. + */ +@discardableResult public func getIpaInfoPlistValue(key: String, + ipa: String) -> String +{ + let keyArg = RubyCommand.Argument(name: "key", value: key, type: nil) + let ipaArg = RubyCommand.Argument(name: "ipa", value: ipa, type: nil) + let array: [RubyCommand.Argument?] = [keyArg, + ipaArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_ipa_info_plist_value", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Obtain publishing rights for custom apps on Managed Google Play Store + + - parameters: + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + + - returns: An URI to obtain publishing rights for custom apps on Managed Play Store + + If you haven't done so before, start by following the first two steps of Googles ["Get started with custom app publishing"](https://developers.google.com/android/work/play/custom-app-api/get-started) -> ["Preliminary setup"](https://developers.google.com/android/work/play/custom-app-api/get-started#preliminary_setup) instructions: + "[Enable the Google Play Custom App Publishing API](https://developers.google.com/android/work/play/custom-app-api/get-started#enable_the_google_play_custom_app_publishing_api)" and "[Create a service account](https://developers.google.com/android/work/play/custom-app-api/get-started#create_a_service_account)". + You need the "service account's private key file" to continue. + Run the action and supply the "private key file" to it as the `json_key` parameter. The command will output a URL to visit. After logging in you are redirected to a page that outputs your "Developer Account ID" - take note of that, you will need it to be able to use [`create_app_on_managed_play_store`](https://docs.fastlane.tools/actions/create_app_on_managed_play_store/). + */ +public func getManagedPlayStorePublishingRights(jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let array: [RubyCommand.Argument?] = [jsonKeyArg, + jsonKeyDataArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_managed_play_store_publishing_rights", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generates a provisioning profile, saving it in the current folder (via _sigh_) + + - parameters: + - adhoc: Setting this flag will generate AdHoc profiles instead of App Store Profiles + - developerId: Setting this flag will generate Developer ID profiles instead of App Store Profiles + - development: Renew the development certificate instead of the production one + - skipInstall: By default, the certificate will be added to your local machine. Setting this flag will skip this action + - force: Renew provisioning profiles regardless of its state - to automatically add all devices for ad hoc profiles + - includeMacInProfiles: Include Apple Silicon Mac devices in provisioning profiles for iOS/iPadOS apps + - appIdentifier: The bundle identifier of your app + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - provisioningName: The name of the profile that is used on the Apple Developer Portal + - ignoreProfilesWithDifferentName: Use in combination with :provisioning_name - when true only profiles matching this exact name will be downloaded + - outputPath: Directory in which the profile should be stored + - certId: The ID of the code signing certificate to use (e.g. 78ADL6LVAA) + - certOwnerName: The certificate name to use for new profiles, or to renew with. (e.g. "Felix Krause") + - filename: Filename to use for the generated provisioning profile (must include .mobileprovision) + - skipFetchProfiles: Skips the verification of existing profiles which is useful if you have thousands of profiles + - includeAllCertificates: Include all matching certificates in the provisioning profile. Works only for the 'development' provisioning profile type + - skipCertificateVerification: Skips the verification of the certificates for every existing profiles. This will make sure the provisioning profile can be used on the local machine + - platform: Set the provisioning profile's platform (i.e. ios, tvos, macos, catalyst) + - readonly: Only fetch existing profile, don't generate new ones + - templateName: The name of provisioning profile template. If the developer account has provisioning profile templates (aka: custom entitlements), the template name can be found by inspecting the Entitlements drop-down while creating/editing a provisioning profile (e.g. "Apple Pay Pass Suppression Development") + - failOnNameTaken: Should the command fail if it was about to create a duplicate of an existing provisioning profile. It can happen due to issues on Apple Developer Portal, when profile to be recreated was not properly deleted first + + - returns: The UUID of the profile sigh just fetched/generated + + **Note**: It is recommended to use [match](https://docs.fastlane.tools/actions/match/) according to the [codesigning.guide](https://codesigning.guide) for generating and maintaining your provisioning profiles. Use _sigh_ directly only if you want full control over what's going on and know more about codesigning. + */ +@discardableResult public func getProvisioningProfile(adhoc: OptionalConfigValue = .fastlaneDefault(false), + developerId: OptionalConfigValue = .fastlaneDefault(false), + development: OptionalConfigValue = .fastlaneDefault(false), + skipInstall: OptionalConfigValue = .fastlaneDefault(false), + force: OptionalConfigValue = .fastlaneDefault(false), + includeMacInProfiles: OptionalConfigValue = .fastlaneDefault(false), + appIdentifier: String, + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + provisioningName: OptionalConfigValue = .fastlaneDefault(nil), + ignoreProfilesWithDifferentName: OptionalConfigValue = .fastlaneDefault(false), + outputPath: String = ".", + certId: OptionalConfigValue = .fastlaneDefault(nil), + certOwnerName: OptionalConfigValue = .fastlaneDefault(nil), + filename: OptionalConfigValue = .fastlaneDefault(nil), + skipFetchProfiles: OptionalConfigValue = .fastlaneDefault(false), + includeAllCertificates: OptionalConfigValue = .fastlaneDefault(false), + skipCertificateVerification: OptionalConfigValue = .fastlaneDefault(false), + platform: Any = "ios", + readonly: OptionalConfigValue = .fastlaneDefault(false), + templateName: OptionalConfigValue = .fastlaneDefault(nil), + failOnNameTaken: OptionalConfigValue = .fastlaneDefault(false)) -> String +{ + let adhocArg = adhoc.asRubyArgument(name: "adhoc", type: nil) + let developerIdArg = developerId.asRubyArgument(name: "developer_id", type: nil) + let developmentArg = development.asRubyArgument(name: "development", type: nil) + let skipInstallArg = skipInstall.asRubyArgument(name: "skip_install", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let includeMacInProfilesArg = includeMacInProfiles.asRubyArgument(name: "include_mac_in_profiles", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let provisioningNameArg = provisioningName.asRubyArgument(name: "provisioning_name", type: nil) + let ignoreProfilesWithDifferentNameArg = ignoreProfilesWithDifferentName.asRubyArgument(name: "ignore_profiles_with_different_name", type: nil) + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let certIdArg = certId.asRubyArgument(name: "cert_id", type: nil) + let certOwnerNameArg = certOwnerName.asRubyArgument(name: "cert_owner_name", type: nil) + let filenameArg = filename.asRubyArgument(name: "filename", type: nil) + let skipFetchProfilesArg = skipFetchProfiles.asRubyArgument(name: "skip_fetch_profiles", type: nil) + let includeAllCertificatesArg = includeAllCertificates.asRubyArgument(name: "include_all_certificates", type: nil) + let skipCertificateVerificationArg = skipCertificateVerification.asRubyArgument(name: "skip_certificate_verification", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let readonlyArg = readonly.asRubyArgument(name: "readonly", type: nil) + let templateNameArg = templateName.asRubyArgument(name: "template_name", type: nil) + let failOnNameTakenArg = failOnNameTaken.asRubyArgument(name: "fail_on_name_taken", type: nil) + let array: [RubyCommand.Argument?] = [adhocArg, + developerIdArg, + developmentArg, + skipInstallArg, + forceArg, + includeMacInProfilesArg, + appIdentifierArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + provisioningNameArg, + ignoreProfilesWithDifferentNameArg, + outputPathArg, + certIdArg, + certOwnerNameArg, + filenameArg, + skipFetchProfilesArg, + includeAllCertificatesArg, + skipCertificateVerificationArg, + platformArg, + readonlyArg, + templateNameArg, + failOnNameTakenArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_provisioning_profile", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Ensure a valid push profile is active, creating a new one if needed (via _pem_) + + - parameters: + - platform: Set certificate's platform. Used for creation of production & development certificates. Supported platforms: ios, macos + - development: Renew the development push certificate instead of the production one + - websitePush: Create a Website Push certificate + - generateP12: Generate a p12 file additionally to a PEM file + - activeDaysLimit: If the current certificate is active for less than this number of days, generate a new one + - force: Create a new push certificate, even if the current one is active for 30 (or PEM_ACTIVE_DAYS_LIMIT) more days + - savePrivateKey: Set to save the private RSA key + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - p12Password: The password that is used for your p12 file + - pemName: The file name of the generated .pem file + - outputPath: The path to a directory in which all certificates and private keys should be stored + - newProfile: Block that is called if there is a new profile + + Additionally to the available options, you can also specify a block that only gets executed if a new profile was created. You can use it to upload the new profile to your server. + Use it like this:| + | + ```ruby| + get_push_certificate(| + new_profile: proc do| + # your upload code| + end| + )| + ```| + >| + */ +public func getPushCertificate(platform: String = "ios", + development: OptionalConfigValue = .fastlaneDefault(false), + websitePush: OptionalConfigValue = .fastlaneDefault(false), + generateP12: OptionalConfigValue = .fastlaneDefault(true), + activeDaysLimit: Int = 30, + force: OptionalConfigValue = .fastlaneDefault(false), + savePrivateKey: OptionalConfigValue = .fastlaneDefault(true), + appIdentifier: String, + username: String, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + p12Password: OptionalConfigValue = .fastlaneDefault(nil), + pemName: OptionalConfigValue = .fastlaneDefault(nil), + outputPath: String = ".", + newProfile: ((String) -> Void)? = nil) +{ + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let developmentArg = development.asRubyArgument(name: "development", type: nil) + let websitePushArg = websitePush.asRubyArgument(name: "website_push", type: nil) + let generateP12Arg = generateP12.asRubyArgument(name: "generate_p12", type: nil) + let activeDaysLimitArg = RubyCommand.Argument(name: "active_days_limit", value: activeDaysLimit, type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let savePrivateKeyArg = savePrivateKey.asRubyArgument(name: "save_private_key", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let p12PasswordArg = p12Password.asRubyArgument(name: "p12_password", type: nil) + let pemNameArg = pemName.asRubyArgument(name: "pem_name", type: nil) + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let newProfileArg = RubyCommand.Argument(name: "new_profile", value: newProfile, type: .stringClosure) + let array: [RubyCommand.Argument?] = [platformArg, + developmentArg, + websitePushArg, + generateP12Arg, + activeDaysLimitArg, + forceArg, + savePrivateKeyArg, + appIdentifierArg, + usernameArg, + teamIdArg, + teamNameArg, + p12PasswordArg, + pemNameArg, + outputPathArg, + newProfileArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_push_certificate", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Get the version number of your project + + - parameters: + - xcodeproj: Path to the Xcode project to read version number from, or its containing directory, optional. If omitted, or if a directory is passed instead, it will use the first Xcode project found within the given directory, or the project root directory if none is passed + - target: Target name, optional. Will be needed if you have more than one non-test target to avoid being prompted to select one + - configuration: Configuration name, optional. Will be needed if you have altered the configurations from the default or your version number depends on the configuration selected + + This action will return the current version number set on your project. It first looks in the plist and then for '$(MARKETING_VERSION)' in the build settings. + */ +@discardableResult public func getVersionNumber(xcodeproj: OptionalConfigValue = .fastlaneDefault(nil), + target: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let targetArg = target.asRubyArgument(name: "target", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let array: [RubyCommand.Argument?] = [xcodeprojArg, + targetArg, + configurationArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "get_version_number", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Directly add the given file or all files + + - parameters: + - path: The file(s) and path(s) you want to add + - shellEscape: Shell escapes paths (set to false if using wildcards or manually escaping spaces in :path) + - pathspec: **DEPRECATED!** Use `--path` instead - The pathspec you want to add files from + */ +public func gitAdd(path: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + shellEscape: OptionalConfigValue = .fastlaneDefault(true), + pathspec: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let pathArg = path.asRubyArgument(name: "path", type: nil) + let shellEscapeArg = shellEscape.asRubyArgument(name: "shell_escape", type: nil) + let pathspecArg = pathspec.asRubyArgument(name: "pathspec", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + shellEscapeArg, + pathspecArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "git_add", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Returns the name of the current git branch, possibly as managed by CI ENV vars + + If no branch could be found, this action will return an empty string. This is a wrapper for the internal action Actions.git_branch + */ +@discardableResult public func gitBranch() -> String { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "git_branch", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Directly commit the given file with the given message + + - parameters: + - path: The file(s) or directory(ies) you want to commit. You can pass an array of multiple file-paths or fileglobs "*.txt" to commit all matching files. The files already staged but not specified and untracked files won't be committed + - message: The commit message that should be used + - skipGitHooks: Set to true to pass `--no-verify` to git + - allowNothingToCommit: Set to true to allow commit without any git changes in the files you want to commit + */ +public func gitCommit(path: [String], + message: String, + skipGitHooks: OptionalConfigValue = .fastlaneDefault(false), + allowNothingToCommit: OptionalConfigValue = .fastlaneDefault(false)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let skipGitHooksArg = skipGitHooks.asRubyArgument(name: "skip_git_hooks", type: nil) + let allowNothingToCommitArg = allowNothingToCommit.asRubyArgument(name: "allow_nothing_to_commit", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + messageArg, + skipGitHooksArg, + allowNothingToCommitArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "git_commit", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Executes a simple git pull command + + - parameters: + - onlyTags: Simply pull the tags, and not bring new commits to the current branch from the remote + - rebase: Rebase on top of the remote branch instead of merge + */ +public func gitPull(onlyTags: OptionalConfigValue = .fastlaneDefault(false), + rebase: OptionalConfigValue = .fastlaneDefault(false)) +{ + let onlyTagsArg = onlyTags.asRubyArgument(name: "only_tags", type: nil) + let rebaseArg = rebase.asRubyArgument(name: "rebase", type: nil) + let array: [RubyCommand.Argument?] = [onlyTagsArg, + rebaseArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "git_pull", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Returns the name of the current git remote default branch + + - parameter remoteName: The remote repository to check + + If no default remote branch could be found, this action will return nil. This is a wrapper for the internal action Actions.git_default_remote_branch_name + */ +@discardableResult public func gitRemoteBranch(remoteName: OptionalConfigValue = .fastlaneDefault(nil)) -> String { + let remoteNameArg = remoteName.asRubyArgument(name: "remote_name", type: nil) + let array: [RubyCommand.Argument?] = [remoteNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "git_remote_branch", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Executes a git submodule update command + + - parameters: + - recursive: Should the submodules be updated recursively? + - init: Should the submodules be initiated before update? + */ +public func gitSubmoduleUpdate(recursive: OptionalConfigValue = .fastlaneDefault(false), + init: OptionalConfigValue = .fastlaneDefault(false)) +{ + let recursiveArg = recursive.asRubyArgument(name: "recursive", type: nil) + let initArg = `init`.asRubyArgument(name: "init", type: nil) + let array: [RubyCommand.Argument?] = [recursiveArg, + initArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "git_submodule_update", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Checks if the git tag with the given name exists in the current repo + + - parameters: + - tag: The tag name that should be checked + - remote: Whether to check remote. Defaults to `false` + - remoteName: The remote to check. Defaults to `origin` + + - returns: Boolean value whether the tag exists or not + */ +@discardableResult public func gitTagExists(tag: String, + remote: OptionalConfigValue = .fastlaneDefault(false), + remoteName: String = "origin") -> Bool +{ + let tagArg = RubyCommand.Argument(name: "tag", value: tag, type: nil) + let remoteArg = remote.asRubyArgument(name: "remote", type: nil) + let remoteNameArg = RubyCommand.Argument(name: "remote_name", value: remoteName, type: nil) + let array: [RubyCommand.Argument?] = [tagArg, + remoteArg, + remoteNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "git_tag_exists", className: nil, args: args) + return parseBool(fromString: runner.executeCommand(command)) +} + +/** + Call a GitHub API endpoint and get the resulting JSON response + + - parameters: + - serverUrl: The server url. e.g. 'https://your.internal.github.host/api/v3' (Default: 'https://api.github.com') + - apiToken: Personal API Token for GitHub - generate one at https://github.com/settings/tokens + - apiBearer: Use a Bearer authorization token. Usually generated by Github Apps, e.g. GitHub Actions GITHUB_TOKEN environment variable + - httpMethod: The HTTP method. e.g. GET / POST + - body: The request body in JSON or hash format + - rawBody: The request body taken verbatim instead of as JSON, useful for file uploads + - path: The endpoint path. e.g. '/repos/:owner/:repo/readme' + - url: The complete full url - used instead of path. e.g. 'https://uploads.github.com/repos/fastlane...' + - errorHandlers: Optional error handling hash based on status code, or pass '*' to handle all errors + - headers: Optional headers to apply + - secure: Optionally disable secure requests (ssl_verify_peer) + + - returns: A hash including the HTTP status code (:status), the response body (:body), and if valid JSON has been returned the parsed JSON (:json). + + Calls any GitHub API endpoint. You must provide your GitHub Personal token (get one from [https://github.com/settings/tokens/new](https://github.com/settings/tokens/new)). + Out parameters provide the status code and the full response JSON if valid, otherwise the raw response body. + Documentation: [https://developer.github.com/v3](https://developer.github.com/v3). + */ +public func githubApi(serverUrl: String = "https://api.github.com", + apiToken: OptionalConfigValue = .fastlaneDefault(nil), + apiBearer: OptionalConfigValue = .fastlaneDefault(nil), + httpMethod: String = "GET", + body: [String: Any] = [:], + rawBody: OptionalConfigValue = .fastlaneDefault(nil), + path: OptionalConfigValue = .fastlaneDefault(nil), + url: OptionalConfigValue = .fastlaneDefault(nil), + errorHandlers: [String: Any] = [:], + headers: [String: Any] = [:], + secure: OptionalConfigValue = .fastlaneDefault(true)) +{ + let serverUrlArg = RubyCommand.Argument(name: "server_url", value: serverUrl, type: nil) + let apiTokenArg = apiToken.asRubyArgument(name: "api_token", type: nil) + let apiBearerArg = apiBearer.asRubyArgument(name: "api_bearer", type: nil) + let httpMethodArg = RubyCommand.Argument(name: "http_method", value: httpMethod, type: nil) + let bodyArg = RubyCommand.Argument(name: "body", value: body, type: nil) + let rawBodyArg = rawBody.asRubyArgument(name: "raw_body", type: nil) + let pathArg = path.asRubyArgument(name: "path", type: nil) + let urlArg = url.asRubyArgument(name: "url", type: nil) + let errorHandlersArg = RubyCommand.Argument(name: "error_handlers", value: errorHandlers, type: nil) + let headersArg = RubyCommand.Argument(name: "headers", value: headers, type: nil) + let secureArg = secure.asRubyArgument(name: "secure", type: nil) + let array: [RubyCommand.Argument?] = [serverUrlArg, + apiTokenArg, + apiBearerArg, + httpMethodArg, + bodyArg, + rawBodyArg, + pathArg, + urlArg, + errorHandlersArg, + headersArg, + secureArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "github_api", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Retrieves release names for a Google Play track + + - parameters: + - packageName: The package name of the application to use + - track: The track of the application to use. The default available tracks are: production, beta, alpha, internal + - key: **DEPRECATED!** Use `--json_key` instead - The p12 File used to authenticate with Google + - issuer: **DEPRECATED!** Use `--json_key` instead - The issuer of the p12 file (email address of the service account) + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - timeout: Timeout for read, open, and send (in seconds) + + - returns: Array of strings representing the release names for the given Google Play track + + More information: [https://docs.fastlane.tools/actions/supply/](https://docs.fastlane.tools/actions/supply/) + */ +public func googlePlayTrackReleaseNames(packageName: String, + track: String = "production", + key: OptionalConfigValue = .fastlaneDefault(nil), + issuer: OptionalConfigValue = .fastlaneDefault(nil), + jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 300) +{ + let packageNameArg = RubyCommand.Argument(name: "package_name", value: packageName, type: nil) + let trackArg = RubyCommand.Argument(name: "track", value: track, type: nil) + let keyArg = key.asRubyArgument(name: "key", type: nil) + let issuerArg = issuer.asRubyArgument(name: "issuer", type: nil) + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [packageNameArg, + trackArg, + keyArg, + issuerArg, + jsonKeyArg, + jsonKeyDataArg, + rootUrlArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "google_play_track_release_names", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Retrieves version codes for a Google Play track + + - parameters: + - packageName: The package name of the application to use + - track: The track of the application to use. The default available tracks are: production, beta, alpha, internal + - key: **DEPRECATED!** Use `--json_key` instead - The p12 File used to authenticate with Google + - issuer: **DEPRECATED!** Use `--json_key` instead - The issuer of the p12 file (email address of the service account) + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - timeout: Timeout for read, open, and send (in seconds) + + - returns: Array of integers representing the version codes for the given Google Play track + + More information: [https://docs.fastlane.tools/actions/supply/](https://docs.fastlane.tools/actions/supply/) + */ +public func googlePlayTrackVersionCodes(packageName: String, + track: String = "production", + key: OptionalConfigValue = .fastlaneDefault(nil), + issuer: OptionalConfigValue = .fastlaneDefault(nil), + jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 300) +{ + let packageNameArg = RubyCommand.Argument(name: "package_name", value: packageName, type: nil) + let trackArg = RubyCommand.Argument(name: "track", value: track, type: nil) + let keyArg = key.asRubyArgument(name: "key", type: nil) + let issuerArg = issuer.asRubyArgument(name: "issuer", type: nil) + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [packageNameArg, + trackArg, + keyArg, + issuerArg, + jsonKeyArg, + jsonKeyDataArg, + rootUrlArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "google_play_track_version_codes", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + All gradle related actions, including building and testing your Android app + + - parameters: + - task: The gradle task you want to execute, e.g. `assemble`, `bundle` or `test`. For tasks such as `assembleMyFlavorRelease` you should use gradle(task: 'assemble', flavor: 'Myflavor', build_type: 'Release') + - flavor: The flavor that you want the task for, e.g. `MyFlavor`. If you are running the `assemble` task in a multi-flavor project, and you rely on Actions.lane_context[SharedValues::GRADLE_APK_OUTPUT_PATH] then you must specify a flavor here or else this value will be undefined + - buildType: The build type that you want the task for, e.g. `Release`. Useful for some tasks such as `assemble` + - tasks: The multiple gradle tasks that you want to execute, e.g. `[assembleDebug, bundleDebug]` + - flags: All parameter flags you want to pass to the gradle command, e.g. `--exitcode --xml file.xml` + - projectDir: The root directory of the gradle project + - gradlePath: The path to your `gradlew`. If you specify a relative path, it is assumed to be relative to the `project_dir` + - properties: Gradle properties to be exposed to the gradle script + - systemProperties: Gradle system properties to be exposed to the gradle script + - serial: Android serial, which device should be used for this command + - printCommand: Control whether the generated Gradle command is printed as output before running it (true/false) + - printCommandOutput: Control whether the output produced by given Gradle command is printed while running (true/false) + + - returns: The output of running the gradle task + + Run `./gradlew tasks` to get a list of all available gradle tasks for your project + */ +public func gradle(task: OptionalConfigValue = .fastlaneDefault(nil), + flavor: OptionalConfigValue = .fastlaneDefault(nil), + buildType: OptionalConfigValue = .fastlaneDefault(nil), + tasks: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + flags: OptionalConfigValue = .fastlaneDefault(nil), + projectDir: String = ".", + gradlePath: OptionalConfigValue = .fastlaneDefault(nil), + properties: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + systemProperties: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + serial: String = "", + printCommand: OptionalConfigValue = .fastlaneDefault(true), + printCommandOutput: OptionalConfigValue = .fastlaneDefault(true)) +{ + let taskArg = task.asRubyArgument(name: "task", type: nil) + let flavorArg = flavor.asRubyArgument(name: "flavor", type: nil) + let buildTypeArg = buildType.asRubyArgument(name: "build_type", type: nil) + let tasksArg = tasks.asRubyArgument(name: "tasks", type: nil) + let flagsArg = flags.asRubyArgument(name: "flags", type: nil) + let projectDirArg = RubyCommand.Argument(name: "project_dir", value: projectDir, type: nil) + let gradlePathArg = gradlePath.asRubyArgument(name: "gradle_path", type: nil) + let propertiesArg = properties.asRubyArgument(name: "properties", type: nil) + let systemPropertiesArg = systemProperties.asRubyArgument(name: "system_properties", type: nil) + let serialArg = RubyCommand.Argument(name: "serial", value: serial, type: nil) + let printCommandArg = printCommand.asRubyArgument(name: "print_command", type: nil) + let printCommandOutputArg = printCommandOutput.asRubyArgument(name: "print_command_output", type: nil) + let array: [RubyCommand.Argument?] = [taskArg, + flavorArg, + buildTypeArg, + tasksArg, + flagsArg, + projectDirArg, + gradlePathArg, + propertiesArg, + systemPropertiesArg, + serialArg, + printCommandArg, + printCommandOutputArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "gradle", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `build_app` action + + - parameters: + - workspace: Path to the workspace file + - project: Path to the project file + - scheme: The project's scheme. Make sure it's marked as `Shared` + - clean: Should the project be cleaned before building it? + - outputDirectory: The directory in which the ipa file should be stored in + - outputName: The name of the resulting ipa file + - configuration: The configuration to use when building the app. Defaults to 'Release' + - silent: Hide all information that's not necessary while building + - codesigningIdentity: The name of the code signing identity to use. It has to match the name exactly. e.g. 'iPhone Distribution: SunApps GmbH' + - skipPackageIpa: Should we skip packaging the ipa? + - skipPackagePkg: Should we skip packaging the pkg? + - includeSymbols: Should the ipa file include symbols? + - includeBitcode: Should the ipa file include bitcode? + - exportMethod: Method used to export the archive. Valid values are: app-store, validation, ad-hoc, package, enterprise, development, developer-id and mac-application + - exportOptions: Path to an export options plist or a hash with export options. Use 'xcodebuild -help' to print the full set of available options + - exportXcargs: Pass additional arguments to xcodebuild for the package phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - skipBuildArchive: Export ipa from previously built xcarchive. Uses archive_path as source + - skipArchive: After building, don't archive, effectively not including -archivePath param + - skipCodesigning: Build without codesigning + - catalystPlatform: Platform to build when using a Catalyst enabled app. Valid values are: ios, macos + - installerCertName: Full name of 3rd Party Mac Developer Installer or Developer ID Installer certificate. Example: `3rd Party Mac Developer Installer: Your Company (ABC1234XWYZ)` + - buildPath: The directory in which the archive should be stored in + - archivePath: The path to the created archive + - derivedDataPath: The directory where built products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - resultBundlePath: Path to the result bundle directory to create. Ignored if `result_bundle` if false + - buildlogPath: The directory where to store the build log + - sdk: The SDK that should be used for building the application + - toolchain: The toolchain that should be used for building the application (e.g. com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a) + - destination: Use a custom destination for building the app + - exportTeamId: Optional: Sometimes you need to specify a team id when exporting the ipa file + - xcargs: Pass additional arguments to xcodebuild for the build phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - disableXcpretty: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Disable xcpretty formatting of build output + - xcprettyTestFormat: Use the test (RSpec style) format for build output + - xcprettyFormatter: A custom xcpretty formatter to use + - xcprettyReportJunit: Have xcpretty create a JUnit-style XML report at the provided path + - xcprettyReportHtml: Have xcpretty create a simple HTML report at the provided path + - xcprettyReportJson: Have xcpretty create a JSON compilation database at the provided path + - xcprettyUtf: Have xcpretty use unicode encoding when reporting builds + - analyzeBuildTime: Analyze the project build time and store the output in 'culprits.txt' file + - skipProfileDetection: Do not try to build a profile mapping from the xcodeproj. Match or a manually provided mapping should be used + - xcodebuildCommand: Allows for override of the default `xcodebuild` command + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - useSystemScm: Lets xcodebuild use system's scm configuration + + - returns: The absolute path to the generated ipa file + + More information: https://fastlane.tools/gym + */ +@discardableResult public func gym(workspace: OptionalConfigValue = .fastlaneDefault(gymfile.workspace), + project: OptionalConfigValue = .fastlaneDefault(gymfile.project), + scheme: OptionalConfigValue = .fastlaneDefault(gymfile.scheme), + clean: OptionalConfigValue = .fastlaneDefault(gymfile.clean), + outputDirectory: String = gymfile.outputDirectory, + outputName: OptionalConfigValue = .fastlaneDefault(gymfile.outputName), + configuration: OptionalConfigValue = .fastlaneDefault(gymfile.configuration), + silent: OptionalConfigValue = .fastlaneDefault(gymfile.silent), + codesigningIdentity: OptionalConfigValue = .fastlaneDefault(gymfile.codesigningIdentity), + skipPackageIpa: OptionalConfigValue = .fastlaneDefault(gymfile.skipPackageIpa), + skipPackagePkg: OptionalConfigValue = .fastlaneDefault(gymfile.skipPackagePkg), + includeSymbols: OptionalConfigValue = .fastlaneDefault(gymfile.includeSymbols), + includeBitcode: OptionalConfigValue = .fastlaneDefault(gymfile.includeBitcode), + exportMethod: OptionalConfigValue = .fastlaneDefault(gymfile.exportMethod), + exportOptions: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(gymfile.exportOptions), + exportXcargs: OptionalConfigValue = .fastlaneDefault(gymfile.exportXcargs), + skipBuildArchive: OptionalConfigValue = .fastlaneDefault(gymfile.skipBuildArchive), + skipArchive: OptionalConfigValue = .fastlaneDefault(gymfile.skipArchive), + skipCodesigning: OptionalConfigValue = .fastlaneDefault(gymfile.skipCodesigning), + catalystPlatform: OptionalConfigValue = .fastlaneDefault(gymfile.catalystPlatform), + installerCertName: OptionalConfigValue = .fastlaneDefault(gymfile.installerCertName), + buildPath: OptionalConfigValue = .fastlaneDefault(gymfile.buildPath), + archivePath: OptionalConfigValue = .fastlaneDefault(gymfile.archivePath), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(gymfile.derivedDataPath), + resultBundle: OptionalConfigValue = .fastlaneDefault(gymfile.resultBundle), + resultBundlePath: OptionalConfigValue = .fastlaneDefault(gymfile.resultBundlePath), + buildlogPath: String = gymfile.buildlogPath, + sdk: OptionalConfigValue = .fastlaneDefault(gymfile.sdk), + toolchain: OptionalConfigValue = .fastlaneDefault(gymfile.toolchain), + destination: OptionalConfigValue = .fastlaneDefault(gymfile.destination), + exportTeamId: OptionalConfigValue = .fastlaneDefault(gymfile.exportTeamId), + xcargs: OptionalConfigValue = .fastlaneDefault(gymfile.xcargs), + xcconfig: OptionalConfigValue = .fastlaneDefault(gymfile.xcconfig), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(gymfile.suppressXcodeOutput), + xcodebuildFormatter: String = gymfile.xcodebuildFormatter, + disableXcpretty: OptionalConfigValue = .fastlaneDefault(gymfile.disableXcpretty), + xcprettyTestFormat: OptionalConfigValue = .fastlaneDefault(gymfile.xcprettyTestFormat), + xcprettyFormatter: OptionalConfigValue = .fastlaneDefault(gymfile.xcprettyFormatter), + xcprettyReportJunit: OptionalConfigValue = .fastlaneDefault(gymfile.xcprettyReportJunit), + xcprettyReportHtml: OptionalConfigValue = .fastlaneDefault(gymfile.xcprettyReportHtml), + xcprettyReportJson: OptionalConfigValue = .fastlaneDefault(gymfile.xcprettyReportJson), + xcprettyUtf: OptionalConfigValue = .fastlaneDefault(gymfile.xcprettyUtf), + analyzeBuildTime: OptionalConfigValue = .fastlaneDefault(gymfile.analyzeBuildTime), + skipProfileDetection: OptionalConfigValue = .fastlaneDefault(gymfile.skipProfileDetection), + xcodebuildCommand: String = gymfile.xcodebuildCommand, + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(gymfile.clonedSourcePackagesPath), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(gymfile.skipPackageDependenciesResolution), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(gymfile.disablePackageAutomaticUpdates), + useSystemScm: OptionalConfigValue = .fastlaneDefault(gymfile.useSystemScm)) -> String +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputNameArg = outputName.asRubyArgument(name: "output_name", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let codesigningIdentityArg = codesigningIdentity.asRubyArgument(name: "codesigning_identity", type: nil) + let skipPackageIpaArg = skipPackageIpa.asRubyArgument(name: "skip_package_ipa", type: nil) + let skipPackagePkgArg = skipPackagePkg.asRubyArgument(name: "skip_package_pkg", type: nil) + let includeSymbolsArg = includeSymbols.asRubyArgument(name: "include_symbols", type: nil) + let includeBitcodeArg = includeBitcode.asRubyArgument(name: "include_bitcode", type: nil) + let exportMethodArg = exportMethod.asRubyArgument(name: "export_method", type: nil) + let exportOptionsArg = exportOptions.asRubyArgument(name: "export_options", type: nil) + let exportXcargsArg = exportXcargs.asRubyArgument(name: "export_xcargs", type: nil) + let skipBuildArchiveArg = skipBuildArchive.asRubyArgument(name: "skip_build_archive", type: nil) + let skipArchiveArg = skipArchive.asRubyArgument(name: "skip_archive", type: nil) + let skipCodesigningArg = skipCodesigning.asRubyArgument(name: "skip_codesigning", type: nil) + let catalystPlatformArg = catalystPlatform.asRubyArgument(name: "catalyst_platform", type: nil) + let installerCertNameArg = installerCertName.asRubyArgument(name: "installer_cert_name", type: nil) + let buildPathArg = buildPath.asRubyArgument(name: "build_path", type: nil) + let archivePathArg = archivePath.asRubyArgument(name: "archive_path", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let resultBundlePathArg = resultBundlePath.asRubyArgument(name: "result_bundle_path", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let toolchainArg = toolchain.asRubyArgument(name: "toolchain", type: nil) + let destinationArg = destination.asRubyArgument(name: "destination", type: nil) + let exportTeamIdArg = exportTeamId.asRubyArgument(name: "export_team_id", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let xcprettyTestFormatArg = xcprettyTestFormat.asRubyArgument(name: "xcpretty_test_format", type: nil) + let xcprettyFormatterArg = xcprettyFormatter.asRubyArgument(name: "xcpretty_formatter", type: nil) + let xcprettyReportJunitArg = xcprettyReportJunit.asRubyArgument(name: "xcpretty_report_junit", type: nil) + let xcprettyReportHtmlArg = xcprettyReportHtml.asRubyArgument(name: "xcpretty_report_html", type: nil) + let xcprettyReportJsonArg = xcprettyReportJson.asRubyArgument(name: "xcpretty_report_json", type: nil) + let xcprettyUtfArg = xcprettyUtf.asRubyArgument(name: "xcpretty_utf", type: nil) + let analyzeBuildTimeArg = analyzeBuildTime.asRubyArgument(name: "analyze_build_time", type: nil) + let skipProfileDetectionArg = skipProfileDetection.asRubyArgument(name: "skip_profile_detection", type: nil) + let xcodebuildCommandArg = RubyCommand.Argument(name: "xcodebuild_command", value: xcodebuildCommand, type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + schemeArg, + cleanArg, + outputDirectoryArg, + outputNameArg, + configurationArg, + silentArg, + codesigningIdentityArg, + skipPackageIpaArg, + skipPackagePkgArg, + includeSymbolsArg, + includeBitcodeArg, + exportMethodArg, + exportOptionsArg, + exportXcargsArg, + skipBuildArchiveArg, + skipArchiveArg, + skipCodesigningArg, + catalystPlatformArg, + installerCertNameArg, + buildPathArg, + archivePathArg, + derivedDataPathArg, + resultBundleArg, + resultBundlePathArg, + buildlogPathArg, + sdkArg, + toolchainArg, + destinationArg, + exportTeamIdArg, + xcargsArg, + xcconfigArg, + suppressXcodeOutputArg, + xcodebuildFormatterArg, + disableXcprettyArg, + xcprettyTestFormatArg, + xcprettyFormatterArg, + xcprettyReportJunitArg, + xcprettyReportHtmlArg, + xcprettyReportJsonArg, + xcprettyUtfArg, + analyzeBuildTimeArg, + skipProfileDetectionArg, + xcodebuildCommandArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "gym", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + This will add a hg tag to the current branch + + - parameter tag: Tag to create + */ +public func hgAddTag(tag: String) { + let tagArg = RubyCommand.Argument(name: "tag", value: tag, type: nil) + let array: [RubyCommand.Argument?] = [tagArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "hg_add_tag", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will commit a version bump to the hg repo + + - parameters: + - message: The commit message when committing the version bump + - xcodeproj: The path to your project file (Not the workspace). If you have only one, this is optional + - force: Forces the commit, even if other files than the ones containing the version number have been modified + - testDirtyFiles: A list of dirty files passed in for testing + - testExpectedFiles: A list of expected changed files passed in for testing + + The mercurial equivalent of the [commit_version_bump](https://docs.fastlane.tools/actions/commit_version_bump/) git action. Like the git version, it is useful in conjunction with [`increment_build_number`](https://docs.fastlane.tools/actions/increment_build_number/). + It checks the repo to make sure that only the relevant files have changed, these are the files that `increment_build_number` (`agvtool`) touches:| + | + >- All `.plist` files| + - The `.xcodeproj/project.pbxproj` file| + >| + Then commits those files to the repo. + Customize the message with the `:message` option, defaults to 'Version Bump' + If you have other uncommitted changes in your repo, this action will fail. If you started off in a clean repo, and used the _ipa_ and or _sigh_ actions, then you can use the [clean_build_artifacts](https://docs.fastlane.tools/actions/clean_build_artifacts/) action to clean those temporary files up before running this action. + */ +public func hgCommitVersionBump(message: String = "Version Bump", + xcodeproj: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + testDirtyFiles: String = "file1, file2", + testExpectedFiles: String = "file1, file2") +{ + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let testDirtyFilesArg = RubyCommand.Argument(name: "test_dirty_files", value: testDirtyFiles, type: nil) + let testExpectedFilesArg = RubyCommand.Argument(name: "test_expected_files", value: testExpectedFiles, type: nil) + let array: [RubyCommand.Argument?] = [messageArg, + xcodeprojArg, + forceArg, + testDirtyFilesArg, + testExpectedFilesArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "hg_commit_version_bump", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Raises an exception if there are uncommitted hg changes + + Along the same lines as the [ensure_git_status_clean](https://docs.fastlane.tools/actions/ensure_git_status_clean/) action, this is a sanity check to ensure the working mercurial repo is clean. Especially useful to put at the beginning of your Fastfile in the `before_all` block. + */ +public func hgEnsureCleanStatus() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "hg_ensure_clean_status", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will push changes to the remote hg repository + + - parameters: + - force: Force push to remote + - destination: The destination to push to + + The mercurial equivalent of [push_to_git_remote](https://docs.fastlane.tools/actions/push_to_git_remote/). Pushes your local commits to a remote mercurial repo. Useful when local changes such as adding a version bump commit or adding a tag are part of your lane’s actions. + */ +public func hgPush(force: OptionalConfigValue = .fastlaneDefault(false), + destination: String = "") +{ + let forceArg = force.asRubyArgument(name: "force", type: nil) + let destinationArg = RubyCommand.Argument(name: "destination", value: destination, type: nil) + let array: [RubyCommand.Argument?] = [forceArg, + destinationArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "hg_push", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Send a error/success message to [HipChat](https://www.hipchat.com/) + + - parameters: + - message: The message to post on HipChat + - channel: The room or @username + - apiToken: Hipchat API Token + - customColor: Specify a custom color, this overrides the success boolean. Can be one of 'yellow', 'red', 'green', 'purple', 'gray', or 'random' + - success: Was this build successful? (true/false) + - version: Version of the Hipchat API. Must be 1 or 2 + - notifyRoom: Should the people in the room be notified? (true/false) + - apiHost: The host of the HipChat-Server API + - messageFormat: Format of the message to post. Must be either 'html' or 'text' + - includeHtmlHeader: Should html formatted messages include a preformatted header? (true/false) + - from: Name the message will appear to be sent from + + Send a message to **room** (by default) or a direct message to **@username** with success (green) or failure (red) status. + */ +public func hipchat(message: String = "", + channel: String, + apiToken: String, + customColor: OptionalConfigValue = .fastlaneDefault(nil), + success: OptionalConfigValue = .fastlaneDefault(true), + version: String, + notifyRoom: OptionalConfigValue = .fastlaneDefault(false), + apiHost: String = "api.hipchat.com", + messageFormat: String = "html", + includeHtmlHeader: OptionalConfigValue = .fastlaneDefault(true), + from: String = "fastlane") +{ + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let channelArg = RubyCommand.Argument(name: "channel", value: channel, type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let customColorArg = customColor.asRubyArgument(name: "custom_color", type: nil) + let successArg = success.asRubyArgument(name: "success", type: nil) + let versionArg = RubyCommand.Argument(name: "version", value: version, type: nil) + let notifyRoomArg = notifyRoom.asRubyArgument(name: "notify_room", type: nil) + let apiHostArg = RubyCommand.Argument(name: "api_host", value: apiHost, type: nil) + let messageFormatArg = RubyCommand.Argument(name: "message_format", value: messageFormat, type: nil) + let includeHtmlHeaderArg = includeHtmlHeader.asRubyArgument(name: "include_html_header", type: nil) + let fromArg = RubyCommand.Argument(name: "from", value: from, type: nil) + let array: [RubyCommand.Argument?] = [messageArg, + channelArg, + apiTokenArg, + customColorArg, + successArg, + versionArg, + notifyRoomArg, + apiHostArg, + messageFormatArg, + includeHtmlHeaderArg, + fromArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "hipchat", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Refer to [App Center](https://github.com/Microsoft/fastlane-plugin-appcenter/) + + - parameters: + - apk: Path to your APK file + - apiToken: API Token for Hockey Access + - ipa: Path to your IPA file. Optional if you use the _gym_ or _xcodebuild_ action. For Mac zip the .app. For Android provide path to .apk file. In addition you could use this to upload .msi, .zip, .pkg, etc if you use the 'create_update' mechanism + - dsym: Path to your symbols file. For iOS and Mac provide path to app.dSYM.zip. For Android provide path to mappings.txt file + - createUpdate: Set true if you want to create then update your app as opposed to just upload it. You will need the 'public_identifier', 'bundle_version' and 'bundle_short_version' + - notes: Beta Notes + - notify: Notify testers? "1" for yes + - status: Download status: "1" = No user can download; "2" = Available for download (only possible with full-access token) + - createStatus: Download status for initial version creation when create_update is true: "1" = No user can download; "2" = Available for download (only possible with full-access token) + - notesType: Notes type for your :notes, "0" = Textile, "1" = Markdown (default) + - releaseType: Release type of the app: "0" = Beta (default), "1" = Store, "2" = Alpha, "3" = Enterprise + - mandatory: Set to "1" to make this update mandatory + - teams: Comma separated list of team ID numbers to which this build will be restricted + - users: Comma separated list of user ID numbers to which this build will be restricted + - tags: Comma separated list of tags which will receive access to the build + - bundleShortVersion: The bundle_short_version of your application, required when using `create_update` + - bundleVersion: The bundle_version of your application, required when using `create_update` + - publicIdentifier: App id of the app you are targeting, usually you won't need this value. Required, if `upload_dsym_only` set to `true` + - commitSha: The Git commit SHA for this build + - repositoryUrl: The URL of your source repository + - buildServerUrl: The URL of the build job on your build server + - uploadDsymOnly: Flag to upload only the dSYM file to hockey app + - ownerId: ID for the owner of the app + - strategy: Strategy: 'add' = to add the build as a new build even if it has the same build number (default); 'replace' = to replace a build with the same build number + - timeout: Request timeout in seconds + - bypassCdn: Flag to bypass Hockey CDN when it uploads successfully but reports error + - dsaSignature: DSA signature for sparkle updates for macOS + + HockeyApp will be no longer supported and will be transitioned into App Center on November 16, 2019. + Please migrate over to [App Center](https://github.com/Microsoft/fastlane-plugin-appcenter/) + + Symbols will also be uploaded automatically if a `app.dSYM.zip` file is found next to `app.ipa`. In case it is located in a different place you can specify the path explicitly in the `:dsym` parameter. + More information about the available options can be found in the [HockeyApp Docs](http://support.hockeyapp.net/kb/api/api-versions#upload-version). + */ +public func hockey(apk: OptionalConfigValue = .fastlaneDefault(nil), + apiToken: String, + ipa: OptionalConfigValue = .fastlaneDefault(nil), + dsym: OptionalConfigValue = .fastlaneDefault(nil), + createUpdate: OptionalConfigValue = .fastlaneDefault(false), + notes: String = "No changelog given", + notify: String = "1", + status: String = "2", + createStatus: String = "2", + notesType: String = "1", + releaseType: String = "0", + mandatory: String = "0", + teams: OptionalConfigValue = .fastlaneDefault(nil), + users: OptionalConfigValue = .fastlaneDefault(nil), + tags: OptionalConfigValue = .fastlaneDefault(nil), + bundleShortVersion: OptionalConfigValue = .fastlaneDefault(nil), + bundleVersion: OptionalConfigValue = .fastlaneDefault(nil), + publicIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + commitSha: OptionalConfigValue = .fastlaneDefault(nil), + repositoryUrl: OptionalConfigValue = .fastlaneDefault(nil), + buildServerUrl: OptionalConfigValue = .fastlaneDefault(nil), + uploadDsymOnly: OptionalConfigValue = .fastlaneDefault(false), + ownerId: OptionalConfigValue = .fastlaneDefault(nil), + strategy: String = "add", + timeout: OptionalConfigValue = .fastlaneDefault(nil), + bypassCdn: OptionalConfigValue = .fastlaneDefault(false), + dsaSignature: String = "") +{ + let apkArg = apk.asRubyArgument(name: "apk", type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let dsymArg = dsym.asRubyArgument(name: "dsym", type: nil) + let createUpdateArg = createUpdate.asRubyArgument(name: "create_update", type: nil) + let notesArg = RubyCommand.Argument(name: "notes", value: notes, type: nil) + let notifyArg = RubyCommand.Argument(name: "notify", value: notify, type: nil) + let statusArg = RubyCommand.Argument(name: "status", value: status, type: nil) + let createStatusArg = RubyCommand.Argument(name: "create_status", value: createStatus, type: nil) + let notesTypeArg = RubyCommand.Argument(name: "notes_type", value: notesType, type: nil) + let releaseTypeArg = RubyCommand.Argument(name: "release_type", value: releaseType, type: nil) + let mandatoryArg = RubyCommand.Argument(name: "mandatory", value: mandatory, type: nil) + let teamsArg = teams.asRubyArgument(name: "teams", type: nil) + let usersArg = users.asRubyArgument(name: "users", type: nil) + let tagsArg = tags.asRubyArgument(name: "tags", type: nil) + let bundleShortVersionArg = bundleShortVersion.asRubyArgument(name: "bundle_short_version", type: nil) + let bundleVersionArg = bundleVersion.asRubyArgument(name: "bundle_version", type: nil) + let publicIdentifierArg = publicIdentifier.asRubyArgument(name: "public_identifier", type: nil) + let commitShaArg = commitSha.asRubyArgument(name: "commit_sha", type: nil) + let repositoryUrlArg = repositoryUrl.asRubyArgument(name: "repository_url", type: nil) + let buildServerUrlArg = buildServerUrl.asRubyArgument(name: "build_server_url", type: nil) + let uploadDsymOnlyArg = uploadDsymOnly.asRubyArgument(name: "upload_dsym_only", type: nil) + let ownerIdArg = ownerId.asRubyArgument(name: "owner_id", type: nil) + let strategyArg = RubyCommand.Argument(name: "strategy", value: strategy, type: nil) + let timeoutArg = timeout.asRubyArgument(name: "timeout", type: nil) + let bypassCdnArg = bypassCdn.asRubyArgument(name: "bypass_cdn", type: nil) + let dsaSignatureArg = RubyCommand.Argument(name: "dsa_signature", value: dsaSignature, type: nil) + let array: [RubyCommand.Argument?] = [apkArg, + apiTokenArg, + ipaArg, + dsymArg, + createUpdateArg, + notesArg, + notifyArg, + statusArg, + createStatusArg, + notesTypeArg, + releaseTypeArg, + mandatoryArg, + teamsArg, + usersArg, + tagsArg, + bundleShortVersionArg, + bundleVersionArg, + publicIdentifierArg, + commitShaArg, + repositoryUrlArg, + buildServerUrlArg, + uploadDsymOnlyArg, + ownerIdArg, + strategyArg, + timeoutArg, + bypassCdnArg, + dsaSignatureArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "hockey", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Connect to the [IFTTT Maker Channel](https://ifttt.com/maker) + + - parameters: + - apiKey: API key + - eventName: The name of the event that will be triggered + - value1: Extra data sent with the event + - value2: Extra data sent with the event + - value3: Extra data sent with the event + + Connect to the IFTTT [Maker Channel](https://ifttt.com/maker). An IFTTT Recipe has two components: a Trigger and an Action. In this case, the Trigger will fire every time the Maker Channel receives a web request (made by this _fastlane_ action) to notify it of an event. The Action can be anything that IFTTT supports: email, SMS, etc. + */ +public func ifttt(apiKey: String, + eventName: String, + value1: OptionalConfigValue = .fastlaneDefault(nil), + value2: OptionalConfigValue = .fastlaneDefault(nil), + value3: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiKeyArg = RubyCommand.Argument(name: "api_key", value: apiKey, type: nil) + let eventNameArg = RubyCommand.Argument(name: "event_name", value: eventName, type: nil) + let value1Arg = value1.asRubyArgument(name: "value1", type: nil) + let value2Arg = value2.asRubyArgument(name: "value2", type: nil) + let value3Arg = value3.asRubyArgument(name: "value3", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyArg, + eventNameArg, + value1Arg, + value2Arg, + value3Arg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ifttt", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Import certificate from inputfile into a keychain + + - parameters: + - certificatePath: Path to certificate + - certificatePassword: Certificate password + - keychainName: Keychain the items should be imported to + - keychainPath: Path to the Keychain file to which the items should be imported + - keychainPassword: The password for the keychain. Note that for the login keychain this is your user's password + - logOutput: If output should be logged to the console + + Import certificates (and private keys) into the current default keychain. Use the `create_keychain` action to create a new keychain. + */ +public func importCertificate(certificatePath: String, + certificatePassword: OptionalConfigValue = .fastlaneDefault(nil), + keychainName: String, + keychainPath: OptionalConfigValue = .fastlaneDefault(nil), + keychainPassword: OptionalConfigValue = .fastlaneDefault(nil), + logOutput: OptionalConfigValue = .fastlaneDefault(false)) +{ + let certificatePathArg = RubyCommand.Argument(name: "certificate_path", value: certificatePath, type: nil) + let certificatePasswordArg = certificatePassword.asRubyArgument(name: "certificate_password", type: nil) + let keychainNameArg = RubyCommand.Argument(name: "keychain_name", value: keychainName, type: nil) + let keychainPathArg = keychainPath.asRubyArgument(name: "keychain_path", type: nil) + let keychainPasswordArg = keychainPassword.asRubyArgument(name: "keychain_password", type: nil) + let logOutputArg = logOutput.asRubyArgument(name: "log_output", type: nil) + let array: [RubyCommand.Argument?] = [certificatePathArg, + certificatePasswordArg, + keychainNameArg, + keychainPathArg, + keychainPasswordArg, + logOutputArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "import_certificate", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Increment the build number of your project + + - parameters: + - buildNumber: Change to a specific version. When you provide this parameter, Apple Generic Versioning does not have to be enabled + - skipInfoPlist: Don't update Info.plist files when updating the build version + - xcodeproj: optional, you must specify the path to your main Xcode project if it is not in the project root directory + + - returns: The new build number + */ +@discardableResult public func incrementBuildNumber(buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + skipInfoPlist: OptionalConfigValue = .fastlaneDefault(false), + xcodeproj: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let skipInfoPlistArg = skipInfoPlist.asRubyArgument(name: "skip_info_plist", type: nil) + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let array: [RubyCommand.Argument?] = [buildNumberArg, + skipInfoPlistArg, + xcodeprojArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "increment_build_number", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Increment the version number of your project + + - parameters: + - bumpType: The type of this version bump. Available: patch, minor, major + - versionNumber: Change to a specific version. This will replace the bump type value + - xcodeproj: optional, you must specify the path to your main Xcode project if it is not in the project root directory + + - returns: The new version number + + This action will increment the version number. + You first have to set up your Xcode project, if you haven't done it already: [https://developer.apple.com/library/ios/qa/qa1827/_index.html](https://developer.apple.com/library/ios/qa/qa1827/_index.html). + */ +@discardableResult public func incrementVersionNumber(bumpType: String = "bump", + versionNumber: OptionalConfigValue = .fastlaneDefault(nil), + xcodeproj: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let bumpTypeArg = RubyCommand.Argument(name: "bump_type", value: bumpType, type: nil) + let versionNumberArg = versionNumber.asRubyArgument(name: "version_number", type: nil) + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let array: [RubyCommand.Argument?] = [bumpTypeArg, + versionNumberArg, + xcodeprojArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "increment_version_number", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Installs an .ipa file on a connected iOS-device via usb or wifi + + - parameters: + - extra: Extra Commandline arguments passed to ios-deploy + - deviceId: id of the device / if not set defaults to first found device + - skipWifi: Do not search for devices via WiFi + - ipa: The IPA file to put on the device + + Installs the ipa on the device. If no id is given, the first found iOS device will be used. Works via USB or Wi-Fi. This requires `ios-deploy` to be installed. Please have a look at [ios-deploy](https://github.com/ios-control/ios-deploy). To quickly install it, use `npm -g i ios-deploy` + */ +public func installOnDevice(extra: OptionalConfigValue = .fastlaneDefault(nil), + deviceId: OptionalConfigValue = .fastlaneDefault(nil), + skipWifi: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let extraArg = extra.asRubyArgument(name: "extra", type: nil) + let deviceIdArg = deviceId.asRubyArgument(name: "device_id", type: nil) + let skipWifiArg = skipWifi.asRubyArgument(name: "skip_wifi", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let array: [RubyCommand.Argument?] = [extraArg, + deviceIdArg, + skipWifiArg, + ipaArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "install_on_device", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Install provisioning profile from path + + - parameter path: Path to provisioning profile + + - returns: The absolute path to the installed provisioning profile + + Install provisioning profile from path for current user + */ +@discardableResult public func installProvisioningProfile(path: String) -> String { + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "install_provisioning_profile", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Install an Xcode plugin for the current user + + - parameters: + - url: URL for Xcode plugin ZIP file + - github: GitHub repository URL for Xcode plugin + */ +public func installXcodePlugin(url: String, + github: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let urlArg = RubyCommand.Argument(name: "url", value: url, type: nil) + let githubArg = github.asRubyArgument(name: "github", type: nil) + let array: [RubyCommand.Argument?] = [urlArg, + githubArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "install_xcode_plugin", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload a new build to [Installr](http://installrapp.com/) + + - parameters: + - apiToken: API Token for Installr Access + - ipa: Path to your IPA file. Optional if you use the _gym_ or _xcodebuild_ action + - notes: Release notes + - notify: Groups to notify (e.g. 'dev,qa') + - add: Groups to add (e.g. 'exec,ops') + */ +public func installr(apiToken: String, + ipa: String, + notes: OptionalConfigValue = .fastlaneDefault(nil), + notify: OptionalConfigValue = .fastlaneDefault(nil), + add: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let ipaArg = RubyCommand.Argument(name: "ipa", value: ipa, type: nil) + let notesArg = notes.asRubyArgument(name: "notes", type: nil) + let notifyArg = notify.asRubyArgument(name: "notify", type: nil) + let addArg = add.asRubyArgument(name: "add", type: nil) + let array: [RubyCommand.Argument?] = [apiTokenArg, + ipaArg, + notesArg, + notifyArg, + addArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "installr", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Easily build and sign your app using shenzhen + + - parameters: + - workspace: WORKSPACE Workspace (.xcworkspace) file to use to build app (automatically detected in current directory) + - project: Project (.xcodeproj) file to use to build app (automatically detected in current directory, overridden by --workspace option, if passed) + - configuration: Configuration used to build + - scheme: Scheme used to build app + - clean: Clean project before building + - archive: Archive project after building + - destination: Build destination. Defaults to current directory + - embed: Sign .ipa file with .mobileprovision + - identity: Identity to be used along with --embed + - sdk: Use SDK as the name or path of the base SDK when building the project + - ipa: Specify the name of the .ipa file to generate (including file extension) + - xcconfig: Use an extra XCCONFIG file to build the app + - xcargs: Pass additional arguments to xcodebuild when building the app. Be sure to quote multiple args + */ +public func ipa(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + clean: OptionalConfigValue = .fastlaneDefault(nil), + archive: OptionalConfigValue = .fastlaneDefault(nil), + destination: OptionalConfigValue = .fastlaneDefault(nil), + embed: OptionalConfigValue = .fastlaneDefault(nil), + identity: OptionalConfigValue = .fastlaneDefault(nil), + sdk: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let archiveArg = archive.asRubyArgument(name: "archive", type: nil) + let destinationArg = destination.asRubyArgument(name: "destination", type: nil) + let embedArg = embed.asRubyArgument(name: "embed", type: nil) + let identityArg = identity.asRubyArgument(name: "identity", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + configurationArg, + schemeArg, + cleanArg, + archiveArg, + destinationArg, + embedArg, + identityArg, + sdkArg, + ipaArg, + xcconfigArg, + xcargsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ipa", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Is the current run being executed on a CI system, like Jenkins or Travis + + The return value of this method is true if fastlane is currently executed on Travis, Jenkins, Circle or a similar CI service + */ +@discardableResult public func isCi() -> Bool { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "is_ci", className: nil, args: args) + return parseBool(fromString: runner.executeCommand(command)) +} + +/** + Generate docs using Jazzy + + - parameters: + - config: Path to jazzy config file + - moduleVersion: Version string to use as part of the the default docs title and inside the docset + */ +public func jazzy(config: OptionalConfigValue = .fastlaneDefault(nil), + moduleVersion: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let configArg = config.asRubyArgument(name: "config", type: nil) + let moduleVersionArg = moduleVersion.asRubyArgument(name: "module_version", type: nil) + let array: [RubyCommand.Argument?] = [configArg, + moduleVersionArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "jazzy", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Leave a comment on a Jira ticket + + - parameters: + - url: URL for Jira instance + - contextPath: Appends to the url (ex: "/jira") + - username: Username for Jira instance + - password: Password or API token for Jira + - ticketId: Ticket ID for Jira, i.e. IOS-123 + - commentText: Text to add to the ticket as a comment + - failOnError: Should an error adding the Jira comment cause a failure? + + - returns: A hash containing all relevant information of the Jira comment + Access Jira comment 'id', 'author', 'body', and more + */ +@discardableResult public func jira(url: String, + contextPath: String = "", + username: String, + password: String, + ticketId: String, + commentText: String, + failOnError: OptionalConfigValue = .fastlaneDefault(true)) -> [String: Any] +{ + let urlArg = RubyCommand.Argument(name: "url", value: url, type: nil) + let contextPathArg = RubyCommand.Argument(name: "context_path", value: contextPath, type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let passwordArg = RubyCommand.Argument(name: "password", value: password, type: nil) + let ticketIdArg = RubyCommand.Argument(name: "ticket_id", value: ticketId, type: nil) + let commentTextArg = RubyCommand.Argument(name: "comment_text", value: commentText, type: nil) + let failOnErrorArg = failOnError.asRubyArgument(name: "fail_on_error", type: nil) + let array: [RubyCommand.Argument?] = [urlArg, + contextPathArg, + usernameArg, + passwordArg, + ticketIdArg, + commentTextArg, + failOnErrorArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "jira", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Access lane context values + + Access the fastlane lane context values. + More information about how the lane context works: [https://docs.fastlane.tools/advanced/#lane-context](https://docs.fastlane.tools/advanced/#lane-context). + */ +@discardableResult public func laneContext() -> [String: Any] { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "lane_context", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Return last git commit hash, abbreviated commit hash, commit message and author + + - returns: Returns the following dict: {commit_hash: "commit hash", abbreviated_commit_hash: "abbreviated commit hash" author: "Author", author_email: "author email", message: "commit message"}. Example: {:message=>"message", :author=>"author", :author_email=>"author_email", :commit_hash=>"commit_hash", :abbreviated_commit_hash=>"short_hash"} + */ +@discardableResult public func lastGitCommit() -> [String: String] { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "last_git_commit", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Get the most recent git tag + + - parameter pattern: Pattern to filter tags when looking for last one. Limit tags to ones matching given shell glob. If pattern lacks ?, *, or [, * at the end is implied + + If you are using this action on a **shallow clone**, *the default with some CI systems like Bamboo*, you need to ensure that you have also pulled all the git tags appropriately. Assuming your git repo has the correct remote set you can issue `sh('git fetch --tags')`. + Pattern parameter allows you to filter to a subset of tags. + */ +@discardableResult public func lastGitTag(pattern: OptionalConfigValue = .fastlaneDefault(nil)) -> String { + let patternArg = pattern.asRubyArgument(name: "pattern", type: nil) + let array: [RubyCommand.Argument?] = [patternArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "last_git_tag", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Fetches most recent build number from TestFlight + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - live: Query the live version (ready-for-sale) + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - version: The version number whose latest build number we want + - platform: The platform to use (optional) + - initialBuildNumber: sets the build number to given value if no build is in current train + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + + - returns: Integer representation of the latest build number uploaded to TestFlight. Example: 2 + + Provides a way to have `increment_build_number` be based on the latest build you uploaded to iTC. + Fetches the most recent build number from TestFlight based on the version number. Provides a way to have `increment_build_number` be based on the latest build you uploaded to iTC. + */ +@discardableResult public func latestTestflightBuildNumber(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + live: OptionalConfigValue = .fastlaneDefault(false), + appIdentifier: String, + username: OptionalConfigValue = .fastlaneDefault(nil), + version: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + initialBuildNumber: Int = 1, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil)) -> Int +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let liveArg = live.asRubyArgument(name: "live", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let versionArg = version.asRubyArgument(name: "version", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let initialBuildNumberArg = RubyCommand.Argument(name: "initial_build_number", value: initialBuildNumber, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + liveArg, + appIdentifierArg, + usernameArg, + versionArg, + platformArg, + initialBuildNumberArg, + teamIdArg, + teamNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "latest_testflight_build_number", className: nil, args: args) + return parseInt(fromString: runner.executeCommand(command)) +} + +/** + Generates coverage data using lcov + + - parameters: + - projectName: Name of the project + - scheme: Scheme of the project + - arch: The build arch where will search .gcda files + - outputDir: The output directory that coverage data will be stored. If not passed will use coverage_reports as default value + */ +public func lcov(projectName: String, + scheme: String, + arch: String = "i386", + outputDir: String = "coverage_reports") +{ + let projectNameArg = RubyCommand.Argument(name: "project_name", value: projectName, type: nil) + let schemeArg = RubyCommand.Argument(name: "scheme", value: scheme, type: nil) + let archArg = RubyCommand.Argument(name: "arch", value: arch, type: nil) + let outputDirArg = RubyCommand.Argument(name: "output_dir", value: outputDir, type: nil) + let array: [RubyCommand.Argument?] = [projectNameArg, + schemeArg, + archArg, + outputDirArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "lcov", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Send a success/error message to an email group + + - parameters: + - mailgunSandboxDomain: Mailgun sandbox domain postmaster for your mail. Please use postmaster instead + - mailgunSandboxPostmaster: Mailgun sandbox domain postmaster for your mail. Please use postmaster instead + - mailgunApikey: Mailgun apikey for your mail. Please use postmaster instead + - postmaster: Mailgun sandbox domain postmaster for your mail + - apikey: Mailgun apikey for your mail + - to: Destination of your mail + - from: Mailgun sender name + - message: Message of your mail + - subject: Subject of your mail + - success: Was this build successful? (true/false) + - appLink: App Release link + - ciBuildLink: CI Build Link + - templatePath: Mail HTML template + - replyTo: Mail Reply to + - attachment: Mail Attachment filenames, either an array or just one string + - customPlaceholders: Placeholders for template given as a hash + */ +public func mailgun(mailgunSandboxDomain: OptionalConfigValue = .fastlaneDefault(nil), + mailgunSandboxPostmaster: OptionalConfigValue = .fastlaneDefault(nil), + mailgunApikey: OptionalConfigValue = .fastlaneDefault(nil), + postmaster: String, + apikey: String, + to: String, + from: String = "Mailgun Sandbox", + message: String, + subject: String = "fastlane build", + success: OptionalConfigValue = .fastlaneDefault(true), + appLink: String, + ciBuildLink: OptionalConfigValue = .fastlaneDefault(nil), + templatePath: OptionalConfigValue = .fastlaneDefault(nil), + replyTo: OptionalConfigValue = .fastlaneDefault(nil), + attachment: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + customPlaceholders: [String: Any] = [:]) +{ + let mailgunSandboxDomainArg = mailgunSandboxDomain.asRubyArgument(name: "mailgun_sandbox_domain", type: nil) + let mailgunSandboxPostmasterArg = mailgunSandboxPostmaster.asRubyArgument(name: "mailgun_sandbox_postmaster", type: nil) + let mailgunApikeyArg = mailgunApikey.asRubyArgument(name: "mailgun_apikey", type: nil) + let postmasterArg = RubyCommand.Argument(name: "postmaster", value: postmaster, type: nil) + let apikeyArg = RubyCommand.Argument(name: "apikey", value: apikey, type: nil) + let toArg = RubyCommand.Argument(name: "to", value: to, type: nil) + let fromArg = RubyCommand.Argument(name: "from", value: from, type: nil) + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let subjectArg = RubyCommand.Argument(name: "subject", value: subject, type: nil) + let successArg = success.asRubyArgument(name: "success", type: nil) + let appLinkArg = RubyCommand.Argument(name: "app_link", value: appLink, type: nil) + let ciBuildLinkArg = ciBuildLink.asRubyArgument(name: "ci_build_link", type: nil) + let templatePathArg = templatePath.asRubyArgument(name: "template_path", type: nil) + let replyToArg = replyTo.asRubyArgument(name: "reply_to", type: nil) + let attachmentArg = attachment.asRubyArgument(name: "attachment", type: nil) + let customPlaceholdersArg = RubyCommand.Argument(name: "custom_placeholders", value: customPlaceholders, type: nil) + let array: [RubyCommand.Argument?] = [mailgunSandboxDomainArg, + mailgunSandboxPostmasterArg, + mailgunApikeyArg, + postmasterArg, + apikeyArg, + toArg, + fromArg, + messageArg, + subjectArg, + successArg, + appLinkArg, + ciBuildLinkArg, + templatePathArg, + replyToArg, + attachmentArg, + customPlaceholdersArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "mailgun", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generate a changelog using the Changes section from the current Jenkins build + + - parameters: + - fallbackChangelog: Fallback changelog if there is not one on Jenkins, or it couldn't be read + - includeCommitBody: Include the commit body along with the summary + + This is useful when deploying automated builds. The changelog from Jenkins lists all the commit messages since the last build. + */ +public func makeChangelogFromJenkins(fallbackChangelog: String = "", + includeCommitBody: OptionalConfigValue = .fastlaneDefault(true)) +{ + let fallbackChangelogArg = RubyCommand.Argument(name: "fallback_changelog", value: fallbackChangelog, type: nil) + let includeCommitBodyArg = includeCommitBody.asRubyArgument(name: "include_commit_body", type: nil) + let array: [RubyCommand.Argument?] = [fallbackChangelogArg, + includeCommitBodyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "make_changelog_from_jenkins", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `sync_code_signing` action + + - parameters: + - type: Define the profile type, can be appstore, adhoc, development, enterprise, developer_id, mac_installer_distribution + - additionalCertTypes: Create additional cert types needed for macOS installers (valid values: mac_installer_distribution, developer_id_installer) + - readonly: Only fetch existing certificates and profiles, don't generate new ones + - generateAppleCerts: Create a certificate type for Xcode 11 and later (Apple Development or Apple Distribution) + - skipProvisioningProfiles: Skip syncing provisioning profiles + - appIdentifier: The bundle identifier(s) of your app (comma-separated string or array of strings) + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - storageMode: Define where you want to store your certificates + - gitUrl: URL to the git repo containing all the certificates + - gitBranch: Specific git branch to use + - gitFullName: git user full name to commit + - gitUserEmail: git user email to commit + - shallowClone: Make a shallow clone of the repository (truncate the history to 1 revision) + - cloneBranchDirectly: Clone just the branch specified, instead of the whole repo. This requires that the branch already exists. Otherwise the command will fail + - gitBasicAuthorization: Use a basic authorization header to access the git repo (e.g.: access via HTTPS, GitHub Actions, etc), usually a string in Base64 + - gitBearerAuthorization: Use a bearer authorization header to access the git repo (e.g.: access to an Azure DevOps repository), usually a string in Base64 + - gitPrivateKey: Use a private key to access the git repo (e.g.: access to GitHub repository via Deploy keys), usually a id_rsa named file or the contents hereof + - googleCloudBucketName: Name of the Google Cloud Storage bucket to use + - googleCloudKeysFile: Path to the gc_keys.json file + - googleCloudProjectId: ID of the Google Cloud project to use for authentication + - skipGoogleCloudAccountConfirmation: Skips confirming to use the system google account + - s3Region: Name of the S3 region + - s3AccessKey: S3 access key + - s3SecretAccessKey: S3 secret access key + - s3Bucket: Name of the S3 bucket + - s3ObjectPrefix: Prefix to be used on all objects uploaded to S3 + - gitlabProject: GitLab Project Path (i.e. 'gitlab-org/gitlab') + - keychainName: Keychain the items should be imported to + - keychainPassword: This might be required the first time you access certificates on a new mac. For the login/default keychain this is your macOS account password + - force: Renew the provisioning profiles every time you run match + - forceForNewDevices: Renew the provisioning profiles if the device count on the developer portal has changed. Ignored for profile types 'appstore' and 'developer_id' + - includeMacInProfiles: Include Apple Silicon Mac devices in provisioning profiles for iOS/iPadOS apps + - includeAllCertificates: Include all matching certificates in the provisioning profile. Works only for the 'development' provisioning profile type + - forceForNewCertificates: Renew the provisioning profiles if the certificate count on the developer portal has changed. Works only for the 'development' provisioning profile type. Requires 'include_all_certificates' option to be 'true' + - skipConfirmation: Disables confirmation prompts during nuke, answering them with yes + - safeRemoveCerts: Remove certs from repository during nuke without revoking them on the developer portal + - skipDocs: Skip generation of a README.md for the created git repository + - platform: Set the provisioning profile's platform to work with (i.e. ios, tvos, macos, catalyst) + - deriveCatalystAppIdentifier: Enable this if you have the Mac Catalyst capability enabled and your project was created with Xcode 11.3 or earlier. Prepends 'maccatalyst.' to the app identifier for the provisioning profile mapping + - templateName: The name of provisioning profile template. If the developer account has provisioning profile templates (aka: custom entitlements), the template name can be found by inspecting the Entitlements drop-down while creating/editing a provisioning profile (e.g. "Apple Pay Pass Suppression Development") + - profileName: A custom name for the provisioning profile. This will replace the default provisioning profile name if specified + - failOnNameTaken: Should the command fail if it was about to create a duplicate of an existing provisioning profile. It can happen due to issues on Apple Developer Portal, when profile to be recreated was not properly deleted first + - skipCertificateMatching: Set to true if there is no access to Apple developer portal but there are certificates, keys and profiles provided. Only works with match import action + - outputPath: Path in which to export certificates, key and profile + - skipSetPartitionList: Skips setting the partition list (which can sometimes take a long time). Setting the partition list is usually needed to prevent Xcode from prompting to allow a cert to be used for signing + - verbose: Print out extra information and all commands + + More information: https://docs.fastlane.tools/actions/match/ + */ +public func match(type: String = matchfile.type, + additionalCertTypes: OptionalConfigValue<[String]?> = .fastlaneDefault(matchfile.additionalCertTypes), + readonly: OptionalConfigValue = .fastlaneDefault(matchfile.readonly), + generateAppleCerts: OptionalConfigValue = .fastlaneDefault(matchfile.generateAppleCerts), + skipProvisioningProfiles: OptionalConfigValue = .fastlaneDefault(matchfile.skipProvisioningProfiles), + appIdentifier: [String] = matchfile.appIdentifier, + apiKeyPath: OptionalConfigValue = .fastlaneDefault(matchfile.apiKeyPath), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(matchfile.apiKey), + username: OptionalConfigValue = .fastlaneDefault(matchfile.username), + teamId: OptionalConfigValue = .fastlaneDefault(matchfile.teamId), + teamName: OptionalConfigValue = .fastlaneDefault(matchfile.teamName), + storageMode: String = matchfile.storageMode, + gitUrl: String = matchfile.gitUrl, + gitBranch: String = matchfile.gitBranch, + gitFullName: OptionalConfigValue = .fastlaneDefault(matchfile.gitFullName), + gitUserEmail: OptionalConfigValue = .fastlaneDefault(matchfile.gitUserEmail), + shallowClone: OptionalConfigValue = .fastlaneDefault(matchfile.shallowClone), + cloneBranchDirectly: OptionalConfigValue = .fastlaneDefault(matchfile.cloneBranchDirectly), + gitBasicAuthorization: OptionalConfigValue = .fastlaneDefault(matchfile.gitBasicAuthorization), + gitBearerAuthorization: OptionalConfigValue = .fastlaneDefault(matchfile.gitBearerAuthorization), + gitPrivateKey: OptionalConfigValue = .fastlaneDefault(matchfile.gitPrivateKey), + googleCloudBucketName: OptionalConfigValue = .fastlaneDefault(matchfile.googleCloudBucketName), + googleCloudKeysFile: OptionalConfigValue = .fastlaneDefault(matchfile.googleCloudKeysFile), + googleCloudProjectId: OptionalConfigValue = .fastlaneDefault(matchfile.googleCloudProjectId), + skipGoogleCloudAccountConfirmation: OptionalConfigValue = .fastlaneDefault(matchfile.skipGoogleCloudAccountConfirmation), + s3Region: OptionalConfigValue = .fastlaneDefault(matchfile.s3Region), + s3AccessKey: OptionalConfigValue = .fastlaneDefault(matchfile.s3AccessKey), + s3SecretAccessKey: OptionalConfigValue = .fastlaneDefault(matchfile.s3SecretAccessKey), + s3Bucket: OptionalConfigValue = .fastlaneDefault(matchfile.s3Bucket), + s3ObjectPrefix: OptionalConfigValue = .fastlaneDefault(matchfile.s3ObjectPrefix), + gitlabProject: OptionalConfigValue = .fastlaneDefault(matchfile.gitlabProject), + keychainName: String = matchfile.keychainName, + keychainPassword: OptionalConfigValue = .fastlaneDefault(matchfile.keychainPassword), + force: OptionalConfigValue = .fastlaneDefault(matchfile.force), + forceForNewDevices: OptionalConfigValue = .fastlaneDefault(matchfile.forceForNewDevices), + includeMacInProfiles: OptionalConfigValue = .fastlaneDefault(matchfile.includeMacInProfiles), + includeAllCertificates: OptionalConfigValue = .fastlaneDefault(matchfile.includeAllCertificates), + forceForNewCertificates: OptionalConfigValue = .fastlaneDefault(matchfile.forceForNewCertificates), + skipConfirmation: OptionalConfigValue = .fastlaneDefault(matchfile.skipConfirmation), + safeRemoveCerts: OptionalConfigValue = .fastlaneDefault(matchfile.safeRemoveCerts), + skipDocs: OptionalConfigValue = .fastlaneDefault(matchfile.skipDocs), + platform: String = matchfile.platform, + deriveCatalystAppIdentifier: OptionalConfigValue = .fastlaneDefault(matchfile.deriveCatalystAppIdentifier), + templateName: OptionalConfigValue = .fastlaneDefault(matchfile.templateName), + profileName: OptionalConfigValue = .fastlaneDefault(matchfile.profileName), + failOnNameTaken: OptionalConfigValue = .fastlaneDefault(matchfile.failOnNameTaken), + skipCertificateMatching: OptionalConfigValue = .fastlaneDefault(matchfile.skipCertificateMatching), + outputPath: OptionalConfigValue = .fastlaneDefault(matchfile.outputPath), + skipSetPartitionList: OptionalConfigValue = .fastlaneDefault(matchfile.skipSetPartitionList), + verbose: OptionalConfigValue = .fastlaneDefault(matchfile.verbose)) +{ + let typeArg = RubyCommand.Argument(name: "type", value: type, type: nil) + let additionalCertTypesArg = additionalCertTypes.asRubyArgument(name: "additional_cert_types", type: nil) + let readonlyArg = readonly.asRubyArgument(name: "readonly", type: nil) + let generateAppleCertsArg = generateAppleCerts.asRubyArgument(name: "generate_apple_certs", type: nil) + let skipProvisioningProfilesArg = skipProvisioningProfiles.asRubyArgument(name: "skip_provisioning_profiles", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let storageModeArg = RubyCommand.Argument(name: "storage_mode", value: storageMode, type: nil) + let gitUrlArg = RubyCommand.Argument(name: "git_url", value: gitUrl, type: nil) + let gitBranchArg = RubyCommand.Argument(name: "git_branch", value: gitBranch, type: nil) + let gitFullNameArg = gitFullName.asRubyArgument(name: "git_full_name", type: nil) + let gitUserEmailArg = gitUserEmail.asRubyArgument(name: "git_user_email", type: nil) + let shallowCloneArg = shallowClone.asRubyArgument(name: "shallow_clone", type: nil) + let cloneBranchDirectlyArg = cloneBranchDirectly.asRubyArgument(name: "clone_branch_directly", type: nil) + let gitBasicAuthorizationArg = gitBasicAuthorization.asRubyArgument(name: "git_basic_authorization", type: nil) + let gitBearerAuthorizationArg = gitBearerAuthorization.asRubyArgument(name: "git_bearer_authorization", type: nil) + let gitPrivateKeyArg = gitPrivateKey.asRubyArgument(name: "git_private_key", type: nil) + let googleCloudBucketNameArg = googleCloudBucketName.asRubyArgument(name: "google_cloud_bucket_name", type: nil) + let googleCloudKeysFileArg = googleCloudKeysFile.asRubyArgument(name: "google_cloud_keys_file", type: nil) + let googleCloudProjectIdArg = googleCloudProjectId.asRubyArgument(name: "google_cloud_project_id", type: nil) + let skipGoogleCloudAccountConfirmationArg = skipGoogleCloudAccountConfirmation.asRubyArgument(name: "skip_google_cloud_account_confirmation", type: nil) + let s3RegionArg = s3Region.asRubyArgument(name: "s3_region", type: nil) + let s3AccessKeyArg = s3AccessKey.asRubyArgument(name: "s3_access_key", type: nil) + let s3SecretAccessKeyArg = s3SecretAccessKey.asRubyArgument(name: "s3_secret_access_key", type: nil) + let s3BucketArg = s3Bucket.asRubyArgument(name: "s3_bucket", type: nil) + let s3ObjectPrefixArg = s3ObjectPrefix.asRubyArgument(name: "s3_object_prefix", type: nil) + let gitlabProjectArg = gitlabProject.asRubyArgument(name: "gitlab_project", type: nil) + let keychainNameArg = RubyCommand.Argument(name: "keychain_name", value: keychainName, type: nil) + let keychainPasswordArg = keychainPassword.asRubyArgument(name: "keychain_password", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let forceForNewDevicesArg = forceForNewDevices.asRubyArgument(name: "force_for_new_devices", type: nil) + let includeMacInProfilesArg = includeMacInProfiles.asRubyArgument(name: "include_mac_in_profiles", type: nil) + let includeAllCertificatesArg = includeAllCertificates.asRubyArgument(name: "include_all_certificates", type: nil) + let forceForNewCertificatesArg = forceForNewCertificates.asRubyArgument(name: "force_for_new_certificates", type: nil) + let skipConfirmationArg = skipConfirmation.asRubyArgument(name: "skip_confirmation", type: nil) + let safeRemoveCertsArg = safeRemoveCerts.asRubyArgument(name: "safe_remove_certs", type: nil) + let skipDocsArg = skipDocs.asRubyArgument(name: "skip_docs", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let deriveCatalystAppIdentifierArg = deriveCatalystAppIdentifier.asRubyArgument(name: "derive_catalyst_app_identifier", type: nil) + let templateNameArg = templateName.asRubyArgument(name: "template_name", type: nil) + let profileNameArg = profileName.asRubyArgument(name: "profile_name", type: nil) + let failOnNameTakenArg = failOnNameTaken.asRubyArgument(name: "fail_on_name_taken", type: nil) + let skipCertificateMatchingArg = skipCertificateMatching.asRubyArgument(name: "skip_certificate_matching", type: nil) + let outputPathArg = outputPath.asRubyArgument(name: "output_path", type: nil) + let skipSetPartitionListArg = skipSetPartitionList.asRubyArgument(name: "skip_set_partition_list", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let array: [RubyCommand.Argument?] = [typeArg, + additionalCertTypesArg, + readonlyArg, + generateAppleCertsArg, + skipProvisioningProfilesArg, + appIdentifierArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + storageModeArg, + gitUrlArg, + gitBranchArg, + gitFullNameArg, + gitUserEmailArg, + shallowCloneArg, + cloneBranchDirectlyArg, + gitBasicAuthorizationArg, + gitBearerAuthorizationArg, + gitPrivateKeyArg, + googleCloudBucketNameArg, + googleCloudKeysFileArg, + googleCloudProjectIdArg, + skipGoogleCloudAccountConfirmationArg, + s3RegionArg, + s3AccessKeyArg, + s3SecretAccessKeyArg, + s3BucketArg, + s3ObjectPrefixArg, + gitlabProjectArg, + keychainNameArg, + keychainPasswordArg, + forceArg, + forceForNewDevicesArg, + includeMacInProfilesArg, + includeAllCertificatesArg, + forceForNewCertificatesArg, + skipConfirmationArg, + safeRemoveCertsArg, + skipDocsArg, + platformArg, + deriveCatalystAppIdentifierArg, + templateNameArg, + profileNameArg, + failOnNameTakenArg, + skipCertificateMatchingArg, + outputPathArg, + skipSetPartitionListArg, + verboseArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "match", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Easily nuke your certificate and provisioning profiles (via _match_) + + - parameters: + - type: Define the profile type, can be appstore, adhoc, development, enterprise, developer_id, mac_installer_distribution + - additionalCertTypes: Create additional cert types needed for macOS installers (valid values: mac_installer_distribution, developer_id_installer) + - readonly: Only fetch existing certificates and profiles, don't generate new ones + - generateAppleCerts: Create a certificate type for Xcode 11 and later (Apple Development or Apple Distribution) + - skipProvisioningProfiles: Skip syncing provisioning profiles + - appIdentifier: The bundle identifier(s) of your app (comma-separated string or array of strings) + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - storageMode: Define where you want to store your certificates + - gitUrl: URL to the git repo containing all the certificates + - gitBranch: Specific git branch to use + - gitFullName: git user full name to commit + - gitUserEmail: git user email to commit + - shallowClone: Make a shallow clone of the repository (truncate the history to 1 revision) + - cloneBranchDirectly: Clone just the branch specified, instead of the whole repo. This requires that the branch already exists. Otherwise the command will fail + - gitBasicAuthorization: Use a basic authorization header to access the git repo (e.g.: access via HTTPS, GitHub Actions, etc), usually a string in Base64 + - gitBearerAuthorization: Use a bearer authorization header to access the git repo (e.g.: access to an Azure DevOps repository), usually a string in Base64 + - gitPrivateKey: Use a private key to access the git repo (e.g.: access to GitHub repository via Deploy keys), usually a id_rsa named file or the contents hereof + - googleCloudBucketName: Name of the Google Cloud Storage bucket to use + - googleCloudKeysFile: Path to the gc_keys.json file + - googleCloudProjectId: ID of the Google Cloud project to use for authentication + - skipGoogleCloudAccountConfirmation: Skips confirming to use the system google account + - s3Region: Name of the S3 region + - s3AccessKey: S3 access key + - s3SecretAccessKey: S3 secret access key + - s3Bucket: Name of the S3 bucket + - s3ObjectPrefix: Prefix to be used on all objects uploaded to S3 + - gitlabProject: GitLab Project Path (i.e. 'gitlab-org/gitlab') + - keychainName: Keychain the items should be imported to + - keychainPassword: This might be required the first time you access certificates on a new mac. For the login/default keychain this is your macOS account password + - force: Renew the provisioning profiles every time you run match + - forceForNewDevices: Renew the provisioning profiles if the device count on the developer portal has changed. Ignored for profile types 'appstore' and 'developer_id' + - includeMacInProfiles: Include Apple Silicon Mac devices in provisioning profiles for iOS/iPadOS apps + - includeAllCertificates: Include all matching certificates in the provisioning profile. Works only for the 'development' provisioning profile type + - forceForNewCertificates: Renew the provisioning profiles if the certificate count on the developer portal has changed. Works only for the 'development' provisioning profile type. Requires 'include_all_certificates' option to be 'true' + - skipConfirmation: Disables confirmation prompts during nuke, answering them with yes + - safeRemoveCerts: Remove certs from repository during nuke without revoking them on the developer portal + - skipDocs: Skip generation of a README.md for the created git repository + - platform: Set the provisioning profile's platform to work with (i.e. ios, tvos, macos, catalyst) + - deriveCatalystAppIdentifier: Enable this if you have the Mac Catalyst capability enabled and your project was created with Xcode 11.3 or earlier. Prepends 'maccatalyst.' to the app identifier for the provisioning profile mapping + - templateName: The name of provisioning profile template. If the developer account has provisioning profile templates (aka: custom entitlements), the template name can be found by inspecting the Entitlements drop-down while creating/editing a provisioning profile (e.g. "Apple Pay Pass Suppression Development") + - profileName: A custom name for the provisioning profile. This will replace the default provisioning profile name if specified + - failOnNameTaken: Should the command fail if it was about to create a duplicate of an existing provisioning profile. It can happen due to issues on Apple Developer Portal, when profile to be recreated was not properly deleted first + - skipCertificateMatching: Set to true if there is no access to Apple developer portal but there are certificates, keys and profiles provided. Only works with match import action + - outputPath: Path in which to export certificates, key and profile + - skipSetPartitionList: Skips setting the partition list (which can sometimes take a long time). Setting the partition list is usually needed to prevent Xcode from prompting to allow a cert to be used for signing + - verbose: Print out extra information and all commands + + Use the match_nuke action to revoke your certificates and provisioning profiles. + Don't worry, apps that are already available in the App Store / TestFlight will still work. + Builds distributed via Ad Hoc or Enterprise will be disabled after nuking your account, so you'll have to re-upload a new build. + After clearing your account you'll start from a clean state, and you can run match to generate your certificates and profiles again. + More information: https://docs.fastlane.tools/actions/match/ + */ +public func matchNuke(type: String = "development", + additionalCertTypes: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + readonly: OptionalConfigValue = .fastlaneDefault(false), + generateAppleCerts: OptionalConfigValue = .fastlaneDefault(true), + skipProvisioningProfiles: OptionalConfigValue = .fastlaneDefault(false), + appIdentifier: [String], + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + storageMode: String = "git", + gitUrl: String, + gitBranch: String = "master", + gitFullName: OptionalConfigValue = .fastlaneDefault(nil), + gitUserEmail: OptionalConfigValue = .fastlaneDefault(nil), + shallowClone: OptionalConfigValue = .fastlaneDefault(false), + cloneBranchDirectly: OptionalConfigValue = .fastlaneDefault(false), + gitBasicAuthorization: OptionalConfigValue = .fastlaneDefault(nil), + gitBearerAuthorization: OptionalConfigValue = .fastlaneDefault(nil), + gitPrivateKey: OptionalConfigValue = .fastlaneDefault(nil), + googleCloudBucketName: OptionalConfigValue = .fastlaneDefault(nil), + googleCloudKeysFile: OptionalConfigValue = .fastlaneDefault(nil), + googleCloudProjectId: OptionalConfigValue = .fastlaneDefault(nil), + skipGoogleCloudAccountConfirmation: OptionalConfigValue = .fastlaneDefault(false), + s3Region: OptionalConfigValue = .fastlaneDefault(nil), + s3AccessKey: OptionalConfigValue = .fastlaneDefault(nil), + s3SecretAccessKey: OptionalConfigValue = .fastlaneDefault(nil), + s3Bucket: OptionalConfigValue = .fastlaneDefault(nil), + s3ObjectPrefix: OptionalConfigValue = .fastlaneDefault(nil), + gitlabProject: OptionalConfigValue = .fastlaneDefault(nil), + keychainName: String = "login.keychain", + keychainPassword: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + forceForNewDevices: OptionalConfigValue = .fastlaneDefault(false), + includeMacInProfiles: OptionalConfigValue = .fastlaneDefault(false), + includeAllCertificates: OptionalConfigValue = .fastlaneDefault(false), + forceForNewCertificates: OptionalConfigValue = .fastlaneDefault(false), + skipConfirmation: OptionalConfigValue = .fastlaneDefault(false), + safeRemoveCerts: OptionalConfigValue = .fastlaneDefault(false), + skipDocs: OptionalConfigValue = .fastlaneDefault(false), + platform: String = "ios", + deriveCatalystAppIdentifier: OptionalConfigValue = .fastlaneDefault(false), + templateName: OptionalConfigValue = .fastlaneDefault(nil), + profileName: OptionalConfigValue = .fastlaneDefault(nil), + failOnNameTaken: OptionalConfigValue = .fastlaneDefault(false), + skipCertificateMatching: OptionalConfigValue = .fastlaneDefault(false), + outputPath: OptionalConfigValue = .fastlaneDefault(nil), + skipSetPartitionList: OptionalConfigValue = .fastlaneDefault(false), + verbose: OptionalConfigValue = .fastlaneDefault(false)) +{ + let typeArg = RubyCommand.Argument(name: "type", value: type, type: nil) + let additionalCertTypesArg = additionalCertTypes.asRubyArgument(name: "additional_cert_types", type: nil) + let readonlyArg = readonly.asRubyArgument(name: "readonly", type: nil) + let generateAppleCertsArg = generateAppleCerts.asRubyArgument(name: "generate_apple_certs", type: nil) + let skipProvisioningProfilesArg = skipProvisioningProfiles.asRubyArgument(name: "skip_provisioning_profiles", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let storageModeArg = RubyCommand.Argument(name: "storage_mode", value: storageMode, type: nil) + let gitUrlArg = RubyCommand.Argument(name: "git_url", value: gitUrl, type: nil) + let gitBranchArg = RubyCommand.Argument(name: "git_branch", value: gitBranch, type: nil) + let gitFullNameArg = gitFullName.asRubyArgument(name: "git_full_name", type: nil) + let gitUserEmailArg = gitUserEmail.asRubyArgument(name: "git_user_email", type: nil) + let shallowCloneArg = shallowClone.asRubyArgument(name: "shallow_clone", type: nil) + let cloneBranchDirectlyArg = cloneBranchDirectly.asRubyArgument(name: "clone_branch_directly", type: nil) + let gitBasicAuthorizationArg = gitBasicAuthorization.asRubyArgument(name: "git_basic_authorization", type: nil) + let gitBearerAuthorizationArg = gitBearerAuthorization.asRubyArgument(name: "git_bearer_authorization", type: nil) + let gitPrivateKeyArg = gitPrivateKey.asRubyArgument(name: "git_private_key", type: nil) + let googleCloudBucketNameArg = googleCloudBucketName.asRubyArgument(name: "google_cloud_bucket_name", type: nil) + let googleCloudKeysFileArg = googleCloudKeysFile.asRubyArgument(name: "google_cloud_keys_file", type: nil) + let googleCloudProjectIdArg = googleCloudProjectId.asRubyArgument(name: "google_cloud_project_id", type: nil) + let skipGoogleCloudAccountConfirmationArg = skipGoogleCloudAccountConfirmation.asRubyArgument(name: "skip_google_cloud_account_confirmation", type: nil) + let s3RegionArg = s3Region.asRubyArgument(name: "s3_region", type: nil) + let s3AccessKeyArg = s3AccessKey.asRubyArgument(name: "s3_access_key", type: nil) + let s3SecretAccessKeyArg = s3SecretAccessKey.asRubyArgument(name: "s3_secret_access_key", type: nil) + let s3BucketArg = s3Bucket.asRubyArgument(name: "s3_bucket", type: nil) + let s3ObjectPrefixArg = s3ObjectPrefix.asRubyArgument(name: "s3_object_prefix", type: nil) + let gitlabProjectArg = gitlabProject.asRubyArgument(name: "gitlab_project", type: nil) + let keychainNameArg = RubyCommand.Argument(name: "keychain_name", value: keychainName, type: nil) + let keychainPasswordArg = keychainPassword.asRubyArgument(name: "keychain_password", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let forceForNewDevicesArg = forceForNewDevices.asRubyArgument(name: "force_for_new_devices", type: nil) + let includeMacInProfilesArg = includeMacInProfiles.asRubyArgument(name: "include_mac_in_profiles", type: nil) + let includeAllCertificatesArg = includeAllCertificates.asRubyArgument(name: "include_all_certificates", type: nil) + let forceForNewCertificatesArg = forceForNewCertificates.asRubyArgument(name: "force_for_new_certificates", type: nil) + let skipConfirmationArg = skipConfirmation.asRubyArgument(name: "skip_confirmation", type: nil) + let safeRemoveCertsArg = safeRemoveCerts.asRubyArgument(name: "safe_remove_certs", type: nil) + let skipDocsArg = skipDocs.asRubyArgument(name: "skip_docs", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let deriveCatalystAppIdentifierArg = deriveCatalystAppIdentifier.asRubyArgument(name: "derive_catalyst_app_identifier", type: nil) + let templateNameArg = templateName.asRubyArgument(name: "template_name", type: nil) + let profileNameArg = profileName.asRubyArgument(name: "profile_name", type: nil) + let failOnNameTakenArg = failOnNameTaken.asRubyArgument(name: "fail_on_name_taken", type: nil) + let skipCertificateMatchingArg = skipCertificateMatching.asRubyArgument(name: "skip_certificate_matching", type: nil) + let outputPathArg = outputPath.asRubyArgument(name: "output_path", type: nil) + let skipSetPartitionListArg = skipSetPartitionList.asRubyArgument(name: "skip_set_partition_list", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let array: [RubyCommand.Argument?] = [typeArg, + additionalCertTypesArg, + readonlyArg, + generateAppleCertsArg, + skipProvisioningProfilesArg, + appIdentifierArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + storageModeArg, + gitUrlArg, + gitBranchArg, + gitFullNameArg, + gitUserEmailArg, + shallowCloneArg, + cloneBranchDirectlyArg, + gitBasicAuthorizationArg, + gitBearerAuthorizationArg, + gitPrivateKeyArg, + googleCloudBucketNameArg, + googleCloudKeysFileArg, + googleCloudProjectIdArg, + skipGoogleCloudAccountConfirmationArg, + s3RegionArg, + s3AccessKeyArg, + s3SecretAccessKeyArg, + s3BucketArg, + s3ObjectPrefixArg, + gitlabProjectArg, + keychainNameArg, + keychainPasswordArg, + forceArg, + forceForNewDevicesArg, + includeMacInProfilesArg, + includeAllCertificatesArg, + forceForNewCertificatesArg, + skipConfirmationArg, + safeRemoveCertsArg, + skipDocsArg, + platformArg, + deriveCatalystAppIdentifierArg, + templateNameArg, + profileNameArg, + failOnNameTakenArg, + skipCertificateMatchingArg, + outputPathArg, + skipSetPartitionListArg, + verboseArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "match_nuke", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Verifies the minimum fastlane version required + + Add this to your `Fastfile` to require a certain version of _fastlane_. + Use it if you use an action that just recently came out and you need it. + */ +public func minFastlaneVersion() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "min_fastlane_version", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Modifies the services of the app created on Developer Portal + + - parameters: + - username: Your Apple ID Username + - appIdentifier: App Identifier (Bundle ID, e.g. com.krausefx.app) + - services: Array with Spaceship App Services (e.g. access_wifi: (on|off)(:on|:off)(true|false), app_attest: (on|off)(:on|:off)(true|false), app_group: (on|off)(:on|:off)(true|false), apple_pay: (on|off)(:on|:off)(true|false), associated_domains: (on|off)(:on|:off)(true|false), auto_fill_credential: (on|off)(:on|:off)(true|false), class_kit: (on|off)(:on|:off)(true|false), icloud: (legacy|cloudkit)(:on|:off)(true|false), custom_network_protocol: (on|off)(:on|:off)(true|false), data_protection: (complete|unlessopen|untilfirstauth)(:on|:off)(true|false), extended_virtual_address_space: (on|off)(:on|:off)(true|false), family_controls: (on|off)(:on|:off)(true|false), file_provider_testing_mode: (on|off)(:on|:off)(true|false), fonts: (on|off)(:on|:off)(true|false), game_center: (ios|mac)(:on|:off)(true|false), health_kit: (on|off)(:on|:off)(true|false), hls_interstitial_preview: (on|off)(:on|:off)(true|false), home_kit: (on|off)(:on|:off)(true|false), hotspot: (on|off)(:on|:off)(true|false), in_app_purchase: (on|off)(:on|:off)(true|false), inter_app_audio: (on|off)(:on|:off)(true|false), low_latency_hls: (on|off)(:on|:off)(true|false), managed_associated_domains: (on|off)(:on|:off)(true|false), maps: (on|off)(:on|:off)(true|false), multipath: (on|off)(:on|:off)(true|false), network_extension: (on|off)(:on|:off)(true|false), nfc_tag_reading: (on|off)(:on|:off)(true|false), personal_vpn: (on|off)(:on|:off)(true|false), passbook: (on|off)(:on|:off)(true|false), push_notification: (on|off)(:on|:off)(true|false), sign_in_with_apple: (on)(:on|:off)(true|false), siri_kit: (on|off)(:on|:off)(true|false), system_extension: (on|off)(:on|:off)(true|false), user_management: (on|off)(:on|:off)(true|false), vpn_configuration: (on|off)(:on|:off)(true|false), wallet: (on|off)(:on|:off)(true|false), wireless_accessory: (on|off)(:on|:off)(true|false), car_play_audio_app: (on|off)(:on|:off)(true|false), car_play_messaging_app: (on|off)(:on|:off)(true|false), car_play_navigation_app: (on|off)(:on|:off)(true|false), car_play_voip_calling_app: (on|off)(:on|:off)(true|false), critical_alerts: (on|off)(:on|:off)(true|false), hotspot_helper: (on|off)(:on|:off)(true|false), driver_kit: (on|off)(:on|:off)(true|false), driver_kit_endpoint_security: (on|off)(:on|:off)(true|false), driver_kit_family_hid_device: (on|off)(:on|:off)(true|false), driver_kit_family_networking: (on|off)(:on|:off)(true|false), driver_kit_family_serial: (on|off)(:on|:off)(true|false), driver_kit_hid_event_service: (on|off)(:on|:off)(true|false), driver_kit_transport_hid: (on|off)(:on|:off)(true|false), multitasking_camera_access: (on|off)(:on|:off)(true|false), sf_universal_link_api: (on|off)(:on|:off)(true|false), vp9_decoder: (on|off)(:on|:off)(true|false), music_kit: (on|off)(:on|:off)(true|false), shazam_kit: (on|off)(:on|:off)(true|false), communication_notifications: (on|off)(:on|:off)(true|false), group_activities: (on|off)(:on|:off)(true|false), health_kit_estimate_recalibration: (on|off)(:on|:off)(true|false), time_sensitive_notifications: (on|off)(:on|:off)(true|false)) + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + + The options are the same as `:enable_services` in the [produce action](https://docs.fastlane.tools/actions/produce/#parameters_1) + */ +public func modifyServices(username: String, + appIdentifier: String, + services: [String: Any] = [:], + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let servicesArg = RubyCommand.Argument(name: "services", value: services, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + appIdentifierArg, + servicesArg, + teamIdArg, + teamNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "modify_services", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload a file to [Sonatype Nexus platform](https://www.sonatype.com) + + - parameters: + - file: File to be uploaded to Nexus + - repoId: Nexus repository id e.g. artefacts + - repoGroupId: Nexus repository group id e.g. com.company + - repoProjectName: Nexus repository commandect name. Only letters, digits, underscores(_), hyphens(-), and dots(.) are allowed + - repoProjectVersion: Nexus repository commandect version + - repoClassifier: Nexus repository artifact classifier (optional) + - endpoint: Nexus endpoint e.g. http://nexus:8081 + - mountPath: Nexus mount path (Nexus 3 instances have this configured as empty by default) + - username: Nexus username + - password: Nexus password + - sslVerify: Verify SSL + - nexusVersion: Nexus major version + - verbose: Make detailed output + - proxyUsername: Proxy username + - proxyPassword: Proxy password + - proxyAddress: Proxy address + - proxyPort: Proxy port + */ +public func nexusUpload(file: String, + repoId: String, + repoGroupId: String, + repoProjectName: String, + repoProjectVersion: String, + repoClassifier: OptionalConfigValue = .fastlaneDefault(nil), + endpoint: String, + mountPath: String = "/nexus", + username: String, + password: String, + sslVerify: OptionalConfigValue = .fastlaneDefault(true), + nexusVersion: Int = 2, + verbose: OptionalConfigValue = .fastlaneDefault(false), + proxyUsername: OptionalConfigValue = .fastlaneDefault(nil), + proxyPassword: OptionalConfigValue = .fastlaneDefault(nil), + proxyAddress: OptionalConfigValue = .fastlaneDefault(nil), + proxyPort: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let fileArg = RubyCommand.Argument(name: "file", value: file, type: nil) + let repoIdArg = RubyCommand.Argument(name: "repo_id", value: repoId, type: nil) + let repoGroupIdArg = RubyCommand.Argument(name: "repo_group_id", value: repoGroupId, type: nil) + let repoProjectNameArg = RubyCommand.Argument(name: "repo_project_name", value: repoProjectName, type: nil) + let repoProjectVersionArg = RubyCommand.Argument(name: "repo_project_version", value: repoProjectVersion, type: nil) + let repoClassifierArg = repoClassifier.asRubyArgument(name: "repo_classifier", type: nil) + let endpointArg = RubyCommand.Argument(name: "endpoint", value: endpoint, type: nil) + let mountPathArg = RubyCommand.Argument(name: "mount_path", value: mountPath, type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let passwordArg = RubyCommand.Argument(name: "password", value: password, type: nil) + let sslVerifyArg = sslVerify.asRubyArgument(name: "ssl_verify", type: nil) + let nexusVersionArg = RubyCommand.Argument(name: "nexus_version", value: nexusVersion, type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let proxyUsernameArg = proxyUsername.asRubyArgument(name: "proxy_username", type: nil) + let proxyPasswordArg = proxyPassword.asRubyArgument(name: "proxy_password", type: nil) + let proxyAddressArg = proxyAddress.asRubyArgument(name: "proxy_address", type: nil) + let proxyPortArg = proxyPort.asRubyArgument(name: "proxy_port", type: nil) + let array: [RubyCommand.Argument?] = [fileArg, + repoIdArg, + repoGroupIdArg, + repoProjectNameArg, + repoProjectVersionArg, + repoClassifierArg, + endpointArg, + mountPathArg, + usernameArg, + passwordArg, + sslVerifyArg, + nexusVersionArg, + verboseArg, + proxyUsernameArg, + proxyPasswordArg, + proxyAddressArg, + proxyPortArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "nexus_upload", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Notarizes a macOS app + + - parameters: + - package: Path to package to notarize, e.g. .app bundle or disk image + - useNotarytool: Whether to `xcrun notarytool` or `xcrun altool` + - tryEarlyStapling: Whether to try early stapling while the notarization request is in progress + - skipStapling: Do not staple the notarization ticket to the artifact; useful for single file executables and ZIP archives + - bundleId: Bundle identifier to uniquely identify the package + - username: Apple ID username + - ascProvider: Provider short name for accounts associated with multiple providers + - printLog: Whether to print notarization log file, listing issues on failure and warnings on success + - verbose: Whether to log requests + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + */ +public func notarize(package: String, + useNotarytool: OptionalConfigValue = .fastlaneDefault(true), + tryEarlyStapling: OptionalConfigValue = .fastlaneDefault(false), + skipStapling: OptionalConfigValue = .fastlaneDefault(false), + bundleId: OptionalConfigValue = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + ascProvider: OptionalConfigValue = .fastlaneDefault(nil), + printLog: OptionalConfigValue = .fastlaneDefault(false), + verbose: OptionalConfigValue = .fastlaneDefault(false), + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil)) +{ + let packageArg = RubyCommand.Argument(name: "package", value: package, type: nil) + let useNotarytoolArg = useNotarytool.asRubyArgument(name: "use_notarytool", type: nil) + let tryEarlyStaplingArg = tryEarlyStapling.asRubyArgument(name: "try_early_stapling", type: nil) + let skipStaplingArg = skipStapling.asRubyArgument(name: "skip_stapling", type: nil) + let bundleIdArg = bundleId.asRubyArgument(name: "bundle_id", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let ascProviderArg = ascProvider.asRubyArgument(name: "asc_provider", type: nil) + let printLogArg = printLog.asRubyArgument(name: "print_log", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let array: [RubyCommand.Argument?] = [packageArg, + useNotarytoolArg, + tryEarlyStaplingArg, + skipStaplingArg, + bundleIdArg, + usernameArg, + ascProviderArg, + printLogArg, + verboseArg, + apiKeyPathArg, + apiKeyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "notarize", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Display a macOS notification with custom message and title + + - parameters: + - title: The title to display in the notification + - subtitle: A subtitle to display in the notification + - message: The message to display in the notification + - sound: The name of a sound to play when the notification appears (names are listed in Sound Preferences) + - activate: Bundle identifier of application to be opened when the notification is clicked + - appIcon: The URL of an image to display instead of the application icon (Mavericks+ only) + - contentImage: The URL of an image to display attached to the notification (Mavericks+ only) + - open: URL of the resource to be opened when the notification is clicked + - execute: Shell command to run when the notification is clicked + */ +public func notification(title: String = "fastlane", + subtitle: OptionalConfigValue = .fastlaneDefault(nil), + message: String, + sound: OptionalConfigValue = .fastlaneDefault(nil), + activate: OptionalConfigValue = .fastlaneDefault(nil), + appIcon: OptionalConfigValue = .fastlaneDefault(nil), + contentImage: OptionalConfigValue = .fastlaneDefault(nil), + open: OptionalConfigValue = .fastlaneDefault(nil), + execute: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let titleArg = RubyCommand.Argument(name: "title", value: title, type: nil) + let subtitleArg = subtitle.asRubyArgument(name: "subtitle", type: nil) + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let soundArg = sound.asRubyArgument(name: "sound", type: nil) + let activateArg = activate.asRubyArgument(name: "activate", type: nil) + let appIconArg = appIcon.asRubyArgument(name: "app_icon", type: nil) + let contentImageArg = contentImage.asRubyArgument(name: "content_image", type: nil) + let openArg = open.asRubyArgument(name: "open", type: nil) + let executeArg = execute.asRubyArgument(name: "execute", type: nil) + let array: [RubyCommand.Argument?] = [titleArg, + subtitleArg, + messageArg, + soundArg, + activateArg, + appIconArg, + contentImageArg, + openArg, + executeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "notification", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Shows a macOS notification - use `notification` instead + */ +public func notify() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "notify", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Return the number of commits in current git branch + + - parameter all: Returns number of all commits instead of current branch + + - returns: The total number of all commits in current git branch + + You can use this action to get the number of commits of this branch. This is useful if you want to set the build number to the number of commits. See `fastlane actions number_of_commits` for more details. + */ +@discardableResult public func numberOfCommits(all: OptionalConfigValue = .fastlaneDefault(nil)) -> Int { + let allArg = all.asRubyArgument(name: "all", type: nil) + let array: [RubyCommand.Argument?] = [allArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "number_of_commits", className: nil, args: args) + return parseInt(fromString: runner.executeCommand(command)) +} + +/** + Lints implementation files with OCLint + + - parameters: + - oclintPath: The path to oclint binary + - compileCommands: The json compilation database, use xctool reporter 'json-compilation-database' + - selectReqex: **DEPRECATED!** Use `:select_regex` instead - Select all files matching this reqex + - selectRegex: Select all files matching this regex + - excludeRegex: Exclude all files matching this regex + - reportType: The type of the report (default: html) + - reportPath: The reports file path + - listEnabledRules: List enabled rules + - rc: Override the default behavior of rules + - thresholds: List of rule thresholds to override the default behavior of rules + - enableRules: List of rules to pick explicitly + - disableRules: List of rules to disable + - maxPriority1: The max allowed number of priority 1 violations + - maxPriority2: The max allowed number of priority 2 violations + - maxPriority3: The max allowed number of priority 3 violations + - enableClangStaticAnalyzer: Enable Clang Static Analyzer, and integrate results into OCLint report + - enableGlobalAnalysis: Compile every source, and analyze across global contexts (depends on number of source files, could results in high memory load) + - allowDuplicatedViolations: Allow duplicated violations in the OCLint report + - extraArg: Additional argument to append to the compiler command line + + Run the static analyzer tool [OCLint](http://oclint.org) for your project. You need to have a `compile_commands.json` file in your _fastlane_ directory or pass a path to your file. + */ +public func oclint(oclintPath: String = "oclint", + compileCommands: String = "compile_commands.json", + selectReqex: OptionalConfigValue = .fastlaneDefault(nil), + selectRegex: OptionalConfigValue = .fastlaneDefault(nil), + excludeRegex: OptionalConfigValue = .fastlaneDefault(nil), + reportType: String = "html", + reportPath: OptionalConfigValue = .fastlaneDefault(nil), + listEnabledRules: OptionalConfigValue = .fastlaneDefault(false), + rc: OptionalConfigValue = .fastlaneDefault(nil), + thresholds: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + enableRules: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + disableRules: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + maxPriority1: OptionalConfigValue = .fastlaneDefault(nil), + maxPriority2: OptionalConfigValue = .fastlaneDefault(nil), + maxPriority3: OptionalConfigValue = .fastlaneDefault(nil), + enableClangStaticAnalyzer: OptionalConfigValue = .fastlaneDefault(false), + enableGlobalAnalysis: OptionalConfigValue = .fastlaneDefault(false), + allowDuplicatedViolations: OptionalConfigValue = .fastlaneDefault(false), + extraArg: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let oclintPathArg = RubyCommand.Argument(name: "oclint_path", value: oclintPath, type: nil) + let compileCommandsArg = RubyCommand.Argument(name: "compile_commands", value: compileCommands, type: nil) + let selectReqexArg = selectReqex.asRubyArgument(name: "select_reqex", type: nil) + let selectRegexArg = selectRegex.asRubyArgument(name: "select_regex", type: nil) + let excludeRegexArg = excludeRegex.asRubyArgument(name: "exclude_regex", type: nil) + let reportTypeArg = RubyCommand.Argument(name: "report_type", value: reportType, type: nil) + let reportPathArg = reportPath.asRubyArgument(name: "report_path", type: nil) + let listEnabledRulesArg = listEnabledRules.asRubyArgument(name: "list_enabled_rules", type: nil) + let rcArg = rc.asRubyArgument(name: "rc", type: nil) + let thresholdsArg = thresholds.asRubyArgument(name: "thresholds", type: nil) + let enableRulesArg = enableRules.asRubyArgument(name: "enable_rules", type: nil) + let disableRulesArg = disableRules.asRubyArgument(name: "disable_rules", type: nil) + let maxPriority1Arg = maxPriority1.asRubyArgument(name: "max_priority_1", type: nil) + let maxPriority2Arg = maxPriority2.asRubyArgument(name: "max_priority_2", type: nil) + let maxPriority3Arg = maxPriority3.asRubyArgument(name: "max_priority_3", type: nil) + let enableClangStaticAnalyzerArg = enableClangStaticAnalyzer.asRubyArgument(name: "enable_clang_static_analyzer", type: nil) + let enableGlobalAnalysisArg = enableGlobalAnalysis.asRubyArgument(name: "enable_global_analysis", type: nil) + let allowDuplicatedViolationsArg = allowDuplicatedViolations.asRubyArgument(name: "allow_duplicated_violations", type: nil) + let extraArgArg = extraArg.asRubyArgument(name: "extra_arg", type: nil) + let array: [RubyCommand.Argument?] = [oclintPathArg, + compileCommandsArg, + selectReqexArg, + selectRegexArg, + excludeRegexArg, + reportTypeArg, + reportPathArg, + listEnabledRulesArg, + rcArg, + thresholdsArg, + enableRulesArg, + disableRulesArg, + maxPriority1Arg, + maxPriority2Arg, + maxPriority3Arg, + enableClangStaticAnalyzerArg, + enableGlobalAnalysisArg, + allowDuplicatedViolationsArg, + extraArgArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "oclint", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Create or update a new [OneSignal](https://onesignal.com/) application + + - parameters: + - appId: OneSignal App ID. Setting this updates an existing app + - authToken: OneSignal Authorization Key + - appName: OneSignal App Name. This is required when creating an app (in other words, when `:app_id` is not set, and optional when updating an app + - androidToken: ANDROID GCM KEY + - androidGcmSenderId: GCM SENDER ID + - apnsP12: APNS P12 File (in .p12 format) + - apnsP12Password: APNS P12 password + - apnsEnv: APNS environment + - organizationId: OneSignal Organization ID + + You can use this action to automatically create or update a OneSignal application. You can also upload a `.p12` with password, a GCM key, or both. + */ +public func onesignal(appId: OptionalConfigValue = .fastlaneDefault(nil), + authToken: String, + appName: OptionalConfigValue = .fastlaneDefault(nil), + androidToken: OptionalConfigValue = .fastlaneDefault(nil), + androidGcmSenderId: OptionalConfigValue = .fastlaneDefault(nil), + apnsP12: OptionalConfigValue = .fastlaneDefault(nil), + apnsP12Password: OptionalConfigValue = .fastlaneDefault(nil), + apnsEnv: String = "production", + organizationId: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let appIdArg = appId.asRubyArgument(name: "app_id", type: nil) + let authTokenArg = RubyCommand.Argument(name: "auth_token", value: authToken, type: nil) + let appNameArg = appName.asRubyArgument(name: "app_name", type: nil) + let androidTokenArg = androidToken.asRubyArgument(name: "android_token", type: nil) + let androidGcmSenderIdArg = androidGcmSenderId.asRubyArgument(name: "android_gcm_sender_id", type: nil) + let apnsP12Arg = apnsP12.asRubyArgument(name: "apns_p12", type: nil) + let apnsP12PasswordArg = apnsP12Password.asRubyArgument(name: "apns_p12_password", type: nil) + let apnsEnvArg = RubyCommand.Argument(name: "apns_env", value: apnsEnv, type: nil) + let organizationIdArg = organizationId.asRubyArgument(name: "organization_id", type: nil) + let array: [RubyCommand.Argument?] = [appIdArg, + authTokenArg, + appNameArg, + androidTokenArg, + androidGcmSenderIdArg, + apnsP12Arg, + apnsP12PasswordArg, + apnsEnvArg, + organizationIdArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "onesignal", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will prevent reports from being uploaded when _fastlane_ crashes + + _fastlane_ doesn't have crash reporting any more. Feel free to remove `opt_out_crash_reporting` from your Fastfile. + */ +public func optOutCrashReporting() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "opt_out_crash_reporting", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will stop uploading the information which actions were run + + By default, _fastlane_ will track what actions are being used. No personal/sensitive information is recorded. + Learn more at [https://docs.fastlane.tools/#metrics](https://docs.fastlane.tools/#metrics). + Add `opt_out_usage` at the top of your Fastfile to disable metrics collection. + */ +public func optOutUsage() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "opt_out_usage", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `get_push_certificate` action + + - parameters: + - platform: Set certificate's platform. Used for creation of production & development certificates. Supported platforms: ios, macos + - development: Renew the development push certificate instead of the production one + - websitePush: Create a Website Push certificate + - generateP12: Generate a p12 file additionally to a PEM file + - activeDaysLimit: If the current certificate is active for less than this number of days, generate a new one + - force: Create a new push certificate, even if the current one is active for 30 (or PEM_ACTIVE_DAYS_LIMIT) more days + - savePrivateKey: Set to save the private RSA key + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - p12Password: The password that is used for your p12 file + - pemName: The file name of the generated .pem file + - outputPath: The path to a directory in which all certificates and private keys should be stored + - newProfile: Block that is called if there is a new profile + + Additionally to the available options, you can also specify a block that only gets executed if a new profile was created. You can use it to upload the new profile to your server. + Use it like this:| + | + ```ruby| + get_push_certificate(| + new_profile: proc do| + # your upload code| + end| + )| + ```| + >| + */ +public func pem(platform: String = "ios", + development: OptionalConfigValue = .fastlaneDefault(false), + websitePush: OptionalConfigValue = .fastlaneDefault(false), + generateP12: OptionalConfigValue = .fastlaneDefault(true), + activeDaysLimit: Int = 30, + force: OptionalConfigValue = .fastlaneDefault(false), + savePrivateKey: OptionalConfigValue = .fastlaneDefault(true), + appIdentifier: String, + username: String, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + p12Password: OptionalConfigValue = .fastlaneDefault(nil), + pemName: OptionalConfigValue = .fastlaneDefault(nil), + outputPath: String = ".", + newProfile: ((String) -> Void)? = nil) +{ + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let developmentArg = development.asRubyArgument(name: "development", type: nil) + let websitePushArg = websitePush.asRubyArgument(name: "website_push", type: nil) + let generateP12Arg = generateP12.asRubyArgument(name: "generate_p12", type: nil) + let activeDaysLimitArg = RubyCommand.Argument(name: "active_days_limit", value: activeDaysLimit, type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let savePrivateKeyArg = savePrivateKey.asRubyArgument(name: "save_private_key", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let p12PasswordArg = p12Password.asRubyArgument(name: "p12_password", type: nil) + let pemNameArg = pemName.asRubyArgument(name: "pem_name", type: nil) + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let newProfileArg = RubyCommand.Argument(name: "new_profile", value: newProfile, type: .stringClosure) + let array: [RubyCommand.Argument?] = [platformArg, + developmentArg, + websitePushArg, + generateP12Arg, + activeDaysLimitArg, + forceArg, + savePrivateKeyArg, + appIdentifierArg, + usernameArg, + teamIdArg, + teamNameArg, + p12PasswordArg, + pemNameArg, + outputPathArg, + newProfileArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "pem", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `upload_to_testflight` action + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - appIdentifier: The bundle identifier of the app to upload or manage testers (optional) + - appPlatform: The platform to use (optional) + - appleId: Apple ID property in the App Information section in App Store Connect + - ipa: Path to the ipa file to upload + - pkg: Path to your pkg file + - demoAccountRequired: Do you need a demo account when Apple does review? + - betaAppReviewInfo: Beta app review information for contact info and demo account + - localizedAppInfo: Localized beta app test info for description, feedback email, marketing url, and privacy policy + - betaAppDescription: Provide the 'Beta App Description' when uploading a new build + - betaAppFeedbackEmail: Provide the beta app email when uploading a new build + - localizedBuildInfo: Localized beta app test info for what's new + - changelog: Provide the 'What to Test' text when uploading a new build + - skipSubmission: Skip the distributing action of pilot and only upload the ipa file + - skipWaitingForBuildProcessing: If set to true, the `distribute_external` option won't work and no build will be distributed to testers. (You might want to use this option if you are using this action on CI and have to pay for 'minutes used' on your CI plan). If set to `true` and a changelog is provided, it will partially wait for the build to appear on AppStore Connect so the changelog can be set, and skip the remaining processing steps + - updateBuildInfoOnUpload: **DEPRECATED!** Update build info immediately after validation. This is deprecated and will be removed in a future release. App Store Connect no longer supports setting build info until after build processing has completed, which is when build info is updated by default + - distributeOnly: Distribute a previously uploaded build (equivalent to the `fastlane pilot distribute` command) + - usesNonExemptEncryption: Provide the 'Uses Non-Exempt Encryption' for export compliance. This is used if there is 'ITSAppUsesNonExemptEncryption' is not set in the Info.plist + - distributeExternal: Should the build be distributed to external testers? If set to true, use of `groups` option is required + - notifyExternalTesters: Should notify external testers? (Not setting a value will use App Store Connect's default which is to notify) + - appVersion: The version number of the application build to distribute. If the version number is not specified, then the most recent build uploaded to TestFlight will be distributed. If specified, the most recent build for the version number will be distributed + - buildNumber: The build number of the application build to distribute. If the build number is not specified, the most recent build is distributed + - expirePreviousBuilds: Should expire previous builds? + - firstName: The tester's first name + - lastName: The tester's last name + - email: The tester's email + - testersFilePath: Path to a CSV file of testers + - groups: Associate tester to one group or more by group name / group id. E.g. `-g "Team 1","Team 2"` This is required when `distribute_external` option is set to true or when we want to add a tester to one or more external testing groups + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - devPortalTeamId: The short ID of your team in the developer portal, if you're in multiple teams. Different from your iTC team ID! + - itcProvider: The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + - waitProcessingInterval: Interval in seconds to wait for App Store Connect processing + - waitProcessingTimeoutDuration: Timeout duration in seconds to wait for App Store Connect processing. If set, after exceeding timeout duration, this will `force stop` to wait for App Store Connect processing and exit with exception + - waitForUploadedBuild: **DEPRECATED!** No longer needed with the transition over to the App Store Connect API - Use version info from uploaded ipa file to determine what build to use for distribution. If set to false, latest processing or any latest build will be used + - rejectBuildWaitingForReview: Expire previous if it's 'waiting for review' + - submitBetaReview: Send the build for a beta review + + More details can be found on https://docs.fastlane.tools/actions/pilot/. + This integration will only do the TestFlight upload. + */ +public func pilot(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + appPlatform: OptionalConfigValue = .fastlaneDefault(nil), + appleId: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil), + pkg: OptionalConfigValue = .fastlaneDefault(nil), + demoAccountRequired: OptionalConfigValue = .fastlaneDefault(nil), + betaAppReviewInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + localizedAppInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + betaAppDescription: OptionalConfigValue = .fastlaneDefault(nil), + betaAppFeedbackEmail: OptionalConfigValue = .fastlaneDefault(nil), + localizedBuildInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + changelog: OptionalConfigValue = .fastlaneDefault(nil), + skipSubmission: OptionalConfigValue = .fastlaneDefault(false), + skipWaitingForBuildProcessing: OptionalConfigValue = .fastlaneDefault(false), + updateBuildInfoOnUpload: OptionalConfigValue = .fastlaneDefault(false), + distributeOnly: OptionalConfigValue = .fastlaneDefault(false), + usesNonExemptEncryption: OptionalConfigValue = .fastlaneDefault(false), + distributeExternal: OptionalConfigValue = .fastlaneDefault(false), + notifyExternalTesters: Any? = nil, + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + expirePreviousBuilds: OptionalConfigValue = .fastlaneDefault(false), + firstName: OptionalConfigValue = .fastlaneDefault(nil), + lastName: OptionalConfigValue = .fastlaneDefault(nil), + email: OptionalConfigValue = .fastlaneDefault(nil), + testersFilePath: String = "./testers.csv", + groups: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + teamId: Any? = nil, + teamName: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamId: OptionalConfigValue = .fastlaneDefault(nil), + itcProvider: OptionalConfigValue = .fastlaneDefault(nil), + waitProcessingInterval: Int = 30, + waitProcessingTimeoutDuration: OptionalConfigValue = .fastlaneDefault(nil), + waitForUploadedBuild: OptionalConfigValue = .fastlaneDefault(false), + rejectBuildWaitingForReview: OptionalConfigValue = .fastlaneDefault(false), + submitBetaReview: OptionalConfigValue = .fastlaneDefault(true)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let appPlatformArg = appPlatform.asRubyArgument(name: "app_platform", type: nil) + let appleIdArg = appleId.asRubyArgument(name: "apple_id", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let pkgArg = pkg.asRubyArgument(name: "pkg", type: nil) + let demoAccountRequiredArg = demoAccountRequired.asRubyArgument(name: "demo_account_required", type: nil) + let betaAppReviewInfoArg = betaAppReviewInfo.asRubyArgument(name: "beta_app_review_info", type: nil) + let localizedAppInfoArg = localizedAppInfo.asRubyArgument(name: "localized_app_info", type: nil) + let betaAppDescriptionArg = betaAppDescription.asRubyArgument(name: "beta_app_description", type: nil) + let betaAppFeedbackEmailArg = betaAppFeedbackEmail.asRubyArgument(name: "beta_app_feedback_email", type: nil) + let localizedBuildInfoArg = localizedBuildInfo.asRubyArgument(name: "localized_build_info", type: nil) + let changelogArg = changelog.asRubyArgument(name: "changelog", type: nil) + let skipSubmissionArg = skipSubmission.asRubyArgument(name: "skip_submission", type: nil) + let skipWaitingForBuildProcessingArg = skipWaitingForBuildProcessing.asRubyArgument(name: "skip_waiting_for_build_processing", type: nil) + let updateBuildInfoOnUploadArg = updateBuildInfoOnUpload.asRubyArgument(name: "update_build_info_on_upload", type: nil) + let distributeOnlyArg = distributeOnly.asRubyArgument(name: "distribute_only", type: nil) + let usesNonExemptEncryptionArg = usesNonExemptEncryption.asRubyArgument(name: "uses_non_exempt_encryption", type: nil) + let distributeExternalArg = distributeExternal.asRubyArgument(name: "distribute_external", type: nil) + let notifyExternalTestersArg = RubyCommand.Argument(name: "notify_external_testers", value: notifyExternalTesters, type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let expirePreviousBuildsArg = expirePreviousBuilds.asRubyArgument(name: "expire_previous_builds", type: nil) + let firstNameArg = firstName.asRubyArgument(name: "first_name", type: nil) + let lastNameArg = lastName.asRubyArgument(name: "last_name", type: nil) + let emailArg = email.asRubyArgument(name: "email", type: nil) + let testersFilePathArg = RubyCommand.Argument(name: "testers_file_path", value: testersFilePath, type: nil) + let groupsArg = groups.asRubyArgument(name: "groups", type: nil) + let teamIdArg = RubyCommand.Argument(name: "team_id", value: teamId, type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let devPortalTeamIdArg = devPortalTeamId.asRubyArgument(name: "dev_portal_team_id", type: nil) + let itcProviderArg = itcProvider.asRubyArgument(name: "itc_provider", type: nil) + let waitProcessingIntervalArg = RubyCommand.Argument(name: "wait_processing_interval", value: waitProcessingInterval, type: nil) + let waitProcessingTimeoutDurationArg = waitProcessingTimeoutDuration.asRubyArgument(name: "wait_processing_timeout_duration", type: nil) + let waitForUploadedBuildArg = waitForUploadedBuild.asRubyArgument(name: "wait_for_uploaded_build", type: nil) + let rejectBuildWaitingForReviewArg = rejectBuildWaitingForReview.asRubyArgument(name: "reject_build_waiting_for_review", type: nil) + let submitBetaReviewArg = submitBetaReview.asRubyArgument(name: "submit_beta_review", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + appPlatformArg, + appleIdArg, + ipaArg, + pkgArg, + demoAccountRequiredArg, + betaAppReviewInfoArg, + localizedAppInfoArg, + betaAppDescriptionArg, + betaAppFeedbackEmailArg, + localizedBuildInfoArg, + changelogArg, + skipSubmissionArg, + skipWaitingForBuildProcessingArg, + updateBuildInfoOnUploadArg, + distributeOnlyArg, + usesNonExemptEncryptionArg, + distributeExternalArg, + notifyExternalTestersArg, + appVersionArg, + buildNumberArg, + expirePreviousBuildsArg, + firstNameArg, + lastNameArg, + emailArg, + testersFilePathArg, + groupsArg, + teamIdArg, + teamNameArg, + devPortalTeamIdArg, + itcProviderArg, + waitProcessingIntervalArg, + waitProcessingTimeoutDurationArg, + waitForUploadedBuildArg, + rejectBuildWaitingForReviewArg, + submitBetaReviewArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "pilot", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + No description provided + + - parameters: + - outputPath: + - templatePath: + - cachePath: + */ +public func pluginScores(outputPath: String, + templatePath: String, + cachePath: String) +{ + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let templatePathArg = RubyCommand.Argument(name: "template_path", value: templatePath, type: nil) + let cachePathArg = RubyCommand.Argument(name: "cache_path", value: cachePath, type: nil) + let array: [RubyCommand.Argument?] = [outputPathArg, + templatePathArg, + cachePathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "plugin_scores", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Pod lib lint + + - parameters: + - useBundleExec: Use bundle exec when there is a Gemfile presented + - podspec: Path of spec to lint + - verbose: Allow output detail in console + - allowWarnings: Allow warnings during pod lint + - sources: The sources of repos you want the pod spec to lint with, separated by commas + - subspec: A specific subspec to lint instead of the entire spec + - includePodspecs: A Glob of additional ancillary podspecs which are used for linting via :path (available since cocoapods >= 1.7) + - externalPodspecs: A Glob of additional ancillary podspecs which are used for linting via :podspec. If there are --include-podspecs, then these are removed from them (available since cocoapods >= 1.7) + - swiftVersion: The SWIFT_VERSION that should be used to lint the spec. This takes precedence over a .swift-version file + - useLibraries: Lint uses static libraries to install the spec + - useModularHeaders: Lint using modular libraries (available since cocoapods >= 1.6) + - failFast: Lint stops on the first failing platform or subspec + - private: Lint skips checks that apply only to public specs + - quick: Lint skips checks that would require to download and build the spec + - noClean: Lint leaves the build directory intact for inspection + - noSubspecs: Lint skips validation of subspecs + - platforms: Lint against specific platforms (defaults to all platforms supported by the podspec). Multiple platforms must be comma-delimited (available since cocoapods >= 1.6) + - skipImportValidation: Lint skips validating that the pod can be imported (available since cocoapods >= 1.3) + - skipTests: Lint skips building and running tests during validation (available since cocoapods >= 1.3) + - analyze: Validate with the Xcode Static Analysis tool (available since cocoapods >= 1.6.1) + + Test the syntax of your Podfile by linting the pod against the files of its directory + */ +public func podLibLint(useBundleExec: OptionalConfigValue = .fastlaneDefault(true), + podspec: OptionalConfigValue = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(nil), + allowWarnings: OptionalConfigValue = .fastlaneDefault(nil), + sources: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + subspec: OptionalConfigValue = .fastlaneDefault(nil), + includePodspecs: OptionalConfigValue = .fastlaneDefault(nil), + externalPodspecs: OptionalConfigValue = .fastlaneDefault(nil), + swiftVersion: OptionalConfigValue = .fastlaneDefault(nil), + useLibraries: OptionalConfigValue = .fastlaneDefault(false), + useModularHeaders: OptionalConfigValue = .fastlaneDefault(false), + failFast: OptionalConfigValue = .fastlaneDefault(false), + private: OptionalConfigValue = .fastlaneDefault(false), + quick: OptionalConfigValue = .fastlaneDefault(false), + noClean: OptionalConfigValue = .fastlaneDefault(false), + noSubspecs: OptionalConfigValue = .fastlaneDefault(false), + platforms: OptionalConfigValue = .fastlaneDefault(nil), + skipImportValidation: OptionalConfigValue = .fastlaneDefault(false), + skipTests: OptionalConfigValue = .fastlaneDefault(false), + analyze: OptionalConfigValue = .fastlaneDefault(false)) +{ + let useBundleExecArg = useBundleExec.asRubyArgument(name: "use_bundle_exec", type: nil) + let podspecArg = podspec.asRubyArgument(name: "podspec", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let allowWarningsArg = allowWarnings.asRubyArgument(name: "allow_warnings", type: nil) + let sourcesArg = sources.asRubyArgument(name: "sources", type: nil) + let subspecArg = subspec.asRubyArgument(name: "subspec", type: nil) + let includePodspecsArg = includePodspecs.asRubyArgument(name: "include_podspecs", type: nil) + let externalPodspecsArg = externalPodspecs.asRubyArgument(name: "external_podspecs", type: nil) + let swiftVersionArg = swiftVersion.asRubyArgument(name: "swift_version", type: nil) + let useLibrariesArg = useLibraries.asRubyArgument(name: "use_libraries", type: nil) + let useModularHeadersArg = useModularHeaders.asRubyArgument(name: "use_modular_headers", type: nil) + let failFastArg = failFast.asRubyArgument(name: "fail_fast", type: nil) + let privateArg = `private`.asRubyArgument(name: "private", type: nil) + let quickArg = quick.asRubyArgument(name: "quick", type: nil) + let noCleanArg = noClean.asRubyArgument(name: "no_clean", type: nil) + let noSubspecsArg = noSubspecs.asRubyArgument(name: "no_subspecs", type: nil) + let platformsArg = platforms.asRubyArgument(name: "platforms", type: nil) + let skipImportValidationArg = skipImportValidation.asRubyArgument(name: "skip_import_validation", type: nil) + let skipTestsArg = skipTests.asRubyArgument(name: "skip_tests", type: nil) + let analyzeArg = analyze.asRubyArgument(name: "analyze", type: nil) + let array: [RubyCommand.Argument?] = [useBundleExecArg, + podspecArg, + verboseArg, + allowWarningsArg, + sourcesArg, + subspecArg, + includePodspecsArg, + externalPodspecsArg, + swiftVersionArg, + useLibrariesArg, + useModularHeadersArg, + failFastArg, + privateArg, + quickArg, + noCleanArg, + noSubspecsArg, + platformsArg, + skipImportValidationArg, + skipTestsArg, + analyzeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "pod_lib_lint", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Push a Podspec to Trunk or a private repository + + - parameters: + - useBundleExec: Use bundle exec when there is a Gemfile presented + - path: The Podspec you want to push + - repo: The repo you want to push. Pushes to Trunk by default + - allowWarnings: Allow warnings during pod push + - useLibraries: Allow lint to use static libraries to install the spec + - sources: The sources of repos you want the pod spec to lint with, separated by commas + - swiftVersion: The SWIFT_VERSION that should be used to lint the spec. This takes precedence over a .swift-version file + - skipImportValidation: Lint skips validating that the pod can be imported + - skipTests: Lint skips building and running tests during validation + - useJson: Convert the podspec to JSON before pushing it to the repo + - verbose: Show more debugging information + - useModularHeaders: Use modular headers option during validation + - synchronous: If validation depends on other recently pushed pods, synchronize + - noOverwrite: Disallow pushing that would overwrite an existing spec + - localOnly: Does not perform the step of pushing REPO to its remote + */ +public func podPush(useBundleExec: OptionalConfigValue = .fastlaneDefault(false), + path: OptionalConfigValue = .fastlaneDefault(nil), + repo: OptionalConfigValue = .fastlaneDefault(nil), + allowWarnings: OptionalConfigValue = .fastlaneDefault(nil), + useLibraries: OptionalConfigValue = .fastlaneDefault(nil), + sources: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + swiftVersion: OptionalConfigValue = .fastlaneDefault(nil), + skipImportValidation: OptionalConfigValue = .fastlaneDefault(nil), + skipTests: OptionalConfigValue = .fastlaneDefault(nil), + useJson: OptionalConfigValue = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(false), + useModularHeaders: OptionalConfigValue = .fastlaneDefault(nil), + synchronous: OptionalConfigValue = .fastlaneDefault(nil), + noOverwrite: OptionalConfigValue = .fastlaneDefault(nil), + localOnly: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let useBundleExecArg = useBundleExec.asRubyArgument(name: "use_bundle_exec", type: nil) + let pathArg = path.asRubyArgument(name: "path", type: nil) + let repoArg = repo.asRubyArgument(name: "repo", type: nil) + let allowWarningsArg = allowWarnings.asRubyArgument(name: "allow_warnings", type: nil) + let useLibrariesArg = useLibraries.asRubyArgument(name: "use_libraries", type: nil) + let sourcesArg = sources.asRubyArgument(name: "sources", type: nil) + let swiftVersionArg = swiftVersion.asRubyArgument(name: "swift_version", type: nil) + let skipImportValidationArg = skipImportValidation.asRubyArgument(name: "skip_import_validation", type: nil) + let skipTestsArg = skipTests.asRubyArgument(name: "skip_tests", type: nil) + let useJsonArg = useJson.asRubyArgument(name: "use_json", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let useModularHeadersArg = useModularHeaders.asRubyArgument(name: "use_modular_headers", type: nil) + let synchronousArg = synchronous.asRubyArgument(name: "synchronous", type: nil) + let noOverwriteArg = noOverwrite.asRubyArgument(name: "no_overwrite", type: nil) + let localOnlyArg = localOnly.asRubyArgument(name: "local_only", type: nil) + let array: [RubyCommand.Argument?] = [useBundleExecArg, + pathArg, + repoArg, + allowWarningsArg, + useLibrariesArg, + sourcesArg, + swiftVersionArg, + skipImportValidationArg, + skipTestsArg, + useJsonArg, + verboseArg, + useModularHeadersArg, + synchronousArg, + noOverwriteArg, + localOnlyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "pod_push", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Creates or updates an item within your Podio app + + - parameters: + - clientId: Client ID for Podio API (see https://developers.podio.com/api-key) + - clientSecret: Client secret for Podio API (see https://developers.podio.com/api-key) + - appId: App ID of the app you intend to authenticate with (see https://developers.podio.com/authentication/app_auth) + - appToken: App token of the app you intend to authenticate with (see https://developers.podio.com/authentication/app_auth) + - identifyingField: String specifying the field key used for identification of an item + - identifyingValue: String uniquely specifying an item within the app + - otherFields: Dictionary of your app fields. Podio supports several field types, see https://developers.podio.com/doc/items + + Use this action to create or update an item within your Podio app (see [https://help.podio.com/hc/en-us/articles/201019278-Creating-apps-](https://help.podio.com/hc/en-us/articles/201019278-Creating-apps-)). + Pass in dictionary with field keys and their values. + Field key is located under `Modify app` -> `Advanced` -> `Developer` -> `External ID` (see [https://developers.podio.com/examples/items](https://developers.podio.com/examples/items)). + */ +public func podioItem(clientId: String, + clientSecret: String, + appId: String, + appToken: String, + identifyingField: String, + identifyingValue: String, + otherFields: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil)) +{ + let clientIdArg = RubyCommand.Argument(name: "client_id", value: clientId, type: nil) + let clientSecretArg = RubyCommand.Argument(name: "client_secret", value: clientSecret, type: nil) + let appIdArg = RubyCommand.Argument(name: "app_id", value: appId, type: nil) + let appTokenArg = RubyCommand.Argument(name: "app_token", value: appToken, type: nil) + let identifyingFieldArg = RubyCommand.Argument(name: "identifying_field", value: identifyingField, type: nil) + let identifyingValueArg = RubyCommand.Argument(name: "identifying_value", value: identifyingValue, type: nil) + let otherFieldsArg = otherFields.asRubyArgument(name: "other_fields", type: nil) + let array: [RubyCommand.Argument?] = [clientIdArg, + clientSecretArg, + appIdArg, + appTokenArg, + identifyingFieldArg, + identifyingValueArg, + otherFieldsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "podio_item", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `check_app_store_metadata` action + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - platform: The platform to use (optional) + - defaultRuleLevel: The default rule level unless otherwise configured + - includeInAppPurchases: Should check in-app purchases? + - useLive: Should force check live app? + - freeStuffInIap: using text indicating that your IAP is free + + - returns: true if precheck passes, else, false + + More information: https://fastlane.tools/precheck + */ +@discardableResult public func precheck(apiKeyPath: OptionalConfigValue = .fastlaneDefault(precheckfile.apiKeyPath), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(precheckfile.apiKey), + appIdentifier: String = precheckfile.appIdentifier, + username: OptionalConfigValue = .fastlaneDefault(precheckfile.username), + teamId: OptionalConfigValue = .fastlaneDefault(precheckfile.teamId), + teamName: OptionalConfigValue = .fastlaneDefault(precheckfile.teamName), + platform: String = precheckfile.platform, + defaultRuleLevel: Any = precheckfile.defaultRuleLevel, + includeInAppPurchases: OptionalConfigValue = .fastlaneDefault(precheckfile.includeInAppPurchases), + useLive: OptionalConfigValue = .fastlaneDefault(precheckfile.useLive), + freeStuffInIap: Any? = precheckfile.freeStuffInIap) -> Bool +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let defaultRuleLevelArg = RubyCommand.Argument(name: "default_rule_level", value: defaultRuleLevel, type: nil) + let includeInAppPurchasesArg = includeInAppPurchases.asRubyArgument(name: "include_in_app_purchases", type: nil) + let useLiveArg = useLive.asRubyArgument(name: "use_live", type: nil) + let freeStuffInIapArg = RubyCommand.Argument(name: "free_stuff_in_iap", value: freeStuffInIap, type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + appIdentifierArg, + usernameArg, + teamIdArg, + teamNameArg, + platformArg, + defaultRuleLevelArg, + includeInAppPurchasesArg, + useLiveArg, + freeStuffInIapArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "precheck", className: nil, args: args) + return parseBool(fromString: runner.executeCommand(command)) +} + +/** + Alias for the `puts` action + + - parameter message: Message to be printed out + */ +public func println(message: OptionalConfigValue = .fastlaneDefault(nil)) { + let messageArg = message.asRubyArgument(name: "message", type: nil) + let array: [RubyCommand.Argument?] = [messageArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "println", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `create_app_online` action + + - parameters: + - username: Your Apple ID Username + - appIdentifier: App Identifier (Bundle ID, e.g. com.krausefx.app) + - bundleIdentifierSuffix: App Identifier Suffix (Ignored if App Identifier does not end with .*) + - appName: App Name + - appVersion: Initial version number (e.g. '1.0') + - sku: SKU Number (e.g. '1234') + - platform: The platform to use (optional) + - platforms: The platforms to use (optional) + - language: Primary Language (e.g. 'en-US', 'fr-FR') + - companyName: The name of your company. It's used to set company name on App Store Connect team's app pages. Only required if it's the first app you create + - skipItc: Skip the creation of the app on App Store Connect + - itcUsers: Array of App Store Connect users. If provided, you can limit access to this newly created app for users with the App Manager, Developer, Marketer or Sales roles + - enabledFeatures: **DEPRECATED!** Please use `enable_services` instead - Array with Spaceship App Services + - enableServices: Array with Spaceship App Services (e.g. access_wifi: (on|off), app_attest: (on|off), app_group: (on|off), apple_pay: (on|off), associated_domains: (on|off), auto_fill_credential: (on|off), class_kit: (on|off), icloud: (legacy|cloudkit), custom_network_protocol: (on|off), data_protection: (complete|unlessopen|untilfirstauth), extended_virtual_address_space: (on|off), family_controls: (on|off), file_provider_testing_mode: (on|off), fonts: (on|off), game_center: (ios|mac), health_kit: (on|off), hls_interstitial_preview: (on|off), home_kit: (on|off), hotspot: (on|off), in_app_purchase: (on|off), inter_app_audio: (on|off), low_latency_hls: (on|off), managed_associated_domains: (on|off), maps: (on|off), multipath: (on|off), network_extension: (on|off), nfc_tag_reading: (on|off), personal_vpn: (on|off), passbook: (on|off), push_notification: (on|off), sign_in_with_apple: (on), siri_kit: (on|off), system_extension: (on|off), user_management: (on|off), vpn_configuration: (on|off), wallet: (on|off), wireless_accessory: (on|off), car_play_audio_app: (on|off), car_play_messaging_app: (on|off), car_play_navigation_app: (on|off), car_play_voip_calling_app: (on|off), critical_alerts: (on|off), hotspot_helper: (on|off), driver_kit: (on|off), driver_kit_endpoint_security: (on|off), driver_kit_family_hid_device: (on|off), driver_kit_family_networking: (on|off), driver_kit_family_serial: (on|off), driver_kit_hid_event_service: (on|off), driver_kit_transport_hid: (on|off), multitasking_camera_access: (on|off), sf_universal_link_api: (on|off), vp9_decoder: (on|off), music_kit: (on|off), shazam_kit: (on|off), communication_notifications: (on|off), group_activities: (on|off), health_kit_estimate_recalibration: (on|off), time_sensitive_notifications: (on|off)) + - skipDevcenter: Skip the creation of the app on the Apple Developer Portal + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - itcTeamId: The ID of your App Store Connect team if you're in multiple teams + - itcTeamName: The name of your App Store Connect team if you're in multiple teams + + Create new apps on App Store Connect and Apple Developer Portal via _produce_. + If the app already exists, `create_app_online` will not do anything. + For more information about _produce_, visit its documentation page: [https://docs.fastlane.tools/actions/produce/](https://docs.fastlane.tools/actions/produce/). + */ +public func produce(username: String, + appIdentifier: String, + bundleIdentifierSuffix: OptionalConfigValue = .fastlaneDefault(nil), + appName: String, + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + sku: String, + platform: String = "ios", + platforms: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + language: String = "English", + companyName: OptionalConfigValue = .fastlaneDefault(nil), + skipItc: OptionalConfigValue = .fastlaneDefault(false), + itcUsers: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + enabledFeatures: [String: Any] = [:], + enableServices: [String: Any] = [:], + skipDevcenter: OptionalConfigValue = .fastlaneDefault(false), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + itcTeamId: Any? = nil, + itcTeamName: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let bundleIdentifierSuffixArg = bundleIdentifierSuffix.asRubyArgument(name: "bundle_identifier_suffix", type: nil) + let appNameArg = RubyCommand.Argument(name: "app_name", value: appName, type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let skuArg = RubyCommand.Argument(name: "sku", value: sku, type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let platformsArg = platforms.asRubyArgument(name: "platforms", type: nil) + let languageArg = RubyCommand.Argument(name: "language", value: language, type: nil) + let companyNameArg = companyName.asRubyArgument(name: "company_name", type: nil) + let skipItcArg = skipItc.asRubyArgument(name: "skip_itc", type: nil) + let itcUsersArg = itcUsers.asRubyArgument(name: "itc_users", type: nil) + let enabledFeaturesArg = RubyCommand.Argument(name: "enabled_features", value: enabledFeatures, type: nil) + let enableServicesArg = RubyCommand.Argument(name: "enable_services", value: enableServices, type: nil) + let skipDevcenterArg = skipDevcenter.asRubyArgument(name: "skip_devcenter", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let itcTeamIdArg = RubyCommand.Argument(name: "itc_team_id", value: itcTeamId, type: nil) + let itcTeamNameArg = itcTeamName.asRubyArgument(name: "itc_team_name", type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + appIdentifierArg, + bundleIdentifierSuffixArg, + appNameArg, + appVersionArg, + skuArg, + platformArg, + platformsArg, + languageArg, + companyNameArg, + skipItcArg, + itcUsersArg, + enabledFeaturesArg, + enableServicesArg, + skipDevcenterArg, + teamIdArg, + teamNameArg, + itcTeamIdArg, + itcTeamNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "produce", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Ask the user for a value or for confirmation + + - parameters: + - text: The text that will be displayed to the user + - ciInput: The default text that will be used when being executed on a CI service + - boolean: Is that a boolean question (yes/no)? This will add (y/n) at the end + - secureText: Is that a secure text (yes/no)? + - multiLineEndKeyword: Enable multi-line inputs by providing an end text (e.g. 'END') which will stop the user input + + You can use `prompt` to ask the user for a value or to just let the user confirm the next step. + When this is executed on a CI service, the passed `ci_input` value will be returned. + This action also supports multi-line inputs using the `multi_line_end_keyword` option. + */ +@discardableResult public func prompt(text: String = "Please enter some text: ", + ciInput: String = "", + boolean: OptionalConfigValue = .fastlaneDefault(false), + secureText: OptionalConfigValue = .fastlaneDefault(false), + multiLineEndKeyword: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let textArg = RubyCommand.Argument(name: "text", value: text, type: nil) + let ciInputArg = RubyCommand.Argument(name: "ci_input", value: ciInput, type: nil) + let booleanArg = boolean.asRubyArgument(name: "boolean", type: nil) + let secureTextArg = secureText.asRubyArgument(name: "secure_text", type: nil) + let multiLineEndKeywordArg = multiLineEndKeyword.asRubyArgument(name: "multi_line_end_keyword", type: nil) + let array: [RubyCommand.Argument?] = [textArg, + ciInputArg, + booleanArg, + secureTextArg, + multiLineEndKeywordArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "prompt", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Push local tags to the remote - this will only push tags + + - parameters: + - force: Force push to remote + - remote: The remote to push tags to + - tag: The tag to push to remote + + If you only want to push the tags and nothing else, you can use the `push_git_tags` action + */ +public func pushGitTags(force: OptionalConfigValue = .fastlaneDefault(false), + remote: String = "origin", + tag: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let forceArg = force.asRubyArgument(name: "force", type: nil) + let remoteArg = RubyCommand.Argument(name: "remote", value: remote, type: nil) + let tagArg = tag.asRubyArgument(name: "tag", type: nil) + let array: [RubyCommand.Argument?] = [forceArg, + remoteArg, + tagArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "push_git_tags", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Push local changes to the remote branch + + - parameters: + - localBranch: The local branch to push from. Defaults to the current branch + - remoteBranch: The remote branch to push to. Defaults to the local branch + - force: Force push to remote + - forceWithLease: Force push with lease to remote + - tags: Whether tags are pushed to remote + - remote: The remote to push to + - noVerify: Whether or not to use --no-verify + - setUpstream: Whether or not to use --set-upstream + - pushOptions: Array of strings to be passed using the '--push-option' option + + Lets you push your local commits to a remote git repo. Useful if you make local changes such as adding a version bump commit (using `commit_version_bump`) or a git tag (using 'add_git_tag') on a CI server, and you want to push those changes back to your canonical/main repo. + If this is a new branch, use the `set_upstream` option to set the remote branch as upstream. + */ +public func pushToGitRemote(localBranch: OptionalConfigValue = .fastlaneDefault(nil), + remoteBranch: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + forceWithLease: OptionalConfigValue = .fastlaneDefault(false), + tags: OptionalConfigValue = .fastlaneDefault(true), + remote: String = "origin", + noVerify: OptionalConfigValue = .fastlaneDefault(false), + setUpstream: OptionalConfigValue = .fastlaneDefault(false), + pushOptions: [String] = []) +{ + let localBranchArg = localBranch.asRubyArgument(name: "local_branch", type: nil) + let remoteBranchArg = remoteBranch.asRubyArgument(name: "remote_branch", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let forceWithLeaseArg = forceWithLease.asRubyArgument(name: "force_with_lease", type: nil) + let tagsArg = tags.asRubyArgument(name: "tags", type: nil) + let remoteArg = RubyCommand.Argument(name: "remote", value: remote, type: nil) + let noVerifyArg = noVerify.asRubyArgument(name: "no_verify", type: nil) + let setUpstreamArg = setUpstream.asRubyArgument(name: "set_upstream", type: nil) + let pushOptionsArg = RubyCommand.Argument(name: "push_options", value: pushOptions, type: nil) + let array: [RubyCommand.Argument?] = [localBranchArg, + remoteBranchArg, + forceArg, + forceWithLeaseArg, + tagsArg, + remoteArg, + noVerifyArg, + setUpstreamArg, + pushOptionsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "push_to_git_remote", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Prints out the given text + + - parameter message: Message to be printed out + */ +public func puts(message: OptionalConfigValue = .fastlaneDefault(nil)) { + let messageArg = message.asRubyArgument(name: "message", type: nil) + let array: [RubyCommand.Argument?] = [messageArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "puts", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Loads a CocoaPods spec as JSON + + - parameter path: Path to the podspec to be read + + This can be used for only specifying a version string in your podspec - and during your release process you'd read it from the podspec by running `version = read_podspec['version']` at the beginning of your lane. + Loads the specified (or the first found) podspec in the folder as JSON, so that you can inspect its `version`, `files` etc. + This can be useful when basing your release process on the version string only stored in one place - in the podspec. + As one of the first steps you'd read the podspec and its version and the rest of the workflow can use that version string (when e.g. creating a new git tag or a GitHub Release). + */ +@discardableResult public func readPodspec(path: String) -> [String: Any] { + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "read_podspec", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Recreate not shared Xcode project schemes + + - parameter project: The Xcode project + */ +public func recreateSchemes(project: String) { + let projectArg = RubyCommand.Argument(name: "project", value: project, type: nil) + let array: [RubyCommand.Argument?] = [projectArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "recreate_schemes", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Registers a new device to the Apple Dev Portal + + - parameters: + - name: Provide the name of the device to register as + - platform: Provide the platform of the device to register as (ios, mac) + - udid: Provide the UDID of the device to register as + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - username: Optional: Your Apple ID + + This will register an iOS device with the Developer Portal so that you can include it in your provisioning profiles. + This is an optimistic action, in that it will only ever add a device to the member center. If the device has already been registered within the member center, it will be left alone in the member center. + The action will connect to the Apple Developer Portal using the username you specified in your `Appfile` with `apple_id`, but you can override it using the `:username` option. + */ +@discardableResult public func registerDevice(name: String, + platform: String = "ios", + udid: String, + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let nameArg = RubyCommand.Argument(name: "name", value: name, type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let udidArg = RubyCommand.Argument(name: "udid", value: udid, type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let array: [RubyCommand.Argument?] = [nameArg, + platformArg, + udidArg, + apiKeyPathArg, + apiKeyArg, + teamIdArg, + teamNameArg, + usernameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "register_device", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Registers new devices to the Apple Dev Portal + + - parameters: + - devices: A hash of devices, with the name as key and the UDID as value + - devicesFile: Provide a path to a file with the devices to register. For the format of the file see the examples + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - username: Optional: Your Apple ID + - platform: The platform to use (optional) + + This will register iOS/Mac devices with the Developer Portal so that you can include them in your provisioning profiles. + This is an optimistic action, in that it will only ever add new devices to the member center, and never remove devices. If a device which has already been registered within the member center is not passed to this action, it will be left alone in the member center and continue to work. + The action will connect to the Apple Developer Portal using the username you specified in your `Appfile` with `apple_id`, but you can override it using the `username` option, or by setting the env variable `ENV['DELIVER_USER']`. + */ +public func registerDevices(devices: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + devicesFile: OptionalConfigValue = .fastlaneDefault(nil), + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios") +{ + let devicesArg = devices.asRubyArgument(name: "devices", type: nil) + let devicesFileArg = devicesFile.asRubyArgument(name: "devices_file", type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let array: [RubyCommand.Argument?] = [devicesArg, + devicesFileArg, + apiKeyPathArg, + apiKeyArg, + teamIdArg, + teamNameArg, + usernameArg, + platformArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "register_devices", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Resets git repo to a clean state by discarding uncommitted changes + + - parameters: + - files: Array of files the changes should be discarded. If not given, all files will be discarded + - force: Skip verifying of previously clean state of repo. Only recommended in combination with `files` option + - skipClean: Skip 'git clean' to avoid removing untracked files like `.env` + - disregardGitignore: Setting this to true will clean the whole repository, ignoring anything in your local .gitignore. Set this to true if you want the equivalent of a fresh clone, and for all untracked and ignore files to also be removed + - exclude: You can pass a string, or array of, file pattern(s) here which you want to have survive the cleaning process, and remain on disk, e.g. to leave the `artifacts` directory you would specify `exclude: 'artifacts'`. Make sure this pattern is also in your gitignore! See the gitignore documentation for info on patterns + + This action will reset your git repo to a clean state, discarding any uncommitted and untracked changes. Useful in case you need to revert the repo back to a clean state, e.g. after running _fastlane_. + Untracked files like `.env` will also be deleted, unless `:skip_clean` is true. + It's a pretty drastic action so it comes with a sort of safety latch. It will only proceed with the reset if this condition is met:| + | + >- You have called the `ensure_git_status_clean` action prior to calling this action. This ensures that your repo started off in a clean state, so the only things that will get destroyed by this action are files that are created as a byproduct of the fastlane run.| + >| + */ +public func resetGitRepo(files: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + skipClean: OptionalConfigValue = .fastlaneDefault(false), + disregardGitignore: OptionalConfigValue = .fastlaneDefault(true), + exclude: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let filesArg = files.asRubyArgument(name: "files", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let skipCleanArg = skipClean.asRubyArgument(name: "skip_clean", type: nil) + let disregardGitignoreArg = disregardGitignore.asRubyArgument(name: "disregard_gitignore", type: nil) + let excludeArg = exclude.asRubyArgument(name: "exclude", type: nil) + let array: [RubyCommand.Argument?] = [filesArg, + forceArg, + skipCleanArg, + disregardGitignoreArg, + excludeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "reset_git_repo", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Shutdown and reset running simulators + + - parameters: + - ios: **DEPRECATED!** Use `:os_versions` instead - Which OS versions of Simulators you want to reset content and settings, this does not remove/recreate the simulators + - osVersions: Which OS versions of Simulators you want to reset content and settings, this does not remove/recreate the simulators + */ +public func resetSimulatorContents(ios: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + osVersions: OptionalConfigValue<[String]?> = .fastlaneDefault(nil)) +{ + let iosArg = ios.asRubyArgument(name: "ios", type: nil) + let osVersionsArg = osVersions.asRubyArgument(name: "os_versions", type: nil) + let array: [RubyCommand.Argument?] = [iosArg, + osVersionsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "reset_simulator_contents", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Codesign an existing ipa file + + - parameters: + - ipa: Path to the ipa file to resign. Optional if you use the _gym_ or _xcodebuild_ action + - signingIdentity: Code signing identity to use. e.g. `iPhone Distribution: Luka Mirosevic (0123456789)` + - entitlements: Path to the entitlement file to use, e.g. `myApp/MyApp.entitlements` + - provisioningProfile: Path to your provisioning_profile. Optional if you use _sigh_ + - version: Version number to force resigned ipa to use. Updates both `CFBundleShortVersionString` and `CFBundleVersion` values in `Info.plist`. Applies for main app and all nested apps or extensions + - displayName: Display name to force resigned ipa to use + - shortVersion: Short version string to force resigned ipa to use (`CFBundleShortVersionString`) + - bundleVersion: Bundle version to force resigned ipa to use (`CFBundleVersion`) + - bundleId: Set new bundle ID during resign (`CFBundleIdentifier`) + - useAppEntitlements: Extract app bundle codesigning entitlements and combine with entitlements from new provisioning profile + - keychainPath: Provide a path to a keychain file that should be used by `/usr/bin/codesign` + */ +public func resign(ipa: String, + signingIdentity: String, + entitlements: OptionalConfigValue = .fastlaneDefault(nil), + provisioningProfile: String, + version: OptionalConfigValue = .fastlaneDefault(nil), + displayName: OptionalConfigValue = .fastlaneDefault(nil), + shortVersion: OptionalConfigValue = .fastlaneDefault(nil), + bundleVersion: OptionalConfigValue = .fastlaneDefault(nil), + bundleId: OptionalConfigValue = .fastlaneDefault(nil), + useAppEntitlements: OptionalConfigValue = .fastlaneDefault(nil), + keychainPath: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let ipaArg = RubyCommand.Argument(name: "ipa", value: ipa, type: nil) + let signingIdentityArg = RubyCommand.Argument(name: "signing_identity", value: signingIdentity, type: nil) + let entitlementsArg = entitlements.asRubyArgument(name: "entitlements", type: nil) + let provisioningProfileArg = RubyCommand.Argument(name: "provisioning_profile", value: provisioningProfile, type: nil) + let versionArg = version.asRubyArgument(name: "version", type: nil) + let displayNameArg = displayName.asRubyArgument(name: "display_name", type: nil) + let shortVersionArg = shortVersion.asRubyArgument(name: "short_version", type: nil) + let bundleVersionArg = bundleVersion.asRubyArgument(name: "bundle_version", type: nil) + let bundleIdArg = bundleId.asRubyArgument(name: "bundle_id", type: nil) + let useAppEntitlementsArg = useAppEntitlements.asRubyArgument(name: "use_app_entitlements", type: nil) + let keychainPathArg = keychainPath.asRubyArgument(name: "keychain_path", type: nil) + let array: [RubyCommand.Argument?] = [ipaArg, + signingIdentityArg, + entitlementsArg, + provisioningProfileArg, + versionArg, + displayNameArg, + shortVersionArg, + bundleVersionArg, + bundleIdArg, + useAppEntitlementsArg, + keychainPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "resign", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action restore your file that was backuped with the `backup_file` action + + - parameter path: Original file name you want to restore + */ +public func restoreFile(path: String) { + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let array: [RubyCommand.Argument?] = [pathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "restore_file", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Outputs ascii-art for a rocket 🚀 + + Print an ascii Rocket :rocket:. Useful after using _crashlytics_ or _pilot_ to indicate that your new build has been shipped to outer-space. + */ +@discardableResult public func rocket() -> String { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "rocket", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Run tests using rspec + */ +public func rspec() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "rspec", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Rsync files from :source to :destination + + - parameters: + - extra: Port + - source: source file/folder + - destination: destination file/folder + + A wrapper around `rsync`, which is a tool that lets you synchronize files, including permissions and so on. For a more detailed information about `rsync`, please see [rsync(1) man page](https://linux.die.net/man/1/rsync). + */ +public func rsync(extra: String = "-av", + source: String, + destination: String) +{ + let extraArg = RubyCommand.Argument(name: "extra", value: extra, type: nil) + let sourceArg = RubyCommand.Argument(name: "source", value: source, type: nil) + let destinationArg = RubyCommand.Argument(name: "destination", value: destination, type: nil) + let array: [RubyCommand.Argument?] = [extraArg, + sourceArg, + destinationArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "rsync", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs the code style checks + */ +public func rubocop() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "rubocop", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Verifies the minimum ruby version required + + Add this to your `Fastfile` to require a certain version of _ruby_. + Put it at the top of your `Fastfile` to ensure that _fastlane_ is executed appropriately. + */ +public func rubyVersion() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "ruby_version", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Easily run tests of your iOS app (via _scan_) + + - parameters: + - workspace: Path to the workspace file + - project: Path to the project file + - packagePath: Path to the Swift Package + - scheme: The project's scheme. Make sure it's marked as `Shared` + - device: The name of the simulator type you want to run tests on (e.g. 'iPhone 6' or 'iPhone SE (2nd generation) (14.5)') + - devices: Array of devices to run the tests on (e.g. ['iPhone 6', 'iPad Air', 'iPhone SE (2nd generation) (14.5)']) + - skipDetectDevices: Should skip auto detecting of devices if none were specified + - ensureDevicesFound: Should fail if devices not found + - forceQuitSimulator: Enabling this option will automatically killall Simulator processes before the run + - resetSimulator: Enabling this option will automatically erase the simulator before running the application + - disableSlideToType: Enabling this option will disable the simulator from showing the 'Slide to type' prompt + - prelaunchSimulator: Enabling this option will launch the first simulator prior to calling any xcodebuild command + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - appIdentifier: The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + - onlyTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to run + - skipTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to skip + - testplan: The testplan associated with the scheme that should be used for testing + - onlyTestConfigurations: Array of strings matching test plan configurations to run + - skipTestConfigurations: Array of strings matching test plan configurations to skip + - xctestrun: Run tests using the provided `.xctestrun` file + - toolchain: The toolchain that should be used for building the application (e.g. `com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a`) + - clean: Should the project be cleaned before building it? + - codeCoverage: Should code coverage be generated? (Xcode 7 and up) + - addressSanitizer: Should the address sanitizer be turned on? + - threadSanitizer: Should the thread sanitizer be turned on? + - openReport: Should the HTML report be opened when tests are completed? + - outputDirectory: The directory in which all reports will be stored + - outputStyle: Define how the output should look like. Valid values are: standard, basic, rspec, or raw (disables xcpretty during xcodebuild) + - outputTypes: Comma separated list of the output types (e.g. html, junit, json-compilation-database) + - outputFiles: Comma separated list of the output files, corresponding to the types provided by :output_types (order should match). If specifying an output type of json-compilation-database with :use_clang_report_name enabled, that option will take precedence + - buildlogPath: The directory where to store the raw log + - includeSimulatorLogs: If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - outputRemoveRetryAttempts: Remove retry attempts from test results table and the JUnit report (if not using xcpretty) + - disableXcpretty: **DEPRECATED!** Use `output_style: 'raw'` instead - Disable xcpretty formatting of build, similar to `output_style='raw'` but this will also skip the test results table + - formatter: **DEPRECATED!** Use 'xcpretty_formatter' instead - A custom xcpretty formatter to use + - xcprettyFormatter: A custom xcpretty formatter to use + - xcprettyArgs: Pass in xcpretty additional command line arguments (e.g. '--test --no-color' or '--tap --no-utf') + - derivedDataPath: The directory where build products and other derived data will go + - shouldZipBuildProducts: Should zip the derived data build products and place in output path? + - outputXctestrun: Should provide additional copy of .xctestrun file (settings.xctestrun) and place in output path? + - resultBundle: Should an Xcode result bundle be generated in the output directory + - useClangReportName: Generate the json compilation database with clang naming convention (compile_commands.json) + - parallelTesting: Optionally override the per-target setting in the scheme for running tests in parallel. Equivalent to -parallel-testing-enabled + - concurrentWorkers: Specify the exact number of test runners that will be spawned during parallel testing. Equivalent to -parallel-testing-worker-count + - maxConcurrentSimulators: Constrain the number of simulator devices on which to test concurrently. Equivalent to -maximum-concurrent-test-simulator-destinations + - disableConcurrentTesting: Do not run test bundles in parallel on the specified destinations. Testing will occur on each destination serially. Equivalent to -disable-concurrent-testing + - skipBuild: Should debug build be skipped before test build? + - testWithoutBuilding: Test without building, requires a derived data path + - buildForTesting: Build for testing only, does not run tests + - sdk: The SDK that should be used for building the application + - configuration: The configuration to use when building the app. Defaults to 'Release' + - xcargs: Pass additional arguments to xcodebuild. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - appName: App name to use in slack message and logfile name + - deploymentTargetVersion: Target version of the app being build or tested. Used to filter out simulator version + - slackUrl: Create an Incoming WebHook for your Slack group to post results there + - slackChannel: #channel or @username + - slackMessage: The message included with each message posted to slack + - slackUseWebhookConfiguredUsernameAndIcon: Use webhook's default username and icon settings? (true/false) + - slackUsername: Overrides the webhook's username property if slack_use_webhook_configured_username_and_icon is false + - slackIconUrl: Overrides the webhook's image property if slack_use_webhook_configured_username_and_icon is false + - skipSlack: Don't publish to slack, even when an URL is given + - slackOnlyOnFailure: Only post on Slack if the tests fail + - slackDefaultPayloads: Specifies default payloads to include in Slack messages. For more info visit https://docs.fastlane.tools/actions/slack + - destination: Use only if you're a pro, use the other options instead + - catalystPlatform: Platform to build when using a Catalyst enabled app. Valid values are: ios, macos + - customReportFileName: **DEPRECATED!** Use `--output_files` instead - Sets custom full report file name when generating a single report + - xcodebuildCommand: Allows for override of the default `xcodebuild` command + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - useSystemScm: Lets xcodebuild use system's scm configuration + - numberOfRetries: The number of times a test can fail + - failBuild: Should this step stop the build if the tests fail? Set this to false if you're using trainer + + - returns: Outputs hash of results with the following keys: :number_of_tests, :number_of_failures, :number_of_retries, :number_of_tests_excluding_retries, :number_of_failures_excluding_retries + + More information: https://docs.fastlane.tools/actions/scan/ + */ +@discardableResult public func runTests(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + packagePath: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + device: OptionalConfigValue = .fastlaneDefault(nil), + devices: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + skipDetectDevices: OptionalConfigValue = .fastlaneDefault(false), + ensureDevicesFound: OptionalConfigValue = .fastlaneDefault(false), + forceQuitSimulator: OptionalConfigValue = .fastlaneDefault(false), + resetSimulator: OptionalConfigValue = .fastlaneDefault(false), + disableSlideToType: OptionalConfigValue = .fastlaneDefault(true), + prelaunchSimulator: OptionalConfigValue = .fastlaneDefault(nil), + reinstallApp: OptionalConfigValue = .fastlaneDefault(false), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + onlyTesting: Any? = nil, + skipTesting: Any? = nil, + testplan: OptionalConfigValue = .fastlaneDefault(nil), + onlyTestConfigurations: Any? = nil, + skipTestConfigurations: Any? = nil, + xctestrun: OptionalConfigValue = .fastlaneDefault(nil), + toolchain: Any? = nil, + clean: OptionalConfigValue = .fastlaneDefault(false), + codeCoverage: OptionalConfigValue = .fastlaneDefault(nil), + addressSanitizer: OptionalConfigValue = .fastlaneDefault(nil), + threadSanitizer: OptionalConfigValue = .fastlaneDefault(nil), + openReport: OptionalConfigValue = .fastlaneDefault(false), + outputDirectory: String = "./test_output", + outputStyle: OptionalConfigValue = .fastlaneDefault(nil), + outputTypes: String = "html,junit", + outputFiles: OptionalConfigValue = .fastlaneDefault(nil), + buildlogPath: String = "~/Library/Logs/scan", + includeSimulatorLogs: OptionalConfigValue = .fastlaneDefault(false), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(nil), + xcodebuildFormatter: String = "xcbeautify", + outputRemoveRetryAttempts: OptionalConfigValue = .fastlaneDefault(false), + disableXcpretty: OptionalConfigValue = .fastlaneDefault(nil), + formatter: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyFormatter: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyArgs: OptionalConfigValue = .fastlaneDefault(nil), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + shouldZipBuildProducts: OptionalConfigValue = .fastlaneDefault(false), + outputXctestrun: OptionalConfigValue = .fastlaneDefault(false), + resultBundle: OptionalConfigValue = .fastlaneDefault(false), + useClangReportName: OptionalConfigValue = .fastlaneDefault(false), + parallelTesting: OptionalConfigValue = .fastlaneDefault(nil), + concurrentWorkers: OptionalConfigValue = .fastlaneDefault(nil), + maxConcurrentSimulators: OptionalConfigValue = .fastlaneDefault(nil), + disableConcurrentTesting: OptionalConfigValue = .fastlaneDefault(false), + skipBuild: OptionalConfigValue = .fastlaneDefault(false), + testWithoutBuilding: OptionalConfigValue = .fastlaneDefault(nil), + buildForTesting: OptionalConfigValue = .fastlaneDefault(nil), + sdk: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + xcargs: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + appName: OptionalConfigValue = .fastlaneDefault(nil), + deploymentTargetVersion: OptionalConfigValue = .fastlaneDefault(nil), + slackUrl: OptionalConfigValue = .fastlaneDefault(nil), + slackChannel: OptionalConfigValue = .fastlaneDefault(nil), + slackMessage: OptionalConfigValue = .fastlaneDefault(nil), + slackUseWebhookConfiguredUsernameAndIcon: OptionalConfigValue = .fastlaneDefault(false), + slackUsername: String = "fastlane", + slackIconUrl: String = "https://fastlane.tools/assets/img/fastlane_icon.png", + skipSlack: OptionalConfigValue = .fastlaneDefault(false), + slackOnlyOnFailure: OptionalConfigValue = .fastlaneDefault(false), + slackDefaultPayloads: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + destination: Any? = nil, + catalystPlatform: OptionalConfigValue = .fastlaneDefault(nil), + customReportFileName: OptionalConfigValue = .fastlaneDefault(nil), + xcodebuildCommand: String = "env NSUnbufferedIO=YES xcodebuild", + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(nil), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(false), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(false), + useSystemScm: OptionalConfigValue = .fastlaneDefault(false), + numberOfRetries: Int = 0, + failBuild: OptionalConfigValue = .fastlaneDefault(true)) -> [String: Any] +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let packagePathArg = packagePath.asRubyArgument(name: "package_path", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let deviceArg = device.asRubyArgument(name: "device", type: nil) + let devicesArg = devices.asRubyArgument(name: "devices", type: nil) + let skipDetectDevicesArg = skipDetectDevices.asRubyArgument(name: "skip_detect_devices", type: nil) + let ensureDevicesFoundArg = ensureDevicesFound.asRubyArgument(name: "ensure_devices_found", type: nil) + let forceQuitSimulatorArg = forceQuitSimulator.asRubyArgument(name: "force_quit_simulator", type: nil) + let resetSimulatorArg = resetSimulator.asRubyArgument(name: "reset_simulator", type: nil) + let disableSlideToTypeArg = disableSlideToType.asRubyArgument(name: "disable_slide_to_type", type: nil) + let prelaunchSimulatorArg = prelaunchSimulator.asRubyArgument(name: "prelaunch_simulator", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let onlyTestingArg = RubyCommand.Argument(name: "only_testing", value: onlyTesting, type: nil) + let skipTestingArg = RubyCommand.Argument(name: "skip_testing", value: skipTesting, type: nil) + let testplanArg = testplan.asRubyArgument(name: "testplan", type: nil) + let onlyTestConfigurationsArg = RubyCommand.Argument(name: "only_test_configurations", value: onlyTestConfigurations, type: nil) + let skipTestConfigurationsArg = RubyCommand.Argument(name: "skip_test_configurations", value: skipTestConfigurations, type: nil) + let xctestrunArg = xctestrun.asRubyArgument(name: "xctestrun", type: nil) + let toolchainArg = RubyCommand.Argument(name: "toolchain", value: toolchain, type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let codeCoverageArg = codeCoverage.asRubyArgument(name: "code_coverage", type: nil) + let addressSanitizerArg = addressSanitizer.asRubyArgument(name: "address_sanitizer", type: nil) + let threadSanitizerArg = threadSanitizer.asRubyArgument(name: "thread_sanitizer", type: nil) + let openReportArg = openReport.asRubyArgument(name: "open_report", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputStyleArg = outputStyle.asRubyArgument(name: "output_style", type: nil) + let outputTypesArg = RubyCommand.Argument(name: "output_types", value: outputTypes, type: nil) + let outputFilesArg = outputFiles.asRubyArgument(name: "output_files", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let includeSimulatorLogsArg = includeSimulatorLogs.asRubyArgument(name: "include_simulator_logs", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let outputRemoveRetryAttemptsArg = outputRemoveRetryAttempts.asRubyArgument(name: "output_remove_retry_attempts", type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let formatterArg = formatter.asRubyArgument(name: "formatter", type: nil) + let xcprettyFormatterArg = xcprettyFormatter.asRubyArgument(name: "xcpretty_formatter", type: nil) + let xcprettyArgsArg = xcprettyArgs.asRubyArgument(name: "xcpretty_args", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let shouldZipBuildProductsArg = shouldZipBuildProducts.asRubyArgument(name: "should_zip_build_products", type: nil) + let outputXctestrunArg = outputXctestrun.asRubyArgument(name: "output_xctestrun", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let useClangReportNameArg = useClangReportName.asRubyArgument(name: "use_clang_report_name", type: nil) + let parallelTestingArg = parallelTesting.asRubyArgument(name: "parallel_testing", type: nil) + let concurrentWorkersArg = concurrentWorkers.asRubyArgument(name: "concurrent_workers", type: nil) + let maxConcurrentSimulatorsArg = maxConcurrentSimulators.asRubyArgument(name: "max_concurrent_simulators", type: nil) + let disableConcurrentTestingArg = disableConcurrentTesting.asRubyArgument(name: "disable_concurrent_testing", type: nil) + let skipBuildArg = skipBuild.asRubyArgument(name: "skip_build", type: nil) + let testWithoutBuildingArg = testWithoutBuilding.asRubyArgument(name: "test_without_building", type: nil) + let buildForTestingArg = buildForTesting.asRubyArgument(name: "build_for_testing", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let appNameArg = appName.asRubyArgument(name: "app_name", type: nil) + let deploymentTargetVersionArg = deploymentTargetVersion.asRubyArgument(name: "deployment_target_version", type: nil) + let slackUrlArg = slackUrl.asRubyArgument(name: "slack_url", type: nil) + let slackChannelArg = slackChannel.asRubyArgument(name: "slack_channel", type: nil) + let slackMessageArg = slackMessage.asRubyArgument(name: "slack_message", type: nil) + let slackUseWebhookConfiguredUsernameAndIconArg = slackUseWebhookConfiguredUsernameAndIcon.asRubyArgument(name: "slack_use_webhook_configured_username_and_icon", type: nil) + let slackUsernameArg = RubyCommand.Argument(name: "slack_username", value: slackUsername, type: nil) + let slackIconUrlArg = RubyCommand.Argument(name: "slack_icon_url", value: slackIconUrl, type: nil) + let skipSlackArg = skipSlack.asRubyArgument(name: "skip_slack", type: nil) + let slackOnlyOnFailureArg = slackOnlyOnFailure.asRubyArgument(name: "slack_only_on_failure", type: nil) + let slackDefaultPayloadsArg = slackDefaultPayloads.asRubyArgument(name: "slack_default_payloads", type: nil) + let destinationArg = RubyCommand.Argument(name: "destination", value: destination, type: nil) + let catalystPlatformArg = catalystPlatform.asRubyArgument(name: "catalyst_platform", type: nil) + let customReportFileNameArg = customReportFileName.asRubyArgument(name: "custom_report_file_name", type: nil) + let xcodebuildCommandArg = RubyCommand.Argument(name: "xcodebuild_command", value: xcodebuildCommand, type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let numberOfRetriesArg = RubyCommand.Argument(name: "number_of_retries", value: numberOfRetries, type: nil) + let failBuildArg = failBuild.asRubyArgument(name: "fail_build", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + packagePathArg, + schemeArg, + deviceArg, + devicesArg, + skipDetectDevicesArg, + ensureDevicesFoundArg, + forceQuitSimulatorArg, + resetSimulatorArg, + disableSlideToTypeArg, + prelaunchSimulatorArg, + reinstallAppArg, + appIdentifierArg, + onlyTestingArg, + skipTestingArg, + testplanArg, + onlyTestConfigurationsArg, + skipTestConfigurationsArg, + xctestrunArg, + toolchainArg, + cleanArg, + codeCoverageArg, + addressSanitizerArg, + threadSanitizerArg, + openReportArg, + outputDirectoryArg, + outputStyleArg, + outputTypesArg, + outputFilesArg, + buildlogPathArg, + includeSimulatorLogsArg, + suppressXcodeOutputArg, + xcodebuildFormatterArg, + outputRemoveRetryAttemptsArg, + disableXcprettyArg, + formatterArg, + xcprettyFormatterArg, + xcprettyArgsArg, + derivedDataPathArg, + shouldZipBuildProductsArg, + outputXctestrunArg, + resultBundleArg, + useClangReportNameArg, + parallelTestingArg, + concurrentWorkersArg, + maxConcurrentSimulatorsArg, + disableConcurrentTestingArg, + skipBuildArg, + testWithoutBuildingArg, + buildForTestingArg, + sdkArg, + configurationArg, + xcargsArg, + xcconfigArg, + appNameArg, + deploymentTargetVersionArg, + slackUrlArg, + slackChannelArg, + slackMessageArg, + slackUseWebhookConfiguredUsernameAndIconArg, + slackUsernameArg, + slackIconUrlArg, + skipSlackArg, + slackOnlyOnFailureArg, + slackDefaultPayloadsArg, + destinationArg, + catalystPlatformArg, + customReportFileNameArg, + xcodebuildCommandArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + useSystemScmArg, + numberOfRetriesArg, + failBuildArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "run_tests", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Generates a plist file and uploads all to AWS S3 + + - parameters: + - ipa: .ipa file for the build + - dsym: zipped .dsym package for the build + - uploadMetadata: Upload relevant metadata for this build + - plistTemplatePath: plist template path + - plistFileName: uploaded plist filename + - htmlTemplatePath: html erb template path + - htmlFileName: uploaded html filename + - versionTemplatePath: version erb template path + - versionFileName: uploaded version filename + - accessKey: AWS Access Key ID + - secretAccessKey: AWS Secret Access Key + - bucket: AWS bucket name + - region: AWS region (for bucket creation) + - path: S3 'path'. Values from Info.plist will be substituted for keys wrapped in {} + - source: Optional source directory e.g. ./build + - acl: Uploaded object permissions e.g public_read (default), private, public_read_write, authenticated_read + + Upload a new build to Amazon S3 to distribute the build to beta testers. + Works for both Ad Hoc and Enterprise signed applications. This step will generate the necessary HTML, plist, and version files for you. + It is recommended to **not** store the AWS access keys in the `Fastfile`. The uploaded `version.json` file provides an easy way for apps to poll if a new update is available. + */ +public func s3(ipa: OptionalConfigValue = .fastlaneDefault(nil), + dsym: OptionalConfigValue = .fastlaneDefault(nil), + uploadMetadata: OptionalConfigValue = .fastlaneDefault(true), + plistTemplatePath: OptionalConfigValue = .fastlaneDefault(nil), + plistFileName: OptionalConfigValue = .fastlaneDefault(nil), + htmlTemplatePath: OptionalConfigValue = .fastlaneDefault(nil), + htmlFileName: OptionalConfigValue = .fastlaneDefault(nil), + versionTemplatePath: OptionalConfigValue = .fastlaneDefault(nil), + versionFileName: OptionalConfigValue = .fastlaneDefault(nil), + accessKey: OptionalConfigValue = .fastlaneDefault(nil), + secretAccessKey: OptionalConfigValue = .fastlaneDefault(nil), + bucket: OptionalConfigValue = .fastlaneDefault(nil), + region: OptionalConfigValue = .fastlaneDefault(nil), + path: String = "v{CFBundleShortVersionString}_b{CFBundleVersion}/", + source: OptionalConfigValue = .fastlaneDefault(nil), + acl: String = "public_read") +{ + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let dsymArg = dsym.asRubyArgument(name: "dsym", type: nil) + let uploadMetadataArg = uploadMetadata.asRubyArgument(name: "upload_metadata", type: nil) + let plistTemplatePathArg = plistTemplatePath.asRubyArgument(name: "plist_template_path", type: nil) + let plistFileNameArg = plistFileName.asRubyArgument(name: "plist_file_name", type: nil) + let htmlTemplatePathArg = htmlTemplatePath.asRubyArgument(name: "html_template_path", type: nil) + let htmlFileNameArg = htmlFileName.asRubyArgument(name: "html_file_name", type: nil) + let versionTemplatePathArg = versionTemplatePath.asRubyArgument(name: "version_template_path", type: nil) + let versionFileNameArg = versionFileName.asRubyArgument(name: "version_file_name", type: nil) + let accessKeyArg = accessKey.asRubyArgument(name: "access_key", type: nil) + let secretAccessKeyArg = secretAccessKey.asRubyArgument(name: "secret_access_key", type: nil) + let bucketArg = bucket.asRubyArgument(name: "bucket", type: nil) + let regionArg = region.asRubyArgument(name: "region", type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let sourceArg = source.asRubyArgument(name: "source", type: nil) + let aclArg = RubyCommand.Argument(name: "acl", value: acl, type: nil) + let array: [RubyCommand.Argument?] = [ipaArg, + dsymArg, + uploadMetadataArg, + plistTemplatePathArg, + plistFileNameArg, + htmlTemplatePathArg, + htmlFileNameArg, + versionTemplatePathArg, + versionFileNameArg, + accessKeyArg, + secretAccessKeyArg, + bucketArg, + regionArg, + pathArg, + sourceArg, + aclArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "s3", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action speaks the given text out loud + + - parameters: + - text: Text to be spoken out loud (as string or array of strings) + - mute: If say should be muted with text printed out + */ +public func say(text: [String], + mute: OptionalConfigValue = .fastlaneDefault(false)) +{ + let textArg = RubyCommand.Argument(name: "text", value: text, type: nil) + let muteArg = mute.asRubyArgument(name: "mute", type: nil) + let array: [RubyCommand.Argument?] = [textArg, + muteArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "say", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `run_tests` action + + - parameters: + - workspace: Path to the workspace file + - project: Path to the project file + - packagePath: Path to the Swift Package + - scheme: The project's scheme. Make sure it's marked as `Shared` + - device: The name of the simulator type you want to run tests on (e.g. 'iPhone 6' or 'iPhone SE (2nd generation) (14.5)') + - devices: Array of devices to run the tests on (e.g. ['iPhone 6', 'iPad Air', 'iPhone SE (2nd generation) (14.5)']) + - skipDetectDevices: Should skip auto detecting of devices if none were specified + - ensureDevicesFound: Should fail if devices not found + - forceQuitSimulator: Enabling this option will automatically killall Simulator processes before the run + - resetSimulator: Enabling this option will automatically erase the simulator before running the application + - disableSlideToType: Enabling this option will disable the simulator from showing the 'Slide to type' prompt + - prelaunchSimulator: Enabling this option will launch the first simulator prior to calling any xcodebuild command + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - appIdentifier: The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + - onlyTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to run + - skipTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to skip + - testplan: The testplan associated with the scheme that should be used for testing + - onlyTestConfigurations: Array of strings matching test plan configurations to run + - skipTestConfigurations: Array of strings matching test plan configurations to skip + - xctestrun: Run tests using the provided `.xctestrun` file + - toolchain: The toolchain that should be used for building the application (e.g. `com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a`) + - clean: Should the project be cleaned before building it? + - codeCoverage: Should code coverage be generated? (Xcode 7 and up) + - addressSanitizer: Should the address sanitizer be turned on? + - threadSanitizer: Should the thread sanitizer be turned on? + - openReport: Should the HTML report be opened when tests are completed? + - outputDirectory: The directory in which all reports will be stored + - outputStyle: Define how the output should look like. Valid values are: standard, basic, rspec, or raw (disables xcpretty during xcodebuild) + - outputTypes: Comma separated list of the output types (e.g. html, junit, json-compilation-database) + - outputFiles: Comma separated list of the output files, corresponding to the types provided by :output_types (order should match). If specifying an output type of json-compilation-database with :use_clang_report_name enabled, that option will take precedence + - buildlogPath: The directory where to store the raw log + - includeSimulatorLogs: If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - outputRemoveRetryAttempts: Remove retry attempts from test results table and the JUnit report (if not using xcpretty) + - disableXcpretty: **DEPRECATED!** Use `output_style: 'raw'` instead - Disable xcpretty formatting of build, similar to `output_style='raw'` but this will also skip the test results table + - formatter: **DEPRECATED!** Use 'xcpretty_formatter' instead - A custom xcpretty formatter to use + - xcprettyFormatter: A custom xcpretty formatter to use + - xcprettyArgs: Pass in xcpretty additional command line arguments (e.g. '--test --no-color' or '--tap --no-utf') + - derivedDataPath: The directory where build products and other derived data will go + - shouldZipBuildProducts: Should zip the derived data build products and place in output path? + - outputXctestrun: Should provide additional copy of .xctestrun file (settings.xctestrun) and place in output path? + - resultBundle: Should an Xcode result bundle be generated in the output directory + - useClangReportName: Generate the json compilation database with clang naming convention (compile_commands.json) + - parallelTesting: Optionally override the per-target setting in the scheme for running tests in parallel. Equivalent to -parallel-testing-enabled + - concurrentWorkers: Specify the exact number of test runners that will be spawned during parallel testing. Equivalent to -parallel-testing-worker-count + - maxConcurrentSimulators: Constrain the number of simulator devices on which to test concurrently. Equivalent to -maximum-concurrent-test-simulator-destinations + - disableConcurrentTesting: Do not run test bundles in parallel on the specified destinations. Testing will occur on each destination serially. Equivalent to -disable-concurrent-testing + - skipBuild: Should debug build be skipped before test build? + - testWithoutBuilding: Test without building, requires a derived data path + - buildForTesting: Build for testing only, does not run tests + - sdk: The SDK that should be used for building the application + - configuration: The configuration to use when building the app. Defaults to 'Release' + - xcargs: Pass additional arguments to xcodebuild. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - appName: App name to use in slack message and logfile name + - deploymentTargetVersion: Target version of the app being build or tested. Used to filter out simulator version + - slackUrl: Create an Incoming WebHook for your Slack group to post results there + - slackChannel: #channel or @username + - slackMessage: The message included with each message posted to slack + - slackUseWebhookConfiguredUsernameAndIcon: Use webhook's default username and icon settings? (true/false) + - slackUsername: Overrides the webhook's username property if slack_use_webhook_configured_username_and_icon is false + - slackIconUrl: Overrides the webhook's image property if slack_use_webhook_configured_username_and_icon is false + - skipSlack: Don't publish to slack, even when an URL is given + - slackOnlyOnFailure: Only post on Slack if the tests fail + - slackDefaultPayloads: Specifies default payloads to include in Slack messages. For more info visit https://docs.fastlane.tools/actions/slack + - destination: Use only if you're a pro, use the other options instead + - catalystPlatform: Platform to build when using a Catalyst enabled app. Valid values are: ios, macos + - customReportFileName: **DEPRECATED!** Use `--output_files` instead - Sets custom full report file name when generating a single report + - xcodebuildCommand: Allows for override of the default `xcodebuild` command + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - useSystemScm: Lets xcodebuild use system's scm configuration + - numberOfRetries: The number of times a test can fail + - failBuild: Should this step stop the build if the tests fail? Set this to false if you're using trainer + + - returns: Outputs hash of results with the following keys: :number_of_tests, :number_of_failures, :number_of_retries, :number_of_tests_excluding_retries, :number_of_failures_excluding_retries + + More information: https://docs.fastlane.tools/actions/scan/ + */ +@discardableResult public func scan(workspace: OptionalConfigValue = .fastlaneDefault(scanfile.workspace), + project: OptionalConfigValue = .fastlaneDefault(scanfile.project), + packagePath: OptionalConfigValue = .fastlaneDefault(scanfile.packagePath), + scheme: OptionalConfigValue = .fastlaneDefault(scanfile.scheme), + device: OptionalConfigValue = .fastlaneDefault(scanfile.device), + devices: OptionalConfigValue<[String]?> = .fastlaneDefault(scanfile.devices), + skipDetectDevices: OptionalConfigValue = .fastlaneDefault(scanfile.skipDetectDevices), + ensureDevicesFound: OptionalConfigValue = .fastlaneDefault(scanfile.ensureDevicesFound), + forceQuitSimulator: OptionalConfigValue = .fastlaneDefault(scanfile.forceQuitSimulator), + resetSimulator: OptionalConfigValue = .fastlaneDefault(scanfile.resetSimulator), + disableSlideToType: OptionalConfigValue = .fastlaneDefault(scanfile.disableSlideToType), + prelaunchSimulator: OptionalConfigValue = .fastlaneDefault(scanfile.prelaunchSimulator), + reinstallApp: OptionalConfigValue = .fastlaneDefault(scanfile.reinstallApp), + appIdentifier: OptionalConfigValue = .fastlaneDefault(scanfile.appIdentifier), + onlyTesting: Any? = scanfile.onlyTesting, + skipTesting: Any? = scanfile.skipTesting, + testplan: OptionalConfigValue = .fastlaneDefault(scanfile.testplan), + onlyTestConfigurations: Any? = scanfile.onlyTestConfigurations, + skipTestConfigurations: Any? = scanfile.skipTestConfigurations, + xctestrun: OptionalConfigValue = .fastlaneDefault(scanfile.xctestrun), + toolchain: Any? = scanfile.toolchain, + clean: OptionalConfigValue = .fastlaneDefault(scanfile.clean), + codeCoverage: OptionalConfigValue = .fastlaneDefault(scanfile.codeCoverage), + addressSanitizer: OptionalConfigValue = .fastlaneDefault(scanfile.addressSanitizer), + threadSanitizer: OptionalConfigValue = .fastlaneDefault(scanfile.threadSanitizer), + openReport: OptionalConfigValue = .fastlaneDefault(scanfile.openReport), + outputDirectory: String = scanfile.outputDirectory, + outputStyle: OptionalConfigValue = .fastlaneDefault(scanfile.outputStyle), + outputTypes: String = scanfile.outputTypes, + outputFiles: OptionalConfigValue = .fastlaneDefault(scanfile.outputFiles), + buildlogPath: String = scanfile.buildlogPath, + includeSimulatorLogs: OptionalConfigValue = .fastlaneDefault(scanfile.includeSimulatorLogs), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(scanfile.suppressXcodeOutput), + xcodebuildFormatter: String = scanfile.xcodebuildFormatter, + outputRemoveRetryAttempts: OptionalConfigValue = .fastlaneDefault(scanfile.outputRemoveRetryAttempts), + disableXcpretty: OptionalConfigValue = .fastlaneDefault(scanfile.disableXcpretty), + formatter: OptionalConfigValue = .fastlaneDefault(scanfile.formatter), + xcprettyFormatter: OptionalConfigValue = .fastlaneDefault(scanfile.xcprettyFormatter), + xcprettyArgs: OptionalConfigValue = .fastlaneDefault(scanfile.xcprettyArgs), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(scanfile.derivedDataPath), + shouldZipBuildProducts: OptionalConfigValue = .fastlaneDefault(scanfile.shouldZipBuildProducts), + outputXctestrun: OptionalConfigValue = .fastlaneDefault(scanfile.outputXctestrun), + resultBundle: OptionalConfigValue = .fastlaneDefault(scanfile.resultBundle), + useClangReportName: OptionalConfigValue = .fastlaneDefault(scanfile.useClangReportName), + parallelTesting: OptionalConfigValue = .fastlaneDefault(scanfile.parallelTesting), + concurrentWorkers: OptionalConfigValue = .fastlaneDefault(scanfile.concurrentWorkers), + maxConcurrentSimulators: OptionalConfigValue = .fastlaneDefault(scanfile.maxConcurrentSimulators), + disableConcurrentTesting: OptionalConfigValue = .fastlaneDefault(scanfile.disableConcurrentTesting), + skipBuild: OptionalConfigValue = .fastlaneDefault(scanfile.skipBuild), + testWithoutBuilding: OptionalConfigValue = .fastlaneDefault(scanfile.testWithoutBuilding), + buildForTesting: OptionalConfigValue = .fastlaneDefault(scanfile.buildForTesting), + sdk: OptionalConfigValue = .fastlaneDefault(scanfile.sdk), + configuration: OptionalConfigValue = .fastlaneDefault(scanfile.configuration), + xcargs: OptionalConfigValue = .fastlaneDefault(scanfile.xcargs), + xcconfig: OptionalConfigValue = .fastlaneDefault(scanfile.xcconfig), + appName: OptionalConfigValue = .fastlaneDefault(scanfile.appName), + deploymentTargetVersion: OptionalConfigValue = .fastlaneDefault(scanfile.deploymentTargetVersion), + slackUrl: OptionalConfigValue = .fastlaneDefault(scanfile.slackUrl), + slackChannel: OptionalConfigValue = .fastlaneDefault(scanfile.slackChannel), + slackMessage: OptionalConfigValue = .fastlaneDefault(scanfile.slackMessage), + slackUseWebhookConfiguredUsernameAndIcon: OptionalConfigValue = .fastlaneDefault(scanfile.slackUseWebhookConfiguredUsernameAndIcon), + slackUsername: String = scanfile.slackUsername, + slackIconUrl: String = scanfile.slackIconUrl, + skipSlack: OptionalConfigValue = .fastlaneDefault(scanfile.skipSlack), + slackOnlyOnFailure: OptionalConfigValue = .fastlaneDefault(scanfile.slackOnlyOnFailure), + slackDefaultPayloads: OptionalConfigValue<[String]?> = .fastlaneDefault(scanfile.slackDefaultPayloads), + destination: Any? = scanfile.destination, + catalystPlatform: OptionalConfigValue = .fastlaneDefault(scanfile.catalystPlatform), + customReportFileName: OptionalConfigValue = .fastlaneDefault(scanfile.customReportFileName), + xcodebuildCommand: String = scanfile.xcodebuildCommand, + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(scanfile.clonedSourcePackagesPath), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(scanfile.skipPackageDependenciesResolution), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(scanfile.disablePackageAutomaticUpdates), + useSystemScm: OptionalConfigValue = .fastlaneDefault(scanfile.useSystemScm), + numberOfRetries: Int = scanfile.numberOfRetries, + failBuild: OptionalConfigValue = .fastlaneDefault(scanfile.failBuild)) -> [String: Any] +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let packagePathArg = packagePath.asRubyArgument(name: "package_path", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let deviceArg = device.asRubyArgument(name: "device", type: nil) + let devicesArg = devices.asRubyArgument(name: "devices", type: nil) + let skipDetectDevicesArg = skipDetectDevices.asRubyArgument(name: "skip_detect_devices", type: nil) + let ensureDevicesFoundArg = ensureDevicesFound.asRubyArgument(name: "ensure_devices_found", type: nil) + let forceQuitSimulatorArg = forceQuitSimulator.asRubyArgument(name: "force_quit_simulator", type: nil) + let resetSimulatorArg = resetSimulator.asRubyArgument(name: "reset_simulator", type: nil) + let disableSlideToTypeArg = disableSlideToType.asRubyArgument(name: "disable_slide_to_type", type: nil) + let prelaunchSimulatorArg = prelaunchSimulator.asRubyArgument(name: "prelaunch_simulator", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let onlyTestingArg = RubyCommand.Argument(name: "only_testing", value: onlyTesting, type: nil) + let skipTestingArg = RubyCommand.Argument(name: "skip_testing", value: skipTesting, type: nil) + let testplanArg = testplan.asRubyArgument(name: "testplan", type: nil) + let onlyTestConfigurationsArg = RubyCommand.Argument(name: "only_test_configurations", value: onlyTestConfigurations, type: nil) + let skipTestConfigurationsArg = RubyCommand.Argument(name: "skip_test_configurations", value: skipTestConfigurations, type: nil) + let xctestrunArg = xctestrun.asRubyArgument(name: "xctestrun", type: nil) + let toolchainArg = RubyCommand.Argument(name: "toolchain", value: toolchain, type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let codeCoverageArg = codeCoverage.asRubyArgument(name: "code_coverage", type: nil) + let addressSanitizerArg = addressSanitizer.asRubyArgument(name: "address_sanitizer", type: nil) + let threadSanitizerArg = threadSanitizer.asRubyArgument(name: "thread_sanitizer", type: nil) + let openReportArg = openReport.asRubyArgument(name: "open_report", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputStyleArg = outputStyle.asRubyArgument(name: "output_style", type: nil) + let outputTypesArg = RubyCommand.Argument(name: "output_types", value: outputTypes, type: nil) + let outputFilesArg = outputFiles.asRubyArgument(name: "output_files", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let includeSimulatorLogsArg = includeSimulatorLogs.asRubyArgument(name: "include_simulator_logs", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let outputRemoveRetryAttemptsArg = outputRemoveRetryAttempts.asRubyArgument(name: "output_remove_retry_attempts", type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let formatterArg = formatter.asRubyArgument(name: "formatter", type: nil) + let xcprettyFormatterArg = xcprettyFormatter.asRubyArgument(name: "xcpretty_formatter", type: nil) + let xcprettyArgsArg = xcprettyArgs.asRubyArgument(name: "xcpretty_args", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let shouldZipBuildProductsArg = shouldZipBuildProducts.asRubyArgument(name: "should_zip_build_products", type: nil) + let outputXctestrunArg = outputXctestrun.asRubyArgument(name: "output_xctestrun", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let useClangReportNameArg = useClangReportName.asRubyArgument(name: "use_clang_report_name", type: nil) + let parallelTestingArg = parallelTesting.asRubyArgument(name: "parallel_testing", type: nil) + let concurrentWorkersArg = concurrentWorkers.asRubyArgument(name: "concurrent_workers", type: nil) + let maxConcurrentSimulatorsArg = maxConcurrentSimulators.asRubyArgument(name: "max_concurrent_simulators", type: nil) + let disableConcurrentTestingArg = disableConcurrentTesting.asRubyArgument(name: "disable_concurrent_testing", type: nil) + let skipBuildArg = skipBuild.asRubyArgument(name: "skip_build", type: nil) + let testWithoutBuildingArg = testWithoutBuilding.asRubyArgument(name: "test_without_building", type: nil) + let buildForTestingArg = buildForTesting.asRubyArgument(name: "build_for_testing", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let appNameArg = appName.asRubyArgument(name: "app_name", type: nil) + let deploymentTargetVersionArg = deploymentTargetVersion.asRubyArgument(name: "deployment_target_version", type: nil) + let slackUrlArg = slackUrl.asRubyArgument(name: "slack_url", type: nil) + let slackChannelArg = slackChannel.asRubyArgument(name: "slack_channel", type: nil) + let slackMessageArg = slackMessage.asRubyArgument(name: "slack_message", type: nil) + let slackUseWebhookConfiguredUsernameAndIconArg = slackUseWebhookConfiguredUsernameAndIcon.asRubyArgument(name: "slack_use_webhook_configured_username_and_icon", type: nil) + let slackUsernameArg = RubyCommand.Argument(name: "slack_username", value: slackUsername, type: nil) + let slackIconUrlArg = RubyCommand.Argument(name: "slack_icon_url", value: slackIconUrl, type: nil) + let skipSlackArg = skipSlack.asRubyArgument(name: "skip_slack", type: nil) + let slackOnlyOnFailureArg = slackOnlyOnFailure.asRubyArgument(name: "slack_only_on_failure", type: nil) + let slackDefaultPayloadsArg = slackDefaultPayloads.asRubyArgument(name: "slack_default_payloads", type: nil) + let destinationArg = RubyCommand.Argument(name: "destination", value: destination, type: nil) + let catalystPlatformArg = catalystPlatform.asRubyArgument(name: "catalyst_platform", type: nil) + let customReportFileNameArg = customReportFileName.asRubyArgument(name: "custom_report_file_name", type: nil) + let xcodebuildCommandArg = RubyCommand.Argument(name: "xcodebuild_command", value: xcodebuildCommand, type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let numberOfRetriesArg = RubyCommand.Argument(name: "number_of_retries", value: numberOfRetries, type: nil) + let failBuildArg = failBuild.asRubyArgument(name: "fail_build", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + packagePathArg, + schemeArg, + deviceArg, + devicesArg, + skipDetectDevicesArg, + ensureDevicesFoundArg, + forceQuitSimulatorArg, + resetSimulatorArg, + disableSlideToTypeArg, + prelaunchSimulatorArg, + reinstallAppArg, + appIdentifierArg, + onlyTestingArg, + skipTestingArg, + testplanArg, + onlyTestConfigurationsArg, + skipTestConfigurationsArg, + xctestrunArg, + toolchainArg, + cleanArg, + codeCoverageArg, + addressSanitizerArg, + threadSanitizerArg, + openReportArg, + outputDirectoryArg, + outputStyleArg, + outputTypesArg, + outputFilesArg, + buildlogPathArg, + includeSimulatorLogsArg, + suppressXcodeOutputArg, + xcodebuildFormatterArg, + outputRemoveRetryAttemptsArg, + disableXcprettyArg, + formatterArg, + xcprettyFormatterArg, + xcprettyArgsArg, + derivedDataPathArg, + shouldZipBuildProductsArg, + outputXctestrunArg, + resultBundleArg, + useClangReportNameArg, + parallelTestingArg, + concurrentWorkersArg, + maxConcurrentSimulatorsArg, + disableConcurrentTestingArg, + skipBuildArg, + testWithoutBuildingArg, + buildForTestingArg, + sdkArg, + configurationArg, + xcargsArg, + xcconfigArg, + appNameArg, + deploymentTargetVersionArg, + slackUrlArg, + slackChannelArg, + slackMessageArg, + slackUseWebhookConfiguredUsernameAndIconArg, + slackUsernameArg, + slackIconUrlArg, + skipSlackArg, + slackOnlyOnFailureArg, + slackDefaultPayloadsArg, + destinationArg, + catalystPlatformArg, + customReportFileNameArg, + xcodebuildCommandArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + useSystemScmArg, + numberOfRetriesArg, + failBuildArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "scan", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Transfer files via SCP + + - parameters: + - username: Username + - password: Password + - host: Hostname + - port: Port + - upload: Upload + - download: Download + */ +public func scp(username: String, + password: OptionalConfigValue = .fastlaneDefault(nil), + host: String, + port: String = "22", + upload: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + download: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil)) +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let passwordArg = password.asRubyArgument(name: "password", type: nil) + let hostArg = RubyCommand.Argument(name: "host", value: host, type: nil) + let portArg = RubyCommand.Argument(name: "port", value: port, type: nil) + let uploadArg = upload.asRubyArgument(name: "upload", type: nil) + let downloadArg = download.asRubyArgument(name: "download", type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + passwordArg, + hostArg, + portArg, + uploadArg, + downloadArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "scp", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `capture_android_screenshots` action + + - parameters: + - androidHome: Path to the root of your Android SDK installation, e.g. ~/tools/android-sdk-macosx + - buildToolsVersion: **DEPRECATED!** The Android build tools version to use, e.g. '23.0.2' + - locales: A list of locales which should be used + - clearPreviousScreenshots: Enabling this option will automatically clear previously generated screenshots before running screengrab + - outputDirectory: The directory where to store the screenshots + - skipOpenSummary: Don't open the summary after running _screengrab_ + - appPackageName: The package name of the app under test (e.g. com.yourcompany.yourapp) + - testsPackageName: The package name of the tests bundle (e.g. com.yourcompany.yourapp.test) + - useTestsInPackages: Only run tests in these Java packages + - useTestsInClasses: Only run tests in these Java classes + - launchArguments: Additional launch arguments + - testInstrumentationRunner: The fully qualified class name of your test instrumentation runner + - endingLocale: **DEPRECATED!** Return the device to this locale after running tests + - useAdbRoot: **DEPRECATED!** Restarts the adb daemon using `adb root` to allow access to screenshots directories on device. Use if getting 'Permission denied' errors + - appApkPath: The path to the APK for the app under test + - testsApkPath: The path to the APK for the tests bundle + - specificDevice: Use the device or emulator with the given serial number or qualifier + - deviceType: Type of device used for screenshots. Matches Google Play Types (phone, sevenInch, tenInch, tv, wear) + - exitOnTestFailure: Whether or not to exit Screengrab on test failure. Exiting on failure will not copy screenshots to local machine nor open screenshots summary + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - useTimestampSuffix: Add timestamp suffix to screenshot filename + - adbHost: Configure the host used by adb to connect, allows running on remote devices farm + */ +public func screengrab(androidHome: OptionalConfigValue = .fastlaneDefault(screengrabfile.androidHome), + buildToolsVersion: OptionalConfigValue = .fastlaneDefault(screengrabfile.buildToolsVersion), + locales: [String] = screengrabfile.locales, + clearPreviousScreenshots: OptionalConfigValue = .fastlaneDefault(screengrabfile.clearPreviousScreenshots), + outputDirectory: String = screengrabfile.outputDirectory, + skipOpenSummary: OptionalConfigValue = .fastlaneDefault(screengrabfile.skipOpenSummary), + appPackageName: String = screengrabfile.appPackageName, + testsPackageName: OptionalConfigValue = .fastlaneDefault(screengrabfile.testsPackageName), + useTestsInPackages: OptionalConfigValue<[String]?> = .fastlaneDefault(screengrabfile.useTestsInPackages), + useTestsInClasses: OptionalConfigValue<[String]?> = .fastlaneDefault(screengrabfile.useTestsInClasses), + launchArguments: OptionalConfigValue<[String]?> = .fastlaneDefault(screengrabfile.launchArguments), + testInstrumentationRunner: String = screengrabfile.testInstrumentationRunner, + endingLocale: String = screengrabfile.endingLocale, + useAdbRoot: OptionalConfigValue = .fastlaneDefault(screengrabfile.useAdbRoot), + appApkPath: OptionalConfigValue = .fastlaneDefault(screengrabfile.appApkPath), + testsApkPath: OptionalConfigValue = .fastlaneDefault(screengrabfile.testsApkPath), + specificDevice: OptionalConfigValue = .fastlaneDefault(screengrabfile.specificDevice), + deviceType: String = screengrabfile.deviceType, + exitOnTestFailure: OptionalConfigValue = .fastlaneDefault(screengrabfile.exitOnTestFailure), + reinstallApp: OptionalConfigValue = .fastlaneDefault(screengrabfile.reinstallApp), + useTimestampSuffix: OptionalConfigValue = .fastlaneDefault(screengrabfile.useTimestampSuffix), + adbHost: OptionalConfigValue = .fastlaneDefault(screengrabfile.adbHost)) +{ + let androidHomeArg = androidHome.asRubyArgument(name: "android_home", type: nil) + let buildToolsVersionArg = buildToolsVersion.asRubyArgument(name: "build_tools_version", type: nil) + let localesArg = RubyCommand.Argument(name: "locales", value: locales, type: nil) + let clearPreviousScreenshotsArg = clearPreviousScreenshots.asRubyArgument(name: "clear_previous_screenshots", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let skipOpenSummaryArg = skipOpenSummary.asRubyArgument(name: "skip_open_summary", type: nil) + let appPackageNameArg = RubyCommand.Argument(name: "app_package_name", value: appPackageName, type: nil) + let testsPackageNameArg = testsPackageName.asRubyArgument(name: "tests_package_name", type: nil) + let useTestsInPackagesArg = useTestsInPackages.asRubyArgument(name: "use_tests_in_packages", type: nil) + let useTestsInClassesArg = useTestsInClasses.asRubyArgument(name: "use_tests_in_classes", type: nil) + let launchArgumentsArg = launchArguments.asRubyArgument(name: "launch_arguments", type: nil) + let testInstrumentationRunnerArg = RubyCommand.Argument(name: "test_instrumentation_runner", value: testInstrumentationRunner, type: nil) + let endingLocaleArg = RubyCommand.Argument(name: "ending_locale", value: endingLocale, type: nil) + let useAdbRootArg = useAdbRoot.asRubyArgument(name: "use_adb_root", type: nil) + let appApkPathArg = appApkPath.asRubyArgument(name: "app_apk_path", type: nil) + let testsApkPathArg = testsApkPath.asRubyArgument(name: "tests_apk_path", type: nil) + let specificDeviceArg = specificDevice.asRubyArgument(name: "specific_device", type: nil) + let deviceTypeArg = RubyCommand.Argument(name: "device_type", value: deviceType, type: nil) + let exitOnTestFailureArg = exitOnTestFailure.asRubyArgument(name: "exit_on_test_failure", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let useTimestampSuffixArg = useTimestampSuffix.asRubyArgument(name: "use_timestamp_suffix", type: nil) + let adbHostArg = adbHost.asRubyArgument(name: "adb_host", type: nil) + let array: [RubyCommand.Argument?] = [androidHomeArg, + buildToolsVersionArg, + localesArg, + clearPreviousScreenshotsArg, + outputDirectoryArg, + skipOpenSummaryArg, + appPackageNameArg, + testsPackageNameArg, + useTestsInPackagesArg, + useTestsInClassesArg, + launchArgumentsArg, + testInstrumentationRunnerArg, + endingLocaleArg, + useAdbRootArg, + appApkPathArg, + testsApkPathArg, + specificDeviceArg, + deviceTypeArg, + exitOnTestFailureArg, + reinstallAppArg, + useTimestampSuffixArg, + adbHostArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "screengrab", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Set the build number from the current repository + + - parameters: + - useHgRevisionNumber: Use hg revision number instead of hash (ignored for non-hg repos) + - xcodeproj: explicitly specify which xcodeproj to use + + This action will set the **build number** according to what the SCM HEAD reports. + Currently supported SCMs are svn (uses root revision), git-svn (uses svn revision) and git (uses short hash) and mercurial (uses short hash or revision number). + There is an option, `:use_hg_revision_number`, which allows to use mercurial revision number instead of hash. + */ +public func setBuildNumberRepository(useHgRevisionNumber: OptionalConfigValue = .fastlaneDefault(false), + xcodeproj: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let useHgRevisionNumberArg = useHgRevisionNumber.asRubyArgument(name: "use_hg_revision_number", type: nil) + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let array: [RubyCommand.Argument?] = [useHgRevisionNumberArg, + xcodeprojArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "set_build_number_repository", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Set the changelog for all languages on App Store Connect + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - appIdentifier: The bundle identifier of your app + - username: Your Apple ID Username + - version: The version number to create/update + - changelog: Changelog text that should be uploaded to App Store Connect + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - platform: The platform of the app (ios, appletvos, mac) + + This is useful if you have only one changelog for all languages. + You can store the changelog in `./changelog.txt` and it will automatically get loaded from there. This integration is useful if you support e.g. 10 languages and want to use the same "What's new"-text for all languages. + Defining the version is optional. _fastlane_ will try to automatically detect it if you don't provide one. + */ +public func setChangelog(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appIdentifier: String, + username: OptionalConfigValue = .fastlaneDefault(nil), + version: OptionalConfigValue = .fastlaneDefault(nil), + changelog: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios") +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let versionArg = version.asRubyArgument(name: "version", type: nil) + let changelogArg = changelog.asRubyArgument(name: "changelog", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + appIdentifierArg, + usernameArg, + versionArg, + changelogArg, + teamIdArg, + teamNameArg, + platformArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "set_changelog", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This will create a new release on GitHub and upload assets for it + + - parameters: + - repositoryName: The path to your repo, e.g. 'fastlane/fastlane' + - serverUrl: The server url. e.g. 'https://your.internal.github.host/api/v3' (Default: 'https://api.github.com') + - apiToken: Personal API Token for GitHub - generate one at https://github.com/settings/tokens + - apiBearer: Use a Bearer authorization token. Usually generated by Github Apps, e.g. GitHub Actions GITHUB_TOKEN environment variable + - tagName: Pass in the tag name + - name: Name of this release + - commitish: Specifies the commitish value that determines where the Git tag is created from. Can be any branch or commit SHA. Unused if the Git tag already exists. Default: the repository's default branch (usually master) + - description: Description of this release + - isDraft: Whether the release should be marked as draft + - isPrerelease: Whether the release should be marked as prerelease + - isGenerateReleaseNotes: Whether the name and body of this release should be generated automatically + - uploadAssets: Path to assets to be uploaded with the release + + - returns: A hash containing all relevant information of this release + Access things like 'html_url', 'tag_name', 'name', 'body' + + Creates a new release on GitHub. You must provide your GitHub Personal token (get one from [https://github.com/settings/tokens/new](https://github.com/settings/tokens/new)), the repository name and tag name. By default, that's `master`. + If the tag doesn't exist, one will be created on the commit or branch passed in as commitish. + Out parameters provide the release's id, which can be used for later editing and the release HTML link to GitHub. You can also specify a list of assets to be uploaded to the release with the `:upload_assets` parameter. + */ +@discardableResult public func setGithubRelease(repositoryName: String, + serverUrl: String = "https://api.github.com", + apiToken: OptionalConfigValue = .fastlaneDefault(nil), + apiBearer: OptionalConfigValue = .fastlaneDefault(nil), + tagName: String, + name: OptionalConfigValue = .fastlaneDefault(nil), + commitish: OptionalConfigValue = .fastlaneDefault(nil), + description: OptionalConfigValue = .fastlaneDefault(nil), + isDraft: OptionalConfigValue = .fastlaneDefault(false), + isPrerelease: OptionalConfigValue = .fastlaneDefault(false), + isGenerateReleaseNotes: OptionalConfigValue = .fastlaneDefault(false), + uploadAssets: OptionalConfigValue<[String]?> = .fastlaneDefault(nil)) -> [String: Any] +{ + let repositoryNameArg = RubyCommand.Argument(name: "repository_name", value: repositoryName, type: nil) + let serverUrlArg = RubyCommand.Argument(name: "server_url", value: serverUrl, type: nil) + let apiTokenArg = apiToken.asRubyArgument(name: "api_token", type: nil) + let apiBearerArg = apiBearer.asRubyArgument(name: "api_bearer", type: nil) + let tagNameArg = RubyCommand.Argument(name: "tag_name", value: tagName, type: nil) + let nameArg = name.asRubyArgument(name: "name", type: nil) + let commitishArg = commitish.asRubyArgument(name: "commitish", type: nil) + let descriptionArg = description.asRubyArgument(name: "description", type: nil) + let isDraftArg = isDraft.asRubyArgument(name: "is_draft", type: nil) + let isPrereleaseArg = isPrerelease.asRubyArgument(name: "is_prerelease", type: nil) + let isGenerateReleaseNotesArg = isGenerateReleaseNotes.asRubyArgument(name: "is_generate_release_notes", type: nil) + let uploadAssetsArg = uploadAssets.asRubyArgument(name: "upload_assets", type: nil) + let array: [RubyCommand.Argument?] = [repositoryNameArg, + serverUrlArg, + apiTokenArg, + apiBearerArg, + tagNameArg, + nameArg, + commitishArg, + descriptionArg, + isDraftArg, + isPrereleaseArg, + isGenerateReleaseNotesArg, + uploadAssetsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "set_github_release", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Sets value to Info.plist of your project as native Ruby data structures + + - parameters: + - key: Name of key in plist + - subkey: Name of subkey in plist + - value: Value to setup + - path: Path to plist file you want to update + - outputFileName: Path to the output file you want to generate + */ +public func setInfoPlistValue(key: String, + subkey: OptionalConfigValue = .fastlaneDefault(nil), + value: String, + path: String, + outputFileName: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let keyArg = RubyCommand.Argument(name: "key", value: key, type: nil) + let subkeyArg = subkey.asRubyArgument(name: "subkey", type: nil) + let valueArg = RubyCommand.Argument(name: "value", value: value, type: nil) + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let outputFileNameArg = outputFileName.asRubyArgument(name: "output_file_name", type: nil) + let array: [RubyCommand.Argument?] = [keyArg, + subkeyArg, + valueArg, + pathArg, + outputFileNameArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "set_info_plist_value", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Sets a value for a key with cocoapods-keys + + - parameters: + - useBundleExec: Use bundle exec when there is a Gemfile presented + - key: The key to be saved with cocoapods-keys + - value: The value to be saved with cocoapods-keys + - project: The project name + + Adds a key to [cocoapods-keys](https://github.com/orta/cocoapods-keys) + */ +public func setPodKey(useBundleExec: OptionalConfigValue = .fastlaneDefault(true), + key: String, + value: String, + project: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let useBundleExecArg = useBundleExec.asRubyArgument(name: "use_bundle_exec", type: nil) + let keyArg = RubyCommand.Argument(name: "key", value: key, type: nil) + let valueArg = RubyCommand.Argument(name: "value", value: value, type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let array: [RubyCommand.Argument?] = [useBundleExecArg, + keyArg, + valueArg, + projectArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "set_pod_key", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Setup the keychain and match to work with CI + + - parameters: + - force: Force setup, even if not executed by CI + - provider: CI provider. If none is set, the provider is detected automatically + - timeout: Set a custom timeout in seconds for keychain. Set `0` if you want to specify 'no time-out' + + - Creates a new temporary keychain for use with match| + - Switches match to `readonly` mode to not create new profiles/cert on CI| + - Sets up log and test result paths to be easily collectible| + >| + This action helps with CI integration. Add this to the top of your Fastfile if you use CI. + */ +public func setupCi(force: OptionalConfigValue = .fastlaneDefault(false), + provider: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 3600) +{ + let forceArg = force.asRubyArgument(name: "force", type: nil) + let providerArg = provider.asRubyArgument(name: "provider", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [forceArg, + providerArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "setup_ci", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Setup the keychain and match to work with CircleCI + + - parameter force: Force setup, even if not executed by CircleCI + + - Creates a new temporary keychain for use with match| + - Switches match to `readonly` mode to not create new profiles/cert on CI| + - Sets up log and test result paths to be easily collectible| + >| + This action helps with CircleCI integration. Add this to the top of your Fastfile if you use CircleCI. + */ +public func setupCircleCi(force: OptionalConfigValue = .fastlaneDefault(false)) { + let forceArg = force.asRubyArgument(name: "force", type: nil) + let array: [RubyCommand.Argument?] = [forceArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "setup_circle_ci", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Setup xcodebuild, gym and scan for easier Jenkins integration + + - parameters: + - force: Force setup, even if not executed by Jenkins + - unlockKeychain: Unlocks keychain + - addKeychainToSearchList: Add to keychain search list, valid values are true, false, :add, and :replace + - setDefaultKeychain: Set keychain as default + - keychainPath: Path to keychain + - keychainPassword: Keychain password + - setCodeSigningIdentity: Set code signing identity from CODE_SIGNING_IDENTITY environment + - codeSigningIdentity: Code signing identity + - outputDirectory: The directory in which the ipa file should be stored in + - derivedDataPath: The directory where built products and other derived data will go + - resultBundle: Produce the result bundle describing what occurred will be placed + + - Adds and unlocks keychains from Jenkins 'Keychains and Provisioning Profiles Plugin'| + - Sets unlocked keychain to be used by Match| + - Sets code signing identity from Jenkins 'Keychains and Provisioning Profiles Plugin'| + - Sets output directory to './output' (gym, scan and backup_xcarchive)| + - Sets derived data path to './derivedData' (xcodebuild, gym, scan and clear_derived_data, carthage)| + - Produce result bundle (gym and scan)| + >| + This action helps with Jenkins integration. Creates own derived data for each job. All build results like IPA files and archives will be stored in the `./output` directory. + The action also works with [Keychains and Provisioning Profiles Plugin](https://wiki.jenkins-ci.org/display/JENKINS/Keychains+and+Provisioning+Profiles+Plugin), the selected keychain will be automatically unlocked and the selected code signing identity will be used. + [Match](https://docs.fastlane.tools/actions/match/) will be also set up to use the unlocked keychain and set in read-only mode, if its environment variables were not yet defined. + By default this action will only work when _fastlane_ is executed on a CI system. + */ +public func setupJenkins(force: OptionalConfigValue = .fastlaneDefault(false), + unlockKeychain: OptionalConfigValue = .fastlaneDefault(true), + addKeychainToSearchList: String = "replace", + setDefaultKeychain: OptionalConfigValue = .fastlaneDefault(true), + keychainPath: OptionalConfigValue = .fastlaneDefault(nil), + keychainPassword: String, + setCodeSigningIdentity: OptionalConfigValue = .fastlaneDefault(true), + codeSigningIdentity: OptionalConfigValue = .fastlaneDefault(nil), + outputDirectory: String = "./output", + derivedDataPath: String = "./derivedData", + resultBundle: OptionalConfigValue = .fastlaneDefault(true)) +{ + let forceArg = force.asRubyArgument(name: "force", type: nil) + let unlockKeychainArg = unlockKeychain.asRubyArgument(name: "unlock_keychain", type: nil) + let addKeychainToSearchListArg = RubyCommand.Argument(name: "add_keychain_to_search_list", value: addKeychainToSearchList, type: nil) + let setDefaultKeychainArg = setDefaultKeychain.asRubyArgument(name: "set_default_keychain", type: nil) + let keychainPathArg = keychainPath.asRubyArgument(name: "keychain_path", type: nil) + let keychainPasswordArg = RubyCommand.Argument(name: "keychain_password", value: keychainPassword, type: nil) + let setCodeSigningIdentityArg = setCodeSigningIdentity.asRubyArgument(name: "set_code_signing_identity", type: nil) + let codeSigningIdentityArg = codeSigningIdentity.asRubyArgument(name: "code_signing_identity", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let derivedDataPathArg = RubyCommand.Argument(name: "derived_data_path", value: derivedDataPath, type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let array: [RubyCommand.Argument?] = [forceArg, + unlockKeychainArg, + addKeychainToSearchListArg, + setDefaultKeychainArg, + keychainPathArg, + keychainPasswordArg, + setCodeSigningIdentityArg, + codeSigningIdentityArg, + outputDirectoryArg, + derivedDataPathArg, + resultBundleArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "setup_jenkins", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Setup the keychain and match to work with Travis CI + + - parameter force: Force setup, even if not executed by travis + + - Creates a new temporary keychain for use with match| + - Switches match to `readonly` mode to not create new profiles/cert on CI| + >| + This action helps with Travis integration. Add this to the top of your Fastfile if you use Travis. + */ +public func setupTravis(force: OptionalConfigValue = .fastlaneDefault(false)) { + let forceArg = force.asRubyArgument(name: "force", type: nil) + let array: [RubyCommand.Argument?] = [forceArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "setup_travis", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs a shell command + + - parameters: + - command: Shell command to be executed + - log: Determines whether fastlane should print out the executed command itself and output of the executed command. If command line option --troubleshoot is used, then it overrides this option to true + - errorCallback: A callback invoked with the command output if there is a non-zero exit status + + - returns: Outputs the string and executes it. When running in tests, it returns the actual command instead of executing it + + Allows running an arbitrary shell command. + Be aware of a specific behavior of `sh` action with regard to the working directory. For details, refer to [Advanced](https://docs.fastlane.tools/advanced/#directory-behavior). + */ +@discardableResult public func sh(command: String, + log: OptionalConfigValue = .fastlaneDefault(true), + errorCallback: ((String) -> Void)? = nil) -> String +{ + let commandArg = RubyCommand.Argument(name: "command", value: command, type: nil) + let logArg = log.asRubyArgument(name: "log", type: nil) + let errorCallbackArg = RubyCommand.Argument(name: "error_callback", value: errorCallback, type: .stringClosure) + let array: [RubyCommand.Argument?] = [commandArg, + logArg, + errorCallbackArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "sh", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Alias for the `get_provisioning_profile` action + + - parameters: + - adhoc: Setting this flag will generate AdHoc profiles instead of App Store Profiles + - developerId: Setting this flag will generate Developer ID profiles instead of App Store Profiles + - development: Renew the development certificate instead of the production one + - skipInstall: By default, the certificate will be added to your local machine. Setting this flag will skip this action + - force: Renew provisioning profiles regardless of its state - to automatically add all devices for ad hoc profiles + - includeMacInProfiles: Include Apple Silicon Mac devices in provisioning profiles for iOS/iPadOS apps + - appIdentifier: The bundle identifier of your app + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - provisioningName: The name of the profile that is used on the Apple Developer Portal + - ignoreProfilesWithDifferentName: Use in combination with :provisioning_name - when true only profiles matching this exact name will be downloaded + - outputPath: Directory in which the profile should be stored + - certId: The ID of the code signing certificate to use (e.g. 78ADL6LVAA) + - certOwnerName: The certificate name to use for new profiles, or to renew with. (e.g. "Felix Krause") + - filename: Filename to use for the generated provisioning profile (must include .mobileprovision) + - skipFetchProfiles: Skips the verification of existing profiles which is useful if you have thousands of profiles + - includeAllCertificates: Include all matching certificates in the provisioning profile. Works only for the 'development' provisioning profile type + - skipCertificateVerification: Skips the verification of the certificates for every existing profiles. This will make sure the provisioning profile can be used on the local machine + - platform: Set the provisioning profile's platform (i.e. ios, tvos, macos, catalyst) + - readonly: Only fetch existing profile, don't generate new ones + - templateName: The name of provisioning profile template. If the developer account has provisioning profile templates (aka: custom entitlements), the template name can be found by inspecting the Entitlements drop-down while creating/editing a provisioning profile (e.g. "Apple Pay Pass Suppression Development") + - failOnNameTaken: Should the command fail if it was about to create a duplicate of an existing provisioning profile. It can happen due to issues on Apple Developer Portal, when profile to be recreated was not properly deleted first + + - returns: The UUID of the profile sigh just fetched/generated + + **Note**: It is recommended to use [match](https://docs.fastlane.tools/actions/match/) according to the [codesigning.guide](https://codesigning.guide) for generating and maintaining your provisioning profiles. Use _sigh_ directly only if you want full control over what's going on and know more about codesigning. + */ +@discardableResult public func sigh(adhoc: OptionalConfigValue = .fastlaneDefault(false), + developerId: OptionalConfigValue = .fastlaneDefault(false), + development: OptionalConfigValue = .fastlaneDefault(false), + skipInstall: OptionalConfigValue = .fastlaneDefault(false), + force: OptionalConfigValue = .fastlaneDefault(false), + includeMacInProfiles: OptionalConfigValue = .fastlaneDefault(false), + appIdentifier: String, + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + provisioningName: OptionalConfigValue = .fastlaneDefault(nil), + ignoreProfilesWithDifferentName: OptionalConfigValue = .fastlaneDefault(false), + outputPath: String = ".", + certId: OptionalConfigValue = .fastlaneDefault(nil), + certOwnerName: OptionalConfigValue = .fastlaneDefault(nil), + filename: OptionalConfigValue = .fastlaneDefault(nil), + skipFetchProfiles: OptionalConfigValue = .fastlaneDefault(false), + includeAllCertificates: OptionalConfigValue = .fastlaneDefault(false), + skipCertificateVerification: OptionalConfigValue = .fastlaneDefault(false), + platform: Any = "ios", + readonly: OptionalConfigValue = .fastlaneDefault(false), + templateName: OptionalConfigValue = .fastlaneDefault(nil), + failOnNameTaken: OptionalConfigValue = .fastlaneDefault(false)) -> String +{ + let adhocArg = adhoc.asRubyArgument(name: "adhoc", type: nil) + let developerIdArg = developerId.asRubyArgument(name: "developer_id", type: nil) + let developmentArg = development.asRubyArgument(name: "development", type: nil) + let skipInstallArg = skipInstall.asRubyArgument(name: "skip_install", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let includeMacInProfilesArg = includeMacInProfiles.asRubyArgument(name: "include_mac_in_profiles", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let provisioningNameArg = provisioningName.asRubyArgument(name: "provisioning_name", type: nil) + let ignoreProfilesWithDifferentNameArg = ignoreProfilesWithDifferentName.asRubyArgument(name: "ignore_profiles_with_different_name", type: nil) + let outputPathArg = RubyCommand.Argument(name: "output_path", value: outputPath, type: nil) + let certIdArg = certId.asRubyArgument(name: "cert_id", type: nil) + let certOwnerNameArg = certOwnerName.asRubyArgument(name: "cert_owner_name", type: nil) + let filenameArg = filename.asRubyArgument(name: "filename", type: nil) + let skipFetchProfilesArg = skipFetchProfiles.asRubyArgument(name: "skip_fetch_profiles", type: nil) + let includeAllCertificatesArg = includeAllCertificates.asRubyArgument(name: "include_all_certificates", type: nil) + let skipCertificateVerificationArg = skipCertificateVerification.asRubyArgument(name: "skip_certificate_verification", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let readonlyArg = readonly.asRubyArgument(name: "readonly", type: nil) + let templateNameArg = templateName.asRubyArgument(name: "template_name", type: nil) + let failOnNameTakenArg = failOnNameTaken.asRubyArgument(name: "fail_on_name_taken", type: nil) + let array: [RubyCommand.Argument?] = [adhocArg, + developerIdArg, + developmentArg, + skipInstallArg, + forceArg, + includeMacInProfilesArg, + appIdentifierArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + provisioningNameArg, + ignoreProfilesWithDifferentNameArg, + outputPathArg, + certIdArg, + certOwnerNameArg, + filenameArg, + skipFetchProfilesArg, + includeAllCertificatesArg, + skipCertificateVerificationArg, + platformArg, + readonlyArg, + templateNameArg, + failOnNameTakenArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "sigh", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Skip the creation of the fastlane/README.md file when running fastlane + + Tell _fastlane_ to not automatically create a `fastlane/README.md` when running _fastlane_. You can always trigger the creation of this file manually by running `fastlane docs`. + */ +public func skipDocs() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "skip_docs", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Send a success/error message to your [Slack](https://slack.com) group + + - parameters: + - message: The message that should be displayed on Slack. This supports the standard Slack markup language + - pretext: This is optional text that appears above the message attachment block. This supports the standard Slack markup language + - channel: #channel or @username + - useWebhookConfiguredUsernameAndIcon: Use webhook's default username and icon settings? (true/false) + - slackUrl: Create an Incoming WebHook for your Slack group + - username: Overrides the webhook's username property if use_webhook_configured_username_and_icon is false + - iconUrl: Overrides the webhook's image property if use_webhook_configured_username_and_icon is false + - payload: Add additional information to this post. payload must be a hash containing any key with any value + - defaultPayloads: Specifies default payloads to include. Pass an empty array to suppress all the default payloads + - attachmentProperties: Merge additional properties in the slack attachment, see https://api.slack.com/docs/attachments + - success: Was this build successful? (true/false) + - failOnError: Should an error sending the slack notification cause a failure? (true/false) + - linkNames: Find and link channel names and usernames (true/false) + + Create an Incoming WebHook and export this as `SLACK_URL`. Can send a message to **#channel** (by default), a direct message to **@username** or a message to a private group **group** with success (green) or failure (red) status. + */ +public func slack(message: OptionalConfigValue = .fastlaneDefault(nil), + pretext: OptionalConfigValue = .fastlaneDefault(nil), + channel: OptionalConfigValue = .fastlaneDefault(nil), + useWebhookConfiguredUsernameAndIcon: OptionalConfigValue = .fastlaneDefault(false), + slackUrl: String, + username: String = "fastlane", + iconUrl: String = "https://fastlane.tools/assets/img/fastlane_icon.png", + payload: [String: Any] = [:], + defaultPayloads: [String] = ["lane", "test_result", "git_branch", "git_author", "last_git_commit", "last_git_commit_hash"], + attachmentProperties: [String: Any] = [:], + success: OptionalConfigValue = .fastlaneDefault(true), + failOnError: OptionalConfigValue = .fastlaneDefault(true), + linkNames: OptionalConfigValue = .fastlaneDefault(false)) +{ + let messageArg = message.asRubyArgument(name: "message", type: nil) + let pretextArg = pretext.asRubyArgument(name: "pretext", type: nil) + let channelArg = channel.asRubyArgument(name: "channel", type: nil) + let useWebhookConfiguredUsernameAndIconArg = useWebhookConfiguredUsernameAndIcon.asRubyArgument(name: "use_webhook_configured_username_and_icon", type: nil) + let slackUrlArg = RubyCommand.Argument(name: "slack_url", value: slackUrl, type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let iconUrlArg = RubyCommand.Argument(name: "icon_url", value: iconUrl, type: nil) + let payloadArg = RubyCommand.Argument(name: "payload", value: payload, type: nil) + let defaultPayloadsArg = RubyCommand.Argument(name: "default_payloads", value: defaultPayloads, type: nil) + let attachmentPropertiesArg = RubyCommand.Argument(name: "attachment_properties", value: attachmentProperties, type: nil) + let successArg = success.asRubyArgument(name: "success", type: nil) + let failOnErrorArg = failOnError.asRubyArgument(name: "fail_on_error", type: nil) + let linkNamesArg = linkNames.asRubyArgument(name: "link_names", type: nil) + let array: [RubyCommand.Argument?] = [messageArg, + pretextArg, + channelArg, + useWebhookConfiguredUsernameAndIconArg, + slackUrlArg, + usernameArg, + iconUrlArg, + payloadArg, + defaultPayloadsArg, + attachmentPropertiesArg, + successArg, + failOnErrorArg, + linkNamesArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "slack", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Show a train of the fastlane progress + + - returns: A string that is being sent to slack + */ +public func slackTrain() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "slack_train", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + + */ +public func slackTrainCrash() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "slack_train_crash", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Show a train of the fastlane progress + + - parameters: + - distance: How many rails do we need? + - train: Train emoji + - rail: Character or emoji for the rail + - reverseDirection: Pass true if you want the train to go from left to right + */ +public func slackTrainStart(distance: Int = 5, + train: String = "🚝", + rail: String = "=", + reverseDirection: OptionalConfigValue = .fastlaneDefault(false)) +{ + let distanceArg = RubyCommand.Argument(name: "distance", value: distance, type: nil) + let trainArg = RubyCommand.Argument(name: "train", value: train, type: nil) + let railArg = RubyCommand.Argument(name: "rail", value: rail, type: nil) + let reverseDirectionArg = reverseDirection.asRubyArgument(name: "reverse_direction", type: nil) + let array: [RubyCommand.Argument?] = [distanceArg, + trainArg, + railArg, + reverseDirectionArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "slack_train_start", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Use slather to generate a code coverage report + + - parameters: + - buildDirectory: The location of the build output + - proj: The project file that slather looks at + - workspace: The workspace that slather looks at + - scheme: Scheme to use when calling slather + - configuration: Configuration to use when calling slather (since slather-2.4.1) + - inputFormat: The input format that slather should look for + - github: Tell slather that it is running on Github Actions + - buildkite: Tell slather that it is running on Buildkite + - teamcity: Tell slather that it is running on TeamCity + - jenkins: Tell slather that it is running on Jenkins + - travis: Tell slather that it is running on TravisCI + - travisPro: Tell slather that it is running on TravisCI Pro + - circleci: Tell slather that it is running on CircleCI + - coveralls: Tell slather that it should post data to Coveralls + - simpleOutput: Tell slather that it should output results to the terminal + - gutterJson: Tell slather that it should output results as Gutter JSON format + - coberturaXml: Tell slather that it should output results as Cobertura XML format + - sonarqubeXml: Tell slather that it should output results as SonarQube Generic XML format + - llvmCov: Tell slather that it should output results as llvm-cov show format + - json: Tell slather that it should output results as static JSON report + - html: Tell slather that it should output results as static HTML pages + - show: Tell slather that it should open static html pages automatically + - sourceDirectory: Tell slather the location of your source files + - outputDirectory: Tell slather the location of for your output files + - ignore: Tell slather to ignore files matching a path or any path from an array of paths + - verbose: Tell slather to enable verbose mode + - useBundleExec: Use bundle exec to execute slather. Make sure it is in the Gemfile + - binaryBasename: Basename of the binary file, this should match the name of your bundle excluding its extension (i.e. YourApp [for YourApp.app bundle]) + - binaryFile: Binary file name to be used for code coverage + - arch: Specify which architecture the binary file is in. Needed for universal binaries + - sourceFiles: A Dir.glob compatible pattern used to limit the lookup to specific source files. Ignored in gcov mode + - decimals: The amount of decimals to use for % coverage reporting + + Slather works with multiple code coverage formats, including Xcode 7 code coverage. + Slather is available at [https://github.com/SlatherOrg/slather](https://github.com/SlatherOrg/slather). + */ +public func slather(buildDirectory: OptionalConfigValue = .fastlaneDefault(nil), + proj: OptionalConfigValue = .fastlaneDefault(nil), + workspace: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + inputFormat: OptionalConfigValue = .fastlaneDefault(nil), + github: OptionalConfigValue = .fastlaneDefault(nil), + buildkite: OptionalConfigValue = .fastlaneDefault(nil), + teamcity: OptionalConfigValue = .fastlaneDefault(nil), + jenkins: OptionalConfigValue = .fastlaneDefault(nil), + travis: OptionalConfigValue = .fastlaneDefault(nil), + travisPro: OptionalConfigValue = .fastlaneDefault(nil), + circleci: OptionalConfigValue = .fastlaneDefault(nil), + coveralls: OptionalConfigValue = .fastlaneDefault(nil), + simpleOutput: OptionalConfigValue = .fastlaneDefault(nil), + gutterJson: OptionalConfigValue = .fastlaneDefault(nil), + coberturaXml: OptionalConfigValue = .fastlaneDefault(nil), + sonarqubeXml: OptionalConfigValue = .fastlaneDefault(nil), + llvmCov: OptionalConfigValue = .fastlaneDefault(nil), + json: OptionalConfigValue = .fastlaneDefault(nil), + html: OptionalConfigValue = .fastlaneDefault(nil), + show: OptionalConfigValue = .fastlaneDefault(false), + sourceDirectory: OptionalConfigValue = .fastlaneDefault(nil), + outputDirectory: OptionalConfigValue = .fastlaneDefault(nil), + ignore: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(nil), + useBundleExec: OptionalConfigValue = .fastlaneDefault(false), + binaryBasename: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + binaryFile: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + arch: OptionalConfigValue = .fastlaneDefault(nil), + sourceFiles: OptionalConfigValue = .fastlaneDefault(false), + decimals: OptionalConfigValue = .fastlaneDefault(false)) +{ + let buildDirectoryArg = buildDirectory.asRubyArgument(name: "build_directory", type: nil) + let projArg = proj.asRubyArgument(name: "proj", type: nil) + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let inputFormatArg = inputFormat.asRubyArgument(name: "input_format", type: nil) + let githubArg = github.asRubyArgument(name: "github", type: nil) + let buildkiteArg = buildkite.asRubyArgument(name: "buildkite", type: nil) + let teamcityArg = teamcity.asRubyArgument(name: "teamcity", type: nil) + let jenkinsArg = jenkins.asRubyArgument(name: "jenkins", type: nil) + let travisArg = travis.asRubyArgument(name: "travis", type: nil) + let travisProArg = travisPro.asRubyArgument(name: "travis_pro", type: nil) + let circleciArg = circleci.asRubyArgument(name: "circleci", type: nil) + let coverallsArg = coveralls.asRubyArgument(name: "coveralls", type: nil) + let simpleOutputArg = simpleOutput.asRubyArgument(name: "simple_output", type: nil) + let gutterJsonArg = gutterJson.asRubyArgument(name: "gutter_json", type: nil) + let coberturaXmlArg = coberturaXml.asRubyArgument(name: "cobertura_xml", type: nil) + let sonarqubeXmlArg = sonarqubeXml.asRubyArgument(name: "sonarqube_xml", type: nil) + let llvmCovArg = llvmCov.asRubyArgument(name: "llvm_cov", type: nil) + let jsonArg = json.asRubyArgument(name: "json", type: nil) + let htmlArg = html.asRubyArgument(name: "html", type: nil) + let showArg = show.asRubyArgument(name: "show", type: nil) + let sourceDirectoryArg = sourceDirectory.asRubyArgument(name: "source_directory", type: nil) + let outputDirectoryArg = outputDirectory.asRubyArgument(name: "output_directory", type: nil) + let ignoreArg = ignore.asRubyArgument(name: "ignore", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let useBundleExecArg = useBundleExec.asRubyArgument(name: "use_bundle_exec", type: nil) + let binaryBasenameArg = binaryBasename.asRubyArgument(name: "binary_basename", type: nil) + let binaryFileArg = binaryFile.asRubyArgument(name: "binary_file", type: nil) + let archArg = arch.asRubyArgument(name: "arch", type: nil) + let sourceFilesArg = sourceFiles.asRubyArgument(name: "source_files", type: nil) + let decimalsArg = decimals.asRubyArgument(name: "decimals", type: nil) + let array: [RubyCommand.Argument?] = [buildDirectoryArg, + projArg, + workspaceArg, + schemeArg, + configurationArg, + inputFormatArg, + githubArg, + buildkiteArg, + teamcityArg, + jenkinsArg, + travisArg, + travisProArg, + circleciArg, + coverallsArg, + simpleOutputArg, + gutterJsonArg, + coberturaXmlArg, + sonarqubeXmlArg, + llvmCovArg, + jsonArg, + htmlArg, + showArg, + sourceDirectoryArg, + outputDirectoryArg, + ignoreArg, + verboseArg, + useBundleExecArg, + binaryBasenameArg, + binaryFileArg, + archArg, + sourceFilesArg, + decimalsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "slather", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `capture_ios_screenshots` action + + - parameters: + - workspace: Path the workspace file + - project: Path the project file + - xcargs: Pass additional arguments to xcodebuild for the test phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + - xcconfig: Use an extra XCCONFIG file to build your app + - devices: A list of devices you want to take the screenshots from + - languages: A list of languages which should be used + - launchArguments: A list of launch arguments which should be used + - outputDirectory: The directory where to store the screenshots + - outputSimulatorLogs: If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + - iosVersion: By default, the latest version should be used automatically. If you want to change it, do it here + - skipOpenSummary: Don't open the HTML summary after running _snapshot_ + - skipHelperVersionCheck: Do not check for most recent SnapshotHelper code + - clearPreviousScreenshots: Enabling this option will automatically clear previously generated screenshots before running snapshot + - reinstallApp: Enabling this option will automatically uninstall the application before running it + - eraseSimulator: Enabling this option will automatically erase the simulator before running the application + - headless: Enabling this option will prevent displaying the simulator window + - overrideStatusBar: Enabling this option will automatically override the status bar to show 9:41 AM, full battery, and full reception (Adjust 'SNAPSHOT_SIMULATOR_WAIT_FOR_BOOT_TIMEOUT' environment variable if override status bar is not working. Might be because simulator is not fully booted. Defaults to 10 seconds) + - overrideStatusBarArguments: Fully customize the status bar by setting each option here. Requires `override_status_bar` to be set to `true`. See `xcrun simctl status_bar --help` + - localizeSimulator: Enabling this option will configure the Simulator's system language + - darkMode: Enabling this option will configure the Simulator to be in dark mode (false for light, true for dark) + - appIdentifier: The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + - addPhotos: A list of photos that should be added to the simulator before running the application + - addVideos: A list of videos that should be added to the simulator before running the application + - htmlTemplate: A path to screenshots.html template + - buildlogPath: The directory where to store the build log + - clean: Should the project be cleaned before building it? + - testWithoutBuilding: Test without building, requires a derived data path + - configuration: The configuration to use when building the app. Defaults to 'Release' + - sdk: The SDK that should be used for building the application + - scheme: The scheme you want to use, this must be the scheme for the UI Tests + - numberOfRetries: The number of times a test can fail before snapshot should stop retrying + - stopAfterFirstError: Should snapshot stop immediately after the tests completely failed on one device? + - derivedDataPath: The directory where build products and other derived data will go + - resultBundle: Should an Xcode result bundle be generated in the output directory + - testTargetName: The name of the target you want to test (if you desire to override the Target Application from Xcode) + - namespaceLogFiles: Separate the log files per device and per language + - concurrentSimulators: Take snapshots on multiple simulators concurrently. Note: This option is only applicable when running against Xcode 9 + - disableSlideToType: Disable the simulator from showing the 'Slide to type' prompt + - clonedSourcePackagesPath: Sets a custom path for Swift Package Manager dependencies + - skipPackageDependenciesResolution: Skips resolution of Swift Package Manager dependencies + - disablePackageAutomaticUpdates: Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + - testplan: The testplan associated with the scheme that should be used for testing + - onlyTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to run + - skipTesting: Array of strings matching Test Bundle/Test Suite/Test Cases to skip + - xcodebuildFormatter: xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + - xcprettyArgs: **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Additional xcpretty arguments + - disableXcpretty: Disable xcpretty formatting of build + - suppressXcodeOutput: Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + - useSystemScm: Lets xcodebuild use system's scm configuration + */ +public func snapshot(workspace: OptionalConfigValue = .fastlaneDefault(snapshotfile.workspace), + project: OptionalConfigValue = .fastlaneDefault(snapshotfile.project), + xcargs: OptionalConfigValue = .fastlaneDefault(snapshotfile.xcargs), + xcconfig: OptionalConfigValue = .fastlaneDefault(snapshotfile.xcconfig), + devices: OptionalConfigValue<[String]?> = .fastlaneDefault(snapshotfile.devices), + languages: [String] = snapshotfile.languages, + launchArguments: [String] = snapshotfile.launchArguments, + outputDirectory: String = snapshotfile.outputDirectory, + outputSimulatorLogs: OptionalConfigValue = .fastlaneDefault(snapshotfile.outputSimulatorLogs), + iosVersion: OptionalConfigValue = .fastlaneDefault(snapshotfile.iosVersion), + skipOpenSummary: OptionalConfigValue = .fastlaneDefault(snapshotfile.skipOpenSummary), + skipHelperVersionCheck: OptionalConfigValue = .fastlaneDefault(snapshotfile.skipHelperVersionCheck), + clearPreviousScreenshots: OptionalConfigValue = .fastlaneDefault(snapshotfile.clearPreviousScreenshots), + reinstallApp: OptionalConfigValue = .fastlaneDefault(snapshotfile.reinstallApp), + eraseSimulator: OptionalConfigValue = .fastlaneDefault(snapshotfile.eraseSimulator), + headless: OptionalConfigValue = .fastlaneDefault(snapshotfile.headless), + overrideStatusBar: OptionalConfigValue = .fastlaneDefault(snapshotfile.overrideStatusBar), + overrideStatusBarArguments: OptionalConfigValue = .fastlaneDefault(snapshotfile.overrideStatusBarArguments), + localizeSimulator: OptionalConfigValue = .fastlaneDefault(snapshotfile.localizeSimulator), + darkMode: OptionalConfigValue = .fastlaneDefault(snapshotfile.darkMode), + appIdentifier: OptionalConfigValue = .fastlaneDefault(snapshotfile.appIdentifier), + addPhotos: OptionalConfigValue<[String]?> = .fastlaneDefault(snapshotfile.addPhotos), + addVideos: OptionalConfigValue<[String]?> = .fastlaneDefault(snapshotfile.addVideos), + htmlTemplate: OptionalConfigValue = .fastlaneDefault(snapshotfile.htmlTemplate), + buildlogPath: String = snapshotfile.buildlogPath, + clean: OptionalConfigValue = .fastlaneDefault(snapshotfile.clean), + testWithoutBuilding: OptionalConfigValue = .fastlaneDefault(snapshotfile.testWithoutBuilding), + configuration: OptionalConfigValue = .fastlaneDefault(snapshotfile.configuration), + sdk: OptionalConfigValue = .fastlaneDefault(snapshotfile.sdk), + scheme: OptionalConfigValue = .fastlaneDefault(snapshotfile.scheme), + numberOfRetries: Int = snapshotfile.numberOfRetries, + stopAfterFirstError: OptionalConfigValue = .fastlaneDefault(snapshotfile.stopAfterFirstError), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(snapshotfile.derivedDataPath), + resultBundle: OptionalConfigValue = .fastlaneDefault(snapshotfile.resultBundle), + testTargetName: OptionalConfigValue = .fastlaneDefault(snapshotfile.testTargetName), + namespaceLogFiles: Any? = snapshotfile.namespaceLogFiles, + concurrentSimulators: OptionalConfigValue = .fastlaneDefault(snapshotfile.concurrentSimulators), + disableSlideToType: OptionalConfigValue = .fastlaneDefault(snapshotfile.disableSlideToType), + clonedSourcePackagesPath: OptionalConfigValue = .fastlaneDefault(snapshotfile.clonedSourcePackagesPath), + skipPackageDependenciesResolution: OptionalConfigValue = .fastlaneDefault(snapshotfile.skipPackageDependenciesResolution), + disablePackageAutomaticUpdates: OptionalConfigValue = .fastlaneDefault(snapshotfile.disablePackageAutomaticUpdates), + testplan: OptionalConfigValue = .fastlaneDefault(snapshotfile.testplan), + onlyTesting: Any? = snapshotfile.onlyTesting, + skipTesting: Any? = snapshotfile.skipTesting, + xcodebuildFormatter: String = snapshotfile.xcodebuildFormatter, + xcprettyArgs: OptionalConfigValue = .fastlaneDefault(snapshotfile.xcprettyArgs), + disableXcpretty: OptionalConfigValue = .fastlaneDefault(snapshotfile.disableXcpretty), + suppressXcodeOutput: OptionalConfigValue = .fastlaneDefault(snapshotfile.suppressXcodeOutput), + useSystemScm: OptionalConfigValue = .fastlaneDefault(snapshotfile.useSystemScm)) +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let xcargsArg = xcargs.asRubyArgument(name: "xcargs", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let devicesArg = devices.asRubyArgument(name: "devices", type: nil) + let languagesArg = RubyCommand.Argument(name: "languages", value: languages, type: nil) + let launchArgumentsArg = RubyCommand.Argument(name: "launch_arguments", value: launchArguments, type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let outputSimulatorLogsArg = outputSimulatorLogs.asRubyArgument(name: "output_simulator_logs", type: nil) + let iosVersionArg = iosVersion.asRubyArgument(name: "ios_version", type: nil) + let skipOpenSummaryArg = skipOpenSummary.asRubyArgument(name: "skip_open_summary", type: nil) + let skipHelperVersionCheckArg = skipHelperVersionCheck.asRubyArgument(name: "skip_helper_version_check", type: nil) + let clearPreviousScreenshotsArg = clearPreviousScreenshots.asRubyArgument(name: "clear_previous_screenshots", type: nil) + let reinstallAppArg = reinstallApp.asRubyArgument(name: "reinstall_app", type: nil) + let eraseSimulatorArg = eraseSimulator.asRubyArgument(name: "erase_simulator", type: nil) + let headlessArg = headless.asRubyArgument(name: "headless", type: nil) + let overrideStatusBarArg = overrideStatusBar.asRubyArgument(name: "override_status_bar", type: nil) + let overrideStatusBarArgumentsArg = overrideStatusBarArguments.asRubyArgument(name: "override_status_bar_arguments", type: nil) + let localizeSimulatorArg = localizeSimulator.asRubyArgument(name: "localize_simulator", type: nil) + let darkModeArg = darkMode.asRubyArgument(name: "dark_mode", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let addPhotosArg = addPhotos.asRubyArgument(name: "add_photos", type: nil) + let addVideosArg = addVideos.asRubyArgument(name: "add_videos", type: nil) + let htmlTemplateArg = htmlTemplate.asRubyArgument(name: "html_template", type: nil) + let buildlogPathArg = RubyCommand.Argument(name: "buildlog_path", value: buildlogPath, type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let testWithoutBuildingArg = testWithoutBuilding.asRubyArgument(name: "test_without_building", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let numberOfRetriesArg = RubyCommand.Argument(name: "number_of_retries", value: numberOfRetries, type: nil) + let stopAfterFirstErrorArg = stopAfterFirstError.asRubyArgument(name: "stop_after_first_error", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let resultBundleArg = resultBundle.asRubyArgument(name: "result_bundle", type: nil) + let testTargetNameArg = testTargetName.asRubyArgument(name: "test_target_name", type: nil) + let namespaceLogFilesArg = RubyCommand.Argument(name: "namespace_log_files", value: namespaceLogFiles, type: nil) + let concurrentSimulatorsArg = concurrentSimulators.asRubyArgument(name: "concurrent_simulators", type: nil) + let disableSlideToTypeArg = disableSlideToType.asRubyArgument(name: "disable_slide_to_type", type: nil) + let clonedSourcePackagesPathArg = clonedSourcePackagesPath.asRubyArgument(name: "cloned_source_packages_path", type: nil) + let skipPackageDependenciesResolutionArg = skipPackageDependenciesResolution.asRubyArgument(name: "skip_package_dependencies_resolution", type: nil) + let disablePackageAutomaticUpdatesArg = disablePackageAutomaticUpdates.asRubyArgument(name: "disable_package_automatic_updates", type: nil) + let testplanArg = testplan.asRubyArgument(name: "testplan", type: nil) + let onlyTestingArg = RubyCommand.Argument(name: "only_testing", value: onlyTesting, type: nil) + let skipTestingArg = RubyCommand.Argument(name: "skip_testing", value: skipTesting, type: nil) + let xcodebuildFormatterArg = RubyCommand.Argument(name: "xcodebuild_formatter", value: xcodebuildFormatter, type: nil) + let xcprettyArgsArg = xcprettyArgs.asRubyArgument(name: "xcpretty_args", type: nil) + let disableXcprettyArg = disableXcpretty.asRubyArgument(name: "disable_xcpretty", type: nil) + let suppressXcodeOutputArg = suppressXcodeOutput.asRubyArgument(name: "suppress_xcode_output", type: nil) + let useSystemScmArg = useSystemScm.asRubyArgument(name: "use_system_scm", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + xcargsArg, + xcconfigArg, + devicesArg, + languagesArg, + launchArgumentsArg, + outputDirectoryArg, + outputSimulatorLogsArg, + iosVersionArg, + skipOpenSummaryArg, + skipHelperVersionCheckArg, + clearPreviousScreenshotsArg, + reinstallAppArg, + eraseSimulatorArg, + headlessArg, + overrideStatusBarArg, + overrideStatusBarArgumentsArg, + localizeSimulatorArg, + darkModeArg, + appIdentifierArg, + addPhotosArg, + addVideosArg, + htmlTemplateArg, + buildlogPathArg, + cleanArg, + testWithoutBuildingArg, + configurationArg, + sdkArg, + schemeArg, + numberOfRetriesArg, + stopAfterFirstErrorArg, + derivedDataPathArg, + resultBundleArg, + testTargetNameArg, + namespaceLogFilesArg, + concurrentSimulatorsArg, + disableSlideToTypeArg, + clonedSourcePackagesPathArg, + skipPackageDependenciesResolutionArg, + disablePackageAutomaticUpdatesArg, + testplanArg, + onlyTestingArg, + skipTestingArg, + xcodebuildFormatterArg, + xcprettyArgsArg, + disableXcprettyArg, + suppressXcodeOutputArg, + useSystemScmArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "snapshot", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Invokes sonar-scanner to programmatically run SonarQube analysis + + - parameters: + - projectConfigurationPath: The path to your sonar project configuration file; defaults to `sonar-project.properties` + - projectKey: The key sonar uses to identify the project, e.g. `name.gretzki.awesomeApp`. Must either be specified here or inside the sonar project configuration file + - projectName: The name of the project that gets displayed on the sonar report page. Must either be specified here or inside the sonar project configuration file + - projectVersion: The project's version that gets displayed on the sonar report page. Must either be specified here or inside the sonar project configuration file + - sourcesPath: Comma-separated paths to directories containing source files. Must either be specified here or inside the sonar project configuration file + - exclusions: Comma-separated paths to directories to be excluded from the analysis + - projectLanguage: Language key, e.g. objc + - sourceEncoding: Used encoding of source files, e.g., UTF-8 + - sonarRunnerArgs: Pass additional arguments to sonar-scanner. Be sure to provide the arguments with a leading `-D` e.g. FL_SONAR_RUNNER_ARGS="-Dsonar.verbose=true" + - sonarLogin: Pass the Sonar Login token (e.g: xxxxxxprivate_token_XXXXbXX7e) + - sonarUrl: Pass the url of the Sonar server + - sonarOrganization: Key of the organization on SonarCloud + - branchName: Pass the branch name which is getting scanned + - pullRequestBranch: The name of the branch that contains the changes to be merged + - pullRequestBase: The long-lived branch into which the PR will be merged + - pullRequestKey: Unique identifier of your PR. Must correspond to the key of the PR in GitHub or TFS + + - returns: The exit code of the sonar-scanner binary + + See [http://docs.sonarqube.org/display/SCAN/Analyzing+with+SonarQube+Scanner](http://docs.sonarqube.org/display/SCAN/Analyzing+with+SonarQube+Scanner) for details. + It can process unit test results if formatted as junit report as shown in [xctest](https://docs.fastlane.tools/actions/xctest/) action. It can also integrate coverage reports in Cobertura format, which can be transformed into by the [slather](https://docs.fastlane.tools/actions/slather/) action. + */ +public func sonar(projectConfigurationPath: OptionalConfigValue = .fastlaneDefault(nil), + projectKey: OptionalConfigValue = .fastlaneDefault(nil), + projectName: OptionalConfigValue = .fastlaneDefault(nil), + projectVersion: OptionalConfigValue = .fastlaneDefault(nil), + sourcesPath: OptionalConfigValue = .fastlaneDefault(nil), + exclusions: OptionalConfigValue = .fastlaneDefault(nil), + projectLanguage: OptionalConfigValue = .fastlaneDefault(nil), + sourceEncoding: OptionalConfigValue = .fastlaneDefault(nil), + sonarRunnerArgs: OptionalConfigValue = .fastlaneDefault(nil), + sonarLogin: OptionalConfigValue = .fastlaneDefault(nil), + sonarUrl: OptionalConfigValue = .fastlaneDefault(nil), + sonarOrganization: OptionalConfigValue = .fastlaneDefault(nil), + branchName: OptionalConfigValue = .fastlaneDefault(nil), + pullRequestBranch: OptionalConfigValue = .fastlaneDefault(nil), + pullRequestBase: OptionalConfigValue = .fastlaneDefault(nil), + pullRequestKey: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let projectConfigurationPathArg = projectConfigurationPath.asRubyArgument(name: "project_configuration_path", type: nil) + let projectKeyArg = projectKey.asRubyArgument(name: "project_key", type: nil) + let projectNameArg = projectName.asRubyArgument(name: "project_name", type: nil) + let projectVersionArg = projectVersion.asRubyArgument(name: "project_version", type: nil) + let sourcesPathArg = sourcesPath.asRubyArgument(name: "sources_path", type: nil) + let exclusionsArg = exclusions.asRubyArgument(name: "exclusions", type: nil) + let projectLanguageArg = projectLanguage.asRubyArgument(name: "project_language", type: nil) + let sourceEncodingArg = sourceEncoding.asRubyArgument(name: "source_encoding", type: nil) + let sonarRunnerArgsArg = sonarRunnerArgs.asRubyArgument(name: "sonar_runner_args", type: nil) + let sonarLoginArg = sonarLogin.asRubyArgument(name: "sonar_login", type: nil) + let sonarUrlArg = sonarUrl.asRubyArgument(name: "sonar_url", type: nil) + let sonarOrganizationArg = sonarOrganization.asRubyArgument(name: "sonar_organization", type: nil) + let branchNameArg = branchName.asRubyArgument(name: "branch_name", type: nil) + let pullRequestBranchArg = pullRequestBranch.asRubyArgument(name: "pull_request_branch", type: nil) + let pullRequestBaseArg = pullRequestBase.asRubyArgument(name: "pull_request_base", type: nil) + let pullRequestKeyArg = pullRequestKey.asRubyArgument(name: "pull_request_key", type: nil) + let array: [RubyCommand.Argument?] = [projectConfigurationPathArg, + projectKeyArg, + projectNameArg, + projectVersionArg, + sourcesPathArg, + exclusionsArg, + projectLanguageArg, + sourceEncodingArg, + sonarRunnerArgsArg, + sonarLoginArg, + sonarUrlArg, + sonarOrganizationArg, + branchNameArg, + pullRequestBranchArg, + pullRequestBaseArg, + pullRequestKeyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "sonar", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Generate docs using SourceDocs + + - parameters: + - allModules: Generate documentation for all modules in a Swift package + - spmModule: Generate documentation for Swift Package Manager module + - moduleName: Generate documentation for a Swift module + - linkBeginning: The text to begin links with + - linkEnding: The text to end links with (default: .md) + - outputFolder: Output directory to clean (default: Documentation/Reference) + - minAcl: Access level to include in documentation [private, fileprivate, internal, public, open] (default: public) + - moduleNamePath: Include the module name as part of the output folder path + - clean: Delete output folder before generating documentation + - collapsible: Put methods, properties and enum cases inside collapsible blocks + - tableOfContents: Generate a table of contents with properties and methods for each type + - reproducible: Generate documentation that is reproducible: only depends on the sources + - scheme: Create documentation for specific scheme + - sdkPlatform: Create documentation for specific sdk platform + */ +public func sourcedocs(allModules: OptionalConfigValue = .fastlaneDefault(nil), + spmModule: OptionalConfigValue = .fastlaneDefault(nil), + moduleName: OptionalConfigValue = .fastlaneDefault(nil), + linkBeginning: OptionalConfigValue = .fastlaneDefault(nil), + linkEnding: OptionalConfigValue = .fastlaneDefault(nil), + outputFolder: String, + minAcl: OptionalConfigValue = .fastlaneDefault(nil), + moduleNamePath: OptionalConfigValue = .fastlaneDefault(nil), + clean: OptionalConfigValue = .fastlaneDefault(nil), + collapsible: OptionalConfigValue = .fastlaneDefault(nil), + tableOfContents: OptionalConfigValue = .fastlaneDefault(nil), + reproducible: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + sdkPlatform: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let allModulesArg = allModules.asRubyArgument(name: "all_modules", type: nil) + let spmModuleArg = spmModule.asRubyArgument(name: "spm_module", type: nil) + let moduleNameArg = moduleName.asRubyArgument(name: "module_name", type: nil) + let linkBeginningArg = linkBeginning.asRubyArgument(name: "link_beginning", type: nil) + let linkEndingArg = linkEnding.asRubyArgument(name: "link_ending", type: nil) + let outputFolderArg = RubyCommand.Argument(name: "output_folder", value: outputFolder, type: nil) + let minAclArg = minAcl.asRubyArgument(name: "min_acl", type: nil) + let moduleNamePathArg = moduleNamePath.asRubyArgument(name: "module_name_path", type: nil) + let cleanArg = clean.asRubyArgument(name: "clean", type: nil) + let collapsibleArg = collapsible.asRubyArgument(name: "collapsible", type: nil) + let tableOfContentsArg = tableOfContents.asRubyArgument(name: "table_of_contents", type: nil) + let reproducibleArg = reproducible.asRubyArgument(name: "reproducible", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let sdkPlatformArg = sdkPlatform.asRubyArgument(name: "sdk_platform", type: nil) + let array: [RubyCommand.Argument?] = [allModulesArg, + spmModuleArg, + moduleNameArg, + linkBeginningArg, + linkEndingArg, + outputFolderArg, + minAclArg, + moduleNamePathArg, + cleanArg, + collapsibleArg, + tableOfContentsArg, + reproducibleArg, + schemeArg, + sdkPlatformArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "sourcedocs", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Find, print, and copy Spaceship logs + + - parameters: + - latest: Finds only the latest Spaceshop log file if set to true, otherwise returns all + - printContents: Prints the contents of the found Spaceship log file(s) + - printPaths: Prints the paths of the found Spaceship log file(s) + - copyToPath: Copies the found Spaceship log file(s) to a directory + - copyToClipboard: Copies the contents of the found Spaceship log file(s) to the clipboard + + - returns: The array of Spaceship logs + */ +@discardableResult public func spaceshipLogs(latest: OptionalConfigValue = .fastlaneDefault(true), + printContents: OptionalConfigValue = .fastlaneDefault(false), + printPaths: OptionalConfigValue = .fastlaneDefault(false), + copyToPath: OptionalConfigValue = .fastlaneDefault(nil), + copyToClipboard: OptionalConfigValue = .fastlaneDefault(false)) -> [String] +{ + let latestArg = latest.asRubyArgument(name: "latest", type: nil) + let printContentsArg = printContents.asRubyArgument(name: "print_contents", type: nil) + let printPathsArg = printPaths.asRubyArgument(name: "print_paths", type: nil) + let copyToPathArg = copyToPath.asRubyArgument(name: "copy_to_path", type: nil) + let copyToClipboardArg = copyToClipboard.asRubyArgument(name: "copy_to_clipboard", type: nil) + let array: [RubyCommand.Argument?] = [latestArg, + printContentsArg, + printPathsArg, + copyToPathArg, + copyToClipboardArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "spaceship_logs", className: nil, args: args) + return parseArray(fromString: runner.executeCommand(command)) +} + +/** + Print out Spaceship stats from this session (number of request to each domain) + + - parameter printRequestLogs: Print all URLs requested + */ +public func spaceshipStats(printRequestLogs: OptionalConfigValue = .fastlaneDefault(false)) { + let printRequestLogsArg = printRequestLogs.asRubyArgument(name: "print_request_logs", type: nil) + let array: [RubyCommand.Argument?] = [printRequestLogsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "spaceship_stats", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload dSYM file to [Splunk MINT](https://mint.splunk.com/) + + - parameters: + - dsym: dSYM.zip file to upload to Splunk MINT + - apiKey: Splunk MINT App API key e.g. f57a57ca + - apiToken: Splunk MINT API token e.g. e05ba40754c4869fb7e0b61 + - verbose: Make detailed output + - uploadProgress: Show upload progress + - proxyUsername: Proxy username + - proxyPassword: Proxy password + - proxyAddress: Proxy address + - proxyPort: Proxy port + */ +public func splunkmint(dsym: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: String, + apiToken: String, + verbose: OptionalConfigValue = .fastlaneDefault(false), + uploadProgress: OptionalConfigValue = .fastlaneDefault(false), + proxyUsername: OptionalConfigValue = .fastlaneDefault(nil), + proxyPassword: OptionalConfigValue = .fastlaneDefault(nil), + proxyAddress: OptionalConfigValue = .fastlaneDefault(nil), + proxyPort: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let dsymArg = dsym.asRubyArgument(name: "dsym", type: nil) + let apiKeyArg = RubyCommand.Argument(name: "api_key", value: apiKey, type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let uploadProgressArg = uploadProgress.asRubyArgument(name: "upload_progress", type: nil) + let proxyUsernameArg = proxyUsername.asRubyArgument(name: "proxy_username", type: nil) + let proxyPasswordArg = proxyPassword.asRubyArgument(name: "proxy_password", type: nil) + let proxyAddressArg = proxyAddress.asRubyArgument(name: "proxy_address", type: nil) + let proxyPortArg = proxyPort.asRubyArgument(name: "proxy_port", type: nil) + let array: [RubyCommand.Argument?] = [dsymArg, + apiKeyArg, + apiTokenArg, + verboseArg, + uploadProgressArg, + proxyUsernameArg, + proxyPasswordArg, + proxyAddressArg, + proxyPortArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "splunkmint", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs Swift Package Manager on your project + + - parameters: + - command: The swift command (one of: build, test, clean, reset, update, resolve, generate-xcodeproj, init) + - enableCodeCoverage: Enables code coverage for the generated Xcode project when using the 'generate-xcodeproj' and the 'test' command + - buildPath: Specify build/cache directory [default: ./.build] + - packagePath: Change working directory before any other operation + - xcconfig: Use xcconfig file to override swift package generate-xcodeproj defaults + - configuration: Build with configuration (debug|release) [default: debug] + - disableSandbox: Disable using the sandbox when executing subprocesses + - xcprettyOutput: Specifies the output type for xcpretty. eg. 'test', or 'simple' + - xcprettyArgs: Pass in xcpretty additional command line arguments (e.g. '--test --no-color' or '--tap --no-utf'), requires xcpretty_output to be specified also + - verbose: Increase verbosity of informational output + */ +public func spm(command: String = "build", + enableCodeCoverage: OptionalConfigValue = .fastlaneDefault(nil), + buildPath: OptionalConfigValue = .fastlaneDefault(nil), + packagePath: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + disableSandbox: OptionalConfigValue = .fastlaneDefault(false), + xcprettyOutput: OptionalConfigValue = .fastlaneDefault(nil), + xcprettyArgs: OptionalConfigValue = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(false)) +{ + let commandArg = RubyCommand.Argument(name: "command", value: command, type: nil) + let enableCodeCoverageArg = enableCodeCoverage.asRubyArgument(name: "enable_code_coverage", type: nil) + let buildPathArg = buildPath.asRubyArgument(name: "build_path", type: nil) + let packagePathArg = packagePath.asRubyArgument(name: "package_path", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let disableSandboxArg = disableSandbox.asRubyArgument(name: "disable_sandbox", type: nil) + let xcprettyOutputArg = xcprettyOutput.asRubyArgument(name: "xcpretty_output", type: nil) + let xcprettyArgsArg = xcprettyArgs.asRubyArgument(name: "xcpretty_args", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let array: [RubyCommand.Argument?] = [commandArg, + enableCodeCoverageArg, + buildPathArg, + packagePathArg, + xcconfigArg, + configurationArg, + disableSandboxArg, + xcprettyOutputArg, + xcprettyArgsArg, + verboseArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "spm", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Allows remote command execution using ssh + + - parameters: + - username: Username + - password: Password + - host: Hostname + - port: Port + - commands: Commands + - log: Log commands and output + + Lets you execute remote commands via ssh using username/password or ssh-agent. If one of the commands in command-array returns non 0, it fails. + */ +public func ssh(username: String, + password: OptionalConfigValue = .fastlaneDefault(nil), + host: String, + port: String = "22", + commands: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + log: OptionalConfigValue = .fastlaneDefault(true)) +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let passwordArg = password.asRubyArgument(name: "password", type: nil) + let hostArg = RubyCommand.Argument(name: "host", value: host, type: nil) + let portArg = RubyCommand.Argument(name: "port", value: port, type: nil) + let commandsArg = commands.asRubyArgument(name: "commands", type: nil) + let logArg = log.asRubyArgument(name: "log", type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + passwordArg, + hostArg, + portArg, + commandsArg, + logArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "ssh", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `upload_to_play_store` action + + - parameters: + - packageName: The package name of the application to use + - versionName: Version name (used when uploading new apks/aabs) - defaults to 'versionName' in build.gradle or AndroidManifest.xml + - versionCode: Version code (used when updating rollout or promoting specific versions) + - releaseStatus: Release status (used when uploading new apks/aabs) - valid values are completed, draft, halted, inProgress + - track: The track of the application to use. The default available tracks are: production, beta, alpha, internal + - rollout: The percentage of the user fraction when uploading to the rollout track (setting to 1 will complete the rollout) + - metadataPath: Path to the directory containing the metadata files + - key: **DEPRECATED!** Use `--json_key` instead - The p12 File used to authenticate with Google + - issuer: **DEPRECATED!** Use `--json_key` instead - The issuer of the p12 file (email address of the service account) + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - apk: Path to the APK file to upload + - apkPaths: An array of paths to APK files to upload + - aab: Path to the AAB file to upload + - aabPaths: An array of paths to AAB files to upload + - skipUploadApk: Whether to skip uploading APK + - skipUploadAab: Whether to skip uploading AAB + - skipUploadMetadata: Whether to skip uploading metadata, changelogs not included + - skipUploadChangelogs: Whether to skip uploading changelogs + - skipUploadImages: Whether to skip uploading images, screenshots not included + - skipUploadScreenshots: Whether to skip uploading SCREENSHOTS + - trackPromoteTo: The track to promote to. The default available tracks are: production, beta, alpha, internal + - trackPromoteReleaseStatus: Promoted track release status (used when promoting a track) - valid values are completed, draft, halted, inProgress + - validateOnly: Only validate changes with Google Play rather than actually publish + - mapping: Path to the mapping file to upload (mapping.txt or native-debug-symbols.zip alike) + - mappingPaths: An array of paths to mapping files to upload (mapping.txt or native-debug-symbols.zip alike) + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - checkSupersededTracks: **DEPRECATED!** Google Play does this automatically now - Check the other tracks for superseded versions and disable them + - timeout: Timeout for read, open, and send (in seconds) + - deactivateOnPromote: **DEPRECATED!** Google Play does this automatically now - When promoting to a new track, deactivate the binary in the origin track + - versionCodesToRetain: An array of version codes to retain when publishing a new APK + - changesNotSentForReview: Indicates that the changes in this edit will not be reviewed until they are explicitly sent for review from the Google Play Console UI + - rescueChangesNotSentForReview: Catches changes_not_sent_for_review errors when an edit is committed and retries with the configuration that the error message recommended + - inAppUpdatePriority: In-app update priority for all the newly added apks in the release. Can take values between [0,5] + - obbMainReferencesVersion: References version of 'main' expansion file + - obbMainFileSize: Size of 'main' expansion file in bytes + - obbPatchReferencesVersion: References version of 'patch' expansion file + - obbPatchFileSize: Size of 'patch' expansion file in bytes + - ackBundleInstallationWarning: Must be set to true if the bundle installation may trigger a warning on user devices (e.g can only be downloaded over wifi). Typically this is required for bundles over 150MB + + More information: https://docs.fastlane.tools/actions/supply/ + */ +public func supply(packageName: String, + versionName: OptionalConfigValue = .fastlaneDefault(nil), + versionCode: OptionalConfigValue = .fastlaneDefault(nil), + releaseStatus: String = "completed", + track: String = "production", + rollout: OptionalConfigValue = .fastlaneDefault(nil), + metadataPath: OptionalConfigValue = .fastlaneDefault(nil), + key: OptionalConfigValue = .fastlaneDefault(nil), + issuer: OptionalConfigValue = .fastlaneDefault(nil), + jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + apk: OptionalConfigValue = .fastlaneDefault(nil), + apkPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + aab: OptionalConfigValue = .fastlaneDefault(nil), + aabPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + skipUploadApk: OptionalConfigValue = .fastlaneDefault(false), + skipUploadAab: OptionalConfigValue = .fastlaneDefault(false), + skipUploadMetadata: OptionalConfigValue = .fastlaneDefault(false), + skipUploadChangelogs: OptionalConfigValue = .fastlaneDefault(false), + skipUploadImages: OptionalConfigValue = .fastlaneDefault(false), + skipUploadScreenshots: OptionalConfigValue = .fastlaneDefault(false), + trackPromoteTo: OptionalConfigValue = .fastlaneDefault(nil), + trackPromoteReleaseStatus: String = "completed", + validateOnly: OptionalConfigValue = .fastlaneDefault(false), + mapping: OptionalConfigValue = .fastlaneDefault(nil), + mappingPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + checkSupersededTracks: OptionalConfigValue = .fastlaneDefault(false), + timeout: Int = 300, + deactivateOnPromote: OptionalConfigValue = .fastlaneDefault(true), + versionCodesToRetain: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + changesNotSentForReview: OptionalConfigValue = .fastlaneDefault(false), + rescueChangesNotSentForReview: OptionalConfigValue = .fastlaneDefault(true), + inAppUpdatePriority: OptionalConfigValue = .fastlaneDefault(nil), + obbMainReferencesVersion: OptionalConfigValue = .fastlaneDefault(nil), + obbMainFileSize: OptionalConfigValue = .fastlaneDefault(nil), + obbPatchReferencesVersion: OptionalConfigValue = .fastlaneDefault(nil), + obbPatchFileSize: OptionalConfigValue = .fastlaneDefault(nil), + ackBundleInstallationWarning: OptionalConfigValue = .fastlaneDefault(false)) +{ + let packageNameArg = RubyCommand.Argument(name: "package_name", value: packageName, type: nil) + let versionNameArg = versionName.asRubyArgument(name: "version_name", type: nil) + let versionCodeArg = versionCode.asRubyArgument(name: "version_code", type: nil) + let releaseStatusArg = RubyCommand.Argument(name: "release_status", value: releaseStatus, type: nil) + let trackArg = RubyCommand.Argument(name: "track", value: track, type: nil) + let rolloutArg = rollout.asRubyArgument(name: "rollout", type: nil) + let metadataPathArg = metadataPath.asRubyArgument(name: "metadata_path", type: nil) + let keyArg = key.asRubyArgument(name: "key", type: nil) + let issuerArg = issuer.asRubyArgument(name: "issuer", type: nil) + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let apkArg = apk.asRubyArgument(name: "apk", type: nil) + let apkPathsArg = apkPaths.asRubyArgument(name: "apk_paths", type: nil) + let aabArg = aab.asRubyArgument(name: "aab", type: nil) + let aabPathsArg = aabPaths.asRubyArgument(name: "aab_paths", type: nil) + let skipUploadApkArg = skipUploadApk.asRubyArgument(name: "skip_upload_apk", type: nil) + let skipUploadAabArg = skipUploadAab.asRubyArgument(name: "skip_upload_aab", type: nil) + let skipUploadMetadataArg = skipUploadMetadata.asRubyArgument(name: "skip_upload_metadata", type: nil) + let skipUploadChangelogsArg = skipUploadChangelogs.asRubyArgument(name: "skip_upload_changelogs", type: nil) + let skipUploadImagesArg = skipUploadImages.asRubyArgument(name: "skip_upload_images", type: nil) + let skipUploadScreenshotsArg = skipUploadScreenshots.asRubyArgument(name: "skip_upload_screenshots", type: nil) + let trackPromoteToArg = trackPromoteTo.asRubyArgument(name: "track_promote_to", type: nil) + let trackPromoteReleaseStatusArg = RubyCommand.Argument(name: "track_promote_release_status", value: trackPromoteReleaseStatus, type: nil) + let validateOnlyArg = validateOnly.asRubyArgument(name: "validate_only", type: nil) + let mappingArg = mapping.asRubyArgument(name: "mapping", type: nil) + let mappingPathsArg = mappingPaths.asRubyArgument(name: "mapping_paths", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let checkSupersededTracksArg = checkSupersededTracks.asRubyArgument(name: "check_superseded_tracks", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let deactivateOnPromoteArg = deactivateOnPromote.asRubyArgument(name: "deactivate_on_promote", type: nil) + let versionCodesToRetainArg = versionCodesToRetain.asRubyArgument(name: "version_codes_to_retain", type: nil) + let changesNotSentForReviewArg = changesNotSentForReview.asRubyArgument(name: "changes_not_sent_for_review", type: nil) + let rescueChangesNotSentForReviewArg = rescueChangesNotSentForReview.asRubyArgument(name: "rescue_changes_not_sent_for_review", type: nil) + let inAppUpdatePriorityArg = inAppUpdatePriority.asRubyArgument(name: "in_app_update_priority", type: nil) + let obbMainReferencesVersionArg = obbMainReferencesVersion.asRubyArgument(name: "obb_main_references_version", type: nil) + let obbMainFileSizeArg = obbMainFileSize.asRubyArgument(name: "obb_main_file_size", type: nil) + let obbPatchReferencesVersionArg = obbPatchReferencesVersion.asRubyArgument(name: "obb_patch_references_version", type: nil) + let obbPatchFileSizeArg = obbPatchFileSize.asRubyArgument(name: "obb_patch_file_size", type: nil) + let ackBundleInstallationWarningArg = ackBundleInstallationWarning.asRubyArgument(name: "ack_bundle_installation_warning", type: nil) + let array: [RubyCommand.Argument?] = [packageNameArg, + versionNameArg, + versionCodeArg, + releaseStatusArg, + trackArg, + rolloutArg, + metadataPathArg, + keyArg, + issuerArg, + jsonKeyArg, + jsonKeyDataArg, + apkArg, + apkPathsArg, + aabArg, + aabPathsArg, + skipUploadApkArg, + skipUploadAabArg, + skipUploadMetadataArg, + skipUploadChangelogsArg, + skipUploadImagesArg, + skipUploadScreenshotsArg, + trackPromoteToArg, + trackPromoteReleaseStatusArg, + validateOnlyArg, + mappingArg, + mappingPathsArg, + rootUrlArg, + checkSupersededTracksArg, + timeoutArg, + deactivateOnPromoteArg, + versionCodesToRetainArg, + changesNotSentForReviewArg, + rescueChangesNotSentForReviewArg, + inAppUpdatePriorityArg, + obbMainReferencesVersionArg, + obbMainFileSizeArg, + obbPatchReferencesVersionArg, + obbPatchFileSizeArg, + ackBundleInstallationWarningArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "supply", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Run swift code validation using SwiftLint + + - parameters: + - mode: SwiftLint mode: :lint, :fix, :autocorrect or :analyze + - path: Specify path to lint + - outputFile: Path to output SwiftLint result + - configFile: Custom configuration file of SwiftLint + - strict: Fail on warnings? (true/false) + - files: List of files to process + - ignoreExitStatus: Ignore the exit status of the SwiftLint command, so that serious violations don't fail the build (true/false) + - raiseIfSwiftlintError: Raises an error if swiftlint fails, so you can fail CI/CD jobs if necessary (true/false) + - reporter: Choose output reporter. Available: xcode, json, csv, checkstyle, codeclimate, junit, html, emoji, sonarqube, markdown, github-actions-logging + - quiet: Don't print status logs like 'Linting ' & 'Done linting' + - executable: Path to the `swiftlint` executable on your machine + - format: Format code when mode is :autocorrect + - noCache: Ignore the cache when mode is :autocorrect or :lint + - compilerLogPath: Compiler log path when mode is :analyze + */ +public func swiftlint(mode: String = "lint", + path: OptionalConfigValue = .fastlaneDefault(nil), + outputFile: OptionalConfigValue = .fastlaneDefault(nil), + configFile: OptionalConfigValue = .fastlaneDefault(nil), + strict: OptionalConfigValue = .fastlaneDefault(false), + files: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + ignoreExitStatus: OptionalConfigValue = .fastlaneDefault(false), + raiseIfSwiftlintError: OptionalConfigValue = .fastlaneDefault(false), + reporter: OptionalConfigValue = .fastlaneDefault(nil), + quiet: OptionalConfigValue = .fastlaneDefault(false), + executable: OptionalConfigValue = .fastlaneDefault(nil), + format: OptionalConfigValue = .fastlaneDefault(false), + noCache: OptionalConfigValue = .fastlaneDefault(false), + compilerLogPath: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let modeArg = RubyCommand.Argument(name: "mode", value: mode, type: nil) + let pathArg = path.asRubyArgument(name: "path", type: nil) + let outputFileArg = outputFile.asRubyArgument(name: "output_file", type: nil) + let configFileArg = configFile.asRubyArgument(name: "config_file", type: nil) + let strictArg = strict.asRubyArgument(name: "strict", type: nil) + let filesArg = files.asRubyArgument(name: "files", type: nil) + let ignoreExitStatusArg = ignoreExitStatus.asRubyArgument(name: "ignore_exit_status", type: nil) + let raiseIfSwiftlintErrorArg = raiseIfSwiftlintError.asRubyArgument(name: "raise_if_swiftlint_error", type: nil) + let reporterArg = reporter.asRubyArgument(name: "reporter", type: nil) + let quietArg = quiet.asRubyArgument(name: "quiet", type: nil) + let executableArg = executable.asRubyArgument(name: "executable", type: nil) + let formatArg = format.asRubyArgument(name: "format", type: nil) + let noCacheArg = noCache.asRubyArgument(name: "no_cache", type: nil) + let compilerLogPathArg = compilerLogPath.asRubyArgument(name: "compiler_log_path", type: nil) + let array: [RubyCommand.Argument?] = [modeArg, + pathArg, + outputFileArg, + configFileArg, + strictArg, + filesArg, + ignoreExitStatusArg, + raiseIfSwiftlintErrorArg, + reporterArg, + quietArg, + executableArg, + formatArg, + noCacheArg, + compilerLogPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "swiftlint", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Easily sync your certificates and profiles across your team (via _match_) + + - parameters: + - type: Define the profile type, can be appstore, adhoc, development, enterprise, developer_id, mac_installer_distribution + - additionalCertTypes: Create additional cert types needed for macOS installers (valid values: mac_installer_distribution, developer_id_installer) + - readonly: Only fetch existing certificates and profiles, don't generate new ones + - generateAppleCerts: Create a certificate type for Xcode 11 and later (Apple Development or Apple Distribution) + - skipProvisioningProfiles: Skip syncing provisioning profiles + - appIdentifier: The bundle identifier(s) of your app (comma-separated string or array of strings) + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - teamId: The ID of your Developer Portal team if you're in multiple teams + - teamName: The name of your Developer Portal team if you're in multiple teams + - storageMode: Define where you want to store your certificates + - gitUrl: URL to the git repo containing all the certificates + - gitBranch: Specific git branch to use + - gitFullName: git user full name to commit + - gitUserEmail: git user email to commit + - shallowClone: Make a shallow clone of the repository (truncate the history to 1 revision) + - cloneBranchDirectly: Clone just the branch specified, instead of the whole repo. This requires that the branch already exists. Otherwise the command will fail + - gitBasicAuthorization: Use a basic authorization header to access the git repo (e.g.: access via HTTPS, GitHub Actions, etc), usually a string in Base64 + - gitBearerAuthorization: Use a bearer authorization header to access the git repo (e.g.: access to an Azure DevOps repository), usually a string in Base64 + - gitPrivateKey: Use a private key to access the git repo (e.g.: access to GitHub repository via Deploy keys), usually a id_rsa named file or the contents hereof + - googleCloudBucketName: Name of the Google Cloud Storage bucket to use + - googleCloudKeysFile: Path to the gc_keys.json file + - googleCloudProjectId: ID of the Google Cloud project to use for authentication + - skipGoogleCloudAccountConfirmation: Skips confirming to use the system google account + - s3Region: Name of the S3 region + - s3AccessKey: S3 access key + - s3SecretAccessKey: S3 secret access key + - s3Bucket: Name of the S3 bucket + - s3ObjectPrefix: Prefix to be used on all objects uploaded to S3 + - gitlabProject: GitLab Project Path (i.e. 'gitlab-org/gitlab') + - keychainName: Keychain the items should be imported to + - keychainPassword: This might be required the first time you access certificates on a new mac. For the login/default keychain this is your macOS account password + - force: Renew the provisioning profiles every time you run match + - forceForNewDevices: Renew the provisioning profiles if the device count on the developer portal has changed. Ignored for profile types 'appstore' and 'developer_id' + - includeMacInProfiles: Include Apple Silicon Mac devices in provisioning profiles for iOS/iPadOS apps + - includeAllCertificates: Include all matching certificates in the provisioning profile. Works only for the 'development' provisioning profile type + - forceForNewCertificates: Renew the provisioning profiles if the certificate count on the developer portal has changed. Works only for the 'development' provisioning profile type. Requires 'include_all_certificates' option to be 'true' + - skipConfirmation: Disables confirmation prompts during nuke, answering them with yes + - safeRemoveCerts: Remove certs from repository during nuke without revoking them on the developer portal + - skipDocs: Skip generation of a README.md for the created git repository + - platform: Set the provisioning profile's platform to work with (i.e. ios, tvos, macos, catalyst) + - deriveCatalystAppIdentifier: Enable this if you have the Mac Catalyst capability enabled and your project was created with Xcode 11.3 or earlier. Prepends 'maccatalyst.' to the app identifier for the provisioning profile mapping + - templateName: The name of provisioning profile template. If the developer account has provisioning profile templates (aka: custom entitlements), the template name can be found by inspecting the Entitlements drop-down while creating/editing a provisioning profile (e.g. "Apple Pay Pass Suppression Development") + - profileName: A custom name for the provisioning profile. This will replace the default provisioning profile name if specified + - failOnNameTaken: Should the command fail if it was about to create a duplicate of an existing provisioning profile. It can happen due to issues on Apple Developer Portal, when profile to be recreated was not properly deleted first + - skipCertificateMatching: Set to true if there is no access to Apple developer portal but there are certificates, keys and profiles provided. Only works with match import action + - outputPath: Path in which to export certificates, key and profile + - skipSetPartitionList: Skips setting the partition list (which can sometimes take a long time). Setting the partition list is usually needed to prevent Xcode from prompting to allow a cert to be used for signing + - verbose: Print out extra information and all commands + + More information: https://docs.fastlane.tools/actions/match/ + */ +public func syncCodeSigning(type: String = "development", + additionalCertTypes: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + readonly: OptionalConfigValue = .fastlaneDefault(false), + generateAppleCerts: OptionalConfigValue = .fastlaneDefault(true), + skipProvisioningProfiles: OptionalConfigValue = .fastlaneDefault(false), + appIdentifier: [String], + apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + storageMode: String = "git", + gitUrl: String, + gitBranch: String = "master", + gitFullName: OptionalConfigValue = .fastlaneDefault(nil), + gitUserEmail: OptionalConfigValue = .fastlaneDefault(nil), + shallowClone: OptionalConfigValue = .fastlaneDefault(false), + cloneBranchDirectly: OptionalConfigValue = .fastlaneDefault(false), + gitBasicAuthorization: OptionalConfigValue = .fastlaneDefault(nil), + gitBearerAuthorization: OptionalConfigValue = .fastlaneDefault(nil), + gitPrivateKey: OptionalConfigValue = .fastlaneDefault(nil), + googleCloudBucketName: OptionalConfigValue = .fastlaneDefault(nil), + googleCloudKeysFile: OptionalConfigValue = .fastlaneDefault(nil), + googleCloudProjectId: OptionalConfigValue = .fastlaneDefault(nil), + skipGoogleCloudAccountConfirmation: OptionalConfigValue = .fastlaneDefault(false), + s3Region: OptionalConfigValue = .fastlaneDefault(nil), + s3AccessKey: OptionalConfigValue = .fastlaneDefault(nil), + s3SecretAccessKey: OptionalConfigValue = .fastlaneDefault(nil), + s3Bucket: OptionalConfigValue = .fastlaneDefault(nil), + s3ObjectPrefix: OptionalConfigValue = .fastlaneDefault(nil), + gitlabProject: OptionalConfigValue = .fastlaneDefault(nil), + keychainName: String = "login.keychain", + keychainPassword: OptionalConfigValue = .fastlaneDefault(nil), + force: OptionalConfigValue = .fastlaneDefault(false), + forceForNewDevices: OptionalConfigValue = .fastlaneDefault(false), + includeMacInProfiles: OptionalConfigValue = .fastlaneDefault(false), + includeAllCertificates: OptionalConfigValue = .fastlaneDefault(false), + forceForNewCertificates: OptionalConfigValue = .fastlaneDefault(false), + skipConfirmation: OptionalConfigValue = .fastlaneDefault(false), + safeRemoveCerts: OptionalConfigValue = .fastlaneDefault(false), + skipDocs: OptionalConfigValue = .fastlaneDefault(false), + platform: String = "ios", + deriveCatalystAppIdentifier: OptionalConfigValue = .fastlaneDefault(false), + templateName: OptionalConfigValue = .fastlaneDefault(nil), + profileName: OptionalConfigValue = .fastlaneDefault(nil), + failOnNameTaken: OptionalConfigValue = .fastlaneDefault(false), + skipCertificateMatching: OptionalConfigValue = .fastlaneDefault(false), + outputPath: OptionalConfigValue = .fastlaneDefault(nil), + skipSetPartitionList: OptionalConfigValue = .fastlaneDefault(false), + verbose: OptionalConfigValue = .fastlaneDefault(false)) +{ + let typeArg = RubyCommand.Argument(name: "type", value: type, type: nil) + let additionalCertTypesArg = additionalCertTypes.asRubyArgument(name: "additional_cert_types", type: nil) + let readonlyArg = readonly.asRubyArgument(name: "readonly", type: nil) + let generateAppleCertsArg = generateAppleCerts.asRubyArgument(name: "generate_apple_certs", type: nil) + let skipProvisioningProfilesArg = skipProvisioningProfiles.asRubyArgument(name: "skip_provisioning_profiles", type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let storageModeArg = RubyCommand.Argument(name: "storage_mode", value: storageMode, type: nil) + let gitUrlArg = RubyCommand.Argument(name: "git_url", value: gitUrl, type: nil) + let gitBranchArg = RubyCommand.Argument(name: "git_branch", value: gitBranch, type: nil) + let gitFullNameArg = gitFullName.asRubyArgument(name: "git_full_name", type: nil) + let gitUserEmailArg = gitUserEmail.asRubyArgument(name: "git_user_email", type: nil) + let shallowCloneArg = shallowClone.asRubyArgument(name: "shallow_clone", type: nil) + let cloneBranchDirectlyArg = cloneBranchDirectly.asRubyArgument(name: "clone_branch_directly", type: nil) + let gitBasicAuthorizationArg = gitBasicAuthorization.asRubyArgument(name: "git_basic_authorization", type: nil) + let gitBearerAuthorizationArg = gitBearerAuthorization.asRubyArgument(name: "git_bearer_authorization", type: nil) + let gitPrivateKeyArg = gitPrivateKey.asRubyArgument(name: "git_private_key", type: nil) + let googleCloudBucketNameArg = googleCloudBucketName.asRubyArgument(name: "google_cloud_bucket_name", type: nil) + let googleCloudKeysFileArg = googleCloudKeysFile.asRubyArgument(name: "google_cloud_keys_file", type: nil) + let googleCloudProjectIdArg = googleCloudProjectId.asRubyArgument(name: "google_cloud_project_id", type: nil) + let skipGoogleCloudAccountConfirmationArg = skipGoogleCloudAccountConfirmation.asRubyArgument(name: "skip_google_cloud_account_confirmation", type: nil) + let s3RegionArg = s3Region.asRubyArgument(name: "s3_region", type: nil) + let s3AccessKeyArg = s3AccessKey.asRubyArgument(name: "s3_access_key", type: nil) + let s3SecretAccessKeyArg = s3SecretAccessKey.asRubyArgument(name: "s3_secret_access_key", type: nil) + let s3BucketArg = s3Bucket.asRubyArgument(name: "s3_bucket", type: nil) + let s3ObjectPrefixArg = s3ObjectPrefix.asRubyArgument(name: "s3_object_prefix", type: nil) + let gitlabProjectArg = gitlabProject.asRubyArgument(name: "gitlab_project", type: nil) + let keychainNameArg = RubyCommand.Argument(name: "keychain_name", value: keychainName, type: nil) + let keychainPasswordArg = keychainPassword.asRubyArgument(name: "keychain_password", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let forceForNewDevicesArg = forceForNewDevices.asRubyArgument(name: "force_for_new_devices", type: nil) + let includeMacInProfilesArg = includeMacInProfiles.asRubyArgument(name: "include_mac_in_profiles", type: nil) + let includeAllCertificatesArg = includeAllCertificates.asRubyArgument(name: "include_all_certificates", type: nil) + let forceForNewCertificatesArg = forceForNewCertificates.asRubyArgument(name: "force_for_new_certificates", type: nil) + let skipConfirmationArg = skipConfirmation.asRubyArgument(name: "skip_confirmation", type: nil) + let safeRemoveCertsArg = safeRemoveCerts.asRubyArgument(name: "safe_remove_certs", type: nil) + let skipDocsArg = skipDocs.asRubyArgument(name: "skip_docs", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let deriveCatalystAppIdentifierArg = deriveCatalystAppIdentifier.asRubyArgument(name: "derive_catalyst_app_identifier", type: nil) + let templateNameArg = templateName.asRubyArgument(name: "template_name", type: nil) + let profileNameArg = profileName.asRubyArgument(name: "profile_name", type: nil) + let failOnNameTakenArg = failOnNameTaken.asRubyArgument(name: "fail_on_name_taken", type: nil) + let skipCertificateMatchingArg = skipCertificateMatching.asRubyArgument(name: "skip_certificate_matching", type: nil) + let outputPathArg = outputPath.asRubyArgument(name: "output_path", type: nil) + let skipSetPartitionListArg = skipSetPartitionList.asRubyArgument(name: "skip_set_partition_list", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let array: [RubyCommand.Argument?] = [typeArg, + additionalCertTypesArg, + readonlyArg, + generateAppleCertsArg, + skipProvisioningProfilesArg, + appIdentifierArg, + apiKeyPathArg, + apiKeyArg, + usernameArg, + teamIdArg, + teamNameArg, + storageModeArg, + gitUrlArg, + gitBranchArg, + gitFullNameArg, + gitUserEmailArg, + shallowCloneArg, + cloneBranchDirectlyArg, + gitBasicAuthorizationArg, + gitBearerAuthorizationArg, + gitPrivateKeyArg, + googleCloudBucketNameArg, + googleCloudKeysFileArg, + googleCloudProjectIdArg, + skipGoogleCloudAccountConfirmationArg, + s3RegionArg, + s3AccessKeyArg, + s3SecretAccessKeyArg, + s3BucketArg, + s3ObjectPrefixArg, + gitlabProjectArg, + keychainNameArg, + keychainPasswordArg, + forceArg, + forceForNewDevicesArg, + includeMacInProfilesArg, + includeAllCertificatesArg, + forceForNewCertificatesArg, + skipConfirmationArg, + safeRemoveCertsArg, + skipDocsArg, + platformArg, + deriveCatalystAppIdentifierArg, + templateNameArg, + profileNameArg, + failOnNameTakenArg, + skipCertificateMatchingArg, + outputPathArg, + skipSetPartitionListArg, + verboseArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "sync_code_signing", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Specify the Team ID you want to use for the Apple Developer Portal + */ +public func teamId() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "team_id", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Set a team to use by its name + */ +public func teamName() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "team_name", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload a new build to [TestFairy](https://www.testfairy.com/) + + - parameters: + - apiKey: API Key for TestFairy + - ipa: Path to your IPA file for iOS + - apk: Path to your APK file for Android + - symbolsFile: Symbols mapping file + - uploadUrl: API URL for TestFairy + - testersGroups: Array of tester groups to be notified + - metrics: Array of metrics to record (cpu,memory,network,phone_signal,gps,battery,mic,wifi) + - comment: Additional release notes for this upload. This text will be added to email notifications + - autoUpdate: Allows an easy upgrade of all users to the current version. To enable set to 'on' + - notify: Send email to testers + - options: Array of options (shake,video_only_wifi,anonymous) + - custom: Array of custom options. Contact support@testfairy.com for more information + - timeout: Request timeout in seconds + + You can retrieve your API key on [your settings page](https://free.testfairy.com/settings/) + */ +public func testfairy(apiKey: String, + ipa: OptionalConfigValue = .fastlaneDefault(nil), + apk: OptionalConfigValue = .fastlaneDefault(nil), + symbolsFile: OptionalConfigValue = .fastlaneDefault(nil), + uploadUrl: String = "https://upload.testfairy.com", + testersGroups: [String] = [], + metrics: [String] = [], + comment: String = "No comment provided", + autoUpdate: String = "off", + notify: String = "off", + options: [String] = [], + custom: String = "", + timeout: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiKeyArg = RubyCommand.Argument(name: "api_key", value: apiKey, type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let apkArg = apk.asRubyArgument(name: "apk", type: nil) + let symbolsFileArg = symbolsFile.asRubyArgument(name: "symbols_file", type: nil) + let uploadUrlArg = RubyCommand.Argument(name: "upload_url", value: uploadUrl, type: nil) + let testersGroupsArg = RubyCommand.Argument(name: "testers_groups", value: testersGroups, type: nil) + let metricsArg = RubyCommand.Argument(name: "metrics", value: metrics, type: nil) + let commentArg = RubyCommand.Argument(name: "comment", value: comment, type: nil) + let autoUpdateArg = RubyCommand.Argument(name: "auto_update", value: autoUpdate, type: nil) + let notifyArg = RubyCommand.Argument(name: "notify", value: notify, type: nil) + let optionsArg = RubyCommand.Argument(name: "options", value: options, type: nil) + let customArg = RubyCommand.Argument(name: "custom", value: custom, type: nil) + let timeoutArg = timeout.asRubyArgument(name: "timeout", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyArg, + ipaArg, + apkArg, + symbolsFileArg, + uploadUrlArg, + testersGroupsArg, + metricsArg, + commentArg, + autoUpdateArg, + notifyArg, + optionsArg, + customArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "testfairy", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Alias for the `upload_to_testflight` action + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - appIdentifier: The bundle identifier of the app to upload or manage testers (optional) + - appPlatform: The platform to use (optional) + - appleId: Apple ID property in the App Information section in App Store Connect + - ipa: Path to the ipa file to upload + - pkg: Path to your pkg file + - demoAccountRequired: Do you need a demo account when Apple does review? + - betaAppReviewInfo: Beta app review information for contact info and demo account + - localizedAppInfo: Localized beta app test info for description, feedback email, marketing url, and privacy policy + - betaAppDescription: Provide the 'Beta App Description' when uploading a new build + - betaAppFeedbackEmail: Provide the beta app email when uploading a new build + - localizedBuildInfo: Localized beta app test info for what's new + - changelog: Provide the 'What to Test' text when uploading a new build + - skipSubmission: Skip the distributing action of pilot and only upload the ipa file + - skipWaitingForBuildProcessing: If set to true, the `distribute_external` option won't work and no build will be distributed to testers. (You might want to use this option if you are using this action on CI and have to pay for 'minutes used' on your CI plan). If set to `true` and a changelog is provided, it will partially wait for the build to appear on AppStore Connect so the changelog can be set, and skip the remaining processing steps + - updateBuildInfoOnUpload: **DEPRECATED!** Update build info immediately after validation. This is deprecated and will be removed in a future release. App Store Connect no longer supports setting build info until after build processing has completed, which is when build info is updated by default + - distributeOnly: Distribute a previously uploaded build (equivalent to the `fastlane pilot distribute` command) + - usesNonExemptEncryption: Provide the 'Uses Non-Exempt Encryption' for export compliance. This is used if there is 'ITSAppUsesNonExemptEncryption' is not set in the Info.plist + - distributeExternal: Should the build be distributed to external testers? If set to true, use of `groups` option is required + - notifyExternalTesters: Should notify external testers? (Not setting a value will use App Store Connect's default which is to notify) + - appVersion: The version number of the application build to distribute. If the version number is not specified, then the most recent build uploaded to TestFlight will be distributed. If specified, the most recent build for the version number will be distributed + - buildNumber: The build number of the application build to distribute. If the build number is not specified, the most recent build is distributed + - expirePreviousBuilds: Should expire previous builds? + - firstName: The tester's first name + - lastName: The tester's last name + - email: The tester's email + - testersFilePath: Path to a CSV file of testers + - groups: Associate tester to one group or more by group name / group id. E.g. `-g "Team 1","Team 2"` This is required when `distribute_external` option is set to true or when we want to add a tester to one or more external testing groups + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - devPortalTeamId: The short ID of your team in the developer portal, if you're in multiple teams. Different from your iTC team ID! + - itcProvider: The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + - waitProcessingInterval: Interval in seconds to wait for App Store Connect processing + - waitProcessingTimeoutDuration: Timeout duration in seconds to wait for App Store Connect processing. If set, after exceeding timeout duration, this will `force stop` to wait for App Store Connect processing and exit with exception + - waitForUploadedBuild: **DEPRECATED!** No longer needed with the transition over to the App Store Connect API - Use version info from uploaded ipa file to determine what build to use for distribution. If set to false, latest processing or any latest build will be used + - rejectBuildWaitingForReview: Expire previous if it's 'waiting for review' + - submitBetaReview: Send the build for a beta review + + More details can be found on https://docs.fastlane.tools/actions/pilot/. + This integration will only do the TestFlight upload. + */ +public func testflight(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + appPlatform: OptionalConfigValue = .fastlaneDefault(nil), + appleId: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil), + pkg: OptionalConfigValue = .fastlaneDefault(nil), + demoAccountRequired: OptionalConfigValue = .fastlaneDefault(nil), + betaAppReviewInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + localizedAppInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + betaAppDescription: OptionalConfigValue = .fastlaneDefault(nil), + betaAppFeedbackEmail: OptionalConfigValue = .fastlaneDefault(nil), + localizedBuildInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + changelog: OptionalConfigValue = .fastlaneDefault(nil), + skipSubmission: OptionalConfigValue = .fastlaneDefault(false), + skipWaitingForBuildProcessing: OptionalConfigValue = .fastlaneDefault(false), + updateBuildInfoOnUpload: OptionalConfigValue = .fastlaneDefault(false), + distributeOnly: OptionalConfigValue = .fastlaneDefault(false), + usesNonExemptEncryption: OptionalConfigValue = .fastlaneDefault(false), + distributeExternal: OptionalConfigValue = .fastlaneDefault(false), + notifyExternalTesters: Any? = nil, + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + expirePreviousBuilds: OptionalConfigValue = .fastlaneDefault(false), + firstName: OptionalConfigValue = .fastlaneDefault(nil), + lastName: OptionalConfigValue = .fastlaneDefault(nil), + email: OptionalConfigValue = .fastlaneDefault(nil), + testersFilePath: String = "./testers.csv", + groups: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + teamId: Any? = nil, + teamName: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamId: OptionalConfigValue = .fastlaneDefault(nil), + itcProvider: OptionalConfigValue = .fastlaneDefault(nil), + waitProcessingInterval: Int = 30, + waitProcessingTimeoutDuration: OptionalConfigValue = .fastlaneDefault(nil), + waitForUploadedBuild: OptionalConfigValue = .fastlaneDefault(false), + rejectBuildWaitingForReview: OptionalConfigValue = .fastlaneDefault(false), + submitBetaReview: OptionalConfigValue = .fastlaneDefault(true)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let appPlatformArg = appPlatform.asRubyArgument(name: "app_platform", type: nil) + let appleIdArg = appleId.asRubyArgument(name: "apple_id", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let pkgArg = pkg.asRubyArgument(name: "pkg", type: nil) + let demoAccountRequiredArg = demoAccountRequired.asRubyArgument(name: "demo_account_required", type: nil) + let betaAppReviewInfoArg = betaAppReviewInfo.asRubyArgument(name: "beta_app_review_info", type: nil) + let localizedAppInfoArg = localizedAppInfo.asRubyArgument(name: "localized_app_info", type: nil) + let betaAppDescriptionArg = betaAppDescription.asRubyArgument(name: "beta_app_description", type: nil) + let betaAppFeedbackEmailArg = betaAppFeedbackEmail.asRubyArgument(name: "beta_app_feedback_email", type: nil) + let localizedBuildInfoArg = localizedBuildInfo.asRubyArgument(name: "localized_build_info", type: nil) + let changelogArg = changelog.asRubyArgument(name: "changelog", type: nil) + let skipSubmissionArg = skipSubmission.asRubyArgument(name: "skip_submission", type: nil) + let skipWaitingForBuildProcessingArg = skipWaitingForBuildProcessing.asRubyArgument(name: "skip_waiting_for_build_processing", type: nil) + let updateBuildInfoOnUploadArg = updateBuildInfoOnUpload.asRubyArgument(name: "update_build_info_on_upload", type: nil) + let distributeOnlyArg = distributeOnly.asRubyArgument(name: "distribute_only", type: nil) + let usesNonExemptEncryptionArg = usesNonExemptEncryption.asRubyArgument(name: "uses_non_exempt_encryption", type: nil) + let distributeExternalArg = distributeExternal.asRubyArgument(name: "distribute_external", type: nil) + let notifyExternalTestersArg = RubyCommand.Argument(name: "notify_external_testers", value: notifyExternalTesters, type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let expirePreviousBuildsArg = expirePreviousBuilds.asRubyArgument(name: "expire_previous_builds", type: nil) + let firstNameArg = firstName.asRubyArgument(name: "first_name", type: nil) + let lastNameArg = lastName.asRubyArgument(name: "last_name", type: nil) + let emailArg = email.asRubyArgument(name: "email", type: nil) + let testersFilePathArg = RubyCommand.Argument(name: "testers_file_path", value: testersFilePath, type: nil) + let groupsArg = groups.asRubyArgument(name: "groups", type: nil) + let teamIdArg = RubyCommand.Argument(name: "team_id", value: teamId, type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let devPortalTeamIdArg = devPortalTeamId.asRubyArgument(name: "dev_portal_team_id", type: nil) + let itcProviderArg = itcProvider.asRubyArgument(name: "itc_provider", type: nil) + let waitProcessingIntervalArg = RubyCommand.Argument(name: "wait_processing_interval", value: waitProcessingInterval, type: nil) + let waitProcessingTimeoutDurationArg = waitProcessingTimeoutDuration.asRubyArgument(name: "wait_processing_timeout_duration", type: nil) + let waitForUploadedBuildArg = waitForUploadedBuild.asRubyArgument(name: "wait_for_uploaded_build", type: nil) + let rejectBuildWaitingForReviewArg = rejectBuildWaitingForReview.asRubyArgument(name: "reject_build_waiting_for_review", type: nil) + let submitBetaReviewArg = submitBetaReview.asRubyArgument(name: "submit_beta_review", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + appPlatformArg, + appleIdArg, + ipaArg, + pkgArg, + demoAccountRequiredArg, + betaAppReviewInfoArg, + localizedAppInfoArg, + betaAppDescriptionArg, + betaAppFeedbackEmailArg, + localizedBuildInfoArg, + changelogArg, + skipSubmissionArg, + skipWaitingForBuildProcessingArg, + updateBuildInfoOnUploadArg, + distributeOnlyArg, + usesNonExemptEncryptionArg, + distributeExternalArg, + notifyExternalTestersArg, + appVersionArg, + buildNumberArg, + expirePreviousBuildsArg, + firstNameArg, + lastNameArg, + emailArg, + testersFilePathArg, + groupsArg, + teamIdArg, + teamNameArg, + devPortalTeamIdArg, + itcProviderArg, + waitProcessingIntervalArg, + waitProcessingTimeoutDurationArg, + waitForUploadedBuildArg, + rejectBuildWaitingForReviewArg, + submitBetaReviewArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "testflight", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Convert the Xcode plist log to a JUnit report + + - parameters: + - path: Path to the directory that should be converted + - extension: The extension for the newly created file. Usually .xml or .junit + - outputDirectory: Directory in which the xml files should be written to. Same directory as source by default + - outputFilename: Filename the xml file should be written to. Defaults to name of input file. (Only works if one input file is used) + - failBuild: Should this step stop the build if the tests fail? Set this to false if you're handling this with a test reporter + - xcprettyNaming: Produces class name and test name identical to xcpretty naming in junit file + - silent: Silences all output + - outputRemoveRetryAttempts: Doesn't include retry attempts in the output + + - returns: A hash with the key being the path of the generated file, the value being if the tests were successful + */ +public func trainer(path: String = ".", + extension: String = ".xml", + outputDirectory: OptionalConfigValue = .fastlaneDefault(nil), + outputFilename: OptionalConfigValue = .fastlaneDefault(nil), + failBuild: OptionalConfigValue = .fastlaneDefault(true), + xcprettyNaming: OptionalConfigValue = .fastlaneDefault(false), + silent: OptionalConfigValue = .fastlaneDefault(false), + outputRemoveRetryAttempts: OptionalConfigValue = .fastlaneDefault(false)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let extensionArg = RubyCommand.Argument(name: "extension", value: `extension`, type: nil) + let outputDirectoryArg = outputDirectory.asRubyArgument(name: "output_directory", type: nil) + let outputFilenameArg = outputFilename.asRubyArgument(name: "output_filename", type: nil) + let failBuildArg = failBuild.asRubyArgument(name: "fail_build", type: nil) + let xcprettyNamingArg = xcprettyNaming.asRubyArgument(name: "xcpretty_naming", type: nil) + let silentArg = silent.asRubyArgument(name: "silent", type: nil) + let outputRemoveRetryAttemptsArg = outputRemoveRetryAttempts.asRubyArgument(name: "output_remove_retry_attempts", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + extensionArg, + outputDirectoryArg, + outputFilenameArg, + failBuildArg, + xcprettyNamingArg, + silentArg, + outputRemoveRetryAttemptsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "trainer", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload a new build to [Tryouts](https://tryouts.io/) + + - parameters: + - appId: Tryouts application hash + - apiToken: API Token (api_key:api_secret) for Tryouts Access + - buildFile: Path to your IPA or APK file. Optional if you use the _gym_ or _xcodebuild_ action + - notes: Release notes + - notesPath: Release notes text file path. Overrides the :notes parameter + - notify: Notify testers? 0 for no + - status: 2 to make your release public. Release will be distributed to available testers. 1 to make your release private. Release won't be distributed to testers. This also prevents release from showing up for SDK update + + More information: [http://tryouts.readthedocs.org/en/latest/releases.html#create-release](http://tryouts.readthedocs.org/en/latest/releases.html#create-release) + */ +public func tryouts(appId: String, + apiToken: String, + buildFile: String, + notes: OptionalConfigValue = .fastlaneDefault(nil), + notesPath: OptionalConfigValue = .fastlaneDefault(nil), + notify: Int = 1, + status: Int = 2) +{ + let appIdArg = RubyCommand.Argument(name: "app_id", value: appId, type: nil) + let apiTokenArg = RubyCommand.Argument(name: "api_token", value: apiToken, type: nil) + let buildFileArg = RubyCommand.Argument(name: "build_file", value: buildFile, type: nil) + let notesArg = notes.asRubyArgument(name: "notes", type: nil) + let notesPathArg = notesPath.asRubyArgument(name: "notes_path", type: nil) + let notifyArg = RubyCommand.Argument(name: "notify", value: notify, type: nil) + let statusArg = RubyCommand.Argument(name: "status", value: status, type: nil) + let array: [RubyCommand.Argument?] = [appIdArg, + apiTokenArg, + buildFileArg, + notesArg, + notesPathArg, + notifyArg, + statusArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "tryouts", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Post a tweet on [Twitter.com](https://twitter.com) + + - parameters: + - consumerKey: Consumer Key + - consumerSecret: Consumer Secret + - accessToken: Access Token + - accessTokenSecret: Access Token Secret + - message: The tweet + + Post a tweet on Twitter. Requires you to setup an app on [twitter.com](https://twitter.com) and obtain `consumer` and `access_token`. + */ +public func twitter(consumerKey: String, + consumerSecret: String, + accessToken: String, + accessTokenSecret: String, + message: String) +{ + let consumerKeyArg = RubyCommand.Argument(name: "consumer_key", value: consumerKey, type: nil) + let consumerSecretArg = RubyCommand.Argument(name: "consumer_secret", value: consumerSecret, type: nil) + let accessTokenArg = RubyCommand.Argument(name: "access_token", value: accessToken, type: nil) + let accessTokenSecretArg = RubyCommand.Argument(name: "access_token_secret", value: accessTokenSecret, type: nil) + let messageArg = RubyCommand.Argument(name: "message", value: message, type: nil) + let array: [RubyCommand.Argument?] = [consumerKeyArg, + consumerSecretArg, + accessTokenArg, + accessTokenSecretArg, + messageArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "twitter", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Post a message to [Typetalk](https://www.typetalk.com/) + */ +public func typetalk() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "typetalk", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Unlock a keychain + + - parameters: + - path: Path to the keychain file + - password: Keychain password + - addToSearchList: Add to keychain search list, valid values are true, false, :add, and :replace + - setDefault: Set as default keychain + + Unlocks the given keychain file and adds it to the keychain search list. + Keychains can be replaced with `add_to_search_list: :replace`. + */ +public func unlockKeychain(path: String = "login", + password: String, + addToSearchList: OptionalConfigValue = .fastlaneDefault(true), + setDefault: OptionalConfigValue = .fastlaneDefault(false)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let passwordArg = RubyCommand.Argument(name: "password", value: password, type: nil) + let addToSearchListArg = addToSearchList.asRubyArgument(name: "add_to_search_list", type: nil) + let setDefaultArg = setDefault.asRubyArgument(name: "set_default", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + passwordArg, + addToSearchListArg, + setDefaultArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "unlock_keychain", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action changes the app group identifiers in the entitlements file + + - parameters: + - entitlementsFile: The path to the entitlement file which contains the app group identifiers + - appGroupIdentifiers: An Array of unique identifiers for the app groups. Eg. ['group.com.test.testapp'] + + Updates the App Group Identifiers in the given Entitlements file, so you can have app groups for the app store build and app groups for an enterprise build. + */ +public func updateAppGroupIdentifiers(entitlementsFile: String, + appGroupIdentifiers: [String]) +{ + let entitlementsFileArg = RubyCommand.Argument(name: "entitlements_file", value: entitlementsFile, type: nil) + let appGroupIdentifiersArg = RubyCommand.Argument(name: "app_group_identifiers", value: appGroupIdentifiers, type: nil) + let array: [RubyCommand.Argument?] = [entitlementsFileArg, + appGroupIdentifiersArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_app_group_identifiers", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Update the project's bundle identifier + + - parameters: + - xcodeproj: Path to your Xcode project + - plistPath: Path to info plist, relative to your Xcode project + - appIdentifier: The app Identifier you want to set + + Update an app identifier by either setting `CFBundleIdentifier` or `PRODUCT_BUNDLE_IDENTIFIER`, depending on which is already in use. + */ +public func updateAppIdentifier(xcodeproj: String, + plistPath: String, + appIdentifier: String) +{ + let xcodeprojArg = RubyCommand.Argument(name: "xcodeproj", value: xcodeproj, type: nil) + let plistPathArg = RubyCommand.Argument(name: "plist_path", value: plistPath, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let array: [RubyCommand.Argument?] = [xcodeprojArg, + plistPathArg, + appIdentifierArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_app_identifier", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Configures Xcode's Codesigning options + + - parameters: + - path: Path to your Xcode project + - useAutomaticSigning: Defines if project should use automatic signing + - sdk: Build target SDKs (iphoneos*, macosx*, iphonesimulator*) + - teamId: Team ID, is used when upgrading project + - targets: Specify targets you want to toggle the signing mech. (default to all targets) + - buildConfigurations: Specify build_configurations you want to toggle the signing mech. (default to all configurations) + - codeSignIdentity: Code signing identity type (iPhone Developer, iPhone Distribution) + - entitlementsFilePath: Path to your entitlements file + - profileName: Provisioning profile name to use for code signing + - profileUuid: Provisioning profile UUID to use for code signing + - bundleIdentifier: Application Product Bundle Identifier + + - returns: The current status (boolean) of codesigning after modification + + Configures Xcode's Codesigning options of all targets in the project + */ +public func updateCodeSigningSettings(path: String, + useAutomaticSigning: OptionalConfigValue = .fastlaneDefault(false), + sdk: OptionalConfigValue = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + targets: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + buildConfigurations: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + codeSignIdentity: OptionalConfigValue = .fastlaneDefault(nil), + entitlementsFilePath: OptionalConfigValue = .fastlaneDefault(nil), + profileName: OptionalConfigValue = .fastlaneDefault(nil), + profileUuid: OptionalConfigValue = .fastlaneDefault(nil), + bundleIdentifier: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let useAutomaticSigningArg = useAutomaticSigning.asRubyArgument(name: "use_automatic_signing", type: nil) + let sdkArg = sdk.asRubyArgument(name: "sdk", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let targetsArg = targets.asRubyArgument(name: "targets", type: nil) + let buildConfigurationsArg = buildConfigurations.asRubyArgument(name: "build_configurations", type: nil) + let codeSignIdentityArg = codeSignIdentity.asRubyArgument(name: "code_sign_identity", type: nil) + let entitlementsFilePathArg = entitlementsFilePath.asRubyArgument(name: "entitlements_file_path", type: nil) + let profileNameArg = profileName.asRubyArgument(name: "profile_name", type: nil) + let profileUuidArg = profileUuid.asRubyArgument(name: "profile_uuid", type: nil) + let bundleIdentifierArg = bundleIdentifier.asRubyArgument(name: "bundle_identifier", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + useAutomaticSigningArg, + sdkArg, + teamIdArg, + targetsArg, + buildConfigurationsArg, + codeSignIdentityArg, + entitlementsFilePathArg, + profileNameArg, + profileUuidArg, + bundleIdentifierArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_code_signing_settings", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Makes sure fastlane-tools are up-to-date when running fastlane + + - parameters: + - noUpdate: Don't update during this run. This is used internally + - nightly: **DEPRECATED!** Nightly builds are no longer being made available - Opt-in to install and use nightly fastlane builds + + This action will update fastlane to the most recent version - major version updates will not be performed automatically, as they might include breaking changes. If an update was performed, fastlane will be restarted before the run continues. + + If you are using rbenv or rvm, everything should be good to go. However, if you are using the system's default ruby, some additional setup is needed for this action to work correctly. In short, fastlane needs to be able to access your gem library without running in `sudo` mode. + + The simplest possible fix for this is putting the following lines into your `~/.bashrc` or `~/.zshrc` file:| + | + ```bash| + export GEM_HOME=~/.gems| + export PATH=$PATH:~/.gems/bin| + ```| + >| + After the above changes, restart your terminal, then run `mkdir $GEM_HOME` to create the new gem directory. After this, you're good to go! + + Recommended usage of the `update_fastlane` action is at the top inside of the `before_all` block, before running any other action. + */ +public func updateFastlane(noUpdate: OptionalConfigValue = .fastlaneDefault(false), + nightly: OptionalConfigValue = .fastlaneDefault(false)) +{ + let noUpdateArg = noUpdate.asRubyArgument(name: "no_update", type: nil) + let nightlyArg = nightly.asRubyArgument(name: "nightly", type: nil) + let array: [RubyCommand.Argument?] = [noUpdateArg, + nightlyArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_fastlane", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action changes the iCloud container identifiers in the entitlements file + + - parameters: + - entitlementsFile: The path to the entitlement file which contains the iCloud container identifiers + - icloudContainerIdentifiers: An Array of unique identifiers for the iCloud containers. Eg. ['iCloud.com.test.testapp'] + + Updates the iCloud Container Identifiers in the given Entitlements file, so you can use different iCloud containers for different builds like Adhoc, App Store, etc. + */ +public func updateIcloudContainerIdentifiers(entitlementsFile: String, + icloudContainerIdentifiers: [String]) +{ + let entitlementsFileArg = RubyCommand.Argument(name: "entitlements_file", value: entitlementsFile, type: nil) + let icloudContainerIdentifiersArg = RubyCommand.Argument(name: "icloud_container_identifiers", value: icloudContainerIdentifiers, type: nil) + let array: [RubyCommand.Argument?] = [entitlementsFileArg, + icloudContainerIdentifiersArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_icloud_container_identifiers", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Update a Info.plist file with bundle identifier and display name + + - parameters: + - xcodeproj: Path to your Xcode project + - plistPath: Path to info plist + - scheme: Scheme of info plist + - appIdentifier: The App Identifier of your app + - displayName: The Display Name of your app + - block: A block to process plist with custom logic + + This action allows you to modify your `Info.plist` file before building. This may be useful if you want a separate build for alpha, beta or nightly builds, but don't want a separate target. + */ +public func updateInfoPlist(xcodeproj: OptionalConfigValue = .fastlaneDefault(nil), + plistPath: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + displayName: OptionalConfigValue = .fastlaneDefault(nil), + block: ((String) -> Void)? = nil) +{ + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let plistPathArg = plistPath.asRubyArgument(name: "plist_path", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let displayNameArg = displayName.asRubyArgument(name: "display_name", type: nil) + let blockArg = RubyCommand.Argument(name: "block", value: block, type: .stringClosure) + let array: [RubyCommand.Argument?] = [xcodeprojArg, + plistPathArg, + schemeArg, + appIdentifierArg, + displayNameArg, + blockArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_info_plist", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + This action changes the keychain access groups in the entitlements file + + - parameters: + - entitlementsFile: The path to the entitlement file which contains the keychain access groups + - identifiers: An Array of unique identifiers for the keychain access groups. Eg. ['your.keychain.access.groups.identifiers'] + + Updates the Keychain Group Access Groups in the given Entitlements file, so you can have keychain access groups for the app store build and keychain access groups for an enterprise build. + */ +public func updateKeychainAccessGroups(entitlementsFile: String, + identifiers: [String]) +{ + let entitlementsFileArg = RubyCommand.Argument(name: "entitlements_file", value: entitlementsFile, type: nil) + let identifiersArg = RubyCommand.Argument(name: "identifiers", value: identifiers, type: nil) + let array: [RubyCommand.Argument?] = [entitlementsFileArg, + identifiersArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_keychain_access_groups", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Update a plist file + + - parameters: + - plistPath: Path to plist file + - block: A block to process plist with custom logic + + This action allows you to modify any value inside any `plist` file. + */ +public func updatePlist(plistPath: OptionalConfigValue = .fastlaneDefault(nil), + block: ((String) -> Void)? = nil) +{ + let plistPathArg = plistPath.asRubyArgument(name: "plist_path", type: nil) + let blockArg = RubyCommand.Argument(name: "block", value: block, type: .stringClosure) + let array: [RubyCommand.Argument?] = [plistPathArg, + blockArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_plist", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Updated code signing settings from 'Automatic' to a specific profile + + - parameters: + - path: Path to your Xcode project + - udid: **DEPRECATED!** Use `:uuid` instead + - uuid: The UUID of the provisioning profile you want to use + */ +public func updateProjectCodeSigning(path: String, + udid: OptionalConfigValue = .fastlaneDefault(nil), + uuid: String) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let udidArg = udid.asRubyArgument(name: "udid", type: nil) + let uuidArg = RubyCommand.Argument(name: "uuid", value: uuid, type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + udidArg, + uuidArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_project_code_signing", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Update projects code signing settings from your provisioning profile + + - parameters: + - xcodeproj: Path to your Xcode project + - profile: Path to provisioning profile (.mobileprovision) + - targetFilter: A filter for the target name. Use a standard regex + - buildConfigurationFilter: Legacy option, use 'target_filter' instead + - buildConfiguration: A filter for the build configuration name. Use a standard regex. Applied to all configurations if not specified + - certificate: Path to apple root certificate + - codeSigningIdentity: Code sign identity for build configuration + + You should check out the [code signing guide](https://docs.fastlane.tools/codesigning/getting-started/) before using this action. + This action retrieves a provisioning profile UUID from a provisioning profile (`.mobileprovision`) to set up the Xcode projects' code signing settings in `*.xcodeproj/project.pbxproj`. + The `:target_filter` value can be used to only update code signing for the specified targets. + The `:build_configuration` value can be used to only update code signing for the specified build configurations of the targets passing through the `:target_filter`. + Example usage is the WatchKit Extension or WatchKit App, where you need separate provisioning profiles. + Example: `update_project_provisioning(xcodeproj: "..", target_filter: ".*WatchKit App.*")`. + */ +public func updateProjectProvisioning(xcodeproj: OptionalConfigValue = .fastlaneDefault(nil), + profile: String, + targetFilter: OptionalConfigValue = .fastlaneDefault(nil), + buildConfigurationFilter: OptionalConfigValue = .fastlaneDefault(nil), + buildConfiguration: OptionalConfigValue = .fastlaneDefault(nil), + certificate: String = "/tmp/AppleIncRootCertificate.cer", + codeSigningIdentity: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let xcodeprojArg = xcodeproj.asRubyArgument(name: "xcodeproj", type: nil) + let profileArg = RubyCommand.Argument(name: "profile", value: profile, type: nil) + let targetFilterArg = targetFilter.asRubyArgument(name: "target_filter", type: nil) + let buildConfigurationFilterArg = buildConfigurationFilter.asRubyArgument(name: "build_configuration_filter", type: nil) + let buildConfigurationArg = buildConfiguration.asRubyArgument(name: "build_configuration", type: nil) + let certificateArg = RubyCommand.Argument(name: "certificate", value: certificate, type: nil) + let codeSigningIdentityArg = codeSigningIdentity.asRubyArgument(name: "code_signing_identity", type: nil) + let array: [RubyCommand.Argument?] = [xcodeprojArg, + profileArg, + targetFilterArg, + buildConfigurationFilterArg, + buildConfigurationArg, + certificateArg, + codeSigningIdentityArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_project_provisioning", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Update Xcode Development Team ID + + - parameters: + - path: Path to your Xcode project + - targets: Name of the targets you want to update + - teamid: The Team ID you want to use + + This action updates the Developer Team ID of your Xcode project. + */ +public func updateProjectTeam(path: String, + targets: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + teamid: String) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let targetsArg = targets.asRubyArgument(name: "targets", type: nil) + let teamidArg = RubyCommand.Argument(name: "teamid", value: teamid, type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + targetsArg, + teamidArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_project_team", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Set [Urban Airship](https://www.urbanairship.com/) plist configuration values + + - parameters: + - plistPath: Path to Urban Airship configuration Plist + - developmentAppKey: The development app key + - developmentAppSecret: The development app secret + - productionAppKey: The production app key + - productionAppSecret: The production app secret + - detectProvisioningMode: Automatically detect provisioning mode + + This action updates the `AirshipConfig.plist` needed to configure the Urban Airship SDK at runtime, allowing keys and secrets to easily be set for the Enterprise and Production versions of the application. + */ +public func updateUrbanAirshipConfiguration(plistPath: String, + developmentAppKey: OptionalConfigValue = .fastlaneDefault(nil), + developmentAppSecret: OptionalConfigValue = .fastlaneDefault(nil), + productionAppKey: OptionalConfigValue = .fastlaneDefault(nil), + productionAppSecret: OptionalConfigValue = .fastlaneDefault(nil), + detectProvisioningMode: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let plistPathArg = RubyCommand.Argument(name: "plist_path", value: plistPath, type: nil) + let developmentAppKeyArg = developmentAppKey.asRubyArgument(name: "development_app_key", type: nil) + let developmentAppSecretArg = developmentAppSecret.asRubyArgument(name: "development_app_secret", type: nil) + let productionAppKeyArg = productionAppKey.asRubyArgument(name: "production_app_key", type: nil) + let productionAppSecretArg = productionAppSecret.asRubyArgument(name: "production_app_secret", type: nil) + let detectProvisioningModeArg = detectProvisioningMode.asRubyArgument(name: "detect_provisioning_mode", type: nil) + let array: [RubyCommand.Argument?] = [plistPathArg, + developmentAppKeyArg, + developmentAppSecretArg, + productionAppKeyArg, + productionAppSecretArg, + detectProvisioningModeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_urban_airship_configuration", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Updates the URL schemes in the given Info.plist + + - parameters: + - path: The Plist file's path + - urlSchemes: The new URL schemes + - updateUrlSchemes: Block that is called to update schemes with current schemes passed in as parameter + + This action allows you to update the URL schemes of the app before building it. + For example, you can use this to set a different URL scheme for the alpha or beta version of the app. + */ +public func updateUrlSchemes(path: String, + urlSchemes: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + updateUrlSchemes: ((String) -> Void)? = nil) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let urlSchemesArg = urlSchemes.asRubyArgument(name: "url_schemes", type: nil) + let updateUrlSchemesArg = RubyCommand.Argument(name: "update_url_schemes", value: updateUrlSchemes, type: .stringClosure) + let array: [RubyCommand.Argument?] = [pathArg, + urlSchemesArg, + updateUrlSchemesArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "update_url_schemes", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload App Privacy Details for an app in App Store Connect + + - parameters: + - username: Your Apple ID Username for App Store Connect + - appIdentifier: The bundle identifier of your app + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - jsonPath: Path to the app usage data JSON + - outputJsonPath: Path to the app usage data JSON file generated by interactive questions + - skipJsonFileSaving: Whether to skip the saving of the JSON file + - skipUpload: Whether to skip the upload and only create the JSON file with interactive questions + - skipPublish: Whether to skip the publishing + + Upload App Privacy Details for an app in App Store Connect. For more detail information, view https://docs.fastlane.tools/uploading-app-privacy-details + */ +public func uploadAppPrivacyDetailsToAppStore(username: String, + appIdentifier: String, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + jsonPath: OptionalConfigValue = .fastlaneDefault(nil), + outputJsonPath: String = "./fastlane/app_privacy_details.json", + skipJsonFileSaving: OptionalConfigValue = .fastlaneDefault(false), + skipUpload: OptionalConfigValue = .fastlaneDefault(false), + skipPublish: OptionalConfigValue = .fastlaneDefault(false)) +{ + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let appIdentifierArg = RubyCommand.Argument(name: "app_identifier", value: appIdentifier, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let jsonPathArg = jsonPath.asRubyArgument(name: "json_path", type: nil) + let outputJsonPathArg = RubyCommand.Argument(name: "output_json_path", value: outputJsonPath, type: nil) + let skipJsonFileSavingArg = skipJsonFileSaving.asRubyArgument(name: "skip_json_file_saving", type: nil) + let skipUploadArg = skipUpload.asRubyArgument(name: "skip_upload", type: nil) + let skipPublishArg = skipPublish.asRubyArgument(name: "skip_publish", type: nil) + let array: [RubyCommand.Argument?] = [usernameArg, + appIdentifierArg, + teamIdArg, + teamNameArg, + jsonPathArg, + outputJsonPathArg, + skipJsonFileSavingArg, + skipUploadArg, + skipPublishArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_app_privacy_details_to_app_store", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload dSYM symbolication files to Crashlytics + + - parameters: + - dsymPath: Path to the DSYM file or zip to upload + - dsymPaths: Paths to the DSYM files or zips to upload + - apiToken: Crashlytics API Key + - gspPath: Path to GoogleService-Info.plist + - appId: Firebase Crashlytics APP ID + - binaryPath: The path to the upload-symbols file of the Fabric app + - platform: The platform of the app (ios, appletvos, mac) + - dsymWorkerThreads: The number of threads to use for simultaneous dSYM upload + - debug: Enable debug mode for upload-symbols + + This action allows you to upload symbolication files to Crashlytics. It's extra useful if you use it to download the latest dSYM files from Apple when you use Bitcode. This action will not fail the build if one of the uploads failed. The reason for that is that sometimes some of dSYM files are invalid, and we don't want them to fail the complete build. + */ +public func uploadSymbolsToCrashlytics(dsymPath: String = "./spec/fixtures/dSYM/Themoji2.dSYM", + dsymPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + apiToken: OptionalConfigValue = .fastlaneDefault(nil), + gspPath: OptionalConfigValue = .fastlaneDefault(nil), + appId: OptionalConfigValue = .fastlaneDefault(nil), + binaryPath: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + dsymWorkerThreads: Int = 1, + debug: OptionalConfigValue = .fastlaneDefault(false)) +{ + let dsymPathArg = RubyCommand.Argument(name: "dsym_path", value: dsymPath, type: nil) + let dsymPathsArg = dsymPaths.asRubyArgument(name: "dsym_paths", type: nil) + let apiTokenArg = apiToken.asRubyArgument(name: "api_token", type: nil) + let gspPathArg = gspPath.asRubyArgument(name: "gsp_path", type: nil) + let appIdArg = appId.asRubyArgument(name: "app_id", type: nil) + let binaryPathArg = binaryPath.asRubyArgument(name: "binary_path", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let dsymWorkerThreadsArg = RubyCommand.Argument(name: "dsym_worker_threads", value: dsymWorkerThreads, type: nil) + let debugArg = debug.asRubyArgument(name: "debug", type: nil) + let array: [RubyCommand.Argument?] = [dsymPathArg, + dsymPathsArg, + apiTokenArg, + gspPathArg, + appIdArg, + binaryPathArg, + platformArg, + dsymWorkerThreadsArg, + debugArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_symbols_to_crashlytics", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload dSYM symbolication files to Sentry + + - parameters: + - apiHost: API host url for Sentry + - apiKey: API key for Sentry + - authToken: Authentication token for Sentry + - orgSlug: Organization slug for Sentry project + - projectSlug: Project slug for Sentry + - dsymPath: Path to your symbols file. For iOS and Mac provide path to app.dSYM.zip + - dsymPaths: Path to an array of your symbols file. For iOS and Mac provide path to app.dSYM.zip + + - returns: The uploaded dSYM path(s) + + This action allows you to upload symbolication files to Sentry. It's extra useful if you use it to download the latest dSYM files from Apple when you use Bitcode. + */ +public func uploadSymbolsToSentry(apiHost: String = "https://app.getsentry.com/api/0", + apiKey: OptionalConfigValue = .fastlaneDefault(nil), + authToken: OptionalConfigValue = .fastlaneDefault(nil), + orgSlug: String, + projectSlug: String, + dsymPath: OptionalConfigValue = .fastlaneDefault(nil), + dsymPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil)) +{ + let apiHostArg = RubyCommand.Argument(name: "api_host", value: apiHost, type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let authTokenArg = authToken.asRubyArgument(name: "auth_token", type: nil) + let orgSlugArg = RubyCommand.Argument(name: "org_slug", value: orgSlug, type: nil) + let projectSlugArg = RubyCommand.Argument(name: "project_slug", value: projectSlug, type: nil) + let dsymPathArg = dsymPath.asRubyArgument(name: "dsym_path", type: nil) + let dsymPathsArg = dsymPaths.asRubyArgument(name: "dsym_paths", type: nil) + let array: [RubyCommand.Argument?] = [apiHostArg, + apiKeyArg, + authTokenArg, + orgSlugArg, + projectSlugArg, + dsymPathArg, + dsymPathsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_symbols_to_sentry", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload metadata and binary to App Store Connect (via _deliver_) + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - appIdentifier: The bundle identifier of your app + - appVersion: The version that should be edited or created + - ipa: Path to your ipa file + - pkg: Path to your pkg file + - buildNumber: If set the given build number (already uploaded to iTC) will be used instead of the current built one + - platform: The platform to use (optional) + - editLive: Modify live metadata, this option disables ipa upload and screenshot upload + - useLiveVersion: Force usage of live version rather than edit version + - metadataPath: Path to the folder containing the metadata files + - screenshotsPath: Path to the folder containing the screenshots + - skipBinaryUpload: Skip uploading an ipa or pkg to App Store Connect + - skipScreenshots: Don't upload the screenshots + - skipMetadata: Don't upload the metadata (e.g. title, description). This will still upload screenshots + - skipAppVersionUpdate: Don’t create or update the app version that is being prepared for submission + - force: Skip verification of HTML preview file + - overwriteScreenshots: Clear all previously uploaded screenshots before uploading the new ones + - syncScreenshots: Sync screenshots with local ones. This is currently beta option so set true to 'FASTLANE_ENABLE_BETA_DELIVER_SYNC_SCREENSHOTS' environment variable as well + - submitForReview: Submit the new version for Review after uploading everything + - verifyOnly: Verifies archive with App Store Connect without uploading + - rejectIfPossible: Rejects the previously submitted build if it's in a state where it's possible + - automaticRelease: Should the app be automatically released once it's approved? (Can not be used together with `auto_release_date`) + - autoReleaseDate: Date in milliseconds for automatically releasing on pending approval (Can not be used together with `automatic_release`) + - phasedRelease: Enable the phased release feature of iTC + - resetRatings: Reset the summary rating when you release a new version of the application + - priceTier: The price tier of this application + - appRatingConfigPath: Path to the app rating's config + - submissionInformation: Extra information for the submission (e.g. compliance specifications, IDFA settings) + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - devPortalTeamId: The short ID of your Developer Portal team, if you're in multiple teams. Different from your iTC team ID! + - devPortalTeamName: The name of your Developer Portal team if you're in multiple teams + - itcProvider: The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + - runPrecheckBeforeSubmit: Run precheck before submitting to app review + - precheckDefaultRuleLevel: The default precheck rule level unless otherwise configured + - individualMetadataItems: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - An array of localized metadata items to upload individually by language so that errors can be identified. E.g. ['name', 'keywords', 'description']. Note: slow + - appIcon: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the app icon + - appleWatchAppIcon: **DEPRECATED!** Removed after the migration to the new App Store Connect API in June 2020 - Metadata: The path to the Apple Watch app icon + - copyright: Metadata: The copyright notice + - primaryCategory: Metadata: The english name of the primary category (e.g. `Business`, `Books`) + - secondaryCategory: Metadata: The english name of the secondary category (e.g. `Business`, `Books`) + - primaryFirstSubCategory: Metadata: The english name of the primary first sub category (e.g. `Educational`, `Puzzle`) + - primarySecondSubCategory: Metadata: The english name of the primary second sub category (e.g. `Educational`, `Puzzle`) + - secondaryFirstSubCategory: Metadata: The english name of the secondary first sub category (e.g. `Educational`, `Puzzle`) + - secondarySecondSubCategory: Metadata: The english name of the secondary second sub category (e.g. `Educational`, `Puzzle`) + - tradeRepresentativeContactInformation: **DEPRECATED!** This is no longer used by App Store Connect - Metadata: A hash containing the trade representative contact information + - appReviewInformation: Metadata: A hash containing the review information + - appReviewAttachmentFile: Metadata: Path to the app review attachment file + - description: Metadata: The localised app description + - name: Metadata: The localised app name + - subtitle: Metadata: The localised app subtitle + - keywords: Metadata: An array of localised keywords + - promotionalText: Metadata: An array of localised promotional texts + - releaseNotes: Metadata: Localised release notes for this version + - privacyUrl: Metadata: Localised privacy url + - appleTvPrivacyPolicy: Metadata: Localised Apple TV privacy policy text + - supportUrl: Metadata: Localised support url + - marketingUrl: Metadata: Localised marketing url + - languages: Metadata: List of languages to activate + - ignoreLanguageDirectoryValidation: Ignore errors when invalid languages are found in metadata and screenshot directories + - precheckIncludeInAppPurchases: Should precheck check in-app purchases? + - app: The (spaceship) app ID of the app you want to use/modify + + Using _upload_to_app_store_ after _build_app_ and _capture_screenshots_ will automatically upload the latest ipa and screenshots with no other configuration. + + If you don't want to verify an HTML preview for App Store builds, use the `:force` option. + This is useful when running _fastlane_ on your Continuous Integration server: + `_upload_to_app_store_(force: true)` + If your account is on multiple teams and you need to tell the `iTMSTransporter` which 'provider' to use, you can set the `:itc_provider` option to pass this info. + */ +public func uploadToAppStore(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil), + pkg: OptionalConfigValue = .fastlaneDefault(nil), + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + platform: String = "ios", + editLive: OptionalConfigValue = .fastlaneDefault(false), + useLiveVersion: OptionalConfigValue = .fastlaneDefault(false), + metadataPath: OptionalConfigValue = .fastlaneDefault(nil), + screenshotsPath: OptionalConfigValue = .fastlaneDefault(nil), + skipBinaryUpload: OptionalConfigValue = .fastlaneDefault(false), + skipScreenshots: OptionalConfigValue = .fastlaneDefault(false), + skipMetadata: OptionalConfigValue = .fastlaneDefault(false), + skipAppVersionUpdate: OptionalConfigValue = .fastlaneDefault(false), + force: OptionalConfigValue = .fastlaneDefault(false), + overwriteScreenshots: OptionalConfigValue = .fastlaneDefault(false), + syncScreenshots: OptionalConfigValue = .fastlaneDefault(false), + submitForReview: OptionalConfigValue = .fastlaneDefault(false), + verifyOnly: OptionalConfigValue = .fastlaneDefault(false), + rejectIfPossible: OptionalConfigValue = .fastlaneDefault(false), + automaticRelease: OptionalConfigValue = .fastlaneDefault(nil), + autoReleaseDate: OptionalConfigValue = .fastlaneDefault(nil), + phasedRelease: OptionalConfigValue = .fastlaneDefault(false), + resetRatings: OptionalConfigValue = .fastlaneDefault(false), + priceTier: OptionalConfigValue = .fastlaneDefault(nil), + appRatingConfigPath: OptionalConfigValue = .fastlaneDefault(nil), + submissionInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + teamId: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamId: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamName: OptionalConfigValue = .fastlaneDefault(nil), + itcProvider: OptionalConfigValue = .fastlaneDefault(nil), + runPrecheckBeforeSubmit: OptionalConfigValue = .fastlaneDefault(true), + precheckDefaultRuleLevel: String = "warn", + individualMetadataItems: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + appIcon: OptionalConfigValue = .fastlaneDefault(nil), + appleWatchAppIcon: OptionalConfigValue = .fastlaneDefault(nil), + copyright: OptionalConfigValue = .fastlaneDefault(nil), + primaryCategory: OptionalConfigValue = .fastlaneDefault(nil), + secondaryCategory: OptionalConfigValue = .fastlaneDefault(nil), + primaryFirstSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + primarySecondSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + secondaryFirstSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + secondarySecondSubCategory: OptionalConfigValue = .fastlaneDefault(nil), + tradeRepresentativeContactInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appReviewInformation: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appReviewAttachmentFile: OptionalConfigValue = .fastlaneDefault(nil), + description: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + name: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + subtitle: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + keywords: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + promotionalText: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + releaseNotes: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + privacyUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + appleTvPrivacyPolicy: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + supportUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + marketingUrl: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + languages: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + ignoreLanguageDirectoryValidation: OptionalConfigValue = .fastlaneDefault(false), + precheckIncludeInAppPurchases: OptionalConfigValue = .fastlaneDefault(true), + app: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let pkgArg = pkg.asRubyArgument(name: "pkg", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let platformArg = RubyCommand.Argument(name: "platform", value: platform, type: nil) + let editLiveArg = editLive.asRubyArgument(name: "edit_live", type: nil) + let useLiveVersionArg = useLiveVersion.asRubyArgument(name: "use_live_version", type: nil) + let metadataPathArg = metadataPath.asRubyArgument(name: "metadata_path", type: nil) + let screenshotsPathArg = screenshotsPath.asRubyArgument(name: "screenshots_path", type: nil) + let skipBinaryUploadArg = skipBinaryUpload.asRubyArgument(name: "skip_binary_upload", type: nil) + let skipScreenshotsArg = skipScreenshots.asRubyArgument(name: "skip_screenshots", type: nil) + let skipMetadataArg = skipMetadata.asRubyArgument(name: "skip_metadata", type: nil) + let skipAppVersionUpdateArg = skipAppVersionUpdate.asRubyArgument(name: "skip_app_version_update", type: nil) + let forceArg = force.asRubyArgument(name: "force", type: nil) + let overwriteScreenshotsArg = overwriteScreenshots.asRubyArgument(name: "overwrite_screenshots", type: nil) + let syncScreenshotsArg = syncScreenshots.asRubyArgument(name: "sync_screenshots", type: nil) + let submitForReviewArg = submitForReview.asRubyArgument(name: "submit_for_review", type: nil) + let verifyOnlyArg = verifyOnly.asRubyArgument(name: "verify_only", type: nil) + let rejectIfPossibleArg = rejectIfPossible.asRubyArgument(name: "reject_if_possible", type: nil) + let automaticReleaseArg = automaticRelease.asRubyArgument(name: "automatic_release", type: nil) + let autoReleaseDateArg = autoReleaseDate.asRubyArgument(name: "auto_release_date", type: nil) + let phasedReleaseArg = phasedRelease.asRubyArgument(name: "phased_release", type: nil) + let resetRatingsArg = resetRatings.asRubyArgument(name: "reset_ratings", type: nil) + let priceTierArg = priceTier.asRubyArgument(name: "price_tier", type: nil) + let appRatingConfigPathArg = appRatingConfigPath.asRubyArgument(name: "app_rating_config_path", type: nil) + let submissionInformationArg = submissionInformation.asRubyArgument(name: "submission_information", type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let devPortalTeamIdArg = devPortalTeamId.asRubyArgument(name: "dev_portal_team_id", type: nil) + let devPortalTeamNameArg = devPortalTeamName.asRubyArgument(name: "dev_portal_team_name", type: nil) + let itcProviderArg = itcProvider.asRubyArgument(name: "itc_provider", type: nil) + let runPrecheckBeforeSubmitArg = runPrecheckBeforeSubmit.asRubyArgument(name: "run_precheck_before_submit", type: nil) + let precheckDefaultRuleLevelArg = RubyCommand.Argument(name: "precheck_default_rule_level", value: precheckDefaultRuleLevel, type: nil) + let individualMetadataItemsArg = individualMetadataItems.asRubyArgument(name: "individual_metadata_items", type: nil) + let appIconArg = appIcon.asRubyArgument(name: "app_icon", type: nil) + let appleWatchAppIconArg = appleWatchAppIcon.asRubyArgument(name: "apple_watch_app_icon", type: nil) + let copyrightArg = copyright.asRubyArgument(name: "copyright", type: nil) + let primaryCategoryArg = primaryCategory.asRubyArgument(name: "primary_category", type: nil) + let secondaryCategoryArg = secondaryCategory.asRubyArgument(name: "secondary_category", type: nil) + let primaryFirstSubCategoryArg = primaryFirstSubCategory.asRubyArgument(name: "primary_first_sub_category", type: nil) + let primarySecondSubCategoryArg = primarySecondSubCategory.asRubyArgument(name: "primary_second_sub_category", type: nil) + let secondaryFirstSubCategoryArg = secondaryFirstSubCategory.asRubyArgument(name: "secondary_first_sub_category", type: nil) + let secondarySecondSubCategoryArg = secondarySecondSubCategory.asRubyArgument(name: "secondary_second_sub_category", type: nil) + let tradeRepresentativeContactInformationArg = tradeRepresentativeContactInformation.asRubyArgument(name: "trade_representative_contact_information", type: nil) + let appReviewInformationArg = appReviewInformation.asRubyArgument(name: "app_review_information", type: nil) + let appReviewAttachmentFileArg = appReviewAttachmentFile.asRubyArgument(name: "app_review_attachment_file", type: nil) + let descriptionArg = description.asRubyArgument(name: "description", type: nil) + let nameArg = name.asRubyArgument(name: "name", type: nil) + let subtitleArg = subtitle.asRubyArgument(name: "subtitle", type: nil) + let keywordsArg = keywords.asRubyArgument(name: "keywords", type: nil) + let promotionalTextArg = promotionalText.asRubyArgument(name: "promotional_text", type: nil) + let releaseNotesArg = releaseNotes.asRubyArgument(name: "release_notes", type: nil) + let privacyUrlArg = privacyUrl.asRubyArgument(name: "privacy_url", type: nil) + let appleTvPrivacyPolicyArg = appleTvPrivacyPolicy.asRubyArgument(name: "apple_tv_privacy_policy", type: nil) + let supportUrlArg = supportUrl.asRubyArgument(name: "support_url", type: nil) + let marketingUrlArg = marketingUrl.asRubyArgument(name: "marketing_url", type: nil) + let languagesArg = languages.asRubyArgument(name: "languages", type: nil) + let ignoreLanguageDirectoryValidationArg = ignoreLanguageDirectoryValidation.asRubyArgument(name: "ignore_language_directory_validation", type: nil) + let precheckIncludeInAppPurchasesArg = precheckIncludeInAppPurchases.asRubyArgument(name: "precheck_include_in_app_purchases", type: nil) + let appArg = app.asRubyArgument(name: "app", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + appVersionArg, + ipaArg, + pkgArg, + buildNumberArg, + platformArg, + editLiveArg, + useLiveVersionArg, + metadataPathArg, + screenshotsPathArg, + skipBinaryUploadArg, + skipScreenshotsArg, + skipMetadataArg, + skipAppVersionUpdateArg, + forceArg, + overwriteScreenshotsArg, + syncScreenshotsArg, + submitForReviewArg, + verifyOnlyArg, + rejectIfPossibleArg, + automaticReleaseArg, + autoReleaseDateArg, + phasedReleaseArg, + resetRatingsArg, + priceTierArg, + appRatingConfigPathArg, + submissionInformationArg, + teamIdArg, + teamNameArg, + devPortalTeamIdArg, + devPortalTeamNameArg, + itcProviderArg, + runPrecheckBeforeSubmitArg, + precheckDefaultRuleLevelArg, + individualMetadataItemsArg, + appIconArg, + appleWatchAppIconArg, + copyrightArg, + primaryCategoryArg, + secondaryCategoryArg, + primaryFirstSubCategoryArg, + primarySecondSubCategoryArg, + secondaryFirstSubCategoryArg, + secondarySecondSubCategoryArg, + tradeRepresentativeContactInformationArg, + appReviewInformationArg, + appReviewAttachmentFileArg, + descriptionArg, + nameArg, + subtitleArg, + keywordsArg, + promotionalTextArg, + releaseNotesArg, + privacyUrlArg, + appleTvPrivacyPolicyArg, + supportUrlArg, + marketingUrlArg, + languagesArg, + ignoreLanguageDirectoryValidationArg, + precheckIncludeInAppPurchasesArg, + appArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_to_app_store", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload metadata, screenshots and binaries to Google Play (via _supply_) + + - parameters: + - packageName: The package name of the application to use + - versionName: Version name (used when uploading new apks/aabs) - defaults to 'versionName' in build.gradle or AndroidManifest.xml + - versionCode: Version code (used when updating rollout or promoting specific versions) + - releaseStatus: Release status (used when uploading new apks/aabs) - valid values are completed, draft, halted, inProgress + - track: The track of the application to use. The default available tracks are: production, beta, alpha, internal + - rollout: The percentage of the user fraction when uploading to the rollout track (setting to 1 will complete the rollout) + - metadataPath: Path to the directory containing the metadata files + - key: **DEPRECATED!** Use `--json_key` instead - The p12 File used to authenticate with Google + - issuer: **DEPRECATED!** Use `--json_key` instead - The issuer of the p12 file (email address of the service account) + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - apk: Path to the APK file to upload + - apkPaths: An array of paths to APK files to upload + - aab: Path to the AAB file to upload + - aabPaths: An array of paths to AAB files to upload + - skipUploadApk: Whether to skip uploading APK + - skipUploadAab: Whether to skip uploading AAB + - skipUploadMetadata: Whether to skip uploading metadata, changelogs not included + - skipUploadChangelogs: Whether to skip uploading changelogs + - skipUploadImages: Whether to skip uploading images, screenshots not included + - skipUploadScreenshots: Whether to skip uploading SCREENSHOTS + - trackPromoteTo: The track to promote to. The default available tracks are: production, beta, alpha, internal + - trackPromoteReleaseStatus: Promoted track release status (used when promoting a track) - valid values are completed, draft, halted, inProgress + - validateOnly: Only validate changes with Google Play rather than actually publish + - mapping: Path to the mapping file to upload (mapping.txt or native-debug-symbols.zip alike) + - mappingPaths: An array of paths to mapping files to upload (mapping.txt or native-debug-symbols.zip alike) + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - checkSupersededTracks: **DEPRECATED!** Google Play does this automatically now - Check the other tracks for superseded versions and disable them + - timeout: Timeout for read, open, and send (in seconds) + - deactivateOnPromote: **DEPRECATED!** Google Play does this automatically now - When promoting to a new track, deactivate the binary in the origin track + - versionCodesToRetain: An array of version codes to retain when publishing a new APK + - changesNotSentForReview: Indicates that the changes in this edit will not be reviewed until they are explicitly sent for review from the Google Play Console UI + - rescueChangesNotSentForReview: Catches changes_not_sent_for_review errors when an edit is committed and retries with the configuration that the error message recommended + - inAppUpdatePriority: In-app update priority for all the newly added apks in the release. Can take values between [0,5] + - obbMainReferencesVersion: References version of 'main' expansion file + - obbMainFileSize: Size of 'main' expansion file in bytes + - obbPatchReferencesVersion: References version of 'patch' expansion file + - obbPatchFileSize: Size of 'patch' expansion file in bytes + - ackBundleInstallationWarning: Must be set to true if the bundle installation may trigger a warning on user devices (e.g can only be downloaded over wifi). Typically this is required for bundles over 150MB + + More information: https://docs.fastlane.tools/actions/supply/ + */ +public func uploadToPlayStore(packageName: String, + versionName: OptionalConfigValue = .fastlaneDefault(nil), + versionCode: OptionalConfigValue = .fastlaneDefault(nil), + releaseStatus: String = "completed", + track: String = "production", + rollout: OptionalConfigValue = .fastlaneDefault(nil), + metadataPath: OptionalConfigValue = .fastlaneDefault(nil), + key: OptionalConfigValue = .fastlaneDefault(nil), + issuer: OptionalConfigValue = .fastlaneDefault(nil), + jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + apk: OptionalConfigValue = .fastlaneDefault(nil), + apkPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + aab: OptionalConfigValue = .fastlaneDefault(nil), + aabPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + skipUploadApk: OptionalConfigValue = .fastlaneDefault(false), + skipUploadAab: OptionalConfigValue = .fastlaneDefault(false), + skipUploadMetadata: OptionalConfigValue = .fastlaneDefault(false), + skipUploadChangelogs: OptionalConfigValue = .fastlaneDefault(false), + skipUploadImages: OptionalConfigValue = .fastlaneDefault(false), + skipUploadScreenshots: OptionalConfigValue = .fastlaneDefault(false), + trackPromoteTo: OptionalConfigValue = .fastlaneDefault(nil), + trackPromoteReleaseStatus: String = "completed", + validateOnly: OptionalConfigValue = .fastlaneDefault(false), + mapping: OptionalConfigValue = .fastlaneDefault(nil), + mappingPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + checkSupersededTracks: OptionalConfigValue = .fastlaneDefault(false), + timeout: Int = 300, + deactivateOnPromote: OptionalConfigValue = .fastlaneDefault(true), + versionCodesToRetain: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + changesNotSentForReview: OptionalConfigValue = .fastlaneDefault(false), + rescueChangesNotSentForReview: OptionalConfigValue = .fastlaneDefault(true), + inAppUpdatePriority: OptionalConfigValue = .fastlaneDefault(nil), + obbMainReferencesVersion: OptionalConfigValue = .fastlaneDefault(nil), + obbMainFileSize: OptionalConfigValue = .fastlaneDefault(nil), + obbPatchReferencesVersion: OptionalConfigValue = .fastlaneDefault(nil), + obbPatchFileSize: OptionalConfigValue = .fastlaneDefault(nil), + ackBundleInstallationWarning: OptionalConfigValue = .fastlaneDefault(false)) +{ + let packageNameArg = RubyCommand.Argument(name: "package_name", value: packageName, type: nil) + let versionNameArg = versionName.asRubyArgument(name: "version_name", type: nil) + let versionCodeArg = versionCode.asRubyArgument(name: "version_code", type: nil) + let releaseStatusArg = RubyCommand.Argument(name: "release_status", value: releaseStatus, type: nil) + let trackArg = RubyCommand.Argument(name: "track", value: track, type: nil) + let rolloutArg = rollout.asRubyArgument(name: "rollout", type: nil) + let metadataPathArg = metadataPath.asRubyArgument(name: "metadata_path", type: nil) + let keyArg = key.asRubyArgument(name: "key", type: nil) + let issuerArg = issuer.asRubyArgument(name: "issuer", type: nil) + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let apkArg = apk.asRubyArgument(name: "apk", type: nil) + let apkPathsArg = apkPaths.asRubyArgument(name: "apk_paths", type: nil) + let aabArg = aab.asRubyArgument(name: "aab", type: nil) + let aabPathsArg = aabPaths.asRubyArgument(name: "aab_paths", type: nil) + let skipUploadApkArg = skipUploadApk.asRubyArgument(name: "skip_upload_apk", type: nil) + let skipUploadAabArg = skipUploadAab.asRubyArgument(name: "skip_upload_aab", type: nil) + let skipUploadMetadataArg = skipUploadMetadata.asRubyArgument(name: "skip_upload_metadata", type: nil) + let skipUploadChangelogsArg = skipUploadChangelogs.asRubyArgument(name: "skip_upload_changelogs", type: nil) + let skipUploadImagesArg = skipUploadImages.asRubyArgument(name: "skip_upload_images", type: nil) + let skipUploadScreenshotsArg = skipUploadScreenshots.asRubyArgument(name: "skip_upload_screenshots", type: nil) + let trackPromoteToArg = trackPromoteTo.asRubyArgument(name: "track_promote_to", type: nil) + let trackPromoteReleaseStatusArg = RubyCommand.Argument(name: "track_promote_release_status", value: trackPromoteReleaseStatus, type: nil) + let validateOnlyArg = validateOnly.asRubyArgument(name: "validate_only", type: nil) + let mappingArg = mapping.asRubyArgument(name: "mapping", type: nil) + let mappingPathsArg = mappingPaths.asRubyArgument(name: "mapping_paths", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let checkSupersededTracksArg = checkSupersededTracks.asRubyArgument(name: "check_superseded_tracks", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let deactivateOnPromoteArg = deactivateOnPromote.asRubyArgument(name: "deactivate_on_promote", type: nil) + let versionCodesToRetainArg = versionCodesToRetain.asRubyArgument(name: "version_codes_to_retain", type: nil) + let changesNotSentForReviewArg = changesNotSentForReview.asRubyArgument(name: "changes_not_sent_for_review", type: nil) + let rescueChangesNotSentForReviewArg = rescueChangesNotSentForReview.asRubyArgument(name: "rescue_changes_not_sent_for_review", type: nil) + let inAppUpdatePriorityArg = inAppUpdatePriority.asRubyArgument(name: "in_app_update_priority", type: nil) + let obbMainReferencesVersionArg = obbMainReferencesVersion.asRubyArgument(name: "obb_main_references_version", type: nil) + let obbMainFileSizeArg = obbMainFileSize.asRubyArgument(name: "obb_main_file_size", type: nil) + let obbPatchReferencesVersionArg = obbPatchReferencesVersion.asRubyArgument(name: "obb_patch_references_version", type: nil) + let obbPatchFileSizeArg = obbPatchFileSize.asRubyArgument(name: "obb_patch_file_size", type: nil) + let ackBundleInstallationWarningArg = ackBundleInstallationWarning.asRubyArgument(name: "ack_bundle_installation_warning", type: nil) + let array: [RubyCommand.Argument?] = [packageNameArg, + versionNameArg, + versionCodeArg, + releaseStatusArg, + trackArg, + rolloutArg, + metadataPathArg, + keyArg, + issuerArg, + jsonKeyArg, + jsonKeyDataArg, + apkArg, + apkPathsArg, + aabArg, + aabPathsArg, + skipUploadApkArg, + skipUploadAabArg, + skipUploadMetadataArg, + skipUploadChangelogsArg, + skipUploadImagesArg, + skipUploadScreenshotsArg, + trackPromoteToArg, + trackPromoteReleaseStatusArg, + validateOnlyArg, + mappingArg, + mappingPathsArg, + rootUrlArg, + checkSupersededTracksArg, + timeoutArg, + deactivateOnPromoteArg, + versionCodesToRetainArg, + changesNotSentForReviewArg, + rescueChangesNotSentForReviewArg, + inAppUpdatePriorityArg, + obbMainReferencesVersionArg, + obbMainFileSizeArg, + obbPatchReferencesVersionArg, + obbPatchFileSizeArg, + ackBundleInstallationWarningArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_to_play_store", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload binaries to Google Play Internal App Sharing (via _supply_) + + - parameters: + - packageName: The package name of the application to use + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - apk: Path to the APK file to upload + - apkPaths: An array of paths to APK files to upload + - aab: Path to the AAB file to upload + - aabPaths: An array of paths to AAB files to upload + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - timeout: Timeout for read, open, and send (in seconds) + + - returns: Returns a string containing the download URL for the uploaded APK/AAB (or array of strings if multiple were uploaded). + + More information: https://docs.fastlane.tools/actions/upload_to_play_store_internal_app_sharing/ + */ +public func uploadToPlayStoreInternalAppSharing(packageName: String, + jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + apk: OptionalConfigValue = .fastlaneDefault(nil), + apkPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + aab: OptionalConfigValue = .fastlaneDefault(nil), + aabPaths: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 300) +{ + let packageNameArg = RubyCommand.Argument(name: "package_name", value: packageName, type: nil) + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let apkArg = apk.asRubyArgument(name: "apk", type: nil) + let apkPathsArg = apkPaths.asRubyArgument(name: "apk_paths", type: nil) + let aabArg = aab.asRubyArgument(name: "aab", type: nil) + let aabPathsArg = aabPaths.asRubyArgument(name: "aab_paths", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [packageNameArg, + jsonKeyArg, + jsonKeyDataArg, + apkArg, + apkPathsArg, + aabArg, + aabPathsArg, + rootUrlArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_to_play_store_internal_app_sharing", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Upload new binary to App Store Connect for TestFlight beta testing (via _pilot_) + + - parameters: + - apiKeyPath: Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + - apiKey: Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + - username: Your Apple ID Username + - appIdentifier: The bundle identifier of the app to upload or manage testers (optional) + - appPlatform: The platform to use (optional) + - appleId: Apple ID property in the App Information section in App Store Connect + - ipa: Path to the ipa file to upload + - pkg: Path to your pkg file + - demoAccountRequired: Do you need a demo account when Apple does review? + - betaAppReviewInfo: Beta app review information for contact info and demo account + - localizedAppInfo: Localized beta app test info for description, feedback email, marketing url, and privacy policy + - betaAppDescription: Provide the 'Beta App Description' when uploading a new build + - betaAppFeedbackEmail: Provide the beta app email when uploading a new build + - localizedBuildInfo: Localized beta app test info for what's new + - changelog: Provide the 'What to Test' text when uploading a new build + - skipSubmission: Skip the distributing action of pilot and only upload the ipa file + - skipWaitingForBuildProcessing: If set to true, the `distribute_external` option won't work and no build will be distributed to testers. (You might want to use this option if you are using this action on CI and have to pay for 'minutes used' on your CI plan). If set to `true` and a changelog is provided, it will partially wait for the build to appear on AppStore Connect so the changelog can be set, and skip the remaining processing steps + - updateBuildInfoOnUpload: **DEPRECATED!** Update build info immediately after validation. This is deprecated and will be removed in a future release. App Store Connect no longer supports setting build info until after build processing has completed, which is when build info is updated by default + - distributeOnly: Distribute a previously uploaded build (equivalent to the `fastlane pilot distribute` command) + - usesNonExemptEncryption: Provide the 'Uses Non-Exempt Encryption' for export compliance. This is used if there is 'ITSAppUsesNonExemptEncryption' is not set in the Info.plist + - distributeExternal: Should the build be distributed to external testers? If set to true, use of `groups` option is required + - notifyExternalTesters: Should notify external testers? (Not setting a value will use App Store Connect's default which is to notify) + - appVersion: The version number of the application build to distribute. If the version number is not specified, then the most recent build uploaded to TestFlight will be distributed. If specified, the most recent build for the version number will be distributed + - buildNumber: The build number of the application build to distribute. If the build number is not specified, the most recent build is distributed + - expirePreviousBuilds: Should expire previous builds? + - firstName: The tester's first name + - lastName: The tester's last name + - email: The tester's email + - testersFilePath: Path to a CSV file of testers + - groups: Associate tester to one group or more by group name / group id. E.g. `-g "Team 1","Team 2"` This is required when `distribute_external` option is set to true or when we want to add a tester to one or more external testing groups + - teamId: The ID of your App Store Connect team if you're in multiple teams + - teamName: The name of your App Store Connect team if you're in multiple teams + - devPortalTeamId: The short ID of your team in the developer portal, if you're in multiple teams. Different from your iTC team ID! + - itcProvider: The provider short name to be used with the iTMSTransporter to identify your team. This value will override the automatically detected provider short name. To get provider short name run `pathToXcode.app/Contents/Applications/Application\ Loader.app/Contents/itms/bin/iTMSTransporter -m provider -u 'USERNAME' -p 'PASSWORD' -account_type itunes_connect -v off`. The short names of providers should be listed in the second column + - waitProcessingInterval: Interval in seconds to wait for App Store Connect processing + - waitProcessingTimeoutDuration: Timeout duration in seconds to wait for App Store Connect processing. If set, after exceeding timeout duration, this will `force stop` to wait for App Store Connect processing and exit with exception + - waitForUploadedBuild: **DEPRECATED!** No longer needed with the transition over to the App Store Connect API - Use version info from uploaded ipa file to determine what build to use for distribution. If set to false, latest processing or any latest build will be used + - rejectBuildWaitingForReview: Expire previous if it's 'waiting for review' + - submitBetaReview: Send the build for a beta review + + More details can be found on https://docs.fastlane.tools/actions/pilot/. + This integration will only do the TestFlight upload. + */ +public func uploadToTestflight(apiKeyPath: OptionalConfigValue = .fastlaneDefault(nil), + apiKey: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + username: OptionalConfigValue = .fastlaneDefault(nil), + appIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + appPlatform: OptionalConfigValue = .fastlaneDefault(nil), + appleId: OptionalConfigValue = .fastlaneDefault(nil), + ipa: OptionalConfigValue = .fastlaneDefault(nil), + pkg: OptionalConfigValue = .fastlaneDefault(nil), + demoAccountRequired: OptionalConfigValue = .fastlaneDefault(nil), + betaAppReviewInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + localizedAppInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + betaAppDescription: OptionalConfigValue = .fastlaneDefault(nil), + betaAppFeedbackEmail: OptionalConfigValue = .fastlaneDefault(nil), + localizedBuildInfo: OptionalConfigValue<[String: Any]?> = .fastlaneDefault(nil), + changelog: OptionalConfigValue = .fastlaneDefault(nil), + skipSubmission: OptionalConfigValue = .fastlaneDefault(false), + skipWaitingForBuildProcessing: OptionalConfigValue = .fastlaneDefault(false), + updateBuildInfoOnUpload: OptionalConfigValue = .fastlaneDefault(false), + distributeOnly: OptionalConfigValue = .fastlaneDefault(false), + usesNonExemptEncryption: OptionalConfigValue = .fastlaneDefault(false), + distributeExternal: OptionalConfigValue = .fastlaneDefault(false), + notifyExternalTesters: Any? = nil, + appVersion: OptionalConfigValue = .fastlaneDefault(nil), + buildNumber: OptionalConfigValue = .fastlaneDefault(nil), + expirePreviousBuilds: OptionalConfigValue = .fastlaneDefault(false), + firstName: OptionalConfigValue = .fastlaneDefault(nil), + lastName: OptionalConfigValue = .fastlaneDefault(nil), + email: OptionalConfigValue = .fastlaneDefault(nil), + testersFilePath: String = "./testers.csv", + groups: OptionalConfigValue<[String]?> = .fastlaneDefault(nil), + teamId: Any? = nil, + teamName: OptionalConfigValue = .fastlaneDefault(nil), + devPortalTeamId: OptionalConfigValue = .fastlaneDefault(nil), + itcProvider: OptionalConfigValue = .fastlaneDefault(nil), + waitProcessingInterval: Int = 30, + waitProcessingTimeoutDuration: OptionalConfigValue = .fastlaneDefault(nil), + waitForUploadedBuild: OptionalConfigValue = .fastlaneDefault(false), + rejectBuildWaitingForReview: OptionalConfigValue = .fastlaneDefault(false), + submitBetaReview: OptionalConfigValue = .fastlaneDefault(true)) +{ + let apiKeyPathArg = apiKeyPath.asRubyArgument(name: "api_key_path", type: nil) + let apiKeyArg = apiKey.asRubyArgument(name: "api_key", type: nil) + let usernameArg = username.asRubyArgument(name: "username", type: nil) + let appIdentifierArg = appIdentifier.asRubyArgument(name: "app_identifier", type: nil) + let appPlatformArg = appPlatform.asRubyArgument(name: "app_platform", type: nil) + let appleIdArg = appleId.asRubyArgument(name: "apple_id", type: nil) + let ipaArg = ipa.asRubyArgument(name: "ipa", type: nil) + let pkgArg = pkg.asRubyArgument(name: "pkg", type: nil) + let demoAccountRequiredArg = demoAccountRequired.asRubyArgument(name: "demo_account_required", type: nil) + let betaAppReviewInfoArg = betaAppReviewInfo.asRubyArgument(name: "beta_app_review_info", type: nil) + let localizedAppInfoArg = localizedAppInfo.asRubyArgument(name: "localized_app_info", type: nil) + let betaAppDescriptionArg = betaAppDescription.asRubyArgument(name: "beta_app_description", type: nil) + let betaAppFeedbackEmailArg = betaAppFeedbackEmail.asRubyArgument(name: "beta_app_feedback_email", type: nil) + let localizedBuildInfoArg = localizedBuildInfo.asRubyArgument(name: "localized_build_info", type: nil) + let changelogArg = changelog.asRubyArgument(name: "changelog", type: nil) + let skipSubmissionArg = skipSubmission.asRubyArgument(name: "skip_submission", type: nil) + let skipWaitingForBuildProcessingArg = skipWaitingForBuildProcessing.asRubyArgument(name: "skip_waiting_for_build_processing", type: nil) + let updateBuildInfoOnUploadArg = updateBuildInfoOnUpload.asRubyArgument(name: "update_build_info_on_upload", type: nil) + let distributeOnlyArg = distributeOnly.asRubyArgument(name: "distribute_only", type: nil) + let usesNonExemptEncryptionArg = usesNonExemptEncryption.asRubyArgument(name: "uses_non_exempt_encryption", type: nil) + let distributeExternalArg = distributeExternal.asRubyArgument(name: "distribute_external", type: nil) + let notifyExternalTestersArg = RubyCommand.Argument(name: "notify_external_testers", value: notifyExternalTesters, type: nil) + let appVersionArg = appVersion.asRubyArgument(name: "app_version", type: nil) + let buildNumberArg = buildNumber.asRubyArgument(name: "build_number", type: nil) + let expirePreviousBuildsArg = expirePreviousBuilds.asRubyArgument(name: "expire_previous_builds", type: nil) + let firstNameArg = firstName.asRubyArgument(name: "first_name", type: nil) + let lastNameArg = lastName.asRubyArgument(name: "last_name", type: nil) + let emailArg = email.asRubyArgument(name: "email", type: nil) + let testersFilePathArg = RubyCommand.Argument(name: "testers_file_path", value: testersFilePath, type: nil) + let groupsArg = groups.asRubyArgument(name: "groups", type: nil) + let teamIdArg = RubyCommand.Argument(name: "team_id", value: teamId, type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let devPortalTeamIdArg = devPortalTeamId.asRubyArgument(name: "dev_portal_team_id", type: nil) + let itcProviderArg = itcProvider.asRubyArgument(name: "itc_provider", type: nil) + let waitProcessingIntervalArg = RubyCommand.Argument(name: "wait_processing_interval", value: waitProcessingInterval, type: nil) + let waitProcessingTimeoutDurationArg = waitProcessingTimeoutDuration.asRubyArgument(name: "wait_processing_timeout_duration", type: nil) + let waitForUploadedBuildArg = waitForUploadedBuild.asRubyArgument(name: "wait_for_uploaded_build", type: nil) + let rejectBuildWaitingForReviewArg = rejectBuildWaitingForReview.asRubyArgument(name: "reject_build_waiting_for_review", type: nil) + let submitBetaReviewArg = submitBetaReview.asRubyArgument(name: "submit_beta_review", type: nil) + let array: [RubyCommand.Argument?] = [apiKeyPathArg, + apiKeyArg, + usernameArg, + appIdentifierArg, + appPlatformArg, + appleIdArg, + ipaArg, + pkgArg, + demoAccountRequiredArg, + betaAppReviewInfoArg, + localizedAppInfoArg, + betaAppDescriptionArg, + betaAppFeedbackEmailArg, + localizedBuildInfoArg, + changelogArg, + skipSubmissionArg, + skipWaitingForBuildProcessingArg, + updateBuildInfoOnUploadArg, + distributeOnlyArg, + usesNonExemptEncryptionArg, + distributeExternalArg, + notifyExternalTestersArg, + appVersionArg, + buildNumberArg, + expirePreviousBuildsArg, + firstNameArg, + lastNameArg, + emailArg, + testersFilePathArg, + groupsArg, + teamIdArg, + teamNameArg, + devPortalTeamIdArg, + itcProviderArg, + waitProcessingIntervalArg, + waitProcessingTimeoutDurationArg, + waitForUploadedBuildArg, + rejectBuildWaitingForReviewArg, + submitBetaReviewArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "upload_to_testflight", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Validate that the Google Play Store `json_key` works + + - parameters: + - jsonKey: The path to a file containing service account JSON, used to authenticate with Google + - jsonKeyData: The raw service account JSON data used to authenticate with Google + - rootUrl: Root URL for the Google Play API. The provided URL will be used for API calls in place of https://www.googleapis.com/ + - timeout: Timeout for read, open, and send (in seconds) + + Use this action to test and validate your private key json key file used to connect and authenticate with the Google Play API + */ +public func validatePlayStoreJsonKey(jsonKey: OptionalConfigValue = .fastlaneDefault(nil), + jsonKeyData: OptionalConfigValue = .fastlaneDefault(nil), + rootUrl: OptionalConfigValue = .fastlaneDefault(nil), + timeout: Int = 300) +{ + let jsonKeyArg = jsonKey.asRubyArgument(name: "json_key", type: nil) + let jsonKeyDataArg = jsonKeyData.asRubyArgument(name: "json_key_data", type: nil) + let rootUrlArg = rootUrl.asRubyArgument(name: "root_url", type: nil) + let timeoutArg = RubyCommand.Argument(name: "timeout", value: timeout, type: nil) + let array: [RubyCommand.Argument?] = [jsonKeyArg, + jsonKeyDataArg, + rootUrlArg, + timeoutArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "validate_play_store_json_key", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Able to verify various settings in ipa file + + - parameters: + - provisioningType: Required type of provisioning + - provisioningUuid: Required UUID of provisioning profile + - teamIdentifier: Required team identifier + - teamName: Required team name + - appName: Required app name + - bundleIdentifier: Required bundle identifier + - ipaPath: Explicitly set the ipa path + - buildPath: Explicitly set the ipa, app or xcarchive path + + Verifies that the built app was built using the expected build resources. This is relevant for people who build on machines that are used to build apps with different profiles, certificates and/or bundle identifiers to guard against configuration mistakes. + */ +public func verifyBuild(provisioningType: OptionalConfigValue = .fastlaneDefault(nil), + provisioningUuid: OptionalConfigValue = .fastlaneDefault(nil), + teamIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + teamName: OptionalConfigValue = .fastlaneDefault(nil), + appName: OptionalConfigValue = .fastlaneDefault(nil), + bundleIdentifier: OptionalConfigValue = .fastlaneDefault(nil), + ipaPath: OptionalConfigValue = .fastlaneDefault(nil), + buildPath: OptionalConfigValue = .fastlaneDefault(nil)) +{ + let provisioningTypeArg = provisioningType.asRubyArgument(name: "provisioning_type", type: nil) + let provisioningUuidArg = provisioningUuid.asRubyArgument(name: "provisioning_uuid", type: nil) + let teamIdentifierArg = teamIdentifier.asRubyArgument(name: "team_identifier", type: nil) + let teamNameArg = teamName.asRubyArgument(name: "team_name", type: nil) + let appNameArg = appName.asRubyArgument(name: "app_name", type: nil) + let bundleIdentifierArg = bundleIdentifier.asRubyArgument(name: "bundle_identifier", type: nil) + let ipaPathArg = ipaPath.asRubyArgument(name: "ipa_path", type: nil) + let buildPathArg = buildPath.asRubyArgument(name: "build_path", type: nil) + let array: [RubyCommand.Argument?] = [provisioningTypeArg, + provisioningUuidArg, + teamIdentifierArg, + teamNameArg, + appNameArg, + bundleIdentifierArg, + ipaPathArg, + buildPathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "verify_build", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Verifies all keys referenced from the Podfile are non-empty + + Runs a check against all keys specified in your Podfile to make sure they're more than a single character long. This is to ensure you don't deploy with stubbed keys. + */ +public func verifyPodKeys() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "verify_pod_keys", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Verifies that the Xcode installation is properly signed by Apple + + - parameter xcodePath: The path to the Xcode installation to test + + This action was implemented after the recent Xcode attack to make sure you're not using a [hacked Xcode installation](http://researchcenter.paloaltonetworks.com/2015/09/novel-malware-xcodeghost-modifies-xcode-infects-apple-ios-apps-and-hits-app-store/). + */ +public func verifyXcode(xcodePath: String) { + let xcodePathArg = RubyCommand.Argument(name: "xcode_path", value: xcodePath, type: nil) + let array: [RubyCommand.Argument?] = [xcodePathArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "verify_xcode", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Increment or set the version in a podspec file + + - parameters: + - path: You must specify the path to the podspec file to update + - bumpType: The type of this version bump. Available: patch, minor, major + - versionNumber: Change to a specific version. This will replace the bump type value + - versionAppendix: Change version appendix to a specific value. For example 1.4.14.4.1 -> 1.4.14.5 + - requireVariablePrefix: true by default, this is used for non CocoaPods version bumps only + + You can use this action to manipulate any 'version' variable contained in a ruby file. + For example, you can use it to bump the version of a CocoaPods' podspec file. + It also supports versions that are not semantic: `1.4.14.4.1`. + For such versions, there is an option to change the appendix (e.g. `4.1`). + */ +public func versionBumpPodspec(path: String, + bumpType: String = "patch", + versionNumber: OptionalConfigValue = .fastlaneDefault(nil), + versionAppendix: OptionalConfigValue = .fastlaneDefault(nil), + requireVariablePrefix: OptionalConfigValue = .fastlaneDefault(true)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let bumpTypeArg = RubyCommand.Argument(name: "bump_type", value: bumpType, type: nil) + let versionNumberArg = versionNumber.asRubyArgument(name: "version_number", type: nil) + let versionAppendixArg = versionAppendix.asRubyArgument(name: "version_appendix", type: nil) + let requireVariablePrefixArg = requireVariablePrefix.asRubyArgument(name: "require_variable_prefix", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + bumpTypeArg, + versionNumberArg, + versionAppendixArg, + requireVariablePrefixArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "version_bump_podspec", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Receive the version number from a podspec file + + - parameters: + - path: You must specify the path to the podspec file + - requireVariablePrefix: true by default, this is used for non CocoaPods version bumps only + */ +public func versionGetPodspec(path: String, + requireVariablePrefix: OptionalConfigValue = .fastlaneDefault(true)) +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let requireVariablePrefixArg = requireVariablePrefix.asRubyArgument(name: "require_variable_prefix", type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + requireVariablePrefixArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "version_get_podspec", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Archives the project using `xcodebuild` + */ +public func xcarchive() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xcarchive", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Builds the project using `xcodebuild` + */ +public func xcbuild() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xcbuild", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Cleans the project using `xcodebuild` + */ +public func xcclean() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xcclean", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Exports the project using `xcodebuild` + */ +public func xcexport() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xcexport", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Make sure a certain version of Xcode is installed + + - parameters: + - version: The version number of the version of Xcode to install + - username: Your Apple ID Username + - teamId: The ID of your team if you're in multiple teams + - downloadRetryAttempts: Number of times the download will be retried in case of failure + + - returns: The path to the newly installed Xcode version + + Makes sure a specific version of Xcode is installed. If that's not the case, it will automatically be downloaded by the [xcode_install](https://github.com/neonichu/xcode-install) gem. This will make sure to use the correct Xcode for later actions. + */ +@discardableResult public func xcodeInstall(version: String, + username: String, + teamId: OptionalConfigValue = .fastlaneDefault(nil), + downloadRetryAttempts: Int = 3) -> String +{ + let versionArg = RubyCommand.Argument(name: "version", value: version, type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let teamIdArg = teamId.asRubyArgument(name: "team_id", type: nil) + let downloadRetryAttemptsArg = RubyCommand.Argument(name: "download_retry_attempts", value: downloadRetryAttempts, type: nil) + let array: [RubyCommand.Argument?] = [versionArg, + usernameArg, + teamIdArg, + downloadRetryAttemptsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "xcode_install", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Change the xcode-path to use. Useful for beta versions of Xcode + + Select and build with the Xcode installed at the provided path. + Use the `xcodes` action if you want to select an Xcode: + - Based on a version specifier or + - You don't have known, stable paths, as may happen in a CI environment. + */ +public func xcodeSelect() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xcode_select", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Downloads Xcode Bot assets like the `.xcarchive` and logs + + - parameters: + - host: IP Address/Hostname of Xcode Server + - botName: Name of the Bot to pull assets from + - integrationNumber: Optionally you can override which integration's assets should be downloaded. If not provided, the latest integration is used + - username: Username for your Xcode Server + - password: Password for your Xcode Server + - targetFolder: Relative path to a folder into which to download assets + - keepAllAssets: Whether to keep all assets or let the script delete everything except for the .xcarchive + - trustSelfSignedCerts: Whether to trust self-signed certs on your Xcode Server + + This action downloads assets from your Xcode Server Bot (works with Xcode Server using Xcode 6 and 7. By default, this action downloads all assets, unzips them and deletes everything except for the `.xcarchive`. + If you'd like to keep all downloaded assets, pass `keep_all_assets: true`. + This action returns the path to the downloaded assets folder and puts into shared values the paths to the asset folder and to the `.xcarchive` inside it. + */ +@discardableResult public func xcodeServerGetAssets(host: String, + botName: String, + integrationNumber: OptionalConfigValue = .fastlaneDefault(nil), + username: String = "", + password: OptionalConfigValue = .fastlaneDefault(nil), + targetFolder: String = "./xcs_assets", + keepAllAssets: OptionalConfigValue = .fastlaneDefault(false), + trustSelfSignedCerts: OptionalConfigValue = .fastlaneDefault(true)) -> [String] +{ + let hostArg = RubyCommand.Argument(name: "host", value: host, type: nil) + let botNameArg = RubyCommand.Argument(name: "bot_name", value: botName, type: nil) + let integrationNumberArg = integrationNumber.asRubyArgument(name: "integration_number", type: nil) + let usernameArg = RubyCommand.Argument(name: "username", value: username, type: nil) + let passwordArg = password.asRubyArgument(name: "password", type: nil) + let targetFolderArg = RubyCommand.Argument(name: "target_folder", value: targetFolder, type: nil) + let keepAllAssetsArg = keepAllAssets.asRubyArgument(name: "keep_all_assets", type: nil) + let trustSelfSignedCertsArg = trustSelfSignedCerts.asRubyArgument(name: "trust_self_signed_certs", type: nil) + let array: [RubyCommand.Argument?] = [hostArg, + botNameArg, + integrationNumberArg, + usernameArg, + passwordArg, + targetFolderArg, + keepAllAssetsArg, + trustSelfSignedCertsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "xcode_server_get_assets", className: nil, args: args) + return parseArray(fromString: runner.executeCommand(command)) +} + +/** + Use the `xcodebuild` command to build and sign your app + + **Note**: `xcodebuild` is a complex command, so it is recommended to use [_gym_](https://docs.fastlane.tools/actions/gym/) for building your ipa file and [_scan_](https://docs.fastlane.tools/actions/scan/) for testing your app instead. + */ +public func xcodebuild() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xcodebuild", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Make sure a certain version of Xcode is installed, installing it only if needed + + - parameters: + - version: The version number of the version of Xcode to install. Defaults to the value specified in the .xcode-version file + - updateList: Whether the list of available Xcode versions should be updated before running the install command + - selectForCurrentBuildOnly: When true, it won't attempt to install an Xcode version, just find the installed Xcode version that best matches the passed version argument, and select it for the current build steps. It doesn't change the global Xcode version (e.g. via 'xcrun xcode-select'), which would require sudo permissions — when this option is true, this action doesn't require sudo permissions + - binaryPath: Where the xcodes binary lives on your system (full path) + - xcodesArgs: Pass in xcodes command line arguments directly. When present, other parameters are ignored and only this parameter is used to build the command to be executed + + - returns: The path to the newly installed Xcode version + + Makes sure a specific version of Xcode is installed. If that's not the case, it will automatically be downloaded by [xcodes](https://github.com/RobotsAndPencils/xcodes). + This will make sure to use the correct Xcode version for later actions. + Note that this action depends on [xcodes](https://github.com/RobotsAndPencils/xcodes) CLI, so make sure you have it installed in your environment. For the installation guide, see: https://github.com/RobotsAndPencils/xcodes#installation + */ +@discardableResult public func xcodes(version: String, + updateList: OptionalConfigValue = .fastlaneDefault(true), + selectForCurrentBuildOnly: OptionalConfigValue = .fastlaneDefault(false), + binaryPath: String = "/opt/homebrew/bin/xcodes", + xcodesArgs: OptionalConfigValue = .fastlaneDefault(nil)) -> String +{ + let versionArg = RubyCommand.Argument(name: "version", value: version, type: nil) + let updateListArg = updateList.asRubyArgument(name: "update_list", type: nil) + let selectForCurrentBuildOnlyArg = selectForCurrentBuildOnly.asRubyArgument(name: "select_for_current_build_only", type: nil) + let binaryPathArg = RubyCommand.Argument(name: "binary_path", value: binaryPath, type: nil) + let xcodesArgsArg = xcodesArgs.asRubyArgument(name: "xcodes_args", type: nil) + let array: [RubyCommand.Argument?] = [versionArg, + updateListArg, + selectForCurrentBuildOnlyArg, + binaryPathArg, + xcodesArgsArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "xcodes", className: nil, args: args) + return runner.executeCommand(command) +} + +/** + Nice code coverage reports without hassle + + - parameters: + - workspace: Path the workspace file + - project: Path the project file + - scheme: The project's scheme. Make sure it's marked as `Shared` + - configuration: The configuration used when building the app. Defaults to 'Release' + - sourceDirectory: The path to project's root directory + - derivedDataPath: The directory where build products and other derived data will go + - outputDirectory: The directory in which all reports will be stored + - htmlReport: Produce an HTML report + - markdownReport: Produce a Markdown report + - jsonReport: Produce a JSON report + - minimumCoveragePercentage: Raise exception if overall coverage percentage is under this value (ie. 75) + - slackUrl: Create an Incoming WebHook for your Slack group to post results there + - slackChannel: #channel or @username + - skipSlack: Don't publish to slack, even when an URL is given + - slackUsername: The username which is used to publish to slack + - slackMessage: The message which is published together with a successful report + - ignoreFilePath: Relative or absolute path to the file containing the list of ignored files + - includeTestTargets: Enables coverage reports for .xctest targets + - excludeTargets: Comma separated list of targets to exclude from coverage report + - includeTargets: Comma separated list of targets to include in coverage report. If specified then exlude_targets will be ignored + - onlyProjectTargets: Display the coverage only for main project targets (e.g. skip Pods targets) + - disableCoveralls: Add this flag to disable automatic submission to Coveralls + - coverallsServiceName: Name of the CI service compatible with Coveralls. i.e. travis-ci. This option must be defined along with coveralls_service_job_id + - coverallsServiceJobId: Name of the current job running on a CI service compatible with Coveralls. This option must be defined along with coveralls_service_name + - coverallsRepoToken: Repository token to be used by integrations not compatible with Coveralls + - xcconfig: Use an extra XCCONFIG file to build your app + - ideFoundationPath: Absolute path to the IDEFoundation.framework binary + - legacySupport: Whether xcov should parse a xccoverage file instead on xccovreport + + Create nice code coverage reports and post coverage summaries on Slack *(xcov gem is required)*. + More information: [https://github.com/nakiostudio/xcov](https://github.com/nakiostudio/xcov). + */ +public func xcov(workspace: OptionalConfigValue = .fastlaneDefault(nil), + project: OptionalConfigValue = .fastlaneDefault(nil), + scheme: OptionalConfigValue = .fastlaneDefault(nil), + configuration: OptionalConfigValue = .fastlaneDefault(nil), + sourceDirectory: OptionalConfigValue = .fastlaneDefault(nil), + derivedDataPath: OptionalConfigValue = .fastlaneDefault(nil), + outputDirectory: String = "./xcov_report", + htmlReport: OptionalConfigValue = .fastlaneDefault(true), + markdownReport: OptionalConfigValue = .fastlaneDefault(false), + jsonReport: OptionalConfigValue = .fastlaneDefault(false), + minimumCoveragePercentage: Float = 0.0, + slackUrl: OptionalConfigValue = .fastlaneDefault(nil), + slackChannel: OptionalConfigValue = .fastlaneDefault(nil), + skipSlack: OptionalConfigValue = .fastlaneDefault(false), + slackUsername: String = "xcov", + slackMessage: String = "Your *xcov* coverage report", + ignoreFilePath: String = "./.xcovignore", + includeTestTargets: OptionalConfigValue = .fastlaneDefault(false), + excludeTargets: OptionalConfigValue = .fastlaneDefault(nil), + includeTargets: OptionalConfigValue = .fastlaneDefault(nil), + onlyProjectTargets: OptionalConfigValue = .fastlaneDefault(false), + disableCoveralls: OptionalConfigValue = .fastlaneDefault(false), + coverallsServiceName: OptionalConfigValue = .fastlaneDefault(nil), + coverallsServiceJobId: OptionalConfigValue = .fastlaneDefault(nil), + coverallsRepoToken: OptionalConfigValue = .fastlaneDefault(nil), + xcconfig: OptionalConfigValue = .fastlaneDefault(nil), + ideFoundationPath: String = "/Applications/Xcode-13.4.1.app/Contents/Developer/../Frameworks/IDEFoundation.framework/Versions/A/IDEFoundation", + legacySupport: OptionalConfigValue = .fastlaneDefault(false)) +{ + let workspaceArg = workspace.asRubyArgument(name: "workspace", type: nil) + let projectArg = project.asRubyArgument(name: "project", type: nil) + let schemeArg = scheme.asRubyArgument(name: "scheme", type: nil) + let configurationArg = configuration.asRubyArgument(name: "configuration", type: nil) + let sourceDirectoryArg = sourceDirectory.asRubyArgument(name: "source_directory", type: nil) + let derivedDataPathArg = derivedDataPath.asRubyArgument(name: "derived_data_path", type: nil) + let outputDirectoryArg = RubyCommand.Argument(name: "output_directory", value: outputDirectory, type: nil) + let htmlReportArg = htmlReport.asRubyArgument(name: "html_report", type: nil) + let markdownReportArg = markdownReport.asRubyArgument(name: "markdown_report", type: nil) + let jsonReportArg = jsonReport.asRubyArgument(name: "json_report", type: nil) + let minimumCoveragePercentageArg = RubyCommand.Argument(name: "minimum_coverage_percentage", value: minimumCoveragePercentage, type: nil) + let slackUrlArg = slackUrl.asRubyArgument(name: "slack_url", type: nil) + let slackChannelArg = slackChannel.asRubyArgument(name: "slack_channel", type: nil) + let skipSlackArg = skipSlack.asRubyArgument(name: "skip_slack", type: nil) + let slackUsernameArg = RubyCommand.Argument(name: "slack_username", value: slackUsername, type: nil) + let slackMessageArg = RubyCommand.Argument(name: "slack_message", value: slackMessage, type: nil) + let ignoreFilePathArg = RubyCommand.Argument(name: "ignore_file_path", value: ignoreFilePath, type: nil) + let includeTestTargetsArg = includeTestTargets.asRubyArgument(name: "include_test_targets", type: nil) + let excludeTargetsArg = excludeTargets.asRubyArgument(name: "exclude_targets", type: nil) + let includeTargetsArg = includeTargets.asRubyArgument(name: "include_targets", type: nil) + let onlyProjectTargetsArg = onlyProjectTargets.asRubyArgument(name: "only_project_targets", type: nil) + let disableCoverallsArg = disableCoveralls.asRubyArgument(name: "disable_coveralls", type: nil) + let coverallsServiceNameArg = coverallsServiceName.asRubyArgument(name: "coveralls_service_name", type: nil) + let coverallsServiceJobIdArg = coverallsServiceJobId.asRubyArgument(name: "coveralls_service_job_id", type: nil) + let coverallsRepoTokenArg = coverallsRepoToken.asRubyArgument(name: "coveralls_repo_token", type: nil) + let xcconfigArg = xcconfig.asRubyArgument(name: "xcconfig", type: nil) + let ideFoundationPathArg = RubyCommand.Argument(name: "ideFoundationPath", value: ideFoundationPath, type: nil) + let legacySupportArg = legacySupport.asRubyArgument(name: "legacy_support", type: nil) + let array: [RubyCommand.Argument?] = [workspaceArg, + projectArg, + schemeArg, + configurationArg, + sourceDirectoryArg, + derivedDataPathArg, + outputDirectoryArg, + htmlReportArg, + markdownReportArg, + jsonReportArg, + minimumCoveragePercentageArg, + slackUrlArg, + slackChannelArg, + skipSlackArg, + slackUsernameArg, + slackMessageArg, + ignoreFilePathArg, + includeTestTargetsArg, + excludeTargetsArg, + includeTargetsArg, + onlyProjectTargetsArg, + disableCoverallsArg, + coverallsServiceNameArg, + coverallsServiceJobIdArg, + coverallsRepoTokenArg, + xcconfigArg, + ideFoundationPathArg, + legacySupportArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "xcov", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Runs tests on the given simulator + */ +public func xctest() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xctest", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Run tests using xctool + + You can run any `xctool` action. This will require having [xctool](https://github.com/facebook/xctool) installed through [Homebrew](http://brew.sh). + It is recommended to store the build configuration in the `.xctool-args` file. + More information: [https://docs.fastlane.tools/actions/xctool/](https://docs.fastlane.tools/actions/xctool/). + */ +public func xctool() { + let args: [RubyCommand.Argument] = [] + let command = RubyCommand(commandID: "", methodName: "xctool", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Select an Xcode to use by version specifier + + - parameter version: The version of Xcode to select specified as a Gem::Version requirement string (e.g. '~> 7.1.0'). Defaults to the value specified in the .xcode-version file + + Finds and selects a version of an installed Xcode that best matches the provided [`Gem::Version` requirement specifier](http://www.rubydoc.info/github/rubygems/rubygems/Gem/Version) + You can either manually provide a specific version using `version:` or you make use of the `.xcode-version` file. + */ +public func xcversion(version: String) { + let versionArg = RubyCommand.Argument(name: "version", value: version, type: nil) + let array: [RubyCommand.Argument?] = [versionArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "xcversion", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Compress a file or folder to a zip + + - parameters: + - path: Path to the directory or file to be zipped + - outputPath: The name of the resulting zip file + - verbose: Enable verbose output of zipped file + - password: Encrypt the contents of the zip archive using a password + - symlinks: Store symbolic links as such in the zip archive + - include: Array of paths or patterns to include + - exclude: Array of paths or patterns to exclude + + - returns: The path to the output zip file + */ +@discardableResult public func zip(path: String, + outputPath: OptionalConfigValue = .fastlaneDefault(nil), + verbose: OptionalConfigValue = .fastlaneDefault(true), + password: OptionalConfigValue = .fastlaneDefault(nil), + symlinks: OptionalConfigValue = .fastlaneDefault(false), + include: [String] = [], + exclude: [String] = []) -> String +{ + let pathArg = RubyCommand.Argument(name: "path", value: path, type: nil) + let outputPathArg = outputPath.asRubyArgument(name: "output_path", type: nil) + let verboseArg = verbose.asRubyArgument(name: "verbose", type: nil) + let passwordArg = password.asRubyArgument(name: "password", type: nil) + let symlinksArg = symlinks.asRubyArgument(name: "symlinks", type: nil) + let includeArg = RubyCommand.Argument(name: "include", value: include, type: nil) + let excludeArg = RubyCommand.Argument(name: "exclude", value: exclude, type: nil) + let array: [RubyCommand.Argument?] = [pathArg, + outputPathArg, + verboseArg, + passwordArg, + symlinksArg, + includeArg, + excludeArg] + let args: [RubyCommand.Argument] = array + .filter { $0?.value != nil } + .compactMap { $0 } + let command = RubyCommand(commandID: "", methodName: "zip", className: nil, args: args) + return runner.executeCommand(command) +} + +// These are all the parsing functions needed to transform our data into the expected types +func parseArray(fromString: String, function: String = #function) -> [String] { + verbose(message: "parsing an Array from data: \(fromString), from function: \(function)") + let potentialArray: String + if fromString.count < 2 { + potentialArray = "[\(fromString)]" + } else { + potentialArray = fromString + } + let array: [String] = try! JSONSerialization.jsonObject(with: potentialArray.data(using: .utf8)!, options: []) as! [String] + return array +} + +func parseDictionary(fromString: String, function: String = #function) -> [String: String] { + return parseDictionaryHelper(fromString: fromString, function: function) as! [String: String] +} + +func parseDictionary(fromString: String, function: String = #function) -> [String: Any] { + return parseDictionaryHelper(fromString: fromString, function: function) +} + +func parseDictionaryHelper(fromString: String, function: String = #function) -> [String: Any] { + verbose(message: "parsing an Array from data: \(fromString), from function: \(function)") + let potentialDictionary: String + if fromString.count < 2 { + verbose(message: "Dictionary value too small: \(fromString), from function: \(function)") + potentialDictionary = "{}" + } else { + potentialDictionary = fromString + } + let dictionary: [String: Any] = try! JSONSerialization.jsonObject(with: potentialDictionary.data(using: .utf8)!, options: []) as! [String: Any] + return dictionary +} + +func parseBool(fromString: String, function: String = #function) -> Bool { + verbose(message: "parsing a Bool from data: \(fromString), from function: \(function)") + return NSString(string: fromString.trimmingCharacters(in: .punctuationCharacters)).boolValue +} + +func parseInt(fromString: String, function: String = #function) -> Int { + verbose(message: "parsing an Int from data: \(fromString), from function: \(function)") + return NSString(string: fromString.trimmingCharacters(in: .punctuationCharacters)).integerValue +} + +public let deliverfile: Deliverfile = .init() +public let gymfile: Gymfile = .init() +public let matchfile: Matchfile = .init() +public let precheckfile: Precheckfile = .init() +public let scanfile: Scanfile = .init() +public let screengrabfile: Screengrabfile = .init() +public let snapshotfile: Snapshotfile = .init() + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.165] diff --git a/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.pbxproj b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.pbxproj new file mode 100644 index 00000000..74e94fe9 --- /dev/null +++ b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.pbxproj @@ -0,0 +1,511 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 46; + objects = { + +/* Begin PBXBuildFile section */ + 0311E387230AC1B20060BB5C /* Plugins.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0311E386230AC1B20060BB5C /* Plugins.swift */; }; + 0311E38B230AC9490060BB5C /* Actions.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0311E38A230AC9490060BB5C /* Actions.swift */; }; + 1257253924B7992C00E04FA3 /* main.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1257253824B7992B00E04FA3 /* main.swift */; }; + 1267C3F42773A43E004DE48A /* Atomic.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1267C3F32773A43E004DE48A /* Atomic.swift */; }; + 12D2EB8D2620D83C00844013 /* OptionalConfigValue.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12D2EB8C2620D83B00844013 /* OptionalConfigValue.swift */; }; + 1C2D0FA528E4512E0059FF5A /* Version.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1C2D0FA428E4512E0059FF5A /* Version.swift */; }; + 1C2D0FA728E457D10059FF5A /* Test.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1C2D0FA628E457D10059FF5A /* Test.swift */; }; + 8B92CD0428DC62D200A3FF05 /* Match.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8B92CD0328DC62D200A3FF05 /* Match.swift */; }; + 8B92CD0728DC638800A3FF05 /* Constant.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8B92CD0628DC638800A3FF05 /* Constant.swift */; }; + 8B92CD0928DC63A700A3FF05 /* Secret.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8B92CD0828DC63A700A3FF05 /* Secret.swift */; }; + 8B92CD0B28DC63E000A3FF05 /* Keychain.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8B92CD0A28DC63E000A3FF05 /* Keychain.swift */; }; + 8B9386D728E2A8630045D709 /* Symbol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8B9386D628E2A8630045D709 /* Symbol.swift */; }; + 90C4D77B28DDC86800E06274 /* Build.swift in Sources */ = {isa = PBXBuildFile; fileRef = 90C4D77A28DDC86800E06274 /* Build.swift */; }; + 90C4D77D28E0AB2A00E06274 /* Distribution.swift in Sources */ = {isa = PBXBuildFile; fileRef = 90C4D77C28E0AB2A00E06274 /* Distribution.swift */; }; + 90DA64BD291BC7F400BABF4D /* EnvironmentParser.swift in Sources */ = {isa = PBXBuildFile; fileRef = 90DA64BC291BC7F400BABF4D /* EnvironmentParser.swift */; }; + 90F4F4CB28E4455C008BDAE5 /* AppStoreAuthentication.swift in Sources */ = {isa = PBXBuildFile; fileRef = 90F4F4CA28E4455C008BDAE5 /* AppStoreAuthentication.swift */; }; + B302067B1F5E3E9000DE6EBD /* SnapshotfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B30206741F5E3E9000DE6EBD /* SnapshotfileProtocol.swift */; }; + B302067C1F5E3E9000DE6EBD /* GymfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B30206751F5E3E9000DE6EBD /* GymfileProtocol.swift */; }; + B302067D1F5E3E9000DE6EBD /* MatchfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B30206761F5E3E9000DE6EBD /* MatchfileProtocol.swift */; }; + B302067E1F5E3E9000DE6EBD /* PrecheckfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B30206771F5E3E9000DE6EBD /* PrecheckfileProtocol.swift */; }; + B302067F1F5E3E9000DE6EBD /* ScanfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B30206781F5E3E9000DE6EBD /* ScanfileProtocol.swift */; }; + B30206801F5E3E9000DE6EBD /* ScreengrabfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B30206791F5E3E9000DE6EBD /* ScreengrabfileProtocol.swift */; }; + B30206811F5E3E9000DE6EBD /* DeliverfileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B302067A1F5E3E9000DE6EBD /* DeliverfileProtocol.swift */; }; + B3BA65A61F5A269100B34850 /* Fastlane.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA659D1F5A269100B34850 /* Fastlane.swift */; }; + B3BA65A71F5A269100B34850 /* LaneFileProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA659E1F5A269100B34850 /* LaneFileProtocol.swift */; }; + B3BA65A91F5A269100B34850 /* RubyCommand.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA65A01F5A269100B34850 /* RubyCommand.swift */; }; + B3BA65AA1F5A269100B34850 /* Runner.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA65A11F5A269100B34850 /* Runner.swift */; }; + B3BA65AB1F5A269100B34850 /* SocketClient.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA65A21F5A269100B34850 /* SocketClient.swift */; }; + B3BA65AC1F5A269100B34850 /* SocketClientDelegateProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA65A31F5A269100B34850 /* SocketClientDelegateProtocol.swift */; }; + B3BA65AD1F5A269100B34850 /* SocketResponse.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA65A41F5A269100B34850 /* SocketResponse.swift */; }; + B3BA65AF1F5A2D5C00B34850 /* RunnerArgument.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3BA65AE1F5A2D5C00B34850 /* RunnerArgument.swift */; }; + C0459CAC27261897002CDFB9 /* FastlaneRunner in FastlaneRunnerCopySigned */ = {isa = PBXBuildFile; fileRef = D556D6A91F6A08F5003108E3 /* FastlaneRunner */; settings = {ATTRIBUTES = (CodeSignOnCopy, ); }; }; + D55B28C31F6C588300DC42C5 /* Deliverfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28BC1F6C588300DC42C5 /* Deliverfile.swift */; }; + D55B28C41F6C588300DC42C5 /* Gymfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28BD1F6C588300DC42C5 /* Gymfile.swift */; }; + D55B28C51F6C588300DC42C5 /* Matchfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28BE1F6C588300DC42C5 /* Matchfile.swift */; }; + D55B28C61F6C588300DC42C5 /* Precheckfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28BF1F6C588300DC42C5 /* Precheckfile.swift */; }; + D55B28C71F6C588300DC42C5 /* Scanfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28C01F6C588300DC42C5 /* Scanfile.swift */; }; + D55B28C81F6C588300DC42C5 /* Screengrabfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28C11F6C588300DC42C5 /* Screengrabfile.swift */; }; + D55B28C91F6C588300DC42C5 /* Snapshotfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D55B28C21F6C588300DC42C5 /* Snapshotfile.swift */; }; + D5A7C48F1F7C4DAF00A91DE6 /* Appfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5A7C48D1F7C4DAF00A91DE6 /* Appfile.swift */; }; + D5A7C4901F7C4DAF00A91DE6 /* Fastfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5A7C48E1F7C4DAF00A91DE6 /* Fastfile.swift */; }; + D5B8A5B31FFDC49E00536B24 /* ControlCommand.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5B8A5B21FFDC49D00536B24 /* ControlCommand.swift */; }; + D5BAFD121F7DAAFC0030B324 /* ArgumentProcessor.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5BAFD111F7DAAFC0030B324 /* ArgumentProcessor.swift */; }; + D5D1DE991FFEE8EA00502A00 /* RubyCommandable.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5D1DE981FFEE8E900502A00 /* RubyCommandable.swift */; }; +/* End PBXBuildFile section */ + +/* Begin PBXCopyFilesBuildPhase section */ + C0459CAB27261886002CDFB9 /* FastlaneRunnerCopySigned */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = "$SRCROOT/../.."; + dstSubfolderSpec = 0; + files = ( + C0459CAC27261897002CDFB9 /* FastlaneRunner in FastlaneRunnerCopySigned */, + ); + name = FastlaneRunnerCopySigned; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 0311E386230AC1B20060BB5C /* Plugins.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Plugins.swift; path = ../Plugins.swift; sourceTree = ""; }; + 0311E38A230AC9490060BB5C /* Actions.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Actions.swift; path = ../Actions.swift; sourceTree = ""; }; + 1257253824B7992B00E04FA3 /* main.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = main.swift; path = ../main.swift; sourceTree = ""; }; + 1267C3F32773A43E004DE48A /* Atomic.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Atomic.swift; path = ../Atomic.swift; sourceTree = ""; }; + 12D2EB8C2620D83B00844013 /* OptionalConfigValue.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = OptionalConfigValue.swift; path = ../OptionalConfigValue.swift; sourceTree = ""; }; + 1C2D0FA428E4512E0059FF5A /* Version.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Version.swift; path = ../../Helpers/Version.swift; sourceTree = ""; }; + 1C2D0FA628E457D10059FF5A /* Test.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Test.swift; path = ../../Helpers/Test.swift; sourceTree = ""; }; + 8B92CD0328DC62D200A3FF05 /* Match.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Match.swift; path = ../../Helpers/Match.swift; sourceTree = ""; }; + 8B92CD0628DC638800A3FF05 /* Constant.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Constant.swift; path = ../../Constants/Constant.swift; sourceTree = ""; }; + 8B92CD0828DC63A700A3FF05 /* Secret.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Secret.swift; path = ../../Constants/Secret.swift; sourceTree = ""; }; + 8B92CD0A28DC63E000A3FF05 /* Keychain.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Keychain.swift; path = ../../Helpers/Keychain.swift; sourceTree = ""; }; + 8B9386D628E2A8630045D709 /* Symbol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Symbol.swift; path = ../../Helpers/Symbol.swift; sourceTree = ""; }; + 90C4D77A28DDC86800E06274 /* Build.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Build.swift; path = ../../Helpers/Build.swift; sourceTree = ""; }; + 90C4D77C28E0AB2A00E06274 /* Distribution.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Distribution.swift; path = ../../Helpers/Distribution.swift; sourceTree = ""; }; + 90DA64BC291BC7F400BABF4D /* EnvironmentParser.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = EnvironmentParser.swift; path = ../../Helpers/EnvironmentParser.swift; sourceTree = ""; }; + 90F4F4CA28E4455C008BDAE5 /* AppStoreAuthentication.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = AppStoreAuthentication.swift; path = ../../Helpers/AppStoreAuthentication.swift; sourceTree = ""; }; + B30206741F5E3E9000DE6EBD /* SnapshotfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SnapshotfileProtocol.swift; path = ../SnapshotfileProtocol.swift; sourceTree = ""; }; + B30206751F5E3E9000DE6EBD /* GymfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = GymfileProtocol.swift; path = ../GymfileProtocol.swift; sourceTree = ""; }; + B30206761F5E3E9000DE6EBD /* MatchfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = MatchfileProtocol.swift; path = ../MatchfileProtocol.swift; sourceTree = ""; }; + B30206771F5E3E9000DE6EBD /* PrecheckfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = PrecheckfileProtocol.swift; path = ../PrecheckfileProtocol.swift; sourceTree = ""; }; + B30206781F5E3E9000DE6EBD /* ScanfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = ScanfileProtocol.swift; path = ../ScanfileProtocol.swift; sourceTree = ""; }; + B30206791F5E3E9000DE6EBD /* ScreengrabfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = ScreengrabfileProtocol.swift; path = ../ScreengrabfileProtocol.swift; sourceTree = ""; }; + B302067A1F5E3E9000DE6EBD /* DeliverfileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = DeliverfileProtocol.swift; path = ../DeliverfileProtocol.swift; sourceTree = ""; }; + B3144C072005533400470AFE /* README.txt */ = {isa = PBXFileReference; lastKnownFileType = text; path = README.txt; sourceTree = ""; }; + B3144C08200553C800470AFE /* README.txt */ = {isa = PBXFileReference; lastKnownFileType = text; path = README.txt; sourceTree = ""; }; + B3144C09200553D400470AFE /* README.txt */ = {isa = PBXFileReference; lastKnownFileType = text; path = README.txt; sourceTree = ""; }; + B3144C0A200553DC00470AFE /* README.txt */ = {isa = PBXFileReference; lastKnownFileType = text; path = README.txt; sourceTree = ""; }; + B3BA659D1F5A269100B34850 /* Fastlane.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Fastlane.swift; path = ../Fastlane.swift; sourceTree = ""; }; + B3BA659E1F5A269100B34850 /* LaneFileProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = LaneFileProtocol.swift; path = ../LaneFileProtocol.swift; sourceTree = ""; }; + B3BA65A01F5A269100B34850 /* RubyCommand.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = RubyCommand.swift; path = ../RubyCommand.swift; sourceTree = ""; }; + B3BA65A11F5A269100B34850 /* Runner.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Runner.swift; path = ../Runner.swift; sourceTree = ""; }; + B3BA65A21F5A269100B34850 /* SocketClient.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SocketClient.swift; path = ../SocketClient.swift; sourceTree = ""; }; + B3BA65A31F5A269100B34850 /* SocketClientDelegateProtocol.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SocketClientDelegateProtocol.swift; path = ../SocketClientDelegateProtocol.swift; sourceTree = ""; }; + B3BA65A41F5A269100B34850 /* SocketResponse.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SocketResponse.swift; path = ../SocketResponse.swift; sourceTree = ""; }; + B3BA65AE1F5A2D5C00B34850 /* RunnerArgument.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = RunnerArgument.swift; path = ../RunnerArgument.swift; sourceTree = ""; }; + D556D6A91F6A08F5003108E3 /* FastlaneRunner */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = FastlaneRunner; sourceTree = BUILT_PRODUCTS_DIR; }; + D55B28BC1F6C588300DC42C5 /* Deliverfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Deliverfile.swift; path = ../Deliverfile.swift; sourceTree = ""; }; + D55B28BD1F6C588300DC42C5 /* Gymfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Gymfile.swift; path = ../Gymfile.swift; sourceTree = ""; }; + D55B28BE1F6C588300DC42C5 /* Matchfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Matchfile.swift; path = ../Matchfile.swift; sourceTree = ""; }; + D55B28BF1F6C588300DC42C5 /* Precheckfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Precheckfile.swift; path = ../Precheckfile.swift; sourceTree = ""; }; + D55B28C01F6C588300DC42C5 /* Scanfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Scanfile.swift; path = ../Scanfile.swift; sourceTree = ""; }; + D55B28C11F6C588300DC42C5 /* Screengrabfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Screengrabfile.swift; path = ../Screengrabfile.swift; sourceTree = ""; }; + D55B28C21F6C588300DC42C5 /* Snapshotfile.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = Snapshotfile.swift; path = ../Snapshotfile.swift; sourceTree = ""; }; + D5A7C48D1F7C4DAF00A91DE6 /* Appfile.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Appfile.swift; path = ../../Appfile.swift; sourceTree = ""; }; + D5A7C48E1F7C4DAF00A91DE6 /* Fastfile.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = Fastfile.swift; path = ../../Fastfile.swift; sourceTree = ""; }; + D5B8A5B21FFDC49D00536B24 /* ControlCommand.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = ControlCommand.swift; path = ../ControlCommand.swift; sourceTree = ""; }; + D5BAFD111F7DAAFC0030B324 /* ArgumentProcessor.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = ArgumentProcessor.swift; path = ../ArgumentProcessor.swift; sourceTree = ""; }; + D5D1DE981FFEE8E900502A00 /* RubyCommandable.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = RubyCommandable.swift; path = ../RubyCommandable.swift; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + B33BAF541F51F8D90001A751 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 8B92CD0228DC62C100A3FF05 /* Helpers */ = { + isa = PBXGroup; + children = ( + 90F4F4CA28E4455C008BDAE5 /* AppStoreAuthentication.swift */, + 90C4D77A28DDC86800E06274 /* Build.swift */, + 90C4D77C28E0AB2A00E06274 /* Distribution.swift */, + 90DA64BC291BC7F400BABF4D /* EnvironmentParser.swift */, + 8B92CD0A28DC63E000A3FF05 /* Keychain.swift */, + 8B92CD0328DC62D200A3FF05 /* Match.swift */, + 8B9386D628E2A8630045D709 /* Symbol.swift */, + 1C2D0FA628E457D10059FF5A /* Test.swift */, + 1C2D0FA428E4512E0059FF5A /* Version.swift */, + ); + name = Helpers; + sourceTree = ""; + }; + 8B92CD0528DC637400A3FF05 /* Constants */ = { + isa = PBXGroup; + children = ( + 8B92CD0628DC638800A3FF05 /* Constant.swift */, + 8B92CD0828DC63A700A3FF05 /* Secret.swift */, + ); + name = Constants; + sourceTree = ""; + }; + B33BAF4E1F51F8D90001A751 = { + isa = PBXGroup; + children = ( + 8B92CD0528DC637400A3FF05 /* Constants */, + 8B92CD0228DC62C100A3FF05 /* Helpers */, + B3BA65B01F5A324A00B34850 /* Fastlane Runner */, + D556D6A91F6A08F5003108E3 /* FastlaneRunner */, + ); + sourceTree = ""; + }; + B3BA65B01F5A324A00B34850 /* Fastlane Runner */ = { + isa = PBXGroup; + children = ( + D5A7C48D1F7C4DAF00A91DE6 /* Appfile.swift */, + D55B28BC1F6C588300DC42C5 /* Deliverfile.swift */, + D5A7C48E1F7C4DAF00A91DE6 /* Fastfile.swift */, + D55B28BD1F6C588300DC42C5 /* Gymfile.swift */, + D55B28BE1F6C588300DC42C5 /* Matchfile.swift */, + D55B28BF1F6C588300DC42C5 /* Precheckfile.swift */, + D55B28C01F6C588300DC42C5 /* Scanfile.swift */, + D55B28C11F6C588300DC42C5 /* Screengrabfile.swift */, + D55B28C21F6C588300DC42C5 /* Snapshotfile.swift */, + B3BA65B21F5A327B00B34850 /* Autogenerated API */, + B3BA65B31F5A329800B34850 /* Fastfile Components */, + B3BA65B11F5A325E00B34850 /* Networking */, + D512BA011F7C7F40000D2137 /* Runner Code */, + ); + name = "Fastlane Runner"; + sourceTree = ""; + }; + B3BA65B11F5A325E00B34850 /* Networking */ = { + isa = PBXGroup; + children = ( + 1267C3F32773A43E004DE48A /* Atomic.swift */, + B3144C072005533400470AFE /* README.txt */, + D5B8A5B21FFDC49D00536B24 /* ControlCommand.swift */, + B3BA65A01F5A269100B34850 /* RubyCommand.swift */, + D5D1DE981FFEE8E900502A00 /* RubyCommandable.swift */, + B3BA65A11F5A269100B34850 /* Runner.swift */, + B3BA65A21F5A269100B34850 /* SocketClient.swift */, + B3BA65A31F5A269100B34850 /* SocketClientDelegateProtocol.swift */, + B3BA65A41F5A269100B34850 /* SocketResponse.swift */, + ); + name = Networking; + sourceTree = ""; + }; + B3BA65B21F5A327B00B34850 /* Autogenerated API */ = { + isa = PBXGroup; + children = ( + B3144C09200553D400470AFE /* README.txt */, + 0311E38A230AC9490060BB5C /* Actions.swift */, + B3BA659D1F5A269100B34850 /* Fastlane.swift */, + B302067A1F5E3E9000DE6EBD /* DeliverfileProtocol.swift */, + B30206751F5E3E9000DE6EBD /* GymfileProtocol.swift */, + B30206761F5E3E9000DE6EBD /* MatchfileProtocol.swift */, + 0311E386230AC1B20060BB5C /* Plugins.swift */, + B30206771F5E3E9000DE6EBD /* PrecheckfileProtocol.swift */, + B30206781F5E3E9000DE6EBD /* ScanfileProtocol.swift */, + B30206791F5E3E9000DE6EBD /* ScreengrabfileProtocol.swift */, + B30206741F5E3E9000DE6EBD /* SnapshotfileProtocol.swift */, + ); + name = "Autogenerated API"; + sourceTree = ""; + }; + B3BA65B31F5A329800B34850 /* Fastfile Components */ = { + isa = PBXGroup; + children = ( + B3144C08200553C800470AFE /* README.txt */, + B3BA659E1F5A269100B34850 /* LaneFileProtocol.swift */, + 12D2EB8C2620D83B00844013 /* OptionalConfigValue.swift */, + ); + name = "Fastfile Components"; + sourceTree = ""; + }; + D512BA011F7C7F40000D2137 /* Runner Code */ = { + isa = PBXGroup; + children = ( + 1257253824B7992B00E04FA3 /* main.swift */, + B3144C0A200553DC00470AFE /* README.txt */, + D5BAFD111F7DAAFC0030B324 /* ArgumentProcessor.swift */, + B3BA65AE1F5A2D5C00B34850 /* RunnerArgument.swift */, + ); + name = "Runner Code"; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + B33BAF561F51F8D90001A751 /* FastlaneRunner */ = { + isa = PBXNativeTarget; + buildConfigurationList = B33BAF5E1F51F8D90001A751 /* Build configuration list for PBXNativeTarget "FastlaneRunner" */; + buildPhases = ( + B33BAF531F51F8D90001A751 /* Sources */, + B33BAF541F51F8D90001A751 /* Frameworks */, + C0459CAB27261886002CDFB9 /* FastlaneRunnerCopySigned */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = FastlaneRunner; + productName = FastlaneSwiftRunner; + productReference = D556D6A91F6A08F5003108E3 /* FastlaneRunner */; + productType = "com.apple.product-type.tool"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + B33BAF4F1F51F8D90001A751 /* Project object */ = { + isa = PBXProject; + attributes = { + LastSwiftUpdateCheck = 0830; + LastUpgradeCheck = 1300; + ORGANIZATIONNAME = Nimble; + TargetAttributes = { + B33BAF561F51F8D90001A751 = { + CreatedOnToolsVersion = 8.3.3; + LastSwiftMigration = 0900; + ProvisioningStyle = Automatic; + }; + }; + }; + buildConfigurationList = B33BAF521F51F8D90001A751 /* Build configuration list for PBXProject "FastlaneSwiftRunner" */; + compatibilityVersion = "Xcode 3.2"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + en, + Base, + ); + mainGroup = B33BAF4E1F51F8D90001A751; + productRefGroup = B33BAF4E1F51F8D90001A751; + projectDirPath = ""; + projectRoot = ""; + targets = ( + B33BAF561F51F8D90001A751 /* FastlaneRunner */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXSourcesBuildPhase section */ + B33BAF531F51F8D90001A751 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + B3BA65A91F5A269100B34850 /* RubyCommand.swift in Sources */, + 8B92CD0928DC63A700A3FF05 /* Secret.swift in Sources */, + 8B92CD0728DC638800A3FF05 /* Constant.swift in Sources */, + D5D1DE991FFEE8EA00502A00 /* RubyCommandable.swift in Sources */, + D55B28C41F6C588300DC42C5 /* Gymfile.swift in Sources */, + 8B9386D728E2A8630045D709 /* Symbol.swift in Sources */, + 1C2D0FA528E4512E0059FF5A /* Version.swift in Sources */, + 8B92CD0B28DC63E000A3FF05 /* Keychain.swift in Sources */, + B302067D1F5E3E9000DE6EBD /* MatchfileProtocol.swift in Sources */, + 1267C3F42773A43E004DE48A /* Atomic.swift in Sources */, + B3BA65AC1F5A269100B34850 /* SocketClientDelegateProtocol.swift in Sources */, + B3BA65A71F5A269100B34850 /* LaneFileProtocol.swift in Sources */, + D55B28C61F6C588300DC42C5 /* Precheckfile.swift in Sources */, + 8B92CD0428DC62D200A3FF05 /* Match.swift in Sources */, + B302067F1F5E3E9000DE6EBD /* ScanfileProtocol.swift in Sources */, + D55B28C51F6C588300DC42C5 /* Matchfile.swift in Sources */, + B30206801F5E3E9000DE6EBD /* ScreengrabfileProtocol.swift in Sources */, + D5BAFD121F7DAAFC0030B324 /* ArgumentProcessor.swift in Sources */, + B302067C1F5E3E9000DE6EBD /* GymfileProtocol.swift in Sources */, + B302067B1F5E3E9000DE6EBD /* SnapshotfileProtocol.swift in Sources */, + 90DA64BD291BC7F400BABF4D /* EnvironmentParser.swift in Sources */, + D55B28C31F6C588300DC42C5 /* Deliverfile.swift in Sources */, + 90C4D77D28E0AB2A00E06274 /* Distribution.swift in Sources */, + D5A7C4901F7C4DAF00A91DE6 /* Fastfile.swift in Sources */, + 0311E38B230AC9490060BB5C /* Actions.swift in Sources */, + D5A7C48F1F7C4DAF00A91DE6 /* Appfile.swift in Sources */, + B3BA65AB1F5A269100B34850 /* SocketClient.swift in Sources */, + B30206811F5E3E9000DE6EBD /* DeliverfileProtocol.swift in Sources */, + B3BA65AA1F5A269100B34850 /* Runner.swift in Sources */, + B3BA65AF1F5A2D5C00B34850 /* RunnerArgument.swift in Sources */, + 90C4D77B28DDC86800E06274 /* Build.swift in Sources */, + 90F4F4CB28E4455C008BDAE5 /* AppStoreAuthentication.swift in Sources */, + D5B8A5B31FFDC49E00536B24 /* ControlCommand.swift in Sources */, + 1257253924B7992C00E04FA3 /* main.swift in Sources */, + B302067E1F5E3E9000DE6EBD /* PrecheckfileProtocol.swift in Sources */, + B3BA65AD1F5A269100B34850 /* SocketResponse.swift in Sources */, + D55B28C71F6C588300DC42C5 /* Scanfile.swift in Sources */, + 0311E387230AC1B20060BB5C /* Plugins.swift in Sources */, + D55B28C91F6C588300DC42C5 /* Snapshotfile.swift in Sources */, + B3BA65A61F5A269100B34850 /* Fastlane.swift in Sources */, + 1C2D0FA728E457D10059FF5A /* Test.swift in Sources */, + D55B28C81F6C588300DC42C5 /* Screengrabfile.swift in Sources */, + 12D2EB8D2620D83C00844013 /* OptionalConfigValue.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin XCBuildConfiguration section */ + B33BAF5C1F51F8D90001A751 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + CODE_SIGN_IDENTITY = "-"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = dwarf; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_TESTABILITY = YES; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_DYNAMIC_NO_PIC = NO; + GCC_NO_COMMON_BLOCKS = YES; + GCC_OPTIMIZATION_LEVEL = 0; + GCC_PREPROCESSOR_DEFINITIONS = ( + "DEBUG=1", + "$(inherited)", + ); + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + MACOSX_DEPLOYMENT_TARGET = 10.12; + MTL_ENABLE_DEBUG_INFO = YES; + ONLY_ACTIVE_ARCH = YES; + SDKROOT = macosx; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_SWIFT3_OBJC_INFERENCE = Off; + }; + name = Debug; + }; + B33BAF5D1F51F8D90001A751 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + CODE_SIGN_IDENTITY = "-"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + MACOSX_DEPLOYMENT_TARGET = 10.12; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = macosx; + SWIFT_OPTIMIZATION_LEVEL = "-Owholemodule"; + SWIFT_SWIFT3_OBJC_INFERENCE = Off; + }; + name = Release; + }; + B33BAF5F1F51F8D90001A751 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + CLANG_ENABLE_MODULES = YES; + CODE_SIGN_IDENTITY = "-"; + LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/../Frameworks @loader_path/../Frameworks"; + MACOSX_DEPLOYMENT_TARGET = 10.12; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 4.0; + }; + name = Debug; + }; + B33BAF601F51F8D90001A751 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + CLANG_ENABLE_MODULES = YES; + CODE_SIGN_IDENTITY = "-"; + LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/../Frameworks @loader_path/../Frameworks"; + MACOSX_DEPLOYMENT_TARGET = 10.12; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 4.0; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + B33BAF521F51F8D90001A751 /* Build configuration list for PBXProject "FastlaneSwiftRunner" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + B33BAF5C1F51F8D90001A751 /* Debug */, + B33BAF5D1F51F8D90001A751 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + B33BAF5E1F51F8D90001A751 /* Build configuration list for PBXNativeTarget "FastlaneRunner" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + B33BAF5F1F51F8D90001A751 /* Debug */, + B33BAF601F51F8D90001A751 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = B33BAF4F1F51F8D90001A751 /* Project object */; +} diff --git a/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..44be89b1 --- /dev/null +++ b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 00000000..18d98100 --- /dev/null +++ b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/xcshareddata/xcschemes/FastlaneRunner.xcscheme b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/xcshareddata/xcschemes/FastlaneRunner.xcscheme new file mode 100644 index 00000000..94572138 --- /dev/null +++ b/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/xcshareddata/xcschemes/FastlaneRunner.xcscheme @@ -0,0 +1,97 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/fastlane/swift/FastlaneSwiftRunner/README.txt b/fastlane/swift/FastlaneSwiftRunner/README.txt new file mode 100644 index 00000000..5fb55ccd --- /dev/null +++ b/fastlane/swift/FastlaneSwiftRunner/README.txt @@ -0,0 +1,10 @@ +Don't modify the structure of this group including but not limited to: +- renaming this group +- adding sub groups +- removing sub groups +- adding new files +- removing files + +If you modify anything in this folder, future fastlane upgrades may not be able to be applied automatically. + +If you need to add new groups, please add them at the root of the "Fastlane Runner" group. diff --git a/fastlane/swift/Gymfile.swift b/fastlane/swift/Gymfile.swift new file mode 100644 index 00000000..4b6abf0a --- /dev/null +++ b/fastlane/swift/Gymfile.swift @@ -0,0 +1,20 @@ +// Gymfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `gym` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Gymfile: GymfileProtocol { + // If you want to enable `gym`, run `fastlane gym init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/GymfileProtocol.swift b/fastlane/swift/GymfileProtocol.swift new file mode 100644 index 00000000..2b1898ec --- /dev/null +++ b/fastlane/swift/GymfileProtocol.swift @@ -0,0 +1,207 @@ +// GymfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol GymfileProtocol: AnyObject { + /// Path to the workspace file + var workspace: String? { get } + + /// Path to the project file + var project: String? { get } + + /// The project's scheme. Make sure it's marked as `Shared` + var scheme: String? { get } + + /// Should the project be cleaned before building it? + var clean: Bool { get } + + /// The directory in which the ipa file should be stored in + var outputDirectory: String { get } + + /// The name of the resulting ipa file + var outputName: String? { get } + + /// The configuration to use when building the app. Defaults to 'Release' + var configuration: String? { get } + + /// Hide all information that's not necessary while building + var silent: Bool { get } + + /// The name of the code signing identity to use. It has to match the name exactly. e.g. 'iPhone Distribution: SunApps GmbH' + var codesigningIdentity: String? { get } + + /// Should we skip packaging the ipa? + var skipPackageIpa: Bool { get } + + /// Should we skip packaging the pkg? + var skipPackagePkg: Bool { get } + + /// Should the ipa file include symbols? + var includeSymbols: Bool? { get } + + /// Should the ipa file include bitcode? + var includeBitcode: Bool? { get } + + /// Method used to export the archive. Valid values are: app-store, validation, ad-hoc, package, enterprise, development, developer-id and mac-application + var exportMethod: String? { get } + + /// Path to an export options plist or a hash with export options. Use 'xcodebuild -help' to print the full set of available options + var exportOptions: [String: Any]? { get } + + /// Pass additional arguments to xcodebuild for the package phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + var exportXcargs: String? { get } + + /// Export ipa from previously built xcarchive. Uses archive_path as source + var skipBuildArchive: Bool? { get } + + /// After building, don't archive, effectively not including -archivePath param + var skipArchive: Bool? { get } + + /// Build without codesigning + var skipCodesigning: Bool? { get } + + /// Platform to build when using a Catalyst enabled app. Valid values are: ios, macos + var catalystPlatform: String? { get } + + /// Full name of 3rd Party Mac Developer Installer or Developer ID Installer certificate. Example: `3rd Party Mac Developer Installer: Your Company (ABC1234XWYZ)` + var installerCertName: String? { get } + + /// The directory in which the archive should be stored in + var buildPath: String? { get } + + /// The path to the created archive + var archivePath: String? { get } + + /// The directory where built products and other derived data will go + var derivedDataPath: String? { get } + + /// Should an Xcode result bundle be generated in the output directory + var resultBundle: Bool { get } + + /// Path to the result bundle directory to create. Ignored if `result_bundle` if false + var resultBundlePath: String? { get } + + /// The directory where to store the build log + var buildlogPath: String { get } + + /// The SDK that should be used for building the application + var sdk: String? { get } + + /// The toolchain that should be used for building the application (e.g. com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a) + var toolchain: String? { get } + + /// Use a custom destination for building the app + var destination: String? { get } + + /// Optional: Sometimes you need to specify a team id when exporting the ipa file + var exportTeamId: String? { get } + + /// Pass additional arguments to xcodebuild for the build phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + var xcargs: String? { get } + + /// Use an extra XCCONFIG file to build your app + var xcconfig: String? { get } + + /// Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + var suppressXcodeOutput: Bool? { get } + + /// xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + var xcodebuildFormatter: String { get } + + /// **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Disable xcpretty formatting of build output + var disableXcpretty: Bool? { get } + + /// Use the test (RSpec style) format for build output + var xcprettyTestFormat: Bool? { get } + + /// A custom xcpretty formatter to use + var xcprettyFormatter: String? { get } + + /// Have xcpretty create a JUnit-style XML report at the provided path + var xcprettyReportJunit: String? { get } + + /// Have xcpretty create a simple HTML report at the provided path + var xcprettyReportHtml: String? { get } + + /// Have xcpretty create a JSON compilation database at the provided path + var xcprettyReportJson: String? { get } + + /// Have xcpretty use unicode encoding when reporting builds + var xcprettyUtf: Bool? { get } + + /// Analyze the project build time and store the output in 'culprits.txt' file + var analyzeBuildTime: Bool? { get } + + /// Do not try to build a profile mapping from the xcodeproj. Match or a manually provided mapping should be used + var skipProfileDetection: Bool { get } + + /// Allows for override of the default `xcodebuild` command + var xcodebuildCommand: String { get } + + /// Sets a custom path for Swift Package Manager dependencies + var clonedSourcePackagesPath: String? { get } + + /// Skips resolution of Swift Package Manager dependencies + var skipPackageDependenciesResolution: Bool { get } + + /// Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + var disablePackageAutomaticUpdates: Bool { get } + + /// Lets xcodebuild use system's scm configuration + var useSystemScm: Bool { get } +} + +public extension GymfileProtocol { + var workspace: String? { return nil } + var project: String? { return nil } + var scheme: String? { return nil } + var clean: Bool { return false } + var outputDirectory: String { return "." } + var outputName: String? { return nil } + var configuration: String? { return nil } + var silent: Bool { return false } + var codesigningIdentity: String? { return nil } + var skipPackageIpa: Bool { return false } + var skipPackagePkg: Bool { return false } + var includeSymbols: Bool? { return nil } + var includeBitcode: Bool? { return nil } + var exportMethod: String? { return nil } + var exportOptions: [String: Any]? { return nil } + var exportXcargs: String? { return nil } + var skipBuildArchive: Bool? { return nil } + var skipArchive: Bool? { return nil } + var skipCodesigning: Bool? { return nil } + var catalystPlatform: String? { return nil } + var installerCertName: String? { return nil } + var buildPath: String? { return nil } + var archivePath: String? { return nil } + var derivedDataPath: String? { return nil } + var resultBundle: Bool { return false } + var resultBundlePath: String? { return nil } + var buildlogPath: String { return "~/Library/Logs/gym" } + var sdk: String? { return nil } + var toolchain: String? { return nil } + var destination: String? { return nil } + var exportTeamId: String? { return nil } + var xcargs: String? { return nil } + var xcconfig: String? { return nil } + var suppressXcodeOutput: Bool? { return nil } + var xcodebuildFormatter: String { return "xcbeautify" } + var disableXcpretty: Bool? { return nil } + var xcprettyTestFormat: Bool? { return nil } + var xcprettyFormatter: String? { return nil } + var xcprettyReportJunit: String? { return nil } + var xcprettyReportHtml: String? { return nil } + var xcprettyReportJson: String? { return nil } + var xcprettyUtf: Bool? { return nil } + var analyzeBuildTime: Bool? { return nil } + var skipProfileDetection: Bool { return false } + var xcodebuildCommand: String { return "xcodebuild" } + var clonedSourcePackagesPath: String? { return nil } + var skipPackageDependenciesResolution: Bool { return false } + var disablePackageAutomaticUpdates: Bool { return false } + var useSystemScm: Bool { return false } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.115] diff --git a/fastlane/swift/LaneFileProtocol.swift b/fastlane/swift/LaneFileProtocol.swift new file mode 100644 index 00000000..19c17898 --- /dev/null +++ b/fastlane/swift/LaneFileProtocol.swift @@ -0,0 +1,155 @@ +// LaneFileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +public protocol LaneFileProtocol: AnyObject { + var fastlaneVersion: String { get } + static func runLane(from fastfile: LaneFile?, named lane: String, with parameters: [String: String]) -> Bool + + func recordLaneDescriptions() + func beforeAll(with lane: String) + func afterAll(with lane: String) + func onError(currentLane: String, errorInfo: String, errorClass: String?, errorMessage: String?) +} + +public extension LaneFileProtocol { + var fastlaneVersion: String { return "" } // Defaults to "" because that means any is fine + func beforeAll(with _: String) {} // No-op by default + func afterAll(with _: String) {} // No-op by default + func recordLaneDescriptions() {} // No-op by default +} + +@objcMembers +open class LaneFile: NSObject, LaneFileProtocol { + private(set) static var fastfileInstance: LaneFile? + private static var onErrorCalled = Set() + + private static func trimLaneFromName(laneName: String) -> String { + return String(laneName.prefix(laneName.count - 4)) + } + + private static func trimLaneWithOptionsFromName(laneName: String) -> String { + return String(laneName.prefix(laneName.count - 12)) + } + + public func onError(currentLane: String, errorInfo _: String, errorClass _: String?, errorMessage _: String?) { + LaneFile.onErrorCalled.insert(currentLane) + } + + private static var laneFunctionNames: [String] { + var lanes: [String] = [] + var methodCount: UInt32 = 0 + #if !SWIFT_PACKAGE + let methodList = class_copyMethodList(self, &methodCount) + #else + // In SPM we're calling this functions out of the scope of the normal binary that it's + // being built, so *self* in this scope would be the SPM executable instead of the Fastfile + // that we'd normally expect. + let methodList = class_copyMethodList(type(of: fastfileInstance!), &methodCount) + #endif + for i in 0 ..< Int(methodCount) { + let selName = sel_getName(method_getName(methodList![i])) + let name = String(cString: selName) + let lowercasedName = name.lowercased() + if lowercasedName.hasSuffix("lane") || lowercasedName.hasSuffix("lanewithoptions:") { + lanes.append(name) + } + } + return lanes + } + + public static var lanes: [String: String] { + var laneToMethodName: [String: String] = [:] + laneFunctionNames.forEach { name in + let lowercasedName = name.lowercased() + if lowercasedName.hasSuffix("lane") { + laneToMethodName[lowercasedName] = name + let lowercasedNameNoLane = trimLaneFromName(laneName: lowercasedName) + laneToMethodName[lowercasedNameNoLane] = name + } else if lowercasedName.hasSuffix("lanewithoptions:") { + let lowercasedNameNoOptions = trimLaneWithOptionsFromName(laneName: lowercasedName) + laneToMethodName[lowercasedNameNoOptions] = name + let lowercasedNameNoLane = trimLaneFromName(laneName: lowercasedNameNoOptions) + laneToMethodName[lowercasedNameNoLane] = name + } + } + + return laneToMethodName + } + + public static func loadFastfile() { + if fastfileInstance == nil { + let fastfileType: AnyObject.Type = NSClassFromString(className())! + let fastfileAsNSObjectType: NSObject.Type = fastfileType as! NSObject.Type + let currentFastfileInstance: Fastfile? = fastfileAsNSObjectType.init() as? Fastfile + fastfileInstance = currentFastfileInstance + } + } + + public static func runLane(from fastfile: LaneFile?, named lane: String, with parameters: [String: String]) -> Bool { + log(message: "Running lane: \(lane)") + #if !SWIFT_PACKAGE + // When not in SPM environment, we load the Fastfile from its `className()`. + loadFastfile() + guard let fastfileInstance = fastfileInstance as? Fastfile else { + let message = "Unable to instantiate class named: \(className())" + log(message: message) + fatalError(message) + } + #else + // When in SPM environment, we can't load the Fastfile from its `className()` because the executable is in + // another scope, so `className()` won't be the expected Fastfile. Instead, we load the Fastfile as a Lanefile + // in a static way, by parameter. + guard let fastfileInstance = fastfile else { + log(message: "Found nil instance of fastfile") + preconditionFailure() + } + self.fastfileInstance = fastfileInstance + #endif + let currentLanes = lanes + let lowerCasedLaneRequested = lane.lowercased() + + guard let laneMethod = currentLanes[lowerCasedLaneRequested] else { + let laneNames = laneFunctionNames.map { laneFuctionName in + if laneFuctionName.hasSuffix("lanewithoptions:") { + return trimLaneWithOptionsFromName(laneName: laneFuctionName) + } else { + return trimLaneFromName(laneName: laneFuctionName) + } + }.joined(separator: ", ") + + let message = "[!] Could not find lane '\(lane)'. Available lanes: \(laneNames)" + log(message: message) + + let shutdownCommand = ControlCommand(commandType: .cancel(cancelReason: .clientError), message: message) + _ = runner.executeCommand(shutdownCommand) + return false + } + + // Call all methods that need to be called before we start calling lanes. + fastfileInstance.beforeAll(with: lane) + + // We need to catch all possible errors here and display a nice message. + _ = fastfileInstance.perform(NSSelectorFromString(laneMethod), with: parameters) + + // Call only on success. + if !LaneFile.onErrorCalled.contains(lane) { + fastfileInstance.afterAll(with: lane) + } + + log(message: "Done running lane: \(lane) 🚀") + return true + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/MainProcess.swift b/fastlane/swift/MainProcess.swift new file mode 100644 index 00000000..da43cd4f --- /dev/null +++ b/fastlane/swift/MainProcess.swift @@ -0,0 +1,79 @@ +// MainProcess.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation +#if canImport(SwiftShell) + import SwiftShell +#endif + +let argumentProcessor = ArgumentProcessor(args: CommandLine.arguments) +let timeout = argumentProcessor.commandTimeout + +class MainProcess { + var doneRunningLane = false + var thread: Thread! + #if SWIFT_PACKAGE + var lastPrintDate = Date.distantFuture + var timeBetweenPrints = Int.min + var rubySocketCommand: AsyncCommand! + #endif + + @objc func connectToFastlaneAndRunLane(_ fastfile: LaneFile?) { + runner.startSocketThread(port: argumentProcessor.port) + + let completedRun = Fastfile.runLane(from: fastfile, named: argumentProcessor.currentLane, with: argumentProcessor.laneParameters()) + if completedRun { + runner.disconnectFromFastlaneProcess() + } + + doneRunningLane = true + } + + func startFastlaneThread(with fastFile: LaneFile?) { + #if !SWIFT_PACKAGE + thread = Thread(target: self, selector: #selector(connectToFastlaneAndRunLane), object: nil) + #else + thread = Thread(target: self, selector: #selector(connectToFastlaneAndRunLane), object: fastFile) + #endif + thread.name = "worker thread" + #if SWIFT_PACKAGE + let PATH = run("/bin/bash", "-c", "-l", "eval $(/usr/libexec/path_helper -s) ; echo $PATH").stdout + main.env["PATH"] = PATH + let path = main.run(bash: "which fastlane").stdout + let pids = main.run("lsof", "-t", "-i", ":\(argumentProcessor.port)").stdout.split(separator: "\n") + pids.forEach { main.run("kill", "-9", $0) } + rubySocketCommand = main.runAsync(path, "socket_server", "-c", argumentProcessor.commandTimeout, "-p", argumentProcessor.port) + lastPrintDate = Date() + rubySocketCommand.stderror.onStringOutput { print($0) } + rubySocketCommand.stdout.onStringOutput { stdout in + print(stdout) + self.timeBetweenPrints = Int(self.lastPrintDate.timeIntervalSinceNow) + } + + // swiftformat:disable:next redundantSelf + _ = Runner.waitWithPolling(self.timeBetweenPrints, toEventually: { $0 > 5 }, timeout: 10) + thread.start() + #endif + } +} + +public class Main { + let process = MainProcess() + + public init() {} + + public func run(with fastFile: LaneFile?) { + process.startFastlaneThread(with: fastFile) + + while !process.doneRunningLane, RunLoop.current.run(mode: RunLoopMode.defaultRunLoopMode, before: Date(timeIntervalSinceNow: 2)) { + // no op + } + } +} diff --git a/fastlane/swift/Matchfile.swift b/fastlane/swift/Matchfile.swift new file mode 100644 index 00000000..14867097 --- /dev/null +++ b/fastlane/swift/Matchfile.swift @@ -0,0 +1,20 @@ +// Matchfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `match` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Matchfile: MatchfileProtocol { + // If you want to enable `match`, run `fastlane match init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/MatchfileProtocol.swift b/fastlane/swift/MatchfileProtocol.swift new file mode 100644 index 00000000..d7a61866 --- /dev/null +++ b/fastlane/swift/MatchfileProtocol.swift @@ -0,0 +1,211 @@ +// MatchfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol MatchfileProtocol: AnyObject { + /// Define the profile type, can be appstore, adhoc, development, enterprise, developer_id, mac_installer_distribution + var type: String { get } + + /// Create additional cert types needed for macOS installers (valid values: mac_installer_distribution, developer_id_installer) + var additionalCertTypes: [String]? { get } + + /// Only fetch existing certificates and profiles, don't generate new ones + var readonly: Bool { get } + + /// Create a certificate type for Xcode 11 and later (Apple Development or Apple Distribution) + var generateAppleCerts: Bool { get } + + /// Skip syncing provisioning profiles + var skipProvisioningProfiles: Bool { get } + + /// The bundle identifier(s) of your app (comma-separated string or array of strings) + var appIdentifier: [String] { get } + + /// Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + var apiKeyPath: String? { get } + + /// Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + var apiKey: [String: Any]? { get } + + /// Your Apple ID Username + var username: String? { get } + + /// The ID of your Developer Portal team if you're in multiple teams + var teamId: String? { get } + + /// The name of your Developer Portal team if you're in multiple teams + var teamName: String? { get } + + /// Define where you want to store your certificates + var storageMode: String { get } + + /// URL to the git repo containing all the certificates + var gitUrl: String { get } + + /// Specific git branch to use + var gitBranch: String { get } + + /// git user full name to commit + var gitFullName: String? { get } + + /// git user email to commit + var gitUserEmail: String? { get } + + /// Make a shallow clone of the repository (truncate the history to 1 revision) + var shallowClone: Bool { get } + + /// Clone just the branch specified, instead of the whole repo. This requires that the branch already exists. Otherwise the command will fail + var cloneBranchDirectly: Bool { get } + + /// Use a basic authorization header to access the git repo (e.g.: access via HTTPS, GitHub Actions, etc), usually a string in Base64 + var gitBasicAuthorization: String? { get } + + /// Use a bearer authorization header to access the git repo (e.g.: access to an Azure DevOps repository), usually a string in Base64 + var gitBearerAuthorization: String? { get } + + /// Use a private key to access the git repo (e.g.: access to GitHub repository via Deploy keys), usually a id_rsa named file or the contents hereof + var gitPrivateKey: String? { get } + + /// Name of the Google Cloud Storage bucket to use + var googleCloudBucketName: String? { get } + + /// Path to the gc_keys.json file + var googleCloudKeysFile: String? { get } + + /// ID of the Google Cloud project to use for authentication + var googleCloudProjectId: String? { get } + + /// Skips confirming to use the system google account + var skipGoogleCloudAccountConfirmation: Bool { get } + + /// Name of the S3 region + var s3Region: String? { get } + + /// S3 access key + var s3AccessKey: String? { get } + + /// S3 secret access key + var s3SecretAccessKey: String? { get } + + /// Name of the S3 bucket + var s3Bucket: String? { get } + + /// Prefix to be used on all objects uploaded to S3 + var s3ObjectPrefix: String? { get } + + /// GitLab Project Path (i.e. 'gitlab-org/gitlab') + var gitlabProject: String? { get } + + /// Keychain the items should be imported to + var keychainName: String { get } + + /// This might be required the first time you access certificates on a new mac. For the login/default keychain this is your macOS account password + var keychainPassword: String? { get } + + /// Renew the provisioning profiles every time you run match + var force: Bool { get } + + /// Renew the provisioning profiles if the device count on the developer portal has changed. Ignored for profile types 'appstore' and 'developer_id' + var forceForNewDevices: Bool { get } + + /// Include Apple Silicon Mac devices in provisioning profiles for iOS/iPadOS apps + var includeMacInProfiles: Bool { get } + + /// Include all matching certificates in the provisioning profile. Works only for the 'development' provisioning profile type + var includeAllCertificates: Bool { get } + + /// Renew the provisioning profiles if the certificate count on the developer portal has changed. Works only for the 'development' provisioning profile type. Requires 'include_all_certificates' option to be 'true' + var forceForNewCertificates: Bool { get } + + /// Disables confirmation prompts during nuke, answering them with yes + var skipConfirmation: Bool { get } + + /// Remove certs from repository during nuke without revoking them on the developer portal + var safeRemoveCerts: Bool { get } + + /// Skip generation of a README.md for the created git repository + var skipDocs: Bool { get } + + /// Set the provisioning profile's platform to work with (i.e. ios, tvos, macos, catalyst) + var platform: String { get } + + /// Enable this if you have the Mac Catalyst capability enabled and your project was created with Xcode 11.3 or earlier. Prepends 'maccatalyst.' to the app identifier for the provisioning profile mapping + var deriveCatalystAppIdentifier: Bool { get } + + /// The name of provisioning profile template. If the developer account has provisioning profile templates (aka: custom entitlements), the template name can be found by inspecting the Entitlements drop-down while creating/editing a provisioning profile (e.g. "Apple Pay Pass Suppression Development") + var templateName: String? { get } + + /// A custom name for the provisioning profile. This will replace the default provisioning profile name if specified + var profileName: String? { get } + + /// Should the command fail if it was about to create a duplicate of an existing provisioning profile. It can happen due to issues on Apple Developer Portal, when profile to be recreated was not properly deleted first + var failOnNameTaken: Bool { get } + + /// Set to true if there is no access to Apple developer portal but there are certificates, keys and profiles provided. Only works with match import action + var skipCertificateMatching: Bool { get } + + /// Path in which to export certificates, key and profile + var outputPath: String? { get } + + /// Skips setting the partition list (which can sometimes take a long time). Setting the partition list is usually needed to prevent Xcode from prompting to allow a cert to be used for signing + var skipSetPartitionList: Bool { get } + + /// Print out extra information and all commands + var verbose: Bool { get } +} + +public extension MatchfileProtocol { + var type: String { return "development" } + var additionalCertTypes: [String]? { return nil } + var readonly: Bool { return false } + var generateAppleCerts: Bool { return true } + var skipProvisioningProfiles: Bool { return false } + var appIdentifier: [String] { return [] } + var apiKeyPath: String? { return nil } + var apiKey: [String: Any]? { return nil } + var username: String? { return nil } + var teamId: String? { return nil } + var teamName: String? { return nil } + var storageMode: String { return "git" } + var gitUrl: String { return "" } + var gitBranch: String { return "master" } + var gitFullName: String? { return nil } + var gitUserEmail: String? { return nil } + var shallowClone: Bool { return false } + var cloneBranchDirectly: Bool { return false } + var gitBasicAuthorization: String? { return nil } + var gitBearerAuthorization: String? { return nil } + var gitPrivateKey: String? { return nil } + var googleCloudBucketName: String? { return nil } + var googleCloudKeysFile: String? { return nil } + var googleCloudProjectId: String? { return nil } + var skipGoogleCloudAccountConfirmation: Bool { return false } + var s3Region: String? { return nil } + var s3AccessKey: String? { return nil } + var s3SecretAccessKey: String? { return nil } + var s3Bucket: String? { return nil } + var s3ObjectPrefix: String? { return nil } + var gitlabProject: String? { return nil } + var keychainName: String { return "login.keychain" } + var keychainPassword: String? { return nil } + var force: Bool { return false } + var forceForNewDevices: Bool { return false } + var includeMacInProfiles: Bool { return false } + var includeAllCertificates: Bool { return false } + var forceForNewCertificates: Bool { return false } + var skipConfirmation: Bool { return false } + var safeRemoveCerts: Bool { return false } + var skipDocs: Bool { return false } + var platform: String { return "ios" } + var deriveCatalystAppIdentifier: Bool { return false } + var templateName: String? { return nil } + var profileName: String? { return nil } + var failOnNameTaken: Bool { return false } + var skipCertificateMatching: Bool { return false } + var outputPath: String? { return nil } + var skipSetPartitionList: Bool { return false } + var verbose: Bool { return false } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.109] diff --git a/fastlane/swift/OptionalConfigValue.swift b/fastlane/swift/OptionalConfigValue.swift new file mode 100644 index 00000000..f96142dd --- /dev/null +++ b/fastlane/swift/OptionalConfigValue.swift @@ -0,0 +1,101 @@ +// OptionalConfigValue.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +public enum OptionalConfigValue { + case fastlaneDefault(T) + case userDefined(T) + case `nil` + + func asRubyArgument(name: String, type: RubyCommand.Argument.ArgType? = nil) -> RubyCommand.Argument? { + if case let .userDefined(value) = self { + return RubyCommand.Argument(name: name, value: value, type: type) + } + return nil + } +} + +extension OptionalConfigValue: ExpressibleByUnicodeScalarLiteral where T == String? { + public typealias UnicodeScalarLiteralType = String + + public init(unicodeScalarLiteral value: UnicodeScalarLiteralType) { + self = .userDefined(value) + } +} + +extension OptionalConfigValue: ExpressibleByExtendedGraphemeClusterLiteral where T == String? { + public typealias ExtendedGraphemeClusterLiteralType = String + + public init(extendedGraphemeClusterLiteral value: ExtendedGraphemeClusterLiteralType) { + self = .userDefined(value) + } +} + +extension OptionalConfigValue: ExpressibleByStringLiteral where T == String? { + public typealias StringLiteralType = String + + public init(stringLiteral value: StringLiteralType) { + self = .userDefined(value) + } +} + +extension OptionalConfigValue: ExpressibleByStringInterpolation where T == String? {} + +extension OptionalConfigValue: ExpressibleByNilLiteral { + public init(nilLiteral _: ()) { + self = .nil + } +} + +extension OptionalConfigValue: ExpressibleByIntegerLiteral where T == Int? { + public typealias IntegerLiteralType = Int + + public init(integerLiteral value: IntegerLiteralType) { + self = .userDefined(value) + } +} + +extension OptionalConfigValue: ExpressibleByArrayLiteral where T == [String] { + public typealias ArrayLiteralElement = String + + public init(arrayLiteral elements: ArrayLiteralElement...) { + self = .userDefined(elements) + } +} + +extension OptionalConfigValue: ExpressibleByFloatLiteral where T == Float { + public typealias FloatLiteralType = Float + + public init(floatLiteral value: FloatLiteralType) { + self = .userDefined(value) + } +} + +extension OptionalConfigValue: ExpressibleByBooleanLiteral where T == Bool { + public typealias BooleanLiteralType = Bool + + public init(booleanLiteral value: BooleanLiteralType) { + self = .userDefined(value) + } +} + +extension OptionalConfigValue: ExpressibleByDictionaryLiteral where T == [String: Any] { + public typealias Key = String + public typealias Value = Any + + public init(dictionaryLiteral elements: (Key, Value)...) { + var dict: [Key: Value] = [:] + elements.forEach { + dict[$0.0] = $0.1 + } + self = .userDefined(dict) + } +} diff --git a/fastlane/swift/Plugins.swift b/fastlane/swift/Plugins.swift new file mode 100644 index 00000000..57c0eab0 --- /dev/null +++ b/fastlane/swift/Plugins.swift @@ -0,0 +1,241 @@ +import Foundation +/** + Release your beta builds with Firebase App Distribution + + - parameters: + - ipaPath: Path to your IPA file. Optional if you use the _gym_ or _xcodebuild_ action + - googleserviceInfoPlistPath: Path to your GoogleService-Info.plist file, relative to the archived product path + - apkPath: Path to your APK file + - androidArtifactPath: Path to your APK or AAB file + - androidArtifactType: Android artifact type. Set to 'APK' or 'AAB'. Defaults to 'APK' if not set + - app: Your app's Firebase App ID. You can find the App ID in the Firebase console, on the General Settings page + - firebaseCliPath: **DEPRECATED!** This plugin no longer uses the Firebase CLI - The absolute path of the firebase cli command + - groups: The group aliases used for distribution, separated by commas + - groupsFile: The group aliases used for distribution, separated by commas + - testers: Pass email addresses of testers, separated by commas + - testersFile: Pass email addresses of testers, separated by commas + - releaseNotes: Release notes for this build + - releaseNotesFile: Release notes file for this build + - firebaseCliToken: Auth token generated using 'fastlane run firebase_app_distribution_login', or the Firebase CLI's login:ci command + - debug: Print verbose debug output + - serviceCredentialsFile: Path to Google service account json + - uploadTimeout: The amount of seconds before the upload will timeout, if not completed + + Release your beta builds with Firebase App Distribution +*/ +public func firebaseAppDistribution(ipaPath: OptionalConfigValue = .fastlaneDefault(nil), + googleserviceInfoPlistPath: String = "GoogleService-Info.plist", + apkPath: OptionalConfigValue = .fastlaneDefault(nil), + androidArtifactPath: OptionalConfigValue = .fastlaneDefault(nil), + androidArtifactType: String = "APK", + app: OptionalConfigValue = .fastlaneDefault(nil), + firebaseCliPath: OptionalConfigValue = .fastlaneDefault(nil), + groups: OptionalConfigValue = .fastlaneDefault(nil), + groupsFile: OptionalConfigValue = .fastlaneDefault(nil), + testers: OptionalConfigValue = .fastlaneDefault(nil), + testersFile: OptionalConfigValue = .fastlaneDefault(nil), + releaseNotes: OptionalConfigValue = .fastlaneDefault(nil), + releaseNotesFile: OptionalConfigValue = .fastlaneDefault(nil), + firebaseCliToken: OptionalConfigValue = .fastlaneDefault(nil), + debug: OptionalConfigValue = .fastlaneDefault(false), + serviceCredentialsFile: OptionalConfigValue = .fastlaneDefault(nil), + uploadTimeout: Int = 300) { +let ipaPathArg = ipaPath.asRubyArgument(name: "ipa_path", type: nil) +let googleserviceInfoPlistPathArg = RubyCommand.Argument(name: "googleservice_info_plist_path", value: googleserviceInfoPlistPath, type: nil) +let apkPathArg = apkPath.asRubyArgument(name: "apk_path", type: nil) +let androidArtifactPathArg = androidArtifactPath.asRubyArgument(name: "android_artifact_path", type: nil) +let androidArtifactTypeArg = RubyCommand.Argument(name: "android_artifact_type", value: androidArtifactType, type: nil) +let appArg = app.asRubyArgument(name: "app", type: nil) +let firebaseCliPathArg = firebaseCliPath.asRubyArgument(name: "firebase_cli_path", type: nil) +let groupsArg = groups.asRubyArgument(name: "groups", type: nil) +let groupsFileArg = groupsFile.asRubyArgument(name: "groups_file", type: nil) +let testersArg = testers.asRubyArgument(name: "testers", type: nil) +let testersFileArg = testersFile.asRubyArgument(name: "testers_file", type: nil) +let releaseNotesArg = releaseNotes.asRubyArgument(name: "release_notes", type: nil) +let releaseNotesFileArg = releaseNotesFile.asRubyArgument(name: "release_notes_file", type: nil) +let firebaseCliTokenArg = firebaseCliToken.asRubyArgument(name: "firebase_cli_token", type: nil) +let debugArg = debug.asRubyArgument(name: "debug", type: nil) +let serviceCredentialsFileArg = serviceCredentialsFile.asRubyArgument(name: "service_credentials_file", type: nil) +let uploadTimeoutArg = RubyCommand.Argument(name: "upload_timeout", value: uploadTimeout, type: nil) +let array: [RubyCommand.Argument?] = [ipaPathArg, +googleserviceInfoPlistPathArg, +apkPathArg, +androidArtifactPathArg, +androidArtifactTypeArg, +appArg, +firebaseCliPathArg, +groupsArg, +groupsFileArg, +testersArg, +testersFileArg, +releaseNotesArg, +releaseNotesFileArg, +firebaseCliTokenArg, +debugArg, +serviceCredentialsFileArg, +uploadTimeoutArg] +let args: [RubyCommand.Argument] = array +.filter { $0?.value != nil } +.compactMap { $0 } +let command = RubyCommand(commandID: "", methodName: "firebase_app_distribution", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Create testers in bulk from a comma-separated list or a file + + - parameters: + - projectNumber: Your Firebase project number. You can find the project number in the Firebase console, on the General Settings page + - emails: Comma separated list of tester emails to be created. A maximum of 1000 testers can be created at a time + - file: Path to a file containing a comma separated list of tester emails to be created. A maximum of 1000 testers can be deleted at a time + - serviceCredentialsFile: Path to Google service credentials file + - firebaseCliToken: Auth token generated using 'fastlane run firebase_app_distribution_login', or the Firebase CLI's login:ci command + - debug: Print verbose debug output + + Create testers in bulk from a comma-separated list or a file +*/ +public func firebaseAppDistributionAddTesters(projectNumber: Int, + emails: OptionalConfigValue = .fastlaneDefault(nil), + file: OptionalConfigValue = .fastlaneDefault(nil), + serviceCredentialsFile: OptionalConfigValue = .fastlaneDefault(nil), + firebaseCliToken: OptionalConfigValue = .fastlaneDefault(nil), + debug: OptionalConfigValue = .fastlaneDefault(false)) { +let projectNumberArg = RubyCommand.Argument(name: "project_number", value: projectNumber, type: nil) +let emailsArg = emails.asRubyArgument(name: "emails", type: nil) +let fileArg = file.asRubyArgument(name: "file", type: nil) +let serviceCredentialsFileArg = serviceCredentialsFile.asRubyArgument(name: "service_credentials_file", type: nil) +let firebaseCliTokenArg = firebaseCliToken.asRubyArgument(name: "firebase_cli_token", type: nil) +let debugArg = debug.asRubyArgument(name: "debug", type: nil) +let array: [RubyCommand.Argument?] = [projectNumberArg, +emailsArg, +fileArg, +serviceCredentialsFileArg, +firebaseCliTokenArg, +debugArg] +let args: [RubyCommand.Argument] = array +.filter { $0?.value != nil } +.compactMap { $0 } +let command = RubyCommand(commandID: "", methodName: "firebase_app_distribution_add_testers", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Fetches the latest release in Firebase App Distribution + + - parameters: + - app: Your app's Firebase App ID. You can find the App ID in the Firebase console, on the General Settings page + - firebaseCliToken: Auth token generated using 'fastlane run firebase_app_distribution_login', or the Firebase CLI's login:ci command + - serviceCredentialsFile: Path to Google service account json + - debug: Print verbose debug output + + - returns: Hash representation of the lastest release created in Firebase App Distribution (see https://firebase.google.com/docs/reference/app-distribution/rest/v1/projects.apps.releases#resource:-release). Example: {:name=>"projects/123456789/apps/1:1234567890:ios:0a1b2c3d4e5f67890/releases/0a1b2c3d4", :releaseNotes=>{:text=>"Here are some release notes!"}, :displayVersion=>"1.2.3", :buildVersion=>"10", :createTime=>"2021-10-06T15:01:23Z"} + + Fetches information about the most recently created release in App Distribution, including the version and release notes. Returns nil if no releases are found. +*/ +@discardableResult public func firebaseAppDistributionGetLatestRelease(app: String, + firebaseCliToken: OptionalConfigValue = .fastlaneDefault(nil), + serviceCredentialsFile: OptionalConfigValue = .fastlaneDefault(nil), + debug: OptionalConfigValue = .fastlaneDefault(false)) -> [String : Any] { +let appArg = RubyCommand.Argument(name: "app", value: app, type: nil) +let firebaseCliTokenArg = firebaseCliToken.asRubyArgument(name: "firebase_cli_token", type: nil) +let serviceCredentialsFileArg = serviceCredentialsFile.asRubyArgument(name: "service_credentials_file", type: nil) +let debugArg = debug.asRubyArgument(name: "debug", type: nil) +let array: [RubyCommand.Argument?] = [appArg, +firebaseCliTokenArg, +serviceCredentialsFileArg, +debugArg] +let args: [RubyCommand.Argument] = array +.filter { $0?.value != nil } +.compactMap { $0 } +let command = RubyCommand(commandID: "", methodName: "firebase_app_distribution_get_latest_release", className: nil, args: args) + return parseDictionary(fromString: runner.executeCommand(command)) +} + +/** + Download the UDIDs of your Firebase App Distribution testers + + - parameters: + - app: Your app's Firebase App ID. You can find the App ID in the Firebase console, on the General Settings page + - outputFile: The path to the file where the tester UDIDs will be written + - firebaseCliToken: Auth token generated using 'fastlane run firebase_app_distribution_login', or the Firebase CLI's login:ci command + - serviceCredentialsFile: Path to Google service account json + - debug: Print verbose debug output + + Export your testers' device identifiers in a CSV file, so you can add them your provisioning profile. This file can be imported into your Apple developer account using the Register Multiple Devices option. See the [App Distribution docs](https://firebase.google.com/docs/app-distribution/ios/distribute-console#register-tester-devices) for more info. +*/ +public func firebaseAppDistributionGetUdids(app: String, + outputFile: String, + firebaseCliToken: OptionalConfigValue = .fastlaneDefault(nil), + serviceCredentialsFile: OptionalConfigValue = .fastlaneDefault(nil), + debug: OptionalConfigValue = .fastlaneDefault(false)) { +let appArg = RubyCommand.Argument(name: "app", value: app, type: nil) +let outputFileArg = RubyCommand.Argument(name: "output_file", value: outputFile, type: nil) +let firebaseCliTokenArg = firebaseCliToken.asRubyArgument(name: "firebase_cli_token", type: nil) +let serviceCredentialsFileArg = serviceCredentialsFile.asRubyArgument(name: "service_credentials_file", type: nil) +let debugArg = debug.asRubyArgument(name: "debug", type: nil) +let array: [RubyCommand.Argument?] = [appArg, +outputFileArg, +firebaseCliTokenArg, +serviceCredentialsFileArg, +debugArg] +let args: [RubyCommand.Argument] = array +.filter { $0?.value != nil } +.compactMap { $0 } +let command = RubyCommand(commandID: "", methodName: "firebase_app_distribution_get_udids", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Authenticate with Firebase App Distribution using a Google account. + + - parameter port: Port for the local web server which receives the response from Google's authorization server + + Log in to Firebase App Distribution using a Google account to generate an authentication token. This token is stored within an environment variable and used to authenticate with your Firebase project. See https://firebase.google.com/docs/app-distribution/ios/distribute-fastlane for more information. +*/ +public func firebaseAppDistributionLogin(port: String = "8081") { +let portArg = RubyCommand.Argument(name: "port", value: port, type: nil) +let array: [RubyCommand.Argument?] = [portArg] +let args: [RubyCommand.Argument] = array +.filter { $0?.value != nil } +.compactMap { $0 } +let command = RubyCommand(commandID: "", methodName: "firebase_app_distribution_login", className: nil, args: args) + _ = runner.executeCommand(command) +} + +/** + Delete testers in bulk from a comma-separated list or a file + + - parameters: + - projectNumber: Your Firebase project number. You can find the project number in the Firebase console, on the General Settings page + - emails: Comma separated list of tester emails to be deleted. A maximum of 1000 testers can be deleted at a time + - file: Path to a file containing a comma separated list of tester emails to be deleted. A maximum of 1000 testers can be deleted at a time + - serviceCredentialsFile: Path to Google service credentials file + - firebaseCliToken: Auth token generated using 'fastlane run firebase_app_distribution_login', or the Firebase CLI's login:ci command + - debug: Print verbose debug output + + Delete testers in bulk from a comma-separated list or a file +*/ +public func firebaseAppDistributionRemoveTesters(projectNumber: Int, + emails: OptionalConfigValue = .fastlaneDefault(nil), + file: OptionalConfigValue = .fastlaneDefault(nil), + serviceCredentialsFile: OptionalConfigValue = .fastlaneDefault(nil), + firebaseCliToken: OptionalConfigValue = .fastlaneDefault(nil), + debug: OptionalConfigValue = .fastlaneDefault(false)) { +let projectNumberArg = RubyCommand.Argument(name: "project_number", value: projectNumber, type: nil) +let emailsArg = emails.asRubyArgument(name: "emails", type: nil) +let fileArg = file.asRubyArgument(name: "file", type: nil) +let serviceCredentialsFileArg = serviceCredentialsFile.asRubyArgument(name: "service_credentials_file", type: nil) +let firebaseCliTokenArg = firebaseCliToken.asRubyArgument(name: "firebase_cli_token", type: nil) +let debugArg = debug.asRubyArgument(name: "debug", type: nil) +let array: [RubyCommand.Argument?] = [projectNumberArg, +emailsArg, +fileArg, +serviceCredentialsFileArg, +firebaseCliTokenArg, +debugArg] +let args: [RubyCommand.Argument] = array +.filter { $0?.value != nil } +.compactMap { $0 } +let command = RubyCommand(commandID: "", methodName: "firebase_app_distribution_remove_testers", className: nil, args: args) + _ = runner.executeCommand(command) +} diff --git a/fastlane/swift/Precheckfile.swift b/fastlane/swift/Precheckfile.swift new file mode 100644 index 00000000..26902196 --- /dev/null +++ b/fastlane/swift/Precheckfile.swift @@ -0,0 +1,20 @@ +// Precheckfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `precheck` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Precheckfile: PrecheckfileProtocol { + // If you want to enable `precheck`, run `fastlane precheck init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/PrecheckfileProtocol.swift b/fastlane/swift/PrecheckfileProtocol.swift new file mode 100644 index 00000000..68f8174e --- /dev/null +++ b/fastlane/swift/PrecheckfileProtocol.swift @@ -0,0 +1,55 @@ +// PrecheckfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol PrecheckfileProtocol: AnyObject { + /// Path to your App Store Connect API Key JSON file (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-json-file) + var apiKeyPath: String? { get } + + /// Your App Store Connect API Key information (https://docs.fastlane.tools/app-store-connect-api/#using-fastlane-api-key-hash-option) + var apiKey: [String: Any]? { get } + + /// The bundle identifier of your app + var appIdentifier: String { get } + + /// Your Apple ID Username + var username: String? { get } + + /// The ID of your App Store Connect team if you're in multiple teams + var teamId: String? { get } + + /// The name of your App Store Connect team if you're in multiple teams + var teamName: String? { get } + + /// The platform to use (optional) + var platform: String { get } + + /// The default rule level unless otherwise configured + var defaultRuleLevel: String { get } + + /// Should check in-app purchases? + var includeInAppPurchases: Bool { get } + + /// Should force check live app? + var useLive: Bool { get } + + /// using text indicating that your IAP is free + var freeStuffInIap: String? { get } +} + +public extension PrecheckfileProtocol { + var apiKeyPath: String? { return nil } + var apiKey: [String: Any]? { return nil } + var appIdentifier: String { return "" } + var username: String? { return nil } + var teamId: String? { return nil } + var teamName: String? { return nil } + var platform: String { return "ios" } + var defaultRuleLevel: String { return "error" } + var includeInAppPurchases: Bool { return true } + var useLive: Bool { return false } + var freeStuffInIap: String? { return nil } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.108] diff --git a/fastlane/swift/RubyCommand.swift b/fastlane/swift/RubyCommand.swift new file mode 100644 index 00000000..d4e02862 --- /dev/null +++ b/fastlane/swift/RubyCommand.swift @@ -0,0 +1,157 @@ +// RubyCommand.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +struct RubyCommand: RubyCommandable { + var type: CommandType { return .action } + + struct Argument { + enum ArgType { + case stringClosure + + var typeString: String { + switch self { + case .stringClosure: + return "string_closure" // this should match when is in ruby's SocketServerActionCommandExecutor + } + } + } + + let name: String + let value: Any? + let type: ArgType? + + init(name: String, value: Any?, type: ArgType? = nil) { + self.name = name + self.value = value + self.type = type + } + + var hasValue: Bool { + return value != nil + } + + var json: String { + if let someValue = value { + let typeJson: String + if let type = type { + typeJson = ", \"value_type\" : \"\(type.typeString)\"" + } else { + typeJson = "" + } + + if type == .stringClosure { + return "{\"name\" : \"\(name)\", \"value\" : \"ignored_for_closure\"\(typeJson)}" + } else if let array = someValue as? [String] { + return "{\"name\" : \"\(name)\", \"value\" : \(array)\(typeJson)}" + } else if let hash = someValue as? [String: Any] { + let jsonData = try! JSONSerialization.data(withJSONObject: hash, options: []) + let jsonString = String(data: jsonData, encoding: .utf8)! + return "{\"name\" : \"\(name)\", \"value\" : \(jsonString)\(typeJson)}" + } else { + let dictionary = [ + "name": name, + "value": someValue, + ] + let jsonData = try! JSONSerialization.data(withJSONObject: dictionary, options: []) + let jsonString = String(data: jsonData, encoding: .utf8)! + return jsonString + } + } else { + // Just exclude this arg if it doesn't have a value + return "" + } + } + } + + let commandID: String + let methodName: String + let className: String? + let args: [Argument] + let id: String = UUID().uuidString + + var closure: ((String) -> Void)? { + let callbacks = args.filter { ($0.type != nil) && $0.type == .stringClosure } + guard let callback = callbacks.first else { + return nil + } + + guard let callbackArgValue = callback.value else { + return nil + } + + guard let callbackClosure = callbackArgValue as? ((String) -> Void) else { + return nil + } + return callbackClosure + } + + func callbackClosure(_ callbackArg: String) -> ((String) -> Void)? { + // WARNING: This will perform the first callback it receives + let callbacks = args.filter { ($0.type != nil) && $0.type == .stringClosure } + guard let callback = callbacks.first else { + verbose(message: "received call to performCallback with \(callbackArg), but no callback available to perform") + return nil + } + + guard let callbackArgValue = callback.value else { + verbose(message: "received call to performCallback with \(callbackArg), but callback is nil") + return nil + } + + guard let callbackClosure = callbackArgValue as? ((String) -> Void) else { + verbose(message: "received call to performCallback with \(callbackArg), but callback type is unknown \(callbackArgValue.self)") + return nil + } + return callbackClosure + } + + func performCallback(callbackArg: String, socket: SocketClient, completion: @escaping () -> Void) { + verbose(message: "Performing callback with: \(callbackArg)") + socket.leave() + callbackClosure(callbackArg)?(callbackArg) + completion() + } + + var commandJson: String { + let argsArrayJson = args + .map { $0.json } + .filter { $0 != "" } + + let argsJson: String? + if !argsArrayJson.isEmpty { + argsJson = "\"args\" : [\(argsArrayJson.joined(separator: ","))]" + } else { + argsJson = nil + } + + let commandIDJson = "\"commandID\" : \"\(commandID)\"" + let methodNameJson = "\"methodName\" : \"\(methodName)\"" + + var jsonParts = [commandIDJson, methodNameJson] + if let argsJson = argsJson { + jsonParts.append(argsJson) + } + + if let className = className { + let classNameJson = "\"className\" : \"\(className)\"" + jsonParts.append(classNameJson) + } + + let commandJsonString = "{\(jsonParts.joined(separator: ","))}" + + return commandJsonString + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/RubyCommandable.swift b/fastlane/swift/RubyCommandable.swift new file mode 100644 index 00000000..0f35c105 --- /dev/null +++ b/fastlane/swift/RubyCommandable.swift @@ -0,0 +1,43 @@ +// RubyCommandable.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +enum CommandType { + case action + case control + + var token: String { + switch self { + case .action: + return "action" + case .control: + return "control" + } + } +} + +protocol RubyCommandable { + var type: CommandType { get } + var commandJson: String { get } + var id: String { get } +} + +extension RubyCommandable { + var json: String { + return """ + { "commandType": "\(type.token)", "command": \(commandJson) } + """ + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/Runner.swift b/fastlane/swift/Runner.swift new file mode 100644 index 00000000..a83108c7 --- /dev/null +++ b/fastlane/swift/Runner.swift @@ -0,0 +1,279 @@ +// Runner.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +let logger: Logger = .init() + +let runner: Runner = .init() + +func desc(_: String) { + // no-op, this is handled in fastlane/lane_list.rb +} + +class Runner { + private var thread: Thread! + private var socketClient: SocketClient! + private let dispatchGroup = DispatchGroup() + private var returnValue: String? // lol, so safe + private var currentlyExecutingCommand: RubyCommandable? + private var shouldLeaveDispatchGroupDuringDisconnect = false + private var executeNext: AtomicDictionary = { + if #available(macOS 10.12, *) { + return UnfairAtomicDictionary() + } else { + return OSSPinAtomicDictionary() + } + }() + + func executeCommand(_ command: RubyCommandable) -> String { + dispatchGroup.enter() + currentlyExecutingCommand = command + socketClient.send(rubyCommand: command) + + let secondsToWait = DispatchTimeInterval.seconds(SocketClient.defaultCommandTimeoutSeconds) + // swiftformat:disable:next redundantSelf + let timeoutResult = Self.waitWithPolling(self.executeNext[command.id], toEventually: { $0 == true }, timeout: SocketClient.defaultCommandTimeoutSeconds) + executeNext.removeValue(forKey: command.id) + let failureMessage = "command didn't execute in: \(SocketClient.defaultCommandTimeoutSeconds) seconds" + let success = testDispatchTimeoutResult(timeoutResult, failureMessage: failureMessage, timeToWait: secondsToWait) + guard success else { + log(message: "command timeout") + preconditionFailure() + } + + if let _returnValue = returnValue { + return _returnValue + } else { + return "" + } + } + + static func waitWithPolling(_ expression: @autoclosure @escaping () throws -> T, toEventually predicate: @escaping (T) -> Bool, timeout: Int, pollingInterval: DispatchTimeInterval = .milliseconds(4)) -> DispatchTimeoutResult { + func memoizedClosure(_ closure: @escaping () throws -> T) -> (Bool) throws -> T { + var cache: T? + return { withoutCaching in + if withoutCaching || cache == nil { + cache = try closure() + } + guard let cache = cache else { + preconditionFailure() + } + + return cache + } + } + + let runLoop = RunLoop.current + let timeoutDate = Date(timeInterval: TimeInterval(timeout), since: Date()) + var fulfilled = false + let _expression = memoizedClosure(expression) + repeat { + do { + let exp = try _expression(true) + fulfilled = predicate(exp) + } catch { + fatalError("Error raised \(error.localizedDescription)") + } + if !fulfilled { + runLoop.run(until: Date(timeIntervalSinceNow: pollingInterval.timeInterval)) + } else { + break + } + } while Date().compare(timeoutDate) == .orderedAscending + + if fulfilled { + return .success + } else { + return .timedOut + } + } +} + +// Handle threading stuff +extension Runner { + func startSocketThread(port: UInt32) { + let secondsToWait = DispatchTimeInterval.seconds(SocketClient.connectTimeoutSeconds) + + dispatchGroup.enter() + + socketClient = SocketClient(port: port, commandTimeoutSeconds: timeout, socketDelegate: self) + thread = Thread(target: self, selector: #selector(startSocketComs), object: nil) + guard let thread = thread else { + preconditionFailure("Thread did not instantiate correctly") + } + + thread.name = "socket thread" + thread.start() + + let connectTimeout = DispatchTime.now() + secondsToWait + let timeoutResult = dispatchGroup.wait(timeout: connectTimeout) + + let failureMessage = "couldn't start socket thread in: \(SocketClient.connectTimeoutSeconds) seconds" + let success = testDispatchTimeoutResult(timeoutResult, failureMessage: failureMessage, timeToWait: secondsToWait) + guard success else { + log(message: "socket thread timeout") + preconditionFailure() + } + } + + func disconnectFromFastlaneProcess() { + shouldLeaveDispatchGroupDuringDisconnect = true + dispatchGroup.enter() + socketClient.sendComplete() + + let connectTimeout = DispatchTime.now() + 2 + _ = dispatchGroup.wait(timeout: connectTimeout) + } + + @objc func startSocketComs() { + guard let socketClient = socketClient else { + return + } + + socketClient.connectAndOpenStreams() + dispatchGroup.leave() + } + + private func testDispatchTimeoutResult(_ timeoutResult: DispatchTimeoutResult, failureMessage: String, timeToWait _: DispatchTimeInterval) -> Bool { + switch timeoutResult { + case .success: + return true + case .timedOut: + log(message: "timeout: \(failureMessage)") + return false + } + } +} + +extension Runner: SocketClientDelegateProtocol { + func commandExecuted(serverResponse: SocketClientResponse, completion: (SocketClient) -> Void) { + switch serverResponse { + case let .success(returnedObject, closureArgumentValue): + verbose(message: "command executed") + returnValue = returnedObject + if let command = currentlyExecutingCommand as? RubyCommand { + if let closureArgumentValue = closureArgumentValue, !closureArgumentValue.isEmpty { + command.performCallback(callbackArg: closureArgumentValue, socket: socketClient) { + self.executeNext[command.id] = true + } + } else { + executeNext[command.id] = true + } + } + dispatchGroup.leave() + completion(socketClient) + case .clientInitiatedCancelAcknowledged: + verbose(message: "server acknowledged a cancel request") + dispatchGroup.leave() + if let command = currentlyExecutingCommand as? RubyCommand { + executeNext[command.id] = true + } + completion(socketClient) + case .alreadyClosedSockets, .connectionFailure, .malformedRequest, .malformedResponse, .serverError: + log(message: "error encountered while executing command:\n\(serverResponse)") + dispatchGroup.leave() + if let command = currentlyExecutingCommand as? RubyCommand { + executeNext[command.id] = true + } + completion(socketClient) + case let .commandTimeout(timeout): + log(message: "Runner timed out after \(timeout) second(s)") + } + } + + func connectionsOpened() { + DispatchQueue.main.async { + verbose(message: "connected!") + } + } + + func connectionsClosed() { + DispatchQueue.main.async { + if let thread = self.thread { + thread.cancel() + } + self.thread = nil + self.socketClient.closeSession() + self.socketClient = nil + verbose(message: "connection closed!") + if self.shouldLeaveDispatchGroupDuringDisconnect { + self.dispatchGroup.leave() + } + exit(0) + } + } +} + +class Logger { + enum LogMode { + init(logMode: String) { + switch logMode { + case "normal", "default": + self = .normal + case "verbose": + self = .verbose + default: + logger.log(message: "unrecognized log mode: \(logMode), defaulting to 'normal'") + self = .normal + } + } + + case normal + case verbose + } + + public static var logMode: LogMode = .normal + + func log(message: String) { + let timestamp = NSDate().timeIntervalSince1970 + print("[\(timestamp)]: \(message)") + } + + func verbose(message: String) { + if Logger.logMode == .verbose { + let timestamp = NSDate().timeIntervalSince1970 + print("[\(timestamp)]: \(message)") + } + } +} + +func log(message: String) { + logger.log(message: message) +} + +func verbose(message: String) { + logger.verbose(message: message) +} + +private extension DispatchTimeInterval { + var timeInterval: TimeInterval { + var result: TimeInterval = 0 + switch self { + case let .seconds(value): + result = TimeInterval(value) + case let .milliseconds(value): + result = TimeInterval(value) * 0.001 + case let .microseconds(value): + result = TimeInterval(value) * 0.000_001 + case let .nanoseconds(value): + result = TimeInterval(value) * 0.000_000_001 + case .never: + fatalError() + @unknown default: + fatalError() + } + return result + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/RunnerArgument.swift b/fastlane/swift/RunnerArgument.swift new file mode 100644 index 00000000..c0ec84af --- /dev/null +++ b/fastlane/swift/RunnerArgument.swift @@ -0,0 +1,20 @@ +// RunnerArgument.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +struct RunnerArgument { + let name: String + let value: String +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/Scanfile.swift b/fastlane/swift/Scanfile.swift new file mode 100644 index 00000000..ce820ebe --- /dev/null +++ b/fastlane/swift/Scanfile.swift @@ -0,0 +1,20 @@ +// Scanfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `scan` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Scanfile: ScanfileProtocol { + // If you want to enable `scan`, run `fastlane scan init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/ScanfileProtocol.swift b/fastlane/swift/ScanfileProtocol.swift new file mode 100644 index 00000000..001081fc --- /dev/null +++ b/fastlane/swift/ScanfileProtocol.swift @@ -0,0 +1,315 @@ +// ScanfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol ScanfileProtocol: AnyObject { + /// Path to the workspace file + var workspace: String? { get } + + /// Path to the project file + var project: String? { get } + + /// Path to the Swift Package + var packagePath: String? { get } + + /// The project's scheme. Make sure it's marked as `Shared` + var scheme: String? { get } + + /// The name of the simulator type you want to run tests on (e.g. 'iPhone 6' or 'iPhone SE (2nd generation) (14.5)') + var device: String? { get } + + /// Array of devices to run the tests on (e.g. ['iPhone 6', 'iPad Air', 'iPhone SE (2nd generation) (14.5)']) + var devices: [String]? { get } + + /// Should skip auto detecting of devices if none were specified + var skipDetectDevices: Bool { get } + + /// Should fail if devices not found + var ensureDevicesFound: Bool { get } + + /// Enabling this option will automatically killall Simulator processes before the run + var forceQuitSimulator: Bool { get } + + /// Enabling this option will automatically erase the simulator before running the application + var resetSimulator: Bool { get } + + /// Enabling this option will disable the simulator from showing the 'Slide to type' prompt + var disableSlideToType: Bool { get } + + /// Enabling this option will launch the first simulator prior to calling any xcodebuild command + var prelaunchSimulator: Bool? { get } + + /// Enabling this option will automatically uninstall the application before running it + var reinstallApp: Bool { get } + + /// The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + var appIdentifier: String? { get } + + /// Array of strings matching Test Bundle/Test Suite/Test Cases to run + var onlyTesting: String? { get } + + /// Array of strings matching Test Bundle/Test Suite/Test Cases to skip + var skipTesting: String? { get } + + /// The testplan associated with the scheme that should be used for testing + var testplan: String? { get } + + /// Array of strings matching test plan configurations to run + var onlyTestConfigurations: String? { get } + + /// Array of strings matching test plan configurations to skip + var skipTestConfigurations: String? { get } + + /// Run tests using the provided `.xctestrun` file + var xctestrun: String? { get } + + /// The toolchain that should be used for building the application (e.g. `com.apple.dt.toolchain.Swift_2_3, org.swift.30p620160816a`) + var toolchain: String? { get } + + /// Should the project be cleaned before building it? + var clean: Bool { get } + + /// Should code coverage be generated? (Xcode 7 and up) + var codeCoverage: Bool? { get } + + /// Should the address sanitizer be turned on? + var addressSanitizer: Bool? { get } + + /// Should the thread sanitizer be turned on? + var threadSanitizer: Bool? { get } + + /// Should the HTML report be opened when tests are completed? + var openReport: Bool { get } + + /// The directory in which all reports will be stored + var outputDirectory: String { get } + + /// Define how the output should look like. Valid values are: standard, basic, rspec, or raw (disables xcpretty during xcodebuild) + var outputStyle: String? { get } + + /// Comma separated list of the output types (e.g. html, junit, json-compilation-database) + var outputTypes: String { get } + + /// Comma separated list of the output files, corresponding to the types provided by :output_types (order should match). If specifying an output type of json-compilation-database with :use_clang_report_name enabled, that option will take precedence + var outputFiles: String? { get } + + /// The directory where to store the raw log + var buildlogPath: String { get } + + /// If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + var includeSimulatorLogs: Bool { get } + + /// Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + var suppressXcodeOutput: Bool? { get } + + /// xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + var xcodebuildFormatter: String { get } + + /// Remove retry attempts from test results table and the JUnit report (if not using xcpretty) + var outputRemoveRetryAttempts: Bool { get } + + /// **DEPRECATED!** Use `output_style: 'raw'` instead - Disable xcpretty formatting of build, similar to `output_style='raw'` but this will also skip the test results table + var disableXcpretty: Bool? { get } + + /// **DEPRECATED!** Use 'xcpretty_formatter' instead - A custom xcpretty formatter to use + var formatter: String? { get } + + /// A custom xcpretty formatter to use + var xcprettyFormatter: String? { get } + + /// Pass in xcpretty additional command line arguments (e.g. '--test --no-color' or '--tap --no-utf') + var xcprettyArgs: String? { get } + + /// The directory where build products and other derived data will go + var derivedDataPath: String? { get } + + /// Should zip the derived data build products and place in output path? + var shouldZipBuildProducts: Bool { get } + + /// Should provide additional copy of .xctestrun file (settings.xctestrun) and place in output path? + var outputXctestrun: Bool { get } + + /// Should an Xcode result bundle be generated in the output directory + var resultBundle: Bool { get } + + /// Generate the json compilation database with clang naming convention (compile_commands.json) + var useClangReportName: Bool { get } + + /// Optionally override the per-target setting in the scheme for running tests in parallel. Equivalent to -parallel-testing-enabled + var parallelTesting: Bool? { get } + + /// Specify the exact number of test runners that will be spawned during parallel testing. Equivalent to -parallel-testing-worker-count + var concurrentWorkers: Int? { get } + + /// Constrain the number of simulator devices on which to test concurrently. Equivalent to -maximum-concurrent-test-simulator-destinations + var maxConcurrentSimulators: Int? { get } + + /// Do not run test bundles in parallel on the specified destinations. Testing will occur on each destination serially. Equivalent to -disable-concurrent-testing + var disableConcurrentTesting: Bool { get } + + /// Should debug build be skipped before test build? + var skipBuild: Bool { get } + + /// Test without building, requires a derived data path + var testWithoutBuilding: Bool? { get } + + /// Build for testing only, does not run tests + var buildForTesting: Bool? { get } + + /// The SDK that should be used for building the application + var sdk: String? { get } + + /// The configuration to use when building the app. Defaults to 'Release' + var configuration: String? { get } + + /// Pass additional arguments to xcodebuild. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + var xcargs: String? { get } + + /// Use an extra XCCONFIG file to build your app + var xcconfig: String? { get } + + /// App name to use in slack message and logfile name + var appName: String? { get } + + /// Target version of the app being build or tested. Used to filter out simulator version + var deploymentTargetVersion: String? { get } + + /// Create an Incoming WebHook for your Slack group to post results there + var slackUrl: String? { get } + + /// #channel or @username + var slackChannel: String? { get } + + /// The message included with each message posted to slack + var slackMessage: String? { get } + + /// Use webhook's default username and icon settings? (true/false) + var slackUseWebhookConfiguredUsernameAndIcon: Bool { get } + + /// Overrides the webhook's username property if slack_use_webhook_configured_username_and_icon is false + var slackUsername: String { get } + + /// Overrides the webhook's image property if slack_use_webhook_configured_username_and_icon is false + var slackIconUrl: String { get } + + /// Don't publish to slack, even when an URL is given + var skipSlack: Bool { get } + + /// Only post on Slack if the tests fail + var slackOnlyOnFailure: Bool { get } + + /// Specifies default payloads to include in Slack messages. For more info visit https://docs.fastlane.tools/actions/slack + var slackDefaultPayloads: [String]? { get } + + /// Use only if you're a pro, use the other options instead + var destination: String? { get } + + /// Platform to build when using a Catalyst enabled app. Valid values are: ios, macos + var catalystPlatform: String? { get } + + /// **DEPRECATED!** Use `--output_files` instead - Sets custom full report file name when generating a single report + var customReportFileName: String? { get } + + /// Allows for override of the default `xcodebuild` command + var xcodebuildCommand: String { get } + + /// Sets a custom path for Swift Package Manager dependencies + var clonedSourcePackagesPath: String? { get } + + /// Skips resolution of Swift Package Manager dependencies + var skipPackageDependenciesResolution: Bool { get } + + /// Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + var disablePackageAutomaticUpdates: Bool { get } + + /// Lets xcodebuild use system's scm configuration + var useSystemScm: Bool { get } + + /// The number of times a test can fail + var numberOfRetries: Int { get } + + /// Should this step stop the build if the tests fail? Set this to false if you're using trainer + var failBuild: Bool { get } +} + +public extension ScanfileProtocol { + var workspace: String? { return nil } + var project: String? { return nil } + var packagePath: String? { return nil } + var scheme: String? { return nil } + var device: String? { return nil } + var devices: [String]? { return nil } + var skipDetectDevices: Bool { return false } + var ensureDevicesFound: Bool { return false } + var forceQuitSimulator: Bool { return false } + var resetSimulator: Bool { return false } + var disableSlideToType: Bool { return true } + var prelaunchSimulator: Bool? { return nil } + var reinstallApp: Bool { return false } + var appIdentifier: String? { return nil } + var onlyTesting: String? { return nil } + var skipTesting: String? { return nil } + var testplan: String? { return nil } + var onlyTestConfigurations: String? { return nil } + var skipTestConfigurations: String? { return nil } + var xctestrun: String? { return nil } + var toolchain: String? { return nil } + var clean: Bool { return false } + var codeCoverage: Bool? { return nil } + var addressSanitizer: Bool? { return nil } + var threadSanitizer: Bool? { return nil } + var openReport: Bool { return false } + var outputDirectory: String { return "./test_output" } + var outputStyle: String? { return nil } + var outputTypes: String { return "html,junit" } + var outputFiles: String? { return nil } + var buildlogPath: String { return "~/Library/Logs/scan" } + var includeSimulatorLogs: Bool { return false } + var suppressXcodeOutput: Bool? { return nil } + var xcodebuildFormatter: String { return "xcbeautify" } + var outputRemoveRetryAttempts: Bool { return false } + var disableXcpretty: Bool? { return nil } + var formatter: String? { return nil } + var xcprettyFormatter: String? { return nil } + var xcprettyArgs: String? { return nil } + var derivedDataPath: String? { return nil } + var shouldZipBuildProducts: Bool { return false } + var outputXctestrun: Bool { return false } + var resultBundle: Bool { return false } + var useClangReportName: Bool { return false } + var parallelTesting: Bool? { return nil } + var concurrentWorkers: Int? { return nil } + var maxConcurrentSimulators: Int? { return nil } + var disableConcurrentTesting: Bool { return false } + var skipBuild: Bool { return false } + var testWithoutBuilding: Bool? { return nil } + var buildForTesting: Bool? { return nil } + var sdk: String? { return nil } + var configuration: String? { return nil } + var xcargs: String? { return nil } + var xcconfig: String? { return nil } + var appName: String? { return nil } + var deploymentTargetVersion: String? { return nil } + var slackUrl: String? { return nil } + var slackChannel: String? { return nil } + var slackMessage: String? { return nil } + var slackUseWebhookConfiguredUsernameAndIcon: Bool { return false } + var slackUsername: String { return "fastlane" } + var slackIconUrl: String { return "https://fastlane.tools/assets/img/fastlane_icon.png" } + var skipSlack: Bool { return false } + var slackOnlyOnFailure: Bool { return false } + var slackDefaultPayloads: [String]? { return nil } + var destination: String? { return nil } + var catalystPlatform: String? { return nil } + var customReportFileName: String? { return nil } + var xcodebuildCommand: String { return "env NSUnbufferedIO=YES xcodebuild" } + var clonedSourcePackagesPath: String? { return nil } + var skipPackageDependenciesResolution: Bool { return false } + var disablePackageAutomaticUpdates: Bool { return false } + var useSystemScm: Bool { return false } + var numberOfRetries: Int { return 0 } + var failBuild: Bool { return true } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.120] diff --git a/fastlane/swift/Screengrabfile.swift b/fastlane/swift/Screengrabfile.swift new file mode 100644 index 00000000..f96b3985 --- /dev/null +++ b/fastlane/swift/Screengrabfile.swift @@ -0,0 +1,20 @@ +// Screengrabfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `screengrab` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Screengrabfile: ScreengrabfileProtocol { + // If you want to enable `screengrab`, run `fastlane screengrab init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/ScreengrabfileProtocol.swift b/fastlane/swift/ScreengrabfileProtocol.swift new file mode 100644 index 00000000..3a09f5ff --- /dev/null +++ b/fastlane/swift/ScreengrabfileProtocol.swift @@ -0,0 +1,99 @@ +// ScreengrabfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol ScreengrabfileProtocol: AnyObject { + /// Path to the root of your Android SDK installation, e.g. ~/tools/android-sdk-macosx + var androidHome: String? { get } + + /// **DEPRECATED!** The Android build tools version to use, e.g. '23.0.2' + var buildToolsVersion: String? { get } + + /// A list of locales which should be used + var locales: [String] { get } + + /// Enabling this option will automatically clear previously generated screenshots before running screengrab + var clearPreviousScreenshots: Bool { get } + + /// The directory where to store the screenshots + var outputDirectory: String { get } + + /// Don't open the summary after running _screengrab_ + var skipOpenSummary: Bool { get } + + /// The package name of the app under test (e.g. com.yourcompany.yourapp) + var appPackageName: String { get } + + /// The package name of the tests bundle (e.g. com.yourcompany.yourapp.test) + var testsPackageName: String? { get } + + /// Only run tests in these Java packages + var useTestsInPackages: [String]? { get } + + /// Only run tests in these Java classes + var useTestsInClasses: [String]? { get } + + /// Additional launch arguments + var launchArguments: [String]? { get } + + /// The fully qualified class name of your test instrumentation runner + var testInstrumentationRunner: String { get } + + /// **DEPRECATED!** Return the device to this locale after running tests + var endingLocale: String { get } + + /// **DEPRECATED!** Restarts the adb daemon using `adb root` to allow access to screenshots directories on device. Use if getting 'Permission denied' errors + var useAdbRoot: Bool { get } + + /// The path to the APK for the app under test + var appApkPath: String? { get } + + /// The path to the APK for the tests bundle + var testsApkPath: String? { get } + + /// Use the device or emulator with the given serial number or qualifier + var specificDevice: String? { get } + + /// Type of device used for screenshots. Matches Google Play Types (phone, sevenInch, tenInch, tv, wear) + var deviceType: String { get } + + /// Whether or not to exit Screengrab on test failure. Exiting on failure will not copy screenshots to local machine nor open screenshots summary + var exitOnTestFailure: Bool { get } + + /// Enabling this option will automatically uninstall the application before running it + var reinstallApp: Bool { get } + + /// Add timestamp suffix to screenshot filename + var useTimestampSuffix: Bool { get } + + /// Configure the host used by adb to connect, allows running on remote devices farm + var adbHost: String? { get } +} + +public extension ScreengrabfileProtocol { + var androidHome: String? { return nil } + var buildToolsVersion: String? { return nil } + var locales: [String] { return ["en-US"] } + var clearPreviousScreenshots: Bool { return false } + var outputDirectory: String { return "fastlane/metadata/android" } + var skipOpenSummary: Bool { return false } + var appPackageName: String { return "" } + var testsPackageName: String? { return nil } + var useTestsInPackages: [String]? { return nil } + var useTestsInClasses: [String]? { return nil } + var launchArguments: [String]? { return nil } + var testInstrumentationRunner: String { return "androidx.test.runner.AndroidJUnitRunner" } + var endingLocale: String { return "en-US" } + var useAdbRoot: Bool { return false } + var appApkPath: String? { return nil } + var testsApkPath: String? { return nil } + var specificDevice: String? { return nil } + var deviceType: String { return "phone" } + var exitOnTestFailure: Bool { return true } + var reinstallApp: Bool { return false } + var useTimestampSuffix: Bool { return true } + var adbHost: String? { return nil } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.110] diff --git a/fastlane/swift/Snapshotfile.swift b/fastlane/swift/Snapshotfile.swift new file mode 100644 index 00000000..1ef376c2 --- /dev/null +++ b/fastlane/swift/Snapshotfile.swift @@ -0,0 +1,20 @@ +// Snapshotfile.swift +// Copyright (c) 2022 FastlaneTools + +// This class is automatically included in FastlaneRunner during build + +// This autogenerated file will be overwritten or replaced during build time, or when you initialize `snapshot` +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +public class Snapshotfile: SnapshotfileProtocol { + // If you want to enable `snapshot`, run `fastlane snapshot init` + // After, this file will be replaced with a custom implementation that contains values you supplied + // during the `init` process, and you won't see this message +} + +// Generated with fastlane 2.210.0 diff --git a/fastlane/swift/SnapshotfileProtocol.swift b/fastlane/swift/SnapshotfileProtocol.swift new file mode 100644 index 00000000..b3704658 --- /dev/null +++ b/fastlane/swift/SnapshotfileProtocol.swift @@ -0,0 +1,207 @@ +// SnapshotfileProtocol.swift +// Copyright (c) 2022 FastlaneTools + +public protocol SnapshotfileProtocol: AnyObject { + /// Path the workspace file + var workspace: String? { get } + + /// Path the project file + var project: String? { get } + + /// Pass additional arguments to xcodebuild for the test phase. Be sure to quote the setting names and values e.g. OTHER_LDFLAGS="-ObjC -lstdc++" + var xcargs: String? { get } + + /// Use an extra XCCONFIG file to build your app + var xcconfig: String? { get } + + /// A list of devices you want to take the screenshots from + var devices: [String]? { get } + + /// A list of languages which should be used + var languages: [String] { get } + + /// A list of launch arguments which should be used + var launchArguments: [String] { get } + + /// The directory where to store the screenshots + var outputDirectory: String { get } + + /// If the logs generated by the app (e.g. using NSLog, perror, etc.) in the Simulator should be written to the output_directory + var outputSimulatorLogs: Bool { get } + + /// By default, the latest version should be used automatically. If you want to change it, do it here + var iosVersion: String? { get } + + /// Don't open the HTML summary after running _snapshot_ + var skipOpenSummary: Bool { get } + + /// Do not check for most recent SnapshotHelper code + var skipHelperVersionCheck: Bool { get } + + /// Enabling this option will automatically clear previously generated screenshots before running snapshot + var clearPreviousScreenshots: Bool { get } + + /// Enabling this option will automatically uninstall the application before running it + var reinstallApp: Bool { get } + + /// Enabling this option will automatically erase the simulator before running the application + var eraseSimulator: Bool { get } + + /// Enabling this option will prevent displaying the simulator window + var headless: Bool { get } + + /// Enabling this option will automatically override the status bar to show 9:41 AM, full battery, and full reception (Adjust 'SNAPSHOT_SIMULATOR_WAIT_FOR_BOOT_TIMEOUT' environment variable if override status bar is not working. Might be because simulator is not fully booted. Defaults to 10 seconds) + var overrideStatusBar: Bool { get } + + /// Fully customize the status bar by setting each option here. Requires `override_status_bar` to be set to `true`. See `xcrun simctl status_bar --help` + var overrideStatusBarArguments: String? { get } + + /// Enabling this option will configure the Simulator's system language + var localizeSimulator: Bool { get } + + /// Enabling this option will configure the Simulator to be in dark mode (false for light, true for dark) + var darkMode: Bool? { get } + + /// The bundle identifier of the app to uninstall (only needed when enabling reinstall_app) + var appIdentifier: String? { get } + + /// A list of photos that should be added to the simulator before running the application + var addPhotos: [String]? { get } + + /// A list of videos that should be added to the simulator before running the application + var addVideos: [String]? { get } + + /// A path to screenshots.html template + var htmlTemplate: String? { get } + + /// The directory where to store the build log + var buildlogPath: String { get } + + /// Should the project be cleaned before building it? + var clean: Bool { get } + + /// Test without building, requires a derived data path + var testWithoutBuilding: Bool? { get } + + /// The configuration to use when building the app. Defaults to 'Release' + var configuration: String? { get } + + /// The SDK that should be used for building the application + var sdk: String? { get } + + /// The scheme you want to use, this must be the scheme for the UI Tests + var scheme: String? { get } + + /// The number of times a test can fail before snapshot should stop retrying + var numberOfRetries: Int { get } + + /// Should snapshot stop immediately after the tests completely failed on one device? + var stopAfterFirstError: Bool { get } + + /// The directory where build products and other derived data will go + var derivedDataPath: String? { get } + + /// Should an Xcode result bundle be generated in the output directory + var resultBundle: Bool { get } + + /// The name of the target you want to test (if you desire to override the Target Application from Xcode) + var testTargetName: String? { get } + + /// Separate the log files per device and per language + var namespaceLogFiles: String? { get } + + /// Take snapshots on multiple simulators concurrently. Note: This option is only applicable when running against Xcode 9 + var concurrentSimulators: Bool { get } + + /// Disable the simulator from showing the 'Slide to type' prompt + var disableSlideToType: Bool { get } + + /// Sets a custom path for Swift Package Manager dependencies + var clonedSourcePackagesPath: String? { get } + + /// Skips resolution of Swift Package Manager dependencies + var skipPackageDependenciesResolution: Bool { get } + + /// Prevents packages from automatically being resolved to versions other than those recorded in the `Package.resolved` file + var disablePackageAutomaticUpdates: Bool { get } + + /// The testplan associated with the scheme that should be used for testing + var testplan: String? { get } + + /// Array of strings matching Test Bundle/Test Suite/Test Cases to run + var onlyTesting: String? { get } + + /// Array of strings matching Test Bundle/Test Suite/Test Cases to skip + var skipTesting: String? { get } + + /// xcodebuild formatter to use (ex: 'xcbeautify', 'xcbeautify --quieter', 'xcpretty', 'xcpretty -test'). Use empty string (ex: '') to disable any formatter (More information: https://docs.fastlane.tools/best-practices/xcodebuild-formatters/) + var xcodebuildFormatter: String { get } + + /// **DEPRECATED!** Use `xcodebuild_formatter: ''` instead - Additional xcpretty arguments + var xcprettyArgs: String? { get } + + /// Disable xcpretty formatting of build + var disableXcpretty: Bool? { get } + + /// Suppress the output of xcodebuild to stdout. Output is still saved in buildlog_path + var suppressXcodeOutput: Bool? { get } + + /// Lets xcodebuild use system's scm configuration + var useSystemScm: Bool { get } +} + +public extension SnapshotfileProtocol { + var workspace: String? { return nil } + var project: String? { return nil } + var xcargs: String? { return nil } + var xcconfig: String? { return nil } + var devices: [String]? { return nil } + var languages: [String] { return ["en-US"] } + var launchArguments: [String] { return [""] } + var outputDirectory: String { return "screenshots" } + var outputSimulatorLogs: Bool { return false } + var iosVersion: String? { return nil } + var skipOpenSummary: Bool { return false } + var skipHelperVersionCheck: Bool { return false } + var clearPreviousScreenshots: Bool { return false } + var reinstallApp: Bool { return false } + var eraseSimulator: Bool { return false } + var headless: Bool { return true } + var overrideStatusBar: Bool { return false } + var overrideStatusBarArguments: String? { return nil } + var localizeSimulator: Bool { return false } + var darkMode: Bool? { return nil } + var appIdentifier: String? { return nil } + var addPhotos: [String]? { return nil } + var addVideos: [String]? { return nil } + var htmlTemplate: String? { return nil } + var buildlogPath: String { return "~/Library/Logs/snapshot" } + var clean: Bool { return false } + var testWithoutBuilding: Bool? { return nil } + var configuration: String? { return nil } + var sdk: String? { return nil } + var scheme: String? { return nil } + var numberOfRetries: Int { return 1 } + var stopAfterFirstError: Bool { return false } + var derivedDataPath: String? { return nil } + var resultBundle: Bool { return false } + var testTargetName: String? { return nil } + var namespaceLogFiles: String? { return nil } + var concurrentSimulators: Bool { return true } + var disableSlideToType: Bool { return false } + var clonedSourcePackagesPath: String? { return nil } + var skipPackageDependenciesResolution: Bool { return false } + var disablePackageAutomaticUpdates: Bool { return false } + var testplan: String? { return nil } + var onlyTesting: String? { return nil } + var skipTesting: String? { return nil } + var xcodebuildFormatter: String { return "xcbeautify" } + var xcprettyArgs: String? { return nil } + var disableXcpretty: Bool? { return nil } + var suppressXcodeOutput: Bool? { return nil } + var useSystemScm: Bool { return false } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.104] diff --git a/fastlane/swift/SocketClient.swift b/fastlane/swift/SocketClient.swift new file mode 100644 index 00000000..ef8763f4 --- /dev/null +++ b/fastlane/swift/SocketClient.swift @@ -0,0 +1,332 @@ +// SocketClient.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Dispatch +import Foundation + +public enum SocketClientResponse: Error { + case alreadyClosedSockets + case malformedRequest + case malformedResponse + case serverError + case clientInitiatedCancelAcknowledged + case commandTimeout(seconds: Int) + case connectionFailure + case success(returnedObject: String?, closureArgumentValue: String?) +} + +class SocketClient: NSObject { + enum SocketStatus { + case ready + case closed + } + + static let connectTimeoutSeconds = 2 + static let defaultCommandTimeoutSeconds = 10800 // 3 hours + static let doneToken = "done" // TODO: remove these + static let cancelToken = "cancelFastlaneRun" + + fileprivate var inputStream: InputStream! + fileprivate var outputStream: OutputStream! + fileprivate var cleaningUpAfterDone = false + fileprivate let dispatchGroup = DispatchGroup() + fileprivate let readSemaphore = DispatchSemaphore(value: 1) + fileprivate let writeSemaphore = DispatchSemaphore(value: 1) + fileprivate let commandTimeoutSeconds: Int + + private let writeQueue: DispatchQueue + private let readQueue: DispatchQueue + private let streamQueue: DispatchQueue + private let host: String + private let port: UInt32 + + let maxReadLength = 65536 // max for ipc on 10.12 is kern.ipc.maxsockbuf: 8388608 ($sysctl kern.ipc.maxsockbuf) + + private(set) weak var socketDelegate: SocketClientDelegateProtocol? + + public private(set) var socketStatus: SocketStatus + + // localhost only, this prevents other computers from connecting + init(host: String = "localhost", port: UInt32 = 2000, commandTimeoutSeconds: Int = defaultCommandTimeoutSeconds, socketDelegate: SocketClientDelegateProtocol) { + self.host = host + self.port = port + self.commandTimeoutSeconds = commandTimeoutSeconds + readQueue = DispatchQueue(label: "readQueue", qos: .background, attributes: .concurrent) + writeQueue = DispatchQueue(label: "writeQueue", qos: .background, attributes: .concurrent) + streamQueue = DispatchQueue.global(qos: .background) + socketStatus = .closed + self.socketDelegate = socketDelegate + super.init() + } + + func connectAndOpenStreams() { + var readStream: Unmanaged? + var writeStream: Unmanaged? + + streamQueue.sync { + CFStreamCreatePairWithSocketToHost(kCFAllocatorDefault, self.host as CFString, self.port, &readStream, &writeStream) + + self.inputStream = readStream!.takeRetainedValue() + self.outputStream = writeStream!.takeRetainedValue() + + self.inputStream.delegate = self + self.outputStream.delegate = self + + self.inputStream.schedule(in: .main, forMode: .defaultRunLoopMode) + self.outputStream.schedule(in: .main, forMode: .defaultRunLoopMode) + } + + dispatchGroup.enter() + readQueue.sync { + self.inputStream.open() + } + + dispatchGroup.enter() + writeQueue.sync { + self.outputStream.open() + } + + let secondsToWait = DispatchTimeInterval.seconds(SocketClient.connectTimeoutSeconds) + let connectTimeout = DispatchTime.now() + secondsToWait + + let timeoutResult = dispatchGroup.wait(timeout: connectTimeout) + let failureMessage = "Couldn't connect to ruby process within: \(SocketClient.connectTimeoutSeconds) seconds" + + let success = testDispatchTimeoutResult(timeoutResult, failureMessage: failureMessage, timeToWait: secondsToWait) + + guard success else { + socketDelegate?.commandExecuted(serverResponse: .connectionFailure) { _ in } + return + } + + socketStatus = .ready + socketDelegate?.connectionsOpened() + } + + public func send(rubyCommand: RubyCommandable) { + verbose(message: "sending: \(rubyCommand.json)") + send(string: rubyCommand.json) + writeSemaphore.signal() + } + + public func sendComplete() { + closeSession(sendAbort: true) + } + + private func testDispatchTimeoutResult(_ timeoutResult: DispatchTimeoutResult, failureMessage: String, timeToWait: DispatchTimeInterval) -> Bool { + switch timeoutResult { + case .success: + return true + case .timedOut: + log(message: "Timeout: \(failureMessage)") + + if case let .seconds(seconds) = timeToWait { + socketDelegate?.commandExecuted(serverResponse: .commandTimeout(seconds: seconds)) { _ in } + } + return false + } + } + + private func stopInputSession() { + inputStream.close() + } + + private func stopOutputSession() { + outputStream.close() + } + + private func sendThroughQueue(string: String) { + let data = string.data(using: .utf8)! + data.withUnsafeBytes { (buffer: UnsafeRawBufferPointer) in + if let buffer = buffer.baseAddress { + self.outputStream.write(buffer.assumingMemoryBound(to: UInt8.self), maxLength: data.count) + } + } + } + + private func privateSend(string: String) { + writeQueue.sync { + writeSemaphore.wait() + self.sendThroughQueue(string: string) + writeSemaphore.signal() + let timeoutSeconds = self.cleaningUpAfterDone ? 1 : self.commandTimeoutSeconds + let timeToWait = DispatchTimeInterval.seconds(timeoutSeconds) + let commandTimeout = DispatchTime.now() + timeToWait + let timeoutResult = writeSemaphore.wait(timeout: commandTimeout) + + _ = self.testDispatchTimeoutResult(timeoutResult, failureMessage: "Ruby process didn't return after: \(SocketClient.connectTimeoutSeconds) seconds", timeToWait: timeToWait) + } + } + + private func send(string: String) { + guard !cleaningUpAfterDone else { + // This will happen after we abort if there are commands waiting to be executed + // Need to check state of SocketClient in command runner to make sure we can accept `send` + socketDelegate?.commandExecuted(serverResponse: .alreadyClosedSockets) { _ in } + return + } + + if string == SocketClient.doneToken { + cleaningUpAfterDone = true + } + + privateSend(string: string) + } + + func closeSession(sendAbort: Bool = true) { + socketStatus = .closed + + stopInputSession() + + if sendAbort { + send(rubyCommand: ControlCommand(commandType: .done)) + } + + stopOutputSession() + socketDelegate?.connectionsClosed() + } + + public func enter() { + dispatchGroup.enter() + } + + public func leave() { + readSemaphore.signal() + writeSemaphore.signal() + } +} + +extension SocketClient: StreamDelegate { + func stream(_ aStream: Stream, handle eventCode: Stream.Event) { + guard !cleaningUpAfterDone else { + // Still getting response from server even though we are done. + // No big deal, we're closing the streams anyway. + // That being said, we need to balance out the dispatchGroups + dispatchGroup.leave() + return + } + + if aStream === inputStream { + switch eventCode { + case Stream.Event.openCompleted: + dispatchGroup.leave() + + case Stream.Event.errorOccurred: + verbose(message: "input stream error occurred") + closeSession(sendAbort: true) + + case Stream.Event.hasBytesAvailable: + read() + + case Stream.Event.endEncountered: + // nothing special here + break + + case Stream.Event.hasSpaceAvailable: + // we don't care about this + break + + default: + verbose(message: "input stream caused unrecognized event: \(eventCode)") + } + + } else if aStream === outputStream { + switch eventCode { + case Stream.Event.openCompleted: + dispatchGroup.leave() + + case Stream.Event.errorOccurred: + // probably safe to close all the things because Ruby already disconnected + verbose(message: "output stream received error") + + case Stream.Event.endEncountered: + // nothing special here + break + + case Stream.Event.hasSpaceAvailable: + // we don't care about this + break + + default: + verbose(message: "output stream caused unrecognized event: \(eventCode)") + } + } + } + + func read() { + readQueue.sync { + self.readSemaphore.wait() + var buffer = [UInt8](repeating: 0, count: maxReadLength) + var output = "" + while self.inputStream!.hasBytesAvailable { + let bytesRead: Int = inputStream!.read(&buffer, maxLength: buffer.count) + if bytesRead >= 0 { + guard let read = String(bytes: buffer[.. Void) +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/SocketResponse.swift b/fastlane/swift/SocketResponse.swift new file mode 100644 index 00000000..4171c00c --- /dev/null +++ b/fastlane/swift/SocketResponse.swift @@ -0,0 +1,84 @@ +// SocketResponse.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +struct SocketResponse { + enum ResponseType { + case parseFailure(failureInformation: [String]) + case failure(failureInformation: [String], failureClass: String?, failureMessage: String?) + case readyForNext(returnedObject: String?, closureArgumentValue: String?) + case clientInitiatedCancel + + init(statusDictionary: [String: Any]) { + guard let status = statusDictionary["status"] as? String else { + self = .parseFailure(failureInformation: ["Message failed to parse from Ruby server"]) + return + } + + if status == "ready_for_next" { + verbose(message: "ready for next") + let returnedObject = statusDictionary["return_object"] as? String + let closureArgumentValue = statusDictionary["closure_argument_value"] as? String + self = .readyForNext(returnedObject: returnedObject, closureArgumentValue: closureArgumentValue) + return + + } else if status == "cancelled" { + self = .clientInitiatedCancel + return + + } else if status == "failure" { + guard let failureInformation = statusDictionary["failure_information"] as? [String] else { + self = .parseFailure(failureInformation: ["Ruby server indicated failure but Swift couldn't receive it"]) + return + } + + let failureClass = statusDictionary["failure_class"] as? String + let failureMessage = statusDictionary["failure_message"] as? String + self = .failure(failureInformation: failureInformation, failureClass: failureClass, failureMessage: failureMessage) + return + } + self = .parseFailure(failureInformation: ["Message status: \(status) not a supported status"]) + } + } + + let responseType: ResponseType + + init(payload: String) { + guard let data = SocketResponse.convertToDictionary(text: payload) else { + responseType = .parseFailure(failureInformation: ["Unable to parse message from Ruby server"]) + return + } + + guard case let statusDictionary? = data["payload"] as? [String: Any] else { + responseType = .parseFailure(failureInformation: ["Payload missing from Ruby server response"]) + return + } + + responseType = ResponseType(statusDictionary: statusDictionary) + } +} + +extension SocketResponse { + static func convertToDictionary(text: String) -> [String: Any]? { + if let data = text.data(using: .utf8) { + do { + return try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any] + } catch { + log(message: error.localizedDescription) + } + } + return nil + } +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/formatting/Brewfile b/fastlane/swift/formatting/Brewfile new file mode 100644 index 00000000..742b1757 --- /dev/null +++ b/fastlane/swift/formatting/Brewfile @@ -0,0 +1 @@ +brew("swiftformat") diff --git a/fastlane/swift/formatting/Brewfile.lock.json b/fastlane/swift/formatting/Brewfile.lock.json new file mode 100644 index 00000000..d08b6877 --- /dev/null +++ b/fastlane/swift/formatting/Brewfile.lock.json @@ -0,0 +1,73 @@ +{ + "entries": { + "brew": { + "swiftformat": { + "version": "0.49.18", + "bottle": { + "rebuild": 0, + "root_url": "https://ghcr.io/v2/homebrew/core", + "files": { + "arm64_monterey": { + "cellar": ":any_skip_relocation", + "url": "https://ghcr.io/v2/homebrew/core/swiftformat/blobs/sha256:6362f6087bc3821f4271c3d17b3a4f180b1e1326646ddfb60f6d27bfb5a2a357", + "sha256": "6362f6087bc3821f4271c3d17b3a4f180b1e1326646ddfb60f6d27bfb5a2a357" + }, + "arm64_big_sur": { + "cellar": ":any_skip_relocation", + "url": "https://ghcr.io/v2/homebrew/core/swiftformat/blobs/sha256:e94cf1b66df0d712bbfbf509b98efaf31d39a61b82999314e1f3c0e45195c51a", + "sha256": "e94cf1b66df0d712bbfbf509b98efaf31d39a61b82999314e1f3c0e45195c51a" + }, + "monterey": { + "cellar": ":any_skip_relocation", + "url": "https://ghcr.io/v2/homebrew/core/swiftformat/blobs/sha256:456e0c95a565adbb45a29747abfadf41c838a7f09fae052a874e59429a94ef14", + "sha256": "456e0c95a565adbb45a29747abfadf41c838a7f09fae052a874e59429a94ef14" + }, + "big_sur": { + "cellar": ":any_skip_relocation", + "url": "https://ghcr.io/v2/homebrew/core/swiftformat/blobs/sha256:d00204be714789fa8b35d4c6f6eea5813604aa09f3911635059973aa827d2e8c", + "sha256": "d00204be714789fa8b35d4c6f6eea5813604aa09f3911635059973aa827d2e8c" + }, + "catalina": { + "cellar": ":any_skip_relocation", + "url": "https://ghcr.io/v2/homebrew/core/swiftformat/blobs/sha256:b07f7221f3c5225ad0037293cecb95bde4f0dba4fa19797d84a3376dd1ad02ea", + "sha256": "b07f7221f3c5225ad0037293cecb95bde4f0dba4fa19797d84a3376dd1ad02ea" + }, + "x86_64_linux": { + "cellar": "/home/linuxbrew/.linuxbrew/Cellar", + "url": "https://ghcr.io/v2/homebrew/core/swiftformat/blobs/sha256:c4a4ebd2f3f54b8f399551efaf47b3e419db2c729ffaf18a09e64bbf62d82f38", + "sha256": "c4a4ebd2f3f54b8f399551efaf47b3e419db2c729ffaf18a09e64bbf62d82f38" + } + } + } + } + } + }, + "system": { + "macos": { + "catalina": { + "HOMEBREW_VERSION": "3.2.0-77-gd305f72", + "HOMEBREW_PREFIX": "/usr/local", + "Homebrew/homebrew-core": "0b13b342053d414d1b241c2c7a446b74d79cc90e", + "CLT": "11.0.0.33.12", + "Xcode": "12.4", + "macOS": "10.15.7" + }, + "big_sur": { + "HOMEBREW_VERSION": "3.3.9-34-g2e92128", + "HOMEBREW_PREFIX": "/usr/local", + "Homebrew/homebrew-core": "c28163ed56d6e54f2f71ecf678d4b4d33bac23a5", + "CLT": "12.4.0.0.1.1610135815", + "Xcode": "12.5", + "macOS": "11.0.1" + }, + "monterey": { + "HOMEBREW_VERSION": "3.6.1-50-g6eaa510", + "HOMEBREW_PREFIX": "/opt/homebrew", + "Homebrew/homebrew-core": "4d6529affa1851ffb992a33fe5641b0cd739c895", + "CLT": "13.4.0.0.1.1651278267", + "Xcode": "13.4.1", + "macOS": "12.5" + } + } + } +} diff --git a/fastlane/swift/formatting/Rakefile b/fastlane/swift/formatting/Rakefile new file mode 100644 index 00000000..ad91133d --- /dev/null +++ b/fastlane/swift/formatting/Rakefile @@ -0,0 +1,18 @@ +# frozen_string_literal: true + +task(default: %w[setup]) + +task(setup: [:brew, :lint]) + +task(:brew) do + raise '`brew` is required. Please install brew. https://brew.sh/' unless system('which brew') + + puts('➡️ Brew') + sh('brew bundle') +end + +task(:lint) do + Dir.chdir('..') do + sh("swiftformat . --config formatting/.swiftformat --verbose --selfrequired waitWithPolling --exclude Fastfile.swift --swiftversion 4.0") + end +end diff --git a/fastlane/swift/main.swift b/fastlane/swift/main.swift new file mode 100644 index 00000000..df5c3283 --- /dev/null +++ b/fastlane/swift/main.swift @@ -0,0 +1,47 @@ +// main.swift +// Copyright (c) 2022 FastlaneTools + +// +// ** NOTE ** +// This file is provided by fastlane and WILL be overwritten in future updates +// If you want to add extra functionality to this project, create a new file in a +// new group so that it won't be marked for upgrade +// + +import Foundation + +let argumentProcessor = ArgumentProcessor(args: CommandLine.arguments) +let timeout = argumentProcessor.commandTimeout + +class MainProcess { + var doneRunningLane = false + var thread: Thread! + + @objc func connectToFastlaneAndRunLane() { + runner.startSocketThread(port: argumentProcessor.port) + + let completedRun = Fastfile.runLane(from: nil, named: argumentProcessor.currentLane, with: argumentProcessor.laneParameters()) + if completedRun { + runner.disconnectFromFastlaneProcess() + } + + doneRunningLane = true + } + + func startFastlaneThread() { + thread = Thread(target: self, selector: #selector(connectToFastlaneAndRunLane), object: nil) + thread.name = "worker thread" + thread.start() + } +} + +let process = MainProcess() +process.startFastlaneThread() + +while !process.doneRunningLane, RunLoop.current.run(mode: RunLoopMode.defaultRunLoopMode, before: Date(timeIntervalSinceNow: 2)) { + // no op +} + +// Please don't remove the lines below +// They are used to detect outdated files +// FastlaneRunnerAPIVersion [0.9.2] diff --git a/fastlane/swift/upgrade_manifest.json b/fastlane/swift/upgrade_manifest.json new file mode 100644 index 00000000..4cf4ea6c --- /dev/null +++ b/fastlane/swift/upgrade_manifest.json @@ -0,0 +1 @@ +{"Actions.swift":"Autogenerated API","Fastlane.swift":"Autogenerated API","DeliverfileProtocol.swift":"Autogenerated API","GymfileProtocol.swift":"Autogenerated API","MatchfileProtocol.swift":"Autogenerated API","Plugins.swift":"Autogenerated API","PrecheckfileProtocol.swift":"Autogenerated API","ScanfileProtocol.swift":"Autogenerated API","ScreengrabfileProtocol.swift":"Autogenerated API","SnapshotfileProtocol.swift":"Autogenerated API","LaneFileProtocol.swift":"Fastfile Components","OptionalConfigValue.swift":"Fastfile Components","Atomic.swift":"Networking","ControlCommand.swift":"Networking","RubyCommand.swift":"Networking","RubyCommandable.swift":"Networking","Runner.swift":"Networking","SocketClient.swift":"Networking","SocketClientDelegateProtocol.swift":"Networking","SocketResponse.swift":"Networking","main.swift":"Runner Code","ArgumentProcessor.swift":"Runner Code","RunnerArgument.swift":"Runner Code"} \ No newline at end of file diff --git a/set_up_test_testflight.sh b/set_up_test_testflight.sh index 8eb05b41..350baa71 100644 --- a/set_up_test_testflight.sh +++ b/set_up_test_testflight.sh @@ -1,7 +1,5 @@ echo "import('./Tests/Fastfile')" | cat - fastlane/Fastfile | tee fastlane/Fastfile &> /dev/null -bundle exec fastlane add_plugin appicon - readonly CONSTANT_API_KEY_ID="<#API_KEY_ID#>" readonly CONSTANT_ISSUER_ID="<#ISSUER_ID#>" readonly CONSTANT_MATCH_REPO="git@github.com:{organization}\/{repo}.git" diff --git a/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/1024.png b/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/1024.png new file mode 100644 index 00000000..d670474d Binary files /dev/null and b/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/1024.png differ diff --git a/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/Contents.json b/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/Contents.json old mode 100644 new mode 100755 index 9221b9bb..cff1680b --- a/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/Contents.json +++ b/{PROJECT_NAME}/Resources/Assets/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -1,93 +1,9 @@ { "images" : [ { - "idiom" : "iphone", - "scale" : "2x", - "size" : "20x20" - }, - { - "idiom" : "iphone", - "scale" : "3x", - "size" : "20x20" - }, - { - "idiom" : "iphone", - "scale" : "2x", - "size" : "29x29" - }, - { - "idiom" : "iphone", - "scale" : "3x", - "size" : "29x29" - }, - { - "idiom" : "iphone", - "scale" : "2x", - "size" : "40x40" - }, - { - "idiom" : "iphone", - "scale" : "3x", - "size" : "40x40" - }, - { - "idiom" : "iphone", - "scale" : "2x", - "size" : "60x60" - }, - { - "idiom" : "iphone", - "scale" : "3x", - "size" : "60x60" - }, - { - "idiom" : "ipad", - "scale" : "1x", - "size" : "20x20" - }, - { - "idiom" : "ipad", - "scale" : "2x", - "size" : "20x20" - }, - { - "idiom" : "ipad", - "scale" : "1x", - "size" : "29x29" - }, - { - "idiom" : "ipad", - "scale" : "2x", - "size" : "29x29" - }, - { - "idiom" : "ipad", - "scale" : "1x", - "size" : "40x40" - }, - { - "idiom" : "ipad", - "scale" : "2x", - "size" : "40x40" - }, - { - "idiom" : "ipad", - "scale" : "1x", - "size" : "76x76" - }, - { - "idiom" : "ipad", - "scale" : "2x", - "size" : "76x76" - }, - { - "idiom" : "ipad", - "scale" : "2x", - "size" : "83.5x83.5" - }, - { - "idiom" : "ios-marketing", - "scale" : "1x", + "filename" : "1024.png", + "idiom" : "universal", + "platform" : "ios", "size" : "1024x1024" } ],