Module: BlackStack::Infrastructure::NodeModule

Included in:
Node
Defined in:
lib/blackstack-nodes.rb

Overview

this module has attributes an methods used by both classes Node and Node.

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Instance Attribute Details

#ipObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def ip
  @ip
end

#loggerObject

non-database attributes, used for ssh connection and logging



13
14
15
# File 'lib/blackstack-nodes.rb', line 13

def logger
  @logger
end

#nameObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def name
  @name
end

#sshObject

non-database attributes, used for ssh connection and logging



13
14
15
# File 'lib/blackstack-nodes.rb', line 13

def ssh
  @ssh
end

#ssh_passwordObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def ssh_password
  @ssh_password
end

#ssh_portObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def ssh_port
  @ssh_port
end

#ssh_private_key_fileObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def ssh_private_key_file
  @ssh_private_key_file
end

#ssh_usernameObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def ssh_username
  @ssh_username
end

#tagsObject

:name is this is just a descriptive name for the node. It is not the host name, nor the domain, nor any ip.



11
12
13
# File 'lib/blackstack-nodes.rb', line 11

def tags
  @tags
end

Class Method Details

.descriptor_errors(h) ⇒ Object



15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
# File 'lib/blackstack-nodes.rb', line 15

def self.descriptor_errors(h)
  errors = []

  # validate: the parameter h is a hash
  errors << "The parameter h is not a hash" unless h.is_a?(Hash)

  # validate: the parameter h has a key :name
  errors << "The parameter h does not have a key :name" unless h.has_key?(:name)

  # validate: the parameter h[:name] is a string
  errors << "The parameter h[:name] is not a string" unless h[:name].is_a?(String)

  # validate: the paramerer h has a key :ip
  errors << "The parameter h does not have a key :ip" unless h.has_key?(:ip)

  # validate: the paramerer h has a key :ssh_username
  errors << "The parameter h does not have a key :ssh_username" unless h.has_key?(:ssh_username)

  # validate: the parameter h[:ssh_username] is a string
  errors << "The parameter h[:ssh_username] is not a string" unless h[:ssh_username].is_a?(String)

  # if the parameter h has a key :ssh_private_key_file
  if h.has_key?(:ssh_private_key_file) && !h[:ssh_private_key_file].nil?
    # validate: the parameter h[:ssh_private_key_file] is a string
    errors << "The parameter h[:ssh_private_key_file] is not a string" unless h[:ssh_private_key_file].is_a?(String)

    # validate: the parameter h[:ssh_private_key_file] is a string
    errors << "The parameter h[:ssh_private_key_file] is not a string" unless h[:ssh_private_key_file].is_a?(String)
  else
    # validate: the parameter h has a key :ssh_password
    errors << "The parameter h does not have a key :ssh_password nor :ssh_private_key_file" unless h.has_key?(:ssh_password)

    # validate: the parameter h[:ssh_password] is a string
    errors << "The parameter h[:ssh_password] is not a string" unless h[:ssh_password].is_a?(String)
  end

  # if the parameter h has a key :tags
  if h.has_key?(:tags) && !h[:tags].nil?
    # validate: the parameter h[:tags] is an array or a string
    errors << "The parameter h[:tags] is not an array or a string" unless h[:tags].is_a?(Array) || h[:tags].is_a?(String)
  end

  # return
  errors
end

Instance Method Details

#connectObject



115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
# File 'lib/blackstack-nodes.rb', line 115

def connect
  # connect
  if self.using_password?
    self.ssh = Net::SSH.start(
      self.ip, 
      self.ssh_username, 
      :password => self.ssh_password, 
      :port => self.ssh_port,
      :verify_host_key => :never,       # Disable host key verification
      :non_interactive => true,         # Ensure non-interactive mode
      :timeout => 10,                   # Set a connection timeout (in seconds)
      #:verbose => :debug                   # Enable verbose logging
    )
  elsif self.using_private_key_file?
    self.ssh = Net::SSH.start(self.ip, self.ssh_username, :keys => self.ssh_private_key_file, :port => self.ssh_port)
  else
    raise "No ssh credentials available"
  end
  self.ssh
end

#connected?Boolean

Returns true if the SSH connection is established

Returns:

  • (Boolean)


110
111
112
# File 'lib/blackstack-nodes.rb', line 110

def connected?
  !disconnected?
end

#disconnectObject

def connect



136
137
138
# File 'lib/blackstack-nodes.rb', line 136

def disconnect
  self.ssh.close
end

#disconnected?Boolean

Returns true if the SSH connection is not established or inactive

Returns:

  • (Boolean)


105
106
107
# File 'lib/blackstack-nodes.rb', line 105

def disconnected?
  self.ssh.nil? || self.ssh.closed?
end

#exec(command, output_file: "$HOME/.bash-command-stdout-buffer", error_file: "$HOME/.bash-command-stderr-buffer", exit_code_file: "$HOME/.bash-command-exit-code") ⇒ Object

Execute a command on the remote server and return the output. If the exit code of the command is not zero, an exception is raised, with the messages pushed into the stderr. If the exit code of the command is zero, the messages pushed into the stdout are returned.

IMPORTANT: Messages pushed into stderr re not always error. Reference:



149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
# File 'lib/blackstack-nodes.rb', line 149

def exec(
  command, 
  output_file: "$HOME/.bash-command-stdout-buffer", 
  error_file: "$HOME/.bash-command-stderr-buffer",
  exit_code_file: "$HOME/.bash-command-exit-code"
)
  # Construct the remote command with redirection for stdout, stderr, and capturing exit code
  remote_command = "#{command} > #{output_file} 2> #{error_file}; echo $? > #{exit_code_file}"

  # Execute the command on the remote server, truncating output, error, and exit code files
  self.ssh.exec!("truncate -s 0 #{output_file} #{error_file} #{exit_code_file}")

  # Execute the command on the remote server, truncating output, error, and exit code files
  self.ssh.exec!(remote_command)

  # Retrieve the exit code
  exit_code = self.ssh.exec!("cat #{exit_code_file}").to_s.chomp.to_i

  # Retrieve the content of the error file from the remote server
  error_content = self.ssh.exec!("cat #{error_file}").to_s.chomp

  # Check if the exit code is not zero
  if exit_code != 0
    # Raise an exception with the error message
    raise "Command failed with exit code #{exit_code}:\n#{error_content}"
  end

  # Retrieve and return the content of the output file from the remote server
  # Truncate any trailing newline character
  self.ssh.exec!("cat #{output_file}").to_s.chomp
end

#initialize(h, i_logger = nil) ⇒ Object

def self.descriptor_errors(h)



61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
# File 'lib/blackstack-nodes.rb', line 61

def initialize(h, i_logger=nil)
  errors = BlackStack::Infrastructure::NodeModule.descriptor_errors(h)
  # raise an exception if any error happneed
  raise "The node descriptor is not valid: #{errors.uniq.join(".\n")}" if errors.length > 0
  # map attributes
  self.name = h[:name]
  self.ip = h[:ip]
  self.ssh_username = h[:ssh_username]
  self.ssh_password = h[:ssh_password] 
  self.ssh_port = h[:ssh_port]
  self.ssh_private_key_file = h[:ssh_private_key_file]
  # parse the tags
  if h.has_key?(:tags) && !h[:tags].nil?
    self.tags = h[:tags].is_a?(Array) ? h[:tags] : [h[:tags]]
  else
    self.tags = []
  end
  # create a logger
  self.logger = !i_logger.nil? ? i_logger : BlackStack::DummyLogger.new(nil)
end

#rebootObject

def exec



181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
# File 'lib/blackstack-nodes.rb', line 181

def reboot()
  tries = 0
  max_tries = 20
  success = false

  host = self

  logger.logs 'reboot... '
  #stdout = host.reboot
  begin
    stdout = self.exec("reboot")
  rescue
  end
  logger.done #logf("done (#{stdout})")

  while tries < max_tries && !success
      begin
          tries += 1

          delay = 10
          logger.logs "wait #{delay.to_s} seconds... "
          sleep(delay)
          logger.done

          logger.logs "connecting (try #{tries.to_s})... "
          host.connect
          logger.done

          success = true
      rescue => e
          logger.logf e.to_s #error e
      end
  end # while 
  raise 'reboot failed' if !success
end

#tail(filename, n = 10) ⇒ Object

return the latest n` lines of the file specified by the filename parameter



270
271
272
# File 'lib/blackstack-nodes.rb', line 270

def tail(filename, n=10)
  self.ssh.exec!("tail -n #{n.to_s} #{filename}")
end

#to_hashObject

def self.create(h)



82
83
84
85
86
87
88
89
90
91
92
# File 'lib/blackstack-nodes.rb', line 82

def to_hash
  {
    :name => self.name,
    :ip => self.ip,
    :ssh_username => self.ssh_username,
    :ssh_password => self.ssh_password, 
    :ssh_port => self.ssh_port,
    :ssh_private_key_file => self.ssh_private_key_file,
    :tags => self.tags
  }
end

#usageObject

Return a hash descriptor of the status of the node



218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
# File 'lib/blackstack-nodes.rb', line 218

def usage()
  ret = {}

  #self.connect

  ret[:b_total_memory] = self.ssh.exec!('cat /proc/meminfo | grep MemTotal').delete('^0-9').to_i*1024
  ret[:kb_total_memory] = ret[:b_total_memory] / 1024
  ret[:mb_total_memory] = ret[:kb_total_memory] / 1024
  ret[:gb_total_memory] = ret[:mb_total_memory] / 1024

  ret[:kb_free_memory] = self.ssh.exec!('cat /proc/meminfo | grep MemFree').delete('^0-9').to_i
  ret[:mb_free_memory] = ret[:kb_free_memory] / 1024
  ret[:gb_free_memory] = ret[:mb_free_memory] / 1024

  # run bash commend to get the total disk space
  ret[:mb_total_disk] = self.ssh.exec!('df -m / | tail -1 | awk \'{print $2}\'').to_i
  ret[:gb_total_disk] = ret[:mb_total_disk] / 1024
  # run bash command to get the free disk space
  ret[:mb_free_disk] = self.ssh.exec!('df -m / | tail -1 | awk \'{print $4}\'').to_i
  ret[:gb_free_disk] = ret[:mb_free_disk] / 1024
  
  # run bash command to get hostname
  ret[:hostname] = self.ssh.exec!('hostname').strip!

  # run bash command to get the CPU load
  # reference: https://stackoverflow.com/questions/9229333/how-to-get-overall-cpu-usage-e-g-57-on-linux
  ret[:cpu_load_average] = self.ssh.exec!("awk '{u=$2+$4; t=$2+$4+$5; if (NR==1){u1=u; t1=t;} else print ($2+$4-u1) * 100 / (t-t1) \"%\"; }' <(grep 'cpu ' /proc/stat) <(sleep 1;grep 'cpu ' /proc/stat)").to_s

  # TODO: monitor the overall Network I/O load

  # TODO: monitor the overall Disk I/O load

  # mapping cpu status
  ret[:cpu_architecture] = self.ssh.exec!('lscpu | grep Architecture').split(':')[1].strip!
  ret[:cpu_speed] = self.ssh.exec!('lscpu | grep "CPU MHz:"').split(':')[1].strip!.to_f.round
  #ret[:cpu_model] = self.ssh.exec!('lscpu | grep "Model"').split(':')[1].strip!
  #ret[:cpu_type] = ret[:cpu_model].split(' ')[0]
  ret[:cpu_number] = self.ssh.exec!('lscpu | grep "^CPU(s):"').split(':')[1].strip!.to_i

  # mapping disk status
  #self.disk_total = mb_total_disk.to_i
  #self.disk_free = mb_free_disk.to_i

  # mapping lan attributes
  ret[:net_mac_address] = self.ssh.exec!('ifconfig | grep ether').split[1].upcase.strip.gsub(':', '-') 

  #self.disconnect

  ret
end

#using_password?Boolean

return true if the node is all set to connect using ssh user and password.

Returns:

  • (Boolean)


95
96
97
# File 'lib/blackstack-nodes.rb', line 95

def using_password?
  !self.ip.nil? && !self.ssh_username.nil? && !self.ssh_password.nil?
end

#using_private_key_file?Boolean

return true if the node is all set to connect using a private key file.

Returns:

  • (Boolean)


100
101
102
# File 'lib/blackstack-nodes.rb', line 100

def using_private_key_file?
  !self.ip.nil? && !self.ssh_username.nil? && !self.ssh_private_key_file.nil?
end