Class: Fastlane::Actions::UpdateAppIdentifierAction

Inherits:
Fastlane::Action show all
Defined in:
fastlane/lib/fastlane/actions/update_app_identifier.rb

Constant Summary

Constants inherited from Fastlane::Action

Fastlane::Action::AVAILABLE_CATEGORIES, Fastlane::Action::RETURN_TYPES

Documentation collapse

Class Method Summary collapse

Methods inherited from Fastlane::Action

action_name, author, deprecated_notes, lane_context, method_missing, other_action, output, return_type, return_value, sample_return_value, shell_out_should_use_bundle_exec?, step_text

Class Method Details

.authorsObject


104
105
106
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 104

def self.authors
  ['squarefrog', 'tobiasstrebitzer']
end

.available_optionsObject


77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 77

def self.available_options
  [
    FastlaneCore::ConfigItem.new(key: :xcodeproj,
                                 env_name: "FL_UPDATE_APP_IDENTIFIER_PROJECT_PATH",
                                 description: "Path to your Xcode project",
                                 code_gen_sensitive: true,
                                 default_value: Dir['*.xcodeproj'].first,
                                 default_value_dynamic: true,
                                 verify_block: proc do |value|
                                   UI.user_error!("Please pass the path to the project, not the workspace") unless value.end_with?(".xcodeproj")
                                   UI.user_error!("Could not find Xcode project") unless File.exist?(value)
                                 end),
    FastlaneCore::ConfigItem.new(key: :plist_path,
                                 env_name: "FL_UPDATE_APP_IDENTIFIER_PLIST_PATH",
                                 description: "Path to info plist, relative to your Xcode project",
                                 verify_block: proc do |value|
                                   UI.user_error!("Invalid plist file") unless value[-6..-1].casecmp(".plist").zero?
                                 end),
    FastlaneCore::ConfigItem.new(key: :app_identifier,
                                 env_name: 'FL_UPDATE_APP_IDENTIFIER',
                                 description: 'The app Identifier you want to set',
                                 code_gen_sensitive: true,
                                 default_value: ENV['PRODUCE_APP_IDENTIFIER'] || CredentialsManager::AppfileConfig.try_fetch_value(:app_identifier),
                                 default_value_dynamic: true)
  ]
end

.categoryObject


118
119
120
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 118

def self.category
  :project
end

.descriptionObject


69
70
71
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 69

def self.description
  "Update the project's bundle identifier"
end

.detailsObject


73
74
75
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 73

def self.details
  "Update an app identifier by either setting `CFBundleIdentifier` or `PRODUCT_BUNDLE_IDENTIFIER`, depending on which is already in use."
end

.example_codeObject


108
109
110
111
112
113
114
115
116
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 108

def self.example_code
  [
    'update_app_identifier(
      xcodeproj: "Example.xcodeproj", # Optional path to xcodeproj, will use the first .xcodeproj if not set
      plist_path: "Example/Info.plist", # Path to info plist file, relative to xcodeproj
      app_identifier: "com.test.example" # The App Identifier
    )'
  ]
end

.is_supported?(platform) ⇒ Boolean

Returns:


65
66
67
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 65

def self.is_supported?(platform)
  [:ios].include?(platform)
end

.resolve_path(path, xcodeproj_path) ⇒ Object


50
51
52
53
54
55
56
57
58
59
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 50

def self.resolve_path(path, xcodeproj_path)
  return nil unless path
  project_dir = File.dirname(xcodeproj_path)
  # SRCROOT, SOURCE_ROOT and PROJECT_DIR are the same
  %w{SRCROOT SOURCE_ROOT PROJECT_DIR}.each do |variable_name|
    path = path.sub("$(#{variable_name})", project_dir)
  end
  path = File.absolute_path(path, project_dir)
  path
end

.run(params) ⇒ Object


4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
# File 'fastlane/lib/fastlane/actions/update_app_identifier.rb', line 4

def self.run(params)
  require 'plist'
  require 'xcodeproj'

  info_plist_key = 'INFOPLIST_FILE'
  identifier_key = 'PRODUCT_BUNDLE_IDENTIFIER'

  # Read existing plist file
  info_plist_path = resolve_path(params[:plist_path], params[:xcodeproj])
  UI.user_error!("Couldn't find info plist file at path '#{params[:plist_path]}'") unless File.exist?(info_plist_path)
  plist = Plist.parse_xml(info_plist_path)

  # Check if current app identifier product bundle identifier
  if plist['CFBundleIdentifier'] == "$(#{identifier_key})"
    # Load .xcodeproj
    project_path = params[:xcodeproj]
    project = Xcodeproj::Project.open(project_path)

    # Fetch the build configuration objects
    configs = project.objects.select { |obj| obj.isa == 'XCBuildConfiguration' && !obj.build_settings[identifier_key].nil? }
    UI.user_error!("Info plist uses $(#{identifier_key}), but xcodeproj does not") unless configs.count > 0

    configs = configs.select { |obj| resolve_path(obj.build_settings[info_plist_key], params[:xcodeproj]) == info_plist_path }
    UI.user_error!("Xcodeproj doesn't have configuration with info plist #{params[:plist_path]}.") unless configs.count > 0

    # For each of the build configurations, set app identifier
    configs.each do |c|
      c.build_settings[identifier_key] = params[:app_identifier]
    end

    # Write changes to the file
    project.save

    UI.success("Updated #{params[:xcodeproj]} 💾.")
  else
    # Update plist value
    plist['CFBundleIdentifier'] = params[:app_identifier]

    # Write changes to file
    plist_string = Plist::Emit.dump(plist)
    File.write(info_plist_path, plist_string)

    UI.success("Updated #{params[:plist_path]} 💾.")
  end
end