blob: abb41b7f5f16c75c216fa31e8c29183ee384e9e1 [file] [log] [blame]
#!/usr/bin/env python
r"""
This module has functions to support various data structures such as the
boot_table, valid_boot_list and boot_results_table.
"""
import os
import tempfile
import json
import glob
from tally_sheet import *
from robot.libraries.BuiltIn import BuiltIn
try:
from robot.utils import DotDict
except ImportError:
import collections
import gen_print as gp
import gen_robot_print as grp
import gen_valid as gv
import gen_misc as gm
import gen_cmd as gc
import var_funcs as vf
# The code base directory will be one level up from the directory containing
# this module.
code_base_dir_path = os.path.dirname(os.path.dirname(__file__)) + os.sep
def create_boot_table(file_path=None):
r"""
Read the boot table JSON file, convert it to an object and return it.
Note that if the user is running without a global OS_HOST robot variable
specified, this function will remove all of the "os_" start and end state
requirements from the JSON data.
Description of arguments:
file_path The path to the boot_table file. If this value is not
specified, it will be obtained from the "BOOT_TABLE_PATH"
environment variable, if set. Otherwise, it will default to
"data/boot_table.json". If this value is a relative path,
this function will use the code_base_dir_path as the base
directory (see definition above).
"""
if file_path is None:
file_path = os.environ.get('BOOT_TABLE_PATH', 'data/boot_table.json')
if not file_path.startswith("/"):
file_path = code_base_dir_path + file_path
# Pre-process the file by removing blank lines and comment lines.
temp = tempfile.NamedTemporaryFile()
temp_file_path = temp.name
cmd_buf = "egrep -v '^[ ]*$|^[ ]*#' " + file_path + " > " + temp_file_path
gc.cmd_fnc_u(cmd_buf, quiet=1)
boot_file = open(temp_file_path)
boot_table = json.load(boot_file, object_hook=DotDict)
# If the user is running without an OS_HOST, we remove os starting and
# ending state requirements from the boot entries.
os_host = BuiltIn().get_variable_value("${OS_HOST}", default="")
if os_host == "":
for boot in boot_table:
state_keys = ['start', 'end']
for state_key in state_keys:
for sub_state in list(boot_table[boot][state_key]):
if sub_state.startswith("os_"):
boot_table[boot][state_key].pop(sub_state, None)
# For every boot_type we should have a corresponding mfg mode boot type.
enhanced_boot_table = DotDict()
for key, value in boot_table.items():
enhanced_boot_table[key] = value
enhanced_boot_table[key + " (mfg)"] = value
return enhanced_boot_table
def create_valid_boot_list(boot_table):
r"""
Return a list of all of the valid boot types (e.g. ['BMC Power On',
'BMC Power Off', ....]
Description of arguments:
boot_table A boot table such as is returned by the create_boot_table
function.
"""
return list(boot_table.keys())
def read_boot_lists(dir_path="data/boot_lists/"):
r"""
Read the contents of all the boot lists files found in the given boot lists
directory and return dictionary of the lists.
Boot lists are simply files containing a boot test name on each line.
These files are useful for categorizing and organizing boot tests. For
example, there may be a "Power_on" list, a "Power_off" list, etc.
The names of the boot list files will be the keys to the top level
dictionary. Each dictionary entry is a list of all the boot tests found
in the corresponding file.
Here is an abbreviated look at the resulting boot_lists dictionary.
boot_lists:
boot_lists[All]:
boot_lists[All][0]: BMC Power On
boot_lists[All][1]: BMC Power Off
...
boot_lists[Code_update]:
boot_lists[Code_update][0]: BMC oob hpm
boot_lists[Code_update][1]: BMC ib hpm
...
Description of arguments:
dir_path The path to the directory containing the boot list files. If
this value is a relative path, this function will use the
code_base_dir_path as the base directory (see definition above).
"""
if not dir_path.startswith("/"):
# Dir path is relative.
dir_path = code_base_dir_path + dir_path
# Get a list of all file names in the directory.
boot_file_names = os.listdir(dir_path)
boot_lists = DotDict()
for boot_category in boot_file_names:
file_path = gm.which(dir_path + boot_category)
boot_list = gm.file_to_list(file_path, newlines=0, comments=0, trim=1)
boot_lists[boot_category] = boot_list
return boot_lists
def valid_boot_list(boot_list,
valid_boot_types):
r"""
Verify that each entry in boot_list is a supported boot test.
Description of arguments:
boot_list An array (i.e. list) of boot test types
(e.g. "BMC Power On").
valid_boot_types A list of valid boot types such as that returned by
create_valid_boot_list.
"""
for boot_name in boot_list:
boot_name = boot_name.strip(" ")
error_message = gv.svalid_value(boot_name,
valid_values=valid_boot_types,
var_name="boot_name")
if error_message != "":
BuiltIn().fail(gp.sprint_error(error_message))
class boot_results:
r"""
This class defines a boot_results table.
"""
def __init__(self,
boot_table,
boot_pass=0,
boot_fail=0,
obj_name='boot_results'):
r"""
Initialize the boot results object.
Description of arguments:
boot_table Boot table object (see definition above). The boot table
contains all of the valid boot test types. It can be
created with the create_boot_table function.
boot_pass An initial boot_pass value. This program may be called
as part of a larger test suite. As such there may already
have been some successful boot tests that we need to
keep track of.
boot_fail An initial boot_fail value. This program may be called
as part of a larger test suite. As such there may already
have been some unsuccessful boot tests that we need to
keep track of.
obj_name The name of this object.
"""
# Store the method parms as class data.
self.__obj_name = obj_name
self.__initial_boot_pass = boot_pass
self.__initial_boot_fail = boot_fail
# Create boot_results_fields for use in creating boot_results table.
boot_results_fields = DotDict([('total', 0), ('pass', 0), ('fail', 0)])
# Create boot_results table.
self.__boot_results = tally_sheet('boot type',
boot_results_fields,
'boot_test_results')
self.__boot_results.set_sum_fields(['total', 'pass', 'fail'])
self.__boot_results.set_calc_fields(['total=pass+fail'])
# Create one row in the result table for each kind of boot test
# in the boot_table (i.e. for all supported boot tests).
for boot_name in list(boot_table.keys()):
self.__boot_results.add_row(boot_name)
def return_total_pass_fail(self):
r"""
Return the total boot_pass and boot_fail values. This information is
comprised of the pass/fail values from the table plus the initial
pass/fail values.
"""
totals_line = self.__boot_results.calc()
return totals_line['pass'] + self.__initial_boot_pass,\
totals_line['fail'] + self.__initial_boot_fail
def update(self,
boot_type,
boot_status):
r"""
Update our boot_results_table. This includes:
- Updating the record for the given boot_type by incrementing the pass
or fail field.
- Calling the calc method to have the totals calculated.
Description of arguments:
boot_type The type of boot test just done (e.g. "BMC Power On").
boot_status The status of the boot just done. This should be equal to
either "pass" or "fail" (case-insensitive).
"""
self.__boot_results.inc_row_field(boot_type, boot_status.lower())
self.__boot_results.calc()
def sprint_report(self,
header_footer="\n"):
r"""
String-print the formatted boot_resuls_table and return them.
Description of arguments:
header_footer This indicates whether a header and footer are to be
included in the report.
"""
buffer = ""
buffer += gp.sprint(header_footer)
buffer += self.__boot_results.sprint_report()
buffer += gp.sprint(header_footer)
return buffer
def print_report(self,
header_footer="\n"):
r"""
Print the formatted boot_resuls_table to the console.
See sprint_report for details.
"""
grp.rqprint(self.sprint_report(header_footer))
def sprint_obj(self):
r"""
sprint the fields of this object. This would normally be for debug
purposes only.
"""
buffer = ""
buffer += "class name: " + self.__class__.__name__ + "\n"
buffer += gp.sprint_var(self.__obj_name)
buffer += self.__boot_results.sprint_obj()
buffer += gp.sprint_var(self.__initial_boot_pass)
buffer += gp.sprint_var(self.__initial_boot_fail)
return buffer
def print_obj(self):
r"""
Print the fields of this object to stdout. This would normally be for
debug purposes.
"""
grp.rprint(self.sprint_obj())
def create_boot_results_file_path(pgm_name,
openbmc_nickname,
master_pid):
r"""
Create a file path to be used to store a boot_results object.
Description of argument(s):
pgm_name The name of the program. This will form part of the
resulting file name.
openbmc_nickname The name of the system. This could be a nickname, a
hostname, an IP, etc. This will form part of the
resulting file name.
master_pid The master process id which will form part of the file
name.
"""
USER = os.environ.get("USER", "")
dir_path = "/tmp/" + USER + "/"
if not os.path.exists(dir_path):
os.makedirs(dir_path)
file_name_dict = vf.create_var_dict(pgm_name, openbmc_nickname, master_pid)
return vf.create_file_path(file_name_dict, dir_path=dir_path,
file_suffix=":boot_results")
def cleanup_boot_results_file():
r"""
Delete all boot results files whose corresponding pids are no longer
active.
"""
# Use create_boot_results_file_path to create a globex to find all of the
# existing boot results files.
globex = create_boot_results_file_path("*", "*", "*")
file_list = sorted(glob.glob(globex))
for file_path in file_list:
# Use parse_file_path to extract info from the file path.
file_dict = vf.parse_file_path(file_path)
if gm.pid_active(file_dict['master_pid']):
gp.qprint_timen("Preserving " + file_path + ".")
else:
gc.cmd_fnc("rm -f " + file_path)