Пых
2.19K members
41 photos
4 videos
3 files
117 links
Блог @vudaltsov о разработке на PHP

Канал https://t.me/phpyh |
Статьи https://medium.com/phpyh |
Патреон https://patreon.com/phpyh |
Обсуждение https://t.me/joinchat/Ad64MkZkLXybsAlWzKPLlQ |
Статистика https://tgstat.ru/channel/@phpyh
Download Telegram
to view and join the conversation
⚠️ Небольшое сообщение для тех, кто обновил Symfony до 4.4.14 / 5.1.6

Обратите внимание, что рядом с вашим src/Kernel.php появился автогенерируемый файл .preload.php (#38140, #38142, документация).

В соответствии с обновленным рецептом его нужно загитигнорить, а также добавить в исключения статических анализаторов во избежание ошибок, связанных с косвенным анализом классов vendor через этот файл.

Спасибо @Taarim за наводку 😊
Приглашаю всех на стрим о рефакторинге!

В четверг вечером я с Леонидом Корсаковым буду в прямом эфире "наперегонки" комментировать и рефакторить одну известную open source библиотеку. Посмотрим, что из этого выйдет 😅

1 октября / четверг / 19:00

Организаторы из Skyeng обещают призы 🎁:
🐘 фирменного PHP слона за лучшую зрительскую историю о рефакторинге,
🔑 несколько лицензий на PhpStorm за вопросы в эфире голосом!

https://www.youtube.com/watch?v=liMyAuxIoyM
Подкаст { Между скобок } про статический анализ

На прошлой неделе Сергей Жук предложил мне обсудить статанализ, и я был бы не я, если бы упустил возможность ввернуть пару монологов про Psalm 😆

И вот о чём мы поговорили:
• почему PHP пошел в сторону строгой типизации,
• для чего нужны статические анализаторы и как они работают,
• будущее статических анализаторов.

https://soundcloud.com/between-braces/11-valentin-udaltsov-staticheskiy-analiz
Audio
Несколько перлов с подкаста для затравки 🍉
Четверо в YouTube, не считая кота 🐈

Вчера у нас получился любопытный стрим про рефакторинг. Приятная компания, интересное задание и достойная организация. Спасибо всем, кто слушал, комментировал и потом писал в личку

Вкратце для тех, кто ещё не посмотрел. Я поленился по-честному отрефакторить yiisoft/validator, вместо этого воспользовался лазейкой "мажорный релиз" и предложил альтернативную архитектуру для библиотеки 😅

Что у меня получилось: https://github.com/phpyh/validator.
Мои пояснения на вчерашнем стриме: https://youtu.be/liMyAuxIoyM?t=4308.
⚠️ Команда Symfony снова шалит в патч-апдейте 4.4.15 / 5.1.7 😏

Если вы вносили правки, описанные мной в посте от 28 сентября, то придётся их откатить: удалить src/.preload.php и убрать соответствующие правила из .gitignore и конфигов инструментов.

После этого нужно добавить в репозиторий (без игнора) один из вариантов файла config/preload.php из обновленного рецепта. В папку с конфигами у меня статанализ не ходит, поэтому на этот раз исключение добавлять не пришлось.

Если у вас ядро располагается в нестандартном месте, то файл var/cache/prod/App_KernelProdContainer.preload.php будет называться иначе. Вот как получить верный путь:

bin/console c:c --env=prod
find var/cache/prod -name '*.preload.php'
Ванлайнер для получения данных объекта без рефлексии

https://3v4l.org/adhBn

Что почитать на эту тему:
документацию по Closure::call(),
• статью Accessing private PHP class members without reflection от @Ocramius.
В преддверии новой самоизоляции хочу поделиться с вами роликом, который мы сделали в апреле, чтобы улыбнуть и улыбнуться 😊

Всем отличного настроения, несмотря ни на что!

🎶🥁💃🥭

https://www.youtube.com/watch?v=cyZ2bJ0rQQM
Функция для получения всех типов класса

Такая функция потребовалась мне для поиска обработчика объекта по типу. Генератор здесь позволяет не рефлексировать раньше времени.

https://3v4l.org/CBuMQ
Пых
Ванлайнер для получения данных объекта без рефлексии https://3v4l.org/adhBn Что почитать на эту тему: • документацию по Closure::call(), • статью Accessing private PHP class members without reflection от @Ocramius.
Выяснилось, что в предложенном мной виде функция objectToArray неуниверсальна ☹️

Для объекта встроенного класса (например, DateTimeImmutable или stdClass) вызов Closure::call бросит ошибку уровня E_WARNING и вернёт null, так как замыкание нельзя привязать к области видимости встроенного класса.

"Встроенность" можно проверить через ReflectionClass::isInternal, а можно просто подавить ошибку и обработать null, выбирайте по религии ситуации 😉

В чистом виде get_object_vars применима к любым объектам и всегда возвращает массив публичных нестатических свойств. У объектов встроенных классов кроме stdClass он, как правило, пустой.

Обновлённый вариант функции с рефлексией https://3v4l.org/2mtXC и без https://3v4l.org/RqDHu.
Не игнорьте .idea в проекте

До сих пор встречаю проекты, где в .gitignore есть правила типа .idea/, .vscode/ или *.swp, поэтому решил написать этот пост.

Представим, что каждый разработчик решил использовать свою IDE с уникальными временными файлами и предложил PR на изменение .gitignore. Думаю, проблема очевидна. Код должен быть IDE-агностик, то есть не зависеть от среды разработки.

Это элементарно достигается настройкой глобального .gitignore на машине разработчика. Каждый прописывает себе те исключения, которые необходимы для его ОС, IDE и прочих инструментов, а затем спокойно открывает любой проект, не боясь закоммитить лишнее.

Всего лишь две команды в чек-лист вашего онбоардинга...

git config --global core.excludesfile ~/.gitignore
echo '.idea/' >> ~/.gitignore
Уже месяц используем Composer 2 RC в продакшне, рекомендую!

Главное улучшение — это, конечно, превосходная скорость без всяких плагинов вроде hirak/prestissimo или symfony/flex.

Понравилась возможность выполнить composer require/remove <name> --dry-run и безболезненно узнать, что изменится.

Также меня заинтересовало, почему задепрекейтили мою любимую опцию --no-suggest. Пояснение "из уст" автора нашёл только в issue #8615: в новой версии предложения необъёмные и ненадоедливые.

Подробно об изменениях:
• на русском https://sergeymukhin.com/blog/composer-2-chto-novogo-i-izmenennogo,
• на английском https://php.watch/articles/composer-2.

composer selfupdate --preview
Параметры vs Аргументы

Параметр — это переменная в сигнатуре функции. В PHP параметры характеризуются именем, типом, позицией, значением по умолчанию, вариативностью (...) и возможностью передачи по ссылке (см. ReflectionParameter).

Аргумент — это выражение/значение, используемое при вызове функции.

Именно поэтому $reflectionMethod->getParameters(), но func_get_args и InvalidArgumentException.

На мой взгляд простой, но важный нюанс 👌
Завтра выступаю на PHPFest 🚀 с докладом про утечки памяти 💧

Буду говорить и лайвкодить на следующие темы:
• как следить за памятью в консольных командах;
• эффективная декомпозиция кода;
• copy-on-write;
• $real_usage в memory_get_usage();
• циклические ссылки наглядно;
• WeakReference, WeakMap;
• как следить за памятью в воркерах.

https://2020.phpfest.ru/lecture/3

Буду ждать вас на странице трансляции в 10:00 по Москве (в 14:00 по Новосибирску)!
Разыгрываю билет на PHPFest 2020!

Задача

https://gist.github.com/vudaltsov/4c1fe1c2c2ad0f2a3850f5ef228e00db

Перед вами код фабрики тяжёлых отчётов. Отчёт генерируется на лету по запросу. В процессе фабрика может быть вызвана несколько раз для одного и того же экземпляра данных, поэтому предусмотрен in-memory кэш. В качестве сервера используется RoadRunner.

Как отрефакторить код, чтобы память не текла в PHP 7? В PHP 8?

Условия участия

• Нужно прислать правильные ответы на оба вопроса через @phpyh_phpfest2020_contest_bot (не через личные сообщения).
• Ответ для PHP 8 должен отличаться от ответа для PHP 7.
• Ответы принимаются сегодня до полуночи по Москве.
• Чтобы повысить шансы, нужно прислать простое и лаконичное решение как можно раньше.

Приз

Победителя определю я и опубликую его имя и решение утром. Герой получит доступ в личный кабинет и возможность посмотреть все доклады в записи в удобное время.

Удачи! 😊
Итоги конкурса!

Как я проверял ваши решения 🤓

Если ответ был текстовый, то я оценивал правильность идеи. Если ответ был кодом, то я тестировал его при помощи InMemoryCachingReportFactoryTest.php, предварительно адаптировав тест, если были изменены названия классов или добавлены зависимости. Если тест крашился из-за банальной ошибки (например, неинициализированного типизированного свойства), я такой ответ не засчитывал. Если в коде визуально были неточности, но мой несложный тест он проходил, я засчитывал. Думаю, это справедливо, всё-таки совсем непротестированный код на прод заливать не стоит 😉

Победитель 🏆

Первый правильный ответ пришёл от Andrew (@Groonya):
https://gist.github.com/aivchen/39a956afc47634c7f8bb2de4688aa7c1 (PHP 7),
https://gist.github.com/aivchen/5c8d51bed4188910ed34660c3b92fea0 (PHP 8).

Интересно, что в первом сниппете Андрея тоже есть неинициализированное свойство $reports, однако оператор ??= на это по понятным причинам не ругается, поэтому формально код корректный.

Пояснение от меня 💡

Эту задачу я рассмотрю сегодня в рамках своего доклада. Присоединяйтесь к PHPFest через 2 часа по ссылке https://2020.phpfest.ru/showtime/, всех жду!
Сегодня в начале выступления была заминка из-за того, что у меня не работала команда bin/console demo в специально подготовленном для лайвкодинга проекте.

По скрину поймёте, почему 😂

Как известно, Symfony пробегает все файлы при прогреве кэша в dev. Соответственно, при автозагрузке демонстрационного класса App\Worker в конце файла запускался этот самый воркер и начинал свой бесконечный цикл. Добавил эти строки я на последнем прогоне доклада и конечно же забыл удалить, поэтому сегодня утром "подвисание" команды для меня было полной неожиданностью...

Неудивительно, что перезагрузка мака меня не спасла. В итоге я успешно показал всё на рабочем проекте, пока фоновый процесс в моей голове пытался понять, почему тут работает, а там нет.

🤦🤦‍♂️🤦‍♀️🙈