| 1 |  |  | """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 2 |  |  | BasicNeeds - useful functions library | 
            
                                                                                                            
                            
            
                                    
            
            
                | 3 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 4 |  |  | This library has functions useful to keep main logic short and simple | 
            
                                                                                                            
                            
            
                                    
            
            
                | 5 |  |  | """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 6 |  |  | # package to handle date and times | 
            
                                                                                                            
                            
            
                                    
            
            
                | 7 |  |  | from datetime import datetime, timedelta | 
            
                                                                                                            
                            
            
                                    
            
            
                | 8 |  |  | # package to add support for multi-language (i18n) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 9 |  |  | import gettext | 
            
                                                                                                            
                            
            
                                    
            
            
                | 10 |  |  | # package to handle files/folders and related metadata/operations | 
            
                                                                                                            
                            
            
                                    
            
            
                | 11 |  |  | import os | 
            
                                                                                                            
                            
            
                                    
            
            
                | 12 |  |  | # package regular expressions | 
            
                                                                                                            
                            
            
                                    
            
            
                | 13 |  |  | import re | 
            
                                                                                                            
                            
            
                                    
            
            
                | 14 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 15 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 16 |  |  | class BasicNeeds: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 17 |  |  |     locale = None | 
            
                                                                                                            
                            
            
                                    
            
            
                | 18 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 19 |  |  |     def __init__(self, in_language='en_US'): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 20 |  |  |         current_script = os.path.basename(__file__).replace('.py', '') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 21 |  |  |         lang_folder = os.path.join(os.path.dirname(__file__), current_script + '_Locale') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 22 |  |  |         self.locale = gettext.translation(current_script, lang_folder, languages=[in_language]) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 23 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 24 |  |  |     def fn_add_value_to_dictionary(self, in_list, adding_value, adding_type, reference_column): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 25 |  |  |         add_type = adding_type.lower() | 
            
                                                                                                            
                            
            
                                    
            
            
                | 26 |  |  |         total_columns = len(in_list.copy()) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 27 |  |  |         reference_indexes = { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 28 |  |  |             'add': {'after': 0, 'before': 0}, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 29 |  |  |             'cycle_down_to': {'after': 0, 'before': 0} | 
            
                                                                                                            
                            
            
                                    
            
            
                | 30 |  |  |         } | 
            
                                                                                                            
                            
            
                                    
            
            
                | 31 |  |  |         if reference_column is not None: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 32 |  |  |             reference_indexes = { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 33 |  |  |                 'add': { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 34 |  |  |                     'after': in_list.copy().index(reference_column) + 1, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 35 |  |  |                     'before': in_list.copy().index(reference_column), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 36 |  |  |                 }, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 37 |  |  |                 'cycle_down_to': { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 38 |  |  |                     'after': in_list.copy().index(reference_column), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 39 |  |  |                     'before': in_list.copy().index(reference_column), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 40 |  |  |                 } | 
            
                                                                                                            
                            
            
                                    
            
            
                | 41 |  |  |             } | 
            
                                                                                                            
                            
            
                                    
            
            
                | 42 |  |  |         positions = { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 43 |  |  |             'after': { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 44 |  |  |                 'cycle_down_to': reference_indexes.get('cycle_down_to').get('after'), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 45 |  |  |                 'add': reference_indexes.get('add').get('after'), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 46 |  |  |             }, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 47 |  |  |             'before': { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 48 |  |  |                 'cycle_down_to': reference_indexes.get('cycle_down_to').get('before'), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 49 |  |  |                 'add': reference_indexes.get('add').get('before'), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 50 |  |  |             }, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 51 |  |  |             'first': { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 52 |  |  |                 'cycle_down_to': 0, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 53 |  |  |                 'add': 0, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 54 |  |  |             }, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 55 |  |  |             'last': { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 56 |  |  |                 'cycle_down_to': total_columns, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 57 |  |  |                 'add': total_columns, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 58 |  |  |             } | 
            
                                                                                                            
                            
            
                                    
            
            
                | 59 |  |  |         } | 
            
                                                                                                            
                            
            
                                    
            
            
                | 60 |  |  |         return self.add_value_to_dictionary_by_position({ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 61 |  |  |             'adding_value': adding_value, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 62 |  |  |             'list': in_list.copy(), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 63 |  |  |             'position_to_add': positions.get(add_type).get('add'), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 64 |  |  |             'position_to_cycle_down_to': positions.get(add_type).get('cycle_down_to'), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 65 |  |  |             'total_columns': total_columns, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 66 |  |  |         }) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 67 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 68 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 69 |  |  |     def add_value_to_dictionary_by_position(adding_dictionary): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 70 |  |  |         list_with_values = adding_dictionary['list'] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 71 |  |  |         list_with_values.append(adding_dictionary['total_columns']) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 72 |  |  |         for counter in range(adding_dictionary['total_columns'], | 
            
                                                                                                            
                            
            
                                    
            
            
                | 73 |  |  |                              adding_dictionary['position_to_cycle_down_to'], -1): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 74 |  |  |             list_with_values[counter] = list_with_values[(counter - 1)] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 75 |  |  |         list_with_values[adding_dictionary['position_to_add']] = adding_dictionary['adding_value'] | 
            
                                                                                                            
                            
            
                                    
            
            
                | 76 |  |  |         return list_with_values | 
            
                                                                                                            
                            
            
                                    
            
            
                | 77 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 78 |  |  |     def fn_check_inputs(self, input_parameters): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 79 |  |  |         if input_parameters.output_log_file is not None: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 80 |  |  |             # checking log folder first as there's all further messages will be stored | 
            
                                                                                                            
                            
            
                                    
            
            
                | 81 |  |  |             self.fn_validate_single_value( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 82 |  |  |                     os.path.dirname(input_parameters.output_log_file), 'folder') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 83 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 84 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 85 |  |  |     def fn_decide_by_omission_or_specific_true(in_dictionary, key_decision_factor): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 86 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 87 |  |  |         Evaluates if a property is specified in a Dict structure | 
            
                                                                                                            
                            
            
                                    
            
            
                | 88 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 89 |  |  |         @param in_dictionary: input Dict structure | 
            
                                                                                                            
                            
            
                                    
            
            
                | 90 |  |  |         @param key_decision_factor: key used to search value in Dict structure | 
            
                                                                                                            
                            
            
                                    
            
            
                | 91 |  |  |         @return: True|False | 
            
                                                                                                            
                            
            
                                    
            
            
                | 92 |  |  |         """ | 
            
                                                                                                            
                            
            
                                    
            
            
                | 93 |  |  |         final_decision = False | 
            
                                                                                                            
                            
            
                                    
            
            
                | 94 |  |  |         if key_decision_factor in in_dictionary: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 95 |  |  |             final_decision = True | 
            
                                                                                                            
                            
            
                                    
            
            
                | 96 |  |  |         elif in_dictionary[key_decision_factor]: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 97 |  |  |             final_decision = True | 
            
                                                                                                            
                            
            
                                    
            
            
                | 98 |  |  |         return final_decision | 
            
                                                                                                            
                            
            
                                    
            
            
                | 99 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 100 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 101 |  |  |     def fn_evaluate_dict_values(in_dict): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 102 |  |  |         true_counted = 0 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 103 |  |  |         for crt_value in in_dict: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 104 |  |  |             if in_dict[crt_value]: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 105 |  |  |                 true_counted += 1 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 106 |  |  |         all_true = False | 
            
                                                                                                            
                            
            
                                    
            
            
                | 107 |  |  |         if true_counted == len(in_dict): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 108 |  |  |             all_true = True | 
            
                                                                                                            
                            
            
                                    
            
            
                | 109 |  |  |         return all_true | 
            
                                                                                                            
                            
            
                                    
            
            
                | 110 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 111 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 112 |  |  |     def fn_evaluate_list_values(in_list): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 113 |  |  |         true_counted = 0 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 114 |  |  |         for crt_value in in_list: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 115 |  |  |             if crt_value: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 116 |  |  |                 true_counted += 1 | 
            
                                                                                                            
                            
            
                                    
            
            
                | 117 |  |  |         all_true = False | 
            
                                                                                                            
                            
            
                                    
            
            
                | 118 |  |  |         if true_counted == len(in_list): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 119 |  |  |             all_true = True | 
            
                                                                                                            
                            
            
                                    
            
            
                | 120 |  |  |         return all_true | 
            
                                                                                                            
                            
            
                                    
            
            
                | 121 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 122 |  |  |     def fn_final_message(self, local_logger, log_file_name, performance_in_seconds): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 123 |  |  |         total_time_string = str(timedelta(seconds=performance_in_seconds)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 124 |  |  |         if log_file_name == 'None': | 
            
                                                                                                            
                            
            
                                    
            
            
                | 125 |  |  |             self.fn_timestamped_print(self.locale.gettext( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 126 |  |  |                 'Application finished, whole script took {total_time_string}') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 127 |  |  |                                       .replace('{total_time_string}', total_time_string)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 128 |  |  |         else: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 129 |  |  |             local_logger.info(self.locale.gettext('Total execution time was {total_time_string}') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 130 |  |  |                               .replace('{total_time_string}', total_time_string)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 131 |  |  |             self.fn_timestamped_print(self.locale.gettext( | 
            
                                                                                                            
                            
            
                                    
            
            
                | 132 |  |  |                 'Application finished, for complete logged details please check {log_file_name}') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 133 |  |  |                                       .replace('{log_file_name}', log_file_name)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 134 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 135 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 136 |  |  |     def fn_multi_line_string_to_single(input_string): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 137 |  |  |         string_to_return = input_string.replace('\n', ' ').replace('\r', ' ') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 138 |  |  |         return re.sub(r'\s{2,100}', ' ', string_to_return).replace(' , ', ', ').strip() | 
            
                                                                                                            
                            
            
                                    
            
            
                | 139 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 140 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 141 |  |  |     def fn_numbers_with_leading_zero(input_number_as_string, digits): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 142 |  |  |         final_number = input_number_as_string | 
            
                                                                                                            
                            
            
                                    
            
            
                | 143 |  |  |         if len(input_number_as_string) < digits: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 144 |  |  |             final_number = '0' * (digits - len(input_number_as_string)) + input_number_as_string | 
            
                                                                                                            
                            
            
                                    
            
            
                | 145 |  |  |         return final_number | 
            
                                                                                                            
                                                                
            
                                    
            
            
                | 146 |  |  |  | 
            
                                                                        
                            
            
                                    
            
            
                | 147 |  |  |     def fn_optional_print(self, boolean_variable, string_to_print): | 
            
                                                                        
                            
            
                                    
            
            
                | 148 |  |  |         if boolean_variable: | 
            
                                                                        
                            
            
                                    
            
            
                | 149 |  |  |             self.fn_timestamped_print(string_to_print) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 150 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 151 |  |  |     @staticmethod | 
            
                                                                                                            
                            
            
                                    
            
            
                | 152 |  |  |     def fn_timestamped_print(string_to_print): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 153 |  |  |         print(datetime.utcnow().strftime("%Y-%m-%d %H:%M:%S.%f %Z") + '- ' + string_to_print) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 154 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 155 |  |  |     def fn_validate_one_value(self, value_to_validate, validation_type): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 156 |  |  |         is_invalid = False | 
            
                                                                                                            
                            
            
                                    
            
            
                | 157 |  |  |         message = '' | 
            
                                                                                                            
                            
            
                                    
            
            
                | 158 |  |  |         if validation_type == 'file': | 
            
                                                                                                            
                            
            
                                    
            
            
                | 159 |  |  |             is_invalid = (not os.path.isfile(value_to_validate)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 160 |  |  |             message = self.locale.gettext('Given file "{value_to_validate}" does not exist') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 161 |  |  |         elif validation_type == 'folder': | 
            
                                                                                                            
                            
            
                                    
            
            
                | 162 |  |  |             is_invalid = (not os.path.isdir(value_to_validate)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 163 |  |  |             message = self.locale.gettext('Given folder "{value_to_validate}" does not exist') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 164 |  |  |         elif validation_type == 'url': | 
            
                                                                                                            
                            
            
                                    
            
            
                | 165 |  |  |             url_reg_expression = 'https?://(?:www)?(?:[\\w-]{2,255}(?:\\.\\w{2,66}){1,2})' | 
            
                                                                                                            
                            
            
                                    
            
            
                | 166 |  |  |             is_invalid = (not re.match(url_reg_expression, value_to_validate)) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 167 |  |  |             message = self.locale.gettext('Given url "{value_to_validate}" is not valid') | 
            
                                                                                                            
                            
            
                                    
            
            
                | 168 |  |  |         return { | 
            
                                                                                                            
                            
            
                                    
            
            
                | 169 |  |  |             'is_invalid': is_invalid, | 
            
                                                                                                            
                            
            
                                    
            
            
                | 170 |  |  |             'message': message.replace('{value_to_validate}', value_to_validate), | 
            
                                                                                                            
                            
            
                                    
            
            
                | 171 |  |  |         } | 
            
                                                                                                            
                            
            
                                    
            
            
                | 172 |  |  |  | 
            
                                                                                                            
                            
            
                                    
            
            
                | 173 |  |  |     def fn_validate_single_value(self, value_to_validate, validation_type): | 
            
                                                                                                            
                            
            
                                    
            
            
                | 174 |  |  |         validation_details = self.fn_validate_one_value(value_to_validate, validation_type) | 
            
                                                                                                            
                            
            
                                    
            
            
                | 175 |  |  |         if validation_details['is_invalid']: | 
            
                                                                                                            
                            
            
                                    
            
            
                | 176 |  |  |             self.fn_timestamped_print(validation_details['message']) | 
            
                                                                                                            
                                                                
            
                                    
            
            
                | 177 |  |  |             exit(1) | 
            
                                                        
            
                                    
            
            
                | 178 |  |  |  |