通过Java修改consul配置(保留注释、保留缩进)
直接上代码了,找了很久也没找到保留注释的三方包,snakeyaml 的缩进一直也有问题,就自己写了个正则方式的
consul也没有相关接口,只接受整个的
传key和value,替换相应value值,
大佬有更好的方法希望能告诉我
1 2 3 4 5 | <dependency> <groupId>com.orbitz.consul</groupId> <artifactId>consul-client</artifactId> <version> 1.5 . 3 </version> </dependency> |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 | String regex = buildRegexForKey(keyToFind); Pattern pattern = Pattern.compile(regex, Pattern.MULTILINE | Pattern.DOTALL); Matcher matcher = pattern.matcher(yamlContent); if (matcher.find()){ String content = matcher.group( 0 ); String targetLine = this .getLine(keyToFind, content); if ( "" .equals(targetLine)) { // 按原值查询 result = this .findByOriginal( yamlContent, keyToFind, newValue, result ); replacement1 = result.getReplacement1(); replacement2 = result.getReplacement2(); addFlag = result.isAddFlag(); } else { String[] parts = keyToFind.split( "\\." ); String param = parts[parts.length - 1 ]; Pair<String, String> pair = this .handleTargetLine(targetLine, param); // 获取到左右部分,可以赋值 replacement1 = targetLine; replacement2 = pair.getKey() + " " + newValue; } } else { // 按原值查询 result = this .findByOriginal( yamlContent, keyToFind, newValue, result ); replacement1 = result.getReplacement1(); replacement2 = result.getReplacement2(); addFlag = result.isAddFlag(); } private String buildRegexForKey(String key) { String[] parts = key.split( "\\." ); StringBuilder regexBuilder = new StringBuilder(); // 匹配每个部分的键和冒号,考虑到YAML的缩进 for ( int i = 0 ; i < parts.length; i++) { if (i == parts.length - 1 ) { // 如果不是第一个部分,添加换行符和缩进 regexBuilder.append( "\\b" ).append(parts[i]).append( ":\\s*" ); } else { regexBuilder.append( "\\b" ).append(parts[i]).append( ":\\s*\\n.*?" ); } } // regexBuilder.append("[^\\s]+"); // 匹配值直到下一个空白字符或换行符 regexBuilder.append( "([^\\n]*)" ); // 匹配值直到下一个换行符 return regexBuilder.toString(); } private String getLine(String key, String yamlContent) { String returnStr = "" ; String[] lines = yamlContent.split( "\n" ); String[] keyParts = key.split( "\\." ); int length = keyParts.length; String lastKey = keyParts[length - 1 ]; // 步骤1: 找到 'springlm.datasourcelm' 所在行的下面一行 String parentKey; if (length >= 2 ) { parentKey = keyParts[length - 2 ]; } else { parentKey = keyParts[length - 1 ]; } String nextLine = null ; int indentSpacesCount = 0 ; for ( int i = 0 ; i < lines.length; i++) { String line = lines[i].trim(); if (line.startsWith(parentKey + ":" )) { // 确保下一行不是空行或注释 String candidateLine = (i + 1 < lines.length) ? lines[i + 1 ] : "" ; if (!candidateLine.isEmpty() && !candidateLine.startsWith( "#" )) { nextLine = candidateLine; break ; } } } // 如果没有找到 下一行,则返回null if (nextLine == null ) { return returnStr; } // 步骤2: 获取下一行的空格数量 indentSpacesCount = countLeadingSpaces(nextLine); // 步骤3: 构建用于匹配目标行的空格字符串 String indentSpaces = buildSpaces(indentSpacesCount); // 步骤4: 根据空格数量和最后一个.后面的key,获取这个key的所在行 for (String line : lines) { if (line.startsWith(indentSpaces + lastKey + ":" )) { return line; } } return returnStr; // 如果没有找到匹配的行,返回null } // 辅助方法,用于计算字符串中前导空格的数量 private int countLeadingSpaces(String line) { int count = 0 ; while (count < line.length() && line.charAt(count) == ' ' ) { count++; } return count; } // 辅助方法,用于构建由空格组成的字符串 private String buildSpaces( int count) { StringBuilder sb = new StringBuilder(); for ( int i = 0 ; i < count; i++) { sb.append( ' ' ); } return sb.toString(); } public Pair<String, String> handleTargetLine(String line, String key) { String regex = "(\\s*" + key + ":)\\s+(.+)" ; Pattern pattern = Pattern.compile(regex); Matcher matcher = pattern.matcher(line); if (matcher.find()) { String prefix = matcher.group( 1 ); // "urllm:" 及其前面的空格 String url = matcher.group( 2 ); // URL 的其余部分 return new Pair<>(prefix, url); } else { return null ; // 或者抛出一个异常,取决于您的错误处理策略 } } public UpdateResultDto findByOriginal( String yamlContent, String keyToFind, String newValue, UpdateResultDto result ) { // String pathRegex = "(\\b" + keyToFind + ")\\s*:\\s*([^\\n]+)"; String pathRegex = "^(" + keyToFind + ")\\s*:\\s*([^\n]+)" ; Pattern pathPattern = Pattern.compile(pathRegex, Pattern.MULTILINE); Matcher pathMatcher = pathPattern.matcher(yamlContent); String replacement1 = null ; String replacement2 = null ; boolean addFlag = false ; if (pathMatcher.find()) { replacement1 = pathMatcher.group( 0 ); // 匹配到的整个键值对 replacement2 = keyToFind + ": " + newValue; // 新的键值对 } else { // 如果没有匹配到,说明是新增 addFlag = true ; } result.setReplacement1(replacement1); result.setReplacement2(replacement2); result.setAddFlag(addFlag); return result; } |
By:努力向前的菜B
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
2020-04-18 记第一个项目结束时的感想