VariableReplacer.java
- /*
- * Copyright 2010-2023 the original author or authors.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * https://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- package org.apache.ibatis.migration;
- import java.util.Arrays;
- import java.util.Collections;
- import java.util.List;
- import java.util.Map;
- import java.util.Objects;
- import java.util.stream.Collectors;
- /**
- * @author Clinton Begin
- */
- public class VariableReplacer {
- private static final String OPEN_TOKEN = "${";
- private static final String CLOSE_TOKEN = "}";
- private final List<Map<? extends Object, ? extends Object>> variablesList;
- public VariableReplacer(Map<? extends Object, ? extends Object> variablesList) {
- this(Arrays.asList(variablesList));
- }
- public VariableReplacer(List<Map<? extends Object, ? extends Object>> variablesList) {
- this.variablesList = variablesList == null ? Collections.emptyList()
- : variablesList.stream().filter(Objects::nonNull).collect(Collectors.toList());
- }
- public String replace(String text) {
- if (text == null || text.isEmpty()) {
- return "";
- }
- // search open token
- int start = text.indexOf(OPEN_TOKEN);
- if (start == -1) {
- return text;
- }
- char[] src = text.toCharArray();
- int offset = 0;
- final StringBuilder builder = new StringBuilder();
- StringBuilder expression = null;
- while (start > -1) {
- if (start > 0 && src[start - 1] == '\\') {
- // this open token is escaped. remove the backslash and continue.
- builder.append(src, offset, start - offset - 1).append(OPEN_TOKEN);
- offset = start + OPEN_TOKEN.length();
- } else {
- // found open token. let's search close token.
- if (expression == null) {
- expression = new StringBuilder();
- } else {
- expression.setLength(0);
- }
- builder.append(src, offset, start - offset);
- offset = start + OPEN_TOKEN.length();
- int end = text.indexOf(CLOSE_TOKEN, offset);
- while (end > -1) {
- if (end <= offset || src[end - 1] != '\\') {
- expression.append(src, offset, end - offset);
- break;
- }
- // this close token is escaped. remove the backslash and continue.
- expression.append(src, offset, end - offset - 1).append(CLOSE_TOKEN);
- offset = end + CLOSE_TOKEN.length();
- end = text.indexOf(CLOSE_TOKEN, offset);
- }
- if (end == -1) {
- // close token was not found.
- builder.append(src, start, src.length - start);
- offset = src.length;
- } else {
- appendWithReplace(builder, expression.toString());
- offset = end + CLOSE_TOKEN.length();
- }
- }
- start = text.indexOf(OPEN_TOKEN, offset);
- }
- if (offset < src.length) {
- builder.append(src, offset, src.length - offset);
- }
- return builder.toString();
- }
- private StringBuilder appendWithReplace(StringBuilder builder, String key) {
- String value = null;
- for (Map<? extends Object, ? extends Object> variables : variablesList) {
- value = (String) variables.get(key);
- if (value != null) {
- builder.append(value);
- break;
- }
- }
- if (value == null) {
- builder.append(OPEN_TOKEN).append(key).append(CLOSE_TOKEN);
- }
- return builder;
- }
- }