/* * 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.TaskDefinition; 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 * @author Andy Clement */ public class ConfigurationPropertyValueHintTaskRecoveryStrategy extends StacktraceFingerprintingTaskRecoveryStrategy<CheckPointedParseException> { private final AppRegistry appRegistry; private final ApplicationConfigurationMetadataResolver metadataResolver; @Autowired private ValueHintProvider[] valueHintProviders = new ValueHintProvider[0]; ConfigurationPropertyValueHintTaskRecoveryStrategy(AppRegistry appRegistry, ApplicationConfigurationMetadataResolver metadataResolver) { super(CheckPointedParseException.class, "foo --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 appRegistration = lookupLastApp(exception); if (appRegistration == null) { // Not a valid app name, do nothing return; } Resource metadataResource = appRegistration.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(); TaskDefinition taskDefinition = new TaskDefinition("__dummy", safe); String appName = taskDefinition.getRegisteredAppName(); AppRegistration appRegistration = this.appRegistry.find(appName, ApplicationType.task); return appRegistration; } 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(); } }