|
import re
|
|
|
|
def process_template(input_text):
|
|
"""
|
|
Process a text template with macro instructions and variable substitution.
|
|
Supports multiple values for variables to generate multiple output versions.
|
|
Each section between macro lines is treated as a separate template.
|
|
|
|
Args:
|
|
input_text (str): The input template text
|
|
|
|
Returns:
|
|
tuple: (output_text, error_message)
|
|
- output_text: Processed output with variables substituted, or empty string if error
|
|
- error_message: Error description and problematic line, or empty string if no error
|
|
"""
|
|
lines = input_text.strip().split('\n')
|
|
current_variables = {}
|
|
current_template_lines = []
|
|
all_output_lines = []
|
|
error_message = ""
|
|
|
|
|
|
line_number = 0
|
|
while line_number < len(lines):
|
|
orig_line = lines[line_number]
|
|
line = orig_line.strip()
|
|
line_number += 1
|
|
|
|
|
|
if not line or line.startswith('#'):
|
|
continue
|
|
|
|
|
|
if line.startswith('!'):
|
|
|
|
if current_template_lines:
|
|
|
|
template_output, err = process_current_template(current_template_lines, current_variables)
|
|
if err:
|
|
return "", err
|
|
all_output_lines.extend(template_output)
|
|
current_template_lines = []
|
|
|
|
|
|
current_variables = {}
|
|
|
|
|
|
macro_line = line[1:].strip()
|
|
|
|
|
|
open_braces = macro_line.count('{')
|
|
close_braces = macro_line.count('}')
|
|
if open_braces != close_braces:
|
|
error_message = f"Unmatched braces: {open_braces} opening '{{' and {close_braces} closing '}}' braces\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
if macro_line.count('"') % 2 != 0:
|
|
error_message = f"Unclosed double quotes\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
var_sections = re.split(r'\s*:\s*', macro_line)
|
|
|
|
for section in var_sections:
|
|
section = section.strip()
|
|
if not section:
|
|
continue
|
|
|
|
|
|
var_match = re.search(r'\{([^}]+)\}', section)
|
|
if not var_match:
|
|
if '{' in section or '}' in section:
|
|
error_message = f"Malformed variable declaration\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
continue
|
|
|
|
var_name = var_match.group(1).strip()
|
|
if not var_name:
|
|
error_message = f"Empty variable name\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
value_part = section[section.find('}')+1:].strip()
|
|
if not value_part.startswith('='):
|
|
error_message = f"Missing '=' after variable '{{{var_name}}}'\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
var_values = re.findall(r'"([^"]*)"', value_part)
|
|
|
|
|
|
if not var_values:
|
|
error_message = f"No quoted values found for variable '{{{var_name}}}'\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
|
|
if re.search(r'"[^,]*"[^,]*"', value_part):
|
|
error_message = f"Missing comma between values for variable '{{{var_name}}}'\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
current_variables[var_name] = var_values
|
|
|
|
|
|
else:
|
|
|
|
var_references = re.findall(r'\{([^}]+)\}', line)
|
|
for var_ref in var_references:
|
|
if var_ref not in current_variables:
|
|
error_message = f"Unknown variable '{{{var_ref}}}' in template\nLine: '{orig_line}'"
|
|
return "", error_message
|
|
|
|
|
|
current_template_lines.append(line)
|
|
|
|
|
|
if current_template_lines:
|
|
template_output, err = process_current_template(current_template_lines, current_variables)
|
|
if err:
|
|
return "", err
|
|
all_output_lines.extend(template_output)
|
|
|
|
return '\n'.join(all_output_lines), ""
|
|
|
|
def process_current_template(template_lines, variables):
|
|
"""
|
|
Process a set of template lines with the current variables.
|
|
|
|
Args:
|
|
template_lines (list): List of template lines to process
|
|
variables (dict): Dictionary of variable names to lists of values
|
|
|
|
Returns:
|
|
tuple: (output_lines, error_message)
|
|
"""
|
|
if not variables or not template_lines:
|
|
return template_lines, ""
|
|
|
|
output_lines = []
|
|
|
|
|
|
max_values = max(len(values) for values in variables.values())
|
|
|
|
|
|
for i in range(max_values):
|
|
for template in template_lines:
|
|
output_line = template
|
|
for var_name, var_values in variables.items():
|
|
|
|
value_index = i % len(var_values)
|
|
var_value = var_values[value_index]
|
|
output_line = output_line.replace(f"{{{var_name}}}", var_value)
|
|
output_lines.append(output_line)
|
|
|
|
return output_lines, ""
|
|
|
|
|
|
def extract_variable_names(macro_line):
|
|
"""
|
|
Extract all variable names from a macro line.
|
|
|
|
Args:
|
|
macro_line (str): A macro line (with or without the leading '!')
|
|
|
|
Returns:
|
|
tuple: (variable_names, error_message)
|
|
- variable_names: List of variable names found in the macro
|
|
- error_message: Error description if any, empty string if no error
|
|
"""
|
|
|
|
if macro_line.startswith('!'):
|
|
macro_line = macro_line[1:].strip()
|
|
|
|
variable_names = []
|
|
|
|
|
|
open_braces = macro_line.count('{')
|
|
close_braces = macro_line.count('}')
|
|
if open_braces != close_braces:
|
|
return [], f"Unmatched braces: {open_braces} opening '{{' and {close_braces} closing '}}' braces"
|
|
|
|
|
|
var_sections = re.split(r'\s*:\s*', macro_line)
|
|
|
|
for section in var_sections:
|
|
section = section.strip()
|
|
if not section:
|
|
continue
|
|
|
|
|
|
var_matches = re.findall(r'\{([^}]+)\}', section)
|
|
for var_name in var_matches:
|
|
new_var = var_name.strip()
|
|
if not new_var in variable_names:
|
|
variable_names.append(new_var)
|
|
|
|
return variable_names, ""
|
|
|
|
def extract_variable_values(macro_line):
|
|
"""
|
|
Extract all variable names and their values from a macro line.
|
|
|
|
Args:
|
|
macro_line (str): A macro line (with or without the leading '!')
|
|
|
|
Returns:
|
|
tuple: (variables_dict, error_message)
|
|
- variables_dict: Dictionary mapping variable names to their values
|
|
- error_message: Error description if any, empty string if no error
|
|
"""
|
|
|
|
if macro_line.startswith('!'):
|
|
macro_line = macro_line[1:].strip()
|
|
|
|
variables = {}
|
|
|
|
|
|
open_braces = macro_line.count('{')
|
|
close_braces = macro_line.count('}')
|
|
if open_braces != close_braces:
|
|
return {}, f"Unmatched braces: {open_braces} opening '{{' and {close_braces} closing '}}' braces"
|
|
|
|
|
|
if macro_line.count('"') % 2 != 0:
|
|
return {}, "Unclosed double quotes"
|
|
|
|
|
|
var_sections = re.split(r'\s*:\s*', macro_line)
|
|
|
|
for section in var_sections:
|
|
section = section.strip()
|
|
if not section:
|
|
continue
|
|
|
|
|
|
var_match = re.search(r'\{([^}]+)\}', section)
|
|
if not var_match:
|
|
if '{' in section or '}' in section:
|
|
return {}, "Malformed variable declaration"
|
|
continue
|
|
|
|
var_name = var_match.group(1).strip()
|
|
if not var_name:
|
|
return {}, "Empty variable name"
|
|
|
|
|
|
value_part = section[section.find('}')+1:].strip()
|
|
if not value_part.startswith('='):
|
|
return {}, f"Missing '=' after variable '{{{var_name}}}'"
|
|
|
|
|
|
var_values = re.findall(r'"([^"]*)"', value_part)
|
|
|
|
|
|
if not var_values:
|
|
return {}, f"No quoted values found for variable '{{{var_name}}}'"
|
|
|
|
|
|
if re.search(r'"[^,]*"[^,]*"', value_part):
|
|
return {}, f"Missing comma between values for variable '{{{var_name}}}'"
|
|
|
|
variables[var_name] = var_values
|
|
|
|
return variables, ""
|
|
|
|
def generate_macro_line(variables_dict):
|
|
"""
|
|
Generate a macro line from a dictionary of variable names and their values.
|
|
|
|
Args:
|
|
variables_dict (dict): Dictionary mapping variable names to lists of values
|
|
|
|
Returns:
|
|
str: A formatted macro line (including the leading '!')
|
|
"""
|
|
sections = []
|
|
|
|
for var_name, values in variables_dict.items():
|
|
|
|
quoted_values = [f'"{value}"' for value in values]
|
|
|
|
values_str = ','.join(quoted_values)
|
|
|
|
section = f"{{{var_name}}}={values_str}"
|
|
sections.append(section)
|
|
|
|
|
|
return "! " + " : ".join(sections) |