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
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: d4026c63fa0871d4eed11bba4a6ebdea8663af7779f63cdf2057af32934b41c4
         | 
| 4 | 
            +
              data.tar.gz: 36e4bbff971201e8ea36b887e3bb1e9102edfee78aeeb9c9183c7c35c01ec1f6
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 7538cc9db1cd0238951d7417d85cfaf23b1c00d95f7fda6edc937308fe471389fa75c7813c995fa0087a710af7e476905171ccfc4027d245384df305e2b882fe
         | 
| 7 | 
            +
              data.tar.gz: dd97bc1054145663273e427e043e8c3d470a13277a1893710fa4894d525fd944359f98f32502a06b0d7b4f6397d5d03eff4cc84c68f1d691e693d14262d7b549
         | 
    
        data/Gemfile
    CHANGED
    
    
    
        data/Rakefile
    CHANGED
    
    
| @@ -1,55 +1,61 @@ | |
| 1 1 | 
             
            class Terminal
         | 
| 2 | 
            -
              class Size; VERSION = '0.0.6' end
         | 
| 2 | 
            +
              class Size; VERSION = '0.0.6'.freeze end
         | 
| 3 3 | 
             
              class << self
         | 
| 4 4 | 
             
                def size
         | 
| 5 5 | 
             
                  size_via_low_level_ioctl or size_via_stty or nil
         | 
| 6 6 | 
             
                end
         | 
| 7 | 
            -
             | 
| 7 | 
            +
             | 
| 8 | 
            +
                def size!
         | 
| 9 | 
            +
                  size or _height_width_hash_from 25, 80
         | 
| 10 | 
            +
                end
         | 
| 8 11 |  | 
| 9 12 | 
             
                # These are experimental
         | 
| 10 | 
            -
                def resize | 
| 13 | 
            +
                def resize(direction, magnitude)
         | 
| 11 14 | 
             
                  tmux 'resize-pane', "-#{direction}", magnitude
         | 
| 12 15 | 
             
                end
         | 
| 13 16 |  | 
| 14 17 | 
             
                def tmux *cmd
         | 
| 15 | 
            -
                  system 'tmux', * | 
| 18 | 
            +
                  system 'tmux', *cmd.map(&:to_s)
         | 
| 16 19 | 
             
                end
         | 
| 17 20 |  | 
| 18 | 
            -
                IOCTL_INPUT_BUF = "\x00"*8
         | 
| 21 | 
            +
                IOCTL_INPUT_BUF = "\x00" * 8
         | 
| 19 22 | 
             
                def size_via_low_level_ioctl
         | 
| 20 23 | 
             
                  # Thanks to runpaint for the general approach to this
         | 
| 21 24 | 
             
                  return unless $stdin.respond_to? :ioctl
         | 
| 25 | 
            +
             | 
| 22 26 | 
             
                  code = tiocgwinsz_value_for RUBY_PLATFORM
         | 
| 23 27 | 
             
                  return unless code
         | 
| 28 | 
            +
             | 
| 24 29 | 
             
                  buf = IOCTL_INPUT_BUF.dup
         | 
| 25 | 
            -
                  return  | 
| 26 | 
            -
                  return if  | 
| 30 | 
            +
                  return if $stdout.ioctl(code, buf) != 0
         | 
| 31 | 
            +
                  return if buf == IOCTL_INPUT_BUF
         | 
| 32 | 
            +
             | 
| 27 33 | 
             
                  got = buf.unpack('S4')[0..1]
         | 
| 28 | 
            -
                  _height_width_hash_from | 
| 29 | 
            -
                rescue
         | 
| 34 | 
            +
                  _height_width_hash_from(*got)
         | 
| 35 | 
            +
                rescue StandardError
         | 
| 30 36 | 
             
                  nil
         | 
| 31 37 | 
             
                end
         | 
| 32 38 |  | 
| 33 | 
            -
                def tiocgwinsz_value_for | 
| 39 | 
            +
                def tiocgwinsz_value_for(platform)
         | 
| 34 40 | 
             
                  # This is as reported by <sys/ioctl.h>
         | 
| 35 41 | 
             
                  # Hard-coding because it seems like overkll to acutally involve C for this.
         | 
| 36 42 | 
             
                  {
         | 
| 37 43 | 
             
                    /linux/ => 0x5413,
         | 
| 38 | 
            -
                    /darwin/ => 0x40087468 | 
| 39 | 
            -
                  }.find{|k, | 
| 44 | 
            +
                    /darwin/ => 0x40087468 # thanks to brandon@brandon.io for the lookup!
         | 
| 45 | 
            +
                  }.find { |k, _v| platform[k] }
         | 
| 40 46 | 
             
                end
         | 
| 41 47 |  | 
| 42 48 | 
             
                def size_via_stty
         | 
| 43 | 
            -
                  ints = `stty size`.scan(/\d+/).map | 
| 44 | 
            -
                  _height_width_hash_from | 
| 45 | 
            -
                rescue
         | 
| 49 | 
            +
                  ints = `stty size`.scan(/\d+/).map(&:to_i)
         | 
| 50 | 
            +
                  _height_width_hash_from(*ints)
         | 
| 51 | 
            +
                rescue StandardError
         | 
| 46 52 | 
             
                  nil
         | 
| 47 53 | 
             
                end
         | 
| 48 54 |  | 
| 49 55 | 
             
                private
         | 
| 56 | 
            +
             | 
| 50 57 | 
             
                def _height_width_hash_from *dimensions
         | 
| 51 | 
            -
                  { : | 
| 58 | 
            +
                  { height: dimensions[0], width: dimensions[1] }
         | 
| 52 59 | 
             
                end
         | 
| 53 | 
            -
             | 
| 54 60 | 
             
              end
         | 
| 55 61 | 
             
            end
         | 
| @@ -21,7 +21,7 @@ module VpsAdmin::CLI::Commands | |
| 21 21 | 
             
                    attempts: 10,
         | 
| 22 22 | 
             
                    checksum: true,
         | 
| 23 23 | 
             
                    delete_after: true,
         | 
| 24 | 
            -
                    sudo: true | 
| 24 | 
            +
                    sudo: true
         | 
| 25 25 | 
             
                  }
         | 
| 26 26 |  | 
| 27 27 | 
             
                  opts.on('-p', '--pretend', 'Print what would the program do') do
         | 
| @@ -93,14 +93,14 @@ module VpsAdmin::CLI::Commands | |
| 93 93 |  | 
| 94 94 | 
             
                    ds_id = read_dataset_id(fs)
         | 
| 95 95 |  | 
| 96 | 
            -
                    if ds_id
         | 
| 97 | 
            -
             | 
| 98 | 
            -
             | 
| 99 | 
            -
             | 
| 100 | 
            -
             | 
| 96 | 
            +
                    ds = if ds_id
         | 
| 97 | 
            +
                           @api.dataset.show(ds_id)
         | 
| 98 | 
            +
                         else
         | 
| 99 | 
            +
                           dataset_chooser
         | 
| 100 | 
            +
                         end
         | 
| 101 101 |  | 
| 102 102 | 
             
                  elsif args.size != 2
         | 
| 103 | 
            -
                    warn  | 
| 103 | 
            +
                    warn 'Provide DATASET_ID and FILESYSTEM arguments'
         | 
| 104 104 | 
             
                    exit(false)
         | 
| 105 105 |  | 
| 106 106 | 
             
                  else
         | 
| @@ -144,15 +144,15 @@ module VpsAdmin::CLI::Commands | |
| 144 144 | 
             
                  # This is the first run within this history id, no local snapshots are
         | 
| 145 145 | 
             
                  # present
         | 
| 146 146 | 
             
                  if !latest_local_snapshot && @opts[:init_snapshots]
         | 
| 147 | 
            -
                    remote_state[ds.current_history_id] = | 
| 147 | 
            +
                    remote_state[ds.current_history_id] =
         | 
| 148 148 | 
             
                      remote_state[ds.current_history_id].last(@opts[:init_snapshots])
         | 
| 149 149 | 
             
                  end
         | 
| 150 150 |  | 
| 151 151 | 
             
                  remote_state[ds.current_history_id].each do |snap|
         | 
| 152 152 | 
             
                    found = false
         | 
| 153 153 |  | 
| 154 | 
            -
                    local_state. | 
| 155 | 
            -
                      found =  | 
| 154 | 
            +
                    local_state.each_value do |local_snapshots|
         | 
| 155 | 
            +
                      found = local_snapshots.detect { |s| s.name == snap.name }
         | 
| 156 156 | 
             
                      break if found
         | 
| 157 157 | 
             
                    end
         | 
| 158 158 |  | 
| @@ -170,33 +170,33 @@ module VpsAdmin::CLI::Commands | |
| 170 170 | 
             
                  if for_transfer.empty?
         | 
| 171 171 | 
             
                    if found_latest
         | 
| 172 172 | 
             
                      exit_msg(
         | 
| 173 | 
            -
                         | 
| 173 | 
            +
                        'Nothing to transfer: all snapshots with history id ' \
         | 
| 174 174 | 
             
                        "#{ds.current_history_id} are already present locally",
         | 
| 175 175 | 
             
                        error: @opts[:no_snapshots_error]
         | 
| 176 176 | 
             
                      )
         | 
| 177 177 |  | 
| 178 178 | 
             
                    else
         | 
| 179 | 
            -
                      exit_msg( | 
| 180 | 
            -
            Unable to transfer: the common snapshot has not been found
         | 
| 179 | 
            +
                      exit_msg(<<~END
         | 
| 180 | 
            +
                        Unable to transfer: the common snapshot has not been found
         | 
| 181 181 |  | 
| 182 | 
            -
            This can happen when the latest local snapshot was deleted from the server,
         | 
| 183 | 
            -
            i.e. you have not backed up this dataset for quite some time.
         | 
| 182 | 
            +
                        This can happen when the latest local snapshot was deleted from the server,
         | 
| 183 | 
            +
                        i.e. you have not backed up this dataset for quite some time.
         | 
| 184 184 |  | 
| 185 | 
            -
            You can either rename or destroy the whole current history id:
         | 
| 185 | 
            +
                        You can either rename or destroy the whole current history id:
         | 
| 186 186 |  | 
| 187 | 
            -
             | 
| 187 | 
            +
                          zfs rename #{fs}/#{ds.current_history_id} #{fs}/#{ds.current_history_id}.old
         | 
| 188 188 |  | 
| 189 | 
            -
            or
         | 
| 189 | 
            +
                        or
         | 
| 190 190 |  | 
| 191 | 
            -
             | 
| 192 | 
            -
             | 
| 191 | 
            +
                          zfs list -r -t all #{fs}/#{ds.current_history_id}
         | 
| 192 | 
            +
                          zfs destroy -r #{fs}/#{ds.current_history_id}
         | 
| 193 193 |  | 
| 194 | 
            -
            which will destroy all snapshots with this history id.
         | 
| 194 | 
            +
                        which will destroy all snapshots with this history id.
         | 
| 195 195 |  | 
| 196 | 
            -
            You can also destroy the local backup completely or backup to another dataset
         | 
| 197 | 
            -
            and start anew.
         | 
| 198 | 
            -
            END
         | 
| 199 | 
            -
             | 
| 196 | 
            +
                        You can also destroy the local backup completely or backup to another dataset
         | 
| 197 | 
            +
                        and start anew.
         | 
| 198 | 
            +
                      END
         | 
| 199 | 
            +
                              )
         | 
| 200 200 | 
             
                    end
         | 
| 201 201 | 
             
                  end
         | 
| 202 202 |  | 
| @@ -226,9 +226,7 @@ END | |
| 226 226 | 
             
                    if shared_name
         | 
| 227 227 | 
             
                      shared = remote_state[ds.current_history_id].detect { |s| s.name == shared_name }
         | 
| 228 228 |  | 
| 229 | 
            -
                      if shared && !for_transfer.detect { |s| s.id == shared.id }
         | 
| 230 | 
            -
                        for_transfer.insert(0, shared)
         | 
| 231 | 
            -
                      end
         | 
| 229 | 
            +
                      for_transfer.insert(0, shared) if shared && !for_transfer.detect { |s| s.id == shared.id }
         | 
| 232 230 | 
             
                    end
         | 
| 233 231 |  | 
| 234 232 | 
             
                    write_dataset_id!(ds, fs) unless written_dataset_id?
         | 
| @@ -238,13 +236,12 @@ END | |
| 238 236 | 
             
                end
         | 
| 239 237 |  | 
| 240 238 | 
             
                protected
         | 
| 239 | 
            +
             | 
| 241 240 | 
             
                def transfer(local_state, snapshots, hist_id, fs)
         | 
| 242 241 | 
             
                  ds = "#{fs}/#{hist_id}"
         | 
| 243 242 | 
             
                  no_local_snapshots = local_state[hist_id].nil? || local_state[hist_id].empty?
         | 
| 244 243 |  | 
| 245 | 
            -
                  if local_state[hist_id].nil?
         | 
| 246 | 
            -
                    zfs(:create, nil, ds)
         | 
| 247 | 
            -
                  end
         | 
| 244 | 
            +
                  zfs(:create, nil, ds) if local_state[hist_id].nil?
         | 
| 248 245 |  | 
| 249 246 | 
             
                  if no_local_snapshots
         | 
| 250 247 | 
             
                    msg "Performing a full receive of @#{snapshots.first.name} to #{ds}"
         | 
| @@ -255,60 +252,59 @@ END | |
| 255 252 | 
             
                    else
         | 
| 256 253 | 
             
                      run_piped(zfs_cmd(:recv, '-F', ds)) do
         | 
| 257 254 | 
             
                        SnapshotSend.new({}, @api).do_exec({
         | 
| 258 | 
            -
             | 
| 259 | 
            -
             | 
| 260 | 
            -
             | 
| 261 | 
            -
             | 
| 262 | 
            -
             | 
| 263 | 
            -
             | 
| 264 | 
            -
             | 
| 255 | 
            +
                                                             snapshot: snapshots.first.id,
         | 
| 256 | 
            +
                                                             send_mail: false,
         | 
| 257 | 
            +
                                                             delete_after: @opts[:delete_after],
         | 
| 258 | 
            +
                                                             max_rate: @opts[:max_rate],
         | 
| 259 | 
            +
                                                             checksum: @opts[:checksum],
         | 
| 260 | 
            +
                                                             quiet: @opts[:quiet]
         | 
| 261 | 
            +
                                                           })
         | 
| 265 262 | 
             
                      end || exit_msg('Receive failed')
         | 
| 266 263 | 
             
                    end
         | 
| 267 264 | 
             
                  end
         | 
| 268 265 |  | 
| 269 | 
            -
                   | 
| 270 | 
            -
                    msg "Performing an incremental receive of "+
         | 
| 271 | 
            -
                        "@#{snapshots.first.name} - @#{snapshots.last.name} to #{ds}"
         | 
| 266 | 
            +
                  return unless !no_local_snapshots || snapshots.size > 1
         | 
| 272 267 |  | 
| 273 | 
            -
             | 
| 274 | 
            -
                       | 
| 268 | 
            +
                  msg 'Performing an incremental receive of ' \
         | 
| 269 | 
            +
                      "@#{snapshots.first.name} - @#{snapshots.last.name} to #{ds}"
         | 
| 275 270 |  | 
| 276 | 
            -
             | 
| 277 | 
            -
             | 
| 278 | 
            -
             | 
| 279 | 
            -
             | 
| 280 | 
            -
             | 
| 281 | 
            -
             | 
| 282 | 
            -
             | 
| 283 | 
            -
             | 
| 284 | 
            -
             | 
| 285 | 
            -
             | 
| 286 | 
            -
             | 
| 287 | 
            -
             | 
| 288 | 
            -
             | 
| 271 | 
            +
                  if @opts[:safe]
         | 
| 272 | 
            +
                    safe_download(ds, snapshots.last, snapshots.first)
         | 
| 273 | 
            +
             | 
| 274 | 
            +
                  else
         | 
| 275 | 
            +
                    run_piped(zfs_cmd(:recv, '-F', ds)) do
         | 
| 276 | 
            +
                      SnapshotSend.new({}, @api).do_exec({
         | 
| 277 | 
            +
                                                           snapshot: snapshots.last.id,
         | 
| 278 | 
            +
                                                           from_snapshot: snapshots.first.id,
         | 
| 279 | 
            +
                                                           send_mail: false,
         | 
| 280 | 
            +
                                                           delete_after: @opts[:delete_after],
         | 
| 281 | 
            +
                                                           max_rate: @opts[:max_rate],
         | 
| 282 | 
            +
                                                           checksum: @opts[:checksum],
         | 
| 283 | 
            +
                                                           quiet: @opts[:quiet]
         | 
| 284 | 
            +
                                                         })
         | 
| 285 | 
            +
                    end || exit_msg('Receive failed')
         | 
| 289 286 | 
             
                  end
         | 
| 290 287 | 
             
                end
         | 
| 291 288 |  | 
| 292 289 | 
             
                def safe_download(ds, snapshot, from_snapshot = nil)
         | 
| 293 290 | 
             
                  part, full = snapshot_tmp_file(snapshot, from_snapshot)
         | 
| 294 291 |  | 
| 295 | 
            -
                   | 
| 292 | 
            +
                  unless File.exist?(full)
         | 
| 296 293 | 
             
                    attempts = 0
         | 
| 297 294 |  | 
| 298 295 | 
             
                    begin
         | 
| 299 296 | 
             
                      SnapshotDownload.new({}, @api).do_exec({
         | 
| 300 | 
            -
             | 
| 301 | 
            -
             | 
| 302 | 
            -
             | 
| 303 | 
            -
             | 
| 304 | 
            -
             | 
| 305 | 
            -
             | 
| 306 | 
            -
             | 
| 307 | 
            -
             | 
| 308 | 
            -
             | 
| 309 | 
            -
             | 
| 310 | 
            -
             | 
| 311 | 
            -
             | 
| 297 | 
            +
                                                               snapshot: snapshot.id,
         | 
| 298 | 
            +
                                                               from_snapshot: from_snapshot && from_snapshot.id,
         | 
| 299 | 
            +
                                                               format: from_snapshot ? :incremental_stream : :stream,
         | 
| 300 | 
            +
                                                               file: part,
         | 
| 301 | 
            +
                                                               max_rate: @opts[:max_rate],
         | 
| 302 | 
            +
                                                               checksum: @opts[:checksum],
         | 
| 303 | 
            +
                                                               quiet: @opts[:quiet],
         | 
| 304 | 
            +
                                                               resume: true,
         | 
| 305 | 
            +
                                                               delete_after: @opts[:delete_after],
         | 
| 306 | 
            +
                                                               send_mail: false
         | 
| 307 | 
            +
                                                             })
         | 
| 312 308 | 
             
                    rescue Errno::ECONNREFUSED,
         | 
| 313 309 | 
             
                           Errno::ETIMEDOUT,
         | 
| 314 310 | 
             
                           Errno::EHOSTUNREACH,
         | 
| @@ -318,11 +314,11 @@ END | |
| 318 314 | 
             
                      attempts += 1
         | 
| 319 315 |  | 
| 320 316 | 
             
                      if attempts >= @opts[:attempts]
         | 
| 321 | 
            -
                        warn  | 
| 317 | 
            +
                        warn 'Run out of attempts'
         | 
| 322 318 | 
             
                        exit(false)
         | 
| 323 319 |  | 
| 324 320 | 
             
                      else
         | 
| 325 | 
            -
                        warn  | 
| 321 | 
            +
                        warn 'Retry in 60 seconds'
         | 
| 326 322 | 
             
                        sleep(60)
         | 
| 327 323 | 
             
                        retry
         | 
| 328 324 | 
             
                      end
         | 
| @@ -339,15 +335,15 @@ END | |
| 339 335 | 
             
                end
         | 
| 340 336 |  | 
| 341 337 | 
             
                def rotate(fs, pretend: false)
         | 
| 342 | 
            -
                  msg  | 
| 343 | 
            -
                  local_state = pretend  | 
| 338 | 
            +
                  msg 'Rotating snapshots'
         | 
| 339 | 
            +
                  local_state = pretend || parse_tree(fs)
         | 
| 344 340 |  | 
| 345 341 | 
             
                  # Order snapshots by date of creation
         | 
| 346 342 | 
             
                  snapshots = local_state.values.flatten.sort do |a, b|
         | 
| 347 343 | 
             
                    a.creation <=> b.creation
         | 
| 348 344 | 
             
                  end
         | 
| 349 345 |  | 
| 350 | 
            -
                  cnt = local_state.values.inject(0) { |sum,  | 
| 346 | 
            +
                  cnt = local_state.values.inject(0) { |sum, local_snapshots| sum + local_snapshots.count }
         | 
| 351 347 | 
             
                  deleted = 0
         | 
| 352 348 | 
             
                  oldest = Time.now.to_i - (@opts[:max_age] * 60 * 60 * 24)
         | 
| 353 349 |  | 
| @@ -363,16 +359,16 @@ END | |
| 363 359 | 
             
                    local_state[s.hist_id].delete(s)
         | 
| 364 360 |  | 
| 365 361 | 
             
                    msg "Destroying #{ds}@#{s.name}"
         | 
| 366 | 
            -
                    zfs(:destroy, nil, "#{ds}@#{s.name}", pretend: | 
| 362 | 
            +
                    zfs(:destroy, nil, "#{ds}@#{s.name}", pretend:)
         | 
| 367 363 | 
             
                  end
         | 
| 368 364 |  | 
| 369 | 
            -
                  local_state.each do |hist_id,  | 
| 370 | 
            -
                    next unless  | 
| 365 | 
            +
                  local_state.each do |hist_id, local_snapshots|
         | 
| 366 | 
            +
                    next unless local_snapshots.empty?
         | 
| 371 367 |  | 
| 372 368 | 
             
                    ds = "#{fs}/#{hist_id}"
         | 
| 373 369 |  | 
| 374 370 | 
             
                    msg "Destroying #{ds}"
         | 
| 375 | 
            -
                    zfs(:destroy, nil, ds, pretend: | 
| 371 | 
            +
                    zfs(:destroy, nil, ds, pretend:)
         | 
| 376 372 | 
             
                  end
         | 
| 377 373 | 
             
                end
         | 
| 378 374 |  | 
| @@ -382,15 +378,15 @@ END | |
| 382 378 | 
             
                  # This is intentionally done by two zfs commands, because -d2 would include
         | 
| 383 379 | 
             
                  # nested subdatasets, which should not be there, but the user might create
         | 
| 384 380 | 
             
                  # them and it could confuse the program.
         | 
| 385 | 
            -
                  zfs(:list, '-r -d1 -tfilesystem -H -oname', fs).split("\n")[1 | 
| 381 | 
            +
                  zfs(:list, '-r -d1 -tfilesystem -H -oname', fs).split("\n")[1..].each do |name|
         | 
| 386 382 | 
             
                    last_name = name.split('/').last
         | 
| 387 383 | 
             
                    ret[last_name.to_i] = [] if dataset?(last_name)
         | 
| 388 384 | 
             
                  end
         | 
| 389 385 |  | 
| 390 386 | 
             
                  zfs(
         | 
| 391 | 
            -
             | 
| 392 | 
            -
             | 
| 393 | 
            -
             | 
| 387 | 
            +
                    :get,
         | 
| 388 | 
            +
                    '-Hrp -d2 -tsnapshot -oname,property,value name,creation',
         | 
| 389 | 
            +
                    fs
         | 
| 394 390 | 
             
                  ).split("\n").each do |line|
         | 
| 395 391 | 
             
                    name, property, value = line.split
         | 
| 396 392 | 
             
                    ds, snap_name = name.split('@')
         | 
| @@ -399,7 +395,7 @@ END | |
| 399 395 |  | 
| 400 396 | 
             
                    hist_id = ds_name.to_i
         | 
| 401 397 |  | 
| 402 | 
            -
                    if snap = ret[hist_id].detect { |s| s.name == snap_name }
         | 
| 398 | 
            +
                    if (snap = ret[hist_id].detect { |s| s.name == snap_name })
         | 
| 403 399 | 
             
                      snap.send("#{property}=", value)
         | 
| 404 400 |  | 
| 405 401 | 
             
                    else
         | 
| @@ -418,14 +414,15 @@ END | |
| 418 414 | 
             
                def read_dataset_id(fs)
         | 
| 419 415 | 
             
                  ds_id = zfs(:get, '-H -ovalue cz.vpsfree.vpsadmin:dataset_id', fs).strip
         | 
| 420 416 | 
             
                  return nil if ds_id == '-'
         | 
| 417 | 
            +
             | 
| 421 418 | 
             
                  @dataset_id = ds_id.to_i
         | 
| 422 419 | 
             
                end
         | 
| 423 420 |  | 
| 424 421 | 
             
                def check_dataset_id!(ds, fs)
         | 
| 425 | 
            -
                   | 
| 426 | 
            -
             | 
| 427 | 
            -
             | 
| 428 | 
            -
                   | 
| 422 | 
            +
                  return unless @dataset_id && @dataset_id != ds.id
         | 
| 423 | 
            +
             | 
| 424 | 
            +
                  warn "Dataset '#{fs}' is used to backup remote dataset with id '#{@dataset_id}', not '#{ds.id}'"
         | 
| 425 | 
            +
                  exit(false)
         | 
| 429 426 | 
             
                end
         | 
| 430 427 |  | 
| 431 428 | 
             
                def written_dataset_id?
         | 
| @@ -444,13 +441,13 @@ END | |
| 444 441 |  | 
| 445 442 | 
             
                  pids << Process.fork do
         | 
| 446 443 | 
             
                    r.close
         | 
| 447 | 
            -
                     | 
| 444 | 
            +
                    $stdout.reopen(w)
         | 
| 448 445 | 
             
                    block.call
         | 
| 449 446 | 
             
                  end
         | 
| 450 447 |  | 
| 451 448 | 
             
                  pids << Process.fork do
         | 
| 452 449 | 
             
                    w.close
         | 
| 453 | 
            -
                     | 
| 450 | 
            +
                    $stdin.reopen(r)
         | 
| 454 451 | 
             
                    Process.exec(cmd2)
         | 
| 455 452 | 
             
                  end
         | 
| 456 453 |  | 
| @@ -500,11 +497,12 @@ END | |
| 500 497 | 
             
                  ds_map = {}
         | 
| 501 498 |  | 
| 502 499 | 
             
                  @api.dataset.index(user: user.id).each do |ds|
         | 
| 503 | 
            -
                    if vps = vps_map[ds.id]
         | 
| 500 | 
            +
                    if (vps = vps_map[ds.id])
         | 
| 504 501 | 
             
                      puts "(#{i}) VPS ##{vps.id}"
         | 
| 505 502 |  | 
| 506 503 | 
             
                    else
         | 
| 507 504 | 
             
                      next if vps_only
         | 
| 505 | 
            +
             | 
| 508 506 | 
             
                      puts "(#{i}) Dataset #{ds.name}"
         | 
| 509 507 | 
             
                    end
         | 
| 510 508 |  | 
| @@ -513,10 +511,10 @@ END | |
| 513 511 | 
             
                  end
         | 
| 514 512 |  | 
| 515 513 | 
             
                  loop do
         | 
| 516 | 
            -
                     | 
| 517 | 
            -
                     | 
| 514 | 
            +
                    $stdout.write('Pick a dataset to backup: ')
         | 
| 515 | 
            +
                    $stdout.flush
         | 
| 518 516 |  | 
| 519 | 
            -
                    i =  | 
| 517 | 
            +
                    i = $stdin.readline.strip.to_i
         | 
| 520 518 | 
             
                    next if i <= 0 || ds_map[i].nil?
         | 
| 521 519 |  | 
| 522 520 | 
             
                    return ds_map[i]
         | 
| @@ -524,12 +522,12 @@ END | |
| 524 522 | 
             
                end
         | 
| 525 523 |  | 
| 526 524 | 
             
                def snapshot_tmp_file(s, from_s = nil)
         | 
| 527 | 
            -
                  if from_s
         | 
| 528 | 
            -
             | 
| 525 | 
            +
                  base = if from_s
         | 
| 526 | 
            +
                           ".snapshot_#{from_s.id}-#{s.id}.inc.dat.gz"
         | 
| 529 527 |  | 
| 530 | 
            -
             | 
| 531 | 
            -
             | 
| 532 | 
            -
             | 
| 528 | 
            +
                         else
         | 
| 529 | 
            +
                           ".snapshot_#{s.id}.dat.gz"
         | 
| 530 | 
            +
                         end
         | 
| 533 531 |  | 
| 534 532 | 
             
                  ["#{base}.part", base]
         | 
| 535 533 | 
             
                end
         | 
| @@ -7,15 +7,15 @@ module VpsAdmin::CLI::Commands | |
| 7 7 | 
             
                end
         | 
| 8 8 |  | 
| 9 9 | 
             
                protected
         | 
| 10 | 
            +
             | 
| 10 11 | 
             
                def find_or_create_dl(opts, do_create = true)
         | 
| 11 12 | 
             
                  @api.snapshot_download.index(snapshot: opts[:snapshot]).each do |r|
         | 
| 12 | 
            -
                     | 
| 13 | 
            -
             | 
| 14 | 
            -
             | 
| 15 | 
            -
                      end
         | 
| 16 | 
            -
             | 
| 17 | 
            -
                      return [r, false]
         | 
| 13 | 
            +
                    next unless opts[:from_snapshot] == (r.from_snapshot && r.from_snapshot_id)
         | 
| 14 | 
            +
                    if r.format != opts[:format].to_s
         | 
| 15 | 
            +
                      raise "SnapshotDownload id=#{r.id} is in unusable format '#{r.format}' (needs '#{opts[:format]}')"
         | 
| 18 16 | 
             
                    end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                    return [r, false]
         | 
| 19 19 | 
             
                  end
         | 
| 20 20 |  | 
| 21 21 | 
             
                  if do_create
         |