forked from bitrise-steplib/steps-amazon-s3-deploy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy paths3deploy.rb
241 lines (187 loc) · 7.13 KB
/
s3deploy.rb
1
2
3
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
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
# --------------------------
# --- Constants & Variables
# --------------------------
@this_script_path = File.expand_path(File.dirname(__FILE__))
# --------------------------
# --- Functions
# --------------------------
def log_fail(message)
puts
puts "\e[31m#{message}\e[0m"
exit(1)
end
def log_warn(message)
puts "\e[33m#{message}\e[0m"
end
def log_info(message)
puts
puts "\e[34m#{message}\e[0m"
end
def log_details(message)
puts " #{message}"
end
def log_done(message)
puts " \e[32m#{message}\e[0m"
end
def s3_object_uri_for_bucket_and_path(bucket_name, path_in_bucket)
return "s3://#{bucket_name}/#{path_in_bucket}"
end
def public_url_for_bucket_and_path(bucket_name, bucket_region, path_in_bucket)
if bucket_region.to_s == '' || bucket_region.to_s == 'us-east-1'
return "https://s3.amazonaws.com/#{bucket_name}/#{path_in_bucket}"
end
return "https://s3-#{bucket_region}.amazonaws.com/#{bucket_name}/#{path_in_bucket}"
end
def export_output(out_key, out_value)
IO.popen("envman add --key #{out_key.to_s}", 'r+') { |f|
f.write(out_value.to_s)
f.close_write
f.read
}
end
def do_s3upload(sourcepth, full_destpth, aclstr)
return system(%Q{aws s3 cp "#{sourcepth}" "#{full_destpth}" --acl "#{aclstr}"})
end
# -----------------------
# --- Main
# -----------------------
options = {
ipa: ENV['ipa_path'],
dsym: ENV['dsym_path'],
app_slug: ENV['app_slug'],
build_slug: ENV['build_slug'],
access_key: ENV['aws_access_key'],
secret_key: ENV['aws_secret_key'],
bucket_name: ENV['bucket_name'],
bucket_region: ENV['bucket_region'],
path_in_bucket: ENV['path_in_bucket'],
acl: ENV['file_access_level']
}
#
# Print options
log_info('Configs:')
log_details("* ipa_path: #{options[:ipa]}")
log_details("* dsym_path: #{options[:dsym]}")
log_details("* app_slug: #{options[:app_slug]}")
log_details("* build_slug: #{options[:build_slug]}")
log_details('* aws_access_key: ') if options[:access_key].to_s == ''
log_details('* aws_access_key: ***') unless options[:access_key].to_s == ''
log_details('* aws_secret_key: ') if options[:secret_key].to_s == ''
log_details('* aws_secret_key: ***') unless options[:secret_key].to_s == ''
log_details("* bucket_name: #{options[:bucket_name]}")
log_details("* bucket_region: #{options[:bucket_region]}")
log_details("* path_in_bucket: #{options[:path_in_bucket]}")
log_details("* file_access_level: #{options[:acl]}")
status = 'success'
begin
#
# Validate options
fail 'No IPA found to deploy. Terminating.' unless File.exist?(options[:ipa])
unless File.exist?(options[:dsym])
options[:dsym] = nil
log_warn("DSYM file not found. To generate debug symbols (dSYM) go to your Xcode Project's Settings - Build Settings - Debug Information Format and set it to DWARF with dSYM File.")
end
fail 'Missing required input: app_slug' if options[:app_slug].to_s.eql?('')
fail 'Missing required input: build_slug' if options[:build_slug].to_s.eql?('')
fail 'Missing required input: aws_access_key' if options[:access_key].to_s.eql?('')
fail 'Missing required input: aws_secret_key' if options[:secret_key].to_s.eql?('')
fail 'Missing required input: bucket_name' if options[:bucket_name].to_s.eql?('')
fail 'Missing required input: file_access_level' if options[:acl].to_s.eql?('')
#
# AWS configs
ENV['AWS_ACCESS_KEY_ID'] = options[:access_key]
ENV['AWS_SECRET_ACCESS_KEY'] = options[:secret_key]
ENV['AWS_DEFAULT_REGION'] = options[:bucket_region] unless options[:bucket_region].to_s.eql?('')
#
# define object path
base_path_in_bucket = ''
if options[:path_in_bucket]
base_path_in_bucket = options[:path_in_bucket]
else
utc_timestamp = Time.now.utc.to_i
base_path_in_bucket = "bitrise_#{options[:app_slug]}/#{utc_timestamp}_build_#{options[:build_slug]}"
end
#
# supported: private, public_read
acl_arg = 'public-read'
if options[:acl]
case options[:acl]
when 'public_read'
acl_arg = 'public-read'
when 'private'
acl_arg = 'private'
else
fail "Invalid ACL option: #{options[:acl]}"
end
end
#
# ipa upload
log_info('Uploading IPA...')
ipa_path_in_bucket = "#{base_path_in_bucket}/#{File.basename(options[:ipa])}"
ipa_full_s3_path = s3_object_uri_for_bucket_and_path(options[:bucket_name], ipa_path_in_bucket)
public_url_ipa = public_url_for_bucket_and_path(options[:bucket_name], options[:bucket_region], ipa_path_in_bucket)
fail 'Failed to upload IPA' unless do_s3upload(options[:ipa], ipa_full_s3_path, acl_arg)
export_output('S3_DEPLOY_STEP_URL_IPA', public_url_ipa)
log_done('IPA upload success')
#
# dsym upload
if options[:dsym]
log_info('Uploading dSYM...')
dsym_path_in_bucket = "#{base_path_in_bucket}/#{File.basename(options[:dsym])}"
dsym_full_s3_path = s3_object_uri_for_bucket_and_path(options[:bucket_name], dsym_path_in_bucket)
public_url_dsym = public_url_for_bucket_and_path(options[:bucket_name], options[:bucket_region], dsym_path_in_bucket)
fail 'Failed to upload dSYM' unless do_s3upload(options[:dsym], dsym_full_s3_path, acl_arg)
export_output('S3_DEPLOY_STEP_URL_DSYM', public_url_dsym)
log_done('dSYM upload success')
end
ENV['S3_DEPLOY_STEP_URL_IPA'] = "#{public_url_ipa}"
#
# plist generation - we have to run it after we have obtained the public url to the ipa
log_info('Generating Info.plist...')
success = system("sh #{@this_script_path}/gen_plist.sh")
fail 'Failed to generate info.plist' unless success
log_done('Generating Info.plist succed')
#
# plist upload
plist_local_path = 'Info.plist'
public_url_plist = ''
if File.exist?(plist_local_path)
log_info('Uploading Info.plist...')
plist_path_in_bucket = "#{base_path_in_bucket}/Info.plist"
plist_full_s3_path = "s3://#{options[:bucket_name]}/#{plist_path_in_bucket}"
public_url_plist = public_url_for_bucket_and_path(options[:bucket_name], options[:bucket_region], plist_path_in_bucket)
fail 'Failed to upload Info.plist' unless do_s3upload(plist_local_path, plist_full_s3_path, acl_arg)
fail 'Failed to remove Plist' unless system(%Q{rm "#{plist_local_path}"})
log_done('Info.plist upload success')
else
log_warn('NO Info.plist generated :<')
end
export_output('S3_DEPLOY_STEP_URL_PLIST', public_url_plist)
email_ready_link_url = "itms-services://?action=download-manifest&url=#{public_url_plist}"
export_output('S3_DEPLOY_STEP_EMAIL_READY_URL', email_ready_link_url)
#
# Print deploy infos
log_info 'Deploy infos:'
log_details("* Access Level: #{options[:acl]}")
log_details("* IPA: #{public_url_ipa}")
if options[:dsym]
log_details("* DSYM: #{public_url_dsym}")
else
log_warn(%Q{DSYM file not found.
To generate debug symbols (dSYM) go to your
Xcode Project's Settings - `Build Settings - Debug Information Format`
and set it to **DWARF with dSYM File**.})
end
log_details("* Plist: #{public_url_plist}")
log_info('Install link:')
log_details("#{email_ready_link_url}")
puts
log_details('Open this link on an iOS device to install the app')
rescue => ex
status = 'failed'
log_fail("#{ex}")
ensure
export_output('S3_DEPLOY_STEP_STATUS', status)
puts
log_done("#{status}")
end