Skip to content

Commit

Permalink
Merge pull request #294 from senacor/bugfix/#293_trailing-newlines-wi…
Browse files Browse the repository at this point in the history
…ll-no-longer-be-removed-from-migration-scripts

bugfix #293: trailing newlines will no longer be removed from migrati…
  • Loading branch information
xtermi2 authored May 2, 2024
2 parents c4dc633 + 3d80bc4 commit 4d059d6
Show file tree
Hide file tree
Showing 14 changed files with 168 additions and 66 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/maven-matrix.yml
Original file line number Diff line number Diff line change
Expand Up @@ -40,8 +40,8 @@ jobs:
build-with-es:
strategy:
matrix:
elasticsearchVersion: [ "8.12.2", "8.11.4", "8.10.4", "8.9.2", "8.8.2", "8.7.1", "8.6.2", "8.5.3", "8.4.3", "8.3.3", "8.2.3", "8.1.3", "8.0.1",
"7.17.18", "7.5.2" ]
elasticsearchVersion: [ "8.13.0", "8.12.2", "8.11.4", "8.10.4", "8.9.2", "8.8.2", "8.7.1", "8.6.2", "8.5.3", "8.4.3", "8.3.3", "8.2.3", "8.1.3", "8.0.1",
"7.17.20", "7.5.2" ]
fail-fast: false
runs-on: ubuntu-22.04
steps:
Expand Down
13 changes: 5 additions & 8 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ Successful executed migration scripts will not be executed again!

- tested on Java 8, 11, 17, and 21
- runs on Spring-Boot 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7, 3.0, 3.1 and 3.2 (and of course without Spring-Boot)
- runs on Elasticsearch version 7.5.x - 8.12.x
- runs on Elasticsearch version 7.5.x - 8.13.x
- runs on Opensearch version 1.x and 2.x
- highly configurable (e.g. location(s) of your migration files, migration files format pattern)
- placeholder substitution in migration scripts
Expand All @@ -33,16 +33,11 @@ Successful executed migration scripts will not be executed again!

| Compatibility | Spring Boot | Elasticsearch | Opensearch |
|----------------------------------|--------------------------------------------------|----------------------|------------|
| elasticsearch-evolution >= 0.4.2 | 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7, 3.0, 3.1, 3.2 | 7.5.x - 8.12.x | 1.x - 2.x |
| elasticsearch-evolution >= 0.4.2 | 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7, 3.0, 3.1, 3.2 | 7.5.x - 8.13.x | 1.x - 2.x |
| elasticsearch-evolution >= 0.4.0 | 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7 | 7.5.x - 8.6.x | 1.x - 2.x |
| elasticsearch-evolution 0.3.x | 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7 | 7.5.x - 7.17.x | |
| elasticsearch-evolution 0.2.x | 1.5, 2.0, 2.1, 2.2, 2.3, 2.4 | 7.0.x - 7.4.x, 6.8.x | |

NOTE: When you run on Java 11 and using spring-boot 2.2 or 2.3 and you hit [this issue](https://github.com/ronmamo/reflections/issues/279), you have 2 options:

- downgrade to `org.reflections:reflections:0.9.11` via defining this dependency explicitly
- use spring boot 2.3.6+

## 3 Quickstart

### 3.1 Quickstart with Spring-Boot starter
Expand Down Expand Up @@ -293,7 +288,9 @@ ElasticsearchEvolution.configure()

### v0.5.2-SNAPSHOT

- ...
- bugfix ([#293](https://github.com/senacor/elasticsearch-evolution/issues/293)): trailing newlines will no longer be removed from migration scripts.
- added regression tests against OpenSearch 2.13
- added regression tests against ElasticSearch 8.13

### v0.5.1

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,8 @@ protected MigrationScriptParser createMigrationScriptParser() {
getConfig().getPlaceholders(),
getConfig().getPlaceholderPrefix(),
getConfig().getPlaceholderSuffix(),
getConfig().isPlaceholderReplacement()
getConfig().isPlaceholderReplacement(),
getConfig().getLineSeparator()
);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,6 @@
import com.senacor.elasticsearch.evolution.core.internal.model.migration.ParsedMigrationScript;
import com.senacor.elasticsearch.evolution.core.internal.model.migration.RawMigrationScript;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.StringReader;
import java.util.Collection;
import java.util.List;
import java.util.Map;
Expand All @@ -21,7 +18,6 @@

import static com.senacor.elasticsearch.evolution.core.internal.utils.AssertionUtils.requireCondition;
import static com.senacor.elasticsearch.evolution.core.internal.utils.AssertionUtils.requireNotBlank;
import static java.lang.System.lineSeparator;
import static java.util.Objects.requireNonNull;

/**
Expand All @@ -39,6 +35,7 @@ public class MigrationScriptParserImpl implements MigrationScriptParser {
private final String placeholderPrefix;
private final String placeholderSuffix;
private final boolean placeholderReplacement;
private final String lineSeparator;

/**
* create Parser
Expand All @@ -48,13 +45,15 @@ public MigrationScriptParserImpl(String esMigrationPrefix,
Map<String, String> placeholders,
String placeholderPrefix,
String placeholderSuffix,
boolean placeholderReplacement) {
boolean placeholderReplacement,
String lineSeparator) {
this.esMigrationPrefix = esMigrationPrefix;
this.esMigrationSuffixes = esMigrationSuffixes;
this.placeholders = placeholders;
this.placeholderPrefix = placeholderPrefix;
this.placeholderSuffix = placeholderSuffix;
this.placeholderReplacement = placeholderReplacement;
this.lineSeparator = lineSeparator;
}

@Override
Expand All @@ -79,36 +78,31 @@ private MigrationScriptRequest parseContent(RawMigrationScript script) {
: script.getContent();
MigrationScriptRequest res = new MigrationScriptRequest();

try (BufferedReader reader = new BufferedReader(new StringReader(contentReplaced))) {
final AtomicReference<ParseState> state = new AtomicReference<>(ParseState.METHOD_PATH);
reader.lines()
// filter out comment lines
.filter(line -> !line.trim().startsWith("#") && !line.trim().startsWith("//"))
.forEachOrdered(line -> {
switch (state.get()) {
case METHOD_PATH:
parseMethodWithPath(res, line);
state.set(ParseState.HEADER);
break;
case HEADER:
if (line.trim().isEmpty()) {
state.set(ParseState.CONTENT);
} else {
parseHeader(res, line);
}
break;
case CONTENT:
if (!res.isBodyEmpty()) {
res.addToBody(lineSeparator());
}
res.addToBody(line);
break;
default:
throw new UnsupportedOperationException("state '" + state + "' not supportet");
final AtomicReference<ParseState> state = new AtomicReference<>(ParseState.METHOD_PATH);
for (String line : contentReplaced.split(lineSeparator, -1)) {
if (!line.trim().startsWith("#") && !line.trim().startsWith("//")) {
switch (state.get()) {
case METHOD_PATH:
parseMethodWithPath(res, line);
state.set(ParseState.HEADER);
break;
case HEADER:
if (line.trim().isEmpty()) {
state.set(ParseState.CONTENT);
} else {
parseHeader(res, line);
}
});
} catch (IOException e) {
throw new MigrationException("failed parsing content of " + script.getFileName(), e);
break;
case CONTENT:
if (!res.isBodyEmpty()) {
res.addToBody(lineSeparator);
}
res.addToBody(line);
break;
default:
throw new UnsupportedOperationException("state '" + state + "' not supportet");
}
}
}

return res;
Expand All @@ -125,7 +119,7 @@ private void parseHeader(MigrationScriptRequest res, String line) {
}

private void parseMethodWithPath(MigrationScriptRequest res, String line) {
String[] methodAndPath = line.trim().split("[ ]+", 2);
String[] methodAndPath = line.trim().split(" +", 2);
if (methodAndPath.length != 2) {
throw new MigrationException(String.format(
"can't parse method and path: '%s'. Method and path must be separated by space and should look like this: 'PUT /my_index'",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -151,10 +151,15 @@ private Stream<RawMigrationScript> readScriptsFromClassPath(String location) {
return res.stream();
}

private Stream<RawMigrationScript> read(BufferedReader reader, String filename) {
String content = reader.lines()
// use static line separator ('\n' per default) to get predictable and system independent checksum later
.collect(Collectors.joining(lineSeparator));
Stream<RawMigrationScript> read(BufferedReader reader, String filename) throws IOException {
StringBuilder sb = new StringBuilder();
int ch;
while ((ch = reader.read()) != -1) {
sb.append((char) ch);
}
// use static line separator ('\n' per default) to get predictable and system independent checksum later
String content = sb.toString().replaceAll("\\R", lineSeparator);

if (content.isEmpty()) {
return Stream.empty();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,10 +63,10 @@ void migrate_OK(String versionInfo, EsUtils esUtils, RestHighLevelClient restHig
assertSoftly(softly -> {
softly.assertThat(underTest.migrate())
.as("# of successful executed scripts ")
.isEqualTo(7);
.isEqualTo(8);
softly.assertThat(historyRepository.findAll())
.as("# of historyIndex entries and all are successful")
.hasSize(7)
.hasSize(8)
.allMatch(MigrationScriptProtocol::isSuccess);
});
esUtils.refreshIndices();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ void migrate_historyMaxQuerySizeToLow() throws IOException {

assertThatThrownBy(underTest::migrate)
.isInstanceOf(MigrationException.class)
.hasMessage("configured historyMaxQuerySize of '%s' is to low for the number of migration scripts of '%s'", historyMaxQuerySize, 7);
.hasMessage("configured historyMaxQuerySize of '%s' is to low for the number of migration scripts of '%s'", historyMaxQuerySize, 8);

InOrder order = inOrder(restHighLevelClient, restClient);
order.verify(restHighLevelClient, times(2)).getLowLevelClient();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,8 @@ void withOneReplacement_isReplaced() {
Maps.newHashMap("index", "myIndex"),
"${",
"}",
true);
true,
"\n");

String replaced = parser.replaceParams(template);

Expand All @@ -51,7 +52,8 @@ void withMultipleReplacementsOfSameKey_isReplaced() {
Maps.newHashMap("index", "myIndex"),
"${",
"}",
true);
true,
"\n");

String replaced = parser.replaceParams(template);

Expand All @@ -68,7 +70,8 @@ void withMultipleReplacementsOfDifferentKey_isReplaced() {
placeholders,
"${",
"}",
true);
true,
"\n");

String replaced = parser.replaceParams(template);

Expand All @@ -85,7 +88,8 @@ class parseFilename {
null,
null,
null,
false);
false,
"\n");

@Test
void isParsable() {
Expand Down Expand Up @@ -147,7 +151,8 @@ class parseCollection {
null,
null,
null,
false);
false,
"\n");

@Test
void success() {
Expand Down Expand Up @@ -186,7 +191,8 @@ class parseSingle {
null,
null,
null,
false);
false,
"\n");

@Test
void success_withMethodAndPathAndHeaderAndBody() {
Expand Down Expand Up @@ -358,7 +364,8 @@ void success_replacePlaceholders() {
placeholders,
"${",
"}",
true)
true,
"\n")
.parse(new RawMigrationScript()
.setFileName(fileName)
.setContent(defaultContent));
Expand Down Expand Up @@ -391,6 +398,55 @@ void success_replacePlaceholders() {
});
}

@Test
void success_NotRemoveTrailingNewlines() {
String defaultContent = "PUT /my-index" + lineSeparator() +
lineSeparator() +
"{" + lineSeparator() + "\"index\":\"my-index\"" + lineSeparator() + "}" +
lineSeparator();
String fileName = "V1__create.http";
HashMap<String, String> placeholders = new HashMap<>();

ParsedMigrationScript res = new MigrationScriptParserImpl(
"V",
Collections.singletonList(".http"),
placeholders,
"${",
"}",
false,
"\n")
.parse(new RawMigrationScript()
.setFileName(fileName)
.setContent(defaultContent));

assertSoftly(softly -> {
softly.assertThat(res.getChecksum())
.as("Checksum")
.isEqualTo(defaultContent.hashCode());
softly.assertThat(res.getFileNameInfo().getVersion())
.as("version")
.isEqualTo(MigrationVersion.fromVersion("1"));
softly.assertThat(res.getFileNameInfo().getScriptName())
.as("scriptName")
.isEqualTo(fileName);
softly.assertThat(res.getFileNameInfo().getDescription())
.as("description")
.isEqualTo("create");
softly.assertThat(res.getMigrationScriptRequest().getHttpMethod())
.as("methot")
.isEqualTo(HttpMethod.PUT);
softly.assertThat(res.getMigrationScriptRequest().getPath())
.as("path")
.isEqualTo("/my-index");
softly.assertThat(res.getMigrationScriptRequest().getHttpHeader())
.as("header")
.isEmpty();
softly.assertThat(res.getMigrationScriptRequest().getBody())
.as("body")
.isEqualTo("{" + lineSeparator() + "\"index\":\"my-index\"" + lineSeparator() + "}" + lineSeparator());
});
}

@Test
void failed_MethodAndPathInvalid() {
String defaultContent = "/";
Expand Down
Loading

0 comments on commit 4d059d6

Please sign in to comment.