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.
@@ -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
@@ -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
+ }
@@ -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