All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.talend.sdk.component.server.service.VirtualDependenciesService Maven / Gradle / Ivy

/**
 * Copyright (C) 2006-2021 Talend Inc. - www.talend.com
 *
 * 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
 *
 * http://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.talend.sdk.component.server.service;

import static java.util.Collections.emptyMap;
import static java.util.Collections.emptySet;
import static java.util.Collections.unmodifiableSet;
import static java.util.Locale.ROOT;
import static java.util.Optional.ofNullable;
import static java.util.function.Function.identity;
import static java.util.stream.Collectors.joining;
import static java.util.stream.Collectors.toList;
import static java.util.stream.Collectors.toMap;
import static lombok.AccessLevel.PACKAGE;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.io.StringReader;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardCopyOption;
import java.nio.file.StandardOpenOption;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Supplier;
import java.util.jar.Attributes;
import java.util.jar.JarEntry;
import java.util.jar.JarOutputStream;
import java.util.jar.Manifest;
import java.util.stream.Stream;

import javax.annotation.PostConstruct;
import javax.enterprise.context.ApplicationScoped;
import javax.enterprise.inject.spi.CDI;
import javax.inject.Inject;

import org.talend.sdk.component.api.service.configuration.LocalConfiguration;
import org.talend.sdk.component.classloader.ConfigurableClassLoader;
import org.talend.sdk.component.container.Container;
import org.talend.sdk.component.dependencies.maven.Artifact;
import org.talend.sdk.component.path.PathFactory;
import org.talend.sdk.component.runtime.manager.ComponentManager;
import org.talend.sdk.component.server.configuration.ComponentServerConfiguration;

import lombok.Getter;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;

@Slf4j
@ApplicationScoped
public class VirtualDependenciesService {

    @Getter(PACKAGE)
    private final String virtualGroupId = "virtual.talend.component.server.generated.";

    @Getter(PACKAGE)
    private final String configurationArtifactIdPrefix = "user-local-configuration-";

    private final Enrichment noCustomization = new Enrichment(false, null, null, null);

    @Inject
    private ComponentServerConfiguration configuration;

    private final Map enrichmentsPerContainer = new HashMap<>();

    @Getter
    private final Map artifactMapping = new ConcurrentHashMap<>();

    private final Map> configurationArtifactMapping = new ConcurrentHashMap<>();

    private Path provisioningM2Base;

    @PostConstruct
    private void init() {
        final String m2 = configuration.getUserExtensionsAutoM2Provisioning();
        switch (m2) {
        case "skip":
            provisioningM2Base = null;
            break;
        case "auto":
            provisioningM2Base = findStudioM2();
            break;
        default:
            provisioningM2Base = PathFactory.get(m2);
        }
        log.debug("m2 provisioning base: {}", provisioningM2Base);
    }

    public void onDeploy(final String pluginId) {
        if (!configuration.getUserExtensions().isPresent()) {
            enrichmentsPerContainer.put(pluginId, noCustomization);
            return;
        }
        final Path extensions = PathFactory
                .get(configuration.getUserExtensions().orElseThrow(IllegalArgumentException::new))
                .resolve(pluginId);
        if (!Files.exists(extensions)) {
            log.debug("'{}' does not exist so no extension will be added to family '{}'", extensions, pluginId);
            enrichmentsPerContainer.put(pluginId, noCustomization);
            return;
        }

        final Path userConfig = extensions.resolve("user-configuration.properties");
        final Map userJars = findJars(extensions, pluginId);
        final Properties userConfiguration = loadUserConfiguration(pluginId, userConfig, userJars);
        if (userConfiguration.isEmpty() && userJars.isEmpty()) {
            log.debug("No customization for container '{}'", pluginId);
            enrichmentsPerContainer.put(pluginId, noCustomization);
            return;
        }

        final Map customConfigAsMap = userConfiguration
                .stringPropertyNames()
                .stream()
                .collect(toMap(identity(), userConfiguration::getProperty));
        log
                .debug("Set up customization for container '{}' (has-configuration={}, jars={})", pluginId,
                        !userConfiguration.isEmpty(), userJars);

        if (userConfiguration.isEmpty()) {
            enrichmentsPerContainer.put(pluginId, new Enrichment(true, customConfigAsMap, null, userJars.keySet()));
        } else {
            final byte[] localConfigurationJar = generateConfigurationJar(pluginId, userConfiguration);
            final Artifact configurationArtifact;
            try {
                configurationArtifact = new Artifact(groupIdFor(pluginId), configurationArtifactIdPrefix + pluginId,
                        "jar", "", Long.toString(Files.getLastModifiedTime(userConfig).toMillis()), "compile");
            } catch (final IOException e) {
                throw new IllegalStateException(e);
            }
            doProvision(configurationArtifact, () -> new ByteArrayInputStream(localConfigurationJar));
            enrichmentsPerContainer
                    .put(pluginId, new Enrichment(true, customConfigAsMap, configurationArtifact, userJars.keySet()));
            configurationArtifactMapping
                    .put(configurationArtifact, () -> new ByteArrayInputStream(localConfigurationJar));
        }
        userJars.forEach((artifact, file) -> doProvision(artifact, () -> {
            try {
                return Files.newInputStream(file, StandardOpenOption.READ);
            } catch (final IOException e) {
                throw new IllegalStateException(e);
            }
        }));

        artifactMapping.putAll(userJars);
    }

    public void onUnDeploy(final Container plugin) {
        final Enrichment enrichment = enrichmentsPerContainer.remove(plugin.getId());
        if (enrichment == null || enrichment == noCustomization) {
            return;
        }

        if (enrichment.userArtifacts != null) {
            enrichment.userArtifacts.forEach(artifactMapping::remove);
            enrichment.userArtifacts.clear();
        }
        if (enrichment.configurationArtifact != null) {
            configurationArtifactMapping.remove(enrichment.configurationArtifact);
        }
    }

    public boolean isVirtual(final String gav) {
        return gav.startsWith(virtualGroupId);
    }

    public Enrichment getEnrichmentFor(final String pluginId) {
        return enrichmentsPerContainer.get(pluginId);
    }

    public Stream userArtifactsFor(final String pluginId) {
        final Enrichment enrichment = enrichmentsPerContainer.get(pluginId);
        if (enrichment == null || !enrichment.customized) {
            return Stream.empty();
        }
        final Stream userJars = enrichment.userArtifacts.stream();
        if (enrichment.configurationArtifact != null) {
            // config is added but will be ignored cause not physically here
            // however it ensures our rest service returns right data
            return Stream.concat(userJars, Stream.of(enrichment.configurationArtifact));
        }
        return userJars;
    }

    public Supplier retrieveArtifact(final Artifact artifact) {
        return ofNullable(artifactMapping.get(artifact)).map(it -> (Supplier) () -> {
            try {
                return Files.newInputStream(it);
            } catch (final IOException e) {
                throw new IllegalStateException(e);
            }
        }).orElseGet(() -> configurationArtifactMapping.get(artifact));
    }

    public String groupIdFor(final String family) {
        return virtualGroupId + sanitizedForGav(family);
    }

    private byte[] generateConfigurationJar(final String family, final Properties userConfiguration) {
        final ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
        final Manifest manifest = new Manifest();
        final Attributes mainAttributes = manifest.getMainAttributes();
        mainAttributes.put(Attributes.Name.MANIFEST_VERSION, "1.0");
        mainAttributes.putValue("Created-By", "Talend Component Kit Server");
        mainAttributes.putValue("Talend-Time", Long.toString(System.currentTimeMillis()));
        mainAttributes.putValue("Talend-Family-Name", family);
        try (final JarOutputStream jar = new JarOutputStream(new BufferedOutputStream(outputStream), manifest)) {
            jar.putNextEntry(new JarEntry("TALEND-INF/local-configuration.properties"));
            userConfiguration.store(jar, "Configuration of the family " + family);
            jar.closeEntry();
        } catch (final IOException e) {
            throw new IllegalStateException(e);
        }
        return outputStream.toByteArray();
    }

    private Properties loadUserConfiguration(final String plugin, final Path userConfig,
            final Map userJars) {
        final Properties properties = new Properties();
        if (!Files.exists(userConfig)) {
            return properties;
        }
        final String content;
        try (final BufferedReader stream = Files.newBufferedReader(userConfig)) {
            content = stream.lines().collect(joining("\n"));
        } catch (final IOException e) {
            throw new IllegalStateException(e);
        }
        try (final Reader reader = new StringReader(replaceByGav(plugin, content, userJars))) {
            properties.load(reader);
        } catch (final IOException e) {
            throw new IllegalStateException(e);
        }
        return properties;
    }

    // handle "userJar(name)" function in the config, normally not needed since jars are in the context already
    // note: if we make it more complex, switch to a real parser or StrSubstitutor
    String replaceByGav(final String plugin, final String content, final Map userJars) {
        final StringBuilder output = new StringBuilder();
        final String prefixFn = "userJar(";
        int fnIdx = content.indexOf(prefixFn);
        int previousEnd = 0;
        if (fnIdx < 0) {
            output.append(content);
        } else {
            while (fnIdx >= 0) {
                final int end = content.indexOf(')', fnIdx);
                output.append(content, previousEnd, fnIdx);
                output.append(toGav(plugin, content.substring(fnIdx + prefixFn.length(), end), userJars));
                fnIdx = content.indexOf(prefixFn, end);
                if (fnIdx < 0) {
                    if (end < content.length() - 1) {
                        output.append(content, end + 1, content.length());
                    }
                } else {
                    previousEnd = end + 1;
                }
            }
        }
        return output.toString();
    }

    private String toGav(final String plugin, final String jarNameWithoutExtension,
            final Map userJars) {
        return groupIdFor(plugin) + ':' + jarNameWithoutExtension + ":jar:"
                + userJars
                        .keySet()
                        .stream()
                        .filter(it -> it.getArtifact().equals(jarNameWithoutExtension))
                        .findFirst()
                        .map(Artifact::getVersion)
                        .orElse("unknown");
    }

    private Map findJars(final Path familyFolder, final String family) {
        if (!Files.isDirectory(familyFolder)) {
            return emptyMap();
        }
        try {
            return Files
                    .list(familyFolder)
                    .filter(file -> file.getFileName().toString().endsWith(".jar"))
                    .collect(toMap(it -> {
                        try {
                            return new Artifact(groupIdFor(family), toArtifact(it), "jar", "",
                                    Long.toString(Files.getLastModifiedTime(it).toMillis()), "compile");
                        } catch (final IOException e) {
                            throw new IllegalStateException(e);
                        }
                    }, identity()));
        } catch (final IOException e) {
            throw new IllegalStateException(e);
        }
    }

    private String toArtifact(final Path file) {
        final String name = file.getFileName().toString();
        return name.substring(0, name.length() - ".jar".length());
    }

    private String sanitizedForGav(final String name) {
        return name.replace(' ', '_').toLowerCase(ROOT);
    }

    // note: we don't want to provision based on our real m2, only studio one for now
    private Path findStudioM2() {
        if (System.getProperty("talend.studio.version") != null && System.getProperty("osgi.bundles") != null) {
            final Path localM2 = PathFactory.get(System.getProperty("talend.component.server.maven.repository", ""));
            if (Files.isDirectory(localM2)) {
                return localM2;
            }
        }
        return null;
    }

    private void doProvision(final Artifact artifact, final Supplier newInputStream) {
        if (provisioningM2Base == null) {
            log.debug("No m2 to provision, skipping {}", artifact);
            return;
        }
        final Path target = provisioningM2Base.resolve(artifact.toPath());
        if (target.toFile().exists()) {
            log.debug("{} already exists, skipping", target);
            return;
        }
        final Path parentFile = target.getParent();
        if (!Files.exists(parentFile)) {
            try {
                Files.createDirectories(parentFile);
            } catch (final IOException e) {
                throw new IllegalArgumentException("Can't create " + parentFile, e);
            }
        }
        try (final InputStream stream = new BufferedInputStream(newInputStream.get())) {
            Files.copy(stream, target, StandardCopyOption.REPLACE_EXISTING, StandardCopyOption.COPY_ATTRIBUTES);
        } catch (final IOException e) {
            throw new IllegalStateException(e);
        }
    }

    @RequiredArgsConstructor
    private static class Enrichment {

        private final boolean customized;

        private final Map customConfiguration;

        private final Artifact configurationArtifact;

        private final Collection userArtifacts;
    }

    public static class LocalConfigurationImpl implements LocalConfiguration {

        private final VirtualDependenciesService delegate;

        public LocalConfigurationImpl() {
            delegate = CDI.current().select(VirtualDependenciesService.class).get();
        }

        @Override
        public String get(final String key) {
            if (key == null || !key.contains(".")) {
                return null;
            }
            final String plugin = key.substring(0, key.indexOf('.'));
            final Enrichment enrichment = delegate.getEnrichmentFor(plugin);
            if (enrichment == null || enrichment.customConfiguration == null) {
                return null;
            }
            return enrichment.customConfiguration.get(key.substring(plugin.length() + 1));
        }

        @Override
        public Set keys() {
            final ClassLoader loader = Thread.currentThread().getContextClassLoader();
            if (!ConfigurableClassLoader.class.isInstance(loader)) {
                return emptySet();
            }
            final String id = ConfigurableClassLoader.class.cast(loader).getId();
            final Enrichment enrichment = delegate.getEnrichmentFor(id);
            if (enrichment == null || enrichment.customConfiguration == null) {
                return emptySet();
            }
            return unmodifiableSet(enrichment.customConfiguration.keySet());
        }
    }

    public static class UserContainerClasspathContributor implements ComponentManager.ContainerClasspathContributor {

        private final VirtualDependenciesService delegate;

        public UserContainerClasspathContributor() {
            delegate = CDI.current().select(VirtualDependenciesService.class).get();
        }

        @Override
        public Collection findContributions(final String pluginId) {
            delegate.onDeploy(pluginId);
            return delegate.userArtifactsFor(pluginId).collect(toList());
        }

        @Override
        public boolean canResolve(final String path) {
            return delegate.isVirtual(path.replace('/', '.'));
        }

        @Override
        public Path resolve(final String path) {
            if (path.contains('/' + delegate.getConfigurationArtifactIdPrefix())) {
                return null; // not needed, will be enriched on the fly, see LocalConfigurationImpl
            }
            final String[] segments = path.split("/");
            if (segments.length < 9) {
                return null;
            }
            // ex: virtual.talend.component.server.generated.::jar:dynamic
            final String group = delegate
                    .groupIdFor(Stream.of(segments).skip(5).limit(segments.length - 5 - 3).collect(joining(".")));
            final String artifact = segments[segments.length - 3];
            return delegate
                    .getArtifactMapping()
                    .entrySet()
                    .stream()
                    .filter(it -> it.getKey().getGroup().equals(group) && it.getKey().getArtifact().equals(artifact))
                    .findFirst()
                    .map(Map.Entry::getValue)
                    .orElse(null);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy