remotus 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.github/workflows/main.yml +35 -0
- data/.gitignore +11 -0
- data/.rspec +3 -0
- data/.rubocop.yml +41 -0
- data/CHANGELOG.md +5 -0
- data/Gemfile +6 -0
- data/Gemfile.lock +103 -0
- data/LICENSE.txt +21 -0
- data/README.md +112 -0
- data/Rakefile +12 -0
- data/bin/console +15 -0
- data/bin/setup +8 -0
- data/lib/remotus.rb +142 -0
- data/lib/remotus/auth.rb +66 -0
- data/lib/remotus/auth/credential.rb +145 -0
- data/lib/remotus/auth/hash_store.rb +56 -0
- data/lib/remotus/auth/store.rb +43 -0
- data/lib/remotus/host_pool.rb +181 -0
- data/lib/remotus/logger.rb +9 -0
- data/lib/remotus/pool.rb +189 -0
- data/lib/remotus/result.rb +83 -0
- data/lib/remotus/ssh_connection.rb +447 -0
- data/lib/remotus/version.rb +6 -0
- data/lib/remotus/winrm_connection.rb +186 -0
- data/remotus.gemspec +45 -0
- metadata +226 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: a719c4faff9f018cf95614e68886b53475218bb27a73cde4579bbf7eb90d638b
|
4
|
+
data.tar.gz: f66e7025b43d7d343f269c076824eda4061b7565b418ee3637a57e17c404d9d6
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: a078c9a5e63f20059d6148e48356efead6f2520f47d0225c94a33cf3721b5294f9f6e893e166e53fb4a4609f3e5c62dee9872ea023d460d0731581d812eeb9bf
|
7
|
+
data.tar.gz: b7595d1ab8c6c7df4506e590a19a6ab0653d27edcfec4a1eef1764c21e3d9eb2d2edaffae6beff4bbf09465b2d84d029bf15c89a7a5cb944104e712543ea5347
|
@@ -0,0 +1,35 @@
|
|
1
|
+
name: Ruby
|
2
|
+
|
3
|
+
on: [push,pull_request]
|
4
|
+
|
5
|
+
jobs:
|
6
|
+
build:
|
7
|
+
runs-on: ubuntu-latest
|
8
|
+
steps:
|
9
|
+
- uses: actions/checkout@v2
|
10
|
+
- name: Set up Ruby
|
11
|
+
uses: ruby/setup-ruby@v1
|
12
|
+
with:
|
13
|
+
ruby-version: 2.7.2
|
14
|
+
- name: Run unit tests and code linting
|
15
|
+
run: |
|
16
|
+
gem install bundler -v 2.2.9
|
17
|
+
bundle install
|
18
|
+
bundle exec rake
|
19
|
+
bundle exec yard
|
20
|
+
|
21
|
+
- name: Deploy documentation
|
22
|
+
if: github.ref == 'refs/heads/main'
|
23
|
+
uses: crazy-max/ghaction-github-pages@v2.2.0
|
24
|
+
with:
|
25
|
+
target_branch: gh-pages
|
26
|
+
build_dir: doc
|
27
|
+
env:
|
28
|
+
GITHUB_TOKEN: ${{secrets.GITHUB_TOKEN}}
|
29
|
+
|
30
|
+
- name: Release Gem
|
31
|
+
if: contains(github.ref, 'refs/tags/v')
|
32
|
+
uses: dawidd6/action-publish-gem@v1.2.0
|
33
|
+
with:
|
34
|
+
github_token: ${{secrets.GITHUB_TOKEN}}
|
35
|
+
api_key: ${{secrets.RUBYGEMS_API_KEY}}
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/.rubocop.yml
ADDED
@@ -0,0 +1,41 @@
|
|
1
|
+
AllCops:
|
2
|
+
TargetRubyVersion: 2.4
|
3
|
+
NewCops: enable
|
4
|
+
|
5
|
+
Style/StringLiterals:
|
6
|
+
Enabled: true
|
7
|
+
EnforcedStyle: double_quotes
|
8
|
+
|
9
|
+
Style/StringLiteralsInInterpolation:
|
10
|
+
Enabled: true
|
11
|
+
EnforcedStyle: double_quotes
|
12
|
+
|
13
|
+
Layout/LineLength:
|
14
|
+
Max: 150
|
15
|
+
|
16
|
+
Metrics/ClassLength:
|
17
|
+
Max: 250
|
18
|
+
|
19
|
+
Metrics/BlockLength:
|
20
|
+
Exclude:
|
21
|
+
- spec/**/*.rb
|
22
|
+
- remotus.gemspec
|
23
|
+
- lib/remotus/ssh_connection.rb
|
24
|
+
|
25
|
+
Metrics/AbcSize:
|
26
|
+
Max: 25
|
27
|
+
Exclude:
|
28
|
+
- lib/remotus/ssh_connection.rb
|
29
|
+
|
30
|
+
Metrics/MethodLength:
|
31
|
+
Max: 20
|
32
|
+
Exclude:
|
33
|
+
- lib/remotus/ssh_connection.rb
|
34
|
+
|
35
|
+
Metrics/CyclomaticComplexity:
|
36
|
+
Exclude:
|
37
|
+
- lib/remotus/ssh_connection.rb
|
38
|
+
|
39
|
+
Metrics/PerceivedComplexity:
|
40
|
+
Exclude:
|
41
|
+
- lib/remotus/ssh_connection.rb
|
data/CHANGELOG.md
ADDED
data/Gemfile
ADDED
data/Gemfile.lock
ADDED
@@ -0,0 +1,103 @@
|
|
1
|
+
PATH
|
2
|
+
remote: .
|
3
|
+
specs:
|
4
|
+
remotus (0.1.0)
|
5
|
+
connection_pool (~> 2.2)
|
6
|
+
net-scp (~> 3.0)
|
7
|
+
net-ssh (~> 6.1)
|
8
|
+
winrm (~> 2.3)
|
9
|
+
winrm-fs (~> 1.3)
|
10
|
+
|
11
|
+
GEM
|
12
|
+
remote: https://rubygems.org/
|
13
|
+
specs:
|
14
|
+
ast (2.4.2)
|
15
|
+
builder (3.2.4)
|
16
|
+
connection_pool (2.2.3)
|
17
|
+
diff-lcs (1.4.4)
|
18
|
+
erubi (1.10.0)
|
19
|
+
ffi (1.15.0)
|
20
|
+
gssapi (1.3.1)
|
21
|
+
ffi (>= 1.0.1)
|
22
|
+
gyoku (1.3.1)
|
23
|
+
builder (>= 2.1.2)
|
24
|
+
httpclient (2.8.3)
|
25
|
+
little-plugger (1.1.4)
|
26
|
+
logging (2.3.0)
|
27
|
+
little-plugger (~> 1.1)
|
28
|
+
multi_json (~> 1.14)
|
29
|
+
multi_json (1.15.0)
|
30
|
+
net-scp (3.0.0)
|
31
|
+
net-ssh (>= 2.6.5, < 7.0.0)
|
32
|
+
net-ssh (6.1.0)
|
33
|
+
nori (2.6.0)
|
34
|
+
parallel (1.20.1)
|
35
|
+
parser (3.0.0.0)
|
36
|
+
ast (~> 2.4.1)
|
37
|
+
rainbow (3.0.0)
|
38
|
+
rake (13.0.3)
|
39
|
+
regexp_parser (2.1.1)
|
40
|
+
rexml (3.2.4)
|
41
|
+
rspec (3.10.0)
|
42
|
+
rspec-core (~> 3.10.0)
|
43
|
+
rspec-expectations (~> 3.10.0)
|
44
|
+
rspec-mocks (~> 3.10.0)
|
45
|
+
rspec-core (3.10.1)
|
46
|
+
rspec-support (~> 3.10.0)
|
47
|
+
rspec-expectations (3.10.1)
|
48
|
+
diff-lcs (>= 1.2.0, < 2.0)
|
49
|
+
rspec-support (~> 3.10.0)
|
50
|
+
rspec-mocks (3.10.2)
|
51
|
+
diff-lcs (>= 1.2.0, < 2.0)
|
52
|
+
rspec-support (~> 3.10.0)
|
53
|
+
rspec-support (3.10.2)
|
54
|
+
rubocop (1.11.0)
|
55
|
+
parallel (~> 1.10)
|
56
|
+
parser (>= 3.0.0.0)
|
57
|
+
rainbow (>= 2.2.2, < 4.0)
|
58
|
+
regexp_parser (>= 1.8, < 3.0)
|
59
|
+
rexml
|
60
|
+
rubocop-ast (>= 1.2.0, < 2.0)
|
61
|
+
ruby-progressbar (~> 1.7)
|
62
|
+
unicode-display_width (>= 1.4.0, < 3.0)
|
63
|
+
rubocop-ast (1.4.1)
|
64
|
+
parser (>= 2.7.1.5)
|
65
|
+
rubocop-rake (0.5.1)
|
66
|
+
rubocop
|
67
|
+
rubocop-rspec (2.2.0)
|
68
|
+
rubocop (~> 1.0)
|
69
|
+
rubocop-ast (>= 1.1.0)
|
70
|
+
ruby-progressbar (1.11.0)
|
71
|
+
rubyntlm (0.6.3)
|
72
|
+
rubyzip (2.3.0)
|
73
|
+
unicode-display_width (2.0.0)
|
74
|
+
winrm (2.3.6)
|
75
|
+
builder (>= 2.1.2)
|
76
|
+
erubi (~> 1.8)
|
77
|
+
gssapi (~> 1.2)
|
78
|
+
gyoku (~> 1.0)
|
79
|
+
httpclient (~> 2.2, >= 2.2.0.2)
|
80
|
+
logging (>= 1.6.1, < 3.0)
|
81
|
+
nori (~> 2.0)
|
82
|
+
rubyntlm (~> 0.6.0, >= 0.6.3)
|
83
|
+
winrm-fs (1.3.5)
|
84
|
+
erubi (~> 1.8)
|
85
|
+
logging (>= 1.6.1, < 3.0)
|
86
|
+
rubyzip (~> 2.0)
|
87
|
+
winrm (~> 2.0)
|
88
|
+
yard (0.9.26)
|
89
|
+
|
90
|
+
PLATFORMS
|
91
|
+
x86_64-linux
|
92
|
+
|
93
|
+
DEPENDENCIES
|
94
|
+
rake (~> 13.0)
|
95
|
+
remotus!
|
96
|
+
rspec (~> 3.0)
|
97
|
+
rubocop (~> 1.7)
|
98
|
+
rubocop-rake (~> 0.5)
|
99
|
+
rubocop-rspec (~> 2.2)
|
100
|
+
yard (~> 0.9)
|
101
|
+
|
102
|
+
BUNDLED WITH
|
103
|
+
2.2.9
|
data/LICENSE.txt
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2021 Matthew Newell
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,112 @@
|
|
1
|
+
# Remotus
|
2
|
+
|
3
|
+
Remotus provides a simple ruby interface for pooling remote SSH and WinRM connections and managing remote credentials. Custom authentication stores may be added to allow remote credentials to be automatically retrieved.
|
4
|
+
|
5
|
+
## Installation
|
6
|
+
|
7
|
+
Add this line to your application's Gemfile:
|
8
|
+
|
9
|
+
```ruby
|
10
|
+
gem 'remotus'
|
11
|
+
```
|
12
|
+
|
13
|
+
And then execute:
|
14
|
+
|
15
|
+
$ bundle install
|
16
|
+
|
17
|
+
Or install it yourself as:
|
18
|
+
|
19
|
+
$ gem install remotus
|
20
|
+
|
21
|
+
## Usage
|
22
|
+
|
23
|
+
```ruby
|
24
|
+
# Initialize a new connection pool to remotehost.local and auto-determine whether to use SSH or WinRM
|
25
|
+
connection = Remotus.connect("remotehost.local")
|
26
|
+
|
27
|
+
# Initialize a new connection pool to remotehost.local with a defined protocol and port
|
28
|
+
connection = Remotus.connect("remotehost.local", proto: :ssh, port: 2222)
|
29
|
+
|
30
|
+
# Create a credential for the new connection pool
|
31
|
+
connection.credential = Remotus::Auth::Credential.new("username", "password")
|
32
|
+
|
33
|
+
# Run a command on the remote host
|
34
|
+
result = connection.run("hostname")
|
35
|
+
|
36
|
+
result.command
|
37
|
+
# => "hostname"
|
38
|
+
|
39
|
+
result.stdout
|
40
|
+
# => "remotehost.local\n"
|
41
|
+
|
42
|
+
result.stderr
|
43
|
+
# => ""
|
44
|
+
|
45
|
+
result.output
|
46
|
+
# => "remotehost.local\n"
|
47
|
+
|
48
|
+
result.exit_code
|
49
|
+
# => 0
|
50
|
+
|
51
|
+
# Run a command on the remote host with sudo (Linux only, requires password to be specified)
|
52
|
+
result = connection.run("ls /root", sudo: true)
|
53
|
+
|
54
|
+
# Run a script on the remote host
|
55
|
+
connection.run_script("/local/script.sh", "/remote/path/script.sh")
|
56
|
+
|
57
|
+
# Run a script on the remote host with arguments
|
58
|
+
connection.run_script("/local/script.sh", "/remote/path/script.sh", "arg1", "arg2")
|
59
|
+
|
60
|
+
# Upload a file to the remote host
|
61
|
+
connection.upload("/local/file.txt", "/remote/path/file.txt")
|
62
|
+
|
63
|
+
# Download a file from the remote host
|
64
|
+
connection.download("/remote/path/file.txt", "/local/file.txt")
|
65
|
+
```
|
66
|
+
|
67
|
+
Full documentation is available in the [Remotus GitHub Pages](https://wheatevo.github.io/remotus/).
|
68
|
+
|
69
|
+
### Extending Remotus
|
70
|
+
|
71
|
+
Remotus may be extended by adding more authentication stores to gather remote credential data from centralized services.
|
72
|
+
|
73
|
+
#### Authentication Stores
|
74
|
+
|
75
|
+
More authentication stores may be added by creating a new class that inherits from `Remotus::Auth::Store` and implementing the `credential` method. The `credential` method should receive a `Remotus::SshConnection` or `Remotus::WinrmConnection` and an options hash and return a `Remotus::Auth::Credential` or `nil` if no credential can be found.
|
76
|
+
|
77
|
+
Once a new authentication store has been defined, ensure it is used by Remotus by adding it to `Remotus::Auth.stores`.
|
78
|
+
|
79
|
+
```ruby
|
80
|
+
# Define a new authentication store that returns "<hostname>_password" for any connection
|
81
|
+
#
|
82
|
+
# This is for demonstration purposes only, please do not use any password with your hostname or "password" in it :)
|
83
|
+
require "remotus"
|
84
|
+
|
85
|
+
class SimpleStore < Remotus::Auth::Store
|
86
|
+
def credential(connection, **options)
|
87
|
+
"#{connection.host}_password"
|
88
|
+
end
|
89
|
+
end
|
90
|
+
|
91
|
+
# Add the authentication store to the array of existing authentication stores
|
92
|
+
Remotus::Auth.stores << SimpleStore.new
|
93
|
+
|
94
|
+
# Use the authentication store
|
95
|
+
connection = Remotus.connect("remotehost.local", proto: :ssh)
|
96
|
+
Remotus::Auth.credential(connection)
|
97
|
+
# => "remotehost.local_password"
|
98
|
+
```
|
99
|
+
|
100
|
+
## Development
|
101
|
+
|
102
|
+
After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
|
103
|
+
|
104
|
+
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
105
|
+
|
106
|
+
## Contributing
|
107
|
+
|
108
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/wheatevo/remotus.
|
109
|
+
|
110
|
+
## License
|
111
|
+
|
112
|
+
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
|
data/Rakefile
ADDED
data/bin/console
ADDED
@@ -0,0 +1,15 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# frozen_string_literal: true
|
3
|
+
|
4
|
+
require "bundler/setup"
|
5
|
+
require "remotus"
|
6
|
+
|
7
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
8
|
+
# with your gem easier. You can also use a different console, if you like.
|
9
|
+
|
10
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
11
|
+
# require "pry"
|
12
|
+
# Pry.start
|
13
|
+
|
14
|
+
require "irb"
|
15
|
+
IRB.start(__FILE__)
|
data/bin/setup
ADDED
data/lib/remotus.rb
ADDED
@@ -0,0 +1,142 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "remotus/version"
|
4
|
+
require "remotus/logger"
|
5
|
+
require "remotus/pool"
|
6
|
+
require "remotus/auth"
|
7
|
+
|
8
|
+
# Contains classes and methods for creating remote connections and running commands
|
9
|
+
module Remotus
|
10
|
+
#
|
11
|
+
# Creates/gets a host pool for the host that can be used to perform remote operations.
|
12
|
+
# After creation, the host pool will persist for future connections.
|
13
|
+
#
|
14
|
+
# @param [String] host hostname
|
15
|
+
# @param [Hash] options options hash
|
16
|
+
# @option options [Integer] :size number of connections in the pool
|
17
|
+
# @option options [Integer] :timeout amount of time to wait for a connection from the pool
|
18
|
+
# @option options [Integer] :port port to use for the connection
|
19
|
+
# @option options [Symbol] :proto protocol to use for the connection (:winrm, :ssh), must be specified if port is specified
|
20
|
+
#
|
21
|
+
# @return [Remotus::HostPool] Newly created or retrieved host pool
|
22
|
+
#
|
23
|
+
def self.connect(host, **options)
|
24
|
+
Remotus::Pool.connect(host, **options)
|
25
|
+
end
|
26
|
+
|
27
|
+
#
|
28
|
+
# Number of host pools in the connection pool
|
29
|
+
#
|
30
|
+
# @return [Integer] number of host pools
|
31
|
+
#
|
32
|
+
def self.count
|
33
|
+
Remotus::Pool.count
|
34
|
+
end
|
35
|
+
|
36
|
+
#
|
37
|
+
# Removes all host pools from the pool in a thread-safe manner
|
38
|
+
#
|
39
|
+
# @return [Integer] number of host pools removed
|
40
|
+
#
|
41
|
+
def self.clear
|
42
|
+
Remotus::Pool.clear
|
43
|
+
end
|
44
|
+
|
45
|
+
#
|
46
|
+
# Checks if a remote port is open
|
47
|
+
#
|
48
|
+
# @param [String] host remote host
|
49
|
+
# @param [Integer] port remote port
|
50
|
+
# @param [Integer] timeout amount of time to wait in seconds
|
51
|
+
#
|
52
|
+
# @return [Boolean] true if port is open, false otherwise
|
53
|
+
#
|
54
|
+
def self.port_open?(host, port, timeout: 1)
|
55
|
+
logger.debug { "Checking if #{host}:#{port} is accessible" }
|
56
|
+
Socket.tcp(host, port, connect_timeout: timeout) { true }
|
57
|
+
rescue StandardError
|
58
|
+
logger.debug { "#{host}:#{port} is inaccessible" }
|
59
|
+
false
|
60
|
+
end
|
61
|
+
|
62
|
+
#
|
63
|
+
# Determine remote host type by checking common ports
|
64
|
+
#
|
65
|
+
# @param [String] host hostname
|
66
|
+
# @param [Integer] timeout amount of time to wait in seconds
|
67
|
+
#
|
68
|
+
# @return [Symbol, nil] :ssh, :winrm, or nil if it cannot be determined
|
69
|
+
#
|
70
|
+
def self.host_type(host, timeout: 1)
|
71
|
+
return :ssh if port_open?(host, SshConnection::REMOTE_PORT, timeout: timeout)
|
72
|
+
|
73
|
+
return :winrm if port_open?(host, WinrmConnection::REMOTE_PORT, timeout: timeout)
|
74
|
+
|
75
|
+
nil
|
76
|
+
end
|
77
|
+
|
78
|
+
#
|
79
|
+
# Gets the remotus logger
|
80
|
+
#
|
81
|
+
# @return [Remotus::Logger] current logger
|
82
|
+
#
|
83
|
+
def self.logger
|
84
|
+
@logger ||= Remotus::Logger.new($stdout, level: Logger::INFO)
|
85
|
+
end
|
86
|
+
|
87
|
+
#
|
88
|
+
# Sets the remotus logger
|
89
|
+
#
|
90
|
+
# @param [::Logger] logger logger to set
|
91
|
+
#
|
92
|
+
def self.logger=(logger)
|
93
|
+
if logger.nil?
|
94
|
+
self.logger.level = Logger::FATAL
|
95
|
+
return self.logger
|
96
|
+
end
|
97
|
+
@logger = logger
|
98
|
+
end
|
99
|
+
|
100
|
+
#
|
101
|
+
# Gets the remotus log formatter
|
102
|
+
#
|
103
|
+
# @return [::Logger::Formatter] current log formatter
|
104
|
+
#
|
105
|
+
def self.log_formatter
|
106
|
+
@log_formatter ||= logger.formatter
|
107
|
+
end
|
108
|
+
|
109
|
+
#
|
110
|
+
# Sets the remotus log formatter
|
111
|
+
#
|
112
|
+
# @param [::Logger::Formatter] log_formatter new log formatter
|
113
|
+
#
|
114
|
+
def self.log_formatter=(log_formatter)
|
115
|
+
@log_formatter = log_formatter
|
116
|
+
logger.formatter = log_formatter
|
117
|
+
end
|
118
|
+
|
119
|
+
# Generic base class for Remotus errors
|
120
|
+
class Error < StandardError; end
|
121
|
+
|
122
|
+
# Failed to obtain PTY during SSH connection
|
123
|
+
class PtyError < Error; end
|
124
|
+
|
125
|
+
# Failed to determine remote node host type
|
126
|
+
class HostTypeDeterminationError < Error; end
|
127
|
+
|
128
|
+
# Failed to authenticate to remote host
|
129
|
+
class AuthenticationError < Error; end
|
130
|
+
|
131
|
+
# Raised when a RemoteCommand::Result has an error
|
132
|
+
class ResultError < Error; end
|
133
|
+
|
134
|
+
# Raised when a derived class is missing an override method
|
135
|
+
class MissingOverride < Error; end
|
136
|
+
|
137
|
+
# Failed to find credential in credential store
|
138
|
+
class MissingCredential < Error; end
|
139
|
+
|
140
|
+
# Failed to find credential password when executing sudo command
|
141
|
+
class MissingSudoPassword < Error; end
|
142
|
+
end
|