Пишем REST API сервис на Go - УЛЬТИМАТИВНЫЙ гайд

  Переглядів 92,790

Николай Тузов — Golang

Николай Тузов — Golang

День тому

Пишем полноценный REST API сервис URL Shortener - это будет не игрушечный проект, а полностью готовый к использованию:
- выберем для него актуальный http-роутер: go-chi/chi
- Позаботимся о логах: slog
- Напишем тесты - unit-тесты, тесты хэндлеров и функциональные
- Настроим автоматический деплой через GitHub Actions, напишем для этого workflow
- и др.
Облачный сервер, который я использовал: slc.tl/torpo
Репозиторий проекта: github.com/GolangLessons/url-...
Другие мои ролики, дополняющие текущий:
Интерфейсы по месту использования: • Почему интерфейсы лучш...
Моки и их генерация: • Генерация и использова...
Использование SQLite в Go: • SQLite в Golang - как ...
✍Текстовый вариант гайда: habr.com/ru/companies/selecte...
Буду очень благодарен за вашу поддержку и там ❤
----
👾 t.me/ntuzov - мой канал в Telegram-канал.
Пишу в нём много интересного: гайды, которых нет на UKposts, интересные мысли про разработку, новости и анонсы всех моих активностей и др.
👀 GoLang Digest: t.me/golang_digest - мои регулярные подборки интересных материалов по Go.
🗣️ Наше сообщество GopherClub: t.me/+zsSZ63wEJDs3NGVi
Лучшее русскоязычное Go-сообщество с очень приятной атмосферой, без токсиков. Вежливо и терпеливо помогаем новичкам, конструктивно дискутируем с профессионалами и т.п.
Здесь также присутствуют все звезды Go-сообщества и представители интересных компаний 😄
❤️ Если у вас есть желание поддержать развитие канала:
Секретный телеграм-канал:
- В рублях: t.me/+1UPXV_DGnG1mODJi
- В евро: t.me/+hedI8LevYTc5MDM6
Boosty: boosty.to/nikolay.tuzov
Patreon: / tuzov
----
Тайм-коды:
00:00 Вступление
01:28 Почему мой Telegram-канал очень важен
03:05 Про папку cmd
03:46 План работ и описание используемых библиотек
07:46 Конфигурация приложения и работа с конфигами - CleanEnv
21:34 Настройка логгера - slog
30:25 Пишем Storage - БД / хранилище данных сервиса - SQLite
43:56 SaveURL() - пишем метод Storage для сохранения URLов
50:01 GetURL() - метод Storage для получения URLов
51:40 DeleteURL() - упражнение для самостоятельной работы
52:29 Создаём роутер - Chi
53:27 Middleware для роутера - что это?
54:30 Подключаем Middleware: RequestID и RealIP
56:19 Middleware для логирования запросов
01:02:43 Middleware: Recover и URLFormat - удобный парсинг URL-параметров
01:04:20 Pretty Logger - крутые красивые логи для локальной разработки
01:08:35 Handler: Save - обработчик запросов на сохранение URL
01:35:09 Создание и запуск HTTP сервера
01:38:08 Пишем тест для хэндлера Save
01:52:13 Функциональные тесты - что это такое, и чем они лучше?
01:53:37 Handler: Redirect - редиректим пользователя на сохранённый URL
02:00:23 Handler: Delete - упражнение для самостоятельной работы
02:01:37 Авторизация - ограничение прав доступа к некоторым хэндлерам
02:07:03 Авторизация: как её протестировать с помощью Postman
02:08:06 Пишем тест для хэндлера Redirect
02:13:18 Функциональные тесты - тестируем приложение как черную коробку
02:28:23 Настраиваем деплой проекта на удалённый сервер
02:28:46 Покупаем сервер у Selectel
02:36:00 GitHub Actions: настройка автоматического деплоя проекта
02:37:38 GitHub Actions: Пишем Worflow для деплоя
02:47:27 systemd: настройка автоматического запуска сервиса
02:48:58 Запускаем и проверяем деплой через наш Workflow
02:50:32 GitHub Secrets: хранение приватной информации для деплоя
02:52:44 Успешный деплой через наш Workflow
02:53:18 Тестируем наш сервис на удалённом сервере
02:55:21 Заключение
#golang #ntuzov

КОМЕНТАРІ: 204
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
👾Мой канал в Telegram: t.me/ntuzov Пишу там новости, анонсы своих активностей и просто интересные мысли Также с его помощью я получаю от вас оперативный фидбэк по роликам - что нравится, что не нравится, какой ролик делать следующим и т.п. ❤ Если у вас есть желание поддержать развитие канала: Секретный телеграм-канал: - В рублях: t.me/+1UPXV_DGnG1mODJi - В евро: t.me/+hedI8LevYTc5MDM6 boosty.to/nikolay.tuzov www.patreon.com/tuzov
@STAP2011
@STAP2011 10 місяців тому
Приятно видеть полезные содержательные видео для новичков на великой Гошечке без всяких ужасностей - сразу к делу, качественно на уровне продакшн кода и без воды, уважаемо!
@vitiok78
@vitiok78 10 місяців тому
Только начал смотреть, но уже по содержанию вижу, что это золотое видео! Спасибо!
@wat4mon
@wat4mon 10 місяців тому
ЛУЧШЕЕ ЧТО МОЖЕТ БЫТЬ В СУББОТНЕЕ УТРО. СПАСИБО ОГРОМНОЕ ЗА КОНТЕНТ. ПРОСТО ЛУЧШИЙ!!!!
@paniciour
@paniciour 10 місяців тому
э, что угодно?..
@vladosssss1088
@vladosssss1088 8 місяців тому
Спасибо за такой отличный урок, впервые просмотрела урок до конца. Узнала очень много нового и полезного, очень помогают твои подробные объяснения что и для чего нужно. Спасибо
@user-qp3lt4ps8c
@user-qp3lt4ps8c 5 місяців тому
Отличный гайд, Николай. Очень познавательно и интересно смотреть твое творчество. Респект!
@Iongjump
@Iongjump 10 місяців тому
Спасибо за контент ❤
@oliverswanick4965
@oliverswanick4965 10 місяців тому
Даешь видео про миграции!
@radikovichkz2470
@radikovichkz2470 7 місяців тому
В golang нет миграций
@console.g
@console.g 3 місяці тому
В Турцию можно мигрировать
@unicoxr5tj417
@unicoxr5tj417 10 місяців тому
даешь практику для народа! Таким видосам тока лайк
@user-fd9bg1gm8z
@user-fd9bg1gm8z 10 місяців тому
Спасибо огромное за контент!
@user-ir3eg9uc9g
@user-ir3eg9uc9g 10 місяців тому
Спасибо за контент!
@idgalushko
@idgalushko Місяць тому
Николай, спасибо огромное за ваш труд и творчество! Благодаря вам, научился создавать веб-сервисы, которые очень удобно масштабировать, рефакторить и поддерживать, пересаживать на любые базы данных и реализации кэша, а также подключать несколько серверов, реализующих различные протоколы обмена данными! А всё благодаря разделению общей архитектуры на слои и интерфейсам! Просто топ, ещё больше стал любить писать на Go благодаря вам!
@nikolay_tuzov
@nikolay_tuzov Місяць тому
Рад, что помог ❤️ Всегда очень приятно видеть такие отзывы
@y0oshi2
@y0oshi2 2 місяці тому
круто объясняет и самое главное, наглядно все показывает на практике👌
@alpenveg
@alpenveg 10 місяців тому
Спасибо за видос!
@dmitriybogdanov6291
@dmitriybogdanov6291 10 місяців тому
Спасибо за труд!
@AndrewYurchenko
@AndrewYurchenko 8 місяців тому
Блин. А в видео - это где можно увидеть? )) Хорошо, что уже не первый день обучалки смотрю и знаю о таких "выкрутасах от блогеров" - читать в надежде комменты. Но, все же. Это баг!!! )))
@user-sp1gr6xb7q
@user-sp1gr6xb7q Місяць тому
Мужик, ты крут, спасибо тебе за такие видео. Действительно очень полезные!
@samoiloff90
@samoiloff90 10 місяців тому
Просто огонь контент
@mini_clop
@mini_clop 10 місяців тому
Утро наступило!
@RusFarFaz
@RusFarFaz 9 місяців тому
Хотелось бы про concurrence и паттерны их использования. Например воркеры, fan-in , fan-out
@svfastunov
@svfastunov 10 місяців тому
Очень классный и полезный материал. Темы по наблюдению за всем этим хозяйством не хватает.
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
По мониторингу точно будет отдельное видео. У меня уже есть материал, я хотел в этом видео о нем рассказать. Но потом решил всё же отдельно, т.к. ролик слишком большой получался.
@purplecorvette
@purplecorvette 8 місяців тому
почему не сетится CONFIG_PATH фатал печатает что конфиг пас не установлен(
@aleksandrpetrov3938
@aleksandrpetrov3938 Місяць тому
Видимо имеется ввиду что запускать мы будем основной скрипт так CONFIG_PATH=./config/local.yaml go run main.go - и у нас есть свобода выбора какой путь к конфигу указать. Если на проде, то файл уже другой просто указываем и всё
@user-wv7gz3lg4b
@user-wv7gz3lg4b 4 місяці тому
Видео очень хорошее, спасибо автору! Если есть возможность, можно ещё видео про деплой не через гитхаб, а через гитлаб ci/cd? Было бы 🔥
@dn.kolesnikov
@dn.kolesnikov 10 місяців тому
Про slog видео будет в паблике? Интересно твоё видение его кастомизации и настройки.
@crazy_fedor
@crazy_fedor 5 місяців тому
Николай, материл от вас - просто супер. Выпустите пожалуйста видео про миграции! 🌷
@nikolay_tuzov
@nikolay_tuzov 5 місяців тому
А что именно интеренсо про миграции услышать? Помимо того, что было показано в этом видео
@user-ed6qc1pn5o
@user-ed6qc1pn5o 10 місяців тому
Огромное спасибо! Таких видео очень не хватает, особенно на русском. Сегодня вечером просмотрю видео и сам пройду все шаги. У вас вижу GoLand, а я в vscode пишу. Но, думаю, из-за этого сложностей не будет.
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Сложностей быть не должно. IDE - это лишь вспомогательный инструмент. Можно хоть в блокноте писать =)
@paniciour
@paniciour 10 місяців тому
​@@nikolay_tuzov не, не можно) интеграция гита, бд, рефакторинг, навигация - у вскода из коробки ничего этого нет, можно обвешаться гирляндой расширений, но зачем если можно просто переактивировать голанд раз в месяц на диспозабл емейл 💀
@jojogay7297
@jojogay7297 10 місяців тому
@@paniciour а че так можно реактивировать?
@paniciour
@paniciour 10 місяців тому
@@jojogay7297 я этого не говорил, меня взломали)
@olegk4041
@olegk4041 10 місяців тому
@@paniciourили neovim поставить, только выиграешь в долгосрок
@dizelvinable
@dizelvinable 10 місяців тому
Ух, супер!
@dimmodddimmodd7199
@dimmodddimmodd7199 10 місяців тому
Спасибо!
@Evg2002
@Evg2002 2 місяці тому
спасибо, все очень круто!
@soundcloudlover
@soundcloudlover 9 місяців тому
Будут ли видео по внутренностям go? А именно как точно устроена сборка мусора и как работает планировщик/аллокаторы? спасибо
@tumenit
@tumenit 10 місяців тому
Спасибо!!!
@PlayGameToday
@PlayGameToday 8 місяців тому
Привет, как называется модель кресла твоя? )
@vitiok78
@vitiok78 10 місяців тому
Миграции очень нужны!
@alexsandr9444
@alexsandr9444 8 місяців тому
Да нужно про миграции
@ginger.samurai
@ginger.samurai 10 місяців тому
ЛУЧШИЙ
@alexobzor
@alexobzor 8 місяців тому
По поводу тестирования хэндлера save, почему запрос NewRequest берётся из библиотеки http, а не httptest? В основе теста же лежит именно Response Recorder, соответственно не создаётся сервера на loop back, ну и нужды в реальном запросе нет, тогда почему бы не юзать из httptest?
@eleimt
@eleimt 7 місяців тому
подскажите, про миграции есть видео?
@artmon2004
@artmon2004 10 місяців тому
Супер! Пробовал пройти cоревнование Codeforces на стажировку в OZON, занял 300 место из 900 примерно, народу тьма в Go идет))))))
@hurricane-rus
@hurricane-rus 4 місяці тому
Хорошее видео, разобрано много интересных подходов, спасибо. До хэндлеров было в целом понятно, потом объяснения пошли очень сжато и кратко, приходилось переслушивать 3-4 раза, чтобы стало понятно. Код хэндлеров очень плохо читается из-за кучи логов - может быть, их можно обработать как-то поэлегантнее? В будущих видео хочется как можно меньше моментов в стиле "просто добавьте сюда этот код, который я не буду объяснять" - по мне такой код лучше вообще не добавлять в проект (ну или потратить дополнительное время на его объяснение). P.S. Не понял, зачем нужно было делать столько лишних папок в проекте? (фактически каждый файл лежит в отдельной папке) Т.е., например, зачем папка sqlite в папке storage? Можно сразу storage.go кинуть в storage. Или в lib/logger папка sl кажется лишней - можно сразу sl.go кинуть в logger, и все. Если поудалять все эти лишние папки, дерево проекта будет намного меньше. Чем мне мешают эти дополнительные папки - они затрудняют ориентирование в проекте и сильно раздувают дерево проекта. А поскольку программист в основном читает код, то чем компактнее будут уложены файлы, тем лучше.
@ThePirateHistory
@ThePirateHistory 5 місяців тому
Было бы классно ещё понимать как описать то что ты сделал в комите, допустим сделали логер, он работает для теста вывели пару сообщений, как записать просто "Initialized logger" и всё, или допустим создали какую-то структуру, вывели возможно менять конфигурацию логера в конфиг файл
@jimshtepa5423
@jimshtepa5423 9 місяців тому
в самом начале когда вы перешли к написанию кода и начали создавать папку cmd, вы использовали какой то пакет который генерирует шаблон для проекта? (типа как vitejs на javascript чтобы собрать шаблон проекта?). или вы сами создали паку url-shortener и внутри создали файлы вручную?
@user-zg8ij3kt1h
@user-zg8ij3kt1h 8 місяців тому
Я ещё не досмотрел, но вопрос, пока не забыл: как сюда подключить ещё и обработку параметров запуска из консоли (ключей типа --config)? Как их правильно с обработкой конфига дружить?
@mrfofaify
@mrfofaify 7 місяців тому
Сделайте видео с проектом по чистой архитектуре
@vitiok78
@vitiok78 10 місяців тому
Последнее время я всё реже использую автоматические id в базе данных. Именно из-за того, что id может быть неизвестен сразу после вставки. Я генерирую uuid, который не полностью рандомный, а типа последовательный. Таким образом, я заранее знаю id, и этот id не так сильно влияет на производительность базы, т.к. он уже отсортирован. Да, такое поле увеличивает размер базы, но оно слишком удобное, чтобы от него отказаться.
@vasyarodionov1369
@vasyarodionov1369 15 днів тому
а как же метод returnid?
@vitiok78
@vitiok78 15 днів тому
@@vasyarodionov1369 Не всегда возможно использовать. Особенно, когда в проекте ORM какая-то используется
@paniciour
@paniciour 10 місяців тому
Го видик по b-tree?) Не, го целый видик по имплементации базки с конкаренси, б-трии и фильтром блума!
@georgiy_kulagin
@georgiy_kulagin 10 місяців тому
кайф
@dovm5d
@dovm5d 2 місяці тому
48:08 - mySQL как раз таки поддерживает стандартным драйвером. PostgreSql стандартным драйвером точно не поддерживает, надо в конец запроса добавлять RETURNING id и вместо экзека отправлять запрос через QueryRow
@jimshtepa5423
@jimshtepa5423 9 місяців тому
какую команду выполнили когда запустили программу в самом начале чтобы убедиться что все конфиг настройки были учтены? просто git init? или что то еще
@cartfpv4610
@cartfpv4610 9 місяців тому
1. Rest коды не используютя по какой то причине? 2. Динамические/настраиваемые моки языком в принципе не поддерживаются? Все создавать в статике/ генерировать?
@vladimireliseev7602
@vladimireliseev7602 10 місяців тому
Здравствуйте, Николай! Благодарю за видео! Хотелось бы узнать - почему Вы возвращаете именно ссылку на конфиг, а не значение из функции MustLoad?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Поправлю - указатель, а не ссылку* Вообще, это тема для частых дискуссий - возвращать указатель или значение. У обоих вариантов есть свои плюсы и минусы, но в большинстве случаев разницы почти нет. Если возвращаем значение, то при возврате и передаче мы каждый раз будем копировать все значения структуры. Зато в случае указателя можем накосячить с разыменованием.
@vladimireliseev7602
@vladimireliseev7602 10 місяців тому
@@nikolay_tuzov благодарю!
@williba100n5
@williba100n5 2 місяці тому
Спасибо
@kotlinjava5228
@kotlinjava5228 10 місяців тому
Большое спасибо за ролик,и уважение за флаг Казахстана на фоне!
@olegk4041
@olegk4041 10 місяців тому
Прикольно, но отдельный ролик для https ты маханул конечно. Self-signed сертификат прям с Голанг идет из коробки.
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Self-signed не совсем то. Я обычно через Cloud Flare делаю, но при этом сильно тоже не заморачиваюсь, выбираю гибридный вариант (от клиента до CF по https, от CF до моего сервиса http) Заодно там можно будет показать, как настроить работу с доменом. Но в целом да, ролик получится короткий и простенький.
@8Sapphire8
@8Sapphire8 7 місяців тому
Ролик бесценный что тут сказать. 5 из 5
@nikolay_tuzov
@nikolay_tuzov 7 місяців тому
Спасибо ❤️
@kodikoff
@kodikoff 10 місяців тому
Очень ждём миграции
@bobmoclaat8965
@bobmoclaat8965 2 місяці тому
Почему config_path не найден пишет? Зачем мы тогда local.yaml создавали
@quiqlerbeats5933
@quiqlerbeats5933 22 дні тому
Тебе нужно путь до него в переменной окружения указать
@deuse7222
@deuse7222 9 місяців тому
Код по работе с БД и сетью же работает в один поток синхронно, если не ошибаюсь. В продакшен коде стоит сразу писать через горутины?
@nikolay_tuzov
@nikolay_tuzov 9 місяців тому
Нет, обычно в БД не ходят асинхронно, тк смысла нет - без похода туда обычно нечего делать дальше. А сам запрос обрабатывается в отдельной горутине и так
@ThePirateHistory
@ThePirateHistory 5 місяців тому
Решил делать как говорили в начале чтобы было аля своё, решил добавил логер в ямл конфиг, сделал как с http_server, всё работает за исключение что уровень логера с нулём просто не берёт выдаёт ошибку, ставлю допустим -4(дебагинг) всё работает, всё выводится как по видео, но если в кфг ямла ставлю 0 то всё, сам struct логера уровень логера int8
@datokhojava6421
@datokhojava6421 6 місяців тому
в ОП ubuntu имеет значение где я создам папку?
@alexsandr9444
@alexsandr9444 8 місяців тому
Привет спасибо за видео)Скажи а как задается "CONFIG_PATH" в окружение?
@boomy842
@boomy842 8 місяців тому
Если ты на линуксе, то export CONFIG_PATH= Если на виндовс то set CONFIG_PATH=
@kirillgrossberg6950
@kirillgrossberg6950 8 місяців тому
@@boomy842 Я правильно понял, тут получается set CONFIG_PATH=./config/local.yaml go run .\cmd\shrt\main.go ?
@user-pb2nv3ye9w
@user-pb2nv3ye9w 3 місяці тому
а это надо в терминал писать или куда? @@boomy842
@petery6775
@petery6775 8 місяців тому
подскажи, а чт оу тебя за тема и шрифт ?
@nikolay_tuzov
@nikolay_tuzov 8 місяців тому
Всё стандартное для IDE GoLand. В видео я использую presentation mode
@flexxx8586
@flexxx8586 3 дні тому
Подскажите пожалуйста, почему в каждом методе Storage , вы пересоздаете подготовленные выражения (stmt) при каждом вызове метода, а не ,например создаете специальный метод в котором эти выражения будут подготавливаться и уже потом сразу использоваться в методах?
@Damir-nl6tf
@Damir-nl6tf 4 місяці тому
Здравствуйте! Только начал изучать Go, хотелось бы узнать, то что пишет Николай в этом видео, обычно пишут Джуны Миддлы или Сеньоры? Спасибо всем за ответы!
@user-gy5lg4vp9i
@user-gy5lg4vp9i 3 місяці тому
ждём, бро. Но думаю без подобных проектов на тебя HR даже не посмотрят
@Daloshka
@Daloshka 2 місяці тому
export CONFIG_PATH=./config/local.yaml
@user-yt2sb9tl3p
@user-yt2sb9tl3p 10 місяців тому
38:39 почему используется log.Error и Os.Exit, экзит же насколько я понимаю отменяет все defer. Не лучше ли тут кидать панику?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Да, согласен, os.Exit может быть не лучшим вариантом, если используется defer. Тут нужно быть аккуратным.
@user-bb5xw8bd8w
@user-bb5xw8bd8w 10 місяців тому
@@nikolay_tuzov прибивать пустым мешком все открытые ресурсы - самый дубовый вариант
@madbad1310
@madbad1310 7 місяців тому
Николай, уже октябрь. Где анонсированное отдельное видео по log/slog ? )
@nikolay_tuzov
@nikolay_tuzov 7 місяців тому
Работы много, на видео очень мало времени остаётся. Но всё будет, планы в силе) Точную дату пока не могу назвать
@madbad1310
@madbad1310 7 місяців тому
@@nikolay_tuzov Николай, у вас очень полезные видео. В любом случае, спасибо вам большое.
@galahad667
@galahad667 Місяць тому
Лучший канал по гошке на русском ютубе!!!
@Roman-tm6qp
@Roman-tm6qp День тому
2:00:10 - насколько правильно передавать объект storage в хендлер роута? Не будет ли лучше это делать внутри хендлера (там где мы работаем с базой непосредственно)?
@alexeymatveev9031
@alexeymatveev9031 Місяць тому
Посмотрел анонс, но понял, что приснилось. Не может быть столько крутоты.
@nikolay_tuzov
@nikolay_tuzov Місяць тому
Сон наяву)
@anmill
@anmill 10 місяців тому
Ролик про миграции - плизззз!
@ThePirateHistory
@ThePirateHistory 5 місяців тому
1:14:09 как реализовать так чтобы наоборот, чтобы интерфейс был в отдельном файле в пакете sqlite. папка sqlite в том же месте что и у вас, затем внутри папка interfaces и там допустим save, и соответственно это надо как-то зарегистрировать в одноимённом файле пакета sqlite, как это сделать? Чтобы мы могли брать из одноименного файла sqlite и функцию save для сохранения url, а так же чтобы могли взять интерфейс в хандлере что в сервер, подскажите куда копать пожалуйста
@nikolay_tuzov
@nikolay_tuzov 5 місяців тому
Советую все подобные вопросы писать в Gopher Club. Они довольно сложные, не для комментов.
@ThePirateHistory
@ThePirateHistory 4 місяці тому
@@nikolay_tuzov Тг хорошо, но безумно не удобно, в тг просто 1 чат, и там люди общаются о своём, ты что-то запостишь какой-то вопрос он почти моментом скроется, а спамить этим вопрос это плохо, даже просто реплаить по кд такое себе, лучше дискорд, там можно создать каналы, чтобы просто общения было отделено от тех вопросов. Но как помню или тут в начале видео или из тех что в описании, был такой момент что был интерфейс Users в котором были как раз таки все методы и они были в отдельных папках, и получается что туда вручную записываются, то в файле допустим по уроку sqlite будет интерфейс по типу Users в котором будут интерфейсы, но как это правильно называется чтобы лучше разобраться вроде понял а вроде не до конца. И если сможете, то подскажите как называется практика при которой такие вещи они регистрируется, чтобы был допустим интерфейс в котором можно запушить свой другой интерфейс, чтобы вручную всё не писать, чтобы не было сотни интерфейсов которые содержат интерфейсы которые ссылаются на интерфейсы. Допустим User User.signUpMethod(saveDefaultData) и потом можно в любом месте User.saveDefaultData('...', '...')
@MrLotrus
@MrLotrus 9 місяців тому
Насколько распространена практика хранения констант с названием и путем функций для логирования? Выглядит как возможная мина и гемор при рефакторинге. Другого способа нет? А если получать в рантайме это считается плохой практикой?
@user-wh6ui3pz9q
@user-wh6ui3pz9q 10 місяців тому
Спасибо за видео! Если у storage'a и URLSaver'a будут немного отличаться сигнатуры методов, то где описывать адаптер?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
А зачем им отличаться? Их сигнатура должна совпадать. Может, такое и бывает, но я не сталкивался.
@user-wh6ui3pz9q
@user-wh6ui3pz9q 10 місяців тому
@@nikolay_tuzov если например сторедж в другом пакете, не логично же интерфейс по месту использования подстраивать под этот сторедж🤔
@jordenskraften8273
@jordenskraften8273 10 місяців тому
Будет ли что-то по golang+graphQL?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Возможно, будет. Но пока есть более важные и интересные темы на очереди
@bladesquirtsize2541
@bladesquirtsize2541 5 місяців тому
У меня ошибка когда я запускал программу после того как мы сделали функцию mustLoad выдавало ошибку что configPath пустой .Мне нужно создать env переменную CONFIG_PATH ?Чтобы решить эту проблему
@KirMC
@KirMC 3 місяці тому
Да, и она должна указывать путь до файла (понимаю вам скорее уже не надо, но вдруг кто увидит)
@user-rj5kt3ft8w
@user-rj5kt3ft8w 10 місяців тому
вместо op можно в логгерах включить caller чтоб показывало файл с конкретной строкой
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Это дорогостоящая операция, поэтому оно не всегда того стоит.
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Хотя, это не чрезвычайно дорого, и каких-то случаях допустимо, согласен. К примеру, обычно логгеры добавляют стек вызовов в случае Error-сообщений
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Кроме того, видеть имя функции вместо номера строки удобней. Потому что код может измениться, и номера строк сдвинутся. Придется выяснять, какая версия была задеплоена и т.п.
@user-rj5kt3ft8w
@user-rj5kt3ft8w 10 місяців тому
​@@nikolay_tuzov ну не знаю, можно и выводить какая функция выполняется, насчёт накладных расходов - не знаю, в zerolog\zap врубаю во время разработки или когда приложение на тестовом контуре
@paniciour
@paniciour 10 місяців тому
@@nikolay_tuzov op в upspin подсмотрел?) имхо полусомнительная приставка, тк вызывающий код и так знает название вызываемой функи, ему интереснее какой именно ее бит сбойнул. ты из каждой функи ретурнишь обернутые опом ошибки?
@batyrakpanbet6292
@batyrakpanbet6292 5 місяців тому
Видос шикарный, долго искал для практики. Сзадий флаг Казахстана?)
@nikolay_tuzov
@nikolay_tuzov 5 місяців тому
Да)
@abdulmagomedov
@abdulmagomedov 8 місяців тому
Как op расшифровывается?
@nikolay_tuzov
@nikolay_tuzov 8 місяців тому
operation
@user-zg8ij3kt1h
@user-zg8ij3kt1h 10 місяців тому
19:10 А почему нельзя сначала проинициализировать логгер? Как принято вообще?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Потому что мы инициализируем логгер различными способами, в зависимости от текущего конфига. Получается, для инициализации логгера нужно сначала получить конфиг.
@StreetWorkout62
@StreetWorkout62 10 місяців тому
А какой плагин ai используется для подсказки кода?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
GitHub Copilot
@StreetWorkout62
@StreetWorkout62 10 місяців тому
@@nikolay_tuzovон платный?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
@@StreetWorkout62 да, платный. Но там дают пару месяцев бесплатно попробовать. Если нужен бесплатный, то посмотри Codeium, говорят тоже хорош (но я не пробовал)
@user-lv9ko1ru7e
@user-lv9ko1ru7e 10 місяців тому
На мой взгляд коду рано в реальное использование 1) неоправданно сложная структура проекта(было тяжело читать код) + сервис очень напоминает код node+express 2) явное отсутствие слоёв приложения 3) разделение интерфейсов для обращения к бд на отдельные операции??? 4) отстутсвие контекста в методах для работы с бд(если бд находится локально, это не значит, что с таймингами будет всё ок. Например при обращении к бд может быть нагрузка на диск 100% и запрос будет схлопнут таймаутом который указали при конфигурации http сервера, что является не лучшим решением) + в серьёзных сервисах конфигурируют таймауты под каждый слой 5) цитата: "изза одного единственного запроса хэндлера не должно приложение падать целиком". Полностью несогласен! Для такого приложения с простой логикой все запросы "одинаковые" и если у нас происходит гдето паника, значит она будет происхдить постоянно, а в таком случае приложение работать не должно(его надо чинить) + дефолтный хттп сервер сам отлавливает паники 6) приложение никак не валидирует урлы, такое чувство, что это не сокращатель урлов, а переосмысленный редис. 7) time.Now() используется без часового пояса 8) нету Readme и Swagger(пришлось лезть в код, чтобы понять как запускать сервис) 9) save_test.go:83 require.Equal(t, rr.Code, http.StatusOK) - если быть невнимательным, то создаст весёлую проблему 10) хэлсчек! Как минимум это первое что проверяют если с сервисом проблемы, лучше всего делать через вызов db.PingWithContext(ctx)
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Спасибо за фидбэк, но с большинством пунктов в корне не согласен: 1) очень абстрактное заявление, довольно субъективный. В чем конкретно сложность? Что именно тяжело читать? 2) В предыдущем пункте вы жаловались на сложность, а добавление слоёв сложность увеличит. Их потому и нет - чтобы не переусложнять приложение. На мой взгля, в подобное проекте можно прекрасно обойтись без них. Если вы с этим не согласны, можем обсудить - приведите примеры, в которых у нас будут проблемы из-за отсутствия слоёв? 3) Да, а что с этим не так? Если подход непонятен, смотрите мой ролик на эту тему (ссылка есть в описании) 4) Тут согласен, контексты стоило добавить, это мой промах 5) "если у нас происходит гдето паника, значит она будет происхдить постоянно" - абсолютно неверное утверждение. У нас есть тесты, а значит, как минимум, happy path и некоторые fail-кейсы будут работать. Если паника и появляется, то точно не во всех случаях. Так зачем падать всему приложению целиком? Чтобы перестать обрабатывать даже корректные запросы? Что нам это падение вообще даст? 6) Посмотрите внимательней, валидация урлов - это вообще единственный валидатор, который у нас есть (помимо required), в хэндлере Save. 7) А зачем нам часовой пояс? Вы точно смотрели ролик, и просто бегло про коду прошлись? Во всем проекте всего 2 точки использования данной функции: подсчет дельты и задание сида в рандомайзере. Для чего там часовой пояс? 8) Readme не помешал бы, согласен. Но это явно не must have фича для работы приложения в продакшене, а скорее помощь себе в будущем. 9) Не очень понял, о какой проблеме речь? 10) Да, тут согласен, стоило сделать что-то простенькое. Но вообще, полноценный мониторинг будем делать в отдельном ролике
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Если хотите обсудить подробней, приглашаю вас в наш чат Gopher Club (ссылка в описании). Приходите, подискутируем. Заодно и сообщество рассудит, кто из нас прав. Там присутствуют и гораздо более опытные коллеги, чем я.
@user-lv9ko1ru7e
@user-lv9ko1ru7e 10 місяців тому
@nikolay_tuzov 1) да, тяжело читается, хотя кода немного 2) я думаю большинство тех, кто смотрел видео это джун или тот кто хочет устроиться джуном, и на мой взгляд стоило бы показать как "правильно делать", потому что придут они на работу и начнут слой данных выпиливать ссылаясь на это видео). Если это попытка в хайлоад/уменьшение расхода памяти/времени отклика, то стоило бы и с остальным кодом то же самое делать Про проблему отсутствия слоёв, она возникнет когда захотите сменить бд, sqlite хороша, но когда нужно будет уходить от неё, то всплывёт проблема отсутствия слоя данных(возможно не такая острая как в серьёзных сервисах, где извращаются над запросами в целях оптимизаций, но будет, как минимум сейчас нет возможности вынести в конфиг тип базы для сервиса и переключиться например на редис или монгу) 3) ничего не имею против разделения интерфейсов, но для сервиса на пару таблиц ваш подход избыточен(возможно было бы лучше оставить как есть и обернуть их в общий интерфейс) 5) грубо говоря сервис каждый раз проходит одни и те же проверки и если передавая два набора валидных данных, в котором один успешно проходит, а второй падает в панике, то это говорит о серьёзной ошибке в коде, которую надо чинить, а не пытаться восстановить прилагу(возможно гдето данные сломались и тогда нет никакого смыслы хранить битые данные) 6) сорри, не заметил 7) сорри, привычка после работы с распределёнными сервисами, не актуально для одного инстанса 9) стоит почитать сигнатуру require.Equal() func Equal(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) {
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
@@user-lv9ko1ru7e 1) Я всё ещё не вижу конкретных аргументов - что именно там тяжело? Приведите пример того, как было бы легко в данном случае? 2) Такое ощущение, что у вас есть некое универсальное правило, какая архитектура правильная. А это не так, она сильно зависит от проекта. Писать пет-проект с кучей слоёв, которые там не пригодятся - НЕ правильно. И допускать оверинжениринг в пет-проектах, это тоже плохо. Это не попытка оптимизации производительности, это отсутствие бессмысленного повышения сложности проекта - кода меньше, он проще, понятней. 3) Не соглашусь с этим утверждением 5) А если в сервис приходит 99 запросов, и лишь 1 из них проблемный и ломает приложение? Нам нужно уронить сервис, чтобы он обрабатывал 0 запросов, пока мы чиним тот единственный проблемный? Вы действительно считаете это хорошим подходом? 7) дело даже не в распределенных сервисах, а в том, зачем вообще используется функция time.Now(). Если она нужна только для получения дельты времени, внутри одной и той же функции, то при чем тут распределенность? В распределенном сервисе я бы тоже не стал думать здесь о временных зонах 9) Я знаю сигнатуру, и всё ещё не понимаю, о какой конкретно проблеме речь.
@user-lv9ko1ru7e
@user-lv9ko1ru7e 10 місяців тому
@@nikolay_tuzov лучше вынесите обсуждение в канал
@user-sp1gr6xb7q
@user-sp1gr6xb7q Місяць тому
Отличное видео, но я так и не понял 1:43:30 почему надо писать tc := tc?
@grigorym6107
@grigorym6107 9 місяців тому
Подскажите пожалуйста, где вы устанавливаете значение CONFIG_PATH, чтобы потом читать его в файле internal/config/config.go, в строке configPath := os.Getenv("CONFIG_PATH")?
@gorsaroyan1060
@gorsaroyan1060 7 місяців тому
Я запускаю так CONFIG_PATH=./config/local.yaml go run ./...
@alonewalker4932
@alonewalker4932 5 місяців тому
Привет! Если ты разобрался с этим подскажи пожалуйста решение данной проблемы
@user-pb2nv3ye9w
@user-pb2nv3ye9w 3 місяці тому
вы это делаете в терминале? @@gorsaroyan1060
@user-pb2nv3ye9w
@user-pb2nv3ye9w 3 місяці тому
а это в терминале вы запускаете или где? @@gorsaroyan1060
@TheWorpX
@TheWorpX 2 місяці тому
Я пофиксил таким образом (GoLand) Там где у нас есть ранер, тыкаем на стрелочку и тыкаем Edit Configurations Там у нас высветиться автоматически созданые ранер (это когда на зеленую стрелочку play нажимаешь около func main) Находим там поле Environment и прописываем туда CONFIG_PATH=./config/local.yaml Тыкаем Apply и стартуем
@user-ex2mo2sn8c
@user-ex2mo2sn8c 10 місяців тому
я возможно что то незаметил... но тут 2 слоя.. где usecases? прям из handler вызывать функции storage.. как то мне кажется не чисто
@paniciour
@paniciour 10 місяців тому
будь прагматичнее, если юзкейс дергает бд и ничего больше, то зачем плодить слои ради слоев
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Кажется, что в проектах такого уровня слишком много слоёв будет скорее вредно. Но это дискутивная тема, вряд ли тут есть однозначный ответ.
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Для иллюстрации приведу обратную крайность. В ООП языках порой встречаются люди, которые прочитали книгу по паттернам, и в первых своих проектах пытаются реализовать сразу все эти паттерны. Получается нечто ужасное и сложное в поддержке. Это называется - оверинжиниринг. С опытом же начинаешь лучше искать баланс между простотой, практичностью
@s1ovac308
@s1ovac308 10 місяців тому
Николай, а zap менее актуален, нежели slog?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Zap сейчас самый популярный, и slog до него далеко. Но slog актуальней в том плане, что он скоро будет в стандартной библиотеке, поэтому лучше начинать привыкать к нему. Но их сравнивать не очень корректно, тк slog сам по себе не совсем логгер, это лишь обертка. Можно Zap внутри slog использовать, например.
@danilakhtarov
@danilakhtarov 3 місяці тому
Можно вместо alias использовать id и каждой цифре присвоить свой символ, а-ля base64
@Trapmaloj
@Trapmaloj 20 днів тому
Подсобите кто сможет, сделал 1:1 но падает на этапе деплоя в VM, говорит неправильный ключ
@xlzpm9722
@xlzpm9722 10 місяців тому
кто может помочь, я уже тупо переписал код, а в консоли все равно пишется CONFIG_PATH is not set, go в vs code словно ничего не видет
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Нужно при запуске приложения указать эту переменную окружения. Заходи в наш чат (ссылка есть в описании), там уже разбирали этот вопрос подробно, и не раз
@AndrewYurchenko
@AndrewYurchenko 8 місяців тому
@@nikolay_tuzov 😂😂😂 вот где кроется маленький продажник, внутри прогера. Этот ответ, заслуживает оваций!!!
@AndrewYurchenko
@AndrewYurchenko 8 місяців тому
Блин, сел осваивать язык. Пока ищу, что можно тупо повторить, попутно въезжая в смысл. А тут роутер не стандарт... Хотелось бы пока с тем, что из коробки предоставляется разобраться. Но, что поделать... По любому благодарность!
@user-gy5lg4vp9i
@user-gy5lg4vp9i 4 місяці тому
как успехи броу? похожая сейчас ситуация у меня)
@AndrewYurchenko
@AndrewYurchenko 4 місяці тому
@@user-gy5lg4vp9i на тот момент отложил изучение Go, узнал про Rust. На нем освоил написание API. Но, за неимением вакансий и пока скудный спрос на рынке, вот попутно решил опять и Go добивать. После Rust намного легче вроде теперь заходит. Во всяком случае понимание работы с памятью там пришлось намного лучше освоить. Но, пока именно по Go сказать нечего. Как только доберусь до реализации, отпишусь. 👍
@user-gy5lg4vp9i
@user-gy5lg4vp9i 4 місяці тому
@@AndrewYurchenko в путь )
@tyapka
@tyapka Місяць тому
Запрос на следующие уроки: - Как сделать асинхронную работу (пусть запускается через API вызов) в кластере, чтоб она 1) по какому-то ключу была только одна во всем кластере и 2) востанавливалась после падения сервера (можно с использованием ZooKepera или etcd) - Реал лайф пример работы с Dragonboat (RAFT либа).
@conskykek
@conskykek 10 місяців тому
Оргазм
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
@fprotimaru1944
@fprotimaru1944 10 місяців тому
стоит ли потратить 10 баксов на copilot?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Стоит, конечно)
@fprotimaru1944
@fprotimaru1944 10 місяців тому
@@nikolay_tuzov спасибо за ответ
@megasuperlexa2
@megasuperlexa2 10 місяців тому
13:30 это kebab case
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
Да, вы правы, спасибо
@BabaykaMoscow
@BabaykaMoscow 10 місяців тому
А где определять интерфейс, если мест его использования - 17? о.О
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
А у вас часто такое бывает? На мой взгляд, это редкий кейс. И разбираться с ним нужно в каждом конкретном случае отдельно.
@BabaykaMoscow
@BabaykaMoscow 10 місяців тому
@@nikolay_tuzov Нет, мне просто интересно как это будет работать в Го :) После Жавы такой подход кажется необычным.
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
@@BabaykaMoscow в Го такое встречается намного реже. Если не ошибаюсь, я даже не встречал ни разу. Но тут важно понимать, что оно потому и встречается редко, что интерфейс описан в месте использования. Я об этом подробно рассказывал в ролике на данную тему, советую посомтреть
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
ukposts.info/have/v-deo/nYl5d3mfq26WuWg.html
@qoonmax
@qoonmax 13 днів тому
В чем смысл использовать файл конфига? Почему нельзя ограничится env + дефолтные значения в env-default. Выглядит как лишний слой.
@ntldrzic
@ntldrzic 10 місяців тому
почему не gin?
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
chi более минималистичный и лучше совместим с net/http. При этом в нём есть всё необходимое для такого проекта. Но вообще, мы обсуждали выбор конкретного варианта в моём ТГ-канале (ссылка есть в описании). gin тоже был среди кандидатов.
@research_Development
@research_Development 10 місяців тому
А что у тебя за кресло ?)
@nikolay_tuzov
@nikolay_tuzov 10 місяців тому
dxracer air
@brawlstarsbro5301
@brawlstarsbro5301 Місяць тому
афигеть он умный
@gooseman5578
@gooseman5578 5 місяців тому
idle-timeout = keep-alive
@Daloshka
@Daloshka 2 місяці тому
1ч 42 мин смотрю уже 8ч. Очень годный видос. Не понравилось тольок что во время написания кода, его не было, просто копипаст
@TechBusinessDev
@TechBusinessDev 24 дні тому
Поцаны на ларке все это за 15 минут сделают Когда уже фреймворк будет нормальный на гошке? Был бы опыт побольше, сам бы написал уже
@VakaramGolang
@VakaramGolang 6 місяців тому
почему губы с голосом не совпадают (
@nikolay_tuzov
@nikolay_tuzov 6 місяців тому
Бывает иногда такой рассинхрон, с ним порой сложно бороться, т.к. видео пишется на внешнюю камеру. Но, вроде, тут это почти незаметно, да и не очень важно.
@user-ju8xk9sh2m
@user-ju8xk9sh2m 5 місяців тому
Это видео сделано искусственным интеллектом, если ты не понял
Voloshyn - ЗУСИЛЛЯ (прем'єра треку 2024)
06:17
VOLOSHYN
Переглядів 700 тис.
Разбираем основы Kafka и RabbitMQ
26:54
Digital train | Alex Babin
Переглядів 8 тис.
Как на самом деле устроены каналы в Golang? | Golang channels internals
41:31
Николай Тузов — Golang
Переглядів 48 тис.