Vkontakte через rhythmbox

March 25th, 2010

Скриншот с конечной реализацией:
image

Должен сразу заметить, что официальная документация довольно скудная, поэтому за образец был взят rhythmbox-youtube, который к слову, у меня при запуске никаких поисковых результатов не выдал, ну да и ладно, основную свою роль как образца он выполнил.

Код плагина тут: http://github.com/grunichev/rhythmbox-vkontakte

Плагин не требует авторизации вконтакте, работает через API.

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

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 для отображения картинок

September 27th, 2009

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

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

который сам будет отображать ту или иную картинку.
Приступим. Read the rest of this entry »

Получение root привелегий при сохранении файла

September 20th, 2009

В очередной раз редактировав конфигурационный файл в /etc директории при сохранении обнаружил что забыл вызвать vim через sudo и как следствие – permission denied.
Понятно что можно извратиться – сохранить в другое место, потом переписать и пр. но оказывается есть способ намного элегантнее.
Основан на использовании команды tee, о которой до этого не подозревал.
Ну и само решение:

:w !sudo tee %

% заменяется на имя редактируемого файла, остальное все должно понятно.
vim обнаружит что файл был изменен и выдаст соответсвующее предупреждение:
Warning: File “file.name” has changed and the buffer was changed in Vim as well
и сразу можете спокойно выходить – изменения сохранены.

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

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