clm_generator/excel_to_clm.py
import os
from datetime import datetime
import re
import json
from difflib import unified_diff
from pathlib import Path
from shutil import copy2
from openpyxl import load_workbook
import xlrd
from jinja2 import Template
from clm_config_updater import CLMRangeSynchronizer
class ExcelToCLMConverter:
import os
import json
def init(self, config_path=“config/config.json”, output_dir=“output”, locale_id=None, locale_display_name=None): # === Step 1: 解析配置文件路径(基于当前文件位置)=== if not os.path.isabs(config_path): # 获取当前 .py 文件所在目录(例如 F:\issue) current_dir = os.path.dirname(os.path.abspath(file)) config_path = os.path.join(current_dir, config_path) self.config_file_path = os.path.abspath(config_path) if not os.path.exists(self.config_file_path): raise FileNotFoundError(f"配置文件不存在: {self.config_file_path}“) with open(self.config_file_path, ‘r’, encoding=‘utf-8’) as f: self.config = json.load(f) print(f”✅ 配置文件已加载: {self.config_file_path}“) # === 计算项目根目录:config 文件所在的父目录(通常是项目根)=== project_root = os.path.dirname(os.path.dirname(self.config_file_path)) # 如果 config 在 config/ 子目录,则 parent 就是项目根 F:\issue # === Step 2: 处理 target_c_file === rel_c_path = self.config.get(“target_c_file”, “input/wlc_clm_data_6726b0.c”) if not os.path.isabs(rel_c_path): # 使用 project_root 作为基准 self.target_c_file = os.path.normpath(os.path.join(project_root, rel_c_path)) else: self.target_c_file = os.path.normpath(rel_c_path) if not os.path.exists(self.target_c_file): raise FileNotFoundError(f"配置中指定的 C 源文件不存在: {self.target_c_file}”) print(f"🔧 已定位目标 C 文件: {self.target_c_file}“) # === Step 3: 初始化其他属性 === self.output_dir = output_dir if not os.path.isabs(self.output_dir): self.output_dir = os.path.normpath(os.path.join(project_root, self.output_dir)) os.makedirs(self.output_dir, exist_ok=True) print(f"📁 输出目录: {self.output_dir}”) self.locale_id = locale_id or self.config.get(“DEFAULT_LOCALE_ID”, “DEFAULT”) self.locale_display_name = ( locale_display_name or self.config.get(“DEFAULT_DISPLAY_NAME”) or self.locale_id.replace(‘-’, ‘').upper() ) # === Step 4: 加载 channel_set_map === persisted_map = self.config.get(“channel_set_map”) if persisted_map is None: raise KeyError( “❌ 配置文件缺少必需字段 ‘channel_set_map’。\n” “请在 config.json 中显式添加该字段,例如:\n” ‘“channel_set_map”: {}\n’ “💡 提示:禁止通过 channel_sets 自动重建,防止状态混乱。” ) if not isinstance(persisted_map, dict): raise TypeError(f"❌ channel_set_map 必须是字典类型,当前类型: {type(persisted_map)}“) self.channel_set_map = {str(k): int(v) for k, v in persisted_map.items()} print(f"🔁 成功加载 channel_set_map (共 {len(self.channel_set_map)} 项): {dict(self.channel_set_map)}”) # === 其他初始化 === self.tx_limit_entries = [] self.eirp_entries = [] self.global_ch_min = None self.global_ch_max = None self.generated_ranges = [] # ==================== 新增工具方法:大小写安全查询 ==================== def ci_get(self, data_dict, key): “”" Case-insensitive 字典查找 “”" for k, v in data_dict.items(): if k.lower() == key.lower(): return v return None def ci_contains(self, data_list, item): “”" Case-insensitive 判断元素是否在列表中 “”" return any(x.lower() == item.lower() for x in data_list) # ==================== 原有 parse_mode_cell 方法保持不变 ==================== def parse_mode_cell(self, cell_value): if not cell_value: return None val = str(cell_value).strip() val = re.sub(r’\s+‘, ’ ‘, val.replace(’\n’, ’ ‘).replace(’\r’, ’ ‘)) val_upper = val.upper() found_modes = [] # ✅ 改进:使用 match + 允许后续内容(比如 20M),不再要求全匹配 if re.match(r’^11AC\s*/\sAX’, val_upper) or re.match(r’^11AX\s/\sAC’, val_upper): found_modes = [‘11AC’, ‘11AX’] print(f"🔍 解析复合模式 ‘{val}’ → {found_modes}") # ======== 一般情况:正则匹配标准模式 ======== else: mode_patterns = [ (r’\b11BE\b|\bEHT\b’, ‘11BE’), (r’\b11AX\b|\bHE\b’, ‘11AX’), (r’\b11AC\b|\bVHT\b’, ‘11AC’), # 自动匹配 11AC 或 VHT (r’\b11N\b|\bHT\b’, ‘11N’), (r’\b11G\b|\bERP\b’, ‘11G’), (r’\b11B\b|\bDSSS\b|\bCCK\b’, ‘11B’) ] for pattern, canonical in mode_patterns: if re.search(pattern, val_upper) and canonical not in found_modes: found_modes.append(canonical) # ======== 提取带宽 ======== bw_match = re.search(r’\b(20|40|80|160)\s(?:MHZ|M)?\b’, val_upper) bw = bw_match.group(1) if bw_match else None # fallback 带宽 if not bw: if all(m in [‘11B’, ‘11G’] for m in found_modes): bw = ‘20’ else: bw = ‘20’ if not found_modes: print(f"🟡 无法识别物理模式: ‘{cell_value}’“) return None return { “phy_mode_list”: found_modes, “bw”: bw } def format_phy_mode(self, mode: str) -> str: “”” 自定义物理层模式输出格式: - 11B/G/N 输出为小写:11b / 11g / 11n - 其他保持原样(如 11AC, 11BE) “”" return { ‘11B’: ‘11b’, ‘11G’: ‘11g’, ‘11N’: ‘11n’ }.get(mode, mode) def load_config(self, path=“config/config.json”): “”“加载配置文件,并在此时设置默认 locale_id”“” if not os.path.exists(path): raise FileNotFoundError(f"配置文件不存在: {path}“) with open(path, ‘r’, encoding=‘utf-8’) as f: self.config = json.load(f) # ✅ 只有在这里才安全地使用 self.config.get() if not self.locale_id: self.locale_id = self.config.get(“DEFAULT_LOCALE_ID”, “DEFAULT”) print(f”✅ 配置文件加载成功: {path}“) print(f"🌍 使用 Locale ID: {self.locale_id}”) def col_to_letter(self, col): col += 1 result = “” while col > 0: col -= 1 result = chr(col % 26 + ord(‘A’)) + result col //= 26 return result def is_valid_power(self, value): try: float(value) return True except (ValueError, TypeError): return False def get_cell_value(self, ws_obj, row_idx, col_idx): fmt = ws_obj[“format”] if fmt == “xls”: return str(ws_obj[“sheet”].cell_value(row_idx, col_idx)).strip() else: cell = ws_obj[“sheet”].cell(row=row_idx + 1, column=col_idx + 1) val = cell.value return str(val).strip() if val is not None else “” def find_table_header_row(self, ws_obj): “”“查找包含 ‘Mode’ 和 ‘Rate’ 的表头行”“” fmt = ws_obj[“format”] ws = ws_obj[“sheet”] for r in range(15): mode_col = rate_col = None if fmt == “xlsx”: if r + 1 > ws.max_row: continue for c in range(1, ws.max_column + 1): cell = ws.cell(row=r + 1, column=c) if not cell.value: continue val = str(cell.value).strip() if val == “Mode”: mode_col = c elif val == “Rate”: rate_col = c if mode_col and rate_col and abs(mode_col - rate_col) == 1: print(f"✅ 找到表头行: 第 {r+1} 行") return r, mode_col - 1, rate_col - 1 # 转为 0-based else: if r >= ws.nrows: continue for c in range(ws.ncols): val = ws.cell_value(r, c) if not val: continue val = str(val).strip() if val == “Mode”: mode_col = c elif val == “Rate”: rate_col = c if mode_col and rate_col and abs(mode_col - rate_col) == 1: print(f"✅ 找到表头行: 第 {r+1} 行") return r, mode_col, rate_col return None, None, None def find_auth_power_above_row(self, ws_obj, start_row): “”“查找 ‘认证功率’ 所在的合并单元格及其列范围”“” fmt = ws_obj[“format”] ws = ws_obj[“sheet”] print(f"🔍 开始向上查找 ‘认证功率’,扫描第 0 ~ {start_row} 行…“) if fmt == “xlsx”: for mr in ws.merged_cells.ranges: top_left = ws.cell(row=mr.min_row, column=mr.min_col) val = str(top_left.value) if top_left.value else “” if “证功率” in val or “Cert” in val: r_idx = mr.min_row - 1 if r_idx <= start_row: start_col = mr.min_col - 1 end_col = mr.max_col - 1 print(f"📌 发现合并单元格含 ‘证功率’: ‘{val}’ → {self.col_to_letter(start_col)}{mr.min_row}”) return start_col, end_col, r_idx # fallback:普通单元格 for r in range(start_row + 1): for c in range(1, ws.max_column + 1): cell = ws.cell(row=r + 1, column=c) if cell.value and (“证功率” in str(cell.value)): print(f"📌 普通单元格发现 ‘证功率’: ‘{cell.value}’ @ R{r+1}C{c}“) return c - 1, c - 1, r else: for r in range(min(ws.nrows, start_row + 1)): for c in range(ws.ncols): val = ws.cell_value(r, c) if val and (“证功率” in str(val)): print(f"📌 发现 ‘证功率’: ‘{val}’ @ R{r+1}C{c+1}”) return c, c, r return None, None, None def parse_ch_columns_under_auth(self, ws_obj, ch_row_idx, auth_start_col, auth_end_col): “”" 只解析位于 [auth_start_col, auth_end_col] 区间内的 CHx 列 “”" fmt = ws_obj[“format”] ws = ws_obj[“sheet”] ch_map = {} print(f"🔍 解析 CH 行(第 {ch_row_idx + 1} 行),限定列范围: Col {auth_start_col} ~ {auth_end_col}“) if fmt == “xlsx”: for c in range(auth_start_col, auth_end_col + 1): cell = ws.cell(row=ch_row_idx + 1, column=c + 1) val = self.get_cell_value(ws_obj, ch_row_idx, c) match = re.search(r"CH(\d+)”, val, re.I) if match: ch_num = int(match.group(1)) ch_map[ch_num] = c print(f" 👉 发现 CH{ch_num} @ Col{c}“) else: for c in range(auth_start_col, auth_end_col + 1): val = self.get_cell_value(ws_obj, ch_row_idx, c) match = re.search(r"CH(\d+)”, val, re.I) if match: ch_num = int(match.group(1)) ch_map[ch_num] = c print(f" 👉 发现 CH{ch_num} @ Col{c}“) if not ch_map: print(”❌ 在指定区域内未找到任何 CHx 列") else: chs = sorted(ch_map.keys()) print(f"✔️ 成功提取 CH{min(chs)}-{max(chs)} 共 {len(chs)} 个信道") return ch_map def encode_power(self, dbm): return int(round((float(dbm) + 1.5) * 4)) def merge_consecutive_channels(self, ch_list): if not ch_list: return [] sorted_ch = sorted(ch_list) ranges = [] start = end = sorted_ch[0] for ch in sorted_ch[1:]: if ch == end + 1: end = ch else: ranges.append((start, end)) start = end = ch ranges.append((start, end)) return ranges # ==================== 修改 collect_tx_limit_data ==================== def collect_tx_limit_data(self, ws_obj, sheet_config, header_row_idx, auth_row, auth_start, auth_end, mode_col, rate_col): ch_row_idx = auth_row + 2 nrows = ws_obj[“sheet”].nrows if ws_obj[“format”] == “xls” else ws_obj[“sheet”].max_row if ch_row_idx >= nrows: print(f"❌ CH 行 ({ch_row_idx + 1}) 超出范围") return [] # ✅ 提取认证功率下方的 CH 列映射 ch_map = self.parse_ch_columns_under_auth(ws_obj, ch_row_idx, auth_start, auth_end) if not ch_map: return [] entries = [] row_mode_info = {} # {row_index: parsed_mode_info} fmt = ws_obj[“format”] ws = ws_obj[“sheet”] # ======== 第一步:构建 row_mode_info —— 使用新解析器 ======== if fmt == “xlsx”: merged_cells_map = {} for mr in ws.merged_cells.ranges: for r in range(mr.min_row - 1, mr.max_row): for c in range(mr.min_col - 1, mr.max_col): merged_cells_map[(r, c)] = mr for row_idx in range(header_row_idx + 1, nrows): cell_value = None is_merged = (row_idx, mode_col) in merged_cells_map if is_merged: mr = merged_cells_map[(row_idx, mode_col)] top_cell = ws.cell(row=mr.min_row, column=mr.min_col) cell_value = top_cell.value else: raw_cell = ws.cell(row=row_idx + 1, column=mode_col + 1) cell_value = raw_cell.value mode_info = self.parse_mode_cell(cell_value) if mode_info: if is_merged: mr = merged_cells_map[(row_idx, mode_col)] for r in range(mr.min_row - 1, mr.max_row): if header_row_idx < r < nrows: row_mode_info[r] = mode_info.copy() else: row_mode_info[row_idx] = mode_info.copy() else: for row_idx in range(header_row_idx + 1, ws.nrows): cell_value = self.get_cell_value(ws_obj, row_idx, mode_col) mode_info = self.parse_mode_cell(cell_value) if mode_info: row_mode_info[row_idx] = mode_info.copy() # ======== 第二步:生成条目(关键修改区)======== for row_idx in range(header_row_idx + 1, nrows): mode_info = row_mode_info.get(row_idx) if not mode_info: continue bw_clean = mode_info[“bw”] has_valid_power = False for ch, col_idx in ch_map.items(): power_val = self.get_cell_value(ws_obj, row_idx, col_idx) if self.is_valid_power(power_val): has_valid_power = True break if not has_valid_power: print(f"🗑️ 跳过空行: 第 {row_idx + 1} 行(无任何有效功率值)“) continue # ---- 遍历每个 phy_mode ---- for phy_mode in mode_info[“phy_mode_list”]: formatted_mode = self.format_phy_mode(phy_mode) mode_key = f”{formatted_mode}M" # ✅ 改为大小写不敏感判断 if not self.ci_contains(sheet_config.get(“modes”, []), mode_key): print(f"⚠️ 忽略不支持的模式: {mode_key}") continue # === 获取 rate_set 定义(可能是 str 或 list)=== raw_rate_set = self.ci_get(sheet_config[“rate_set_map”], mode_key) if not raw_rate_set: print(f"❌ 找不到 rate_set 映射: {mode_key}“) continue # 统一转为 list 处理 if isinstance(raw_rate_set, str): rate_set_list = [raw_rate_set] elif isinstance(raw_rate_set, list): rate_set_list = raw_rate_set else: continue # 非法类型跳过 for rate_set_macro in rate_set_list: ch_count = 0 for ch, col_idx in ch_map.items(): power_val = self.get_cell_value(ws_obj, row_idx, col_idx) if not self.is_valid_power(power_val): continue try: power_dbm = float(power_val) except: continue encoded_power = self.encode_power(power_dbm) entries.append({ “ch”: ch, “power_dbm”: round(power_dbm, 2), “encoded_power”: encoded_power, “rate_set_macro”: rate_set_macro, # <<< 每个 macro 单独一条记录 “mode”: phy_mode, “bw”: bw_clean, “src_row”: row_idx + 1, “band”: sheet_config[“band”] }) ch_count += 1 print( f"📊 已采集第 {row_idx + 1} 行 → {formatted_mode} {bw_clean}M, {ch_count} 个信道, 使用宏: {rate_set_macro}” ) return entries def compress_tx_limit_entries(self, raw_entries, sheet_config): “”" 压缩TX限制条目。 Args: raw_entries (list): 原始条目列表。 sheet_config (dict): Excel表格配置字典。 Returns: list: 压缩后的条目列表。 “”" from collections import defaultdict modes_order = sheet_config[“modes”] # ✅ 构建小写映射用于排序(key: “11n_20M”) mode_lower_to_index = {mode.lower(): idx for idx, mode in enumerate(modes_order)} range_template = sheet_config[“range_macro_template”] group_key = lambda e: (e[“encoded_power”], e[“rate_set_macro”]) groups = defaultdict(list) for e in raw_entries: groups[group_key(e)].append(e) compressed = [] for (encoded_power, rate_set_macro), entries_in_group in groups.items(): first = entries_in_group[0] power_dbm = first[“power_dbm”] mode = first[“mode”] # 如 ‘11N’ bw = first[“bw”] # 如 ‘20’ 或 ‘40’ ch_list = sorted(e[“ch”] for e in entries_in_group) for start, end in self.merge_consecutive_channels(ch_list): range_macro = range_template.format( band=sheet_config[“band”], bw=bw, start=start, end=end ) # === 🔥 新增:查找或分配 CHANNEL_SET_ID === assigned_id = -1 # 表示:这不是 regulatory 范围,无需映射 # === 🔥 新增:记录到 generated_ranges === segment_ch_list = list(range(start, end + 1)) self.record_generated_range( range_macro=range_macro, band=sheet_config[“band”], bw=bw, ch_start=start, ch_end=end, channels=segment_ch_list ) # ✅ 格式化物理层模式(如 ‘11N’ -> ‘11n’) formatted_mode = self.format_phy_mode(mode) # ✅ 构造 mode_key 用于查找排序优先级 mode_key = f"{formatted_mode}M" mode_order_idx = mode_lower_to_index.get(mode_key.lower(), 999) # ✅ 生成注释 comment = f"/* {power_dbm:5.2f}dBm, CH{start}-{end}, {formatted_mode} @ {bw}MHz */" # ✅ 新增:生成该段落的实际信道列表 segment_ch_list = list(range(start, end + 1)) compressed.append({ “encoded_power”: encoded_power, “range_macro”: range_macro, “rate_set_macro”: rate_set_macro, “comment”: comment, "mode_order": mode_order_idx, # — 👇 新增:保留关键字段供模板使用 — “bw”: bw, # 带宽数字(字符串) “mode”: formatted_mode, # 统一格式化的模式名 “ch_start”: start, “ch_end”: end, “power_dbm”: round(power_dbm, 2), “ch_list”: segment_ch_list, # ✅ 关键!用于 global_ch_min/max 统计 }) # 排序后删除临时字段 compressed.sort(key=lambda x: x["mode_order"]) for item in compressed: del item["mode_order"] return compressed def record_generated_range(self, range_macro, band, bw, ch_start, ch_end, channels): “”" 记录生成的 RANGE 宏信息,供后续输出 manifest 使用 注意:不再记录 channel_set_id “”" self.generated_ranges.append({ “range_macro”: range_macro, “band”: band, “bandwidth”: int(bw), “channels”: sorted(channels), “start_channel”: ch_start, “end_channel”: int(ch_end), “source_sheet”: getattr(self, ‘current_sheet_name’, ‘unknown’) }) def clean_sheet_name(self, name): cleaned = re.sub(r’\w.=\u4e00-\u9fa5’, ‘’, str(name)) return cleaned def match_sheet_to_config(self, sheet_name): cleaned = self.clean_sheet_name(sheet_name) for cfg in self.config[“sheets”]: for pat in cfg[“pattern”]: if re.search(pat, cleaned, re.I): print(f"🧹 ‘{sheet_name}’ → 清洗后: ‘{cleaned}’“) print(f”✅ 匹配成功!‘{sheet_name}’ → [{cfg[‘band’]}] 配置") return cfg print(f"🧹 ‘{sheet_name}’ → 清洗后: ‘{cleaned}’“) print(f"🟡 未匹配到 ‘{sheet_name}’ 的模式,跳过…”) return None def convert_sheet_with_config(self, ws_obj, sheet_name, sheet_config): self.current_sheet_name = sheet_name # 👈 设置当前 sheet 名,供 record_generated_range 使用 header_row_idx, mode_col, rate_col = self.find_table_header_row(ws_obj) if header_row_idx is None: print(f"🟡 跳过 ‘{sheet_name}’:未找到 ‘Mode’ 和 ‘Rate’“) return auth_start, auth_end, auth_row = self.find_auth_power_above_row(ws_obj, header_row_idx) if auth_start is None: print(f"🟡 跳过 ‘{sheet_name}’:未找到 ‘认证功率’”) return raw_entries = self.collect_tx_limit_data( ws_obj, sheet_config, header_row_idx, auth_row, auth_start, auth_end, mode_col, rate_col ) if not raw_entries: print(f"⚠️ 从 ‘{sheet_name}’ 未收集到有效数据") return compressed = self.compress_tx_limit_entries(raw_entries, sheet_config) # ✅ 新增:仅对 2.4G 频段进行信道边界统计 band = str(sheet_config.get(“band”, “”)).strip().upper() if band in [“2G”, “2.4G”, “2.4GHZ”, “BGN”]: # 执行信道统计 for entry in compressed: ch_range = entry.get(“ch_list”) or [] if not ch_range: continue ch_start = min(ch_range) ch_end = max(ch_range) # 更新全局最小最大值 if self.global_ch_min is None or ch_start < self.global_ch_min: self.global_ch_min = ch_start if self.global_ch_max is None or ch_end > self.global_ch_max: self.global_ch_max = ch_end # ✅ 强制打印当前状态 print(f"📊 [Band={band}] 累计 2.4G 信道范围: CH{self.global_ch_min} – CH{self.global_ch_max}“) self.tx_limit_entries.extend(compressed) print(f”✔️ 成功从 ‘{sheet_name}’ 添加 {len(compressed)} 条压缩后 TX 限幅条目") # 可选调试输出 if band == “2G” and self.global_ch_min is not None: print(f"📊 当前累计 2.4G 信道范围: CH{self.global_ch_min} – CH{self.global_ch_max}“) def render_from_template(self, template_path, context, output_path): “”” 根据模板生成文件。 Args: template_path (str): 模板文件路径。 context (dict): 渲染模板所需的上下文数据。 output_path (str): 输出文件的路径。 Returns: None Raises: FileNotFoundError: 如果指定的模板文件不存在。 IOError: 如果在读取或写入文件时发生错误。 “”" with open(template_path, ‘r’, encoding=‘utf-8’) as f: template = Template(f.read()) content = template.render(**context) os.makedirs(os.path.dirname(output_path), exist_ok=True) with open(output_path, ‘w’, encoding=‘utf-8’) as f: f.write(content) print(f"🎉 已生成: {output_path}“) def generate_outputs(self): print(“🔧 正在执行 generate_outputs()…”) if not self.tx_limit_entries: print(”⚠️ 无 TX 限幅数据可输出") return # === Step 1: 使用 “HT” 分类 entries === normal_entries = [] ht_entries = [] for e in self.tx_limit_entries: macro = e[“rate_set_macro”] if “HT” in macro: ht_entries.append(e) else: normal_entries.append(e) print(f"📊 自动分类结果:“) print(f” ├─ Normal 模式(不含 HT): {len(normal_entries)} 条") print(f" └─ HT 模式(含 HT): {len(ht_entries)} 条") # === Step 2: 构建 g_tx_limit_normal 结构(按 bw 排序)=== def build_normal_structure(entries): from collections import defaultdict grouped = defaultdict(list) for e in entries: grouped[e[“bw”]].append(e) result = [] for bw in [“20”, “40”, “80”, “160”]: if bw in grouped: sorted_entries = sorted(grouped[bw], key=lambda x: (x[“ch_start”], x[“encoded_power”])) result.append((bw, sorted_entries)) return result normal_struct = build_normal_structure(normal_entries) # === Step 3: 构建 g_tx_limit_ht 结构(严格顺序)=== def build_ht_structure(entries): from collections import defaultdict groups = defaultdict(list) for e in entries: if “EXT4” in e[“rate_set_macro”]: level = “ext4” elif “EXT” in e[“rate_set_macro”]: level = “ext” else: level = “base” groups[(level, e[“bw”])].append(e) order = [ (“base”, “20”), (“base”, “40”), (“ext”, “20”), (“ext”, “40”), (“ext4”, “20”), (“ext4”, “40”) ] segments = [] active_segment_count = sum(1 for key in order if key in groups) for idx, (level, bw) in enumerate(order): key = (level, bw) if key not in groups: continue seg_entries = sorted(groups[key], key=lambda x: (x[“ch_start”], x[“encoded_power”])) count = len(seg_entries) header_flags = f"CLM_DATA_FLAG_WIDTH | CLM_DATA_FLAG_MEAS_COND" if idx < active_segment_count - 1: header_flags += " | CLM_DATA_FLAG_MORE" if level != “base”: header_flags += " | CLM_DATA_FLAG_FLAG2" segment = { “header_flags”: header_flags, “count”: count, “entries”: seg_entries } if level == “ext”: segment[“flag2”] = “CLM_DATA_FLAG2_RATE_TYPE_EXT” elif level == “ext4”: segment[“flag2”] = “CLM_DATA_FLAG2_RATE_TYPE_EXT4” segments.append(segment) return segments ht_segments = build_ht_structure(ht_entries) # === Step 4: fallback range 和 CHANNEL_SET 自动创建逻辑 === fallback_range_macro = “RANGE_EIRP_DUMMY” fallback_ch_start = fallback_ch_end = 1 fallback_channel_set_id = 1 channel_set_comment = “Fallback 2.4GHz channel set (default)” if self.global_ch_min is not None and self.global_ch_max is not None: fallback_range_macro = f"RANGE_2G_20M" fallback_ch_start = self.global_ch_min fallback_ch_end = self.global_ch_max print(f"🔧 正在设置监管 fallback 范围: {fallback_range_macro}“) fallback_channel_set_id = 1 self.channel_set_map[fallback_range_macro] = fallback_channel_set_id print(f”✅ 已绑定监管 fallback: {fallback_range_macro} → CHANNEL_SET") else: fallback_range_macro = “RANGE_2G_20M_1_11” fallback_ch_start = 1 fallback_ch_end = 11 fallback_channel_set_id = 1 self.channel_set_map[fallback_range_macro] = fallback_channel_set_id print(“⚠️ 未检测到有效的 2.4G 信道范围,使用默认 fallback: RANGE_2G_20M_1_11 → CHANNEL_SET_1”) # === Step 5: 渲染上下文集合 === timestamp = datetime.now().strftime(“%Y-%m-%d %H:%M:%S”) locale_id_safe = self.locale_id.replace(‘-’, '‘) context_clm = { “locale_id”: locale_id_safe, “eirp_entries”: self.eirp_entries or [], “fallback_encoded_eirp”: 30, “fallback_range_macro”: fallback_range_macro, “fallback_ch_start”: fallback_ch_start, “fallback_ch_end”: fallback_ch_end, “entries_grouped_by_bw”: normal_struct, } context_tables = { “timestamp”: timestamp, “locale_id”: locale_id_safe, “locale_display_name”: self.locale_display_name, “normal_table”: normal_struct, “ht_segments”: ht_segments, “fallback_encoded_eirp”: 30, “fallback_range_macro”: fallback_range_macro, “fallback_ch_start”: fallback_ch_start, “fallback_ch_end”: fallback_ch_end, “fallback_channel_set_id”: fallback_channel_set_id, “channel_set_comment”: channel_set_comment, } os.makedirs(self.output_dir, exist_ok=True) # ================================ # 🔍 新增:分析 tx_limit_table.c 的变更 # ================================ output_path = Path(self.output_dir) / “tx_limit_table.c” template_path = “templates/tx_limit_table.c.j2” # 1️⃣ 读取原始文件内容(如果存在) if output_path.exists(): original_lines = output_path.read_text(encoding=‘utf-8’).splitlines() else: original_lines = [] print(f"🆕 将创建新文件: {output_path}") # 2️⃣ 生成新内容(不直接写入) new_content = self.render_from_template_string( template_path=template_path, context=context_tables ) new_lines = new_content.splitlines() # 3️⃣ 计算差异 diff = list(unified_diff( original_lines, new_lines, fromfile=“before.c”, tofile=“after.c”, lineterm=’', n=0 )) # 4️⃣ 提取变更信息 changes = { ‘added_ranges’: set(), ‘removed_ranges’: set(), ‘modified_ranges’: set(), ‘other_additions’: [], ‘other_deletions’: [] } range_pattern = re.compile(r’RANGE\w+‘) for line in diff: if line.startswith(’—‘) or line.startswith(’+‘): continue matches = range_pattern.findall(line) if line.startswith(’+ ‘) and not line.startswith(’+‘): for m in matches: changes[‘added_ranges’].add(m) if not matches: changes[‘other_additions’].append(line[2:]) elif line.startswith(’- ‘) and not line.startswith(’—'): for m in matches: changes[‘removed_ranges’].add(m) if not matches: changes[‘other_deletions’].append(line[2:]) # 推断修改:出现在删除和添加中的 RANGE common = changes[‘added_ranges’] & changes[‘removed_ranges’] changes[‘modified_ranges’] = common changes[‘added_ranges’] -= common changes[‘removed_ranges’] -= common # 5️⃣ 输出变更摘要 print("\n" + “=” * 60) print(“📝 变更摘要 - tx_limit_table.c”) print(“=” * 60) total_changes = ( len(changes[‘added_ranges’]) + len(changes[‘removed_ranges’]) + len(changes[‘modified_ranges’]) + len(changes[‘other_additions’]) + len(changes[‘other_deletions’]) ) if total_changes == 0: print(“🟢 文件无变化,已是最新状态”) else: if changes[‘added_ranges’]: print(f"🟢 新增 {len(changes[‘added_ranges’])} 个 RANGE:“) for r in sorted(changes[‘added_ranges’]): print(f” → {r}“) if changes[‘removed_ranges’]: print(f"🔴 删除 {len(changes[‘removed_ranges’])} 个 RANGE:”) for r in sorted(changes[‘removed_ranges’]): print(f" → {r}“) if changes[‘modified_ranges’]: print(f"🟡 修改 {len(changes[‘modified_ranges’])} 个 RANGE:”) for r in sorted(changes[‘modified_ranges’]): print(f" → {r}“) other_count = len(changes[‘other_additions’]) + len(changes[‘other_deletions’]) if other_count > 0: print(f"🔵 其他变更 ({other_count} 处)😊 for add in changes[‘other_additions’][:3]: print(f” ➕ {add}“) for rem in changes[‘other_deletions’][:3]: print(f” ➖ {rem}“) if len(changes[‘other_additions’]) > 3 or len(changes[‘other_deletions’]) > 3: print(f” … 还有 {other_count - 6} 处未显示") print(“=” * 60) # === 🔁 新增:写入日志文件 === self.log_changes_to_file( changes=changes, output_dir=self.output_dir, locale_id=self.locale_id, total_entries=len(self.tx_limit_entries) ) # === Step 6: 实际渲染并写入文件 === self.render_from_template( “templates/clm_locale.c.j2”, context_clm, os.path.join(self.output_dir, f"locale.c") ) # 写入新内容 output_path.write_text(new_content, encoding=‘utf-8’) print(f"💾 已写入 → {output_path}“) self.render_from_template( “templates/clm_macros.h.j2”, context_tables, os.path.join(self.output_dir, “clm_macros.h”) ) # === manifest:提取所有在 TX 表中实际使用的 RANGE 宏(去重)=== used_range_macros = sorted(set(entry[“range_macro”] for entry in self.tx_limit_entries)) manifest_data = { “timestamp”: datetime.now().isoformat(), “used_ranges”: used_range_macros, “count”: len(used_range_macros) } manifest_path = os.path.join(self.output_dir, “generated_ranges_manifest.json”) with open(manifest_path, ‘w’, encoding=‘utf-8’) as f: json.dump(manifest_data, f, indent=4, ensure_ascii=False) print(f”✅ 已生成精简 manifest 文件: {manifest_path}“) print(f"📊 共 {len(used_range_macros)} 个唯一 RANGE 宏被使用:”) for macro in used_range_macros: print(f" - {macro}“) # ✅ 暴露 manifest 路径,供外部 UI 或模块使用 self.last_generated_manifest = manifest_path print(”✅ 所有输出文件生成完成。“) self.save_channel_set_map_to_config() def render_from_template_string(self, template_path, context): “”“仅渲染模板为字符串,不写入文件””" import jinja2 env = jinja2.Environment(loader=jinja2.FileSystemLoader(searchpath=“.”)) template = env.get_template(template_path) return template.render(**context) def log_changes_to_file(self, changes, output_dir, locale_id, total_entries): “”“将变更摘要写入日志文件”“” log_dir = Path(output_dir) / “change_logs” log_dir.mkdir(exist_ok=True) # ✅ 使用时间戳生成唯一文件名 timestamp_str = datetime.now().strftime("%Y%m%d%H%M%S") log_path = log_dir / f"change{timestamp_str}.log" timestamp = datetime.now().strftime(“%Y-%m-%d %H:%M:%S”) with open(log_path, ‘w’, encoding=‘utf-8’) as f: # 覆盖写入最新变更 f.write(f"\n") f.write(f"CLM 变更日志\n") f.write(f"\n") f.write(f"时间: {timestamp}\n") f.write(f"地区码: {locale_id}\n") f.write(f"总 TX 条目数: {total_entries}\n") f.write(f"\n") if not any(changes.values()): f.write(“🟢 本次运行无任何变更,所有文件已是最新状态。\n”) else: if changes[‘added_ranges’]: f.write(f"🟢 新增 RANGE ({len(changes[‘added_ranges’])}):\n") for r in sorted(changes[‘added_ranges’]): f.write(f" → {r}\n") f.write(f"\n") if changes[‘removed_ranges’]: f.write(f"🔴 删除 RANGE ({len(changes[‘removed_ranges’])}):\n") for r in sorted(changes[‘removed_ranges’]): f.write(f" → {r}\n") f.write(f"\n") if changes[‘modified_ranges’]: f.write(f"🟡 修改 RANGE ({len(changes[‘modified_ranges’])}):\n") for r in sorted(changes[‘modified_ranges’]): f.write(f" → {r}\n") f.write(f"\n") other_adds = changes[‘other_additions’] other_dels = changes[‘other_deletions’] if other_adds or other_dels: f.write(f"🔵 其他变更:\n") for line in other_adds[:10]: f.write(f" ➕ {line}\n") for line in other_dels[:10]: f.write(f" ➖ {line}\n") if len(other_adds) > 10 or len(other_dels) > 10: f.write(f" … 还有 {len(other_adds) + len(other_dels) - 20} 处未显示\n") f.write(f"\n") f.write(f"📌 输出目录: {output_dir}\n") f.write(f"备份文件: {Path(self.target_c_file).with_suffix(‘.c.bak’)}\n") f.write(f"========================================\n") print(f"📄 已保存变更日志 → {log_path}“) def save_channel_set_map_to_config(self): “”“将当前 channel_set_map 写回 config.json 的 channel_set_map 字段””" try: # 🔽 清理:只保留 fallback 类型的 RANGE(可正则匹配) valid_keys = [ k for k in self.channel_set_map.keys() if re.match(r’RANGE[\dA-Z]+\d+M\d+_\d+’, k) # 如 RANGE_2G_20M_1_11 ] filtered_map = {k: v for k, v in self.channel_set_map.items() if k in valid_keys} # 🔼 更新主配置中的字段 self.config[“channel_set_map”] = filtered_map # 使用过滤后的版本 with open(self.config_file_path, ‘w’, encoding=‘utf-8’) as f: json.dump(self.config, f, indent=4, ensure_ascii=False) print(f"💾 已成功将精简后的 channel_set_map 写回配置文件: {filtered_map}“) except Exception as e: print(f”❌ 写入配置文件失败: {e}“) raise def convert(self, file_path): # =============== 🔐 每次都更新备份 C 文件 =============== c_source = Path(self.target_c_file) c_backup = c_source.with_suffix(c_source.suffix + “.bak”) if not c_source.exists(): raise FileNotFoundError(f"目标 C 文件不存在: {c_source}”) ext = os.path.splitext(file_path)[-1].lower() if ext == “.xlsx”: wb = load_workbook(file_path, data_only=True) sheets = [{“sheet”: ws, “format”: “xlsx”} for ws in wb.worksheets] elif ext == “.xls”: wb = xlrd.open_workbook(file_path) sheets = [{“sheet”: ws, “format”: “xls”} for ws in wb.sheets()] else: raise ValueError(“仅支持 .xls 或 .xlsx 文件”) for i, ws_obj in enumerate(sheets): sheet_name = wb.sheet_names()[i] if ext == “.xls” else ws_obj[“sheet”].title config = self.match_sheet_to_config(sheet_name) if config: self.convert_sheet_with_config(ws_obj, sheet_name, config) self.generate_outputs() def parse_excel(self): “”" 【UI 兼容】供 PyQt UI 调用的入口方法 将当前 self.input_file 中的数据解析并填充到 tx_limit_entries “”" if not hasattr(self, ‘input_file’) or not self.input_file: raise ValueError(“未设置 input_file 属性!”) if not os.path.exists(self.input_file): raise FileNotFoundError(f"文件不存在: {self.input_file}“) print(f"📊 开始解析 Excel 文件: {self.input_file}”) try: self.convert(self.input_file) # 调用已有逻辑 print(f"✅ Excel 解析完成,共生成 {len(self.tx_limit_entries)} 条 TX 限幅记录") except Exception as e: print(f"❌ 解析失败: {e}") raise
if name == “main”:
import argparse
import os
切换到脚本所在目录 script_dir = os.path.dirname(file) os.chdir(script_dir) # 定义命令行参数解析器 parser = argparse.ArgumentParser(description=“Convert Excel to CLM C code.”) parser.add_argument( “input”, nargs=“?”, default=“input/Archer BE900US 2.xlsx”, help=“Input Excel file (default: input/Archer BE900US 2.xlsx)” ) parser.add_argument( “–config”, default=“config/config.json”, help=“Path to config.json (default: config/config.json)” ) parser.add_argument( “–output-dir”, default=“output”, help=“Output directory (default: output)” ) parser.add_argument( “–locale-id”, default=None, help=‘Locale ID, e.g., “US”, “CN-2G” (default: from config or “DEFAULT”)’ ) parser.add_argument( “–display-name”, default=None, help=‘Display name in generated code, e.g., “FCC_Core” (default: derived from locale_id)’ ) args = parser.parse_args() # 创建转换器实例,并传入所有参数 converter = ExcelToCLMConverter( config_path=args.config, output_dir=args.output_dir, locale_id=args.locale_id, locale_display_name=args.display_name ) # 执行转换 converter.convert(args.input)排个版发给我,不要修改任何东西