当前位置: 首页 > 知识库问答 >
问题:

@WithUserDetails和spring boot 1.4 TestEntityManager存在问题

罗安和
2023-03-14

我有一个问题与Spring Boot的TestEntityManager@SusUser细节注释。

这是我的测试套件:

public class AdvertisementAuthorizationTest extends AbstractIntegrationTest {

    private static final String IMPERSONATOR_EMAIL = "joe.hacker@gmail.com";

    private final static String OWNER_OF_ADVERTISEMENT_EMAIL = "john.nice@gmail.com";

    @Autowired
    private TestEntityManager testEntityManager;

    @Autowired
    private MockMvc mockMvc;

    private Advertisement advertisement;

    private UserAccount impersonator;

    private ObjectMapper mapper = new ObjectMapper();

    @Before
    public void setUp() {
        advertisement = testEntityManager.persist(createAdvertisement(OWNER_OF_ADVERTISEMENT_EMAIL));
        impersonator = testEntityManager.persist(createUserAccount(IMPERSONATOR_EMAIL));
    }

    @Test
    @WithUserDetails(IMPERSONATOR_EMAIL)
    public void shouldNotAllowAdvertisementModification() throws Exception {
        String jsonAdvertisement = mapper.writeValueAsString(advertisement);
        mockMvc.perform(put("/api/advertisement/{id}", advertisement.getId())//
                .contentType(MediaType.APPLICATION_JSON)//
                .content(jsonAdvertisement))//
                .andDo(print())//
                .andExpect(status().isForbidden());//
    }

    @Test
    @WithUserDetails(OWNER_OF_ADVERTISEMENT_EMAIL)
    public void shouldAllowAdvertisementModification() throws Exception {
        String jsonAdvertisement = mapper.writeValueAsString(advertisement);
        mockMvc.perform(put("/api/advertisement/{id}", advertisement.getId())//
                .contentType(MediaType.APPLICATION_JSON)//
                .content(jsonAdvertisement))//
                .andDo(print())//
                .andExpect(status().isOk());//
    }

}

这是超级课堂:

@AutoConfigureMockMvc
@AutoConfigureTestEntityManager
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = {TestApplication.class})
@WebAppConfiguration
@ActiveProfiles(Profiles.TEST)
@Transactional
public abstract class AbstractIntegrationTest {

}

当我运行测试时,实体似乎没有被TestEntityManager持久化到H2,如以下错误消息所示:

Hibernate: select useraccoun0_.id as id1_9_, useraccoun0_.address_id as address13_9_, useraccoun0_.email as email2_9_, useraccoun0_.email_notification as email_no3_9_, useraccoun0_.enabled as enabled4_9_, useraccoun0_.first_name as first_na5_9_, useraccoun0_.last_connection_date as last_con6_9_, useraccoun0_.password as password7_9_, useraccoun0_.registration_date as registra8_9_, useraccoun0_.role as role9_9_, useraccoun0_.token as token10_9_, useraccoun0_.user_account_type as user_ac11_9_, useraccoun0_.version as version12_9_ from user_account useraccoun0_ where useraccoun0_.email=?
22:52:39.943 [Test worker] WARN  o.s.test.context.TestContextManager - Caught exception while allowing TestExecutionListener [org.springframework.security.test.context.support.WithSecurityContextTestExecutionListener@43ec4dcd] to process 'before' execution of test method [public void com.bignibou.it.web.security.advertisement.AdvertisementAuthorizationTest.shouldNotAllowAdvertisementModification() throws java.lang.Exception] for test instance [com.bignibou.it.web.security.advertisement.AdvertisementAuthorizationTest@646496bc]
java.lang.IllegalStateException: Unable to create SecurityContext using @org.springframework.security.test.context.support.WithUserDetails(value=joe.hacker@gmail.com, userDetailsServiceBeanName=)
    at org.springframework.security.test.context.support.WithSecurityContextTestExecutionListener.createSecurityContext(WithSecurityContextTestExecutionListener.java:79)
    at org.springframework.security.test.context.support.WithSecurityContextTestExecutionListener.beforeTestMethod(WithSecurityContextTestExecutionListener.java:56)
    at org.springframework.test.context.TestContextManager.beforeTestMethod(TestContextManager.java:269)
    at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:74)
    at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:86)
    at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:84)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:252)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:94)
    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.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:191)
    at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecuter.runTestClass(JUnitTestClassExecuter.java:112)
    at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecuter.execute(JUnitTestClassExecuter.java:56)
    at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassProcessor.processTestClass(JUnitTestClassProcessor.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:497)
    at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35)
    at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24)
    at org.gradle.messaging.dispatch.ContextClassLoaderDispatch.dispatch(ContextClassLoaderDispatch.java:32)
    at org.gradle.messaging.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:109)
    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:497)
    at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35)
    at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24)
    at org.gradle.messaging.remote.internal.hub.MessageHub$Handler.run(MessageHub.java:364)
    at org.gradle.internal.concurrent.ExecutorPolicy$CatchAndRecordFailures.onExecute(ExecutorPolicy.java:54)
    at org.gradle.internal.concurrent.StoppableExecutorImpl$1.run(StoppableExecutorImpl.java:40)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)
Caused by: org.springframework.security.core.userdetails.UsernameNotFoundException: Username: joe.hacker@gmail.com not found!
    at com.bignibou.service.security.UserAccountUserDetailsService.loadUserByUsername(UserAccountUserDetailsService.java:21)
    at org.springframework.security.test.context.support.WithUserDetailsSecurityContextFactory.createSecurityContext(WithUserDetailsSecurityContextFactory.java:56)
    at org.springframework.security.test.context.support.WithUserDetailsSecurityContextFactory.createSecurityContext(WithUserDetailsSecurityContextFactory.java:39)
    at org.springframework.security.test.context.support.WithSecurityContextTestExecutionListener.createSecurityContext(WithSecurityContextTestExecutionListener.java:76)
    ... 43 common frames omitted

有人能帮忙吗?

共有1个答案

桓风史
2023-03-14

这是关于TestExecutionListener回调和@前测试方法的计时问题。

Spring Security的带有Security ContextTestExecutionListener的@WithUserDetails受支持,它永远不会在方法之前运行。因此,Spring Security不可能在方法中看到您持久化到数据库的用户。这基本上就是异常告诉您的:Spring Security试图在数据库存在之前从数据库中读取用户。

解决此问题的一种方法是迁移到Sql支持,以便在数据库中插入测试数据。您可能发现这不如简单地持久化实体那么舒服,但Sql方法允许在测试管理的事务中创建测试数据(即,不需要手动清理)。请注意,您必须升级到Spring Security 4.1.1才能正常工作。

解决此问题的另一种方法是以@BeforeTransaction方法将您的实体持久化在用户管理的事务中——例如,使用Spring的TransactionTemboard。但是,您需要以类似的方式在@AfterTransaction方法中手动清理数据库。此外,您仍然需要升级到Spring Security 4.1.1才能正常工作。

像下面这样的东西应该可以做到:

@Autowired
private TestEntityManager testEntityManager;

@Autowired
PlatformTransactionManager transactionManager;

@BeforeTransaction
public void setUp() {
    new TransactionTemplate(transactionManager).execute(status -> {
        UserAccount owner = testEntityManager.persist(createUserAccount(OWNER_OF_ADVERTISEMENT_EMAIL));
        Language language = testEntityManager.persist(createLanguage("Français"));
        DayToTimeSlot dayToTimeSlot = testEntityManager.persist(createDayToTimeSlot());

        advertisement = testEntityManager.persist(createAdvertisement(owner, language, dayToTimeSlot));
        impersonator = testEntityManager.persist(createUserAccount(IMPERSONATOR_EMAIL));

        return null;
    });
}

@AfterTransaction
public void tearDown() {
    new TransactionTemplate(transactionManager).execute(status -> {
        testEntityManager.remove(testEntityManager.find(Advertisement.class, advertisement.getId()));

        UserAccount owner = advertisement.getUserAccount();
        testEntityManager.remove(testEntityManager.find(UserAccount.class, owner.getId()));

        testEntityManager.remove(testEntityManager.find(UserAccount.class, impersonator.getId()));

        return null;
    });
}

当做

山姆

 类似资料:
  • 在使用Spring MockMVC的JUnit测试中,有两种方法用于作为Spring Security用户进行身份验证:使用提供的凭据创建一个虚拟用户,使用用户名并使用自定义()将其解析为正确的自定义实现。 在我的示例中,从数据库加载用户。我要使用的用户是插入到测试套件的方法中的。 但是,我的找不到该用户。 在我的中,我像这样插入用户: 并且在中:

  • 问题内容: 我的服务器时间是格林尼治标准时间(GMT),每当有人在线时,我都会执行以下操作。 所以,我的问题是……每当有人执行某项操作时,它都会将日期/时间存储在用户本地时间中……这给我带来了很多麻烦。 我想要的是将所有内容存储在GMT中,但要让用户在其本地时区查看数据并与之交互。 编辑: 这是我更新用户状态的方法: 这是我将日期戳转换为秒的功能。 最后,我如何比较日期: 问题答案: 这里的技巧是

  • 完整的程序代码在此处:单击此处 启动时,我会遇到以下异常: 请告诉我如何解决这个问题?为什么会出现这种情况? 对评论的答复: @Kandy“您使用的数据库是什么,您能告诉驱动程序名还是显示您的数据库配置代码?” 我正在使用Postgresql。 pom.xml 和配置应用程序。性质

  • 我目前正在用Javascript编写代码,我是一名新手,我需要代码方面的帮助。我想创建一个包含节点(显然)的树,它指向一个链表,这个列表将包含节点子节点。在链表之外,我唯一需要的节点是我的根节点。 无论如何,我的问题是在我父亲节点的链接列表中添加一个节点。示例: 根- 当我试图在链表中添加一个节点时,我的第一个节点被覆盖。 代码: 打印我的给我:

  • 我有以下课程: DBEntity.java 使用者JAVA 用户服务。JAVA UserRepository.java