continuent-tools-core 0.9.0 → 0.10.6
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/README.md +23 -2
- data/bin/tungsten_analyze_thl_index +23 -10
- data/bin/tungsten_create_load +38 -7
- data/bin/tungsten_directory +379 -0
- data/bin/tungsten_manage_configuration +1654 -0
- data/bin/tungsten_migrate_schema +123 -0
- data/lib/continuent-tools-core.rb +123 -0
- data/lib/ipparse.rb +144 -0
- data/lib/ipparse/README +18 -0
- data/lib/ipparse/platforms/linux.rb +103 -0
- data/lib/tungsten/datasources/mysql.rb +4 -36
- data/lib/tungsten/exec.rb +16 -16
- data/lib/tungsten/script.rb +57 -121
- data/lib/tungsten/util.rb +0 -9
- data/providers/awsec2.rb +141 -0
- metadata +12 -2
@@ -0,0 +1,123 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# Copyright (C) 2014 Continuent, Inc.
|
3
|
+
#
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
5
|
+
# not use this file except in compliance with the License. You may obtain
|
6
|
+
# a copy of the License at
|
7
|
+
#
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
#
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
12
|
+
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
13
|
+
# License for the specific language governing permissions and limitations
|
14
|
+
# under the License.
|
15
|
+
#
|
16
|
+
# Initial developer(s): Jeff Mace
|
17
|
+
# Contributor(s):
|
18
|
+
|
19
|
+
begin
|
20
|
+
require 'rubygems'
|
21
|
+
gem 'continuent-tools-core'
|
22
|
+
rescue LoadError
|
23
|
+
end
|
24
|
+
|
25
|
+
require 'continuent-tools-core'
|
26
|
+
require 'tempfile'
|
27
|
+
|
28
|
+
class TungstenMigrateSchema
|
29
|
+
include TungstenScript
|
30
|
+
include MySQLServiceScript
|
31
|
+
include OfflineSingleServiceScript
|
32
|
+
private
|
33
|
+
|
34
|
+
def main
|
35
|
+
f = Tempfile.new("#{script_name()}")
|
36
|
+
|
37
|
+
TU.debug("Create mysqldump in #{f.path()}")
|
38
|
+
if opt(:log_updates) == false
|
39
|
+
TU.cmd_result("echo \"SET SESSION SQL_LOG_BIN=0;\n\" >> #{f.path()}")
|
40
|
+
end
|
41
|
+
|
42
|
+
if opt(:drop_original_schema) == true
|
43
|
+
TU.cmd_result("echo \"DROP SCHEMA IF EXISTS \\`#{opt(:to)}\\`;\n\" >> #{f.path()}")
|
44
|
+
end
|
45
|
+
|
46
|
+
TU.cmd_result("echo \"CREATE SCHEMA \\`#{opt(:to)}\\`;\n\" >> #{f.path()}")
|
47
|
+
TU.cmd_result("echo \"USE '#{opt(:to)}';\n\" >> #{f.path()}")
|
48
|
+
TU.cmd_result("#{get_mysqldump_command(opt(:from))} >> #{f.path()}")
|
49
|
+
|
50
|
+
if opt(:drop_target_schema) == true
|
51
|
+
TU.cmd_result("echo \"DROP SCHEMA \\`#{opt(:from)}\\`;\n\" >> #{f.path()}")
|
52
|
+
end
|
53
|
+
|
54
|
+
TU.cmd_result("cat #{f.path()} | #{get_mysql_command()}")
|
55
|
+
end
|
56
|
+
|
57
|
+
def get_mysqldump_command(schema)
|
58
|
+
"mysqldump --defaults-file=#{@options[:my_cnf]} --host=#{@options[:mysqlhost]} --port=#{@options[:mysqlport]} --opt --single-transaction #{schema}"
|
59
|
+
end
|
60
|
+
|
61
|
+
def configure
|
62
|
+
super()
|
63
|
+
|
64
|
+
add_option(:from, {
|
65
|
+
:on => "--from-schema String",
|
66
|
+
:help => "The existing schema name to use",
|
67
|
+
:required => true,
|
68
|
+
})
|
69
|
+
|
70
|
+
add_option(:to, {
|
71
|
+
:on => "--to-schema String",
|
72
|
+
:help => "The schema name to create",
|
73
|
+
:required => true,
|
74
|
+
})
|
75
|
+
|
76
|
+
add_option(:log_updates, {
|
77
|
+
:on => "--log-updates String",
|
78
|
+
:parse => method(:parse_boolean_option),
|
79
|
+
:default => false,
|
80
|
+
:help => "Force the script to log updates to the database",
|
81
|
+
})
|
82
|
+
|
83
|
+
add_option(:drop_original_schema, {
|
84
|
+
:on => "--drop-original-schema String",
|
85
|
+
:parse => method(:parse_boolean_option),
|
86
|
+
:default => false,
|
87
|
+
:help => "Drop the schema identified by --from-schema",
|
88
|
+
})
|
89
|
+
|
90
|
+
add_option(:drop_target_schema, {
|
91
|
+
:on => "--drop-target-schema String",
|
92
|
+
:parse => method(:parse_boolean_option),
|
93
|
+
:default => false,
|
94
|
+
:help => "Drop the schema identified by --to-schema",
|
95
|
+
})
|
96
|
+
end
|
97
|
+
|
98
|
+
def validate
|
99
|
+
super()
|
100
|
+
|
101
|
+
unless TU.is_valid?()
|
102
|
+
return TU.is_valid?()
|
103
|
+
end
|
104
|
+
|
105
|
+
unless opt(:drop_original_schema) == true
|
106
|
+
if get_mysql_result("SHOW SCHEMAS LIKE '#{opt(:from)}'") == ""
|
107
|
+
TU.error("Unable to find an existing '#{opt(:from)}' schema")
|
108
|
+
end
|
109
|
+
end
|
110
|
+
|
111
|
+
unless opt(:drop_target_schema) == true
|
112
|
+
if get_mysql_result("SHOW SCHEMAS LIKE '#{opt(:to)}'") != ""
|
113
|
+
TU.error("There is already an existing '#{opt(:to)}' schema")
|
114
|
+
end
|
115
|
+
end
|
116
|
+
end
|
117
|
+
|
118
|
+
def script_name
|
119
|
+
"tungsten_migrate_schema"
|
120
|
+
end
|
121
|
+
|
122
|
+
self.new().run()
|
123
|
+
end
|
@@ -20,6 +20,22 @@
|
|
20
20
|
require 'tungsten'
|
21
21
|
|
22
22
|
class TungstenUtil
|
23
|
+
# Override the method imported from tpm to include '>>' to avoid detecting
|
24
|
+
# an error that wasn't thrown by TungstenScript
|
25
|
+
def parse_log_level(line)
|
26
|
+
prefix = line[0,8]
|
27
|
+
|
28
|
+
case prefix.strip
|
29
|
+
when "ERROR >>" then Logger::ERROR
|
30
|
+
when "WARN >>" then Logger::WARN
|
31
|
+
when "DEBUG >>" then Logger::DEBUG
|
32
|
+
when "NOTE >>" then Logger::NOTICE
|
33
|
+
when "INFO >>" then Logger::INFO
|
34
|
+
else
|
35
|
+
nil
|
36
|
+
end
|
37
|
+
end
|
38
|
+
|
23
39
|
# A wrapper for running another Tungsten script. This will automatically
|
24
40
|
# forward messages to the console and add any TungstenScript options to
|
25
41
|
# the command.
|
@@ -139,6 +155,113 @@ class TungstenUtil
|
|
139
155
|
|
140
156
|
return result
|
141
157
|
end
|
158
|
+
|
159
|
+
def scp_result(local_file, remote_file, host, user)
|
160
|
+
unless File.file?(local_file)
|
161
|
+
debug("Unable to copy '#{local_file}' because it doesn't exist")
|
162
|
+
raise MessageError.new("Unable to copy '#{local_file}' because it doesn't exist")
|
163
|
+
end
|
164
|
+
|
165
|
+
if is_localhost?(host) &&
|
166
|
+
user == whoami()
|
167
|
+
debug("Copy #{local_file} to #{remote_file}")
|
168
|
+
return FileUtils.cp(local_file, remote_file)
|
169
|
+
end
|
170
|
+
|
171
|
+
self.synchronize() {
|
172
|
+
unless defined?(Net::SCP)
|
173
|
+
begin
|
174
|
+
require "openssl"
|
175
|
+
rescue LoadError
|
176
|
+
raise("Unable to find the Ruby openssl library. Try installing the openssl package for your version of Ruby (libopenssl-ruby#{RUBY_VERSION[0,3]}).")
|
177
|
+
end
|
178
|
+
require 'net/scp'
|
179
|
+
end
|
180
|
+
}
|
181
|
+
|
182
|
+
ssh_user = get_ssh_user(user)
|
183
|
+
if user != ssh_user
|
184
|
+
debug("SCP user changed to #{ssh_user}")
|
185
|
+
end
|
186
|
+
|
187
|
+
connection_error = "Net::SCP was unable to copy #{local_file} to #{host}:#{remote_file} as #{ssh_user}. Check that #{host} is online, #{ssh_user} exists and your SSH private keyfile or ssh-agent settings. Try adding --net-ssh-option=port=<SSH port number> if you are using an SSH port other than 22. Review http://docs.continuent.com/helpwithsshandtpm for more help on diagnosing SSH problems."
|
188
|
+
debug("Copy #{local_file} to #{host}:#{remote_file} as #{ssh_user}")
|
189
|
+
begin
|
190
|
+
Net::SCP.start(host, ssh_user, get_ssh_options) do |scp|
|
191
|
+
scp.upload!(local_file, remote_file, get_ssh_options)
|
192
|
+
end
|
193
|
+
|
194
|
+
if user != ssh_user
|
195
|
+
ssh_result("sudo -n chown -R #{user} #{remote_file}", host, ssh_user)
|
196
|
+
end
|
197
|
+
|
198
|
+
return true
|
199
|
+
rescue Errno::ENOENT => ee
|
200
|
+
raise MessageError.new("Net::SCP was unable to find a private key to use for SSH authenticaton. Try creating a private keyfile or setting up ssh-agent.")
|
201
|
+
rescue OpenSSL::PKey::RSAError
|
202
|
+
raise MessageError.new(connection_error)
|
203
|
+
rescue Net::SSH::AuthenticationFailed
|
204
|
+
raise MessageError.new(connection_error)
|
205
|
+
rescue Errno::ECONNREFUSED, Errno::EHOSTUNREACH
|
206
|
+
raise MessageError.new(connection_error)
|
207
|
+
rescue Timeout::Error
|
208
|
+
raise MessageError.new(connection_error)
|
209
|
+
rescue Exception => e
|
210
|
+
raise RemoteCommandError.new(user, host, "scp #{local_file} #{ssh_user}@#{host}:#{remote_file}", nil, '')
|
211
|
+
end
|
212
|
+
end
|
213
|
+
|
214
|
+
def scp_download(remote_file, local_file, host, user)
|
215
|
+
if is_localhost?(host) && user == whoami()
|
216
|
+
debug("Copy #{remote_file} to #{local_file}")
|
217
|
+
return FileUtils.cp(remote_file, local_file)
|
218
|
+
end
|
219
|
+
|
220
|
+
begin
|
221
|
+
exists = ssh_result("if [ -f #{remote_file} ]; then echo 0; else echo 1; fi", host, user)
|
222
|
+
if exists == "1"
|
223
|
+
raise MessageError.new("Unable to download '#{remote_file}' because the file does not exist on #{host}")
|
224
|
+
end
|
225
|
+
rescue CommandError
|
226
|
+
raise MessageError.new("Unable to check if '#{remote_file}' exists on #{host}")
|
227
|
+
end
|
228
|
+
|
229
|
+
self.synchronize() {
|
230
|
+
unless defined?(Net::SCP)
|
231
|
+
begin
|
232
|
+
require "openssl"
|
233
|
+
rescue LoadError
|
234
|
+
raise("Unable to find the Ruby openssl library. Try installing the openssl package for your version of Ruby (libopenssl-ruby#{RUBY_VERSION[0,3]}).")
|
235
|
+
end
|
236
|
+
require 'net/scp'
|
237
|
+
end
|
238
|
+
}
|
239
|
+
|
240
|
+
ssh_user = get_ssh_user(user)
|
241
|
+
if user != ssh_user
|
242
|
+
debug("SCP user changed to #{ssh_user}")
|
243
|
+
end
|
244
|
+
|
245
|
+
connection_error = "Net::SCP was unable to copy to #{host}:#{remote_file} #{local_file} as #{ssh_user}. Check that #{host} is online, #{ssh_user} exists and your SSH private keyfile or ssh-agent settings. Try adding --net-ssh-option=port=<SSH port number> if you are using an SSH port other than 22. Review http://docs.continuent.com/helpwithsshandtpm for more help on diagnosing SSH problems."
|
246
|
+
debug("Copy #{host}:#{remote_file} to #{local_file} as #{ssh_user}")
|
247
|
+
begin
|
248
|
+
Net::SCP.download!(host, ssh_user, remote_file, local_file, get_ssh_options)
|
249
|
+
|
250
|
+
return true
|
251
|
+
rescue Errno::ENOENT => ee
|
252
|
+
raise MessageError.new("Net::SCP was unable to find a private key to use for SSH authenticaton. Try creating a private keyfile or setting up ssh-agent.")
|
253
|
+
rescue OpenSSL::PKey::RSAError
|
254
|
+
raise MessageError.new(connection_error)
|
255
|
+
rescue Net::SSH::AuthenticationFailed
|
256
|
+
raise MessageError.new(connection_error)
|
257
|
+
rescue Errno::ECONNREFUSED, Errno::EHOSTUNREACH
|
258
|
+
raise MessageError.new(connection_error)
|
259
|
+
rescue Timeout::Error
|
260
|
+
raise MessageError.new(connection_error)
|
261
|
+
rescue Exception => e
|
262
|
+
raise RemoteCommandError.new(user, host, "scp #{ssh_user}@#{host}:#{remote_file} #{local_file}", nil, '')
|
263
|
+
end
|
264
|
+
end
|
142
265
|
end
|
143
266
|
|
144
267
|
module TungstenScript
|
data/lib/ipparse.rb
ADDED
@@ -0,0 +1,144 @@
|
|
1
|
+
# VMware Continuent Tungsten Replicator
|
2
|
+
# Copyright (C) 2015 VMware, Inc. All rights reserved.
|
3
|
+
#
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
5
|
+
# you may not use this file except in compliance with the License.
|
6
|
+
# You may obtain a copy of the License at
|
7
|
+
#
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
#
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
13
|
+
# See the License for the specific language governing permissions and
|
14
|
+
# limitations under the License.
|
15
|
+
#
|
16
|
+
# Initial developer(s): Jeff Mace
|
17
|
+
|
18
|
+
class IPParse
|
19
|
+
IPV4 = :ipv4
|
20
|
+
IPV6 = :ipv6
|
21
|
+
|
22
|
+
def initialize
|
23
|
+
@klass = get_platform_klass().new()
|
24
|
+
@parsed = nil
|
25
|
+
end
|
26
|
+
|
27
|
+
# Return a nested array of the parsed interface information
|
28
|
+
# {
|
29
|
+
# "eth0" => {
|
30
|
+
# :ipv4 => { :address => "1.2.3.4", :netmask => "255.255.255.0" },
|
31
|
+
# :ipv6 => { :address => "fe80::a00:27ff:fe1f:84a5", :netmask => "64" }
|
32
|
+
# }
|
33
|
+
# }
|
34
|
+
def get_interfaces
|
35
|
+
parse()
|
36
|
+
return @parsed
|
37
|
+
end
|
38
|
+
|
39
|
+
def get_interface_address(interface, type = IPV4)
|
40
|
+
parse()
|
41
|
+
|
42
|
+
unless @parsed.has_key?(interface)
|
43
|
+
return nil
|
44
|
+
end
|
45
|
+
|
46
|
+
unless @parsed[interface].has_key?(type)
|
47
|
+
return nil
|
48
|
+
end
|
49
|
+
|
50
|
+
return @parsed[interface][type]
|
51
|
+
end
|
52
|
+
|
53
|
+
private
|
54
|
+
|
55
|
+
def get_platform
|
56
|
+
RUBY_PLATFORM
|
57
|
+
end
|
58
|
+
|
59
|
+
# Have the platform class parse the IP configuration information
|
60
|
+
# it just collected. If the information has already been parsed
|
61
|
+
# the steps can be skipped.
|
62
|
+
def parse
|
63
|
+
if @parsed != nil
|
64
|
+
return
|
65
|
+
end
|
66
|
+
|
67
|
+
@parsed = @klass.parse(get_raw_ip_configuration())
|
68
|
+
end
|
69
|
+
|
70
|
+
# Have the platform class to collect IP configuration information
|
71
|
+
def get_raw_ip_configuration
|
72
|
+
@klass.get_raw_ip_configuration()
|
73
|
+
end
|
74
|
+
|
75
|
+
# Iterate through all of the platform classes until one declares
|
76
|
+
# it is able to support the current system.
|
77
|
+
def get_platform_klass(platform = nil)
|
78
|
+
if platform == nil
|
79
|
+
platform = get_platform()
|
80
|
+
end
|
81
|
+
|
82
|
+
platform = platform.downcase()
|
83
|
+
|
84
|
+
IPParsePlatform.subclasses().each{
|
85
|
+
|klass|
|
86
|
+
begin
|
87
|
+
supports_platform = klass.supports_platform?(platform)
|
88
|
+
if supports_platform == true
|
89
|
+
return klass
|
90
|
+
end
|
91
|
+
rescue NoMethodError
|
92
|
+
# Eat this error because the platform didn't define the required method
|
93
|
+
end
|
94
|
+
}
|
95
|
+
|
96
|
+
throw Exception.new("Unable to support the #{platform} platform")
|
97
|
+
end
|
98
|
+
end
|
99
|
+
|
100
|
+
class IPParsePlatform
|
101
|
+
def initialize
|
102
|
+
@interfaces = {}
|
103
|
+
end
|
104
|
+
|
105
|
+
def add_ipv4(name, address, netmask)
|
106
|
+
unless @interfaces.has_key?(name)
|
107
|
+
@interfaces[name] = {}
|
108
|
+
end
|
109
|
+
|
110
|
+
@interfaces[name][IPParse::IPV4] = {
|
111
|
+
:address => address,
|
112
|
+
:netmask => netmask
|
113
|
+
}
|
114
|
+
end
|
115
|
+
|
116
|
+
def add_ipv6(name, address, netmask)
|
117
|
+
unless @interfaces.has_key?(name)
|
118
|
+
@interfaces[name] = {}
|
119
|
+
end
|
120
|
+
|
121
|
+
@interfaces[name][IPParse::IPV6] = {
|
122
|
+
:address => address,
|
123
|
+
:netmask => netmask
|
124
|
+
}
|
125
|
+
end
|
126
|
+
|
127
|
+
def get_raw_ip_configuration
|
128
|
+
throw Exception.new("Undefined method #{self.class.name}:get_raw_ip_configuration")
|
129
|
+
end
|
130
|
+
|
131
|
+
def self.inherited(subclass)
|
132
|
+
@subclasses ||= []
|
133
|
+
@subclasses << subclass
|
134
|
+
end
|
135
|
+
|
136
|
+
def self.subclasses
|
137
|
+
@subclasses
|
138
|
+
end
|
139
|
+
end
|
140
|
+
|
141
|
+
# Load all platform classes so they can be registered
|
142
|
+
Dir.glob("#{File.dirname(__FILE__)}/ipparse/platforms/*.rb").each{|platform|
|
143
|
+
require platform
|
144
|
+
}
|
data/lib/ipparse/README
ADDED
@@ -0,0 +1,18 @@
|
|
1
|
+
About
|
2
|
+
===
|
3
|
+
|
4
|
+
This library provides functionality to parse the IP information for a host. It
|
5
|
+
is designed with a limited feature set based on what is necessary for the
|
6
|
+
VMware Continuent project.
|
7
|
+
|
8
|
+
Platform Specifics
|
9
|
+
===
|
10
|
+
|
11
|
+
On Linux platforms, the library will only return values where the link encapsulation is 'Ethernet'. This eliminates loopback and specialty devices.
|
12
|
+
|
13
|
+
Known Limitations
|
14
|
+
===
|
15
|
+
|
16
|
+
The ipparse library does not support interfaces that include multiple IPV6
|
17
|
+
addresses on a single interface. It will identify the first address and return
|
18
|
+
that. Any additional addresses will be ignored.
|
@@ -0,0 +1,103 @@
|
|
1
|
+
# VMware Continuent Tungsten Replicator
|
2
|
+
# Copyright (C) 2015 VMware, Inc. All rights reserved.
|
3
|
+
#
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
5
|
+
# you may not use this file except in compliance with the License.
|
6
|
+
# You may obtain a copy of the License at
|
7
|
+
#
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
#
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
13
|
+
# See the License for the specific language governing permissions and
|
14
|
+
# limitations under the License.
|
15
|
+
#
|
16
|
+
# Initial developer(s): Jeff Mace
|
17
|
+
|
18
|
+
class LinuxIPParsePlatform < IPParsePlatform
|
19
|
+
def self.supports_platform?(platform)
|
20
|
+
if platform.downcase() =~ /linux/
|
21
|
+
true
|
22
|
+
else
|
23
|
+
false
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
def get_raw_ip_configuration
|
28
|
+
path = `which ifconfig 2>/dev/null`.chomp()
|
29
|
+
if path == ""
|
30
|
+
path = "/sbin/ifconfig"
|
31
|
+
end
|
32
|
+
|
33
|
+
results = `export LANG=en_US; #{path} -a`
|
34
|
+
if results == false
|
35
|
+
raise "Unable to collect IP configuration from ifconfig"
|
36
|
+
else
|
37
|
+
return results
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def parse(raw)
|
42
|
+
name_regex = Regexp.compile(/^([a-zA-Z0-9]+)/)
|
43
|
+
encapsulation_regex = Regexp.compile(/Link encap:(\S+)/)
|
44
|
+
flags_regex = Regexp.compile(/flags=([0-9]+)\<([A-Z,]+)\>/)
|
45
|
+
inet4_regex1 = Regexp.compile(/inet addr:([0-9\.]+)[ ]+(Bcast:[0-9\.]+ )?[ ]+Mask:([0-9\.]+)/)
|
46
|
+
inet4_regex2 = Regexp.compile(/inet ([0-9\.]+)[ ]+netmask ([0-9\.]+)[ ]+(broadcast [0-9\.]+ )?/)
|
47
|
+
inet6_regex1 = Regexp.compile(/inet6 addr:[ ]*([a-f0-9:]+)\/([0-9]+)/)
|
48
|
+
inet6_regex2 = Regexp.compile(/inet6 ([a-f0-9:]+)[ ]*prefixlen ([0-9]+)/)
|
49
|
+
|
50
|
+
raw.split("\n\n").each{
|
51
|
+
|ifconfig|
|
52
|
+
include_interface = false
|
53
|
+
|
54
|
+
begin
|
55
|
+
encapsulation = encapsulation_regex.match(ifconfig)[1]
|
56
|
+
|
57
|
+
if encapsulation.downcase() == "ethernet"
|
58
|
+
include_interface = true
|
59
|
+
end
|
60
|
+
rescue
|
61
|
+
# Catch the exception and move on
|
62
|
+
end
|
63
|
+
|
64
|
+
begin
|
65
|
+
flags = flags_regex.match(ifconfig)[2]
|
66
|
+
flags = flags.split(",")
|
67
|
+
|
68
|
+
if flags.include?("LOOPBACK") == false
|
69
|
+
include_interface = true
|
70
|
+
end
|
71
|
+
rescue
|
72
|
+
# Catch the exception and move on
|
73
|
+
end
|
74
|
+
|
75
|
+
if include_interface == false
|
76
|
+
next
|
77
|
+
end
|
78
|
+
|
79
|
+
name = name_regex.match(ifconfig)[1]
|
80
|
+
if name == nil
|
81
|
+
raise "Unable to parse IP configuration because a valid name does not exist"
|
82
|
+
end
|
83
|
+
|
84
|
+
m1 = inet4_regex1.match(ifconfig)
|
85
|
+
m2 = inet4_regex2.match(ifconfig)
|
86
|
+
if m1
|
87
|
+
add_ipv4(name, m1[1], m1[3])
|
88
|
+
elsif m2
|
89
|
+
add_ipv4(name, m2[1], m2[2])
|
90
|
+
end
|
91
|
+
|
92
|
+
m1 = inet6_regex1.match(ifconfig)
|
93
|
+
m2 = inet6_regex2.match(ifconfig)
|
94
|
+
if m1
|
95
|
+
add_ipv6(name, m1[1], m1[2])
|
96
|
+
elsif m2
|
97
|
+
add_ipv6(name, m2[1], m2[2])
|
98
|
+
end
|
99
|
+
}
|
100
|
+
|
101
|
+
return @interfaces
|
102
|
+
end
|
103
|
+
end
|