b2-client 1.0.5 → 1.0.6
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/lib/b2.rb +20 -11
- data/lib/b2/api_connection.rb +101 -0
- data/lib/b2/bucket.rb +14 -4
- data/lib/b2/connection.rb +56 -81
- data/lib/b2/errors.rb +3 -0
- data/lib/b2/file.rb +1 -1
- data/lib/b2/version.rb +1 -1
- metadata +4 -4
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 405cc51a4b94788687e8ccfa7aa774df40c29135b9ac89251405f56c32d73835
|
4
|
+
data.tar.gz: 820653192390b83efbb713f4104d739b496e75d3ae9c39dec6b99b3ac2e1af77
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 73a212dbb80684eca0097f9e784ab8815a559e3a810cdb448eaf74fb9ec49a594c5f3f78790870fc05ce3fe47dc0accfb62f931dce6811178b771f63884fcdd1
|
7
|
+
data.tar.gz: 7f6a23851372cd9fa317faab23fa09fcca35344e024fe8c1496c3204bffdb258e75197f1e115b45c5b5f400b0f872b738827973bb5469b2d2b11d7a1bc9c6892
|
data/lib/b2.rb
CHANGED
@@ -5,37 +5,46 @@ require 'net/http'
|
|
5
5
|
require File.expand_path('../b2/errors', __FILE__)
|
6
6
|
require File.expand_path('../b2/file', __FILE__)
|
7
7
|
require File.expand_path('../b2/bucket', __FILE__)
|
8
|
+
require File.expand_path('../b2/api_connection', __FILE__)
|
8
9
|
require File.expand_path('../b2/connection', __FILE__)
|
9
10
|
require File.expand_path('../b2/upload_chunker', __FILE__)
|
10
11
|
|
11
12
|
class B2
|
12
13
|
|
13
|
-
def initialize(
|
14
|
-
@
|
15
|
-
|
14
|
+
def initialize(key_id: , secret: )
|
15
|
+
@connection = B2::Connection.new(key_id, secret)
|
16
|
+
end
|
17
|
+
|
18
|
+
def account_id
|
19
|
+
@connection.account_id
|
16
20
|
end
|
17
21
|
|
18
22
|
def buckets
|
19
23
|
@connection.buckets
|
20
24
|
end
|
21
25
|
|
26
|
+
def bucket(name)
|
27
|
+
bs = @connection.post('/b2api/v2/b2_list_buckets', {accountId: account_id, bucketName: name})['buckets']
|
28
|
+
B2::Bucket.new(bs.first, @connection)
|
29
|
+
end
|
30
|
+
|
22
31
|
def file(bucket, key)
|
23
32
|
bucket_id = @connection.lookup_bucket_id(bucket)
|
24
33
|
|
25
|
-
file = @connection.post('/b2api/
|
34
|
+
file = @connection.post('/b2api/v2/b2_list_file_names', {
|
26
35
|
bucketId: bucket_id,
|
27
36
|
startFileName: key
|
28
37
|
})['files'].find {|f| f['fileName'] == key }
|
29
38
|
|
30
|
-
file ? B2::File.new(file.merge({'bucketId' => bucket_id})) : nil
|
39
|
+
file ? B2::File.new(file.merge({'bucketId' => bucket_id}), @connection) : nil
|
31
40
|
end
|
32
41
|
|
33
42
|
def delete(bucket, key)
|
34
43
|
object = file(bucket, key)
|
35
44
|
if object
|
36
|
-
@connection.post('/b2api/
|
37
|
-
fileName:
|
38
|
-
fileId:
|
45
|
+
@connection.post('/b2api/v2/b2_delete_file_version', {
|
46
|
+
fileName: object.name,
|
47
|
+
fileId: object.id
|
39
48
|
})
|
40
49
|
else
|
41
50
|
false
|
@@ -43,7 +52,7 @@ class B2
|
|
43
52
|
end
|
44
53
|
|
45
54
|
def get_upload_token(bucket)
|
46
|
-
@connection.post("/b2api/
|
55
|
+
@connection.post("/b2api/v2/b2_get_upload_url", {
|
47
56
|
bucketId: @connection.lookup_bucket_id(bucket)
|
48
57
|
})
|
49
58
|
end
|
@@ -85,11 +94,11 @@ class B2
|
|
85
94
|
|
86
95
|
|
87
96
|
def download_to_file(bucket, key, filename)
|
88
|
-
file = File.open(filename, '
|
97
|
+
file = ::File.open(filename, 'wb')
|
89
98
|
download(bucket, key) do |chunk|
|
90
99
|
file << chunk
|
91
100
|
end
|
92
101
|
file.close
|
93
102
|
end
|
94
103
|
|
95
|
-
end
|
104
|
+
end
|
@@ -0,0 +1,101 @@
|
|
1
|
+
class B2
|
2
|
+
class APIConnection
|
3
|
+
|
4
|
+
attr_reader :key_id, :key_secret, :download_url
|
5
|
+
|
6
|
+
def initialize(key_id, secret)
|
7
|
+
@key_id = key_id
|
8
|
+
@key_secret = secret
|
9
|
+
end
|
10
|
+
|
11
|
+
def connect!
|
12
|
+
conn = Net::HTTP.new('api.backblazeb2.com', 443)
|
13
|
+
conn.use_ssl = true
|
14
|
+
|
15
|
+
req = Net::HTTP::Get.new('/b2api/v2/b2_authorize_account')
|
16
|
+
req.basic_auth(@key_id, @key_secret)
|
17
|
+
|
18
|
+
key_expiration = Time.now.to_i + 86_400 #24hr expiry
|
19
|
+
resp = conn.start { |http| http.request(req) }
|
20
|
+
if resp.is_a?(Net::HTTPSuccess)
|
21
|
+
resp = JSON.parse(resp.body)
|
22
|
+
else
|
23
|
+
raise "Error connecting to B2 API"
|
24
|
+
end
|
25
|
+
|
26
|
+
uri = URI.parse(resp['apiUrl'])
|
27
|
+
@connection = Net::HTTP.new(uri.host, uri.port)
|
28
|
+
@connection.use_ssl = uri.scheme == 'https'
|
29
|
+
@connection.start
|
30
|
+
|
31
|
+
@auth_token_expires_at = key_expiration
|
32
|
+
@account_id = resp['accountId']
|
33
|
+
@minimum_part_size = resp['absoluteMinimumPartSize']
|
34
|
+
@recommended_part_size = resp['recommendedPartSize']
|
35
|
+
@auth_token = resp['authorizationToken']
|
36
|
+
@download_url = resp['downloadUrl']
|
37
|
+
@buckets_cache = []
|
38
|
+
end
|
39
|
+
|
40
|
+
def account_id
|
41
|
+
return @account_id if !@account_id.nil?
|
42
|
+
|
43
|
+
connect!
|
44
|
+
@account_id
|
45
|
+
end
|
46
|
+
|
47
|
+
def disconnect!
|
48
|
+
if @connection
|
49
|
+
@connection.finish if @connection.active?
|
50
|
+
@connection = nil
|
51
|
+
end
|
52
|
+
end
|
53
|
+
|
54
|
+
def reconnect!
|
55
|
+
disconnect!
|
56
|
+
connect!
|
57
|
+
end
|
58
|
+
|
59
|
+
def authorization_token
|
60
|
+
if @auth_token_expires_at.nil? || @auth_token_expires_at <= Time.now.to_i
|
61
|
+
reconnect!
|
62
|
+
end
|
63
|
+
@auth_token
|
64
|
+
end
|
65
|
+
|
66
|
+
def active?
|
67
|
+
!@connection.nil? && @connection.active?
|
68
|
+
end
|
69
|
+
|
70
|
+
def connection
|
71
|
+
reconnect! if !active?
|
72
|
+
@connection
|
73
|
+
end
|
74
|
+
|
75
|
+
def send_request(request, body=nil, &block)
|
76
|
+
request['Authorization'] = authorization_token
|
77
|
+
request.body = (body.is_a?(String) ? body : JSON.generate(body)) if body
|
78
|
+
|
79
|
+
return_value = nil
|
80
|
+
close_connection = false
|
81
|
+
connection.request(request) do |response|
|
82
|
+
close_connection = response['Connection'] == 'close'
|
83
|
+
|
84
|
+
case response
|
85
|
+
when Net::HTTPSuccess
|
86
|
+
if block_given?
|
87
|
+
return_value = yield(response)
|
88
|
+
else
|
89
|
+
return_value = JSON.parse(response.body)
|
90
|
+
end
|
91
|
+
else
|
92
|
+
raise "Error connecting to B2 API #{response.body}"
|
93
|
+
end
|
94
|
+
end
|
95
|
+
disconnect! if close_connection
|
96
|
+
|
97
|
+
return_value
|
98
|
+
end
|
99
|
+
|
100
|
+
end
|
101
|
+
end
|
data/lib/b2/bucket.rb
CHANGED
@@ -14,7 +14,7 @@ class B2
|
|
14
14
|
end
|
15
15
|
|
16
16
|
def get_upload_token
|
17
|
-
@connection.post("/b2api/
|
17
|
+
@connection.post("/b2api/v2/b2_get_upload_url", { bucketId: @id })
|
18
18
|
end
|
19
19
|
|
20
20
|
def upload_file(key, io_or_string, mime_type: nil, sha1: nil, content_disposition: nil, info: {})
|
@@ -46,9 +46,19 @@ class B2
|
|
46
46
|
|
47
47
|
B2::File.new(result, @connection)
|
48
48
|
end
|
49
|
+
|
50
|
+
def keys(prefix: nil, delimiter: nil)
|
51
|
+
#TODO: add abilty to get all names
|
52
|
+
@connection.post('/b2api/v2/b2_list_file_names', {
|
53
|
+
bucketId: @id,
|
54
|
+
maxFileCount: 1000,
|
55
|
+
prefix: prefix,
|
56
|
+
delimiter: delimiter
|
57
|
+
})['files'].map{ |f| f['fileName'] }
|
58
|
+
end
|
49
59
|
|
50
60
|
def has_key?(key)
|
51
|
-
!@connection.post('/b2api/
|
61
|
+
!@connection.post('/b2api/v2/b2_list_file_names', {
|
52
62
|
bucketId: @id,
|
53
63
|
startFileName: key,
|
54
64
|
maxFileCount: 1,
|
@@ -57,7 +67,7 @@ class B2
|
|
57
67
|
end
|
58
68
|
|
59
69
|
def file(key)
|
60
|
-
file = @connection.post('/b2api/
|
70
|
+
file = @connection.post('/b2api/v2/b2_list_file_names', {
|
61
71
|
bucketId: @id,
|
62
72
|
startFileName: key,
|
63
73
|
maxFileCount: 1,
|
@@ -80,4 +90,4 @@ class B2
|
|
80
90
|
end
|
81
91
|
|
82
92
|
end
|
83
|
-
end
|
93
|
+
end
|
data/lib/b2/connection.rb
CHANGED
@@ -1,93 +1,67 @@
|
|
1
1
|
require 'cgi'
|
2
|
+
require 'thread'
|
2
3
|
|
3
4
|
class B2
|
4
5
|
class Connection
|
5
6
|
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
@
|
10
|
-
@
|
11
|
-
|
12
|
-
|
13
|
-
def connect!
|
14
|
-
conn = Net::HTTP.new('api.backblazeb2.com', 443)
|
15
|
-
conn.use_ssl = true
|
7
|
+
def initialize(key_id, secret, pool: 5, timeout: 5)
|
8
|
+
@mutex = Mutex.new
|
9
|
+
@availability = ConditionVariable.new
|
10
|
+
@max = pool
|
11
|
+
@timeout = timeout
|
12
|
+
@free_pool = []
|
13
|
+
@used_pool = []
|
16
14
|
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
key_expiration = Time.now.to_i + 86_400 #24hr expiry
|
21
|
-
resp = conn.start { |http| http.request(req) }
|
22
|
-
if resp.is_a?(Net::HTTPSuccess)
|
23
|
-
resp = JSON.parse(resp.body)
|
24
|
-
else
|
25
|
-
raise "Error connecting to B2 API"
|
26
|
-
end
|
27
|
-
|
28
|
-
uri = URI.parse(resp['apiUrl'])
|
29
|
-
@connection = Net::HTTP.new(uri.host, uri.port)
|
30
|
-
@connection.use_ssl = uri.scheme == 'https'
|
31
|
-
@connection.start
|
15
|
+
@key_id = key_id
|
16
|
+
@key_secret = secret
|
32
17
|
|
33
|
-
@auth_token_expires_at = key_expiration
|
34
|
-
@minimum_part_size = resp['absoluteMinimumPartSize']
|
35
|
-
@recommended_part_size = resp['recommendedPartSize']
|
36
|
-
@auth_token = resp['authorizationToken']
|
37
|
-
@download_url = resp['downloadUrl']
|
38
18
|
@buckets_cache = []
|
39
19
|
end
|
40
20
|
|
41
|
-
def
|
42
|
-
if
|
43
|
-
|
44
|
-
|
45
|
-
end
|
21
|
+
def account_id
|
22
|
+
return @account_id if !@account_id.nil?
|
23
|
+
|
24
|
+
@account_id = with_connection { |conn| conn.account_id }
|
46
25
|
end
|
47
|
-
|
48
|
-
def
|
49
|
-
|
50
|
-
|
26
|
+
|
27
|
+
def with_connection
|
28
|
+
conn = @mutex.synchronize do
|
29
|
+
cxn = if !@free_pool.empty?
|
30
|
+
@free_pool.shift
|
31
|
+
elsif @free_pool.size + @used_pool.size < @max
|
32
|
+
B2::APIConnection.new(@key_id, @key_secret)
|
33
|
+
else
|
34
|
+
@availability.wait(@mutex, @timeout)
|
35
|
+
@free_pool.shift || B2::APIConnection.new(@key_id, @key_secret)
|
36
|
+
end
|
37
|
+
|
38
|
+
@used_pool << cxn
|
39
|
+
cxn
|
40
|
+
end
|
41
|
+
|
42
|
+
yield conn
|
43
|
+
ensure
|
44
|
+
@mutex.synchronize do
|
45
|
+
@used_pool.delete(conn)
|
46
|
+
@free_pool << conn if conn.active?
|
47
|
+
@availability.signal()
|
48
|
+
end
|
51
49
|
end
|
52
50
|
|
53
51
|
def authorization_token
|
54
|
-
|
55
|
-
reconnect!
|
56
|
-
end
|
57
|
-
@auth_token
|
52
|
+
with_connection { |conn| conn.authorization_token }
|
58
53
|
end
|
59
54
|
|
60
|
-
def active?
|
61
|
-
!@connection.nil? && @connection.active?
|
62
|
-
end
|
63
|
-
|
64
55
|
def send_request(request, body=nil, &block)
|
65
|
-
request
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
@connection.request(request) do |response|
|
71
|
-
close_connection = response['Connection'] == 'close'
|
72
|
-
|
73
|
-
case response
|
74
|
-
when Net::HTTPSuccess
|
75
|
-
if block_given?
|
76
|
-
return_value = yield(response)
|
77
|
-
else
|
78
|
-
return_value = JSON.parse(response.body)
|
79
|
-
end
|
80
|
-
else
|
81
|
-
raise "Error connecting to B2 API #{response.body}"
|
82
|
-
end
|
83
|
-
end
|
84
|
-
@connection.finish if close_connection
|
85
|
-
|
86
|
-
return_value
|
56
|
+
with_connection { |conn| conn.send_request(request, body, &block) }
|
57
|
+
end
|
58
|
+
|
59
|
+
def download_url
|
60
|
+
with_connection { |conn| conn.download_url }
|
87
61
|
end
|
88
62
|
|
89
63
|
def buckets
|
90
|
-
post('/b2api/
|
64
|
+
post('/b2api/v2/b2_list_buckets', {accountId: account_id})['buckets'].map do |b|
|
91
65
|
B2::Bucket.new(b, self)
|
92
66
|
end
|
93
67
|
end
|
@@ -95,30 +69,30 @@ class B2
|
|
95
69
|
def lookup_bucket_id(name)
|
96
70
|
bucket = @buckets_cache.find{ |b| b.name == name }
|
97
71
|
return bucket.id if bucket
|
98
|
-
|
72
|
+
|
99
73
|
@buckets_cache = buckets
|
100
74
|
@buckets_cache.find{ |b| b.name == name }&.id
|
101
75
|
end
|
102
76
|
|
103
77
|
def get_download_url(bucket, filename, expires_in: 3_600, disposition: nil)
|
104
|
-
response = post("/b2api/
|
78
|
+
response = post("/b2api/v2/b2_get_download_authorization", {
|
105
79
|
bucketId: lookup_bucket_id(bucket),
|
106
80
|
fileNamePrefix: filename,
|
107
81
|
validDurationInSeconds: expires_in,
|
108
82
|
b2ContentDisposition: disposition
|
109
83
|
})
|
110
|
-
url =
|
84
|
+
url = download_url + '/file/' + bucket + '/' + filename + "?Authorization=" + response['authorizationToken']
|
111
85
|
url += "&b2ContentDisposition=#{CGI.escape(disposition)}" if disposition
|
112
86
|
url
|
113
87
|
end
|
114
88
|
|
115
|
-
def download(bucket, key, to=nil
|
89
|
+
def download(bucket, key, to=nil)
|
116
90
|
opened_file = (to && to.is_a?(String))
|
117
91
|
to = ::File.open(to, 'wb') if to.is_a?(String)
|
118
92
|
digestor = Digest::SHA1.new
|
119
93
|
data = ""
|
120
94
|
|
121
|
-
uri = URI.parse(
|
95
|
+
uri = URI.parse(download_url)
|
122
96
|
conn = Net::HTTP.new(uri.host, uri.port)
|
123
97
|
conn.use_ssl = uri.scheme == 'https'
|
124
98
|
|
@@ -132,15 +106,15 @@ class B2
|
|
132
106
|
digestor << chunk
|
133
107
|
if to
|
134
108
|
to << chunk
|
135
|
-
elsif
|
136
|
-
|
109
|
+
elsif block_given?
|
110
|
+
yield(chunk)
|
137
111
|
else
|
138
112
|
data << chunk
|
139
113
|
end
|
140
114
|
end
|
141
115
|
|
142
|
-
if digestor.hexdigest != response['X-Bz-Content-Sha1']
|
143
|
-
|
116
|
+
if response['X-Bz-Content-Sha1'] != 'none' && digestor.hexdigest != response['X-Bz-Content-Sha1']
|
117
|
+
rase B2::FileIntegrityError.new("SHA1 Mismatch, expected: \"#{response['X-Bz-Content-Sha1']}\", actual: \"#{digestor.hexdigest}\"")
|
144
118
|
end
|
145
119
|
when Net::HTTPNotFound
|
146
120
|
raise B2::NotFound.new(JSON.parse(response.body)['message'])
|
@@ -164,7 +138,7 @@ class B2
|
|
164
138
|
elsif to
|
165
139
|
to.flush
|
166
140
|
end
|
167
|
-
|
141
|
+
!block_given? && to.nil? ? data : nil
|
168
142
|
end
|
169
143
|
|
170
144
|
def get(path, body=nil, &block)
|
@@ -180,4 +154,5 @@ class B2
|
|
180
154
|
end
|
181
155
|
|
182
156
|
end
|
183
|
-
end
|
157
|
+
end
|
158
|
+
|
data/lib/b2/errors.rb
CHANGED
data/lib/b2/file.rb
CHANGED
data/lib/b2/version.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: b2-client
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.0.
|
4
|
+
version: 1.0.6
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Jon Bracy
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2020-04-06 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rake
|
@@ -36,6 +36,7 @@ files:
|
|
36
36
|
- README.md
|
37
37
|
- b2-client.gemspec
|
38
38
|
- lib/b2.rb
|
39
|
+
- lib/b2/api_connection.rb
|
39
40
|
- lib/b2/bucket.rb
|
40
41
|
- lib/b2/client.rb
|
41
42
|
- lib/b2/connection.rb
|
@@ -62,8 +63,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
62
63
|
- !ruby/object:Gem::Version
|
63
64
|
version: '0'
|
64
65
|
requirements: []
|
65
|
-
|
66
|
-
rubygems_version: 2.7.4
|
66
|
+
rubygems_version: 3.0.6
|
67
67
|
signing_key:
|
68
68
|
specification_version: 4
|
69
69
|
summary: Backblaze B2 Client
|