io.micronaut.flyway.StaticResourceProvider Maven / Gradle / Ivy
/*
* Copyright 2017-2023 original 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 io.micronaut.flyway;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.Reader;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.nio.charset.StandardCharsets;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Enumeration;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import io.micronaut.context.exceptions.ConfigurationException;
import io.micronaut.core.annotation.Internal;
import io.micronaut.core.annotation.NonNull;
import io.micronaut.core.annotation.Nullable;
import org.flywaydb.core.api.ResourceProvider;
import org.flywaydb.core.api.resource.LoadableResource;
import org.flywaydb.core.internal.util.StringUtils;
import org.graalvm.nativeimage.ImageSingletons;
/**
* Internal class for resolving migrations at build time with GraalVM.
*/
@Internal
final class StaticResourceProvider implements ResourceProvider {
private static final String FLYWAY_LOCATIONS = "flyway.locations";
private static final String DEFAULT_FLYWAY_LOCATIONS = "classpath:db/migration";
private static final String CLASSPATH_APPLICATION_MIGRATIONS_PROTOCOL = "classpath";
private static final String JAR_APPLICATION_MIGRATIONS_PROTOCOL = "jar";
private static final String FILE_APPLICATION_MIGRATIONS_PROTOCOL = "file";
private final List resources;
private StaticResourceProvider(List resources) {
this.resources = resources;
}
static StaticResourceProvider get() {
return findStaticResourceProvider();
}
static void install(ClassLoader classLoader) {
if (hasImageSingletons()) {
StaticResourceProvider staticResourceProvider = create(classLoader);
ImageSingletons.add(StaticResourceProvider.class, staticResourceProvider);
}
}
static StaticResourceProvider create(ClassLoader classLoader) {
List locations = Stream
.of(System.getProperty(FLYWAY_LOCATIONS, DEFAULT_FLYWAY_LOCATIONS).split(","))
.toList();
try {
List resources = discoverApplicationMigrations(locations, classLoader);
return new StaticResourceProvider(resources);
} catch (IOException | URISyntaxException e) {
throw new ConfigurationException("Error loading Flyway migrations: " + e.getMessage(), e);
}
}
@Override
public LoadableResource getResource(String name) {
return resources.stream()
.filter(resource -> resource.getAbsolutePath().equals(name))
.findFirst()
.orElse(null);
}
@Override
public Collection getResources(String prefix, String[] suffixes) {
return resources.stream()
.filter(resource -> StringUtils.startsAndEndsWith(resource.getFilename(), prefix, suffixes))
.map(LoadableResource.class::cast)
.toList();
}
@Nullable
private static StaticResourceProvider findStaticResourceProvider() {
return hasImageSingletons() && ImageSingletons.contains(StaticResourceProvider.class) ?
ImageSingletons.lookup(StaticResourceProvider.class) : null;
}
@SuppressWarnings("java:S1181")
private static boolean hasImageSingletons() {
try {
//noinspection ConstantValue
return ImageSingletons.class != null;
} catch (Throwable e) {
// not present or not a GraalVM JDK
return false;
}
}
private static List discoverApplicationMigrations(List locations, ClassLoader classLoader) throws IOException, URISyntaxException {
List applicationMigrationResources = new ArrayList<>();
// Locations can be a comma separated list
for (String location : locations) {
// Strip any 'classpath:' protocol prefixes because they are assumed
// but not recognized by ClassLoader.getResources()
if (location != null && location.startsWith(CLASSPATH_APPLICATION_MIGRATIONS_PROTOCOL + ':')) {
location = location.substring(CLASSPATH_APPLICATION_MIGRATIONS_PROTOCOL.length() + 1);
}
Enumeration migrations = classLoader.getResources(location);
while (migrations.hasMoreElements()) {
URL path = migrations.nextElement();
applicationMigrations(path, location)
.ifPresent(applicationMigrationResources::addAll);
}
}
return applicationMigrationResources;
}
private static Optional> applicationMigrations(@NonNull URL path, String location) throws URISyntaxException, IOException {
if (JAR_APPLICATION_MIGRATIONS_PROTOCOL.equals(path.getProtocol())) {
try (FileSystem fileSystem = initFileSystem(path.toURI())) {
return Optional.of(getApplicationMigrationsFromPath(location, path));
}
} else if (FILE_APPLICATION_MIGRATIONS_PROTOCOL.equals(path.getProtocol())) {
return Optional.of(getApplicationMigrationsFromPath(location, path));
}
return Optional.empty();
}
private static Set getApplicationMigrationsFromPath(final String location, final URL path)
throws IOException, URISyntaxException {
try (Stream pathStream = Files.walk(Paths.get(path.toURI()))) {
return pathStream.filter(Files::isRegularFile)
// we don't want windows paths here since the paths are going to be used as classpath paths anyway
.map(it -> new StaticLoadableResource(location, it))
.collect(Collectors.toSet());
}
}
private static FileSystem initFileSystem(final URI uri) throws IOException {
return FileSystems.newFileSystem(uri, Collections.singletonMap("create", "true"));
}
static final class StaticLoadableResource extends LoadableResource {
private final String location;
private final String absolutePath;
private final String fileName;
private final byte[] bytes;
StaticLoadableResource(String location, Path it) {
byte[] readBytes;
this.location = location;
this.absolutePath = it.toAbsolutePath().toString();
this.fileName = it.getFileName().toString();
try {
readBytes = Files.readAllBytes(it);
} catch (IOException e) {
readBytes = new byte[0];
}
this.bytes = readBytes;
}
@Override
public Reader read() {
return new InputStreamReader(new ByteArrayInputStream(bytes), StandardCharsets.UTF_8);
}
@Override
public String getAbsolutePath() {
return absolutePath;
}
@Override
public String getAbsolutePathOnDisk() {
return getAbsolutePath();
}
@Override
public String getFilename() {
return fileName;
}
@Override
public String getRelativePath() {
return location + File.pathSeparator + fileName;
}
@Override
public String toString() {
return "StaticLoadableResource{" +
"location='" + location + '\'' +
", absolutePath='" + absolutePath + '\'' +
", fileName='" + fileName + '\'' +
'}';
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy