vpsadmin-client 3.0.0.master.20231229.pre.0.51d41b07 → 3.0.0.master.20240728.pre.0.dc5474cc
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/Gemfile +4 -0
- data/Rakefile +0 -1
- data/lib/{terminal-size.rb → terminal_size.rb} +23 -17
- data/lib/vpsadmin/cli/commands/backup_dataset.rb +96 -98
- data/lib/vpsadmin/cli/commands/backup_vps.rb +1 -1
- data/lib/vpsadmin/cli/commands/base_download.rb +6 -6
- data/lib/vpsadmin/cli/commands/network_top.rb +65 -59
- data/lib/vpsadmin/cli/commands/snapshot_download.rb +19 -20
- data/lib/vpsadmin/cli/commands/snapshot_send.rb +6 -7
- data/lib/vpsadmin/cli/commands/vps_migrate_many.rb +10 -12
- data/lib/vpsadmin/cli/commands/vps_remote_console.rb +29 -30
- data/lib/vpsadmin/cli/stream_downloader.rb +38 -45
- data/lib/vpsadmin/cli.rb +2 -2
- data/lib/vpsadmin/client/version.rb +1 -1
- data/lib/vpsadmin/client.rb +2 -3
- data/shell.nix +6 -6
- data/vpsadmin-client.gemspec +7 -10
- metadata +10 -38
@@ -1,8 +1,8 @@
|
|
1
|
-
require 'thread'
|
2
1
|
require 'time'
|
3
2
|
require 'json'
|
4
3
|
require 'base64'
|
5
|
-
require '
|
4
|
+
require 'io/wait'
|
5
|
+
require 'terminal_size'
|
6
6
|
|
7
7
|
module VpsAdmin::CLI::Commands
|
8
8
|
class VpsRemoteControl < HaveAPI::CLI::Command
|
@@ -14,7 +14,7 @@ module VpsAdmin::CLI::Commands
|
|
14
14
|
def initialize(http_client)
|
15
15
|
@http_client = http_client
|
16
16
|
@private_buffer = ''
|
17
|
-
@end_seq = ["\r", "\e",
|
17
|
+
@end_seq = ["\r", "\e", '.']
|
18
18
|
@end_i = 0
|
19
19
|
@stop = false
|
20
20
|
end
|
@@ -37,6 +37,7 @@ module VpsAdmin::CLI::Commands
|
|
37
37
|
end
|
38
38
|
|
39
39
|
protected
|
40
|
+
|
40
41
|
attr_reader :http_client
|
41
42
|
|
42
43
|
# Data is checked on the presence of the end sequence. The first character
|
@@ -52,10 +53,10 @@ module VpsAdmin::CLI::Commands
|
|
52
53
|
buffer = ''
|
53
54
|
|
54
55
|
data.each_char do |char|
|
55
|
-
if char == @end_seq[
|
56
|
+
if char == @end_seq[@end_i]
|
56
57
|
if @end_i == @end_seq.size - 1
|
57
58
|
@stop = true
|
58
|
-
return
|
59
|
+
return # rubocop:disable Lint/NonLocalExitFromIterator
|
59
60
|
end
|
60
61
|
|
61
62
|
@end_i += 1
|
@@ -111,9 +112,7 @@ module VpsAdmin::CLI::Commands
|
|
111
112
|
@error != false
|
112
113
|
end
|
113
114
|
|
114
|
-
|
115
|
-
@error
|
116
|
-
end
|
115
|
+
attr_reader :error
|
117
116
|
|
118
117
|
def join
|
119
118
|
stop
|
@@ -132,13 +131,14 @@ module VpsAdmin::CLI::Commands
|
|
132
131
|
end
|
133
132
|
|
134
133
|
protected
|
134
|
+
|
135
135
|
attr_reader :vps, :token, :rate, :width, :height, :write_buffer,
|
136
|
-
|
136
|
+
:mutex, :thread
|
137
137
|
|
138
138
|
def run
|
139
139
|
uri = URI("#{vps.node.location.remote_console_server}/console/feed/#{vps.id}")
|
140
140
|
start_args = [uri.host, uri.port, nil, nil, nil, nil, {
|
141
|
-
use_ssl: uri.scheme == 'https'
|
141
|
+
use_ssl: uri.scheme == 'https'
|
142
142
|
}]
|
143
143
|
|
144
144
|
Net::HTTP.start(*start_args) do |http|
|
@@ -164,14 +164,14 @@ module VpsAdmin::CLI::Commands
|
|
164
164
|
'session' => token,
|
165
165
|
'keys' => keys,
|
166
166
|
'width' => width,
|
167
|
-
'height' => height
|
167
|
+
'height' => height
|
168
168
|
)
|
169
169
|
|
170
170
|
res = http.request(req)
|
171
171
|
|
172
172
|
unless res.is_a?(Net::HTTPSuccess)
|
173
173
|
set_error(
|
174
|
-
|
174
|
+
'Console server returned error: ' \
|
175
175
|
"HTTP #{res.code} - #{res.message}\n\n#{res.body}"
|
176
176
|
)
|
177
177
|
stop
|
@@ -182,7 +182,7 @@ module VpsAdmin::CLI::Commands
|
|
182
182
|
|
183
183
|
unless ret[:session]
|
184
184
|
$stdout.write(ret[:data])
|
185
|
-
set_error(
|
185
|
+
set_error('Session closed.')
|
186
186
|
stop
|
187
187
|
return
|
188
188
|
end
|
@@ -216,38 +216,36 @@ module VpsAdmin::CLI::Commands
|
|
216
216
|
|
217
217
|
def exec(args)
|
218
218
|
if args.empty?
|
219
|
-
puts
|
219
|
+
puts 'provide VPS ID as an argument'
|
220
220
|
exit(false)
|
221
221
|
end
|
222
222
|
|
223
223
|
vps_id = args.first.to_i
|
224
224
|
|
225
|
-
write
|
225
|
+
write 'Locating VPS..'
|
226
226
|
begin
|
227
227
|
vps = @api.vps.show(vps_id, meta: { includes: 'node__location' })
|
228
|
-
|
229
228
|
rescue HaveAPI::Client::ActionFailed => e
|
230
|
-
puts
|
229
|
+
puts ' error'
|
231
230
|
puts e.message
|
232
231
|
exit(false)
|
233
232
|
end
|
234
233
|
|
235
234
|
puts " VPS is on #{vps.node.domain_name}, located in #{vps.node.location.label}."
|
236
235
|
puts "Console server URL is #{vps.node.location.remote_console_server}"
|
237
|
-
write
|
236
|
+
write 'Obtaining authentication token...'
|
238
237
|
|
239
238
|
begin
|
240
239
|
t = vps.console_token.create
|
241
|
-
|
242
240
|
rescue HaveAPI::Client::ActionFailed => e
|
243
|
-
puts
|
241
|
+
puts ' error'
|
244
242
|
puts e.message
|
245
243
|
exit(false)
|
246
244
|
end
|
247
245
|
|
248
246
|
puts
|
249
|
-
puts
|
250
|
-
puts
|
247
|
+
puts 'Connecting to the remote console...'
|
248
|
+
puts 'Press ENTER ESC . to exit'
|
251
249
|
puts
|
252
250
|
|
253
251
|
raw_mode do
|
@@ -256,6 +254,7 @@ module VpsAdmin::CLI::Commands
|
|
256
254
|
end
|
257
255
|
|
258
256
|
protected
|
257
|
+
|
259
258
|
attr_reader :client
|
260
259
|
|
261
260
|
def raw_mode
|
@@ -289,19 +288,19 @@ module VpsAdmin::CLI::Commands
|
|
289
288
|
client.start
|
290
289
|
|
291
290
|
loop do
|
292
|
-
res =
|
291
|
+
res = $stdin.wait_readable(1)
|
293
292
|
input.read_from($stdin) if res
|
294
293
|
|
295
|
-
|
296
|
-
client.join
|
294
|
+
next unless input.stop? || client.stop?
|
297
295
|
|
298
|
-
|
299
|
-
write("\n")
|
300
|
-
write(client.error)
|
301
|
-
end
|
296
|
+
client.join
|
302
297
|
|
303
|
-
|
298
|
+
if client.error?
|
299
|
+
write("\n")
|
300
|
+
write(client.error)
|
304
301
|
end
|
302
|
+
|
303
|
+
return
|
305
304
|
end
|
306
305
|
end
|
307
306
|
|
@@ -4,15 +4,15 @@ require 'ruby-progressbar'
|
|
4
4
|
require 'digest'
|
5
5
|
|
6
6
|
module VpsAdmin::CLI
|
7
|
-
class DownloadError < StandardError
|
7
|
+
class DownloadError < StandardError; end
|
8
8
|
|
9
9
|
class StreamDownloader
|
10
|
-
def self.download(
|
11
|
-
new(
|
10
|
+
def self.download(*, **)
|
11
|
+
new(*, **)
|
12
12
|
end
|
13
13
|
|
14
|
-
def initialize(api, dl, io, progress:
|
15
|
-
|
14
|
+
def initialize(api, dl, io, progress: $stdout, position: 0, max_rate: nil,
|
15
|
+
checksum: true)
|
16
16
|
downloaded = position
|
17
17
|
uri = URI(dl.url)
|
18
18
|
digest = Digest::SHA256.new
|
@@ -26,12 +26,12 @@ module VpsAdmin::CLI
|
|
26
26
|
format: '%E %t: [%B] %p%% %r MB/s',
|
27
27
|
rate_scale: ->(rate) { (rate / 1024.0 / 1024.0).round(2) },
|
28
28
|
throttle_rate: 0.2,
|
29
|
-
output: progress
|
29
|
+
output: progress
|
30
30
|
)
|
31
31
|
end
|
32
32
|
|
33
33
|
read = 0
|
34
|
-
step = 1*1024*1024
|
34
|
+
step = 1 * 1024 * 1024
|
35
35
|
io.seek(0)
|
36
36
|
|
37
37
|
while read < position
|
@@ -56,20 +56,20 @@ module VpsAdmin::CLI
|
|
56
56
|
throttle_rate: 0.2,
|
57
57
|
starting_at: downloaded,
|
58
58
|
autofinish: false,
|
59
|
-
output: progress
|
59
|
+
output: progress
|
60
60
|
)
|
61
61
|
end
|
62
62
|
|
63
|
-
args = [uri.host] + Array.new(5, nil) + [{use_ssl: uri.scheme == 'https'}]
|
63
|
+
args = [uri.host] + Array.new(5, nil) + [{ use_ssl: uri.scheme == 'https' }]
|
64
64
|
|
65
65
|
Net::HTTP.start(*args) do |http|
|
66
66
|
loop do
|
67
67
|
begin
|
68
68
|
dl_check = api.snapshot_download.show(dl.id)
|
69
69
|
|
70
|
-
if @pb && (dl_check.ready || (dl_check.size && dl_check.size > 0))
|
70
|
+
if @pb && (dl_check.ready || (dl_check.size && dl_check.size > 0)) # rubocop:disable all
|
71
71
|
total = dl_check.size * 1024 * 1024
|
72
|
-
@pb.total = @pb.progress
|
72
|
+
@pb.total = [@pb.progress, total].max
|
73
73
|
|
74
74
|
@download_size = (dl_check.size / 1024.0).round(2)
|
75
75
|
|
@@ -81,7 +81,6 @@ module VpsAdmin::CLI
|
|
81
81
|
self.format = "%E %t ~#{@download_size} GB: [%B] %p%% %r kB/s"
|
82
82
|
end
|
83
83
|
end
|
84
|
-
|
85
84
|
rescue HaveAPI::Client::ActionFailed => e
|
86
85
|
# The SnapshotDownload object no longer exists, the transaction
|
87
86
|
# responsible for its creation must have failed.
|
@@ -94,36 +93,32 @@ module VpsAdmin::CLI
|
|
94
93
|
|
95
94
|
http.request_get(uri.path, headers) do |res|
|
96
95
|
case res.code
|
97
|
-
when '404'
|
98
|
-
if downloaded > 0
|
99
|
-
# This means that the transaction used for preparing the download
|
100
|
-
# has failed, the file to download does not exist anymore, so fail.
|
101
|
-
raise DownloadError, 'The download has failed, most likely transaction failure'
|
96
|
+
when '404' # Not Found
|
97
|
+
raise DownloadError, 'The download has failed, most likely transaction failure' if downloaded > 0
|
102
98
|
|
103
|
-
|
104
|
-
|
105
|
-
# may be queued and it can take some time before it is processed.
|
106
|
-
pause(10)
|
107
|
-
next
|
108
|
-
end
|
99
|
+
# This means that the transaction used for preparing the download
|
100
|
+
# has failed, the file to download does not exist anymore, so fail.
|
109
101
|
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
# because the server is busy. Wait and retry.
|
115
|
-
pause(20)
|
116
|
-
next
|
102
|
+
# The file is not available yet, this is normal, the transaction
|
103
|
+
# may be queued and it can take some time before it is processed.
|
104
|
+
pause(10)
|
105
|
+
next
|
117
106
|
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
107
|
+
when '416' # Range Not Satisfiable
|
108
|
+
raise DownloadError, 'Range not satisfiable' unless downloaded > position
|
109
|
+
|
110
|
+
# We have already managed to download something (at this run, if the trasfer
|
111
|
+
# was resumed) and the server cannot provide more data yet. This can be
|
112
|
+
# because the server is busy. Wait and retry.
|
113
|
+
pause(20)
|
114
|
+
next
|
125
115
|
|
126
|
-
|
116
|
+
# The file is not ready yet - we ask for range that cannot be provided
|
117
|
+
# This happens when we're resuming a download and the file on the
|
118
|
+
# server was deleted meanwhile. The file might not be exactly the same
|
119
|
+
# as the one before, sha256sum would most likely fail.
|
120
|
+
|
121
|
+
when '200', '206' # OK and Partial Content
|
127
122
|
resume
|
128
123
|
|
129
124
|
else
|
@@ -140,10 +135,7 @@ module VpsAdmin::CLI
|
|
140
135
|
downloaded += size
|
141
136
|
|
142
137
|
begin
|
143
|
-
if @pb && (@pb.total.nil? || @pb.progress < @pb.total)
|
144
|
-
@pb.progress += size
|
145
|
-
end
|
146
|
-
|
138
|
+
@pb.progress += size if @pb && (@pb.total.nil? || @pb.progress < @pb.total)
|
147
139
|
rescue ProgressBar::InvalidProgressError
|
148
140
|
# The total value is in MB, it is not precise, so the actual
|
149
141
|
# size may be a little bit bigger.
|
@@ -186,12 +178,13 @@ module VpsAdmin::CLI
|
|
186
178
|
@pb.finish if @pb
|
187
179
|
|
188
180
|
# Verify the checksum
|
189
|
-
|
190
|
-
|
191
|
-
|
181
|
+
return unless checksum && digest.hexdigest != dl_check.sha256sum
|
182
|
+
|
183
|
+
raise DownloadError, 'The sha256sum does not match, retry the download'
|
192
184
|
end
|
193
185
|
|
194
186
|
protected
|
187
|
+
|
195
188
|
def pause(secs)
|
196
189
|
@paused = true
|
197
190
|
|
data/lib/vpsadmin/cli.rb
CHANGED
@@ -3,11 +3,11 @@ require 'vpsadmin/client/version'
|
|
3
3
|
|
4
4
|
module VpsAdmin
|
5
5
|
module CLI
|
6
|
-
module Commands
|
6
|
+
module Commands; end
|
7
7
|
|
8
8
|
class Cli < HaveAPI::CLI::Cli
|
9
9
|
def show_version
|
10
|
-
puts "#{VpsAdmin::Client::VERSION} based on haveapi-client "+
|
10
|
+
puts "#{VpsAdmin::Client::VERSION} based on haveapi-client " +
|
11
11
|
HaveAPI::Client::VERSION
|
12
12
|
end
|
13
13
|
end
|
data/lib/vpsadmin/client.rb
CHANGED
@@ -3,12 +3,11 @@ require 'haveapi/client'
|
|
3
3
|
module VpsAdmin
|
4
4
|
module Client
|
5
5
|
class Client < HaveAPI::Client::Client
|
6
|
-
|
7
6
|
end
|
8
7
|
|
9
8
|
# Shortcut to {VpsAdmin::Client::Client.new}
|
10
|
-
def self.new(*
|
11
|
-
Client.new(*
|
9
|
+
def self.new(*)
|
10
|
+
Client.new(*)
|
12
11
|
end
|
13
12
|
end
|
14
13
|
end
|
data/shell.nix
CHANGED
@@ -5,12 +5,12 @@ let
|
|
5
5
|
in stdenv.mkDerivation rec {
|
6
6
|
name = "vpsadmin-client";
|
7
7
|
|
8
|
-
buildInputs = [
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
8
|
+
buildInputs = with pkgs; [
|
9
|
+
ruby_3_2
|
10
|
+
git
|
11
|
+
zlib
|
12
|
+
openssl
|
13
|
+
ncurses
|
14
14
|
];
|
15
15
|
|
16
16
|
shellHook = ''
|
data/vpsadmin-client.gemspec
CHANGED
@@ -1,6 +1,5 @@
|
|
1
|
-
|
2
|
-
lib
|
3
|
-
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
1
|
+
lib = File.expand_path('lib', __dir__)
|
2
|
+
$:.unshift(lib) unless $:.include?(lib)
|
4
3
|
require 'vpsadmin/client/version'
|
5
4
|
|
6
5
|
Gem::Specification.new do |spec|
|
@@ -9,19 +8,17 @@ Gem::Specification.new do |spec|
|
|
9
8
|
spec.authors = ['Jakub Skokan']
|
10
9
|
spec.email = ['jakub.skokan@vpsfree.cz']
|
11
10
|
spec.summary =
|
12
|
-
|
11
|
+
spec.description = 'Ruby API and CLI for vpsAdmin API'
|
13
12
|
spec.homepage = ''
|
14
13
|
spec.license = 'GPL'
|
15
14
|
|
15
|
+
spec.required_ruby_version = ">= #{File.read('../.ruby-version').strip}"
|
16
|
+
|
16
17
|
spec.files = `git ls-files -z`.split("\x0")
|
17
18
|
spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
|
18
|
-
spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
|
19
19
|
spec.require_paths = ['lib']
|
20
20
|
|
21
|
-
spec.add_development_dependency 'bundler'
|
22
|
-
spec.add_development_dependency 'rake'
|
23
|
-
|
24
|
-
spec.add_runtime_dependency 'haveapi-client', '~> 0.19.0'
|
25
|
-
spec.add_runtime_dependency 'json'
|
26
21
|
spec.add_runtime_dependency 'curses'
|
22
|
+
spec.add_runtime_dependency 'haveapi-client', '~> 0.23.5'
|
23
|
+
spec.add_runtime_dependency 'json'
|
27
24
|
end
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: vpsadmin-client
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 3.0.0.master.
|
4
|
+
version: 3.0.0.master.20240728.pre.0.dc5474cc
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Jakub Skokan
|
@@ -11,27 +11,13 @@ cert_chain: []
|
|
11
11
|
date: 1980-01-01 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
|
-
name:
|
15
|
-
requirement: !ruby/object:Gem::Requirement
|
16
|
-
requirements:
|
17
|
-
- - ">="
|
18
|
-
- !ruby/object:Gem::Version
|
19
|
-
version: '0'
|
20
|
-
type: :development
|
21
|
-
prerelease: false
|
22
|
-
version_requirements: !ruby/object:Gem::Requirement
|
23
|
-
requirements:
|
24
|
-
- - ">="
|
25
|
-
- !ruby/object:Gem::Version
|
26
|
-
version: '0'
|
27
|
-
- !ruby/object:Gem::Dependency
|
28
|
-
name: rake
|
14
|
+
name: curses
|
29
15
|
requirement: !ruby/object:Gem::Requirement
|
30
16
|
requirements:
|
31
17
|
- - ">="
|
32
18
|
- !ruby/object:Gem::Version
|
33
19
|
version: '0'
|
34
|
-
type: :
|
20
|
+
type: :runtime
|
35
21
|
prerelease: false
|
36
22
|
version_requirements: !ruby/object:Gem::Requirement
|
37
23
|
requirements:
|
@@ -44,14 +30,14 @@ dependencies:
|
|
44
30
|
requirements:
|
45
31
|
- - "~>"
|
46
32
|
- !ruby/object:Gem::Version
|
47
|
-
version: 0.
|
33
|
+
version: 0.23.5
|
48
34
|
type: :runtime
|
49
35
|
prerelease: false
|
50
36
|
version_requirements: !ruby/object:Gem::Requirement
|
51
37
|
requirements:
|
52
38
|
- - "~>"
|
53
39
|
- !ruby/object:Gem::Version
|
54
|
-
version: 0.
|
40
|
+
version: 0.23.5
|
55
41
|
- !ruby/object:Gem::Dependency
|
56
42
|
name: json
|
57
43
|
requirement: !ruby/object:Gem::Requirement
|
@@ -66,20 +52,6 @@ dependencies:
|
|
66
52
|
- - ">="
|
67
53
|
- !ruby/object:Gem::Version
|
68
54
|
version: '0'
|
69
|
-
- !ruby/object:Gem::Dependency
|
70
|
-
name: curses
|
71
|
-
requirement: !ruby/object:Gem::Requirement
|
72
|
-
requirements:
|
73
|
-
- - ">="
|
74
|
-
- !ruby/object:Gem::Version
|
75
|
-
version: '0'
|
76
|
-
type: :runtime
|
77
|
-
prerelease: false
|
78
|
-
version_requirements: !ruby/object:Gem::Requirement
|
79
|
-
requirements:
|
80
|
-
- - ">="
|
81
|
-
- !ruby/object:Gem::Version
|
82
|
-
version: '0'
|
83
55
|
description: Ruby API and CLI for vpsAdmin API
|
84
56
|
email:
|
85
57
|
- jakub.skokan@vpsfree.cz
|
@@ -95,7 +67,7 @@ files:
|
|
95
67
|
- README.md
|
96
68
|
- Rakefile
|
97
69
|
- bin/vpsadminctl
|
98
|
-
- lib/
|
70
|
+
- lib/terminal_size.rb
|
99
71
|
- lib/vpsadmin/cli.rb
|
100
72
|
- lib/vpsadmin/cli/commands/backup_dataset.rb
|
101
73
|
- lib/vpsadmin/cli/commands/backup_vps.rb
|
@@ -122,14 +94,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
122
94
|
requirements:
|
123
95
|
- - ">="
|
124
96
|
- !ruby/object:Gem::Version
|
125
|
-
version:
|
97
|
+
version: 3.2.0
|
126
98
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
127
99
|
requirements:
|
128
|
-
- - "
|
100
|
+
- - ">="
|
129
101
|
- !ruby/object:Gem::Version
|
130
|
-
version:
|
102
|
+
version: '0'
|
131
103
|
requirements: []
|
132
|
-
rubygems_version: 3.
|
104
|
+
rubygems_version: 3.5.9
|
133
105
|
signing_key:
|
134
106
|
specification_version: 4
|
135
107
|
summary: Ruby API and CLI for vpsAdmin API
|