使用 LiipFunctionalTestBundle 和固定装置进行独立功能测试?

Independent functional tests with LiipFunctionalTestBundle and fixtures?

我正在向 Symfony2 项目添加测试。以前我在 devtest 环境中使用了相同的数据库,它使用了一个 MySQL 数据库,该数据库已经填充了与生产服务器上相同的数据。

测试是相互依赖的,因为一些测试依赖于以前的测试。例如,如果我有一个商店网站,我在购物车中添加了一个产品,然后从购物车中删除了该产品。所以我需要使用表单插入数据,然后才能删除它。

现在我想使用独立的功能测试,因为那是 recommended way (by one of Symfony2's developers)

我已配置 LiipFunctionalTestBundle correctly to use a SQLite database in the test environment and I've started to add fixtures with DoctrineFixturesBundle

但我不知道每次功能测试需要加载多少数据。我应该在测试开始时加载什么夹具?当实体因为表之间的关系而依赖于其他实体时,如何处理CRUD操作?

假设我正在开发一家商店,我想进行一些测试:

  1. 用户在购物车中添加了一些产品
  2. 用户从购物车中移除一件产品
  3. 用户订购剩余商品

我应该为每个步骤创建不同的夹具吗?这意味着我的固定装置将需要以多种不同的状态存在:空车、订购了一种产品的车等。这对我来说似乎是正确的,但非常耗时,所以我想知道我的想法是否有效。

对于每个测试用例,为了隔离和性能,最好加载尽可能少的夹具(测试套件可能会非常慢)。

当 fixture 相互依赖时,您只需使用 doctrine reference 和 link 管理它们,同时注意顺序。 例如,假设简单的用户和角色关系。

用于管理角色夹具的通用 class:

abstract class BaseLoadRoleData extends AbstractFixture implements OrderedFixtureInterface
{


    public function getOrder()
    {
        return 1;
    }

    protected function createRole(ObjectManager $manager, $rolename)
    {
        $role= new Role();
        $role->setName($rolename);

        $manager->persist($role);
        $manager->flush();
        $this->setReference('role-' . $rolename, $role);
    }
}

简单角色的专用class

class LoadSimpleRoleData extends BaseLoadRoleData
{
    public function load(ObjectManager $manager)
    {
        $this->createRole($manager, Role::SIMPLE);
    }
}

管理员角色的专用class

class LoadAdminRoleData extends BaseLoadRoleData
{
    public function load(ObjectManager $manager)
    {
        $this->createRole($manager, Role::ADMIN);
    }

}

以及用户: 用于管理用户设备的通用 class:

abstract class BaseLoadUserData extends AbstractFixture implements OrderedFixtureInterface
{

    /**
     * @var ContainerInterface
     */
    private $container;

    /**
     * {@inheritDoc}
     */
    public function setContainer(ContainerInterface $container = null)
    {
        $this->container = $container;
    }

    public function getOrder()
    {
        return 2;
    }

    protected function buildUser($username, $firstName = "",$lastName ="")
    {
        $user= new User();
        $user->setUsername($username);
        $user->setFirstName($firstName);
        $user->setLastName($lastName);

        return $user;

    }
}

简单用户专用class

class LoadSimpleUserData extends BaseLoadUserData {

    /**
     * Load data fixtures with the passed EntityManager
     *
     * @param Doctrine\Common\Persistence\ObjectManager $manager
     */
    function load(ObjectManager $manager)
    {
        $user = $this->buildUser($manager, "simple@example.com");
        $user->addRole($this->getReference('role-'.Role::SIMPLE));
        $manager->persist($user);
        $manager->flush();
        $this->setReference('user-' . "admin@example.com", $user);

    }
}

管理员用户专用class

class LoadAdminUserData extends BaseLoadUserData {

    /**
     * Load data fixtures with the passed EntityManager
     *
     * @param Doctrine\Common\Persistence\ObjectManager $manager
     */
    function load(ObjectManager $manager)
    {
        $user = $this->buildUser($manager, "admin@example.com");
        $user->addRole($this->getReference('role-'.Role::ADMIN));
        $manager->persist($user);
        $manager->flush();
        $this->setReference('user-' . "admin@example.com", $user);

    }

现在可以单独使用了,例如,基于Liip Functional Test Bundle:

class LoginControllerTest {

    public function testAdminUserLogin()
    {
        $this->loadFixtures(array(
            'Acme\DemoBundle\DataFixtures\ORM\LoadAdminRoleData',
            'Acme\DemoBundle\DataFixtures\ORM\LoadAdminUserData'
        ));

        // you can now run your functional tests with a populated database
        $client = static::createClient();
        // ...

        // test the login with admin credential
    }

    public function testSimpleUserLogin()
    {
        // add all your fixtures classes that implement
        // Doctrine\Common\DataFixtures\FixtureInterface
        $this->loadFixtures(array(
            'Acme\DemoBundle\DataFixtures\ORM\LoadSimpleRoleData',
            'Acme\DemoBundle\DataFixtures\ORM\LoadSimpleUserData'
        ));

        // you can now run your functional tests with a populated database
        $client = static::createClient();
        // ...

        // test the login with simple user credential

    }

}

希望对您有所帮助。