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

#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

#net_remote_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 net_remote_ip
  @net_remote_ip
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 :net_remote_ip
  errors << "The parameter h does not have a key :net_remote_ip" unless h.has_key?(:net_remote_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



104
105
106
107
108
109
110
111
112
113
114
# File 'lib/blackstack-nodes.rb', line 104

def connect
  # connect
  if self.using_password?
    self.ssh = Net::SSH.start(self.net_remote_ip, self.ssh_username, :password => self.ssh_password, :port => self.ssh_port)
  elsif self.using_private_key_file?
    self.ssh = Net::SSH.start(self.net_remote_ip, self.ssh_username, :keys => self.ssh_private_key_file, :port => self.ssh_port)
  else
    raise "No ssh credentials available"
  end
  self.ssh
end

#disconnectObject

def connect



116
117
118
# File 'lib/blackstack-nodes.rb', line 116

def disconnect
  self.ssh.close
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:



129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
# File 'lib/blackstack-nodes.rb', line 129

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.net_remote_ip = h[:net_remote_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



161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
# File 'lib/blackstack-nodes.rb', line 161

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



250
251
252
# File 'lib/blackstack-nodes.rb', line 250

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,
    :net_remote_ip => self.net_remote_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



198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
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
# File 'lib/blackstack-nodes.rb', line 198

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.net_remote_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.net_remote_ip.nil? && !self.ssh_username.nil? && !self.ssh_private_key_file.nil?
end