active_encode 0.7.0 → 0.8.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.rubocop.yml +1 -0
- data/.rubocop_todo.yml +1 -0
- data/Gemfile +2 -1
- data/README.md +69 -0
- data/active_encode.gemspec +5 -2
- data/lib/active_encode/engine_adapters.rb +1 -0
- data/lib/active_encode/engine_adapters/elastic_transcoder_adapter.rb +1 -1
- data/lib/active_encode/engine_adapters/ffmpeg_adapter.rb +12 -2
- data/lib/active_encode/engine_adapters/media_convert_adapter.rb +372 -0
- data/lib/active_encode/engine_adapters/media_convert_output.rb +104 -0
- data/lib/active_encode/polling.rb +1 -1
- data/lib/active_encode/spec/shared_specs.rb +2 -0
- data/{spec/shared_specs/engine_adapter_specs.rb → lib/active_encode/spec/shared_specs/engine_adapter.rb} +36 -36
- data/lib/active_encode/version.rb +1 -1
- data/lib/file_locator.rb +1 -1
- data/spec/fixtures/media_convert/endpoints.json +1 -0
- data/spec/fixtures/media_convert/job_canceled.json +412 -0
- data/spec/fixtures/media_convert/job_canceling.json +1 -0
- data/spec/fixtures/media_convert/job_completed.json +359 -0
- data/spec/fixtures/media_convert/job_completed_detail.json +1 -0
- data/spec/fixtures/media_convert/job_completed_detail_query.json +1 -0
- data/spec/fixtures/media_convert/job_created.json +408 -0
- data/spec/fixtures/media_convert/job_failed.json +406 -0
- data/spec/fixtures/media_convert/job_progressing.json +414 -0
- data/spec/integration/elastic_transcoder_adapter_spec.rb +4 -4
- data/spec/integration/ffmpeg_adapter_spec.rb +2 -2
- data/spec/integration/matterhorn_adapter_spec.rb +39 -39
- data/spec/integration/media_convert_adapter_spec.rb +126 -0
- data/spec/integration/pass_through_adapter_spec.rb +2 -2
- data/spec/integration/zencoder_adapter_spec.rb +198 -198
- data/spec/spec_helper.rb +0 -1
- data/spec/units/core_spec.rb +17 -17
- data/spec/units/file_locator_spec.rb +1 -1
- data/spec/units/global_id_spec.rb +8 -8
- data/spec/units/persistence_spec.rb +9 -9
- metadata +87 -23
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: e36b69edd440383a39009cc2a331f6cda32974e62fde4961353e5abd80e2bc0b
|
4
|
+
data.tar.gz: ce26e5a14a08c96b708d6fea08ff127606f9da7023a674444037374cf9812ef9
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d82bbc86cfc93cc33d699fb943c5b1caab27bc76b8149a599a61514f5257572aa3f0239ab617062f5b9ed4ef5b3874b9764f6786b0700fe2fa08625b917ead07
|
7
|
+
data.tar.gz: 53aabadb2c2337ac2cf8725ca0476b5afb9446aa8102aa156e93913d978317ceece57facf2bce0775fb8d5e3cf6e5a9ad47538c15006aed40d7bb94b98e301ce
|
data/.rubocop.yml
CHANGED
data/.rubocop_todo.yml
CHANGED
data/Gemfile
CHANGED
data/README.md
CHANGED
@@ -124,6 +124,75 @@ end
|
|
124
124
|
|
125
125
|
Engine adapters are shims between ActiveEncode and the back end encoding service. You can add an additional engine by creating an engine adapter class that implements `:create`, `:find`, and `:cancel` and passes the shared specs.
|
126
126
|
|
127
|
+
For example:
|
128
|
+
```ruby
|
129
|
+
# In your application at:
|
130
|
+
# lib/active_encode/engine_adapters/my_custom_adapter.rb
|
131
|
+
module ActiveEncode
|
132
|
+
module EngineAdapters
|
133
|
+
class MyCustomAdapter
|
134
|
+
def create(input_url, options = {})
|
135
|
+
# Start a new encoding job. This may be an external service, or a
|
136
|
+
# locally queued job.
|
137
|
+
|
138
|
+
# Return an instance ActiveEncode::Base (or subclass) that represents
|
139
|
+
# the encoding job that was just started.
|
140
|
+
end
|
141
|
+
|
142
|
+
def find(id, opts = {})
|
143
|
+
# Find the encoding job for the given parameters.
|
144
|
+
|
145
|
+
# Return an instance of ActiveEncode::Base (or subclass) that represents
|
146
|
+
# the found encoding job.
|
147
|
+
end
|
148
|
+
|
149
|
+
def cancel(id)
|
150
|
+
# Cancel the encoding job for the given id.
|
151
|
+
|
152
|
+
# Return an instance of ActiveEncode::Base (or subclass) that represents
|
153
|
+
# the canceled job.
|
154
|
+
end
|
155
|
+
end
|
156
|
+
end
|
157
|
+
end
|
158
|
+
```
|
159
|
+
Then, use the shared specs...
|
160
|
+
```ruby
|
161
|
+
# In your application at...
|
162
|
+
# spec/lib/active_encode/engine_adapters/my_custom_adapter_spec.rb
|
163
|
+
require 'spec_helper'
|
164
|
+
require 'active_encode/spec/shared_specs'
|
165
|
+
RSpec.describe MyCustomAdapter do
|
166
|
+
let(:created_job) {
|
167
|
+
# an instance of ActiveEncode::Base represented a newly created encode job
|
168
|
+
}
|
169
|
+
let(:running_job) {
|
170
|
+
# an instance of ActiveEncode::Base represented a running encode job
|
171
|
+
}
|
172
|
+
let(:canceled_job) {
|
173
|
+
# an instance of ActiveEncode::Base represented a canceled encode job
|
174
|
+
}
|
175
|
+
let(:completed_job) {
|
176
|
+
# an instance of ActiveEncode::Base represented a completed encode job
|
177
|
+
}
|
178
|
+
let(:failed_job) {
|
179
|
+
# an instance of ActiveEncode::Base represented a failed encode job
|
180
|
+
}
|
181
|
+
let(:completed_tech_metadata) {
|
182
|
+
# a hash representing completed technical metadata
|
183
|
+
}
|
184
|
+
let(:completed_output) {
|
185
|
+
# data representing completed output
|
186
|
+
}
|
187
|
+
let(:failed_tech_metadata) {
|
188
|
+
# a hash representing failed technical metadata
|
189
|
+
}
|
190
|
+
|
191
|
+
# Run the shared specs.
|
192
|
+
it_behaves_like 'an ActiveEncode::EngineAdapter'
|
193
|
+
end
|
194
|
+
```
|
195
|
+
|
127
196
|
# Acknowledgments
|
128
197
|
|
129
198
|
This software has been developed by and is brought to you by the Samvera community. Learn more at the
|
data/active_encode.gemspec
CHANGED
@@ -22,7 +22,11 @@ Gem::Specification.new do |spec|
|
|
22
22
|
|
23
23
|
spec.add_dependency "rails"
|
24
24
|
|
25
|
-
spec.add_development_dependency "aws-sdk"
|
25
|
+
spec.add_development_dependency "aws-sdk-cloudwatchevents"
|
26
|
+
spec.add_development_dependency "aws-sdk-cloudwatchlogs"
|
27
|
+
spec.add_development_dependency "aws-sdk-elastictranscoder"
|
28
|
+
spec.add_development_dependency "aws-sdk-mediaconvert"
|
29
|
+
spec.add_development_dependency "aws-sdk-s3"
|
26
30
|
spec.add_development_dependency "bixby", '~> 1.0.0'
|
27
31
|
spec.add_development_dependency "bundler"
|
28
32
|
spec.add_development_dependency "coveralls"
|
@@ -30,7 +34,6 @@ Gem::Specification.new do |spec|
|
|
30
34
|
spec.add_development_dependency "engine_cart", "~> 2.2"
|
31
35
|
spec.add_development_dependency "rake"
|
32
36
|
spec.add_development_dependency "rspec"
|
33
|
-
spec.add_development_dependency "rspec-its"
|
34
37
|
spec.add_development_dependency 'rspec_junit_formatter'
|
35
38
|
spec.add_development_dependency "rspec-rails"
|
36
39
|
|
@@ -34,7 +34,13 @@ module ActiveEncode
|
|
34
34
|
FileUtils.mkdir_p working_path("outputs", new_encode.id)
|
35
35
|
|
36
36
|
# Extract technical metadata from input file
|
37
|
-
|
37
|
+
curl_option = if options && options[:headers]
|
38
|
+
headers = options[:headers].map { |k, v| "#{k}: #{v}" }
|
39
|
+
(["--File_curl=HttpHeader"] + headers).join(",").yield_self { |s| "'#{s}'" }
|
40
|
+
else
|
41
|
+
""
|
42
|
+
end
|
43
|
+
`#{MEDIAINFO_PATH} #{curl_option} --Output=XML --LogFile=#{working_path("input_metadata", new_encode.id)} "#{input_url}"`
|
38
44
|
new_encode.input = build_input new_encode
|
39
45
|
|
40
46
|
if new_encode.input.duration.blank?
|
@@ -206,7 +212,11 @@ module ActiveEncode
|
|
206
212
|
file_name = "outputs/#{sanitized_filename}-#{output[:label]}.#{output[:extension]}"
|
207
213
|
" #{output[:ffmpeg_opt]} #{working_path(file_name, id)}"
|
208
214
|
end.join(" ")
|
209
|
-
|
215
|
+
header_opt = Array(opts[:headers]).map do |k, v|
|
216
|
+
"#{k}: #{v}\r\n"
|
217
|
+
end.join
|
218
|
+
header_opt = "-headers '#{header_opt}'" if header_opt.present?
|
219
|
+
"#{FFMPEG_PATH} #{header_opt} -y -loglevel error -progress #{working_path('progress', id)} -i \"#{input_url}\" #{output_opt}"
|
210
220
|
end
|
211
221
|
|
212
222
|
def sanitize_base(input_url)
|
@@ -0,0 +1,372 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require 'active_encode/engine_adapters/media_convert_output.rb'
|
3
|
+
require 'active_support/core_ext/integer/time'
|
4
|
+
require 'addressable/uri'
|
5
|
+
require 'aws-sdk-cloudwatchevents'
|
6
|
+
require 'aws-sdk-cloudwatchlogs'
|
7
|
+
require 'aws-sdk-mediaconvert'
|
8
|
+
require 'file_locator'
|
9
|
+
|
10
|
+
require 'active_support/json'
|
11
|
+
require 'active_support/time'
|
12
|
+
|
13
|
+
module ActiveEncode
|
14
|
+
module EngineAdapters
|
15
|
+
class MediaConvertAdapter
|
16
|
+
# [AWS Elemental MediaConvert](https://aws.amazon.com/mediaconvert/) doesn't provide detailed
|
17
|
+
# output information in the job description that can be pulled directly from the service.
|
18
|
+
# Instead, it provides that information along with the job status notification when the job
|
19
|
+
# status changes to `COMPLETE`. The only way to capture that notification is through an [Amazon
|
20
|
+
# Eventbridge](https://aws.amazon.com/eventbridge/) rule that forwards the status change
|
21
|
+
# notification to another service for capture and/or handling.
|
22
|
+
#
|
23
|
+
# `ActiveEncode::EngineAdapters::MediaConvert` does this by creating a [CloudWatch Logs]
|
24
|
+
# (https://aws.amazon.com/cloudwatch/) log group and an EventBridge rule to forward status
|
25
|
+
# change notifications to the log group. It can then find the log entry containing the output
|
26
|
+
# details later when the job is complete. This is accomplished by calling the idempotent
|
27
|
+
# `#setup!` method.
|
28
|
+
#
|
29
|
+
# The AWS user/role calling the `#setup!` method will require permissions to create the
|
30
|
+
# necessary CloudWatch and EventBridge resources, and the role passed to the engine adapter
|
31
|
+
# will need access to any S3 buckets where files will be read from or written to during
|
32
|
+
# transcoding.
|
33
|
+
#
|
34
|
+
# Configuration example:
|
35
|
+
#
|
36
|
+
# ActiveEncode::Base.engine_adapter = :media_convert
|
37
|
+
# ActiveEncode::Base.engine_adapter.role = 'arn:aws:iam::123456789012:role/service-role/MediaConvert_Default_Role'
|
38
|
+
# ActiveEncode::Base.engine_adapter.output_bucket = 'output-bucket'
|
39
|
+
# ActiveEncode::Base.engine_adapter.setup!
|
40
|
+
|
41
|
+
JOB_STATES = {
|
42
|
+
"SUBMITTED" => :running, "PROGRESSING" => :running, "CANCELED" => :cancelled,
|
43
|
+
"ERROR" => :failed, "COMPLETE" => :completed
|
44
|
+
}.freeze
|
45
|
+
|
46
|
+
OUTPUT_GROUP_TEMPLATES = {
|
47
|
+
hls: { min_segment_length: 0, segment_control: "SEGMENTED_FILES", segment_length: 10 },
|
48
|
+
dash_iso: { fragment_length: 2, segment_control: "SEGMENTED_FILES", segment_length: 30 },
|
49
|
+
file: {},
|
50
|
+
ms_smooth: { fragment_length: 2 },
|
51
|
+
cmaf: { fragment_length: 2, segment_control: "SEGMENTED_FILES", segment_length: 10 }
|
52
|
+
}.freeze
|
53
|
+
|
54
|
+
attr_accessor :role, :output_bucket
|
55
|
+
attr_writer :log_group, :queue
|
56
|
+
|
57
|
+
def setup!
|
58
|
+
rule_name = "active-encode-mediaconvert-#{queue}"
|
59
|
+
return true if event_rule_exists?(rule_name)
|
60
|
+
|
61
|
+
queue_arn = mediaconvert.get_queue(name: queue).queue.arn
|
62
|
+
|
63
|
+
event_pattern = {
|
64
|
+
source: ["aws.mediaconvert"],
|
65
|
+
"detail-type": ["MediaConvert Job State Change"],
|
66
|
+
detail: {
|
67
|
+
queue: [queue_arn]
|
68
|
+
}
|
69
|
+
}
|
70
|
+
|
71
|
+
log_group_arn = create_log_group(log_group).arn
|
72
|
+
|
73
|
+
cloudwatch_events.put_rule(
|
74
|
+
name: rule_name,
|
75
|
+
event_pattern: event_pattern.to_json,
|
76
|
+
state: "ENABLED",
|
77
|
+
description: "Forward MediaConvert job state changes from queue #{queue} to #{log_group}"
|
78
|
+
)
|
79
|
+
|
80
|
+
cloudwatch_events.put_targets(
|
81
|
+
rule: rule_name,
|
82
|
+
targets: [
|
83
|
+
{
|
84
|
+
id: "Id#{SecureRandom.uuid}",
|
85
|
+
arn: log_group_arn
|
86
|
+
}
|
87
|
+
]
|
88
|
+
)
|
89
|
+
true
|
90
|
+
end
|
91
|
+
|
92
|
+
# Required options:
|
93
|
+
#
|
94
|
+
# * `output_prefix`: The S3 key prefix to use as the base for all outputs.
|
95
|
+
#
|
96
|
+
# * `outputs`: An array of `{preset, modifier}` options defining how to transcode and name the outputs.
|
97
|
+
#
|
98
|
+
# Optional options:
|
99
|
+
#
|
100
|
+
# * `masterfile_bucket`: The bucket to which file-based inputs will be copied before
|
101
|
+
# being passed to MediaConvert. Also used for S3-based inputs
|
102
|
+
# unless `use_original_url` is specified.
|
103
|
+
#
|
104
|
+
# * `use_original_url`: If `true`, any S3 URL passed in as input will be passed directly to
|
105
|
+
# MediaConvert as the file input instead of copying the source to
|
106
|
+
# the `masterfile_bucket`.
|
107
|
+
#
|
108
|
+
# Example:
|
109
|
+
# {
|
110
|
+
# output_prefix: "path/to/output/files",
|
111
|
+
# outputs: [
|
112
|
+
# {preset: "System-Avc_16x9_1080p_29_97fps_8500kbps", modifier: "-1080"},
|
113
|
+
# {preset: "System-Avc_16x9_720p_29_97fps_5000kbps", modifier: "-720"},
|
114
|
+
# {preset: "System-Avc_16x9_540p_29_97fps_3500kbps", modifier: "-540"}
|
115
|
+
# ]
|
116
|
+
# }
|
117
|
+
# }
|
118
|
+
def create(input_url, options = {})
|
119
|
+
input_url = s3_uri(input_url, options)
|
120
|
+
|
121
|
+
input = options[:media_type] == :audio ? make_audio_input(input_url) : make_video_input(input_url)
|
122
|
+
|
123
|
+
create_job_params = {
|
124
|
+
role: role,
|
125
|
+
settings: {
|
126
|
+
inputs: [input],
|
127
|
+
output_groups: make_output_groups(options)
|
128
|
+
}
|
129
|
+
}
|
130
|
+
|
131
|
+
response = mediaconvert.create_job(create_job_params)
|
132
|
+
job = response.job
|
133
|
+
build_encode(job)
|
134
|
+
end
|
135
|
+
|
136
|
+
def find(id, _opts = {})
|
137
|
+
response = mediaconvert.get_job(id: id)
|
138
|
+
job = response.job
|
139
|
+
build_encode(job)
|
140
|
+
rescue Aws::MediaConvert::Errors::NotFound
|
141
|
+
raise ActiveEncode::NotFound, "Job #{id} not found"
|
142
|
+
end
|
143
|
+
|
144
|
+
def cancel(id)
|
145
|
+
mediaconvert.cancel_job(id: id)
|
146
|
+
find(id)
|
147
|
+
end
|
148
|
+
|
149
|
+
def log_group
|
150
|
+
@log_group ||= "/aws/events/active-encode/mediaconvert/#{queue}"
|
151
|
+
end
|
152
|
+
|
153
|
+
def queue
|
154
|
+
@queue ||= "Default"
|
155
|
+
end
|
156
|
+
|
157
|
+
private
|
158
|
+
|
159
|
+
def build_encode(job)
|
160
|
+
return nil if job.nil?
|
161
|
+
encode = ActiveEncode::Base.new(job.settings.inputs.first.file_input, {})
|
162
|
+
encode.id = job.id
|
163
|
+
encode.input.id = job.id
|
164
|
+
encode.state = JOB_STATES[job.status]
|
165
|
+
encode.current_operations = [job.current_phase].compact
|
166
|
+
encode.created_at = job.timing.submit_time
|
167
|
+
encode.updated_at = job.timing.finish_time || job.timing.start_time || encode.created_at
|
168
|
+
encode.percent_complete = convert_percent_complete(job)
|
169
|
+
encode.errors = [job.error_message].compact
|
170
|
+
|
171
|
+
encode.input.created_at = encode.created_at
|
172
|
+
encode.input.updated_at = encode.updated_at
|
173
|
+
|
174
|
+
encode.output = encode.state == :completed ? convert_output(job) : []
|
175
|
+
encode
|
176
|
+
end
|
177
|
+
|
178
|
+
def convert_percent_complete(job)
|
179
|
+
case job.status
|
180
|
+
when "SUBMITTED"
|
181
|
+
5
|
182
|
+
when "PROGRESSING"
|
183
|
+
job.job_percent_complete
|
184
|
+
when "CANCELED", "ERROR"
|
185
|
+
50
|
186
|
+
when "COMPLETE"
|
187
|
+
100
|
188
|
+
else
|
189
|
+
0
|
190
|
+
end
|
191
|
+
end
|
192
|
+
|
193
|
+
def convert_output(job)
|
194
|
+
results = get_encode_results(job)
|
195
|
+
settings = job.settings.output_groups.first.outputs
|
196
|
+
|
197
|
+
outputs = results.dig('detail', 'outputGroupDetails', 0, 'outputDetails').map.with_index do |detail, index|
|
198
|
+
tech_md = MediaConvertOutput.tech_metadata(settings[index], detail)
|
199
|
+
output = ActiveEncode::Output.new
|
200
|
+
|
201
|
+
output.created_at = job.timing.submit_time
|
202
|
+
output.updated_at = job.timing.finish_time || job.timing.start_time || output.created_at
|
203
|
+
|
204
|
+
[:width, :height, :frame_rate, :duration, :checksum, :audio_codec, :video_codec,
|
205
|
+
:audio_bitrate, :video_bitrate, :file_size, :label, :url, :id].each do |field|
|
206
|
+
output.send("#{field}=", tech_md[field])
|
207
|
+
end
|
208
|
+
output.id ||= "#{job.id}-output#{tech_md[:suffix]}"
|
209
|
+
output
|
210
|
+
end
|
211
|
+
|
212
|
+
adaptive_playlist = results.dig('detail', 'outputGroupDetails', 0, 'playlistFilePaths', 0)
|
213
|
+
unless adaptive_playlist.nil?
|
214
|
+
output = ActiveEncode::Output.new
|
215
|
+
output.created_at = job.timing.submit_time
|
216
|
+
output.updated_at = job.timing.finish_time || job.timing.start_time || output.created_at
|
217
|
+
output.id = "#{job.id}-output-auto"
|
218
|
+
|
219
|
+
[:duration, :audio_codec, :video_codec].each do |field|
|
220
|
+
output.send("#{field}=", outputs.first.send(field))
|
221
|
+
end
|
222
|
+
output.label = File.basename(adaptive_playlist)
|
223
|
+
output.url = adaptive_playlist
|
224
|
+
outputs << output
|
225
|
+
end
|
226
|
+
outputs
|
227
|
+
end
|
228
|
+
|
229
|
+
def get_encode_results(job)
|
230
|
+
start_time = job.timing.submit_time
|
231
|
+
end_time = job.timing.finish_time || Time.now.utc
|
232
|
+
|
233
|
+
response = cloudwatch_logs.start_query(
|
234
|
+
log_group_name: log_group,
|
235
|
+
start_time: start_time.to_i,
|
236
|
+
end_time: end_time.to_i,
|
237
|
+
limit: 1,
|
238
|
+
query_string: "fields @message | filter detail.jobId = '#{job.id}' | filter detail.status = 'COMPLETE' | sort @ingestionTime desc"
|
239
|
+
)
|
240
|
+
query_id = response.query_id
|
241
|
+
response = cloudwatch_logs.get_query_results(query_id: query_id)
|
242
|
+
until response.status == "Complete"
|
243
|
+
sleep(0.5)
|
244
|
+
response = cloudwatch_logs.get_query_results(query_id: query_id)
|
245
|
+
end
|
246
|
+
raise ActiveEncode::NotFound, "Unable to load progress for job #{job.id}" if response.results.empty?
|
247
|
+
|
248
|
+
JSON.parse(response.results.first.first.value)
|
249
|
+
end
|
250
|
+
|
251
|
+
def cloudwatch_events
|
252
|
+
@cloudwatch_events ||= Aws::CloudWatchEvents::Client.new
|
253
|
+
end
|
254
|
+
|
255
|
+
def cloudwatch_logs
|
256
|
+
@cloudwatch_logs ||= Aws::CloudWatchLogs::Client.new
|
257
|
+
end
|
258
|
+
|
259
|
+
def mediaconvert
|
260
|
+
endpoint = Aws::MediaConvert::Client.new.describe_endpoints.endpoints.first.url
|
261
|
+
@mediaconvert ||= Aws::MediaConvert::Client.new(endpoint: endpoint)
|
262
|
+
end
|
263
|
+
|
264
|
+
def s3_uri(url, options = {})
|
265
|
+
bucket = options[:masterfile_bucket]
|
266
|
+
|
267
|
+
case Addressable::URI.parse(url).scheme
|
268
|
+
when nil, 'file'
|
269
|
+
upload_to_s3 url, bucket
|
270
|
+
when 's3'
|
271
|
+
return url if options[:use_original_url]
|
272
|
+
check_s3_bucket url, bucket
|
273
|
+
else
|
274
|
+
raise ArgumentError, "Cannot handle source URL: #{url}"
|
275
|
+
end
|
276
|
+
end
|
277
|
+
|
278
|
+
def check_s3_bucket(input_url, source_bucket)
|
279
|
+
# logger.info("Checking `#{input_url}'")
|
280
|
+
s3_object = FileLocator::S3File.new(input_url).object
|
281
|
+
if s3_object.bucket_name == source_bucket
|
282
|
+
# logger.info("Already in bucket `#{source_bucket}'")
|
283
|
+
s3_object.key
|
284
|
+
else
|
285
|
+
s3_key = File.join(SecureRandom.uuid, s3_object.key)
|
286
|
+
# logger.info("Copying to `#{source_bucket}/#{input_url}'")
|
287
|
+
target = Aws::S3::Object.new(bucket_name: source_bucket, key: input_url)
|
288
|
+
target.copy_from(s3_object, multipart_copy: s3_object.size > 15_728_640) # 15.megabytes
|
289
|
+
s3_key
|
290
|
+
end
|
291
|
+
end
|
292
|
+
|
293
|
+
def upload_to_s3(input_url, source_bucket)
|
294
|
+
# original_input = input_url
|
295
|
+
bucket = Aws::S3::Resource.new(client: s3client).bucket(source_bucket)
|
296
|
+
filename = FileLocator.new(input_url).location
|
297
|
+
s3_key = File.join(SecureRandom.uuid, File.basename(filename))
|
298
|
+
# logger.info("Copying `#{original_input}' to `#{source_bucket}/#{input_url}'")
|
299
|
+
obj = bucket.object(s3_key)
|
300
|
+
obj.upload_file filename
|
301
|
+
|
302
|
+
s3_key
|
303
|
+
end
|
304
|
+
|
305
|
+
def event_rule_exists?(rule_name)
|
306
|
+
rule = cloudwatch_events.list_rules(name_prefix: rule_name).rules.find do |existing_rule|
|
307
|
+
existing_rule.name == rule_name
|
308
|
+
end
|
309
|
+
!rule.nil?
|
310
|
+
end
|
311
|
+
|
312
|
+
def find_log_group(name)
|
313
|
+
cloudwatch_logs.describe_log_groups(log_group_name_prefix: name).log_groups.find do |group|
|
314
|
+
group.log_group_name == name
|
315
|
+
end
|
316
|
+
end
|
317
|
+
|
318
|
+
def create_log_group(name)
|
319
|
+
result = find_log_group(name)
|
320
|
+
|
321
|
+
return result unless result.nil?
|
322
|
+
|
323
|
+
cloudwatch_logs.create_log_group(log_group_name: name)
|
324
|
+
find_log_group(name)
|
325
|
+
end
|
326
|
+
|
327
|
+
def make_audio_input(input_url)
|
328
|
+
{
|
329
|
+
audio_selectors: { "Audio Selector 1" => { default_selection: "DEFAULT" } },
|
330
|
+
audio_selector_groups: {
|
331
|
+
"Audio Selector Group 1" => {
|
332
|
+
audio_selector_names: ["Audio Selector 1"]
|
333
|
+
}
|
334
|
+
},
|
335
|
+
file_input: input_url,
|
336
|
+
timecode_source: "ZEROBASED"
|
337
|
+
}
|
338
|
+
end
|
339
|
+
|
340
|
+
def make_video_input(input_url)
|
341
|
+
{
|
342
|
+
audio_selectors: { "Audio Selector 1" => { default_selection: "DEFAULT" } },
|
343
|
+
file_input: input_url,
|
344
|
+
timecode_source: "ZEROBASED",
|
345
|
+
video_selector: {}
|
346
|
+
}
|
347
|
+
end
|
348
|
+
|
349
|
+
def make_output_groups(options)
|
350
|
+
output_type = options[:output_type] || :hls
|
351
|
+
raise ArgumentError, "Unknown output type: #{output_type.inspect}" unless OUTPUT_GROUP_TEMPLATES.keys.include?(output_type)
|
352
|
+
output_group_settings_key = "#{output_type}_group_settings".to_sym
|
353
|
+
output_group_settings = OUTPUT_GROUP_TEMPLATES[output_type].merge(destination: "s3://#{output_bucket}/#{options[:output_prefix]}")
|
354
|
+
|
355
|
+
outputs = options[:outputs].map do |output|
|
356
|
+
{
|
357
|
+
preset: output[:preset],
|
358
|
+
name_modifier: output[:modifier]
|
359
|
+
}
|
360
|
+
end
|
361
|
+
|
362
|
+
[{
|
363
|
+
output_group_settings: {
|
364
|
+
type: output_group_settings_key.upcase,
|
365
|
+
output_group_settings_key => output_group_settings
|
366
|
+
},
|
367
|
+
outputs: outputs
|
368
|
+
}]
|
369
|
+
end
|
370
|
+
end
|
371
|
+
end
|
372
|
+
end
|