代码接收:保持登录状态

2022-08-30 18:27:10

我想在大多数测试之前保留或运行登录名。但是,如果我尝试将登录代码移动到_before它不起作用,因为没有可用的webguy实例。

在多个测试之间保持会话的最佳方法是什么?这是我的代码到目前为止,会很高兴得到一些帮助。我已经谷歌并检查了文档,但我找不到任何关于会话的东西。

<?php
use \WebGuy;

class ProductCest
{

    private $product_id = '1';

    public function _before()
    {
    }

    public function _after()
    {
    }

    // tests
    public function login(WebGuy $I) {
        $I->seeInCurrentUrl('/auth/login');
        $I->fillField("//input[@type='email']", "username@email.com");
        $I->fillField("//input[@type='password']", "1234");
        $I->click('#signIn .submit');
        $I->wait(500);

        $I->seeInCurrentUrl('/account');
    }

    /**
     * @depends login
     */
    public function chooseProduct(WebGuy $I) {
        $I->wantTo('go to products and choose one');
        $I->amOnPage('/?product=' . $this->client_id);
    }

}

答案 1

我认为公认的答案是实现它的一种方式,但不是最好的。这样做,当您只需要共享用户会话时,您将始终必须重现登录到系统的所有步骤。我认为获取会话cookie并通过所需的用户登录测试传递它更好。为此,您需要创建一个登录函数,获取cookie,然后首先根据登录测试进行测试:

<?php
use \AcceptanceTester;

class testingCest
{
    private $cookie = null;

    public function _before(AcceptanceTester $I)
    {
    }

    public function _after(AcceptanceTester $I)
    {
    }


    // tests
    public function login(AcceptanceTester $I)
    {
        $I->wantTo('entrar al sistema');
        $I->amOnPage('/');
        $I->seeInCurrentUrl('/admin/login');
        $I->fillField('user','perry');
        $I->fillField('pass','pass-perry');
        $I->click('Login');
        $I->see('You\'re logged!');
        $this->cookie   = $I->grabCookie('your-session-cookie-name');
    }

    /**
     * @depends login
     */
    public function listUsers(AcceptanceTester $I)
    {
        $I->setCookie( 'your-session-cookie-name', $this->cookie );
        $I->amOnPage('/admin/users');
        $I->seeInCurrentUrl('/admin/users/1');
    }

    /**
     * @depends login
     */
    public function listRols(AcceptanceTester $I)
    {
        $I->setCookie( 'your-session-cookie-name', $this->cookie );
        $I->amOnPage('/admin/rols');
        $I->seeInCurrentUrl('/admin/rols/1');
    }
}

这样,如果登录测试失败,您将不会获得cookie,也不会通过其他测试。

我更喜欢这个注释而不是其他答案中建议的注释,因为如果你使用,你将始终在之前的测试中执行代码,并且测试将仅在登录后执行。@depends@before@depends

更新

下面还有另一个答案,https://stackoverflow.com/a/41109855/1168804 也可能对你有所帮助,因为自编写此答案以来,Codeception的框架已经发展。


答案 2

所有早期的答案都是旧的,现在它是直接用_before方法完成的,该方法将Actor类作为参数。

<?php

namespace Test\Api;

use ApiTester;

class TrainingCest
{
    public function _before(ApiTester $I)
    {
        $I->amLoggedInAs('kgkg');
    }

    public function _after(ApiTester $I)
    {
    }

    // tests
    public function testForLoggedInUser(ApiTester $I)
    {
    }

    public function anotherTestForLoggedInUser(ApiTester $I)
    {
    }

}

如果您只想为所有CEST文件登录一次,则可以使用全局注册表类实现注册表设计模式(请参阅 https://dzone.com/articles/practical-php-patterns/basic/practical-php-patterns-0)以及一些延迟加载。以下是我在Actor类中定义的api集成测试的工作代码(在我的例子中是ApiTester):

public function amLoggedInAs($userLogin)
{
    $I = $this;

    if (Registry::getInstance()->exists($userLogin)) {
        // get data from registry
        $storedUserData = Registry::getInstance()->get($userLogin);
        $newAccessToken = $storedUserData['accessToken'];
        $playerId = $storedUserData['playerId'];
    }
    else {
        // no registry data - log in and save data in registry
        $I->tryToLogin($userLogin);

        $newAccessToken = $I->grabDataFromResponseByJsonPath('data.newToken');
        $playerId = (int)$I->grabDataFromResponseByJsonPath('data.userId');
        Registry::getInstance()->set($userLogin, [
            'accessToken' => $newAccessToken,
            'playerId' => $playerId
        ]);
    }

    // finally set headers and some other data
    $I->haveHttpHeader('X-Token', $newAccessToken);
    $I->havePlayerId($playerId);
}

protected function tryToLogin($userLogin)
{
    $I = $this;

    $I->wantTo('login into api');
    $I->amGoingTo('try to log to API using login and password');
    $I->sendPOST('/system/login', ['login' => $userLogin, 'password' => self::getPassword($userLogin)]);

    // ...some other checking if user was correctly logged in ...
}

此代码基本上在用户首次登录后将 accessToken 以及一些其他数据存储在注册表中。如果调用对$I->amLoggedInAs('kgkg')的另一个调用,它将从注册表中获取这些值。您可以通过这种方式拥有许多已登录的用户,每个用户每个套件仅记录一次。

您可以使用另一种方法进行自动化,而不是自定义令牌,逻辑仍将相同。

此外,如果您使用的是WebDriver(不是PhpBrowser),则可以使用loadSessionSnapshot和saveSessionSnapshot而不是Registry来获得完全相同的结果。


推荐