在SpEL中使用应用程序范围的属性(以编程方式)

时间:2016-01-05 14:59:03

标签: java spring spring-el

我需要构建一个小模块,将SpEL编码的表达式转换为字符串。这是为了抛弃Jexl并访问应用程序上下文。

例如,如果我正确配置的属性文件包含

application.name=AppTest
company.name=ACME Inc.

我希望根据属性

翻译以下路径或类似于它的字符串
/path/to/#{application.name}/#{company.name}

#或$与我无关

通常在Spring中你可以将属性注入bean中,我已经成功了。但是现在我希望用户输入一个模板字符串,该字符串可以使用Application Context中的所有属性进行翻译。目前我不需要访问bean属性,但我将来可能会有。以上是定义文件夹路径时最常见方案的简化变体。运行时参数(例如一天中的时间)增加了很少的复杂性,但我问的问题是一步一步地工作。

所以我试图在JUnit的帮助下通过玩它们来理解Spring Expressions。我编写了以下代码,但无法使其工作

代码

final HashMap<String, Object> propertySource = new HashMap<String, Object>();
private final String FOLDER_PATTERN = "#tmp/appTest/#{company}_#{appname}/q1"; //#tmp is only token being replaced "hardcoded", not passed to Spring

@Before
public void setUp() throws Exception
{

    propertySource.put("appname", APPNAME);
    propertySource.put("company", COMPANY);
    applicationContext.getEnvironment()
                      .getPropertySources()
                      .addLast(new MapPropertySource("test", propertySource));

}

@Test
public void playWithExpression()
{
    ExpressionParser expParser = new SpelExpressionParser();
    StandardEvaluationContext stdEvaluationContext = new StandardEvaluationContext();
    stdEvaluationContext.setBeanResolver(new BeanFactoryResolver(applicationContext.getBeanFactory()));
    // stdEvaluationContext.setVariables(propertySource);

    final TemplateParserContext templateParserContext = new TemplateParserContext();

    String folderPattern = FOLDER_PATTERN.replace("#tmp", SYSTEM_TEMP_DIR);

    String realPath = expParser.parseExpression(folderPattern, templateParserContext)
                               .getValue(stdEvaluationContext, String.class);

    String calculatedPath = folderPattern.replace("#{company}", COMPANY)
                                         .replace("#{appname}", APPNAME);

    assertEquals(calculatedPath, realPath);

}

说明

部分复制并粘贴其他示例,我:

  • 注入自定义属性源以仅为当前JUnit测试添加属性
  • 使用BeanFactory将EvaluationContext实例化为bean解析器
  • 使用使用#{..}表示法
  • 的默认模板解析器上下文解析表达式
  • 使用上述评估上下文检索该表达式的值。 此操作失败

表达式已成功实例化,但getValue抛出异常

org.springframework.expression.spel.SpelEvaluationException: EL1007E:(pos 0): Property or field 'company' cannot be found on null
    at org.springframework.expression.spel.ast.PropertyOrFieldReference.readProperty(PropertyOrFieldReference.java:220)
    at org.springframework.expression.spel.ast.PropertyOrFieldReference.getValueInternal(PropertyOrFieldReference.java:94)
    at org.springframework.expression.spel.ast.PropertyOrFieldReference.getValueInternal(PropertyOrFieldReference.java:81)
    at org.springframework.expression.spel.ast.SpelNodeImpl.getTypedValue(SpelNodeImpl.java:131)
    at org.springframework.expression.spel.standard.SpelExpression.getValue(SpelExpression.java:299)
    at org.springframework.expression.common.CompositeStringExpression.getValue(CompositeStringExpression.java:89)
    at org.springframework.expression.common.CompositeStringExpression.getValue(CompositeStringExpression.java:136)
    at it.phoenix.web.data.managers.test.FoldersManagerTemplateTests.playWithExpression(FoldersManagerTemplateTests.java:80)

从错误判断,似乎Spring正在尝试将company标记解析为根对象的属性,在这种情况下会丢失。

如果我使用#{@...}表示法,那么Spring会尝试将该标记解释为bean标识符。不是我的情况。

如何从表达式中的AppContext的PropertySource解析属性?

1 个答案:

答案 0 :(得分:2)

可能有一个更优雅的解决方案,但

private final String FOLDER_PATTERN = "#tmp/appTest/#{environment.getProperty('company')}"
        + "_#{environment.getProperty('appname')}/q1";  // #tmp is only token being replaced
                                                        // "hardcoded", not passed to Spring

    String realPath = expParser.parseExpression(folderPattern, templateParserContext)
            .getValue(stdEvaluationContext, applicationContext, String.class);

有效(即使用应用程序上下文作为评估中的根对象)。

修改

稍微好一些......

private final String FOLDER_PATTERN = "#tmp/appTest/#{getProperty('company')}"
        + "_#{getProperty('appname')}/q1";  // #tmp is only token being replaced
                                            // "hardcoded", not passed to Spring

    String realPath = expParser.parseExpression(folderPattern, templateParserContext)
            .getValue(stdEvaluationContext, applicationContext.getEnvironment(), String.class);