/* * Copyright 2015-2017 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 * * 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.springframework.cloud.dataflow.completion; import java.io.IOException; import java.net.URLClassLoader; import java.util.List; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.configurationmetadata.ConfigurationMetadataProperty; import org.springframework.boot.configurationmetadata.ValueHint; import org.springframework.cloud.dataflow.configuration.metadata.ApplicationConfigurationMetadataResolver; import org.springframework.cloud.dataflow.core.ApplicationType; import org.springframework.cloud.dataflow.core.StreamAppDefinition; import org.springframework.cloud.dataflow.core.StreamDefinition; import org.springframework.cloud.dataflow.core.dsl.CheckPointedParseException; import org.springframework.cloud.dataflow.core.dsl.Token; import org.springframework.cloud.dataflow.core.dsl.TokenKind; import org.springframework.cloud.dataflow.registry.AppRegistration; import org.springframework.cloud.dataflow.registry.AppRegistry; import org.springframework.core.io.Resource; import static org.springframework.cloud.dataflow.completion.CompletionProposal.expanding; /** * Attempts to fill in possible values after a {@literal --foo=} dangling construct in the * DSL. * * @author Eric Bottard * @author Mark Fisher */ public class ConfigurationPropertyValueHintRecoveryStrategy extends StacktraceFingerprintingRecoveryStrategy<CheckPointedParseException> { private final AppRegistry appRegistry; private final ApplicationConfigurationMetadataResolver metadataResolver; @Autowired private ValueHintProvider[] valueHintProviders = new ValueHintProvider[0]; ConfigurationPropertyValueHintRecoveryStrategy(AppRegistry appRegistry, ApplicationConfigurationMetadataResolver metadataResolver) { super(CheckPointedParseException.class, "foo --bar=", "foo | wizz --bar="); this.appRegistry = appRegistry; this.metadataResolver = metadataResolver; } @Override public void addProposals(String dsl, CheckPointedParseException exception, int detailLevel, List<CompletionProposal> collector) { String propertyName = recoverPropertyName(exception); AppRegistration lastAppRegistration = lookupLastApp(exception); if (lastAppRegistration == null) { // Not a valid app name, do nothing return; } Resource metadataResource = lastAppRegistration.getMetadataResource(); CompletionProposal.Factory proposals = expanding(dsl); List<ConfigurationMetadataProperty> whiteList = metadataResolver.listProperties(metadataResource); URLClassLoader classLoader = null; try { for (ConfigurationMetadataProperty property : metadataResolver.listProperties(metadataResource, true)) { if (CompletionUtils.isMatchingProperty(propertyName, property, whiteList)) { if (classLoader == null) { classLoader = metadataResolver.createAppClassLoader(metadataResource); } for (ValueHintProvider valueHintProvider : valueHintProviders) { for (ValueHint valueHint : valueHintProvider.generateValueHints(property, classLoader)) { collector.add(proposals.withSuffix(String.valueOf(valueHint.getValue()), valueHint.getShortDescription())); } } } } } catch (Exception e) { throw new RuntimeException(e); } finally { if (classLoader != null) { try { classLoader.close(); } catch (IOException e) { // ignore } } } } private AppRegistration lookupLastApp(CheckPointedParseException exception) { String safe = exception.getExpressionStringUntilCheckpoint(); StreamDefinition streamDefinition = new StreamDefinition("__dummy", safe); StreamAppDefinition lastApp = streamDefinition.getDeploymentOrderIterator().next(); String lastAppName = lastApp.getName(); AppRegistration lastAppRegistration = null; for (ApplicationType appType : CompletionUtils.determinePotentialTypes(lastApp)) { lastAppRegistration = this.appRegistry.find(lastAppName, appType); if (lastAppRegistration != null) { break; } } return lastAppRegistration; } private String recoverPropertyName(CheckPointedParseException exception) { List<Token> tokens = exception.getTokens(); int tokenPointer = tokens.size() - 1; while (!tokens.get(tokenPointer - 1).isKind(TokenKind.DOUBLE_MINUS)) { tokenPointer--; } StringBuilder builder; final int equalSignPointer = tokens.size() - 1; for (builder = new StringBuilder(); tokenPointer < equalSignPointer; tokenPointer++) { Token t = tokens.get(tokenPointer); if (t.isIdentifier()) { builder.append(t.stringValue()); } else { builder.append(t.getKind().getTokenChars()); } } return builder.toString(); } }