Не удалось разрешить заполнитель в весеннем приложении XML-config

Я начинающий весной. Я прочитал много вопросов с похожими заголовками на SO, и либо они у меня не работали, либо это была не та конфигурация, и я не мог понять, что не так.

Я пишу простое весеннее приложение в Intellij Idea с Gradle. Все работало хорошо, пока я не решил прочитать некоторые значения из файла .properties. И xml-config файл, и .properties файлы находятся в ресурсах.

Я создал простой тест junit. Реализация LoginService действительна, так как когда электронная почта и пароль были жестко закодированы в xml, тест пройден.

Трассировки стека:

Testing started at 19:08 ...
19:08:27: Executing tasks ':cleanTest :test --tests MyDocumentsLoginTest'...

> Task :cleanTest
> Task :compileJava UP-TO-DATE
> Task :compileGroovy NO-SOURCE
> Task :processResources
> Task :classes
> Task :compileTestJava UP-TO-DATE
> Task :compileTestGroovy NO-SOURCE
> Task :processTestResources NO-SOURCE
> Task :testClasses UP-TO-DATE
> Task :test FAILED
lip 15, 2019 7:08:29 PM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@6d2d086b: startup date [Mon Jul 15 19:08:29 CEST 2019]; root of context hierarchy
lip 15, 2019 7:08:29 PM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [META-INF/spring/MyDocuments-config.xml]
lip 15, 2019 7:08:29 PM org.springframework.beans.factory.config.PropertyPlaceholderConfigurer loadProperties
INFO: Loading properties file from class path resource [META-INF/data/env_dev.properties]

Invalid bean definition with name 'login' defined in class path resource [META-INF/spring/MyDocuments-config.xml]: Could not resolve placeholder 'user.email' in string value "${user.email}"; nested exception is java.lang.IllegalArgumentException: Could not resolve placeholder 'user.email' in string value "${user.email}"
org.springframework.beans.factory.BeanDefinitionStoreException: Invalid bean definition with name 'login' defined in class path resource [META-INF/spring/MyDocuments-config.xml]: Could not resolve placeholder 'user.email' in string value "${user.email}"; nested exception is java.lang.IllegalArgumentException: Could not resolve placeholder 'user.email' in string value "${user.email}"
    at org.springframework.beans.factory.config.PlaceholderConfigurerSupport.doProcessProperties(PlaceholderConfigurerSupport.java:211)
    at org.springframework.beans.factory.config.PropertyPlaceholderConfigurer.processProperties(PropertyPlaceholderConfigurer.java:223)
    at org.springframework.beans.factory.config.PropertyResourceConfigurer.postProcessBeanFactory(PropertyResourceConfigurer.java:86)
    at org.springframework.context.support.PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(PostProcessorRegistrationDelegate.java:265)
    at org.springframework.context.support.PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(PostProcessorRegistrationDelegate.java:162)
    at org.springframework.context.support.AbstractApplicationContext.invokeBeanFactoryPostProcessors(AbstractApplicationContext.java:609)
    at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:464)
    at org.springframework.context.support.ClassPathXmlApplicationContext.<init>(ClassPathXmlApplicationContext.java:139)
    at org.springframework.context.support.ClassPathXmlApplicationContext.<init>(ClassPathXmlApplicationContext.java:83)
    at MyDocumentsLoginTest.setup(MyDocumentsLoginTest.java:19)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:24)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecutor.runTestClass(JUnitTestClassExecutor.java:106)
    at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecutor.execute(JUnitTestClassExecutor.java:58)
    at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecutor.execute(JUnitTestClassExecutor.java:38)
    at org.gradle.api.internal.tasks.testing.junit.AbstractJUnitTestClassProcessor.processTestClass(AbstractJUnitTestClassProcessor.java:66)
    at org.gradle.api.internal.tasks.testing.SuiteTestClassProcessor.processTestClass(SuiteTestClassProcessor.java:51)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.gradle.internal.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35)
    at org.gradle.internal.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24)
    at org.gradle.internal.dispatch.ContextClassLoaderDispatch.dispatch(ContextClassLoaderDispatch.java:32)
    at org.gradle.internal.dispatch.ProxyDispatchAdapter$DispatchingInvocationHandler.invoke(ProxyDispatchAdapter.java:93)
    at com.sun.proxy.$Proxy2.processTestClass(Unknown Source)
    at org.gradle.api.internal.tasks.testing.worker.TestWorker.processTestClass(TestWorker.java:117)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.gradle.internal.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35)
    at org.gradle.internal.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24)
    at org.gradle.internal.remote.internal.hub.MessageHubBackedObjectConnection$DispatchWrapper.dispatch(MessageHubBackedObjectConnection.java:155)
    at org.gradle.internal.remote.internal.hub.MessageHubBackedObjectConnection$DispatchWrapper.dispatch(MessageHubBackedObjectConnection.java:137)
    at org.gradle.internal.remote.internal.hub.MessageHub$Handler.run(MessageHub.java:404)
    at org.gradle.internal.concurrent.ExecutorPolicy$CatchAndRecordFailures.onExecute(ExecutorPolicy.java:63)
    at org.gradle.internal.concurrent.ManagedExecutorImpl$1.run(ManagedExecutorImpl.java:46)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at org.gradle.internal.concurrent.ThreadFactoryImpl$ManagedThreadRunnable.run(ThreadFactoryImpl.java:55)
    at java.lang.Thread.run(Thread.java:748)
Caused by: java.lang.IllegalArgumentException: Could not resolve placeholder 'user.email' in string value "${user.email}"
    at org.springframework.util.PropertyPlaceholderHelper.parseStringValue(PropertyPlaceholderHelper.java:174)
    at org.springframework.util.PropertyPlaceholderHelper.replacePlaceholders(PropertyPlaceholderHelper.java:126)
    at org.springframework.beans.factory.config.PropertyPlaceholderConfigurer$PlaceholderResolvingStringValueResolver.resolveStringValue(PropertyPlaceholderConfigurer.java:259)
    at org.springframework.beans.factory.config.BeanDefinitionVisitor.resolveStringValue(BeanDefinitionVisitor.java:282)
    at org.springframework.beans.factory.config.BeanDefinitionVisitor.resolveValue(BeanDefinitionVisitor.java:204)
    at org.springframework.beans.factory.config.BeanDefinitionVisitor.visitPropertyValues(BeanDefinitionVisitor.java:141)
    at org.springframework.beans.factory.config.BeanDefinitionVisitor.visitBeanDefinition(BeanDefinitionVisitor.java:82)
    at org.springframework.beans.factory.config.PlaceholderConfigurerSupport.doProcessProperties(PlaceholderConfigurerSupport.java:208)
    ... 56 more

MyDocumentsLoginTest > testLogin FAILED
    org.springframework.beans.factory.BeanDefinitionStoreException at MyDocumentsLoginTest.java:19
        Caused by: java.lang.IllegalArgumentException at MyDocumentsLoginTest.java:19
1 test completed, 1 failed
FAILURE: Build failed with an exception.
* What went wrong:
Execution failed for task ':test'.

мой файл конфигурации xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context-4.0.xsd">


    <bean id="environmentProperties" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
        <property name="location" value="classpath:META-INF/data/env_dev.properties" />
    </bean>

    <bean id="login" class="com.doszke.isf.java.spring.service.LoginService">
        <property name="username" value="${user.email}"/>
        <property name="password" value="${user.password}"/>
    </bean>


    <context:component-scan base-package="com.doszke.isf.java" />

    <bean id="engine" class="com.doszke.isf.java.service.SearchEngineService">
        <property name="documentDAO" ref="documentDAO"/>
    </bean>

    <bean id="documentDAO" class="com.doszke.isf.java.spring.service.DocumentRepository">
        <property name="documents">
            <list>
                <ref bean="doc1"/>
                <ref bean="doc2"/>
                <ref bean="doc3"/>
                <ref bean="doc4"/>
            </list>
        </property>
    </bean>

    <bean id="typeDAO" class="com.doszke.isf.java.spring.service.TypeDataRepository">
        <property name="types">
            <map>
                <entry key="webType" value-ref="webType"/>
                <entry key="noteType" value-ref="noteType"/>
                <entry key="pdfType" value-ref="pdfType"/>
            </map>
        </property>
    </bean>

    <!--Document-->
    <bean id="doc1" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Szablon książki" />
        <property name="type" ref="pdfType" />
        <property name="location" value="/Users/doszke/Documents/random/book template.pdf" />
    </bean>

    <bean id="doc2" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Przykładowa umowa" />
        <property name="type" ref="pdfType" />
        <property name="location" value="/Users/doszke/Documents/Contracts/sample contract.pdf" />
    </bean>

    <bean id="doc3" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Moje notatki" />
        <property name="type" ref="noteType" />
        <property name="location" value="/Users/doszke/Documents/random/my note.txt" />
    </bean>

    <bean id="doc4" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Pro Spring Security Book" />
        <property name="type" ref="webType" />
        <property name="location" value="http://www.apress.com/9781430248187" />
    </bean>

    <!--Type-->
    <bean id="webType" class="com.doszke.isf.java.model.Type">
        <property name="name" value="WEB" />
        <property name="desc" value="Łącze sieciowe" />
        <property name="extension" value=".url" />
    </bean>

    <bean id="pdfType" class="com.doszke.isf.java.model.Type">
        <property name="name" value="PDF" />
        <property name="desc" value="Portable Document Format" />
        <property name="extension" value=".pdf" />
    </bean>

    <bean id="noteType" class="com.doszke.isf.java.model.Type">
        <property name="name" value="NOTE" />
        <property name="desc" value="Notatki tekstowe" />
        <property name="extension" value=".txt" />
    </bean>

</beans>

Тестовый класс (покажите, как я читаю xml):

import com.doszke.isf.java.spring.service.Login;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import static org.junit.Assert.assertNotNull;

public class MyDocumentsLoginTest {


    private static final String EMAIL = "[email protected]";
    private static final String PASS = "test123";
    private static final String SUCCESS = "Authorized";
    private static final String FAILURE = "Authorization failed";
    private ClassPathXmlApplicationContext context;

    @Before
    public void setup(){
        context = new ClassPathXmlApplicationContext("META-INF/spring/MyDocuments-config.xml");
    }

    @Test
    public void testLogin(){
        Login login = context.getBean(Login.class);
        assertNotNull(login);
        System.out.println(login.isAuthorized(EMAIL, PASS) ? SUCCESS : FAILURE);
    }

}

Вот как выглядит структура моего проекта: введите здесь описание изображения

@Обновление: файл env_dev.properties:

[email protected]
user.password=test123

person Embid123    schedule 15.07.2019    source источник
comment
Я думаю, стоит упомянуть, что Intellij дает мне подсказки об этих значениях, когда я набираю user.email и т. д.   -  person Embid123    schedule 15.07.2019
comment
Вы уверены, что Maven копирует env_dev.properties в целевую папку при сборке проекта?   -  person Ken Bekov    schedule 15.07.2019
comment
Да, он находится в папке build вместе с другими ресурсами.   -  person Embid123    schedule 15.07.2019
comment
Вы уверены, что в вашем файле свойств действительно есть свойство user.email?   -  person Ivan    schedule 15.07.2019
comment
Да, это имеет. Я обновил основной пост содержимым этого файла   -  person Embid123    schedule 15.07.2019


Ответы (1)


В applicationContext.xml измените на это:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context-4.0.xsd">


    <!-- <bean id="environmentProperties" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
        <property name="location" value="classpath:META-INF/data/env_dev.properties" /></bean>-->

    <!-- Add this instead -->
    <context:property-placeholder location="classpath:env_dev.properties" />

    <bean id="login" class="com.doszke.isf.java.spring.service.LoginService">
        <property name="username" value="${user.email}"/>
        <property name="password" value="${user.password}"/>
    </bean>


    <context:component-scan base-package="com.doszke.isf.java" />

    <bean id="engine" class="com.doszke.isf.java.service.SearchEngineService">
        <property name="documentDAO" ref="documentDAO"/>
    </bean>

    <bean id="documentDAO" class="com.doszke.isf.java.spring.service.DocumentRepository">
        <property name="documents">
            <list>
                <ref bean="doc1"/>
                <ref bean="doc2"/>
                <ref bean="doc3"/>
                <ref bean="doc4"/>
            </list>
        </property>
    </bean>

    <bean id="typeDAO" class="com.doszke.isf.java.spring.service.TypeDataRepository">
        <property name="types">
            <map>
                <entry key="webType" value-ref="webType"/>
                <entry key="noteType" value-ref="noteType"/>
                <entry key="pdfType" value-ref="pdfType"/>
            </map>
        </property>
    </bean>

    <!--Document-->
    <bean id="doc1" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Szablon książki" />
        <property name="type" ref="pdfType" />
        <property name="location" value="/Users/doszke/Documents/random/book template.pdf" />
    </bean>

    <bean id="doc2" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Przykładowa umowa" />
        <property name="type" ref="pdfType" />
        <property name="location" value="/Users/doszke/Documents/Contracts/sample contract.pdf" />
    </bean>

    <bean id="doc3" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Moje notatki" />
        <property name="type" ref="noteType" />
        <property name="location" value="/Users/doszke/Documents/random/my note.txt" />
    </bean>

    <bean id="doc4" class="com.doszke.isf.java.model.Document">
        <property name="name" value="Pro Spring Security Book" />
        <property name="type" ref="webType" />
        <property name="location" value="http://www.apress.com/9781430248187" />
    </bean>

    <!--Type-->
    <bean id="webType" class="com.doszke.isf.java.model.Type">
        <property name="name" value="WEB" />
        <property name="desc" value="Łącze sieciowe" />
        <property name="extension" value=".url" />
    </bean>

    <bean id="pdfType" class="com.doszke.isf.java.model.Type">
        <property name="name" value="PDF" />
        <property name="desc" value="Portable Document Format" />
        <property name="extension" value=".pdf" />
    </bean>

    <bean id="noteType" class="com.doszke.isf.java.model.Type">
        <property name="name" value="NOTE" />
        <property name="desc" value="Notatki tekstowe" />
        <property name="extension" value=".txt" />
    </bean>

</beans>

Это определенно должно решить проблему. Добавьте папку ресурсов в путь к классам.

person Anish B.    schedule 15.07.2019
comment
Он выдает FileNotFound после расширения пути до classpath:META-INF/data/env_dev.properties с тем же эффектом, что и в основном посте. - person Embid123; 15.07.2019
comment
Добавьте папку ресурсов в путь к классам из свойств проекта. - person Anish B.; 15.07.2019
comment
Все еще что-то идет не так. Я добавил в путь к классам каталог src/main/java/resources, добавил тестовый каталог (который был указан как пустая библиотека), перекомпилировал и запустил. Тот же эффект. Вот скриншот свойств проекта imgur.com/a/eRsNvt9. - person Embid123; 15.07.2019
comment
Храните файл свойств вне META-INF. Проверить тогда? - person Anish B.; 15.07.2019
comment
Вкладка «Открытые источники». Пришлите мне скриншот. - person Anish B.; 15.07.2019
comment
imgur.com/a/LZzQMds Вот. Тем временем я добавил ignore-unresolvable="true" к context:property-placeholder и напечатал логин и пароль. Пароль читался из файла, лемейла нет. - person Embid123; 15.07.2019
comment
Я нашел источник проблемы. В начале файла env_dev.properties было несколько оставшихся байтов, и это привело к сбою загрузки user.email. После удаления даже все прошло. Спасибо за потраченное время! PS. Моя прежняя конфигурация xml теперь тоже работает. - person Embid123; 15.07.2019