fastlane_iac_publish 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.yardopts +1 -0
- data/LICENSE +21 -0
- data/README.md +195 -0
- data/bin/bin-proxy +19 -0
- data/bin/fastlane +23 -0
- data/cert/README.md +17 -0
- data/cert/lib/cert.rb +4 -0
- data/cert/lib/cert/commands_generator.rb +59 -0
- data/cert/lib/cert/module.rb +14 -0
- data/cert/lib/cert/options.rb +92 -0
- data/cert/lib/cert/runner.rb +203 -0
- data/credentials_manager/README.md +64 -0
- data/credentials_manager/lib/credentials_manager.rb +7 -0
- data/credentials_manager/lib/credentials_manager/account_manager.rb +157 -0
- data/credentials_manager/lib/credentials_manager/appfile_config.rb +192 -0
- data/credentials_manager/lib/credentials_manager/cli.rb +67 -0
- data/deliver/README.md +17 -0
- data/deliver/lib/assets/DeliverfileDefault +3 -0
- data/deliver/lib/assets/DeliverfileDefault.swift +13 -0
- data/deliver/lib/assets/ScreenshotsHelp +7 -0
- data/deliver/lib/assets/summary.html.erb +299 -0
- data/deliver/lib/deliver.rb +15 -0
- data/deliver/lib/deliver/app_screenshot.rb +352 -0
- data/deliver/lib/deliver/commands_generator.rb +186 -0
- data/deliver/lib/deliver/detect_values.rb +93 -0
- data/deliver/lib/deliver/download_screenshots.rb +49 -0
- data/deliver/lib/deliver/generate_summary.rb +13 -0
- data/deliver/lib/deliver/html_generator.rb +71 -0
- data/deliver/lib/deliver/loader.rb +56 -0
- data/deliver/lib/deliver/module.rb +16 -0
- data/deliver/lib/deliver/options.rb +419 -0
- data/deliver/lib/deliver/runner.rb +202 -0
- data/deliver/lib/deliver/setup.rb +135 -0
- data/deliver/lib/deliver/submit_for_review.rb +122 -0
- data/deliver/lib/deliver/upload_assets.rb +27 -0
- data/deliver/lib/deliver/upload_metadata.rb +391 -0
- data/deliver/lib/deliver/upload_price_tier.rb +23 -0
- data/deliver/lib/deliver/upload_screenshots.rb +158 -0
- data/fastlane/README.md +11 -0
- data/fastlane/lib/assets/ActionDetails.md.erb +106 -0
- data/fastlane/lib/assets/Actions.md.erb +43 -0
- data/fastlane/lib/assets/AppfileTemplate +6 -0
- data/fastlane/lib/assets/AppfileTemplate.swift +7 -0
- data/fastlane/lib/assets/AppfileTemplateAndroid +2 -0
- data/fastlane/lib/assets/DefaultFastfileTemplate +20 -0
- data/fastlane/lib/assets/DefaultFastfileTemplate.swift +13 -0
- data/fastlane/lib/assets/completions/completion.bash +23 -0
- data/fastlane/lib/assets/completions/completion.fish +39 -0
- data/fastlane/lib/assets/completions/completion.sh +12 -0
- data/fastlane/lib/assets/completions/completion.zsh +23 -0
- data/fastlane/lib/assets/custom_action_template.rb +82 -0
- data/fastlane/lib/assets/mailgun_html_template.erb +142 -0
- data/fastlane/lib/assets/report_template.xml.erb +15 -0
- data/fastlane/lib/assets/s3_html_template.erb +105 -0
- data/fastlane/lib/assets/s3_plist_template.erb +31 -0
- data/fastlane/lib/assets/s3_version_template.erb +4 -0
- data/fastlane/lib/fastlane.rb +50 -0
- data/fastlane/lib/fastlane/action.rb +198 -0
- data/fastlane/lib/fastlane/action_collector.rb +35 -0
- data/fastlane/lib/fastlane/actions/README.md +3 -0
- data/fastlane/lib/fastlane/actions/actions_helper.rb +162 -0
- data/fastlane/lib/fastlane/actions/adb.rb +78 -0
- data/fastlane/lib/fastlane/actions/adb_devices.rb +71 -0
- data/fastlane/lib/fastlane/actions/add_extra_platforms.rb +45 -0
- data/fastlane/lib/fastlane/actions/add_git_tag.rb +122 -0
- data/fastlane/lib/fastlane/actions/app_store_build_number.rb +193 -0
- data/fastlane/lib/fastlane/actions/appaloosa.rb +266 -0
- data/fastlane/lib/fastlane/actions/appetize.rb +188 -0
- data/fastlane/lib/fastlane/actions/appetize_viewing_url_generator.rb +146 -0
- data/fastlane/lib/fastlane/actions/appium.rb +206 -0
- data/fastlane/lib/fastlane/actions/appledoc.rb +226 -0
- data/fastlane/lib/fastlane/actions/appstore.rb +14 -0
- data/fastlane/lib/fastlane/actions/apteligent.rb +105 -0
- data/fastlane/lib/fastlane/actions/artifactory.rb +157 -0
- data/fastlane/lib/fastlane/actions/automatic_code_signing.rb +220 -0
- data/fastlane/lib/fastlane/actions/backup_file.rb +41 -0
- data/fastlane/lib/fastlane/actions/backup_xcarchive.rb +135 -0
- data/fastlane/lib/fastlane/actions/badge.rb +155 -0
- data/fastlane/lib/fastlane/actions/build_and_upload_to_appetize.rb +107 -0
- data/fastlane/lib/fastlane/actions/build_android_app.rb +14 -0
- data/fastlane/lib/fastlane/actions/build_app.rb +14 -0
- data/fastlane/lib/fastlane/actions/build_ios_app.rb +151 -0
- data/fastlane/lib/fastlane/actions/bundle_install.rb +156 -0
- data/fastlane/lib/fastlane/actions/capture_android_screenshots.rb +63 -0
- data/fastlane/lib/fastlane/actions/capture_ios_screenshots.rb +61 -0
- data/fastlane/lib/fastlane/actions/capture_screenshots.rb +14 -0
- data/fastlane/lib/fastlane/actions/carthage.rb +215 -0
- data/fastlane/lib/fastlane/actions/cert.rb +14 -0
- data/fastlane/lib/fastlane/actions/changelog_from_git_commits.rb +197 -0
- data/fastlane/lib/fastlane/actions/chatwork.rb +94 -0
- data/fastlane/lib/fastlane/actions/check_app_store_metadata.rb +53 -0
- data/fastlane/lib/fastlane/actions/clean_build_artifacts.rb +68 -0
- data/fastlane/lib/fastlane/actions/clean_cocoapods_cache.rb +52 -0
- data/fastlane/lib/fastlane/actions/clear_derived_data.rb +70 -0
- data/fastlane/lib/fastlane/actions/clipboard.rb +52 -0
- data/fastlane/lib/fastlane/actions/cloc.rb +89 -0
- data/fastlane/lib/fastlane/actions/cocoapods.rb +148 -0
- data/fastlane/lib/fastlane/actions/commit_github_file.rb +186 -0
- data/fastlane/lib/fastlane/actions/commit_version_bump.rb +301 -0
- data/fastlane/lib/fastlane/actions/copy_artifacts.rb +114 -0
- data/fastlane/lib/fastlane/actions/crashlytics.rb +203 -0
- data/fastlane/lib/fastlane/actions/create_app_on_managed_play_store.rb +174 -0
- data/fastlane/lib/fastlane/actions/create_app_online.rb +75 -0
- data/fastlane/lib/fastlane/actions/create_keychain.rb +173 -0
- data/fastlane/lib/fastlane/actions/create_pull_request.rb +190 -0
- data/fastlane/lib/fastlane/actions/danger.rb +124 -0
- data/fastlane/lib/fastlane/actions/debug.rb +32 -0
- data/fastlane/lib/fastlane/actions/default_platform.rb +47 -0
- data/fastlane/lib/fastlane/actions/delete_keychain.rb +68 -0
- data/fastlane/lib/fastlane/actions/deliver.rb +14 -0
- data/fastlane/lib/fastlane/actions/deploygate.rb +210 -0
- data/fastlane/lib/fastlane/actions/device_grid/README.md +157 -0
- data/fastlane/lib/fastlane/actions/docs/build_ios_app.md +245 -0
- data/fastlane/lib/fastlane/actions/docs/capture_android_screenshots.md +259 -0
- data/fastlane/lib/fastlane/actions/docs/capture_ios_screenshots.md +363 -0
- data/fastlane/lib/fastlane/actions/docs/check_app_store_metadata.md +103 -0
- data/fastlane/lib/fastlane/actions/docs/create_app_online.md +282 -0
- data/fastlane/lib/fastlane/actions/docs/frame_screenshots.md +330 -0
- data/fastlane/lib/fastlane/actions/docs/get_certificates.md +86 -0
- data/fastlane/lib/fastlane/actions/docs/get_provisioning_profile.md +223 -0
- data/fastlane/lib/fastlane/actions/docs/get_push_certificate.md +134 -0
- data/fastlane/lib/fastlane/actions/docs/run_tests.md +151 -0
- data/fastlane/lib/fastlane/actions/docs/sync_code_signing.md +479 -0
- data/fastlane/lib/fastlane/actions/docs/upload_to_app_store.md.erb +730 -0
- data/fastlane/lib/fastlane/actions/docs/upload_to_play_store.md +162 -0
- data/fastlane/lib/fastlane/actions/docs/upload_to_testflight.md +231 -0
- data/fastlane/lib/fastlane/actions/dotgpg_environment.rb +57 -0
- data/fastlane/lib/fastlane/actions/download.rb +76 -0
- data/fastlane/lib/fastlane/actions/download_dsyms.rb +266 -0
- data/fastlane/lib/fastlane/actions/download_from_play_store.rb +61 -0
- data/fastlane/lib/fastlane/actions/dsym_zip.rb +93 -0
- data/fastlane/lib/fastlane/actions/echo.rb +14 -0
- data/fastlane/lib/fastlane/actions/ensure_bundle_exec.rb +59 -0
- data/fastlane/lib/fastlane/actions/ensure_git_branch.rb +69 -0
- data/fastlane/lib/fastlane/actions/ensure_git_status_clean.rb +81 -0
- data/fastlane/lib/fastlane/actions/ensure_no_debug_code.rb +135 -0
- data/fastlane/lib/fastlane/actions/ensure_xcode_version.rb +97 -0
- data/fastlane/lib/fastlane/actions/environment_variable.rb +77 -0
- data/fastlane/lib/fastlane/actions/erb.rb +88 -0
- data/fastlane/lib/fastlane/actions/fastlane_version.rb +15 -0
- data/fastlane/lib/fastlane/actions/flock.rb +82 -0
- data/fastlane/lib/fastlane/actions/frame_screenshots.rb +63 -0
- data/fastlane/lib/fastlane/actions/frameit.rb +14 -0
- data/fastlane/lib/fastlane/actions/gcovr.rb +163 -0
- data/fastlane/lib/fastlane/actions/get_build_number.rb +106 -0
- data/fastlane/lib/fastlane/actions/get_build_number_repository.rb +120 -0
- data/fastlane/lib/fastlane/actions/get_certificates.rb +76 -0
- data/fastlane/lib/fastlane/actions/get_github_release.rb +163 -0
- data/fastlane/lib/fastlane/actions/get_info_plist_value.rb +78 -0
- data/fastlane/lib/fastlane/actions/get_ipa_info_plist_value.rb +82 -0
- data/fastlane/lib/fastlane/actions/get_managed_play_store_publishing_rights.rb +120 -0
- data/fastlane/lib/fastlane/actions/get_provisioning_profile.rb +100 -0
- data/fastlane/lib/fastlane/actions/get_push_certificate.rb +87 -0
- data/fastlane/lib/fastlane/actions/get_version_number.rb +170 -0
- data/fastlane/lib/fastlane/actions/git_add.rb +93 -0
- data/fastlane/lib/fastlane/actions/git_branch.rb +58 -0
- data/fastlane/lib/fastlane/actions/git_commit.rb +80 -0
- data/fastlane/lib/fastlane/actions/git_pull.rb +53 -0
- data/fastlane/lib/fastlane/actions/git_submodule_update.rb +52 -0
- data/fastlane/lib/fastlane/actions/git_tag_exists.rb +74 -0
- data/fastlane/lib/fastlane/actions/github_api.rb +262 -0
- data/fastlane/lib/fastlane/actions/google_play_track_version_codes.rb +73 -0
- data/fastlane/lib/fastlane/actions/gradle.rb +249 -0
- data/fastlane/lib/fastlane/actions/gym.rb +14 -0
- data/fastlane/lib/fastlane/actions/hg_add_tag.rb +48 -0
- data/fastlane/lib/fastlane/actions/hg_commit_version_bump.rb +183 -0
- data/fastlane/lib/fastlane/actions/hg_ensure_clean_status.rb +53 -0
- data/fastlane/lib/fastlane/actions/hg_push.rb +64 -0
- data/fastlane/lib/fastlane/actions/hipchat.rb +200 -0
- data/fastlane/lib/fastlane/actions/hockey.rb +406 -0
- data/fastlane/lib/fastlane/actions/ifttt.rb +96 -0
- data/fastlane/lib/fastlane/actions/import.rb +49 -0
- data/fastlane/lib/fastlane/actions/import_certificate.rb +77 -0
- data/fastlane/lib/fastlane/actions/import_from_git.rb +71 -0
- data/fastlane/lib/fastlane/actions/increment_build_number.rb +117 -0
- data/fastlane/lib/fastlane/actions/increment_version_number.rb +177 -0
- data/fastlane/lib/fastlane/actions/install_on_device.rb +95 -0
- data/fastlane/lib/fastlane/actions/install_provisioning_profile.rb +58 -0
- data/fastlane/lib/fastlane/actions/install_xcode_plugin.rb +91 -0
- data/fastlane/lib/fastlane/actions/installr.rb +132 -0
- data/fastlane/lib/fastlane/actions/ipa.rb +246 -0
- data/fastlane/lib/fastlane/actions/is_ci.rb +51 -0
- data/fastlane/lib/fastlane/actions/jazzy.rb +56 -0
- data/fastlane/lib/fastlane/actions/jira.rb +115 -0
- data/fastlane/lib/fastlane/actions/lane_context.rb +60 -0
- data/fastlane/lib/fastlane/actions/last_git_commit.rb +58 -0
- data/fastlane/lib/fastlane/actions/last_git_tag.rb +51 -0
- data/fastlane/lib/fastlane/actions/latest_testflight_build_number.rb +134 -0
- data/fastlane/lib/fastlane/actions/lcov.rb +102 -0
- data/fastlane/lib/fastlane/actions/mailgun.rb +197 -0
- data/fastlane/lib/fastlane/actions/make_changelog_from_jenkins.rb +81 -0
- data/fastlane/lib/fastlane/actions/match.rb +14 -0
- data/fastlane/lib/fastlane/actions/min_fastlane_version.rb +57 -0
- data/fastlane/lib/fastlane/actions/modify_services.rb +173 -0
- data/fastlane/lib/fastlane/actions/nexus_upload.rb +230 -0
- data/fastlane/lib/fastlane/actions/notification.rb +75 -0
- data/fastlane/lib/fastlane/actions/notify.rb +40 -0
- data/fastlane/lib/fastlane/actions/number_of_commits.rb +75 -0
- data/fastlane/lib/fastlane/actions/oclint.rb +269 -0
- data/fastlane/lib/fastlane/actions/onesignal.rb +156 -0
- data/fastlane/lib/fastlane/actions/opt_out_crash_reporting.rb +33 -0
- data/fastlane/lib/fastlane/actions/opt_out_usage.rb +40 -0
- data/fastlane/lib/fastlane/actions/pem.rb +14 -0
- data/fastlane/lib/fastlane/actions/pilot.rb +14 -0
- data/fastlane/lib/fastlane/actions/pod_lib_lint.rb +183 -0
- data/fastlane/lib/fastlane/actions/pod_push.rb +145 -0
- data/fastlane/lib/fastlane/actions/podio_item.rb +217 -0
- data/fastlane/lib/fastlane/actions/precheck.rb +14 -0
- data/fastlane/lib/fastlane/actions/println.rb +14 -0
- data/fastlane/lib/fastlane/actions/produce.rb +14 -0
- data/fastlane/lib/fastlane/actions/prompt.rb +119 -0
- data/fastlane/lib/fastlane/actions/push_git_tags.rb +76 -0
- data/fastlane/lib/fastlane/actions/push_to_git_remote.rb +127 -0
- data/fastlane/lib/fastlane/actions/puts.rb +68 -0
- data/fastlane/lib/fastlane/actions/read_podspec.rb +90 -0
- data/fastlane/lib/fastlane/actions/recreate_schemes.rb +46 -0
- data/fastlane/lib/fastlane/actions/register_device.rb +110 -0
- data/fastlane/lib/fastlane/actions/register_devices.rb +160 -0
- data/fastlane/lib/fastlane/actions/reset_git_repo.rb +121 -0
- data/fastlane/lib/fastlane/actions/reset_simulator_contents.rb +92 -0
- data/fastlane/lib/fastlane/actions/resign.rb +132 -0
- data/fastlane/lib/fastlane/actions/restore_file.rb +44 -0
- data/fastlane/lib/fastlane/actions/rocket.rb +83 -0
- data/fastlane/lib/fastlane/actions/rsync.rb +74 -0
- data/fastlane/lib/fastlane/actions/ruby_version.rb +56 -0
- data/fastlane/lib/fastlane/actions/run_tests.rb +131 -0
- data/fastlane/lib/fastlane/actions/s3.rb +460 -0
- data/fastlane/lib/fastlane/actions/say.rb +56 -0
- data/fastlane/lib/fastlane/actions/scan.rb +14 -0
- data/fastlane/lib/fastlane/actions/scp.rb +114 -0
- data/fastlane/lib/fastlane/actions/screengrab.rb +14 -0
- data/fastlane/lib/fastlane/actions/set_build_number_repository.rb +76 -0
- data/fastlane/lib/fastlane/actions/set_changelog.rb +172 -0
- data/fastlane/lib/fastlane/actions/set_github_release.rb +274 -0
- data/fastlane/lib/fastlane/actions/set_info_plist_value.rb +92 -0
- data/fastlane/lib/fastlane/actions/set_pod_key.rb +78 -0
- data/fastlane/lib/fastlane/actions/setup_ci.rb +121 -0
- data/fastlane/lib/fastlane/actions/setup_circle_ci.rb +58 -0
- data/fastlane/lib/fastlane/actions/setup_jenkins.rb +193 -0
- data/fastlane/lib/fastlane/actions/setup_travis.rb +57 -0
- data/fastlane/lib/fastlane/actions/sh.rb +71 -0
- data/fastlane/lib/fastlane/actions/sigh.rb +14 -0
- data/fastlane/lib/fastlane/actions/skip_docs.rb +52 -0
- data/fastlane/lib/fastlane/actions/slack.rb +288 -0
- data/fastlane/lib/fastlane/actions/slather.rb +314 -0
- data/fastlane/lib/fastlane/actions/snapshot.rb +14 -0
- data/fastlane/lib/fastlane/actions/sonar.rb +138 -0
- data/fastlane/lib/fastlane/actions/spaceship_logs.rb +135 -0
- data/fastlane/lib/fastlane/actions/splunkmint.rb +156 -0
- data/fastlane/lib/fastlane/actions/spm.rb +123 -0
- data/fastlane/lib/fastlane/actions/ssh.rb +162 -0
- data/fastlane/lib/fastlane/actions/supply.rb +14 -0
- data/fastlane/lib/fastlane/actions/swiftlint.rb +182 -0
- data/fastlane/lib/fastlane/actions/sync_code_signing.rb +103 -0
- data/fastlane/lib/fastlane/actions/team_id.rb +42 -0
- data/fastlane/lib/fastlane/actions/team_name.rb +42 -0
- data/fastlane/lib/fastlane/actions/testfairy.rb +263 -0
- data/fastlane/lib/fastlane/actions/testflight.rb +14 -0
- data/fastlane/lib/fastlane/actions/tryouts.rb +149 -0
- data/fastlane/lib/fastlane/actions/twitter.rb +89 -0
- data/fastlane/lib/fastlane/actions/typetalk.rb +93 -0
- data/fastlane/lib/fastlane/actions/unlock_keychain.rb +132 -0
- data/fastlane/lib/fastlane/actions/update_app_group_identifiers.rb +91 -0
- data/fastlane/lib/fastlane/actions/update_app_identifier.rb +124 -0
- data/fastlane/lib/fastlane/actions/update_fastlane.rb +177 -0
- data/fastlane/lib/fastlane/actions/update_icloud_container_identifiers.rb +99 -0
- data/fastlane/lib/fastlane/actions/update_info_plist.rb +143 -0
- data/fastlane/lib/fastlane/actions/update_plist.rb +80 -0
- data/fastlane/lib/fastlane/actions/update_project_code_signing.rb +76 -0
- data/fastlane/lib/fastlane/actions/update_project_provisioning.rb +173 -0
- data/fastlane/lib/fastlane/actions/update_project_team.rb +95 -0
- data/fastlane/lib/fastlane/actions/update_urban_airship_configuration.rb +91 -0
- data/fastlane/lib/fastlane/actions/update_url_schemes.rb +123 -0
- data/fastlane/lib/fastlane/actions/upload_symbols_to_crashlytics.rb +244 -0
- data/fastlane/lib/fastlane/actions/upload_symbols_to_sentry.rb +165 -0
- data/fastlane/lib/fastlane/actions/upload_to_app_store.rb +65 -0
- data/fastlane/lib/fastlane/actions/upload_to_play_store.rb +80 -0
- data/fastlane/lib/fastlane/actions/upload_to_testflight.rb +107 -0
- data/fastlane/lib/fastlane/actions/validate_play_store_json_key.rb +104 -0
- data/fastlane/lib/fastlane/actions/verify_build.rb +199 -0
- data/fastlane/lib/fastlane/actions/verify_pod_keys.rb +67 -0
- data/fastlane/lib/fastlane/actions/verify_xcode.rb +152 -0
- data/fastlane/lib/fastlane/actions/version_bump_podspec.rb +105 -0
- data/fastlane/lib/fastlane/actions/version_get_podspec.rb +71 -0
- data/fastlane/lib/fastlane/actions/xcode_install.rb +105 -0
- data/fastlane/lib/fastlane/actions/xcode_select.rb +63 -0
- data/fastlane/lib/fastlane/actions/xcode_server_get_assets.rb +312 -0
- data/fastlane/lib/fastlane/actions/xcodebuild.rb +641 -0
- data/fastlane/lib/fastlane/actions/xcov.rb +66 -0
- data/fastlane/lib/fastlane/actions/xctool.rb +55 -0
- data/fastlane/lib/fastlane/actions/xcversion.rb +55 -0
- data/fastlane/lib/fastlane/actions/zip.rb +120 -0
- data/fastlane/lib/fastlane/auto_complete.rb +82 -0
- data/fastlane/lib/fastlane/boolean.rb +5 -0
- data/fastlane/lib/fastlane/cli_tools_distributor.rb +185 -0
- data/fastlane/lib/fastlane/command_line_handler.rb +49 -0
- data/fastlane/lib/fastlane/commands_generator.rb +350 -0
- data/fastlane/lib/fastlane/configuration_helper.rb +26 -0
- data/fastlane/lib/fastlane/core_ext/bundler_monkey_patch.rb +14 -0
- data/fastlane/lib/fastlane/documentation/actions_list.rb +214 -0
- data/fastlane/lib/fastlane/documentation/docs_generator.rb +89 -0
- data/fastlane/lib/fastlane/documentation/markdown_docs_generator.rb +228 -0
- data/fastlane/lib/fastlane/environment_printer.rb +297 -0
- data/fastlane/lib/fastlane/erb_template_helper.rb +30 -0
- data/fastlane/lib/fastlane/fast_file.rb +351 -0
- data/fastlane/lib/fastlane/fastlane_require.rb +75 -0
- data/fastlane/lib/fastlane/features.rb +4 -0
- data/fastlane/lib/fastlane/helper/README.md +29 -0
- data/fastlane/lib/fastlane/helper/adb_helper.rb +52 -0
- data/fastlane/lib/fastlane/helper/crashlytics_helper.rb +151 -0
- data/fastlane/lib/fastlane/helper/dotenv_helper.rb +50 -0
- data/fastlane/lib/fastlane/helper/gem_helper.rb +26 -0
- data/fastlane/lib/fastlane/helper/git_helper.rb +135 -0
- data/fastlane/lib/fastlane/helper/gradle_helper.rb +62 -0
- data/fastlane/lib/fastlane/helper/podspec_helper.rb +75 -0
- data/fastlane/lib/fastlane/helper/sh_helper.rb +134 -0
- data/fastlane/lib/fastlane/helper/xcodeproj_helper.rb +12 -0
- data/fastlane/lib/fastlane/helper/xcversion_helper.rb +25 -0
- data/fastlane/lib/fastlane/junit_generator.rb +27 -0
- data/fastlane/lib/fastlane/lane.rb +97 -0
- data/fastlane/lib/fastlane/lane_list.rb +137 -0
- data/fastlane/lib/fastlane/lane_manager.rb +140 -0
- data/fastlane/lib/fastlane/lane_manager_base.rb +92 -0
- data/fastlane/lib/fastlane/markdown_table_formatter.rb +62 -0
- data/fastlane/lib/fastlane/new_action.rb +47 -0
- data/fastlane/lib/fastlane/one_off.rb +45 -0
- data/fastlane/lib/fastlane/other_action.rb +29 -0
- data/fastlane/lib/fastlane/plugins/plugin_fetcher.rb +55 -0
- data/fastlane/lib/fastlane/plugins/plugin_generator.rb +86 -0
- data/fastlane/lib/fastlane/plugins/plugin_generator_ui.rb +19 -0
- data/fastlane/lib/fastlane/plugins/plugin_info.rb +49 -0
- data/fastlane/lib/fastlane/plugins/plugin_info_collector.rb +159 -0
- data/fastlane/lib/fastlane/plugins/plugin_manager.rb +387 -0
- data/fastlane/lib/fastlane/plugins/plugin_search.rb +46 -0
- data/fastlane/lib/fastlane/plugins/plugin_update_manager.rb +70 -0
- data/fastlane/lib/fastlane/plugins/plugins.rb +12 -0
- data/fastlane/lib/fastlane/plugins/template/%gem_name%.gemspec.erb +35 -0
- data/fastlane/lib/fastlane/plugins/template/.circleci/config.yml +43 -0
- data/fastlane/lib/fastlane/plugins/template/.gitignore +12 -0
- data/fastlane/lib/fastlane/plugins/template/.rspec +5 -0
- data/fastlane/lib/fastlane/plugins/template/.rubocop.yml +180 -0
- data/fastlane/lib/fastlane/plugins/template/.travis.yml +4 -0
- data/fastlane/lib/fastlane/plugins/template/Gemfile +6 -0
- data/fastlane/lib/fastlane/plugins/template/LICENSE.erb +21 -0
- data/fastlane/lib/fastlane/plugins/template/README.md.erb +52 -0
- data/fastlane/lib/fastlane/plugins/template/Rakefile +9 -0
- data/fastlane/lib/fastlane/plugins/template/fastlane/Fastfile.erb +3 -0
- data/fastlane/lib/fastlane/plugins/template/fastlane/Pluginfile.erb +1 -0
- data/fastlane/lib/fastlane/plugins/template/lib/fastlane/plugin/%plugin_name%.rb.erb +16 -0
- data/fastlane/lib/fastlane/plugins/template/lib/fastlane/plugin/%plugin_name%/actions/%plugin_name%_action.rb.erb +47 -0
- data/fastlane/lib/fastlane/plugins/template/lib/fastlane/plugin/%plugin_name%/helper/%plugin_name%_helper.rb.erb +16 -0
- data/fastlane/lib/fastlane/plugins/template/lib/fastlane/plugin/%plugin_name%/version.rb.erb +5 -0
- data/fastlane/lib/fastlane/plugins/template/spec/%plugin_name%_action_spec.rb.erb +9 -0
- data/fastlane/lib/fastlane/plugins/template/spec/spec_helper.rb.erb +15 -0
- data/fastlane/lib/fastlane/runner.rb +371 -0
- data/fastlane/lib/fastlane/server/action_command.rb +61 -0
- data/fastlane/lib/fastlane/server/action_command_return.rb +14 -0
- data/fastlane/lib/fastlane/server/command_executor.rb +7 -0
- data/fastlane/lib/fastlane/server/command_parser.rb +44 -0
- data/fastlane/lib/fastlane/server/control_command.rb +23 -0
- data/fastlane/lib/fastlane/server/json_return_value_processor.rb +72 -0
- data/fastlane/lib/fastlane/server/socket_server.rb +232 -0
- data/fastlane/lib/fastlane/server/socket_server_action_command_executor.rb +101 -0
- data/fastlane/lib/fastlane/setup/setup.rb +355 -0
- data/fastlane/lib/fastlane/setup/setup_android.rb +97 -0
- data/fastlane/lib/fastlane/setup/setup_ios.rb +504 -0
- data/fastlane/lib/fastlane/shells.rb +6 -0
- data/fastlane/lib/fastlane/supported_platforms.rb +28 -0
- data/fastlane/lib/fastlane/swift_fastlane_api_generator.rb +383 -0
- data/fastlane/lib/fastlane/swift_fastlane_function.rb +320 -0
- data/fastlane/lib/fastlane/swift_lane_manager.rb +304 -0
- data/fastlane/lib/fastlane/swift_runner_upgrader.rb +206 -0
- data/fastlane/lib/fastlane/tools.rb +33 -0
- data/fastlane/lib/fastlane/version.rb +6 -0
- data/fastlane/swift/Appfile.swift +17 -0
- data/fastlane/swift/ArgumentProcessor.swift +93 -0
- data/fastlane/swift/ControlCommand.swift +76 -0
- data/fastlane/swift/Deliverfile.swift +21 -0
- data/fastlane/swift/DeliverfileProtocol.swift +129 -0
- data/fastlane/swift/Fastfile.swift +12 -0
- data/fastlane/swift/Fastlane.swift +4396 -0
- data/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.pbxproj +425 -0
- data/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/project.xcworkspace/contents.xcworkspacedata +7 -0
- data/fastlane/swift/FastlaneSwiftRunner/FastlaneSwiftRunner.xcodeproj/xcshareddata/xcschemes/FastlaneRunner.xcscheme +103 -0
- data/fastlane/swift/FastlaneSwiftRunner/README.txt +10 -0
- data/fastlane/swift/Gymfile.swift +21 -0
- data/fastlane/swift/GymfileProtocol.swift +87 -0
- data/fastlane/swift/LaneFileProtocol.swift +140 -0
- data/fastlane/swift/Matchfile.swift +21 -0
- data/fastlane/swift/MatchfileProtocol.swift +61 -0
- data/fastlane/swift/Precheckfile.swift +21 -0
- data/fastlane/swift/PrecheckfileProtocol.swift +24 -0
- data/fastlane/swift/RubyCommand.swift +141 -0
- data/fastlane/swift/RubyCommandable.swift +44 -0
- data/fastlane/swift/Runner.swift +200 -0
- data/fastlane/swift/RunnerArgument.swift +24 -0
- data/fastlane/swift/Scanfile.swift +21 -0
- data/fastlane/swift/ScanfileProtocol.swift +117 -0
- data/fastlane/swift/Screengrabfile.swift +21 -0
- data/fastlane/swift/ScreengrabfileProtocol.swift +48 -0
- data/fastlane/swift/Snapshotfile.swift +21 -0
- data/fastlane/swift/SnapshotfileProtocol.swift +75 -0
- data/fastlane/swift/SocketClient.swift +305 -0
- data/fastlane/swift/SocketClientDelegateProtocol.swift +25 -0
- data/fastlane/swift/SocketResponse.swift +85 -0
- data/fastlane/swift/main.swift +52 -0
- data/fastlane/swift/upgrade_manifest.json +1 -0
- data/fastlane_core/README.md +79 -0
- data/fastlane_core/lib/assets/XMLTemplate.xml.erb +12 -0
- data/fastlane_core/lib/fastlane_core.rb +46 -0
- data/fastlane_core/lib/fastlane_core/analytics/action_completion_context.rb +34 -0
- data/fastlane_core/lib/fastlane_core/analytics/action_launch_context.rb +38 -0
- data/fastlane_core/lib/fastlane_core/analytics/analytics_event_builder.rb +23 -0
- data/fastlane_core/lib/fastlane_core/analytics/analytics_ingester_client.rb +54 -0
- data/fastlane_core/lib/fastlane_core/analytics/analytics_session.rb +71 -0
- data/fastlane_core/lib/fastlane_core/analytics/app_identifier_guesser.rb +64 -0
- data/fastlane_core/lib/fastlane_core/android_package_name_guesser.rb +79 -0
- data/fastlane_core/lib/fastlane_core/build_watcher.rb +80 -0
- data/fastlane_core/lib/fastlane_core/cert_checker.rb +116 -0
- data/fastlane_core/lib/fastlane_core/command_executor.rb +99 -0
- data/fastlane_core/lib/fastlane_core/configuration/commander_generator.rb +103 -0
- data/fastlane_core/lib/fastlane_core/configuration/config_item.rb +314 -0
- data/fastlane_core/lib/fastlane_core/configuration/configuration.rb +332 -0
- data/fastlane_core/lib/fastlane_core/configuration/configuration_file.rb +182 -0
- data/fastlane_core/lib/fastlane_core/core_ext/cfpropertylist.rb +23 -0
- data/fastlane_core/lib/fastlane_core/core_ext/shellwords.rb +63 -0
- data/fastlane_core/lib/fastlane_core/core_ext/string.rb +17 -0
- data/fastlane_core/lib/fastlane_core/device_manager.rb +331 -0
- data/fastlane_core/lib/fastlane_core/env.rb +9 -0
- data/fastlane_core/lib/fastlane_core/fastlane_folder.rb +72 -0
- data/fastlane_core/lib/fastlane_core/fastlane_pty.rb +57 -0
- data/fastlane_core/lib/fastlane_core/feature/feature.rb +51 -0
- data/fastlane_core/lib/fastlane_core/features.rb +4 -0
- data/fastlane_core/lib/fastlane_core/globals.rb +27 -0
- data/fastlane_core/lib/fastlane_core/helper.rb +414 -0
- data/fastlane_core/lib/fastlane_core/ios_app_identifier_guesser.rb +112 -0
- data/fastlane_core/lib/fastlane_core/ipa_file_analyser.rb +63 -0
- data/fastlane_core/lib/fastlane_core/ipa_upload_package_builder.rb +52 -0
- data/fastlane_core/lib/fastlane_core/itunes_transporter.rb +520 -0
- data/fastlane_core/lib/fastlane_core/keychain_importer.rb +74 -0
- data/fastlane_core/lib/fastlane_core/languages.rb +16 -0
- data/fastlane_core/lib/fastlane_core/module.rb +31 -0
- data/fastlane_core/lib/fastlane_core/pkg_file_analyser.rb +44 -0
- data/fastlane_core/lib/fastlane_core/pkg_upload_package_builder.rb +48 -0
- data/fastlane_core/lib/fastlane_core/print_table.rb +131 -0
- data/fastlane_core/lib/fastlane_core/project.rb +481 -0
- data/fastlane_core/lib/fastlane_core/provisioning_profile.rb +111 -0
- data/fastlane_core/lib/fastlane_core/string_filters.rb +51 -0
- data/fastlane_core/lib/fastlane_core/swag.rb +85 -0
- data/fastlane_core/lib/fastlane_core/tag_version.rb +31 -0
- data/fastlane_core/lib/fastlane_core/test_parser.rb +107 -0
- data/fastlane_core/lib/fastlane_core/ui/disable_colors.rb +17 -0
- data/fastlane_core/lib/fastlane_core/ui/errors.rb +1 -0
- data/fastlane_core/lib/fastlane_core/ui/errors/fastlane_common_error.rb +19 -0
- data/fastlane_core/lib/fastlane_core/ui/errors/fastlane_crash.rb +11 -0
- data/fastlane_core/lib/fastlane_core/ui/errors/fastlane_error.rb +25 -0
- data/fastlane_core/lib/fastlane_core/ui/errors/fastlane_exception.rb +19 -0
- data/fastlane_core/lib/fastlane_core/ui/errors/fastlane_shell_error.rb +11 -0
- data/fastlane_core/lib/fastlane_core/ui/fastlane_runner.rb +294 -0
- data/fastlane_core/lib/fastlane_core/ui/github_issue_inspector_reporter.rb +62 -0
- data/fastlane_core/lib/fastlane_core/ui/implementations/shell.rb +158 -0
- data/fastlane_core/lib/fastlane_core/ui/interface.rb +205 -0
- data/fastlane_core/lib/fastlane_core/ui/ui.rb +26 -0
- data/fastlane_core/lib/fastlane_core/update_checker/changelog.rb +37 -0
- data/fastlane_core/lib/fastlane_core/update_checker/update_checker.rb +109 -0
- data/frameit/README.md +17 -0
- data/frameit/lib/assets/empty.png +0 -0
- data/frameit/lib/frameit.rb +13 -0
- data/frameit/lib/frameit/commands_generator.rb +114 -0
- data/frameit/lib/frameit/config_parser.rb +114 -0
- data/frameit/lib/frameit/dependency_checker.rb +22 -0
- data/frameit/lib/frameit/device_types.rb +18 -0
- data/frameit/lib/frameit/editor.rb +563 -0
- data/frameit/lib/frameit/frame_downloader.rb +85 -0
- data/frameit/lib/frameit/mac_editor.rb +35 -0
- data/frameit/lib/frameit/module.rb +42 -0
- data/frameit/lib/frameit/offsets.rb +21 -0
- data/frameit/lib/frameit/options.rb +82 -0
- data/frameit/lib/frameit/runner.rb +82 -0
- data/frameit/lib/frameit/screenshot.rb +137 -0
- data/frameit/lib/frameit/strings_parser.rb +46 -0
- data/frameit/lib/frameit/template_finder.rb +42 -0
- data/frameit/lib/frameit/trim_box.rb +35 -0
- data/gym/README.md +17 -0
- data/gym/lib/assets/GymfileTemplate +13 -0
- data/gym/lib/assets/GymfileTemplate.swift +13 -0
- data/gym/lib/assets/wrap_xcodebuild/xcbuild-safe.sh +61 -0
- data/gym/lib/gym.rb +9 -0
- data/gym/lib/gym/code_signing_mapping.rb +181 -0
- data/gym/lib/gym/commands_generator.rb +78 -0
- data/gym/lib/gym/detect_values.rb +171 -0
- data/gym/lib/gym/error_handler.rb +306 -0
- data/gym/lib/gym/generators/README.md +1 -0
- data/gym/lib/gym/generators/build_command_generator.rb +143 -0
- data/gym/lib/gym/generators/package_command_generator.rb +56 -0
- data/gym/lib/gym/generators/package_command_generator_xcode7.rb +193 -0
- data/gym/lib/gym/manager.rb +22 -0
- data/gym/lib/gym/module.rb +36 -0
- data/gym/lib/gym/options.rb +262 -0
- data/gym/lib/gym/runner.rb +264 -0
- data/gym/lib/gym/xcode.rb +27 -0
- data/gym/lib/gym/xcodebuild_fixes/README.md +9 -0
- data/gym/lib/gym/xcodebuild_fixes/generic_archive_fix.rb +35 -0
- data/match/README.md +17 -0
- data/match/lib/assets/MatchfileTemplate +11 -0
- data/match/lib/assets/MatchfileTemplate.swift +9 -0
- data/match/lib/assets/READMETemplate.md +56 -0
- data/match/lib/match.rb +13 -0
- data/match/lib/match/change_password.rb +70 -0
- data/match/lib/match/commands_generator.rb +175 -0
- data/match/lib/match/encryption.rb +45 -0
- data/match/lib/match/encryption/interface.rb +17 -0
- data/match/lib/match/encryption/openssl.rb +161 -0
- data/match/lib/match/generator.rb +91 -0
- data/match/lib/match/migrate.rb +98 -0
- data/match/lib/match/module.rb +27 -0
- data/match/lib/match/nuke.rb +260 -0
- data/match/lib/match/options.rb +198 -0
- data/match/lib/match/runner.rb +333 -0
- data/match/lib/match/setup.rb +40 -0
- data/match/lib/match/spaceship_ensure.rb +83 -0
- data/match/lib/match/storage.rb +42 -0
- data/match/lib/match/storage/git_storage.rb +214 -0
- data/match/lib/match/storage/google_cloud_storage.rb +265 -0
- data/match/lib/match/storage/interface.rb +110 -0
- data/match/lib/match/table_printer.rb +51 -0
- data/match/lib/match/utils.rb +75 -0
- data/pem/README.md +17 -0
- data/pem/lib/pem.rb +4 -0
- data/pem/lib/pem/commands_generator.rb +46 -0
- data/pem/lib/pem/manager.rb +112 -0
- data/pem/lib/pem/module.rb +19 -0
- data/pem/lib/pem/options.rb +103 -0
- data/pilot/README.md +17 -0
- data/pilot/lib/pilot.rb +9 -0
- data/pilot/lib/pilot/build_manager.rb +453 -0
- data/pilot/lib/pilot/commands_generator.rb +174 -0
- data/pilot/lib/pilot/features.rb +0 -0
- data/pilot/lib/pilot/manager.rb +80 -0
- data/pilot/lib/pilot/module.rb +11 -0
- data/pilot/lib/pilot/options.rb +266 -0
- data/pilot/lib/pilot/tester_exporter.rb +58 -0
- data/pilot/lib/pilot/tester_importer.rb +53 -0
- data/pilot/lib/pilot/tester_manager.rb +168 -0
- data/pilot/lib/pilot/tester_util.rb +0 -0
- data/precheck/README.md +17 -0
- data/precheck/lib/assets/PrecheckfileTemplate +27 -0
- data/precheck/lib/assets/PrecheckfileTemplate.swift +19 -0
- data/precheck/lib/precheck.rb +3 -0
- data/precheck/lib/precheck/commands_generator.rb +74 -0
- data/precheck/lib/precheck/item_to_check.rb +58 -0
- data/precheck/lib/precheck/module.rb +21 -0
- data/precheck/lib/precheck/options.rb +79 -0
- data/precheck/lib/precheck/rule.rb +171 -0
- data/precheck/lib/precheck/rule_check_result.rb +19 -0
- data/precheck/lib/precheck/rule_processor.rb +231 -0
- data/precheck/lib/precheck/rules/abstract_text_match_rule.rb +80 -0
- data/precheck/lib/precheck/rules/all.rb +1 -0
- data/precheck/lib/precheck/rules/copyright_date_rule.rb +37 -0
- data/precheck/lib/precheck/rules/curse_words_rule.rb +62 -0
- data/precheck/lib/precheck/rules/custom_text_rule.rb +35 -0
- data/precheck/lib/precheck/rules/free_stuff_iap_rule.rb +31 -0
- data/precheck/lib/precheck/rules/future_functionality_rule.rb +33 -0
- data/precheck/lib/precheck/rules/negative_apple_sentiment_rule.rb +37 -0
- data/precheck/lib/precheck/rules/other_platforms_rule.rb +43 -0
- data/precheck/lib/precheck/rules/placeholder_words_rule.rb +31 -0
- data/precheck/lib/precheck/rules/rules_data/curse_word_hashes/en_us.txt +349 -0
- data/precheck/lib/precheck/rules/test_words_rule.rb +30 -0
- data/precheck/lib/precheck/rules/unreachable_urls_rule.rb +47 -0
- data/precheck/lib/precheck/runner.rb +176 -0
- data/produce/README.md +17 -0
- data/produce/lib/produce.rb +6 -0
- data/produce/lib/produce/available_default_languages.rb +51 -0
- data/produce/lib/produce/cloud_container.rb +82 -0
- data/produce/lib/produce/commands_generator.rb +235 -0
- data/produce/lib/produce/developer_center.rb +134 -0
- data/produce/lib/produce/group.rb +90 -0
- data/produce/lib/produce/itunes_connect.rb +92 -0
- data/produce/lib/produce/manager.rb +15 -0
- data/produce/lib/produce/merchant.rb +121 -0
- data/produce/lib/produce/module.rb +14 -0
- data/produce/lib/produce/options.rb +172 -0
- data/produce/lib/produce/service.rb +313 -0
- data/scan/README.md +17 -0
- data/scan/lib/assets/ScanfileTemplate +16 -0
- data/scan/lib/assets/ScanfileTemplate.swift +13 -0
- data/scan/lib/scan.rb +10 -0
- data/scan/lib/scan/commands_generator.rb +78 -0
- data/scan/lib/scan/detect_values.rb +219 -0
- data/scan/lib/scan/error_handler.rb +52 -0
- data/scan/lib/scan/manager.rb +21 -0
- data/scan/lib/scan/module.rb +32 -0
- data/scan/lib/scan/options.rb +390 -0
- data/scan/lib/scan/runner.rb +173 -0
- data/scan/lib/scan/slack_poster.rb +60 -0
- data/scan/lib/scan/test_command_generator.rb +160 -0
- data/scan/lib/scan/test_result_parser.rb +26 -0
- data/scan/lib/scan/xcpretty_reporter_options_generator.rb +85 -0
- data/screengrab/README.md +17 -0
- data/screengrab/lib/assets/ScreengrabfileTemplate +15 -0
- data/screengrab/lib/assets/ScreengrabfileTemplate.swift +15 -0
- data/screengrab/lib/screengrab.rb +7 -0
- data/screengrab/lib/screengrab/android_environment.rb +91 -0
- data/screengrab/lib/screengrab/commands_generator.rb +70 -0
- data/screengrab/lib/screengrab/dependency_checker.rb +58 -0
- data/screengrab/lib/screengrab/detect_values.rb +16 -0
- data/screengrab/lib/screengrab/module.rb +24 -0
- data/screengrab/lib/screengrab/options.rb +135 -0
- data/screengrab/lib/screengrab/page.html.erb +190 -0
- data/screengrab/lib/screengrab/reports_generator.rb +36 -0
- data/screengrab/lib/screengrab/runner.rb +379 -0
- data/screengrab/lib/screengrab/setup.rb +24 -0
- data/sigh/README.md +17 -0
- data/sigh/lib/assets/resign.sh +888 -0
- data/sigh/lib/sigh.rb +4 -0
- data/sigh/lib/sigh/commands_generator.rb +153 -0
- data/sigh/lib/sigh/download_all.rb +45 -0
- data/sigh/lib/sigh/local_manage.rb +145 -0
- data/sigh/lib/sigh/manager.rb +47 -0
- data/sigh/lib/sigh/module.rb +16 -0
- data/sigh/lib/sigh/options.rb +160 -0
- data/sigh/lib/sigh/repair.rb +34 -0
- data/sigh/lib/sigh/resign.rb +220 -0
- data/sigh/lib/sigh/runner.rb +285 -0
- data/snapshot/README.md +17 -0
- data/snapshot/lib/assets/SnapfileTemplate +34 -0
- data/snapshot/lib/assets/SnapfileTemplate.swift +41 -0
- data/snapshot/lib/assets/SnapshotHelper.swift +303 -0
- data/snapshot/lib/assets/SnapshotHelperXcode8.swift +172 -0
- data/snapshot/lib/snapshot.rb +19 -0
- data/snapshot/lib/snapshot/collector.rb +140 -0
- data/snapshot/lib/snapshot/commands_generator.rb +115 -0
- data/snapshot/lib/snapshot/dependency_checker.rb +69 -0
- data/snapshot/lib/snapshot/detect_values.rb +71 -0
- data/snapshot/lib/snapshot/error_handler.rb +24 -0
- data/snapshot/lib/snapshot/fixes/README.md +5 -0
- data/snapshot/lib/snapshot/fixes/hardware_keyboard_fix.rb +27 -0
- data/snapshot/lib/snapshot/fixes/simulator_zoom_fix.rb +27 -0
- data/snapshot/lib/snapshot/latest_os_version.rb +34 -0
- data/snapshot/lib/snapshot/module.rb +44 -0
- data/snapshot/lib/snapshot/options.rb +223 -0
- data/snapshot/lib/snapshot/page.html.erb +285 -0
- data/snapshot/lib/snapshot/reports_generator.rb +125 -0
- data/snapshot/lib/snapshot/reset_simulators.rb +122 -0
- data/snapshot/lib/snapshot/runner.rb +146 -0
- data/snapshot/lib/snapshot/screenshot_flatten.rb +20 -0
- data/snapshot/lib/snapshot/screenshot_rotate.rb +36 -0
- data/snapshot/lib/snapshot/setup.rb +58 -0
- data/snapshot/lib/snapshot/simulator_launchers/launcher_configuration.rb +54 -0
- data/snapshot/lib/snapshot/simulator_launchers/simulator_launcher.rb +215 -0
- data/snapshot/lib/snapshot/simulator_launchers/simulator_launcher_base.rb +143 -0
- data/snapshot/lib/snapshot/simulator_launchers/simulator_launcher_xcode_8.rb +113 -0
- data/snapshot/lib/snapshot/test_command_generator.rb +102 -0
- data/snapshot/lib/snapshot/test_command_generator_base.rb +117 -0
- data/snapshot/lib/snapshot/test_command_generator_xcode_8.rb +67 -0
- data/snapshot/lib/snapshot/update.rb +36 -0
- data/spaceship/README.md +183 -0
- data/spaceship/lib/assets/displayFamilies.json +181 -0
- data/spaceship/lib/assets/languageMapping.json +306 -0
- data/spaceship/lib/assets/languageMappingReadable.json +30 -0
- data/spaceship/lib/spaceship.rb +26 -0
- data/spaceship/lib/spaceship/babosa_fix.rb +30 -0
- data/spaceship/lib/spaceship/base.rb +310 -0
- data/spaceship/lib/spaceship/client.rb +892 -0
- data/spaceship/lib/spaceship/commands_generator.rb +52 -0
- data/spaceship/lib/spaceship/connect_api.rb +47 -0
- data/spaceship/lib/spaceship/connect_api/client.rb +178 -0
- data/spaceship/lib/spaceship/connect_api/model.rb +152 -0
- data/spaceship/lib/spaceship/connect_api/models/app.rb +97 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_app_localization.rb +28 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_app_review_detail.rb +32 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_app_review_submission.rb +26 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_build_localization.rb +20 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_build_metric.rb +24 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_group.rb +41 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_tester.rb +56 -0
- data/spaceship/lib/spaceship/connect_api/models/beta_tester_metric.rb +43 -0
- data/spaceship/lib/spaceship/connect_api/models/build.rb +144 -0
- data/spaceship/lib/spaceship/connect_api/models/build_beta_detail.rb +56 -0
- data/spaceship/lib/spaceship/connect_api/models/build_delivery.rb +36 -0
- data/spaceship/lib/spaceship/connect_api/models/bundle_id.rb +46 -0
- data/spaceship/lib/spaceship/connect_api/models/bundle_id_capability.rb +37 -0
- data/spaceship/lib/spaceship/connect_api/models/certificate.rb +48 -0
- data/spaceship/lib/spaceship/connect_api/models/device.rb +53 -0
- data/spaceship/lib/spaceship/connect_api/models/pre_release_version.rb +20 -0
- data/spaceship/lib/spaceship/connect_api/models/profile.rb +60 -0
- data/spaceship/lib/spaceship/connect_api/models/user.rb +50 -0
- data/spaceship/lib/spaceship/connect_api/provisioning/client.rb +59 -0
- data/spaceship/lib/spaceship/connect_api/provisioning/provisioning.rb +48 -0
- data/spaceship/lib/spaceship/connect_api/response.rb +67 -0
- data/spaceship/lib/spaceship/connect_api/testflight/client.rb +33 -0
- data/spaceship/lib/spaceship/connect_api/testflight/testflight.rb +322 -0
- data/spaceship/lib/spaceship/connect_api/token.rb +56 -0
- data/spaceship/lib/spaceship/connect_api/users/client.rb +33 -0
- data/spaceship/lib/spaceship/connect_api/users/users.rb +16 -0
- data/spaceship/lib/spaceship/du/du_client.rb +134 -0
- data/spaceship/lib/spaceship/du/upload_file.rb +58 -0
- data/spaceship/lib/spaceship/du/utilities.rb +99 -0
- data/spaceship/lib/spaceship/errors.rb +79 -0
- data/spaceship/lib/spaceship/globals.rb +12 -0
- data/spaceship/lib/spaceship/helper/net_http_generic_request.rb +14 -0
- data/spaceship/lib/spaceship/helper/plist_middleware.rb +16 -0
- data/spaceship/lib/spaceship/helper/rels_middleware.rb +28 -0
- data/spaceship/lib/spaceship/launcher.rb +99 -0
- data/spaceship/lib/spaceship/module.rb +8 -0
- data/spaceship/lib/spaceship/playground.rb +75 -0
- data/spaceship/lib/spaceship/portal/app.rb +192 -0
- data/spaceship/lib/spaceship/portal/app_group.rb +73 -0
- data/spaceship/lib/spaceship/portal/app_service.rb +87 -0
- data/spaceship/lib/spaceship/portal/certificate.rb +351 -0
- data/spaceship/lib/spaceship/portal/cloud_container.rb +74 -0
- data/spaceship/lib/spaceship/portal/device.rb +191 -0
- data/spaceship/lib/spaceship/portal/invite.rb +53 -0
- data/spaceship/lib/spaceship/portal/key.rb +101 -0
- data/spaceship/lib/spaceship/portal/legacy_wrapper.rb +21 -0
- data/spaceship/lib/spaceship/portal/merchant.rb +86 -0
- data/spaceship/lib/spaceship/portal/passbook.rb +78 -0
- data/spaceship/lib/spaceship/portal/person.rb +62 -0
- data/spaceship/lib/spaceship/portal/persons.rb +67 -0
- data/spaceship/lib/spaceship/portal/portal.rb +17 -0
- data/spaceship/lib/spaceship/portal/portal_base.rb +16 -0
- data/spaceship/lib/spaceship/portal/portal_client.rb +848 -0
- data/spaceship/lib/spaceship/portal/provisioning_profile.rb +559 -0
- data/spaceship/lib/spaceship/portal/provisioning_profile_template.rb +52 -0
- data/spaceship/lib/spaceship/portal/spaceship.rb +154 -0
- data/spaceship/lib/spaceship/portal/ui/select_team.rb +116 -0
- data/spaceship/lib/spaceship/portal/website_push.rb +88 -0
- data/spaceship/lib/spaceship/provider.rb +13 -0
- data/spaceship/lib/spaceship/spaceauth_runner.rb +74 -0
- data/spaceship/lib/spaceship/test_flight.rb +10 -0
- data/spaceship/lib/spaceship/test_flight/app_test_info.rb +40 -0
- data/spaceship/lib/spaceship/test_flight/base.rb +39 -0
- data/spaceship/lib/spaceship/test_flight/beta_review_info.rb +19 -0
- data/spaceship/lib/spaceship/test_flight/build.rb +239 -0
- data/spaceship/lib/spaceship/test_flight/build_trains.rb +46 -0
- data/spaceship/lib/spaceship/test_flight/client.rb +361 -0
- data/spaceship/lib/spaceship/test_flight/export_compliance.rb +12 -0
- data/spaceship/lib/spaceship/test_flight/group.rb +131 -0
- data/spaceship/lib/spaceship/test_flight/test_info.rb +56 -0
- data/spaceship/lib/spaceship/test_flight/tester.rb +136 -0
- data/spaceship/lib/spaceship/tunes/app_analytics.rb +168 -0
- data/spaceship/lib/spaceship/tunes/app_details.rb +147 -0
- data/spaceship/lib/spaceship/tunes/app_image.rb +51 -0
- data/spaceship/lib/spaceship/tunes/app_ratings.rb +58 -0
- data/spaceship/lib/spaceship/tunes/app_review.rb +55 -0
- data/spaceship/lib/spaceship/tunes/app_review_attachment.rb +49 -0
- data/spaceship/lib/spaceship/tunes/app_screenshot.rb +12 -0
- data/spaceship/lib/spaceship/tunes/app_status.rb +63 -0
- data/spaceship/lib/spaceship/tunes/app_submission.rb +161 -0
- data/spaceship/lib/spaceship/tunes/app_trailer.rb +64 -0
- data/spaceship/lib/spaceship/tunes/app_version.rb +961 -0
- data/spaceship/lib/spaceship/tunes/app_version_common.rb +35 -0
- data/spaceship/lib/spaceship/tunes/app_version_generated_promocodes.rb +28 -0
- data/spaceship/lib/spaceship/tunes/app_version_history.rb +52 -0
- data/spaceship/lib/spaceship/tunes/app_version_promocodes.rb +27 -0
- data/spaceship/lib/spaceship/tunes/app_version_ref.rb +15 -0
- data/spaceship/lib/spaceship/tunes/app_version_states_history.rb +27 -0
- data/spaceship/lib/spaceship/tunes/application.rb +448 -0
- data/spaceship/lib/spaceship/tunes/availability.rb +132 -0
- data/spaceship/lib/spaceship/tunes/b2b_user.rb +30 -0
- data/spaceship/lib/spaceship/tunes/build.rb +263 -0
- data/spaceship/lib/spaceship/tunes/build_details.rb +42 -0
- data/spaceship/lib/spaceship/tunes/build_train.rb +181 -0
- data/spaceship/lib/spaceship/tunes/developer_response.rb +23 -0
- data/spaceship/lib/spaceship/tunes/device_type.rb +17 -0
- data/spaceship/lib/spaceship/tunes/display_family.rb +84 -0
- data/spaceship/lib/spaceship/tunes/errors.rb +17 -0
- data/spaceship/lib/spaceship/tunes/iap.rb +151 -0
- data/spaceship/lib/spaceship/tunes/iap_detail.rb +243 -0
- data/spaceship/lib/spaceship/tunes/iap_families.rb +57 -0
- data/spaceship/lib/spaceship/tunes/iap_family_details.rb +73 -0
- data/spaceship/lib/spaceship/tunes/iap_family_list.rb +28 -0
- data/spaceship/lib/spaceship/tunes/iap_list.rb +76 -0
- data/spaceship/lib/spaceship/tunes/iap_status.rb +52 -0
- data/spaceship/lib/spaceship/tunes/iap_subscription_pricing_info.rb +46 -0
- data/spaceship/lib/spaceship/tunes/iap_subscription_pricing_tier.rb +25 -0
- data/spaceship/lib/spaceship/tunes/iap_type.rb +45 -0
- data/spaceship/lib/spaceship/tunes/language_converter.rb +65 -0
- data/spaceship/lib/spaceship/tunes/language_item.rb +61 -0
- data/spaceship/lib/spaceship/tunes/legacy_wrapper.rb +13 -0
- data/spaceship/lib/spaceship/tunes/member.rb +84 -0
- data/spaceship/lib/spaceship/tunes/members.rb +35 -0
- data/spaceship/lib/spaceship/tunes/pricing_info.rb +42 -0
- data/spaceship/lib/spaceship/tunes/pricing_tier.rb +25 -0
- data/spaceship/lib/spaceship/tunes/sandbox_tester.rb +74 -0
- data/spaceship/lib/spaceship/tunes/spaceship.rb +39 -0
- data/spaceship/lib/spaceship/tunes/territory.rb +39 -0
- data/spaceship/lib/spaceship/tunes/transit_app_file.rb +23 -0
- data/spaceship/lib/spaceship/tunes/tunes.rb +35 -0
- data/spaceship/lib/spaceship/tunes/tunes_base.rb +17 -0
- data/spaceship/lib/spaceship/tunes/tunes_client.rb +1572 -0
- data/spaceship/lib/spaceship/tunes/user_detail.rb +15 -0
- data/spaceship/lib/spaceship/tunes/version_set.rb +26 -0
- data/spaceship/lib/spaceship/two_step_or_factor_client.rb +287 -0
- data/spaceship/lib/spaceship/ui.rb +27 -0
- data/supply/README.md +17 -0
- data/supply/lib/supply.rb +30 -0
- data/supply/lib/supply/apk_listing.rb +14 -0
- data/supply/lib/supply/client.rb +458 -0
- data/supply/lib/supply/commands_generator.rb +66 -0
- data/supply/lib/supply/languages.rb +88 -0
- data/supply/lib/supply/listing.rb +32 -0
- data/supply/lib/supply/options.rb +266 -0
- data/supply/lib/supply/reader.rb +25 -0
- data/supply/lib/supply/setup.rb +112 -0
- data/supply/lib/supply/uploader.rb +349 -0
- metadata +1772 -0
@@ -0,0 +1,30 @@
|
|
1
|
+
{
|
2
|
+
"Brazilian Portuguese": "Brazilian Portuguese",
|
3
|
+
"Danish": "Danish",
|
4
|
+
"Dutch": "Dutch",
|
5
|
+
"English": "English",
|
6
|
+
"English_Australian": "Australian English",
|
7
|
+
"English_CA": "Canadian English",
|
8
|
+
"English_UK": "UK English",
|
9
|
+
"Finnish": "Finnish",
|
10
|
+
"French": "French",
|
11
|
+
"French_CA": "Canadian French",
|
12
|
+
"German": "German",
|
13
|
+
"Greek": "Greek",
|
14
|
+
"Indonesian": "Indonesian",
|
15
|
+
"Italian": "Italian",
|
16
|
+
"Japanese": "Japanese",
|
17
|
+
"Korean": "Korean",
|
18
|
+
"Malay": "Malay",
|
19
|
+
"Norwegian": "Norwegian",
|
20
|
+
"Portuguese": "Portuguese",
|
21
|
+
"Russian": "Russian",
|
22
|
+
"Simplified Chinese": "Simplified Chinese",
|
23
|
+
"Spanish": "Spanish",
|
24
|
+
"Spanish_MX": "Mexican Spanish",
|
25
|
+
"Swedish": "Swedish",
|
26
|
+
"Thai": "Thai",
|
27
|
+
"Traditional Chinese": "Traditional Chinese",
|
28
|
+
"Turkish": "Turkish",
|
29
|
+
"Vietnamese": "Vietnamese"
|
30
|
+
}
|
@@ -0,0 +1,26 @@
|
|
1
|
+
require_relative 'spaceship/globals'
|
2
|
+
require_relative 'spaceship/base'
|
3
|
+
require_relative 'spaceship/client'
|
4
|
+
require_relative 'spaceship/provider'
|
5
|
+
require_relative 'spaceship/launcher'
|
6
|
+
|
7
|
+
# Dev Portal
|
8
|
+
require_relative 'spaceship/portal/portal'
|
9
|
+
require_relative 'spaceship/portal/spaceship'
|
10
|
+
|
11
|
+
# App Store Connect
|
12
|
+
require_relative 'spaceship/tunes/tunes'
|
13
|
+
require_relative 'spaceship/tunes/spaceship'
|
14
|
+
require_relative 'spaceship/test_flight'
|
15
|
+
require_relative 'spaceship/connect_api'
|
16
|
+
require_relative 'spaceship/spaceauth_runner'
|
17
|
+
|
18
|
+
require_relative 'spaceship/module'
|
19
|
+
|
20
|
+
# Support for legacy wrappers
|
21
|
+
require_relative 'spaceship/portal/legacy_wrapper'
|
22
|
+
require_relative 'spaceship/tunes/legacy_wrapper'
|
23
|
+
|
24
|
+
# For basic user inputs
|
25
|
+
require 'highline/import'
|
26
|
+
require 'colored'
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# Babosa has a conflict with the unicode-string_width gem. unicode-string_width defines
|
2
|
+
# a module called `Unicode`, but Babosa uses the presence of this constant as
|
3
|
+
# the sign that it should try to require the `unicode` gem, which will not be present.
|
4
|
+
#
|
5
|
+
# We don't want to introduce the `unicode` gem because it depends on native extensions.
|
6
|
+
#
|
7
|
+
# This works around the possibility that the unicode-string_width gem may already be
|
8
|
+
# loaded by temporarily undefining the `Unicode` constant while we load Babosa,
|
9
|
+
# then restoring it to its previous state if necessary.
|
10
|
+
#
|
11
|
+
# Can be removed once https://github.com/norman/babosa/pull/42 is merged and released
|
12
|
+
class BabosaFix
|
13
|
+
def apply
|
14
|
+
unicode_removed = false
|
15
|
+
|
16
|
+
if defined? Unicode
|
17
|
+
orig_unicode = Unicode
|
18
|
+
Object.send(:remove_const, :Unicode)
|
19
|
+
unicode_removed = true
|
20
|
+
end
|
21
|
+
|
22
|
+
require 'babosa'
|
23
|
+
|
24
|
+
if unicode_removed
|
25
|
+
Object.send(:const_set, :Unicode, orig_unicode)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
BabosaFix.new.apply
|
@@ -0,0 +1,310 @@
|
|
1
|
+
require_relative 'globals'
|
2
|
+
|
3
|
+
module Spaceship
|
4
|
+
##
|
5
|
+
# Spaceship::Base is the superclass for models in Apple Developer Portal.
|
6
|
+
# It's mainly responsible for mapping responses to objects.
|
7
|
+
#
|
8
|
+
# A class-level attribute `client` is used to maintain the spaceship which we
|
9
|
+
# are using to talk to ADP.
|
10
|
+
#
|
11
|
+
# Example of creating a new ADP model:
|
12
|
+
#
|
13
|
+
# class Widget < Spaceship::Base
|
14
|
+
# attr_accessor :id, :name, :foo_bar, :wiz_baz
|
15
|
+
# attr_mapping({
|
16
|
+
# 'name' => :name,
|
17
|
+
# 'fooBar' => :foo_bar,
|
18
|
+
# 'wizBaz' => :wiz_baz
|
19
|
+
# })
|
20
|
+
# end
|
21
|
+
#
|
22
|
+
# When you want to instantiate a model pass in the parsed response: `Widget.new(widget_json)`
|
23
|
+
class Base
|
24
|
+
class DataHash
|
25
|
+
include Enumerable
|
26
|
+
|
27
|
+
def initialize(hash)
|
28
|
+
@hash = hash || {}
|
29
|
+
end
|
30
|
+
|
31
|
+
def key?(key)
|
32
|
+
@hash.key?(key)
|
33
|
+
end
|
34
|
+
|
35
|
+
def get(*keys)
|
36
|
+
lookup(keys)
|
37
|
+
end
|
38
|
+
|
39
|
+
alias [] get
|
40
|
+
|
41
|
+
def set(keys, value)
|
42
|
+
raise "'keys' must be an array, got #{keys.class} instead" unless keys.kind_of?(Array)
|
43
|
+
last = keys.pop
|
44
|
+
ref = lookup(keys) || @hash
|
45
|
+
ref[last] = value
|
46
|
+
end
|
47
|
+
|
48
|
+
def delete(key)
|
49
|
+
@hash.delete(key)
|
50
|
+
end
|
51
|
+
|
52
|
+
def lookup(keys)
|
53
|
+
head, *tail = *keys
|
54
|
+
if tail.empty?
|
55
|
+
@hash[head]
|
56
|
+
else
|
57
|
+
DataHash.new(@hash[head]).lookup(tail)
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def each(&block)
|
62
|
+
@hash.each(&block)
|
63
|
+
end
|
64
|
+
|
65
|
+
def to_json(*a)
|
66
|
+
h = @hash.dup
|
67
|
+
h.delete(:application)
|
68
|
+
h.to_json(*a)
|
69
|
+
rescue JSON::GeneratorError => e
|
70
|
+
puts("Failed to jsonify #{h} (#{a})") if Spaceship::Globals.verbose?
|
71
|
+
raise e
|
72
|
+
end
|
73
|
+
|
74
|
+
def to_h
|
75
|
+
@hash.dup
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
class << self
|
80
|
+
attr_accessor :client
|
81
|
+
|
82
|
+
##
|
83
|
+
# The client used to make requests.
|
84
|
+
# @return (Spaceship::Client) Defaults to the singleton
|
85
|
+
def client
|
86
|
+
raise "`client` must be implemented in subclasses"
|
87
|
+
end
|
88
|
+
|
89
|
+
##
|
90
|
+
# Sets client and returns self for chaining.
|
91
|
+
# @return (Spaceship::Base)
|
92
|
+
# rubocop:disable Style/AccessorMethodName
|
93
|
+
def set_client(client)
|
94
|
+
self.client = client
|
95
|
+
self
|
96
|
+
end
|
97
|
+
# rubocop:enable Style/AccessorMethodName
|
98
|
+
|
99
|
+
##
|
100
|
+
# Binds attributes getters and setters to underlying data returned from the API.
|
101
|
+
# Setting any properties will alter the `raw_data` hash.
|
102
|
+
#
|
103
|
+
# @return (Module) with the mapped getters and setters defined. Can be `include`, `extend`, or `prepend` into a class or object
|
104
|
+
def mapping_module(attr_mapping)
|
105
|
+
Module.new do
|
106
|
+
attr_mapping.each do |source, dest|
|
107
|
+
getter = dest.to_sym
|
108
|
+
setter = "#{dest}=".to_sym
|
109
|
+
|
110
|
+
define_method(getter) do
|
111
|
+
raw_data.get(*source.split('.'))
|
112
|
+
end
|
113
|
+
|
114
|
+
define_method(setter) do |value|
|
115
|
+
self.raw_data ||= DataHash.new({})
|
116
|
+
raw_data.set(source.split('.'), value)
|
117
|
+
end
|
118
|
+
end
|
119
|
+
end
|
120
|
+
end
|
121
|
+
|
122
|
+
##
|
123
|
+
# Defines the attribute mapping between the response from Apple and our model objects.
|
124
|
+
# Keys are to match keys in the response and the values are to match attributes on the model.
|
125
|
+
#
|
126
|
+
# Example of using `attr_mapping`
|
127
|
+
#
|
128
|
+
# class Widget < Spaceship::Base
|
129
|
+
# attr_accessor :id, :name, :foo_bar, :wiz_baz
|
130
|
+
# attr_mapping({
|
131
|
+
# 'name' => :name,
|
132
|
+
# 'fooBar' => :foo_bar,
|
133
|
+
# 'wizBaz' => :wiz_baz
|
134
|
+
# })
|
135
|
+
# end
|
136
|
+
def attr_mapping(attr_map = nil)
|
137
|
+
if attr_map
|
138
|
+
@attr_mapping = attr_map
|
139
|
+
@attr_mapping.values.each do |method_name|
|
140
|
+
getter = method_name.to_sym
|
141
|
+
setter = "#{method_name}=".to_sym
|
142
|
+
|
143
|
+
# Seems like the `public_instance_methods.include?` doesn't always work
|
144
|
+
# More context https://github.com/fastlane/fastlane/issues/11481
|
145
|
+
# That's why we have the `begin` `rescue` code here
|
146
|
+
begin
|
147
|
+
remove_method(getter) if public_instance_methods.include?(getter)
|
148
|
+
rescue NameError
|
149
|
+
end
|
150
|
+
begin
|
151
|
+
remove_method(setter) if public_instance_methods.include?(setter)
|
152
|
+
rescue NameError
|
153
|
+
end
|
154
|
+
end
|
155
|
+
include(mapping_module(@attr_mapping))
|
156
|
+
else
|
157
|
+
begin
|
158
|
+
@attr_mapping ||= ancestors[1].attr_mapping
|
159
|
+
rescue NoMethodError
|
160
|
+
rescue NameError
|
161
|
+
end
|
162
|
+
end
|
163
|
+
return @attr_mapping
|
164
|
+
end
|
165
|
+
|
166
|
+
##
|
167
|
+
# Call a method to return a subclass constant.
|
168
|
+
#
|
169
|
+
# If `method_sym` is an underscored name of a class,
|
170
|
+
# return the class with the current client passed into it.
|
171
|
+
# If the method does not match, NoMethodError is raised.
|
172
|
+
#
|
173
|
+
# Example:
|
174
|
+
#
|
175
|
+
# Certificate.production_push
|
176
|
+
# #=> Certificate::ProductionPush
|
177
|
+
#
|
178
|
+
# ProvisioningProfile.ad_hoc
|
179
|
+
# #=> ProvisioningProfile::AdHoc
|
180
|
+
#
|
181
|
+
# ProvisioningProfile.some_other_method
|
182
|
+
# #=> NoMethodError: undefined method `some_other_method' for ProvisioningProfile
|
183
|
+
def method_missing(method_sym, *args, &block)
|
184
|
+
module_name = method_sym.to_s
|
185
|
+
module_name.sub!(/^[a-z\d]/) { $&.upcase }
|
186
|
+
module_name.gsub!(%r{(?:_|(/))([a-z\d])}) { $2.upcase }
|
187
|
+
if const_defined?(module_name)
|
188
|
+
klass = const_get(module_name)
|
189
|
+
klass.set_client(@client)
|
190
|
+
else
|
191
|
+
super
|
192
|
+
end
|
193
|
+
end
|
194
|
+
|
195
|
+
##
|
196
|
+
# The factory class-method. This should only be used or overridden in very specific use-cases
|
197
|
+
#
|
198
|
+
# The only time it makes sense to use or override this method is when we want a base class
|
199
|
+
# to return a sub-class based on attributes.
|
200
|
+
#
|
201
|
+
# Here, we define the method to be the same as `Spaceship::Base.new(attrs)`, be it should
|
202
|
+
# be used only by classes that override it.
|
203
|
+
#
|
204
|
+
# Example:
|
205
|
+
#
|
206
|
+
# Certificate.factory(attrs)
|
207
|
+
# #=> #<PushCertificate ... >
|
208
|
+
#
|
209
|
+
def factory(attrs, existing_client = nil)
|
210
|
+
self.new(attrs, existing_client)
|
211
|
+
end
|
212
|
+
end
|
213
|
+
|
214
|
+
##
|
215
|
+
# @return (Spaceship::Client) The current spaceship client used by the model to make requests.
|
216
|
+
attr_reader :client
|
217
|
+
|
218
|
+
##
|
219
|
+
# @return (Hash/Array) Holds the raw data we got from Apple's
|
220
|
+
# server to use it later
|
221
|
+
attr_accessor :raw_data
|
222
|
+
|
223
|
+
##
|
224
|
+
# The initialize method accepts a parsed response from Apple and sets all
|
225
|
+
# attributes that are defined by `attr_mapping`
|
226
|
+
#
|
227
|
+
# Do not override `initialize` in your own models.
|
228
|
+
def initialize(attrs = {}, existing_client = nil)
|
229
|
+
attrs.each do |key, val|
|
230
|
+
self.send("#{key}=", val) if respond_to?("#{key}=")
|
231
|
+
end
|
232
|
+
self.raw_data = DataHash.new(attrs)
|
233
|
+
@client = existing_client || self.class.client
|
234
|
+
self.setup
|
235
|
+
end
|
236
|
+
|
237
|
+
# This method can be used by subclasses to do additional initialisation
|
238
|
+
# using the `raw_data`
|
239
|
+
def setup; end
|
240
|
+
|
241
|
+
#####################################################
|
242
|
+
# @!group Storing the `attr_accessor`
|
243
|
+
#####################################################
|
244
|
+
|
245
|
+
# From https://stackoverflow.com/questions/2487333/fastest-one-liner-way-to-list-attr-accessors-in-ruby
|
246
|
+
# This will store a list of defined attr_accessors to easily access them when inspecting the values
|
247
|
+
def self.attr_accessor(*vars)
|
248
|
+
@attributes ||= []
|
249
|
+
@attributes.concat(vars)
|
250
|
+
super(*vars)
|
251
|
+
end
|
252
|
+
|
253
|
+
def self.attributes
|
254
|
+
@attributes ||= []
|
255
|
+
par = []
|
256
|
+
|
257
|
+
par = (self.superclass.attributes || []) unless self == Base
|
258
|
+
|
259
|
+
@attributes + par
|
260
|
+
end
|
261
|
+
|
262
|
+
def attributes
|
263
|
+
self.class.attributes
|
264
|
+
end
|
265
|
+
|
266
|
+
#####################################################
|
267
|
+
# @!group Inspect related code
|
268
|
+
#####################################################
|
269
|
+
|
270
|
+
def inspect
|
271
|
+
# To avoid circular references, we keep track of the references
|
272
|
+
# of all objects already inspected from the first call to inspect
|
273
|
+
# in this call stack
|
274
|
+
# We use a Thread local storage for multi-thread friendliness
|
275
|
+
thread = Thread.current
|
276
|
+
tree_root = thread[:inspected_objects].nil?
|
277
|
+
thread[:inspected_objects] = Set.new if tree_root
|
278
|
+
|
279
|
+
if thread[:inspected_objects].include?(self)
|
280
|
+
# already inspected objects have a default value,
|
281
|
+
# let's follow Ruby's convention for circular references
|
282
|
+
value = "#<Object ...>"
|
283
|
+
else
|
284
|
+
thread[:inspected_objects].add(self)
|
285
|
+
begin
|
286
|
+
value = inspect_value
|
287
|
+
ensure
|
288
|
+
thread[:inspected_objects] = nil if tree_root
|
289
|
+
end
|
290
|
+
end
|
291
|
+
|
292
|
+
"<#{self.class.name} \n#{value}>"
|
293
|
+
end
|
294
|
+
|
295
|
+
def inspect_value
|
296
|
+
self.attributes.map do |k|
|
297
|
+
v = self.send(k).inspect
|
298
|
+
v.gsub!("\n", "\n\t") # to align nested elements
|
299
|
+
|
300
|
+
"\t#{k}=#{v}"
|
301
|
+
end.join(", \n")
|
302
|
+
end
|
303
|
+
|
304
|
+
def to_s
|
305
|
+
self.inspect
|
306
|
+
end
|
307
|
+
|
308
|
+
private :inspect_value
|
309
|
+
end
|
310
|
+
end
|
@@ -0,0 +1,892 @@
|
|
1
|
+
require 'faraday' # HTTP Client
|
2
|
+
require 'faraday-cookie_jar'
|
3
|
+
require 'faraday_middleware'
|
4
|
+
require 'logger'
|
5
|
+
require 'tmpdir'
|
6
|
+
require 'cgi'
|
7
|
+
require 'tempfile'
|
8
|
+
|
9
|
+
require 'fastlane/version'
|
10
|
+
require_relative 'babosa_fix'
|
11
|
+
require_relative 'helper/net_http_generic_request'
|
12
|
+
require_relative 'helper/plist_middleware'
|
13
|
+
require_relative 'helper/rels_middleware'
|
14
|
+
require_relative 'ui'
|
15
|
+
require_relative 'errors'
|
16
|
+
require_relative 'tunes/errors'
|
17
|
+
require_relative 'globals'
|
18
|
+
require_relative 'provider'
|
19
|
+
|
20
|
+
Faraday::Utils.default_params_encoder = Faraday::FlatParamsEncoder
|
21
|
+
|
22
|
+
module Spaceship
|
23
|
+
# rubocop:disable Metrics/ClassLength
|
24
|
+
class Client
|
25
|
+
PROTOCOL_VERSION = "QH65B2"
|
26
|
+
USER_AGENT = "Spaceship #{Fastlane::VERSION}"
|
27
|
+
AUTH_TYPES = ["sa", "hsa", "non-sa", "hsa2"]
|
28
|
+
|
29
|
+
attr_reader :client
|
30
|
+
|
31
|
+
# The user that is currently logged in
|
32
|
+
attr_accessor :user
|
33
|
+
|
34
|
+
# The email of the user that is currently logged in
|
35
|
+
attr_accessor :user_email
|
36
|
+
|
37
|
+
# The logger in which all requests are logged
|
38
|
+
# /tmp/spaceship[time]_[pid].log by default
|
39
|
+
attr_accessor :logger
|
40
|
+
|
41
|
+
attr_accessor :csrf_tokens
|
42
|
+
|
43
|
+
attr_accessor :provider
|
44
|
+
|
45
|
+
# legacy support
|
46
|
+
BasicPreferredInfoError = Spaceship::BasicPreferredInfoError
|
47
|
+
InvalidUserCredentialsError = Spaceship::InvalidUserCredentialsError
|
48
|
+
NoUserCredentialsError = Spaceship::NoUserCredentialsError
|
49
|
+
ProgramLicenseAgreementUpdated = Spaceship::ProgramLicenseAgreementUpdated
|
50
|
+
InsufficientPermissions = Spaceship::InsufficientPermissions
|
51
|
+
UnexpectedResponse = Spaceship::UnexpectedResponse
|
52
|
+
AppleTimeoutError = Spaceship::AppleTimeoutError
|
53
|
+
UnauthorizedAccessError = Spaceship::UnauthorizedAccessError
|
54
|
+
GatewayTimeoutError = Spaceship::GatewayTimeoutError
|
55
|
+
InternalServerError = Spaceship::InternalServerError
|
56
|
+
BadGatewayError = Spaceship::BadGatewayError
|
57
|
+
|
58
|
+
def self.hostname
|
59
|
+
raise "You must implement self.hostname"
|
60
|
+
end
|
61
|
+
|
62
|
+
#####################################################
|
63
|
+
# @!group Teams + User
|
64
|
+
#####################################################
|
65
|
+
|
66
|
+
# @return (Array) A list of all available teams
|
67
|
+
def teams
|
68
|
+
user_details_data['associatedAccounts'].sort_by do |team|
|
69
|
+
[
|
70
|
+
team['contentProvider']['name'],
|
71
|
+
team['contentProvider']['contentProviderId']
|
72
|
+
]
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
# Fetch the general information of the user, is used by various methods across spaceship
|
77
|
+
# Sample return value
|
78
|
+
# => {"associatedAccounts"=>
|
79
|
+
# [{"contentProvider"=>{"contentProviderId"=>11142800, "name"=>"Felix Krause", "contentProviderTypes"=>["Purple Software"]}, "roles"=>["Developer"], "lastLogin"=>1468784113000}],
|
80
|
+
# "sessionToken"=>{"dsId"=>"8501011116", "contentProviderId"=>18111111, "expirationDate"=>nil, "ipAddress"=>nil},
|
81
|
+
# "permittedActivities"=>
|
82
|
+
# {"EDIT"=>
|
83
|
+
# ["UserManagementSelf",
|
84
|
+
# "GameCenterTestData",
|
85
|
+
# "AppAddonCreation"],
|
86
|
+
# "REPORT"=>
|
87
|
+
# ["UserManagementSelf",
|
88
|
+
# "AppAddonCreation"],
|
89
|
+
# "VIEW"=>
|
90
|
+
# ["TestFlightAppExternalTesterManagement",
|
91
|
+
# ...
|
92
|
+
# "HelpGeneral",
|
93
|
+
# "HelpApplicationLoader"]},
|
94
|
+
# "preferredCurrencyCode"=>"EUR",
|
95
|
+
# "preferredCountryCode"=>nil,
|
96
|
+
# "countryOfOrigin"=>"AT",
|
97
|
+
# "isLocaleNameReversed"=>false,
|
98
|
+
# "feldsparToken"=>nil,
|
99
|
+
# "feldsparChannelName"=>nil,
|
100
|
+
# "hasPendingFeldsparBindingRequest"=>false,
|
101
|
+
# "isLegalUser"=>false,
|
102
|
+
# "userId"=>"1771111155",
|
103
|
+
# "firstname"=>"Detlef",
|
104
|
+
# "lastname"=>"Mueller",
|
105
|
+
# "isEmailInvalid"=>false,
|
106
|
+
# "hasContractInfo"=>false,
|
107
|
+
# "canEditITCUsersAndRoles"=>false,
|
108
|
+
# "canViewITCUsersAndRoles"=>true,
|
109
|
+
# "canEditIAPUsersAndRoles"=>false,
|
110
|
+
# "transporterEnabled"=>false,
|
111
|
+
# "contentProviderFeatures"=>["APP_SILOING", "PROMO_CODE_REDESIGN", ...],
|
112
|
+
# "contentProviderType"=>"Purple Software",
|
113
|
+
# "displayName"=>"Detlef",
|
114
|
+
# "contentProviderId"=>"18742800",
|
115
|
+
# "userFeatures"=>[],
|
116
|
+
# "visibility"=>true,
|
117
|
+
# "DYCVisibility"=>false,
|
118
|
+
# "contentProvider"=>"Felix Krause",
|
119
|
+
# "userName"=>"detlef@krausefx.com"}
|
120
|
+
def user_details_data
|
121
|
+
return @_cached_user_details if @_cached_user_details
|
122
|
+
r = request(:get, '/WebObjects/iTunesConnect.woa/ra/user/detail')
|
123
|
+
@_cached_user_details = parse_response(r, 'data')
|
124
|
+
end
|
125
|
+
|
126
|
+
# @return (String) The currently selected Team ID
|
127
|
+
def team_id
|
128
|
+
return @current_team_id if @current_team_id
|
129
|
+
|
130
|
+
if teams.count > 1
|
131
|
+
puts("The current user is in #{teams.count} teams. Pass a team ID or call `select_team` to choose a team. Using the first one for now.")
|
132
|
+
end
|
133
|
+
@current_team_id ||= teams[0]['contentProvider']['contentProviderId']
|
134
|
+
end
|
135
|
+
|
136
|
+
# Set a new team ID which will be used from now on
|
137
|
+
def team_id=(team_id)
|
138
|
+
# First, we verify the team actually exists, because otherwise iTC would return the
|
139
|
+
# following confusing error message
|
140
|
+
#
|
141
|
+
# invalid content provider id
|
142
|
+
#
|
143
|
+
available_teams = teams.collect do |team|
|
144
|
+
{
|
145
|
+
team_id: (team["contentProvider"] || {})["contentProviderId"],
|
146
|
+
team_name: (team["contentProvider"] || {})["name"]
|
147
|
+
}
|
148
|
+
end
|
149
|
+
|
150
|
+
result = available_teams.find do |available_team|
|
151
|
+
team_id.to_s == available_team[:team_id].to_s
|
152
|
+
end
|
153
|
+
|
154
|
+
unless result
|
155
|
+
error_string = "Could not set team ID to '#{team_id}', only found the following available teams:\n\n#{available_teams.map { |team| "- #{team[:team_id]} (#{team[:team_name]})" }.join("\n")}\n"
|
156
|
+
raise Tunes::Error.new, error_string
|
157
|
+
end
|
158
|
+
|
159
|
+
response = request(:post) do |req|
|
160
|
+
req.url("ra/v1/session/webSession")
|
161
|
+
req.body = {
|
162
|
+
contentProviderId: team_id,
|
163
|
+
dsId: user_detail_data.ds_id # https://github.com/fastlane/fastlane/issues/6711
|
164
|
+
}.to_json
|
165
|
+
req.headers['Content-Type'] = 'application/json'
|
166
|
+
end
|
167
|
+
|
168
|
+
handle_itc_response(response.body)
|
169
|
+
|
170
|
+
@current_team_id = team_id
|
171
|
+
end
|
172
|
+
|
173
|
+
# @return (Hash) Fetches all information of the currently used team
|
174
|
+
def team_information
|
175
|
+
teams.find do |t|
|
176
|
+
t['teamId'] == team_id
|
177
|
+
end
|
178
|
+
end
|
179
|
+
|
180
|
+
# @return (String) Fetches name from currently used team
|
181
|
+
def team_name
|
182
|
+
(team_information || {})['name']
|
183
|
+
end
|
184
|
+
|
185
|
+
#####################################################
|
186
|
+
# @!group Client Init
|
187
|
+
#####################################################
|
188
|
+
|
189
|
+
# Instantiates a client but with a cookie derived from another client.
|
190
|
+
#
|
191
|
+
# HACK: since the `@cookie` is not exposed, we use this hacky way of sharing the instance.
|
192
|
+
def self.client_with_authorization_from(another_client)
|
193
|
+
self.new(cookie: another_client.instance_variable_get(:@cookie), current_team_id: another_client.team_id)
|
194
|
+
end
|
195
|
+
|
196
|
+
def initialize(cookie: nil, current_team_id: nil)
|
197
|
+
options = {
|
198
|
+
request: {
|
199
|
+
timeout: (ENV["SPACESHIP_TIMEOUT"] || 300).to_i,
|
200
|
+
open_timeout: (ENV["SPACESHIP_TIMEOUT"] || 300).to_i
|
201
|
+
}
|
202
|
+
}
|
203
|
+
@current_team_id = current_team_id
|
204
|
+
@cookie = cookie || HTTP::CookieJar.new
|
205
|
+
@client = Faraday.new(self.class.hostname, options) do |c|
|
206
|
+
c.response(:json, content_type: /\bjson$/)
|
207
|
+
c.response(:xml, content_type: /\bxml$/)
|
208
|
+
c.response(:plist, content_type: /\bplist$/)
|
209
|
+
c.use(:cookie_jar, jar: @cookie)
|
210
|
+
c.use(FaradayMiddleware::RelsMiddleware)
|
211
|
+
c.adapter(Faraday.default_adapter)
|
212
|
+
|
213
|
+
if ENV['SPACESHIP_DEBUG']
|
214
|
+
# for debugging only
|
215
|
+
# This enables tracking of networking requests using Charles Web Proxy
|
216
|
+
c.proxy = "https://127.0.0.1:8888"
|
217
|
+
c.ssl[:verify_mode] = OpenSSL::SSL::VERIFY_NONE
|
218
|
+
elsif ENV["SPACESHIP_PROXY"]
|
219
|
+
c.proxy = ENV["SPACESHIP_PROXY"]
|
220
|
+
c.ssl[:verify_mode] = OpenSSL::SSL::VERIFY_NONE if ENV["SPACESHIP_PROXY_SSL_VERIFY_NONE"]
|
221
|
+
end
|
222
|
+
|
223
|
+
if ENV["DEBUG"]
|
224
|
+
puts("To run spaceship through a local proxy, use SPACESHIP_DEBUG")
|
225
|
+
end
|
226
|
+
end
|
227
|
+
end
|
228
|
+
|
229
|
+
#####################################################
|
230
|
+
# @!group Request Logger
|
231
|
+
#####################################################
|
232
|
+
|
233
|
+
# The logger in which all requests are logged
|
234
|
+
# /tmp/spaceship[time]_[pid]_["threadid"].log by default
|
235
|
+
def logger
|
236
|
+
unless @logger
|
237
|
+
if ENV["VERBOSE"]
|
238
|
+
@logger = Logger.new(STDOUT)
|
239
|
+
else
|
240
|
+
# Log to file by default
|
241
|
+
path = "/tmp/spaceship#{Time.now.to_i}_#{Process.pid}_#{Thread.current.object_id}.log"
|
242
|
+
@logger = Logger.new(path)
|
243
|
+
end
|
244
|
+
|
245
|
+
@logger.formatter = proc do |severity, datetime, progname, msg|
|
246
|
+
severity = format('%-5.5s', severity)
|
247
|
+
"#{severity} [#{datetime.strftime('%H:%M:%S')}]: #{msg}\n"
|
248
|
+
end
|
249
|
+
end
|
250
|
+
|
251
|
+
@logger
|
252
|
+
end
|
253
|
+
|
254
|
+
#####################################################
|
255
|
+
# @!group Session Cookie
|
256
|
+
#####################################################
|
257
|
+
|
258
|
+
##
|
259
|
+
# Return the session cookie.
|
260
|
+
#
|
261
|
+
# @return (String) the cookie-string in the RFC6265 format: https://tools.ietf.org/html/rfc6265#section-4.2.1
|
262
|
+
def cookie
|
263
|
+
@cookie.map(&:to_s).join(';')
|
264
|
+
end
|
265
|
+
|
266
|
+
def store_cookie(path: nil)
|
267
|
+
path ||= persistent_cookie_path
|
268
|
+
FileUtils.mkdir_p(File.expand_path("..", path))
|
269
|
+
|
270
|
+
# really important to specify the session to true
|
271
|
+
# otherwise myacinfo and more won't be stored
|
272
|
+
@cookie.save(path, :yaml, session: true)
|
273
|
+
return File.read(path)
|
274
|
+
end
|
275
|
+
|
276
|
+
# This is a duplicate method of fastlane_core/fastlane_core.rb#fastlane_user_dir
|
277
|
+
def fastlane_user_dir
|
278
|
+
path = File.expand_path(File.join(Dir.home, ".fastlane"))
|
279
|
+
FileUtils.mkdir_p(path) unless File.directory?(path)
|
280
|
+
return path
|
281
|
+
end
|
282
|
+
|
283
|
+
# Returns preferred path for storing cookie
|
284
|
+
# for two step verification.
|
285
|
+
def persistent_cookie_path
|
286
|
+
if ENV["SPACESHIP_COOKIE_PATH"]
|
287
|
+
path = File.expand_path(File.join(ENV["SPACESHIP_COOKIE_PATH"], "spaceship", self.user, "cookie"))
|
288
|
+
else
|
289
|
+
[File.join(self.fastlane_user_dir, "spaceship"), "~/.spaceship", "/var/tmp/spaceship", "#{Dir.tmpdir}/spaceship"].each do |dir|
|
290
|
+
dir_parts = File.split(dir)
|
291
|
+
if directory_accessible?(File.expand_path(dir_parts.first))
|
292
|
+
path = File.expand_path(File.join(dir, self.user, "cookie"))
|
293
|
+
break
|
294
|
+
end
|
295
|
+
end
|
296
|
+
end
|
297
|
+
|
298
|
+
return path
|
299
|
+
end
|
300
|
+
|
301
|
+
#####################################################
|
302
|
+
# @!group Automatic Paging
|
303
|
+
#####################################################
|
304
|
+
|
305
|
+
# The page size we want to request, defaults to 500
|
306
|
+
def page_size
|
307
|
+
@page_size ||= 500
|
308
|
+
end
|
309
|
+
|
310
|
+
# Handles the paging for you... for free
|
311
|
+
# Just pass a block and use the parameter as page number
|
312
|
+
def paging
|
313
|
+
page = 0
|
314
|
+
results = []
|
315
|
+
loop do
|
316
|
+
page += 1
|
317
|
+
current = yield(page)
|
318
|
+
|
319
|
+
results += current
|
320
|
+
|
321
|
+
break if (current || []).count < page_size # no more results
|
322
|
+
end
|
323
|
+
|
324
|
+
return results
|
325
|
+
end
|
326
|
+
|
327
|
+
#####################################################
|
328
|
+
# @!group Login and Team Selection
|
329
|
+
#####################################################
|
330
|
+
|
331
|
+
# Authenticates with Apple's web services. This method has to be called once
|
332
|
+
# to generate a valid session. The session will automatically be used from then
|
333
|
+
# on.
|
334
|
+
#
|
335
|
+
# This method will automatically use the username from the Appfile (if available)
|
336
|
+
# and fetch the password from the Keychain (if available)
|
337
|
+
#
|
338
|
+
# @param user (String) (optional): The username (usually the email address)
|
339
|
+
# @param password (String) (optional): The password
|
340
|
+
#
|
341
|
+
# @raise InvalidUserCredentialsError: raised if authentication failed
|
342
|
+
#
|
343
|
+
# @return (Spaceship::Client) The client the login method was called for
|
344
|
+
def self.login(user = nil, password = nil)
|
345
|
+
instance = self.new
|
346
|
+
if instance.login(user, password)
|
347
|
+
instance
|
348
|
+
else
|
349
|
+
raise InvalidUserCredentialsError.new, "Invalid User Credentials"
|
350
|
+
end
|
351
|
+
end
|
352
|
+
|
353
|
+
# Authenticates with Apple's web services. This method has to be called once
|
354
|
+
# to generate a valid session. The session will automatically be used from then
|
355
|
+
# on.
|
356
|
+
#
|
357
|
+
# This method will automatically use the username from the Appfile (if available)
|
358
|
+
# and fetch the password from the Keychain (if available)
|
359
|
+
#
|
360
|
+
# @param user (String) (optional): The username (usually the email address)
|
361
|
+
# @param password (String) (optional): The password
|
362
|
+
#
|
363
|
+
# @raise InvalidUserCredentialsError: raised if authentication failed
|
364
|
+
#
|
365
|
+
# @return (Spaceship::Client) The client the login method was called for
|
366
|
+
def login(user = nil, password = nil)
|
367
|
+
if user.to_s.empty? || password.to_s.empty?
|
368
|
+
require 'credentials_manager/account_manager'
|
369
|
+
|
370
|
+
puts("Reading keychain entry, because either user or password were empty") if Spaceship::Globals.verbose?
|
371
|
+
|
372
|
+
keychain_entry = CredentialsManager::AccountManager.new(user: user, password: password)
|
373
|
+
user ||= keychain_entry.user
|
374
|
+
password = keychain_entry.password
|
375
|
+
end
|
376
|
+
|
377
|
+
if user.to_s.strip.empty? || password.to_s.strip.empty?
|
378
|
+
raise NoUserCredentialsError.new, "No login data provided"
|
379
|
+
end
|
380
|
+
|
381
|
+
self.user = user
|
382
|
+
@password = password
|
383
|
+
begin
|
384
|
+
do_login(user, password) # calls `send_login_request` in sub class (which then will redirect back here to `send_shared_login_request`, below)
|
385
|
+
rescue InvalidUserCredentialsError => ex
|
386
|
+
raise ex unless keychain_entry
|
387
|
+
|
388
|
+
if keychain_entry.invalid_credentials
|
389
|
+
login(user)
|
390
|
+
else
|
391
|
+
raise ex
|
392
|
+
end
|
393
|
+
end
|
394
|
+
end
|
395
|
+
|
396
|
+
# This method is used for both the Apple Dev Portal and App Store Connect
|
397
|
+
# This will also handle 2 step verification and 2 factor authentication
|
398
|
+
#
|
399
|
+
# It is called in `send_login_request` of sub classes (which the method `login`, above, transferred over to via `do_login`)
|
400
|
+
def send_shared_login_request(user, password)
|
401
|
+
# Check if we have a cached/valid session
|
402
|
+
#
|
403
|
+
# Background:
|
404
|
+
# December 4th 2017 Apple introduced a rate limit - which is of course fine by itself -
|
405
|
+
# but unfortunately also rate limits successful logins. If you call multiple tools in a
|
406
|
+
# lane (e.g. call match 5 times), this would lock you out of the account for a while.
|
407
|
+
# By loading existing sessions and checking if they're valid, we're sending less login requests.
|
408
|
+
# More context on why this change was necessary https://github.com/fastlane/fastlane/pull/11108
|
409
|
+
#
|
410
|
+
# If there was a successful manual login before, we have a session on disk
|
411
|
+
if load_session_from_file
|
412
|
+
# Check if the session is still valid here
|
413
|
+
begin
|
414
|
+
# We use the olympus session to determine if the old session is still valid
|
415
|
+
# As this will raise an exception if the old session has expired
|
416
|
+
# If the old session is still valid, we don't have to do anything else in this method
|
417
|
+
# that's why we return true
|
418
|
+
return true if fetch_olympus_session
|
419
|
+
rescue
|
420
|
+
# If the `fetch_olympus_session` method raises an exception
|
421
|
+
# we'll land here, and therefore continue doing a full login process
|
422
|
+
# This happens if the session we loaded from the cache isn't valid any more
|
423
|
+
# which is common, as the session automatically invalidates after x hours (we don't know x)
|
424
|
+
# In this case we don't actually care about the exact exception, and why it was failing
|
425
|
+
# because either way, we'll have to do a fresh login, where we do the actual error handling
|
426
|
+
puts("Available session is not valid any more. Continuing with normal login.")
|
427
|
+
end
|
428
|
+
end
|
429
|
+
#
|
430
|
+
# The user can pass the session via environment variable (Mainly used in CI environments)
|
431
|
+
if load_session_from_env
|
432
|
+
# see above
|
433
|
+
begin
|
434
|
+
# see above
|
435
|
+
return true if fetch_olympus_session
|
436
|
+
rescue
|
437
|
+
puts("Session loaded from environment variable is not valid. Continuing with normal login.")
|
438
|
+
# see above
|
439
|
+
end
|
440
|
+
end
|
441
|
+
#
|
442
|
+
# After this point, we sure have no valid session any more and have to create a new one
|
443
|
+
#
|
444
|
+
|
445
|
+
data = {
|
446
|
+
accountName: user,
|
447
|
+
password: password,
|
448
|
+
rememberMe: true
|
449
|
+
}
|
450
|
+
|
451
|
+
begin
|
452
|
+
# The below workaround is only needed for 2 step verified machines
|
453
|
+
# Due to escaping of cookie values we have a little workaround here
|
454
|
+
# By default the cookie jar would generate the following header
|
455
|
+
# DES5c148...=HSARM.......xaA/O69Ws/CHfQ==SRVT
|
456
|
+
# However we need the following
|
457
|
+
# DES5c148...="HSARM.......xaA/O69Ws/CHfQ==SRVT"
|
458
|
+
# There is no way to get the cookie jar value with " around the value
|
459
|
+
# so we manually modify the cookie (only this one) to be properly escaped
|
460
|
+
# Afterwards we pass this value manually as a header
|
461
|
+
# It's not enough to just modify @cookie, it needs to be done after self.cookie
|
462
|
+
# as a string operation
|
463
|
+
important_cookie = @cookie.store.entries.find { |a| a.name.include?("DES") }
|
464
|
+
if important_cookie
|
465
|
+
modified_cookie = self.cookie # returns a string of all cookies
|
466
|
+
unescaped_important_cookie = "#{important_cookie.name}=#{important_cookie.value}"
|
467
|
+
escaped_important_cookie = "#{important_cookie.name}=\"#{important_cookie.value}\""
|
468
|
+
modified_cookie.gsub!(unescaped_important_cookie, escaped_important_cookie)
|
469
|
+
end
|
470
|
+
|
471
|
+
response = request(:post) do |req|
|
472
|
+
req.url("https://idmsa.apple.com/appleauth/auth/signin")
|
473
|
+
req.body = data.to_json
|
474
|
+
req.headers['Content-Type'] = 'application/json'
|
475
|
+
req.headers['X-Requested-With'] = 'XMLHttpRequest'
|
476
|
+
req.headers['X-Apple-Widget-Key'] = self.itc_service_key
|
477
|
+
req.headers['Accept'] = 'application/json, text/javascript'
|
478
|
+
req.headers["Cookie"] = modified_cookie if modified_cookie
|
479
|
+
end
|
480
|
+
rescue UnauthorizedAccessError
|
481
|
+
raise InvalidUserCredentialsError.new, "Invalid username and password combination. Used '#{user}' as the username."
|
482
|
+
end
|
483
|
+
|
484
|
+
# Now we know if the login is successful or if we need to do 2 factor
|
485
|
+
|
486
|
+
case response.status
|
487
|
+
when 403
|
488
|
+
raise InvalidUserCredentialsError.new, "Invalid username and password combination. Used '#{user}' as the username."
|
489
|
+
when 200
|
490
|
+
fetch_olympus_session
|
491
|
+
return response
|
492
|
+
when 409
|
493
|
+
# 2 step/factor is enabled for this account, first handle that
|
494
|
+
handle_two_step_or_factor(response)
|
495
|
+
# and then get the olympus session
|
496
|
+
fetch_olympus_session
|
497
|
+
return true
|
498
|
+
else
|
499
|
+
if (response.body || "").include?('invalid="true"')
|
500
|
+
# User Credentials are wrong
|
501
|
+
raise InvalidUserCredentialsError.new, "Invalid username and password combination. Used '#{user}' as the username."
|
502
|
+
elsif response.status == 412 && AUTH_TYPES.include?(response.body["authType"])
|
503
|
+
# Need to acknowledge Apple ID and Privacy statement - https://github.com/fastlane/fastlane/issues/12577
|
504
|
+
# Looking for status of 412 might be enough but might be safer to keep looking only at what is being reported
|
505
|
+
raise AppleIDAndPrivacyAcknowledgementNeeded.new, "Need to acknowledge to Apple's Apple ID and Privacy statement. Please manually log into https://appleid.apple.com (or https://appstoreconnect.apple.com) to acknowledge the statement."
|
506
|
+
elsif (response['Set-Cookie'] || "").include?("itctx")
|
507
|
+
raise "Looks like your Apple ID is not enabled for App Store Connect, make sure to be able to login online"
|
508
|
+
else
|
509
|
+
info = [response.body, response['Set-Cookie']]
|
510
|
+
raise Tunes::Error.new, info.join("\n")
|
511
|
+
end
|
512
|
+
end
|
513
|
+
end
|
514
|
+
|
515
|
+
# Get the `itctx` from the new (22nd May 2017) API endpoint "olympus"
|
516
|
+
# Update (29th March 2019) olympus migrates to new appstoreconnect API
|
517
|
+
def fetch_olympus_session
|
518
|
+
response = request(:get, "https://appstoreconnect.apple.com/olympus/v1/session")
|
519
|
+
body = response.body
|
520
|
+
if body
|
521
|
+
body = JSON.parse(body) if body.kind_of?(String)
|
522
|
+
user_map = body["user"]
|
523
|
+
if user_map
|
524
|
+
self.user_email = user_map["emailAddress"]
|
525
|
+
end
|
526
|
+
|
527
|
+
provider = body["provider"]
|
528
|
+
if provider
|
529
|
+
self.provider = Spaceship::Provider.new(provider_hash: provider)
|
530
|
+
return true
|
531
|
+
end
|
532
|
+
end
|
533
|
+
|
534
|
+
return false
|
535
|
+
end
|
536
|
+
|
537
|
+
def itc_service_key
|
538
|
+
return @service_key if @service_key
|
539
|
+
|
540
|
+
# Check if we have a local cache of the key
|
541
|
+
itc_service_key_path = "/tmp/spaceship_itc_service_key.txt"
|
542
|
+
return File.read(itc_service_key_path) if File.exist?(itc_service_key_path)
|
543
|
+
|
544
|
+
# Fixes issue https://github.com/fastlane/fastlane/issues/13281
|
545
|
+
# Even though we are using https://appstoreconnect.apple.com, the service key needs to still use a
|
546
|
+
# hostname through itunesconnect.apple.com
|
547
|
+
response = request(:get, "https://appstoreconnect.apple.com/olympus/v1/app/config?hostname=itunesconnect.apple.com")
|
548
|
+
@service_key = response.body["authServiceKey"].to_s
|
549
|
+
|
550
|
+
raise "Service key is empty" if @service_key.length == 0
|
551
|
+
|
552
|
+
# Cache the key locally
|
553
|
+
File.write(itc_service_key_path, @service_key)
|
554
|
+
|
555
|
+
return @service_key
|
556
|
+
rescue => ex
|
557
|
+
puts(ex.to_s)
|
558
|
+
raise AppleTimeoutError.new, "Could not receive latest API key from App Store Connect, this might be a server issue."
|
559
|
+
end
|
560
|
+
|
561
|
+
#####################################################
|
562
|
+
# @!group Session
|
563
|
+
#####################################################
|
564
|
+
|
565
|
+
def load_session_from_file
|
566
|
+
if File.exist?(persistent_cookie_path)
|
567
|
+
puts("Loading session from '#{persistent_cookie_path}'") if Spaceship::Globals.verbose?
|
568
|
+
@cookie.load(persistent_cookie_path)
|
569
|
+
return true
|
570
|
+
end
|
571
|
+
return false
|
572
|
+
end
|
573
|
+
|
574
|
+
def load_session_from_env
|
575
|
+
return if self.class.spaceship_session_env.to_s.length == 0
|
576
|
+
puts("Loading session from environment variable") if Spaceship::Globals.verbose?
|
577
|
+
|
578
|
+
file = Tempfile.new('cookie.yml')
|
579
|
+
file.write(self.class.spaceship_session_env.gsub("\\n", "\n"))
|
580
|
+
file.close
|
581
|
+
|
582
|
+
begin
|
583
|
+
@cookie.load(file.path)
|
584
|
+
rescue => ex
|
585
|
+
puts("Error loading session from environment")
|
586
|
+
puts("Make sure to pass the session in a valid format")
|
587
|
+
raise ex
|
588
|
+
ensure
|
589
|
+
file.unlink
|
590
|
+
end
|
591
|
+
end
|
592
|
+
|
593
|
+
# Fetch the session cookie from the environment
|
594
|
+
# (if exists)
|
595
|
+
def self.spaceship_session_env
|
596
|
+
ENV["FASTLANE_SESSION"] || ENV["SPACESHIP_SESSION"]
|
597
|
+
end
|
598
|
+
|
599
|
+
# Get contract messages from App Store Connect's "olympus" endpoint
|
600
|
+
def fetch_program_license_agreement_messages
|
601
|
+
all_messages = []
|
602
|
+
|
603
|
+
messages_request = request(:get, "https://appstoreconnect.apple.com/olympus/v1/contractMessages")
|
604
|
+
body = messages_request.body
|
605
|
+
if body
|
606
|
+
body = JSON.parse(body) if body.kind_of?(String)
|
607
|
+
body.map do |messages|
|
608
|
+
all_messages.push(messages["message"])
|
609
|
+
end
|
610
|
+
end
|
611
|
+
|
612
|
+
return all_messages
|
613
|
+
end
|
614
|
+
|
615
|
+
#####################################################
|
616
|
+
# @!group Helpers
|
617
|
+
#####################################################
|
618
|
+
|
619
|
+
def with_retry(tries = 5, &_block)
|
620
|
+
return yield
|
621
|
+
rescue \
|
622
|
+
Faraday::Error::ConnectionFailed,
|
623
|
+
Faraday::Error::TimeoutError, # New Faraday version: Faraday::TimeoutError => ex
|
624
|
+
BadGatewayError,
|
625
|
+
AppleTimeoutError,
|
626
|
+
GatewayTimeoutError => ex
|
627
|
+
tries -= 1
|
628
|
+
unless tries.zero?
|
629
|
+
msg = "Timeout received: '#{ex.class}', '#{ex.message}'. Retrying after 3 seconds (remaining: #{tries})..."
|
630
|
+
puts(msg) if Spaceship::Globals.verbose?
|
631
|
+
logger.warn(msg)
|
632
|
+
|
633
|
+
sleep(3) unless Object.const_defined?("SpecHelper")
|
634
|
+
retry
|
635
|
+
end
|
636
|
+
raise ex # re-raise the exception
|
637
|
+
rescue \
|
638
|
+
Faraday::ParsingError, # <h2>Internal Server Error</h2> with content type json
|
639
|
+
InternalServerError => ex
|
640
|
+
tries -= 1
|
641
|
+
unless tries.zero?
|
642
|
+
msg = "Internal Server Error received: '#{ex.class}', '#{ex.message}'. Retrying after 3 seconds (remaining: #{tries})..."
|
643
|
+
puts(msg) if Spaceship::Globals.verbose?
|
644
|
+
logger.warn(msg)
|
645
|
+
|
646
|
+
sleep(3) unless Object.const_defined?("SpecHelper")
|
647
|
+
retry
|
648
|
+
end
|
649
|
+
raise ex # re-raise the exception
|
650
|
+
rescue UnauthorizedAccessError => ex
|
651
|
+
if @loggedin && !(tries -= 1).zero?
|
652
|
+
msg = "Auth error received: '#{ex.class}', '#{ex.message}'. Login in again then retrying after 3 seconds (remaining: #{tries})..."
|
653
|
+
puts(msg) if Spaceship::Globals.verbose?
|
654
|
+
logger.warn(msg)
|
655
|
+
|
656
|
+
if self.class.spaceship_session_env.to_s.length > 0
|
657
|
+
raise UnauthorizedAccessError.new, "Authentication error, you passed an invalid session using the environment variable FASTLANE_SESSION or SPACESHIP_SESSION"
|
658
|
+
end
|
659
|
+
|
660
|
+
do_login(self.user, @password)
|
661
|
+
sleep(3) unless Object.const_defined?("SpecHelper")
|
662
|
+
retry
|
663
|
+
end
|
664
|
+
raise ex # re-raise the exception
|
665
|
+
end
|
666
|
+
|
667
|
+
# memorize the last csrf tokens from responses
|
668
|
+
def csrf_tokens
|
669
|
+
@csrf_tokens || {}
|
670
|
+
end
|
671
|
+
|
672
|
+
def request(method, url_or_path = nil, params = nil, headers = {}, auto_paginate = false, &block)
|
673
|
+
headers.merge!(csrf_tokens)
|
674
|
+
headers['User-Agent'] = USER_AGENT
|
675
|
+
|
676
|
+
# Before encoding the parameters, log them
|
677
|
+
log_request(method, url_or_path, params, headers, &block)
|
678
|
+
|
679
|
+
# form-encode the params only if there are params, and the block is not supplied.
|
680
|
+
# this is so that certain requests can be made using the block for more control
|
681
|
+
if method == :post && params && !block_given?
|
682
|
+
params, headers = encode_params(params, headers)
|
683
|
+
end
|
684
|
+
|
685
|
+
response = if auto_paginate
|
686
|
+
send_request_auto_paginate(method, url_or_path, params, headers, &block)
|
687
|
+
else
|
688
|
+
send_request(method, url_or_path, params, headers, &block)
|
689
|
+
end
|
690
|
+
|
691
|
+
return response
|
692
|
+
end
|
693
|
+
|
694
|
+
def parse_response(response, expected_key = nil)
|
695
|
+
if response.body
|
696
|
+
# If we have an `expected_key`, select that from response.body Hash
|
697
|
+
# Else, don't.
|
698
|
+
|
699
|
+
# the returned error message and info, is html encoded -> "issued" -> make this readable -> "issued"
|
700
|
+
response.body["userString"] = CGI.unescapeHTML(response.body["userString"]) if response.body["userString"]
|
701
|
+
response.body["resultString"] = CGI.unescapeHTML(response.body["resultString"]) if response.body["resultString"]
|
702
|
+
|
703
|
+
content = expected_key ? response.body[expected_key] : response.body
|
704
|
+
end
|
705
|
+
|
706
|
+
# if content (filled with whole body or just expected_key) is missing
|
707
|
+
if content.nil?
|
708
|
+
detect_most_common_errors_and_raise_exceptions(response.body) if response.body
|
709
|
+
raise UnexpectedResponse, response.body
|
710
|
+
# else if it is a hash and `resultString` includes `NotAllowed`
|
711
|
+
elsif content.kind_of?(Hash) && (content["resultString"] || "").include?("NotAllowed")
|
712
|
+
# example content when doing a Developer Portal action with not enough permission
|
713
|
+
# => {"responseId"=>"e5013d83-c5cb-4ba0-bb62-734a8d56007f",
|
714
|
+
# "resultCode"=>1200,
|
715
|
+
# "resultString"=>"webservice.certificate.downloadNotAllowed",
|
716
|
+
# "userString"=>"You are not permitted to download this certificate.",
|
717
|
+
# "creationTimestamp"=>"2017-01-26T22:44:13Z",
|
718
|
+
# "protocolVersion"=>"QH65B2",
|
719
|
+
# "userLocale"=>"en_US",
|
720
|
+
# "requestUrl"=>"https://developer.apple.com/services-account/QH65B2/account/ios/certificate/downloadCertificateContent.action",
|
721
|
+
# "httpCode"=>200}
|
722
|
+
raise_insufficient_permission_error!(additional_error_string: content["userString"])
|
723
|
+
else
|
724
|
+
store_csrf_tokens(response)
|
725
|
+
content
|
726
|
+
end
|
727
|
+
end
|
728
|
+
|
729
|
+
def detect_most_common_errors_and_raise_exceptions(body)
|
730
|
+
# Check if the failure is due to missing permissions (App Store Connect)
|
731
|
+
if body["messages"] && body["messages"]["error"].include?("Forbidden")
|
732
|
+
raise_insufficient_permission_error!
|
733
|
+
elsif body["messages"] && body["messages"]["error"].include?("insufficient privileges")
|
734
|
+
# Passing a specific `caller_location` here to make sure we return the correct method
|
735
|
+
# With the default location the error would say that `parse_response` is the caller
|
736
|
+
raise_insufficient_permission_error!(caller_location: 3)
|
737
|
+
elsif body.to_s.include?("Internal Server Error - Read")
|
738
|
+
raise InternalServerError, "Received an internal server error from App Store Connect / Developer Portal, please try again later"
|
739
|
+
elsif body.to_s.include?("Gateway Timeout - In read")
|
740
|
+
raise GatewayTimeoutError, "Received a gateway timeout error from App Store Connect / Developer Portal, please try again later"
|
741
|
+
elsif (body["resultString"] || "").include?("Program License Agreement")
|
742
|
+
raise ProgramLicenseAgreementUpdated, "#{body['userString']} Please manually log into your Apple Developer account to review and accept the updated agreement."
|
743
|
+
end
|
744
|
+
end
|
745
|
+
|
746
|
+
# This also gets called from subclasses
|
747
|
+
def raise_insufficient_permission_error!(additional_error_string: nil, caller_location: 2)
|
748
|
+
# get the method name of the request that failed
|
749
|
+
# `block in` is used very often for requests when surrounded for paging or retrying blocks
|
750
|
+
# The ! is part of some methods when they modify or delete a resource, so we don't want to show it
|
751
|
+
# Using `sub` instead of `delete` as we don't want to allow multiple matches
|
752
|
+
calling_method_name = caller_locations(caller_location, 2).first.label.sub("block in", "").delete("!").strip
|
753
|
+
|
754
|
+
# calling the computed property self.team_id can get us into an exception handling loop
|
755
|
+
team_id = @current_team_id ? "(Team ID #{@current_team_id}) " : ""
|
756
|
+
|
757
|
+
error_message = "User #{self.user} #{team_id}doesn't have enough permission for the following action: #{calling_method_name}"
|
758
|
+
error_message += " (#{additional_error_string})" if additional_error_string.to_s.length > 0
|
759
|
+
raise InsufficientPermissions, error_message
|
760
|
+
end
|
761
|
+
|
762
|
+
private
|
763
|
+
|
764
|
+
def directory_accessible?(path)
|
765
|
+
Dir.exist?(File.expand_path(path))
|
766
|
+
end
|
767
|
+
|
768
|
+
def do_login(user, password)
|
769
|
+
@loggedin = false
|
770
|
+
ret = send_login_request(user, password) # different in subclasses
|
771
|
+
@loggedin = true
|
772
|
+
ret
|
773
|
+
end
|
774
|
+
|
775
|
+
# Is called from `parse_response` to store the latest csrf_token (if available)
|
776
|
+
def store_csrf_tokens(response)
|
777
|
+
if response && response.headers
|
778
|
+
tokens = response.headers.select { |k, v| %w(csrf csrf_ts).include?(k) }
|
779
|
+
if tokens && !tokens.empty?
|
780
|
+
@csrf_tokens = tokens
|
781
|
+
end
|
782
|
+
end
|
783
|
+
end
|
784
|
+
|
785
|
+
def log_request(method, url, params, headers = nil, &block)
|
786
|
+
url ||= extract_key_from_block('url', &block)
|
787
|
+
body = extract_key_from_block('body', &block)
|
788
|
+
body_to_log = '[undefined body]'
|
789
|
+
if body
|
790
|
+
begin
|
791
|
+
body = JSON.parse(body)
|
792
|
+
# replace password in body if present
|
793
|
+
body['password'] = '***' if body.kind_of?(Hash) && body.key?("password")
|
794
|
+
body_to_log = body.to_json
|
795
|
+
rescue JSON::ParserError
|
796
|
+
# no json, no password to replace
|
797
|
+
body_to_log = "[non JSON body]"
|
798
|
+
end
|
799
|
+
end
|
800
|
+
params_to_log = Hash(params).dup # to also work with nil
|
801
|
+
params_to_log.delete(:accountPassword) # Dev Portal
|
802
|
+
params_to_log.delete(:theAccountPW) # iTC
|
803
|
+
params_to_log = params_to_log.collect do |key, value|
|
804
|
+
"{#{key}: #{value}}"
|
805
|
+
end
|
806
|
+
logger.info(">> #{method.upcase} #{url}: #{body_to_log} #{params_to_log.join(', ')}")
|
807
|
+
end
|
808
|
+
|
809
|
+
def log_response(method, url, response, headers = nil, &block)
|
810
|
+
url ||= extract_key_from_block('url', &block)
|
811
|
+
body = response.body.kind_of?(String) ? response.body.force_encoding(Encoding::UTF_8) : response.body
|
812
|
+
logger.debug("<< #{method.upcase} #{url}: #{response.status} #{body}")
|
813
|
+
end
|
814
|
+
|
815
|
+
def extract_key_from_block(key, &block)
|
816
|
+
if block_given?
|
817
|
+
obj = Object.new
|
818
|
+
class << obj
|
819
|
+
attr_accessor :body, :headers, :params, :url, :options
|
820
|
+
# rubocop: disable Style/TrivialAccessors
|
821
|
+
# the block calls `url` (not `url=`) so need to define `url` method
|
822
|
+
def url(url)
|
823
|
+
@url = url
|
824
|
+
end
|
825
|
+
|
826
|
+
def options
|
827
|
+
options_obj = Object.new
|
828
|
+
class << options_obj
|
829
|
+
attr_accessor :params_encoder
|
830
|
+
end
|
831
|
+
options_obj
|
832
|
+
end
|
833
|
+
# rubocop: enable Style/TrivialAccessors
|
834
|
+
end
|
835
|
+
obj.headers = {}
|
836
|
+
yield(obj)
|
837
|
+
obj.instance_variable_get("@#{key}")
|
838
|
+
end
|
839
|
+
end
|
840
|
+
|
841
|
+
# Actually sends the request to the remote server
|
842
|
+
# Automatically retries the request up to 3 times if something goes wrong
|
843
|
+
def send_request(method, url_or_path, params, headers, &block)
|
844
|
+
with_retry do
|
845
|
+
response = @client.send(method, url_or_path, params, headers, &block)
|
846
|
+
log_response(method, url_or_path, response, headers, &block)
|
847
|
+
|
848
|
+
resp_hash = response.to_hash
|
849
|
+
if resp_hash[:status] == 401
|
850
|
+
msg = "Auth lost"
|
851
|
+
logger.warn(msg)
|
852
|
+
raise UnauthorizedAccessError.new, "Unauthorized Access"
|
853
|
+
end
|
854
|
+
|
855
|
+
if response.body.to_s.include?("<title>302 Found</title>")
|
856
|
+
raise AppleTimeoutError.new, "Apple 302 detected - this might be temporary server error, check https://developer.apple.com/system-status/ to see if there is a known downtime"
|
857
|
+
end
|
858
|
+
|
859
|
+
if response.body.to_s.include?("<h3>Bad Gateway</h3>")
|
860
|
+
raise BadGatewayError.new, "Apple 502 detected - this might be temporary server error, try again later"
|
861
|
+
end
|
862
|
+
|
863
|
+
return response
|
864
|
+
end
|
865
|
+
end
|
866
|
+
|
867
|
+
def send_request_auto_paginate(method, url_or_path, params, headers, &block)
|
868
|
+
response = send_request(method, url_or_path, params, headers, &block)
|
869
|
+
return response unless should_process_next_rel?(response)
|
870
|
+
last_response = response
|
871
|
+
while last_response.env.rels[:next]
|
872
|
+
last_response = send_request(method, last_response.env.rels[:next], params, headers, &block)
|
873
|
+
break unless should_process_next_rel?(last_response)
|
874
|
+
response.body['data'].concat(last_response.body['data'])
|
875
|
+
end
|
876
|
+
response
|
877
|
+
end
|
878
|
+
|
879
|
+
def should_process_next_rel?(response)
|
880
|
+
response.body.kind_of?(Hash) && response.body['data'].kind_of?(Array)
|
881
|
+
end
|
882
|
+
|
883
|
+
def encode_params(params, headers)
|
884
|
+
params = Faraday::Utils::ParamsHash[params].to_query
|
885
|
+
headers = { 'Content-Type' => 'application/x-www-form-urlencoded' }.merge(headers)
|
886
|
+
return params, headers
|
887
|
+
end
|
888
|
+
end
|
889
|
+
# rubocop:enable Metrics/ClassLength
|
890
|
+
end
|
891
|
+
|
892
|
+
require 'spaceship/two_step_or_factor_client'
|