mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-22 03:33:52 +01:00 
			
		
		
		
	template all the things
This commit is contained in:
		| @@ -14,6 +14,8 @@ from pathlib import Path | |||||||
| import subprocess | import subprocess | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
|  | from jinja2 import Environment, FileSystemLoader | ||||||
|  |  | ||||||
| # Add esphome to path for analyze_memory import | # Add esphome to path for analyze_memory import | ||||||
| sys.path.insert(0, str(Path(__file__).parent.parent)) | sys.path.insert(0, str(Path(__file__).parent.parent)) | ||||||
|  |  | ||||||
| @@ -26,6 +28,22 @@ COMMENT_MARKER = "<!-- esphome-memory-impact-analysis -->" | |||||||
| OVERALL_CHANGE_THRESHOLD = 1.0  # Overall RAM/Flash changes | OVERALL_CHANGE_THRESHOLD = 1.0  # Overall RAM/Flash changes | ||||||
| COMPONENT_CHANGE_THRESHOLD = 3.0  # Component breakdown changes | COMPONENT_CHANGE_THRESHOLD = 3.0  # Component breakdown changes | ||||||
|  |  | ||||||
|  | # Display limits for tables | ||||||
|  | MAX_COMPONENT_BREAKDOWN_ROWS = 20  # Maximum components to show in breakdown table | ||||||
|  | MAX_CHANGED_SYMBOLS_ROWS = 30  # Maximum changed symbols to show | ||||||
|  | MAX_NEW_SYMBOLS_ROWS = 15  # Maximum new symbols to show | ||||||
|  | MAX_REMOVED_SYMBOLS_ROWS = 15  # Maximum removed symbols to show | ||||||
|  |  | ||||||
|  | # Symbol display formatting | ||||||
|  | SYMBOL_DISPLAY_MAX_LENGTH = 100  # Max length before using <details> tag | ||||||
|  | SYMBOL_DISPLAY_TRUNCATE_LENGTH = 97  # Length to truncate in summary | ||||||
|  |  | ||||||
|  | # Component change noise threshold | ||||||
|  | COMPONENT_CHANGE_NOISE_THRESHOLD = 2  # Ignore component changes ≤ this many bytes | ||||||
|  |  | ||||||
|  | # Template directory | ||||||
|  | TEMPLATE_DIR = Path(__file__).parent / "templates" | ||||||
|  |  | ||||||
|  |  | ||||||
| def load_analysis_json(json_path: str) -> dict | None: | def load_analysis_json(json_path: str) -> dict | None: | ||||||
|     """Load memory analysis results from JSON file. |     """Load memory analysis results from JSON file. | ||||||
| @@ -111,35 +129,20 @@ def format_change(before: int, after: int, threshold: float | None = None) -> st | |||||||
|     return f"{emoji} {delta_str} ({pct_str})" |     return f"{emoji} {delta_str} ({pct_str})" | ||||||
|  |  | ||||||
|  |  | ||||||
| def format_symbol_for_display(symbol: str) -> str: | def prepare_symbol_changes_data( | ||||||
|     """Format a symbol name for display in markdown table. |  | ||||||
|  |  | ||||||
|     Args: |  | ||||||
|         symbol: Symbol name to format |  | ||||||
|  |  | ||||||
|     Returns: |  | ||||||
|         Formatted symbol with backticks or HTML details tag for long names |  | ||||||
|     """ |  | ||||||
|     if len(symbol) <= 100: |  | ||||||
|         return f"`{symbol}`" |  | ||||||
|     # Use HTML details for very long symbols (no backticks inside HTML) |  | ||||||
|     return f"<details><summary><code>{symbol[:97]}...</code></summary><code>{symbol}</code></details>" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def create_symbol_changes_table( |  | ||||||
|     target_symbols: dict | None, pr_symbols: dict | None |     target_symbols: dict | None, pr_symbols: dict | None | ||||||
| ) -> str: | ) -> dict | None: | ||||||
|     """Create a markdown table showing symbols that changed size. |     """Prepare symbol changes data for template rendering. | ||||||
|  |  | ||||||
|     Args: |     Args: | ||||||
|         target_symbols: Symbol name to size mapping for target branch |         target_symbols: Symbol name to size mapping for target branch | ||||||
|         pr_symbols: Symbol name to size mapping for PR branch |         pr_symbols: Symbol name to size mapping for PR branch | ||||||
|  |  | ||||||
|     Returns: |     Returns: | ||||||
|         Formatted markdown table |         Dictionary with changed, new, and removed symbols, or None if no changes | ||||||
|     """ |     """ | ||||||
|     if not target_symbols or not pr_symbols: |     if not target_symbols or not pr_symbols: | ||||||
|         return "" |         return None | ||||||
|  |  | ||||||
|     # Find all symbols that exist in both branches or only in one |     # Find all symbols that exist in both branches or only in one | ||||||
|     all_symbols = set(target_symbols.keys()) | set(pr_symbols.keys()) |     all_symbols = set(target_symbols.keys()) | set(pr_symbols.keys()) | ||||||
| @@ -165,113 +168,39 @@ def create_symbol_changes_table( | |||||||
|             changed_symbols.append((symbol, target_size, pr_size, delta)) |             changed_symbols.append((symbol, target_size, pr_size, delta)) | ||||||
|  |  | ||||||
|     if not changed_symbols and not new_symbols and not removed_symbols: |     if not changed_symbols and not new_symbols and not removed_symbols: | ||||||
|         return "" |         return None | ||||||
|  |  | ||||||
|     lines = [ |     # Sort by size/delta | ||||||
|         "", |     changed_symbols.sort(key=lambda x: abs(x[3]), reverse=True) | ||||||
|         "<details>", |     new_symbols.sort(key=lambda x: x[1], reverse=True) | ||||||
|         "<summary>🔍 Symbol-Level Changes (click to expand)</summary>", |     removed_symbols.sort(key=lambda x: x[1], reverse=True) | ||||||
|         "", |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     # Show changed symbols (sorted by absolute delta) |     return { | ||||||
|     if changed_symbols: |         "changed_symbols": changed_symbols, | ||||||
|         changed_symbols.sort(key=lambda x: abs(x[3]), reverse=True) |         "new_symbols": new_symbols, | ||||||
|         lines.extend( |         "removed_symbols": removed_symbols, | ||||||
|             [ |     } | ||||||
|                 "### Changed Symbols", |  | ||||||
|                 "", |  | ||||||
|                 "| Symbol | Target Size | PR Size | Change |", |  | ||||||
|                 "|--------|-------------|---------|--------|", |  | ||||||
|             ] |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         # Show top 30 changes |  | ||||||
|         for symbol, target_size, pr_size, delta in changed_symbols[:30]: |  | ||||||
|             target_str = format_bytes(target_size) |  | ||||||
|             pr_str = format_bytes(pr_size) |  | ||||||
|             change_str = format_change(target_size, pr_size)  # Chart icons only |  | ||||||
|             display_symbol = format_symbol_for_display(symbol) |  | ||||||
|             lines.append( |  | ||||||
|                 f"| {display_symbol} | {target_str} | {pr_str} | {change_str} |" |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         if len(changed_symbols) > 30: |  | ||||||
|             lines.append( |  | ||||||
|                 f"| ... | ... | ... | *({len(changed_symbols) - 30} more changed symbols not shown)* |" |  | ||||||
|             ) |  | ||||||
|         lines.append("") |  | ||||||
|  |  | ||||||
|     # Show new symbols |  | ||||||
|     if new_symbols: |  | ||||||
|         new_symbols.sort(key=lambda x: x[1], reverse=True) |  | ||||||
|         lines.extend( |  | ||||||
|             [ |  | ||||||
|                 "### New Symbols (top 15)", |  | ||||||
|                 "", |  | ||||||
|                 "| Symbol | Size |", |  | ||||||
|                 "|--------|------|", |  | ||||||
|             ] |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         for symbol, size in new_symbols[:15]: |  | ||||||
|             display_symbol = format_symbol_for_display(symbol) |  | ||||||
|             lines.append(f"| {display_symbol} | {format_bytes(size)} |") |  | ||||||
|  |  | ||||||
|         if len(new_symbols) > 15: |  | ||||||
|             total_new_size = sum(s[1] for s in new_symbols) |  | ||||||
|             lines.append( |  | ||||||
|                 f"| *{len(new_symbols) - 15} more new symbols...* | *Total: {format_bytes(total_new_size)}* |" |  | ||||||
|             ) |  | ||||||
|         lines.append("") |  | ||||||
|  |  | ||||||
|     # Show removed symbols |  | ||||||
|     if removed_symbols: |  | ||||||
|         removed_symbols.sort(key=lambda x: x[1], reverse=True) |  | ||||||
|         lines.extend( |  | ||||||
|             [ |  | ||||||
|                 "### Removed Symbols (top 15)", |  | ||||||
|                 "", |  | ||||||
|                 "| Symbol | Size |", |  | ||||||
|                 "|--------|------|", |  | ||||||
|             ] |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         for symbol, size in removed_symbols[:15]: |  | ||||||
|             display_symbol = format_symbol_for_display(symbol) |  | ||||||
|             lines.append(f"| {display_symbol} | {format_bytes(size)} |") |  | ||||||
|  |  | ||||||
|         if len(removed_symbols) > 15: |  | ||||||
|             total_removed_size = sum(s[1] for s in removed_symbols) |  | ||||||
|             lines.append( |  | ||||||
|                 f"| *{len(removed_symbols) - 15} more removed symbols...* | *Total: {format_bytes(total_removed_size)}* |" |  | ||||||
|             ) |  | ||||||
|         lines.append("") |  | ||||||
|  |  | ||||||
|     lines.extend(["</details>", ""]) |  | ||||||
|  |  | ||||||
|     return "\n".join(lines) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def create_detailed_breakdown_table( | def prepare_component_breakdown_data( | ||||||
|     target_analysis: dict | None, pr_analysis: dict | None |     target_analysis: dict | None, pr_analysis: dict | None | ||||||
| ) -> str: | ) -> list[tuple[str, int, int, int]] | None: | ||||||
|     """Create a markdown table showing detailed memory breakdown by component. |     """Prepare component breakdown data for template rendering. | ||||||
|  |  | ||||||
|     Args: |     Args: | ||||||
|         target_analysis: Component memory breakdown for target branch |         target_analysis: Component memory breakdown for target branch | ||||||
|         pr_analysis: Component memory breakdown for PR branch |         pr_analysis: Component memory breakdown for PR branch | ||||||
|  |  | ||||||
|     Returns: |     Returns: | ||||||
|         Formatted markdown table |         List of tuples (component, target_flash, pr_flash, delta), or None if no changes | ||||||
|     """ |     """ | ||||||
|     if not target_analysis or not pr_analysis: |     if not target_analysis or not pr_analysis: | ||||||
|         return "" |         return None | ||||||
|  |  | ||||||
|     # Combine all components from both analyses |     # Combine all components from both analyses | ||||||
|     all_components = set(target_analysis.keys()) | set(pr_analysis.keys()) |     all_components = set(target_analysis.keys()) | set(pr_analysis.keys()) | ||||||
|  |  | ||||||
|     # Filter to components that have changed (ignoring noise ≤2 bytes) |     # Filter to components that have changed (ignoring noise) | ||||||
|     changed_components = [] |     changed_components = [] | ||||||
|     for comp in all_components: |     for comp in all_components: | ||||||
|         target_mem = target_analysis.get(comp, {}) |         target_mem = target_analysis.get(comp, {}) | ||||||
| @@ -280,43 +209,18 @@ def create_detailed_breakdown_table( | |||||||
|         target_flash = target_mem.get("flash_total", 0) |         target_flash = target_mem.get("flash_total", 0) | ||||||
|         pr_flash = pr_mem.get("flash_total", 0) |         pr_flash = pr_mem.get("flash_total", 0) | ||||||
|  |  | ||||||
|         # Only include if component has meaningful change (>2 bytes) |         # Only include if component has meaningful change (above noise threshold) | ||||||
|         delta = pr_flash - target_flash |         delta = pr_flash - target_flash | ||||||
|         if abs(delta) > 2: |         if abs(delta) > COMPONENT_CHANGE_NOISE_THRESHOLD: | ||||||
|             changed_components.append((comp, target_flash, pr_flash, delta)) |             changed_components.append((comp, target_flash, pr_flash, delta)) | ||||||
|  |  | ||||||
|     if not changed_components: |     if not changed_components: | ||||||
|         return "" |         return None | ||||||
|  |  | ||||||
|     # Sort by absolute delta (largest changes first) |     # Sort by absolute delta (largest changes first) | ||||||
|     changed_components.sort(key=lambda x: abs(x[3]), reverse=True) |     changed_components.sort(key=lambda x: abs(x[3]), reverse=True) | ||||||
|  |  | ||||||
|     # Build table - limit to top 20 changes |     return changed_components | ||||||
|     lines = [ |  | ||||||
|         "", |  | ||||||
|         "<details open>", |  | ||||||
|         "<summary>📊 Component Memory Breakdown</summary>", |  | ||||||
|         "", |  | ||||||
|         "| Component | Target Flash | PR Flash | Change |", |  | ||||||
|         "|-----------|--------------|----------|--------|", |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     for comp, target_flash, pr_flash, delta in changed_components[:20]: |  | ||||||
|         target_str = format_bytes(target_flash) |  | ||||||
|         pr_str = format_bytes(pr_flash) |  | ||||||
|         # Only apply threshold to ESPHome components, not framework/infrastructure |  | ||||||
|         threshold = COMPONENT_CHANGE_THRESHOLD if comp.startswith("[esphome]") else None |  | ||||||
|         change_str = format_change(target_flash, pr_flash, threshold=threshold) |  | ||||||
|         lines.append(f"| `{comp}` | {target_str} | {pr_str} | {change_str} |") |  | ||||||
|  |  | ||||||
|     if len(changed_components) > 20: |  | ||||||
|         lines.append( |  | ||||||
|             f"| ... | ... | ... | *({len(changed_components) - 20} more components not shown)* |" |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     lines.extend(["", "</details>", ""]) |  | ||||||
|  |  | ||||||
|     return "\n".join(lines) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def create_comment_body( | def create_comment_body( | ||||||
| @@ -332,7 +236,7 @@ def create_comment_body( | |||||||
|     pr_symbols: dict | None = None, |     pr_symbols: dict | None = None, | ||||||
|     target_cache_hit: bool = False, |     target_cache_hit: bool = False, | ||||||
| ) -> str: | ) -> str: | ||||||
|     """Create the comment body with memory impact analysis. |     """Create the comment body with memory impact analysis using Jinja2 templates. | ||||||
|  |  | ||||||
|     Args: |     Args: | ||||||
|         components: List of component names (merged config) |         components: List of component names (merged config) | ||||||
| @@ -350,57 +254,87 @@ def create_comment_body( | |||||||
|     Returns: |     Returns: | ||||||
|         Formatted comment body |         Formatted comment body | ||||||
|     """ |     """ | ||||||
|     ram_change = format_change(target_ram, pr_ram, threshold=OVERALL_CHANGE_THRESHOLD) |     # Set up Jinja2 environment | ||||||
|     flash_change = format_change( |     env = Environment( | ||||||
|         target_flash, pr_flash, threshold=OVERALL_CHANGE_THRESHOLD |         loader=FileSystemLoader(TEMPLATE_DIR), | ||||||
|  |         trim_blocks=True, | ||||||
|  |         lstrip_blocks=True, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     # Use provided analysis data if available |     # Register custom filters | ||||||
|     component_breakdown = "" |     env.filters["format_bytes"] = format_bytes | ||||||
|     symbol_changes = "" |     env.filters["format_change"] = format_change | ||||||
|  |  | ||||||
|     if target_analysis and pr_analysis: |     # Prepare template context | ||||||
|         component_breakdown = create_detailed_breakdown_table( |     context = { | ||||||
|             target_analysis, pr_analysis |         "comment_marker": COMMENT_MARKER, | ||||||
|         ) |         "platform": platform, | ||||||
|  |         "target_ram": format_bytes(target_ram), | ||||||
|         if target_symbols and pr_symbols: |         "pr_ram": format_bytes(pr_ram), | ||||||
|             symbol_changes = create_symbol_changes_table(target_symbols, pr_symbols) |         "target_flash": format_bytes(target_flash), | ||||||
|     else: |         "pr_flash": format_bytes(pr_flash), | ||||||
|         print("No ELF files provided, skipping detailed analysis", file=sys.stderr) |         "ram_change": format_change( | ||||||
|  |             target_ram, pr_ram, threshold=OVERALL_CHANGE_THRESHOLD | ||||||
|  |         ), | ||||||
|  |         "flash_change": format_change( | ||||||
|  |             target_flash, pr_flash, threshold=OVERALL_CHANGE_THRESHOLD | ||||||
|  |         ), | ||||||
|  |         "target_cache_hit": target_cache_hit, | ||||||
|  |         "component_change_threshold": COMPONENT_CHANGE_THRESHOLD, | ||||||
|  |     } | ||||||
|  |  | ||||||
|     # Format components list |     # Format components list | ||||||
|     if len(components) == 1: |     if len(components) == 1: | ||||||
|         components_str = f"`{components[0]}`" |         context["components_str"] = f"`{components[0]}`" | ||||||
|         config_note = "a representative test configuration" |         context["config_note"] = "a representative test configuration" | ||||||
|     else: |     else: | ||||||
|         components_str = ", ".join(f"`{c}`" for c in sorted(components)) |         context["components_str"] = ", ".join(f"`{c}`" for c in sorted(components)) | ||||||
|         config_note = f"a merged configuration with {len(components)} components" |         context["config_note"] = ( | ||||||
|  |             f"a merged configuration with {len(components)} components" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     # Add cache info note if target was cached |     # Prepare component breakdown if available | ||||||
|     cache_note = "" |     component_breakdown = "" | ||||||
|     if target_cache_hit: |     if target_analysis and pr_analysis: | ||||||
|         cache_note = "\n\n> ⚡ Target branch analysis was loaded from cache (build skipped for faster CI)." |         changed_components = prepare_component_breakdown_data( | ||||||
|  |             target_analysis, pr_analysis | ||||||
|  |         ) | ||||||
|  |         if changed_components: | ||||||
|  |             template = env.get_template("ci_memory_impact_component_breakdown.j2") | ||||||
|  |             component_breakdown = template.render( | ||||||
|  |                 changed_components=changed_components, | ||||||
|  |                 format_bytes=format_bytes, | ||||||
|  |                 format_change=format_change, | ||||||
|  |                 component_change_threshold=COMPONENT_CHANGE_THRESHOLD, | ||||||
|  |                 max_rows=MAX_COMPONENT_BREAKDOWN_ROWS, | ||||||
|  |             ) | ||||||
|  |  | ||||||
|     return f"""{COMMENT_MARKER} |     # Prepare symbol changes if available | ||||||
| ## Memory Impact Analysis |     symbol_changes = "" | ||||||
|  |     if target_symbols and pr_symbols: | ||||||
|  |         symbol_data = prepare_symbol_changes_data(target_symbols, pr_symbols) | ||||||
|  |         if symbol_data: | ||||||
|  |             template = env.get_template("ci_memory_impact_symbol_changes.j2") | ||||||
|  |             symbol_changes = template.render( | ||||||
|  |                 **symbol_data, | ||||||
|  |                 format_bytes=format_bytes, | ||||||
|  |                 format_change=format_change, | ||||||
|  |                 max_changed_rows=MAX_CHANGED_SYMBOLS_ROWS, | ||||||
|  |                 max_new_rows=MAX_NEW_SYMBOLS_ROWS, | ||||||
|  |                 max_removed_rows=MAX_REMOVED_SYMBOLS_ROWS, | ||||||
|  |                 symbol_max_length=SYMBOL_DISPLAY_MAX_LENGTH, | ||||||
|  |                 symbol_truncate_length=SYMBOL_DISPLAY_TRUNCATE_LENGTH, | ||||||
|  |             ) | ||||||
|  |  | ||||||
| **Components:** {components_str} |     if not target_analysis or not pr_analysis: | ||||||
| **Platform:** `{platform}` |         print("No ELF files provided, skipping detailed analysis", file=sys.stderr) | ||||||
|  |  | ||||||
| | Metric | Target Branch | This PR | Change | |     context["component_breakdown"] = component_breakdown | ||||||
| |--------|--------------|---------|--------| |     context["symbol_changes"] = symbol_changes | ||||||
| | **RAM** | {format_bytes(target_ram)} | {format_bytes(pr_ram)} | {ram_change} | |  | ||||||
| | **Flash** | {format_bytes(target_flash)} | {format_bytes(pr_flash)} | {flash_change} | |  | ||||||
| {component_breakdown}{symbol_changes}{cache_note} |  | ||||||
|  |  | ||||||
| --- |     # Render main template | ||||||
| > **Note:** This analysis measures **static RAM and Flash usage** only (compile-time allocation). |     template = env.get_template("ci_memory_impact_comment_template.j2") | ||||||
| > **Dynamic memory (heap)** cannot be measured automatically. |     return template.render(**context) | ||||||
| > **⚠️ You must test this PR on a real device** to measure free heap and ensure no runtime memory issues. |  | ||||||
|  |  | ||||||
| *This analysis runs automatically when components change. Memory usage is measured from {config_note}.* |  | ||||||
| """ |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def find_existing_comment(pr_number: str) -> str | None: | def find_existing_comment(pr_number: str) -> str | None: | ||||||
| @@ -605,9 +539,9 @@ def main() -> int: | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     # Post or update comment |     # Post or update comment | ||||||
|     success = post_or_update_comment(args.pr_number, comment_body) |     post_or_update_comment(args.pr_number, comment_body) | ||||||
|  |  | ||||||
|     return 0 if success else 1 |     return 0 | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								script/templates/ci_memory_impact_comment_template.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								script/templates/ci_memory_impact_comment_template.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | {{ comment_marker }} | ||||||
|  | ## Memory Impact Analysis | ||||||
|  |  | ||||||
|  | **Components:** {{ components_str }} | ||||||
|  | **Platform:** `{{ platform }}` | ||||||
|  |  | ||||||
|  | | Metric | Target Branch | This PR | Change | | ||||||
|  | |--------|--------------|---------|--------| | ||||||
|  | | **RAM** | {{ target_ram }} | {{ pr_ram }} | {{ ram_change }} | | ||||||
|  | | **Flash** | {{ target_flash }} | {{ pr_flash }} | {{ flash_change }} | | ||||||
|  | {% if component_breakdown %} | ||||||
|  | {{ component_breakdown }} | ||||||
|  | {%- endif %} | ||||||
|  | {%- if symbol_changes %} | ||||||
|  | {{ symbol_changes }} | ||||||
|  | {%- endif %} | ||||||
|  | {%- if target_cache_hit %} | ||||||
|  |  | ||||||
|  | > ⚡ Target branch analysis was loaded from cache (build skipped for faster CI). | ||||||
|  | {%- endif %} | ||||||
|  |  | ||||||
|  | --- | ||||||
|  | > **Note:** This analysis measures **static RAM and Flash usage** only (compile-time allocation). | ||||||
|  | > **Dynamic memory (heap)** cannot be measured automatically. | ||||||
|  | > **⚠️ You must test this PR on a real device** to measure free heap and ensure no runtime memory issues. | ||||||
|  |  | ||||||
|  | *This analysis runs automatically when components change. Memory usage is measured from {{ config_note }}.* | ||||||
							
								
								
									
										15
									
								
								script/templates/ci_memory_impact_component_breakdown.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								script/templates/ci_memory_impact_component_breakdown.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  |  | ||||||
|  | <details open> | ||||||
|  | <summary>📊 Component Memory Breakdown</summary> | ||||||
|  |  | ||||||
|  | | Component | Target Flash | PR Flash | Change | | ||||||
|  | |-----------|--------------|----------|--------| | ||||||
|  | {% for comp, target_flash, pr_flash, delta in changed_components[:max_rows] -%} | ||||||
|  | {% set threshold = component_change_threshold if comp.startswith("[esphome]") else none -%} | ||||||
|  | | `{{ comp }}` | {{ target_flash|format_bytes }} | {{ pr_flash|format_bytes }} | {{ format_change(target_flash, pr_flash, threshold=threshold) }} | | ||||||
|  | {% endfor -%} | ||||||
|  | {% if changed_components|length > max_rows -%} | ||||||
|  | | ... | ... | ... | *({{ changed_components|length - max_rows }} more components not shown)* | | ||||||
|  | {% endif -%} | ||||||
|  |  | ||||||
|  | </details> | ||||||
							
								
								
									
										8
									
								
								script/templates/ci_memory_impact_macros.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								script/templates/ci_memory_impact_macros.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | {#- Macro for formatting symbol names in tables -#} | ||||||
|  | {%- macro format_symbol(symbol, max_length, truncate_length) -%} | ||||||
|  | {%- if symbol|length <= max_length -%} | ||||||
|  | `{{ symbol }}` | ||||||
|  | {%- else -%} | ||||||
|  | <details><summary><code>{{ symbol[:truncate_length] }}...</code></summary><code>{{ symbol }}</code></details> | ||||||
|  | {%- endif -%} | ||||||
|  | {%- endmacro -%} | ||||||
							
								
								
									
										51
									
								
								script/templates/ci_memory_impact_symbol_changes.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								script/templates/ci_memory_impact_symbol_changes.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | |||||||
|  | {%- from 'ci_memory_impact_macros.j2' import format_symbol -%} | ||||||
|  |  | ||||||
|  | <details> | ||||||
|  | <summary>🔍 Symbol-Level Changes (click to expand)</summary> | ||||||
|  |  | ||||||
|  | {%- if changed_symbols %} | ||||||
|  |  | ||||||
|  | ### Changed Symbols | ||||||
|  |  | ||||||
|  | | Symbol | Target Size | PR Size | Change | | ||||||
|  | |--------|-------------|---------|--------| | ||||||
|  | {% for symbol, target_size, pr_size, delta in changed_symbols[:max_changed_rows] -%} | ||||||
|  | | {{ format_symbol(symbol, symbol_max_length, symbol_truncate_length) }} | {{ target_size|format_bytes }} | {{ pr_size|format_bytes }} | {{ format_change(target_size, pr_size) }} | | ||||||
|  | {% endfor -%} | ||||||
|  | {% if changed_symbols|length > max_changed_rows -%} | ||||||
|  | | ... | ... | ... | *({{ changed_symbols|length - max_changed_rows }} more changed symbols not shown)* | | ||||||
|  | {% endif -%} | ||||||
|  |  | ||||||
|  | {%- endif %} | ||||||
|  | {%- if new_symbols %} | ||||||
|  |  | ||||||
|  | ### New Symbols (top {{ max_new_rows }}) | ||||||
|  |  | ||||||
|  | | Symbol | Size | | ||||||
|  | |--------|------| | ||||||
|  | {% for symbol, size in new_symbols[:max_new_rows] -%} | ||||||
|  | | {{ format_symbol(symbol, symbol_max_length, symbol_truncate_length) }} | {{ size|format_bytes }} | | ||||||
|  | {% endfor -%} | ||||||
|  | {% if new_symbols|length > max_new_rows -%} | ||||||
|  | {% set total_new_size = new_symbols|sum(attribute=1) -%} | ||||||
|  | | *{{ new_symbols|length - max_new_rows }} more new symbols...* | *Total: {{ total_new_size|format_bytes }}* | | ||||||
|  | {% endif -%} | ||||||
|  |  | ||||||
|  | {%- endif %} | ||||||
|  | {%- if removed_symbols %} | ||||||
|  |  | ||||||
|  | ### Removed Symbols (top {{ max_removed_rows }}) | ||||||
|  |  | ||||||
|  | | Symbol | Size | | ||||||
|  | |--------|------| | ||||||
|  | {% for symbol, size in removed_symbols[:max_removed_rows] -%} | ||||||
|  | | {{ format_symbol(symbol, symbol_max_length, symbol_truncate_length) }} | {{ size|format_bytes }} | | ||||||
|  | {% endfor -%} | ||||||
|  | {% if removed_symbols|length > max_removed_rows -%} | ||||||
|  | {% set total_removed_size = removed_symbols|sum(attribute=1) -%} | ||||||
|  | | *{{ removed_symbols|length - max_removed_rows }} more removed symbols...* | *Total: {{ total_removed_size|format_bytes }}* | | ||||||
|  | {% endif -%} | ||||||
|  |  | ||||||
|  | {%- endif %} | ||||||
|  |  | ||||||
|  | </details> | ||||||
		Reference in New Issue
	
	Block a user