Archive for the ‘PHP’ Category

Передача объектов по ссылкам и значениям в PHP

Saturday, March 26th, 2011

Дописывая свой проект столкнулся с проблемой передачи объектов в PHP, о чем и хотел бы тут все расписать, мало ли кому пригодится.

Общеизвестно, что с 5ой версии PHP все объекты передаются по ссылке. Но что же стоит за этой фразой?
Пример первый:

< ?php
session_start();
class My {
        public $i = 0;
}
 
if (isset($_SESSION['my_int'])) {
        $my_int = $_SESSION['my_int'];
} else {
        $my_int = 0;
        $_SESSION['my_int'] = $my_int;
}
if (isset($_SESSION['my_obj'])) {
        $my_obj = $_SESSION['my_obj'];
} else {
        $my_obj = new My();
        $_SESSION['my_obj'] = $my_obj;
}
echo('my_int=' . $my_int++);
echo(' my_obj.i=' . $my_obj->i++);
?>

(more…)

Модульное тестирование приложений, использующих файловую систему

Monday, October 5th, 2009

Для примера возьмем ситуацию:
у пользователя в профайле есть аватар – либо собственный если присутсвует файл userId.png, либо аватар по умолчанию:

<?php
$avatarPath = $this->baseUrl('/images/avatars/' . $this->userId . '.png');
if(!file_exists($_SERVER['DOCUMENT_ROOT'] . $avatarPath)) {
    $avatarPath = $this->baseUrl('/images/avatars/default.png');
}
?>
<img src="<?php echo $avatarPath; ?>" alt="avatar"></img>

В данном случае логика отображения зависит от наличия или отсутсвия в файловой системе того или иного файла. Эту зависимость в тестах надо устранить. На помощь приходит виртуальная файловая система и wrapper к ней: vfsStream
С помощью данного streamWrapper’a мы полностью исключаем зависимость тестовой от файловой системы.
Пример теста:

    require_once 'vfsStream/vfsStream.php';
 
    private function setUp()
    {
        // prepare vfs
        vfsStreamWrapper::register();
        vfsStreamWrapper::setRoot(new vfsStreamDirectory('document_root'));
        $_SERVER['DOCUMENT_ROOT'] = vfsStream::url('document_root');
        // create test data
        $avatarPath = 'images/avatars/77.png';
        mkdir($_SERVER['DOCUMENT_ROOT'] . DIRECTORY_SEPARATOR . dirname($avatarPath), 777, true);
        $fp = fopen($_SERVER['DOCUMENT_ROOT'] . DIRECTORY_SEPARATOR . $avatarPath, 'w');
        fclose($fp);
    }
 
    public function testGetDefaultAvatar()
    {
        $this->dispatch('/user/avatar/id/10');
        $this->assertXpath("//img[@src = '/images/avatars/default.png']");
    }
 
    public function testGetCustomAvatar()
    {
        $this->dispatch('/user/avatar/id/77');
        $this->assertXpath("//img[@src = '/images/avatars/77.png']");
    }

В заключение следует добавить, что есть случаи в которых vfsStream не поможет, из-за ограничений php-шных stream wrapper’ов.

Создаем Zend_View_Helper для отображения картинок

Sunday, September 27th, 2009

Задача – создать хелпер для отображения картинок во вьюхах, не заботясь каждый раз о путях.
Наше приложение поддерживает различные скины (используются стандартные зендовские layout’ы), структура папок следующая:
/layouts/default/images/myfile.png
/layouts/otherskin/images/myfile.png
в зависимости он настройки resources.layout.layout в конфигурационном файле, будет показана та или иная картинка (равно как та или иная вьюха и пр.)
В клиентском коде же хочется иметь универсальный механизм:

$this->image('myfile.png');

который сам будет отображать ту или иную картинку.
Приступим. (more…)

Валидатор для полей подтверждения в Zend_Form

Sunday, September 13th, 2009

Zend_Framework предоставляет абстрактный класс Zend_Validate_Abstract наследовав который, можно создать собственные валидаторы.

Ключевой метод: isValid на вход которому передается собственно само валидируемое поле первым аргументом и весь исходный набор вторым, что в нашем случае очень удобно.

Создаем класс:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
<?php
/**
 * Zend_Form validator for confirmation any fields
 * Example of usage:
 *     addValidator(new App_Validate_Confirmation('fieldName')
 *
 */
class App_Validate_Confirmation extends Zend_Validate_Abstract
{
    const NOT_MATCH = 'notMatch';
 
    protected $_matchedField;
 
    protected $_messageTemplates = array(
        self::NOT_MATCH => 'confirmation does not match'
    );
 
    public function __construct($fieldName)
    {
        $this->_matchedField = $fieldName;
    }
 
    public function isValid($value, $context = null)
    {
        $value = (string) $value;
        $this->_setValue($value);
 
        if (is_array($context)) {
            if (isset($context[$this->_matchedField])
                && ($value == $context[$this->_matchedField])
            ) {
                return true;
            }
        } elseif (is_string($context) && ($value == $context)) {
            return true;
        }
 
        $this->_error(self::NOT_MATCH);
        return false;
    }
}

Пример использования в форме:

    $password = $form->createElement('password', 'password', array('label' => 'password'));
    $password->addValidator('stringLength', false, array(6, 20))
             ->addValidator(new App_Validate_Confirmation('passwordConfirm'))
             ->setRequired(true);
 
    $passwordConfirm = $form->createElement('password', 'passwordConfirm', array('label' => 'confirm password'));
    $passwordConfirm->addValidator('stringLength', false, array(6, 20))
                    ->setRequired(true);

Данный валидатор можно применять и к нескольким полям формы, например, если также необходимо создать поле подтверждения email’a