aliyun-rails 0.1.13 → 0.1.16

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: 54b953cc3442bb86cdfccfb2fe1cd80f7237280f67384588d27effb11cadb580
4
- data.tar.gz: 17e808edae1fac737e133b2fecf62a6b4923cc21490a5c4bf83547b261dd0be9
3
+ metadata.gz: 3240b73e02b5bf35dfa310205d8dae87e0850fdd55a89f61ffd3ede196b4dd2f
4
+ data.tar.gz: 1b547158d62ab63a243c32e4e4b53b7aa793f29373857585a7e24f1b989e688a
5
5
  SHA512:
6
- metadata.gz: c6642a259e9b175b99a97b7acf03c2931a3ff02a01b03e01118a20cec4f51ac20b5bd250ab1c821fa41103ea78f379d476416b32438dd5fc7c28c2b1a1c9b598
7
- data.tar.gz: 3aa40ca9000cdc6b2ca7618f25d9e2742c99048d8d496f41fd3715ed678f3fb47de60168518facb66d18ebbb90749cd13dc316e1f8deb21188a94265491f60cc
6
+ metadata.gz: 17a9fe79cb951e258a942b1e429ae99d28ca94a8d5b36b7e6df3e3f0a179f1d565c9fa7d5c0f7f88e9323e44810fdf1fe3ec9cca42a26528e9d7116d0a6978f4
7
+ data.tar.gz: ae4449f0facadcecb91d5f752b9129eb3c3f0ac527c1d04d601e0008d39041b718dd58c062560ebc342ebba2c069e550229717e707526cc52a986628d7b386d5
data/README.md CHANGED
@@ -5,23 +5,25 @@
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
- <a href="https://badge.fury.io/rb/aliyunsdkcore"><img src="https://badge.fury.io/rb/aliyunsdkcore.svg" alt="Gem Version"></a>
12
- <a href="https://travis-ci.org/aliyun/openapi-core-ruby-sdk"><img src="https://travis-ci.org/aliyun/openapi-core-ruby-sdk.svg?branch=master" alt="Build Status"></a>
13
- <a href="https://ci.appveyor.com/project/aliyun/openapi-core-ruby-sdk/branch/master"><img src="https://ci.appveyor.com/api/projects/status/uyepkk5bjbynofvu/branch/master?svg=true" alt="Build status"></a>
14
- <a href="https://codecov.io/gh/aliyun/openapi-core-ruby-sdk"><img src="https://codecov.io/gh/aliyun/openapi-core-ruby-sdk/branch/master/graph/badge.svg" alt="codecov"></a>
11
+ <a href="https://badge.fury.io/rb/aliyun-rails"><img src="https://badge.fury.io/rb/aliyun-rails.svg" alt="Gem Version"></a>
12
+ <a href="https://travis-ci.org/ciscolive/aliyun-rails"><img src="https://travis-ci.org/ciscolive/aliyun-rails.svg?branch=master" alt="Build Status"></a>
13
+ <a href="https://ci.appveyor.com/project/ciscolive/aliyun-rails/branch/master"><img src="https://ci.appveyor.com/api/projects/status/uyepkk5bjbynofvu/branch/master?svg=true" alt="Build status"></a>
14
+ <a href="https://codecov.io/gh/ciscolive/aliyun-rails"><img src="https://codecov.io/gh/ciscolive/aliyun-rails/branch/master/graph/badge.svg" alt="codecov"></a>
15
15
  </p>
16
16
 
17
17
 
18
- 支持 Rails 轻松访问阿里云服务,例如:弹性云主机(ECS)、负载均衡(SLB)、云监控(CloudMonitor)等。当前已集成短信服务、语音服务功能,后续可以根据业务需要拓展。
18
+ - 支持 Rails 轻松访问阿里云服务,例如:弹性云主机(ECS)、负载均衡(SLB)、云监控(CloudMonitor)等。
19
+ - 已集成阿里云短信服务和语音服务,满足日常短信和电话告警需求。
19
20
 
20
21
  本文档介绍如何安装和使用 aliyun-rails
21
- 本项目是个人用于集成阿里云短信推送、电话告警功能编排,部分代码借鉴自官方 openapi-core-ruby-sdk
22
22
 
23
23
  ## 使用诊断
24
- [Troubleshoot](https://troubleshoot.api.aliyun.com/?source=github_sdk) 提供 OpenAPI 使用诊断服务,通过 `RequestID` 或 `报错信息` ,帮助开发者快速定位,为开发者提供解决方案。
24
+
25
+ [Troubleshoot](https://troubleshoot.api.aliyun.com/?source=github_sdk) 提供 OpenAPI 使用诊断服务,通过 `RequestID` 或 `报错信息`
26
+ ,帮助开发者快速定位,为开发者提供解决方案。
25
27
 
26
28
  ## 安装
27
29
 
@@ -31,36 +33,90 @@ $ gem install aliyun-rails
31
33
 
32
34
  ## 使用
33
35
 
34
- RPC 示例;
36
+ - 加载模块自动挂载几个常量:Dysms、Dyvms、RPCClient和ROAClient
37
+
38
+ CALL_TTS 实例:
39
+
40
+ ```ruby
41
+ require "aliyun-rails"
42
+
43
+ client = Aliyun::Dyvms.new(
44
+ access_key_id: ENV['ACCESS_KEY_ID'],
45
+ access_key_secret: ENV['ACCESS_KEY_SECRET'],
46
+ )
47
+
48
+ # 可以直接将API参数放到RAILS项目 * config/initializers *下
49
+ # Aliyun.config do |aliyun|
50
+ # aliyun.access_key_id = "XXX"
51
+ # aliyun.access_key_secret = "YYY"
52
+ # end
53
+ # 随后直接初始化
54
+ # client = Aliyun::Dyvms.new
55
+
56
+ # 调用语音方法 single_call_by_tts method
57
+ response = client.single_call_by_tts("075566668888", "13900001234", "TTS_CODE", { TTS_PARAM: 2022 })
58
+
59
+ puts response
60
+
61
+ ```
62
+
63
+ SEND_SMS 示例:
35
64
 
36
65
  ```ruby
37
66
  require "aliyun-rails"
38
67
 
39
- client = Dysms.new(
68
+ client = Aliyun::Dysms.new(
40
69
  access_key_id: ENV['ACCESS_KEY_ID'],
41
70
  access_key_secret: ENV['ACCESS_KEY_SECRET'],
42
71
  )
43
72
 
44
- # 可以直接将API参数放到 initializers
45
- # Aliyun::Rails.config do |i|
46
- # i.access_key_id = "XXX"
47
- # i.access_key_secret = "YYYY"
73
+ # 可以直接将API参数放到RAILS项目 * config/initializers *下
74
+ # Aliyun.config do |aliyun|
75
+ # aliyun.access_key_id = "XXX"
76
+ # aliyun.access_key_secret = "YYY"
48
77
  # end
49
78
  # 随后直接初始化
50
79
  # client = Dysms.new
51
80
 
52
- # then use the send_sms method
53
- response = client.send_sms("1380000000", "SMS_10010", {param1: "11"}, "SIGN_NAME")
81
+ # 调用短信方法 send_sms method
82
+ response = client.send_sms("1380000000", "SMS_10010", { param1: "11" }, "SIGN_NAME")
54
83
 
55
84
  puts response
56
85
  ```
57
86
 
87
+ RPC 示例:
88
+
89
+ ```ruby
90
+ require 'aliyun-rails'
91
+
92
+ # 实例化对象
93
+ client = RPCClient.new(
94
+ endpoint: 'http://ros.aliyuncs.com',
95
+ api_version: '2015-09-01',
96
+ access_key_id: ENV['ACCESS_KEY_ID'],
97
+ access_key_secret: ENV['ACCESS_KEY_SECRET'],
98
+ security_token: 'TOKEN_KEY'
99
+ )
100
+
101
+ # 请求接口查询
102
+ params = { key: (1..11).to_a.map(&:to_s) }
103
+ request_option = { method: 'POST', timeout: 15000 }
104
+ response = client.request(
105
+ action: 'DescribeRegions',
106
+ params: params,
107
+ opts: request_option
108
+ )
109
+
110
+ puts response
111
+
112
+ ```
58
113
 
59
114
  ROA 示例:
60
115
 
61
116
  ```ruby
62
117
  require 'aliyun-rails'
63
118
 
119
+ # 初始化对象
64
120
  client = ROAClient.new(
65
121
  endpoint: 'http://ros.aliyuncs.com',
66
122
  api_version: '2015-09-01',
@@ -68,28 +124,30 @@ client = ROAClient.new(
68
124
  access_key_secret: ENV['ACCESS_KEY_SECRET'],
69
125
  )
70
126
 
127
+ # 请求接口查询
71
128
  response = client.request(
72
- method: 'GET',
73
- uri: '/regions',
129
+ method: 'GET',
130
+ uri: '/regions',
74
131
  options: {
75
132
  timeout: 15000
76
133
  }
77
134
  )
78
135
 
79
- print response.body
136
+ puts response.body
80
137
  ```
81
138
 
82
139
  ## 问题
83
- [提交 Issue](https://github.com/ciscolive/aliyun-rails/issues/new/choose),不符合指南的问题可能会立即关闭。
84
140
 
141
+ [提交 Issue](https://github.com/ciscolive/aliyun-rails/issues/new/choose),不符合指南的问题可能会立即关闭。
85
142
 
86
143
  ## 发行说明
87
- 每个版本的详细更改记录在[发行说明](CHANGELOG.md)中。
88
144
 
145
+ 每个版本的详细更改记录在[发行说明](CHANGELOG.md)中。
89
146
 
90
147
  ## 贡献
91
- 提交 Pull Request 之前请阅读[贡献指南](CONTRIBUTING.md)。
92
148
 
149
+ 提交 Pull Request 之前请阅读[贡献指南](CONTRIBUTING.md)。
93
150
 
94
151
  ## 许可证
95
- [MIT](LICENSE.md)
152
+
153
+ [MIT](LICENSE.txt)
@@ -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
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Aliyun
4
+ VERSION = "0.1.16"
5
+ DEFAULT_UA = "AlibabaCloud (#{Gem::Platform.local.os}; " +
6
+ "#{Gem::Platform.local.cpu}) Ruby/#{RUBY_VERSION} Core/#{VERSION}"
7
+ end
@@ -0,0 +1,26 @@
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
21
+
22
+ # 加载模块,自动初始化的常量
23
+ RPCClient = Aliyun::Connector::RPCClient
24
+ ROAClient = Aliyun::Connector::ROAClient
25
+ Dysms = Aliyun::Dysms
26
+ Dyvms = Aliyun::Dyvms
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.13
4
+ version: 0.1.16
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
@@ -1,9 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module Aliyun
4
- module Rails
5
- VERSION = "0.1.13"
6
- DEFAULT_UA = "AlibabaCloud (#{Gem::Platform.local.os}; " +
7
- "#{Gem::Platform.local.cpu}) Ruby/#{RUBY_VERSION} Core/#{VERSION}"
8
- end
9
- 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