relishable 0.19 → 0.20

Sign up to get free protection for your applications and to get access to all the features.
data/lib/relish.rb CHANGED
@@ -1,4 +1,5 @@
1
1
  require "relish/dynamo_helper"
2
+ require "relish/cloudfront_helper"
2
3
  require "relish/s3_helper"
3
4
  require "relish/encryption_helper"
4
5
  require "relish/release"
@@ -0,0 +1,65 @@
1
+ # Adapted from https://github.com/stlondemand/aws_cf_signer
2
+ # Modified to support in-memory instead of on-disk signing keys
3
+ require 'openssl'
4
+ require 'time'
5
+ require 'base64'
6
+
7
+ class Relish
8
+ class CloudFrontHelper
9
+
10
+ attr_reader :key_pair_id
11
+
12
+ def initialize(pem, key_pair_id)
13
+ @key = OpenSSL::PKey::RSA.new(pem)
14
+ @key_pair_id = key_pair_id
15
+ end
16
+
17
+ def sign(url_to_sign, policy_options = {})
18
+ separator = url_to_sign =~ /\?/ ? '&' : '?'
19
+ if policy_options[:policy_file]
20
+ policy = IO.read(policy_options[:policy_file])
21
+ "#{url_to_sign}#{separator}Policy=#{encode_policy(policy)}&Signature=#{create_signature(policy)}&Key-Pair-Id=#{@key_pair_id}"
22
+ else
23
+ raise ArgumentError.new("'ending' argument is required") if policy_options[:ending].nil?
24
+ if policy_options.keys.size == 1
25
+ # Canned Policy - shorter URL
26
+ expires_at = epoch_time(policy_options[:ending])
27
+ policy = %({"Statement":[{"Resource":"#{url_to_sign}","Condition":{"DateLessThan":{"AWS:EpochTime":#{expires_at}}}}]})
28
+ "#{url_to_sign}#{separator}Expires=#{expires_at}&Signature=#{create_signature(policy)}&Key-Pair-Id=#{@key_pair_id}"
29
+ else
30
+ # Custom Policy
31
+ resource = policy_options[:resource] || url_to_sign
32
+ policy = generate_custom_policy(resource, policy_options)
33
+ "#{url_to_sign}#{separator}Policy=#{encode_policy(policy)}&Signature=#{create_signature(policy)}&Key-Pair-Id=#{@key_pair_id}"
34
+ end
35
+ end
36
+ end
37
+
38
+ def generate_custom_policy(resource, options)
39
+ conditions = ["\"DateLessThan\":{\"AWS:EpochTime\":#{epoch_time(options[:ending])}}"]
40
+ conditions << "\"DateGreaterThan\":{\"AWS:EpochTime\":#{epoch_time(options[:starting])}}" if options[:starting]
41
+ conditions << "\"IpAddress\":{\"AWS:SourceIp\":\"#{options[:ip_range]}\"" if options[:ip_range]
42
+ %({"Statement":[{"Resource":"#{resource}","Condition":{#{conditions.join(',')}}}}]})
43
+ end
44
+
45
+ def epoch_time(timelike)
46
+ case timelike
47
+ when String then Time.parse(timelike).to_i
48
+ when Time then timelike.to_i
49
+ else raise ArgumentError.new("Invalid argument - String or Time required - #{timelike.class} passed.")
50
+ end
51
+ end
52
+
53
+ def encode_policy(policy)
54
+ url_safe(Base64.encode64(policy))
55
+ end
56
+
57
+ def create_signature(policy)
58
+ url_safe(Base64.encode64(@key.sign(OpenSSL::Digest::SHA1.new, (policy))))
59
+ end
60
+
61
+ def url_safe(s)
62
+ s.gsub('+','-').gsub('=','_').gsub('/','~').gsub(/\n/,'').gsub(' ','')
63
+ end
64
+ end
65
+ end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: relishable
3
3
  version: !ruby/object:Gem::Version
4
- version: '0.19'
4
+ version: '0.20'
5
5
  prerelease:
6
6
  platform: ruby
7
7
  authors:
@@ -50,6 +50,7 @@ executables: []
50
50
  extensions: []
51
51
  extra_rdoc_files: []
52
52
  files:
53
+ - lib/relish/cloudfront_helper.rb
53
54
  - lib/relish/dynamo_helper.rb
54
55
  - lib/relish/encryption_helper.rb
55
56
  - lib/relish/release.rb