aliyun-rails 0.1.12 → 0.1.15

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 5dd2919d59678f49ad2cf11ee04b09699998223435406bca266b028450ba498e
4
- data.tar.gz: 339a221b039e588e6e0c7070b1264de67ca43260e1ee1f7f9423e54672b308f2
3
+ metadata.gz: 485e0e5b14e3aefe6c6e0865cf20ff3a043fe38d012c05f067b1d9d59a21940f
4
+ data.tar.gz: d18b4c36f7a38b64a25928a4222c23d7021851267f25f15099c1d8a73c6f9146
5
5
  SHA512:
6
- metadata.gz: 9eecf24324919ed9725024aa9a83091f7544c3fca085773aee5d4f6576f7a2fa7e7a3294d11785784a9e6f318833da01f1a451e6d4db7d6070ab55d33cf22d45
7
- data.tar.gz: 4423ce1c0668764732833a6a0a95f4230243000e91bb5576a09d2780f2a88224a511253bd8242dcd911eed6fadf5e583ba137d4b1cf97246d726f5dcca0ef8af
6
+ metadata.gz: 97971caf232a968deca9341f24613a98f2e671b0d762f2f198fdb266b119bf01994c1858d536817a084e27fb2e143bab3cbc5c1b1da99767f831e6ea6158d64c
7
+ data.tar.gz: 18bf61bee58194b9dcb6f462d08ebbf05d117ca0577854723f3ee0a9e68bfff428c99789a04623ca34742bfe918dbf748536f8777e280d6e1829790083f5fa7e
data/README.md CHANGED
@@ -5,7 +5,7 @@
5
5
  <a href=" https://www.alibabacloud.com"><img src="https://aliyunsdk-pages.alicdn.com/icons/Aliyun.svg"></a>
6
6
  </p>
7
7
 
8
- <h1 align="center">非官方SDK套件-用于RAILS项目管理</h1>
8
+ <h1 align="center">非官方SDK套件-用于RAILS项目管理阿里云资源</h1>
9
9
 
10
10
  <p align="center">
11
11
  <a href="https://badge.fury.io/rb/aliyunsdkcore"><img src="https://badge.fury.io/rb/aliyunsdkcore.svg" alt="Gem Version"></a>
@@ -15,9 +15,10 @@
15
15
  </p>
16
16
 
17
17
 
18
- 支持 Rails 轻松访问阿里云服务,例如:弹性云主机(ECS)、负载均衡(SLB)、云监控(CloudMonitor)等。
18
+ 支持 Rails 轻松访问阿里云服务,例如:弹性云主机(ECS)、负载均衡(SLB)、云监控(CloudMonitor)等。当前已集成短信服务、语音服务功能,后续可以根据业务需要拓展。
19
19
 
20
20
  本文档介绍如何安装和使用 aliyun-rails
21
+ 本项目是个人用于集成阿里云短信推送、电话告警功能编排,部分代码借鉴自官方 openapi-core-ruby-sdk
21
22
 
22
23
  ## 使用诊断
23
24
  [Troubleshoot](https://troubleshoot.api.aliyun.com/?source=github_sdk) 提供 OpenAPI 使用诊断服务,通过 `RequestID` 或 `报错信息` ,帮助开发者快速定位,为开发者提供解决方案。
@@ -0,0 +1,166 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "faraday"
4
+ require "securerandom"
5
+ require "active_support/all"
6
+ require "net/http"
7
+
8
+ module Net::HTTPHeader
9
+ def capitalize(name)
10
+ name
11
+ end
12
+ private :capitalize
13
+ end
14
+
15
+ module Aliyun
16
+ module Connector
17
+ class ROAClient
18
+ attr_accessor :endpoint, :api_version, :access_key_id,
19
+ :access_key_secret, :security_token, :hostname, :opts
20
+
21
+ def initialize(config)
22
+ validate config
23
+
24
+ self.endpoint = config[:endpoint]
25
+ self.api_version = config[:api_version]
26
+ self.access_key_id = config[:access_key_id] || Aliyun.access_key_id
27
+ self.access_key_secret = config[:access_key_secret] || Aliyun.access_key_secret
28
+ self.security_token = config[:security_token]
29
+ end
30
+
31
+ def request(method:, uri:, params: {}, body: {}, headers: {}, options: {})
32
+ # :"Content-Type" => "application/json" to "content-type" => "application/json"
33
+ headers.deep_transform_keys! { |key| key.to_s.downcase }
34
+ mix_headers = default_headers.merge(headers)
35
+
36
+ response = connection.send(method.downcase) do |request|
37
+ request.url uri, params
38
+ if body
39
+ if mix_headers["content-type"].start_with? "application/json"
40
+ request_body = body.to_json
41
+ elsif mix_headers["content-type"].start_with? "application/x-www-form-urlencoded"
42
+ request_body = URI.encode_www_form(body)
43
+ else
44
+ request_body = body
45
+ end
46
+ mix_headers["content-md5"] = Digest::MD5.base64digest request_body
47
+ mix_headers["content-length"] = request_body.length.to_s
48
+ request.body = request_body
49
+ end
50
+ string2sign = string_to_sign(method, uri, mix_headers, params)
51
+ mix_headers[:authorization] = authorization(string2sign)
52
+ mix_headers.each { |key, value| request.headers[key] = value }
53
+ end
54
+
55
+ return response if options.has_key? :raw_body
56
+
57
+ response_content_type = response.headers["Content-Type"] || ""
58
+ if response_content_type.start_with?("application/json")
59
+ if response.status >= 400
60
+ result = JSON.parse(response.body)
61
+ raise StandardError, "code: #{response.status}, #{result['Message']} requestid: #{result['RequestId']}"
62
+ end
63
+ end
64
+
65
+ if response_content_type.start_with?("text/xml")
66
+ result = Hash.from_xml(response.body)
67
+ raise ACSError, result["Error"] if result["Error"]
68
+ end
69
+
70
+ response
71
+ end
72
+
73
+ def connection(adapter = Faraday.default_adapter)
74
+ Faraday.new(url: self.endpoint) { |f| f.adapter adapter }
75
+ end
76
+
77
+ def get(uri: "", headers: {}, params: {}, options: {})
78
+ request(method: :get, uri: uri, params: params, body: {}, headers: headers, options: options)
79
+ end
80
+
81
+ def post(uri: "", headers: {}, params: {}, body: {}, options: {})
82
+ request(method: :post, uri: uri, params: params, body: body, headers: headers, options: options)
83
+ end
84
+
85
+ def put(uri: "", headers: {}, params: {}, body: {}, options: {})
86
+ request(method: :put, uri: uri, params: params, body: body, headers: headers, options: options)
87
+ end
88
+
89
+ def delete(uri: "", headers: {}, params: {}, options: {})
90
+ request(method: :delete, uri: uri, params: params, body: {}, headers: headers, options: options)
91
+ end
92
+
93
+ def default_headers
94
+ default_headers = {
95
+ "accept" => "application/json",
96
+ "date" => Time.now.httpdate,
97
+ "host" => URI(self.endpoint).host,
98
+ "x-acs-signature-nonce" => SecureRandom.hex(16),
99
+ "x-acs-signature-method" => "HMAC-SHA1",
100
+ "x-acs-signature-version" => "1.0",
101
+ "x-acs-version" => self.api_version,
102
+ "x-sdk-client" => "RUBY(#{RUBY_VERSION})", # FIXME: 如何获取Gem的名称和版本号
103
+ "user-agent" => DEFAULT_UA
104
+ }
105
+ if self.security_token
106
+ default_headers["x-acs-accesskey-id"] = self.access_key_id
107
+ default_headers["x-acs-security-token"] = self.security_token
108
+ end
109
+ default_headers
110
+ end
111
+
112
+ private
113
+ def string_to_sign(method, uri, headers, query = {})
114
+ header_string = [
115
+ method,
116
+ headers["accept"],
117
+ headers["content-md5"] || "",
118
+ headers["content-type"] || "",
119
+ headers["date"],
120
+ ].join("\n")
121
+ "#{header_string}\n#{canonicalized_headers(headers)}#{canonicalized_resource(uri, query)}"
122
+ end
123
+
124
+ def canonicalized_headers(headers)
125
+ headers.keys.select { |key| key.to_s.start_with? "x-acs-" }
126
+ .sort.map { |key| "#{key}:#{headers[key].strip}\n" }.join
127
+ end
128
+
129
+ def canonicalized_resource(uri, query_hash = {})
130
+ query_string = query_hash.sort.map { |key, value| "#{key}=#{value}" }.join("&")
131
+ query_string.empty? ? uri : "#{uri}?#{query_string}"
132
+ end
133
+
134
+ def authorization(string_to_sign)
135
+ "acs #{self.access_key_id}:#{signature(string_to_sign)}"
136
+ end
137
+
138
+ def signature(string_to_sign)
139
+ Base64.encode64(OpenSSL::HMAC.digest("sha1", self.access_key_secret, string_to_sign)).strip
140
+ end
141
+
142
+ def validate(config)
143
+ raise ArgumentError, 'must pass "config"' unless config
144
+ raise ArgumentError, 'must pass "config[:endpoint]"' unless config[:endpoint]
145
+ unless config[:endpoint].start_with?("http://") || config[:endpoint].start_with?("https://")
146
+ raise ArgumentError, '"config.endpoint" must starts with \'https://\' or \'http://\'.'
147
+ end
148
+ raise ArgumentError, 'must pass "config[:api_version]"' unless config[:api_version]
149
+ raise ArgumentError, 'must pass "config[:access_key_id]"' unless config[:access_key_id]
150
+ raise ArgumentError, 'must pass "config[:access_key_secret]"' unless config[:access_key_secret]
151
+ end
152
+
153
+ class ACSError < StandardError
154
+ attr_accessor :code
155
+
156
+ def initialize(error)
157
+ self.code = error["Code"]
158
+ message = error["Message"]
159
+ host_id = error["HostId"]
160
+ request_id = error["RequestId"]
161
+ super("#{message} host_id: #{host_id}, request_id: #{request_id}")
162
+ end
163
+ end
164
+ end
165
+ end
166
+ end
@@ -0,0 +1,116 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "set"
4
+ require "openssl"
5
+ require "faraday"
6
+ require "erb"
7
+ require "active_support/all"
8
+
9
+ module Aliyun
10
+ module Connector
11
+ class RPCClient
12
+ attr_accessor :endpoint, :api_version, :access_key_id, :access_key_secret,
13
+ :security_token, :codes, :opts, :verbose
14
+
15
+ # 对象初始化属性
16
+ def initialize(config = {}, verbose = false)
17
+ validate config
18
+
19
+ self.endpoint = config[:endpoint]
20
+ self.api_version = config[:api_version]
21
+ self.access_key_id = config[:access_key_id] || Aliyun.access_key_id
22
+ self.access_key_secret = config[:access_key_secret] || Aliyun.access_key_secret
23
+ self.security_token = config[:security_token]
24
+ self.opts = config[:opts] || {}
25
+ self.verbose = verbose.instance_of?(TrueClass) && verbose
26
+
27
+ self.codes = Set.new [200, "200", "OK", "Success"]
28
+ self.codes.merge config[:codes] if config[:codes]
29
+ end
30
+
31
+ # 通用请求接口
32
+ def request(action:, params: {}, opts: {})
33
+ opts = self.opts.merge(opts)
34
+ action = action.upcase_first if opts[:format_action]
35
+ params = format_params(params) unless opts[:format_params]
36
+ defaults = default_params
37
+ params = { Action: action }.merge(defaults).merge(params)
38
+ method = (opts[:method] || "GET").upcase
39
+ sign = "#{method}&#{encode('/')}&#{encode(params.to_query)}"
40
+ secret = "#{self.access_key_secret}&"
41
+ signature = Base64.encode64(OpenSSL::HMAC.digest("sha1", secret, sign)).strip
42
+ params["Signature"] = signature
43
+
44
+ # 转换为 query 样式
45
+ query_string = params.to_query
46
+
47
+ # 特殊处理 POST
48
+ uri = opts[:method] == "POST" ? "/" : "/?#{query_string}"
49
+
50
+ # 初始化会话
51
+ response = connection.send(method.downcase, uri) do |r|
52
+ if opts[:method] == "POST"
53
+ r.headers["Content-Type"] = "application/x-www-form-urlencoded"
54
+ r.body = query_string
55
+ end
56
+ r.headers["User-Agent"] = DEFAULT_UA
57
+ end
58
+
59
+ # 解析接口响应
60
+ response_body = JSON.parse(response.body)
61
+ if response_body["Code"] && !response_body["Code"].to_s.empty? && !self.codes.include?(response_body["Code"])
62
+ raise StandardError, "Code: #{response_body['Code']}, Message: #{response_body['Message']}, URL: #{uri}"
63
+ end
64
+
65
+ response_body
66
+ end
67
+
68
+ private
69
+ def connection(adapter = Faraday.default_adapter)
70
+ Faraday.new(url: self.endpoint) { |f| f.adapter adapter }
71
+ end
72
+
73
+ # 设置缺省参数
74
+ def default_params
75
+ params = {
76
+ Format: "JSON",
77
+ SignatureMethod: "HMAC-SHA1",
78
+ SignatureNonce: SecureRandom.hex(8),
79
+ SignatureVersion: "1.0",
80
+ Timestamp: Time.now.utc.strftime("%Y-%m-%dT%H:%M:%SZ"),
81
+ AccessKeyId: self.access_key_id,
82
+ Version: self.api_version,
83
+ }
84
+ params[:SecurityToken] = self.security_token if self.security_token.present?
85
+ params
86
+ end
87
+
88
+ # 消息签名需要
89
+ def encode(input)
90
+ ERB::Util.url_encode input
91
+ end
92
+
93
+ # 转换 HASH key 样式
94
+ def format_params(param_hash)
95
+ param_hash.keys.each { |key| param_hash[(key.to_s.upcase_first).to_sym] = param_hash.delete key }
96
+ param_hash
97
+ end
98
+
99
+ def validate(config = {})
100
+ config.with_indifferent_access
101
+ raise ArgumentError, 'must pass "config"' unless config
102
+ raise ArgumentError, 'must pass "config[:endpoint]"' unless config[:endpoint]
103
+ unless config[:endpoint].match?(/^http[s]?:/i)
104
+ raise ArgumentError, '"config.endpoint" must starts with \'https://\' or \'http://\'.'
105
+ end
106
+ raise ArgumentError, 'must pass "config[:api_version]"' unless config[:api_version]
107
+ unless config[:access_key_id] || Aliyun.access_key_id
108
+ raise ArgumentError, 'must pass "config[:access_key_id]" or define "Aliyun.access_key_id"'
109
+ end
110
+ unless config[:access_key_secret] || Aliyun.access_key_secret
111
+ raise ArgumentError, 'must pass "config[:access_key_secret]" or define "Aliyun.access_key_secret"'
112
+ end
113
+ end
114
+ end
115
+ end
116
+ end
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Aliyun
4
+ class Dysms < Aliyun::Connector::RPCClient
5
+ # 本产品(Dysmsapi/2017-05-25)的OpenAPI采用RPC签名风格,签名细节参见签名机制说明。
6
+ # 我们已经为开发者封装了常见编程语言的SDK,开发者可通过下载SDK直接调用本产品OpenAPI而无需关心技术细节。
7
+ def initialize(config = {}, verbose = nil)
8
+ config[:endpoint] ||= "http://dysmsapi.aliyuncs.com"
9
+ config[:api_version] ||= "2017-05-25"
10
+ super(config, verbose)
11
+ end
12
+
13
+ # 发送短信,发送前要申请短信签名和短信模板,并确保签名和模板已审核通过。
14
+ def send_sms(phone_numbers, template_code, template_param, sign_name = "")
15
+ params = {
16
+ PhoneNumbers: phone_numbers,
17
+ SignName: sign_name,
18
+ TemplateCode: template_code,
19
+ TemplateParam: template_param.to_json
20
+ }
21
+ opts = { method: "POST", timeout: 15000 }
22
+ request(action: "SendSms", params: params, opts: opts)
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Aliyun
4
+ class Dyvms < Aliyun::Connector::RPCClient
5
+ # 本产品(Dyvmsapi/2017-05-25)的OpenAPI采用RPC签名风格,签名细节参见签名机制说明。
6
+ # 我们已经为开发者封装了常见编程语言的SDK,开发者可通过下载SDK直接调用本产品OpenAPI而无需关心技术细节。
7
+ def initialize(config = {}, verbose = nil)
8
+ config[:endpoint] ||= "http://dyvmsapi.aliyuncs.com"
9
+ config[:api_version] ||= "2017-05-25"
10
+ super(config, verbose)
11
+ end
12
+
13
+ # 调用SingleCallByTts接口向指定号码发送语音验证码和带参数变量的语音通知
14
+ def single_call_by_tts(called_show_number, called_number, tts_code, tts_param)
15
+ params = {
16
+ CalledShowNumber: called_show_number,
17
+ CalledNumber: called_number,
18
+ TtsCode: tts_code,
19
+ TtsParam: tts_param.to_json
20
+ }
21
+ opts = { method: "POST", timeout: 15000 }
22
+ request(action: "SendSms", params: params, opts: opts)
23
+ end
24
+ end
25
+ end
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Aliyun
4
4
  module Rails
5
- VERSION = "0.1.12"
5
+ VERSION = "0.1.15"
6
6
  DEFAULT_UA = "AlibabaCloud (#{Gem::Platform.local.os}; " +
7
7
  "#{Gem::Platform.local.cpu}) Ruby/#{RUBY_VERSION} Core/#{VERSION}"
8
8
  end
@@ -0,0 +1,20 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "aliyun/version"
4
+ require_relative "aliyun/connector/roa_client"
5
+ require_relative "aliyun/connector/rpc_client"
6
+ require_relative "aliyun/dysms"
7
+ require_relative "aliyun/dyvms"
8
+
9
+ module Aliyun
10
+ class Error < StandardError; end
11
+
12
+ class << self
13
+ attr_accessor :access_key_id, :access_key_secret
14
+ def config
15
+ yield self
16
+ end
17
+ end
18
+
19
+ # Your code goes here...
20
+ end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: aliyun-rails
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.12
4
+ version: 0.1.15
5
5
  platform: ruby
6
6
  authors:
7
7
  - WENWU.YAN
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2022-04-01 00:00:00.000000000 Z
11
+ date: 2022-04-02 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: activesupport
@@ -38,7 +38,7 @@ dependencies:
38
38
  - - ">="
39
39
  - !ruby/object:Gem::Version
40
40
  version: 0.15.4
41
- description: 支持 Rails 轻松访问阿里云服务,例如:弹性云主机(ECS)、负载均衡(SLB)、云监控(CloudMonitor)等
41
+ description: 支持 Rails 轻松访问阿里云服务,例如:弹性云主机(ECS)、负载均衡(SLB)、云监控(CloudMonitor)等。当前已支持短信服务、语音服务,可以直接调用!
42
42
  email:
43
43
  - careline@foxmail.com
44
44
  executables: []
@@ -47,19 +47,19 @@ extra_rdoc_files: []
47
47
  files:
48
48
  - README.md
49
49
  - Rakefile
50
- - lib/aliyun/rails.rb
51
- - lib/aliyun/rails/connector/roa_client.rb
52
- - lib/aliyun/rails/connector/rpc_client.rb
53
- - lib/aliyun/rails/dysms.rb
54
- - lib/aliyun/rails/dyvms.rb
55
- - lib/aliyun/rails/version.rb
50
+ - lib/aliyun-rails.rb
51
+ - lib/aliyun/connector/roa_client.rb
52
+ - lib/aliyun/connector/rpc_client.rb
53
+ - lib/aliyun/dysms.rb
54
+ - lib/aliyun/dyvms.rb
55
+ - lib/aliyun/version.rb
56
56
  homepage: https://github.com/ciscolive/aliyun-rails
57
57
  licenses:
58
58
  - MIT
59
59
  metadata:
60
60
  homepage_uri: https://github.com/ciscolive/aliyun-rails
61
61
  source_code_uri: https://github.com/ciscolive/aliyun-rails
62
- changelog_uri: https://github.com/ciscolive/aliyun-rails/blob/main/README.md
62
+ changelog_uri: https://github.com/ciscolive/aliyun-rails/blob/main/CHANGELOG.md
63
63
  post_install_message:
64
64
  rdoc_options: []
65
65
  require_paths:
@@ -1,168 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- require "faraday"
4
- require "securerandom"
5
- require "active_support/all"
6
- require "net/http"
7
-
8
- module Net::HTTPHeader
9
- def capitalize(name)
10
- name
11
- end
12
- private :capitalize
13
- end
14
-
15
- module Aliyun
16
- module Rails
17
- module Connector
18
- class ROAClient
19
- attr_accessor :endpoint, :api_version, :access_key_id,
20
- :access_key_secret, :security_token, :hostname, :opts
21
-
22
- def initialize(config)
23
- validate config
24
-
25
- self.endpoint = config[:endpoint]
26
- self.api_version = config[:api_version]
27
- self.access_key_id = config[:access_key_id]
28
- self.access_key_secret = config[:access_key_secret]
29
- self.security_token = config[:security_token]
30
- end
31
-
32
- def request(method:, uri:, params: {}, body: {}, headers: {}, options: {})
33
- # :"Content-Type" => "application/json" to "content-type" => "application/json"
34
- headers.deep_transform_keys! { |key| key.to_s.downcase }
35
- mix_headers = default_headers.merge(headers)
36
-
37
- response = connection.send(method.downcase) do |request|
38
- request.url uri, params
39
- if body
40
- if mix_headers["content-type"].start_with? "application/json"
41
- request_body = body.to_json
42
- elsif mix_headers["content-type"].start_with? "application/x-www-form-urlencoded"
43
- request_body = URI.encode_www_form(body)
44
- else
45
- request_body = body
46
- end
47
- mix_headers["content-md5"] = Digest::MD5.base64digest request_body
48
- mix_headers["content-length"] = request_body.length.to_s
49
- request.body = request_body
50
- end
51
- string2sign = string_to_sign(method, uri, mix_headers, params)
52
- mix_headers[:authorization] = authorization(string2sign)
53
- mix_headers.each { |key, value| request.headers[key] = value }
54
- end
55
-
56
- return response if options.has_key? :raw_body
57
-
58
- response_content_type = response.headers["Content-Type"] || ""
59
- if response_content_type.start_with?("application/json")
60
- if response.status >= 400
61
- result = JSON.parse(response.body)
62
- raise StandardError, "code: #{response.status}, #{result['Message']} requestid: #{result['RequestId']}"
63
- end
64
- end
65
-
66
- if response_content_type.start_with?("text/xml")
67
- result = Hash.from_xml(response.body)
68
- raise ACSError, result["Error"] if result["Error"]
69
- end
70
-
71
- response
72
- end
73
-
74
- def connection(adapter = Faraday.default_adapter)
75
- Faraday.new(url: self.endpoint) { |f| f.adapter adapter }
76
- end
77
-
78
- def get(uri: "", headers: {}, params: {}, options: {})
79
- request(method: :get, uri: uri, params: params, body: {}, headers: headers, options: options)
80
- end
81
-
82
- def post(uri: "", headers: {}, params: {}, body: {}, options: {})
83
- request(method: :post, uri: uri, params: params, body: body, headers: headers, options: options)
84
- end
85
-
86
- def put(uri: "", headers: {}, params: {}, body: {}, options: {})
87
- request(method: :put, uri: uri, params: params, body: body, headers: headers, options: options)
88
- end
89
-
90
- def delete(uri: "", headers: {}, params: {}, options: {})
91
- request(method: :delete, uri: uri, params: params, body: {}, headers: headers, options: options)
92
- end
93
-
94
- def default_headers
95
- default_headers = {
96
- "accept" => "application/json",
97
- "date" => Time.now.httpdate,
98
- "host" => URI(self.endpoint).host,
99
- "x-acs-signature-nonce" => SecureRandom.hex(16),
100
- "x-acs-signature-method" => "HMAC-SHA1",
101
- "x-acs-signature-version" => "1.0",
102
- "x-acs-version" => self.api_version,
103
- "x-sdk-client" => "RUBY(#{RUBY_VERSION})", # FIXME: 如何获取Gem的名称和版本号
104
- "user-agent" => DEFAULT_UA
105
- }
106
- if self.security_token
107
- default_headers["x-acs-accesskey-id"] = self.access_key_id
108
- default_headers["x-acs-security-token"] = self.security_token
109
- end
110
- default_headers
111
- end
112
-
113
- private
114
- def string_to_sign(method, uri, headers, query = {})
115
- header_string = [
116
- method,
117
- headers["accept"],
118
- headers["content-md5"] || "",
119
- headers["content-type"] || "",
120
- headers["date"],
121
- ].join("\n")
122
- "#{header_string}\n#{canonicalized_headers(headers)}#{canonicalized_resource(uri, query)}"
123
- end
124
-
125
- def canonicalized_headers(headers)
126
- headers.keys.select { |key| key.to_s.start_with? "x-acs-" }
127
- .sort.map { |key| "#{key}:#{headers[key].strip}\n" }.join
128
- end
129
-
130
- def canonicalized_resource(uri, query_hash = {})
131
- query_string = query_hash.sort.map { |key, value| "#{key}=#{value}" }.join("&")
132
- query_string.empty? ? uri : "#{uri}?#{query_string}"
133
- end
134
-
135
- def authorization(string_to_sign)
136
- "acs #{self.access_key_id}:#{signature(string_to_sign)}"
137
- end
138
-
139
- def signature(string_to_sign)
140
- Base64.encode64(OpenSSL::HMAC.digest("sha1", self.access_key_secret, string_to_sign)).strip
141
- end
142
-
143
- def validate(config)
144
- raise ArgumentError, 'must pass "config"' unless config
145
- raise ArgumentError, 'must pass "config[:endpoint]"' unless config[:endpoint]
146
- unless config[:endpoint].start_with?("http://") || config[:endpoint].start_with?("https://")
147
- raise ArgumentError, '"config.endpoint" must starts with \'https://\' or \'http://\'.'
148
- end
149
- raise ArgumentError, 'must pass "config[:api_version]"' unless config[:api_version]
150
- raise ArgumentError, 'must pass "config[:access_key_id]"' unless config[:access_key_id]
151
- raise ArgumentError, 'must pass "config[:access_key_secret]"' unless config[:access_key_secret]
152
- end
153
-
154
- class ACSError < StandardError
155
- attr_accessor :code
156
-
157
- def initialize(error)
158
- self.code = error["Code"]
159
- message = error["Message"]
160
- host_id = error["HostId"]
161
- request_id = error["RequestId"]
162
- super("#{message} host_id: #{host_id}, request_id: #{request_id}")
163
- end
164
- end
165
- end
166
- end
167
- end
168
- end
@@ -1,117 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- require "set"
4
- require "openssl"
5
- require "faraday"
6
- require "erb"
7
- require "active_support/all"
8
-
9
- module Aliyun
10
- module Rails
11
- module Connector
12
- class RPCClient
13
- attr_accessor :endpoint, :api_version, :access_key_id, :access_key_secret,
14
- :security_token, :codes, :opts, :verbose
15
-
16
- # 对象初始化属性
17
- def initialize(config = {}, verbose = false)
18
- validate config
19
-
20
- self.endpoint = config[:endpoint]
21
- self.api_version = config[:api_version]
22
- self.access_key_id = config[:access_key_id] || Aliyun::Rails.access_key_id
23
- self.access_key_secret = config[:access_key_secret] || Aliyun::Rails.access_key_secret
24
- self.security_token = config[:security_token]
25
- self.opts = config[:opts] || {}
26
- self.verbose = verbose.instance_of?(TrueClass) && verbose
27
- self.codes = Set.new [200, "200", "OK", "Success"]
28
- self.codes.merge config[:codes] if config[:codes]
29
- end
30
-
31
- # 通用请求接口
32
- def request(action:, params: {}, opts: {})
33
- opts = self.opts.merge(opts)
34
- action = action.upcase_first if opts[:format_action]
35
- params = format_params(params) unless opts[:format_params]
36
- defaults = default_params
37
- params = { Action: action }.merge(defaults).merge(params)
38
- method = (opts[:method] || "GET").upcase
39
- sign = "#{method}&#{encode('/')}&#{encode(params.to_query)}"
40
- secret = "#{self.access_key_secret}&"
41
- signature = Base64.encode64(OpenSSL::HMAC.digest("sha1", secret, sign)).strip
42
- params["Signature"] = signature
43
-
44
- # 转换为 query 样式
45
- query_string = params.to_query
46
-
47
- # 特殊处理 POST
48
- uri = opts[:method] == "POST" ? "/" : "/?#{query_string}"
49
-
50
- # 初始化会话
51
- response = connection.send(method.downcase, uri) do |r|
52
- if opts[:method] == "POST"
53
- r.headers["Content-Type"] = "application/x-www-form-urlencoded"
54
- r.body = query_string
55
- end
56
- r.headers["User-Agent"] = DEFAULT_UA
57
- end
58
-
59
- # 解析接口响应
60
- response_body = JSON.parse(response.body)
61
- if response_body["Code"] && !response_body["Code"].to_s.empty? && !self.codes.include?(response_body["Code"])
62
- raise StandardError, "Code: #{response_body['Code']}, Message: #{response_body['Message']}, URL: #{uri}"
63
- end
64
-
65
- response_body
66
- end
67
-
68
- private
69
- def connection(adapter = Faraday.default_adapter)
70
- Faraday.new(url: self.endpoint) { |f| f.adapter adapter }
71
- end
72
-
73
- # 设置缺省参数
74
- def default_params
75
- params = {
76
- Format: "JSON",
77
- SignatureMethod: "HMAC-SHA1",
78
- SignatureNonce: SecureRandom.hex(8),
79
- SignatureVersion: "1.0",
80
- Timestamp: Time.now.utc.strftime("%Y-%m-%dT%H:%M:%SZ"),
81
- AccessKeyId: self.access_key_id,
82
- Version: self.api_version,
83
- }
84
- params[:SecurityToken] = self.security_token if self.security_token.present?
85
- params
86
- end
87
-
88
- # 消息签名需要
89
- def encode(input)
90
- ERB::Util.url_encode input
91
- end
92
-
93
- # 转换 HASH key 样式
94
- def format_params(param_hash)
95
- param_hash.keys.each { |key| param_hash[(key.to_s.upcase_first).to_sym] = param_hash.delete key }
96
- param_hash
97
- end
98
-
99
- def validate(config)
100
- config.with_indifferent_access
101
- raise ArgumentError, 'must pass "config"' unless config
102
- raise ArgumentError, 'must pass "config[:endpoint]"' unless config[:endpoint]
103
- unless config[:endpoint].match?(/^http[s]?:/i)
104
- raise ArgumentError, '"config.endpoint" must starts with \'https://\' or \'http://\'.'
105
- end
106
- raise ArgumentError, 'must pass "config[:api_version]"' unless config[:api_version]
107
- unless config[:access_key_id] || Aliyun::Rails.access_key_id
108
- raise ArgumentError, 'must pass "config[:access_key_id]" or define "Aliyun::Rails.access_key_id"'
109
- end
110
- unless config[:access_key_secret] || Aliyun::Rails.access_key_secret
111
- raise ArgumentError, 'must pass "config[:access_key_secret]" or define "Aliyun::Rails.access_key_secret"'
112
- end
113
- end
114
- end
115
- end
116
- end
117
- end
@@ -1,27 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module Aliyun
4
- module Rails
5
- class Dysms < Aliyun::Rails::Connector::RPCClient
6
- # 本产品(Dysmsapi/2017-05-25)的OpenAPI采用RPC签名风格,签名细节参见签名机制说明。
7
- # 我们已经为开发者封装了常见编程语言的SDK,开发者可通过下载SDK直接调用本产品OpenAPI而无需关心技术细节。
8
- def initialize(config = {}, verbose = nil)
9
- config[:endpoint] ||= "http://dysmsapi.aliyuncs.com"
10
- config[:api_version] ||= "2017-05-25"
11
- super(config, verbose)
12
- end
13
-
14
- # 发送短信,发送前要申请短信签名和短信模板,并确保签名和模板已审核通过。
15
- def send_sms(phone_numbers, template_code, template_param, sign_name = "")
16
- params = {
17
- PhoneNumbers: phone_numbers,
18
- SignName: sign_name,
19
- TemplateCode: template_code,
20
- TemplateParam: template_param.to_json
21
- }
22
- opts = { method: "POST", timeout: 15000 }
23
- request(action: "SendSms", params: params, opts: opts)
24
- end
25
- end
26
- end
27
- end
@@ -1,27 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module Aliyun
4
- module Rails
5
- class Dyvms < Aliyun::Rails::Connector::RPCClient
6
- # 本产品(Dyvmsapi/2017-05-25)的OpenAPI采用RPC签名风格,签名细节参见签名机制说明。
7
- # 我们已经为开发者封装了常见编程语言的SDK,开发者可通过下载SDK直接调用本产品OpenAPI而无需关心技术细节。
8
- def initialize(config = {}, verbose = nil)
9
- config[:endpoint] ||= "http://dyvmsapi.aliyuncs.com"
10
- config[:api_version] ||= "2017-05-25"
11
- super(config, verbose)
12
- end
13
-
14
- # 调用SingleCallByTts接口向指定号码发送语音验证码和带参数变量的语音通知
15
- def single_call_by_tts(called_show_number, called_number, tts_code, tts_param)
16
- params = {
17
- CalledShowNumber: called_show_number,
18
- CalledNumber: called_number,
19
- TtsCode: tts_code,
20
- TtsParam: tts_param.to_json
21
- }
22
- opts = { method: "POST", timeout: 15000 }
23
- request(action: "SendSms", params: params, opts: opts)
24
- end
25
- end
26
- end
27
- end
data/lib/aliyun/rails.rb DELETED
@@ -1,22 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- require_relative "rails/version"
4
- require_relative "rails/connector/roa_client"
5
- require_relative "rails/connector/rpc_client"
6
- require_relative "rails/dyvms"
7
- require_relative "rails/dysms"
8
-
9
- module Aliyun
10
- module Rails
11
- class Error < StandardError; end
12
-
13
- class << self
14
- attr_accessor :access_key_id, :access_key_secret
15
- def config
16
- yield self
17
- end
18
- end
19
-
20
- # Your code goes here...
21
- end
22
- end