ch
Feedback
Библиотека Go-разработчика | Golang

Библиотека Go-разработчика | Golang

前往频道在 Telegram

Все самое полезное для Go-разработчика в одном канале. Учиться у нас: clc.to/qaSdww По рекламе: @proglib_adv Для обратной связи: @proglibrary_feeedback_bot РКН: https://gosuslugi.ru/snet/67a4a8c24689c2151c752af0 #WXSSA

显示更多

📈 Telegram 频道 Библиотека Go-разработчика | Golang 的分析概览

频道 Библиотека Go-разработчика | Golang (@goproglib) 俄语 语言赛道中的 是活跃参与者。目前社区聚集了 23 938 名订阅者,在 技术与应用 类别中位列第 5 633,并在 俄罗斯 地区排名第 27 844

📊 受众指标与增长动态

невідомо 创建以来,项目保持高速增长,吸引了 23 938 名订阅者。

根据 30 六月, 2026 的最新数据,频道保持稳定运转。过去 30 天订阅人数变化为 -86,过去 24 小时变化为 -5,整体触达仍然可观。

  • 认证状态: 未认证
  • 互动率 (ER): 平均受众互动率为 11.78%。内容发布后 24 小时内通常能获得 7.55% 的反应,占订阅者总量。
  • 帖子覆盖: 每篇帖子平均可获得 2 820 次浏览,首日通常累积 1 808 次浏览。
  • 互动与反馈: 受众积极参与,单帖平均反应数为 10
  • 主题关注点: 内容集中在 навигация, лучшее_из_библиотеки_2025, git, string, golive 等核心主题上。

📝 描述与内容策略

作者将该频道定位为表达主观观点的平台:
Все самое полезное для Go-разработчика в одном канале. Учиться у нас: clc.to/qaSdww По рекламе: @proglib_adv Для обратной связи: @proglibrary_feeedback_bot РКН: https://gosuslugi.ru/snet/67a4a8c24689c2151c752af0 #WXSSA

凭借高频更新(最新数据采集于 01 七月, 2026),频道始终保持新鲜度与高覆盖。分析显示受众积极互动,使其成为 技术与应用 类别中的关键影响点。

23 938
订阅者
-524 小时
-77
-8630
吸引订阅者
七月 '26
七月 '26
+4
在1个频道中
六月 '26
+171
在3个频道中
Get PRO
五月 '26
+288
在3个频道中
Get PRO
四月 '26
+249
在1个频道中
Get PRO
三月 '26
+216
在1个频道中
Get PRO
二月 '26
+487
在7个频道中
Get PRO
一月 '26
+346
在3个频道中
Get PRO
十二月 '25
+370
在4个频道中
Get PRO
十一月 '25
+384
在4个频道中
Get PRO
十月 '25
+331
在2个频道中
Get PRO
九月 '25
+290
在0个频道中
Get PRO
八月 '25
+316
在4个频道中
Get PRO
七月 '25
+317
在2个频道中
Get PRO
六月 '25
+315
在0个频道中
Get PRO
五月 '25
+391
在5个频道中
Get PRO
四月 '25
+358
在4个频道中
Get PRO
三月 '25
+625
在53个频道中
Get PRO
二月 '25
+527
在35个频道中
Get PRO
一月 '25
+517
在37个频道中
Get PRO
十二月 '24
+591
在40个频道中
Get PRO
十一月 '24
+605
在52个频道中
Get PRO
十月 '24
+499
在36个频道中
Get PRO
九月 '24
+676
在36个频道中
Get PRO
八月 '24
+580
在36个频道中
Get PRO
七月 '24
+586
在37个频道中
Get PRO
六月 '24
+467
在29个频道中
Get PRO
五月 '24
+674
在35个频道中
Get PRO
四月 '24
+771
在36个频道中
Get PRO
三月 '24
+901
在31个频道中
Get PRO
二月 '24
+729
在31个频道中
Get PRO
一月 '24
+863
在27个频道中
Get PRO
十二月 '23
+850
在29个频道中
Get PRO
十一月 '23
+608
在10个频道中
Get PRO
十月 '23
+697
在25个频道中
Get PRO
九月 '23
+723
在0个频道中
Get PRO
八月 '23
+648
在0个频道中
Get PRO
七月 '23
+503
在0个频道中
Get PRO
六月 '23
+422
在0个频道中
Get PRO
五月 '23
+568
在0个频道中
Get PRO
四月 '23
+365
在0个频道中
Get PRO
三月 '23
+841
在0个频道中
Get PRO
二月 '23
+392
在0个频道中
Get PRO
一月 '23
+395
在0个频道中
Get PRO
十二月 '22
+384
在0个频道中
Get PRO
十一月 '22
+491
在0个频道中
Get PRO
十月 '22
+256
在0个频道中
Get PRO
九月 '22
+292
在0个频道中
Get PRO
八月 '22
+394
在0个频道中
Get PRO
七月 '22
+472
在0个频道中
Get PRO
六月 '22
+443
在0个频道中
Get PRO
五月 '22
+176
在0个频道中
Get PRO
四月 '22
+328
在0个频道中
Get PRO
三月 '22
+242
在0个频道中
Get PRO
二月 '22
+135
在0个频道中
Get PRO
一月 '22
+236
在0个频道中
Get PRO
十二月 '21
+245
在0个频道中
Get PRO
十一月 '21
+219
在0个频道中
Get PRO
十月 '21
+215
在0个频道中
Get PRO
九月 '21
+290
在0个频道中
Get PRO
八月 '21
+298
在0个频道中
Get PRO
七月 '21
+281
在0个频道中
Get PRO
六月 '21
+196
在0个频道中
Get PRO
五月 '21
+389
在0个频道中
Get PRO
四月 '21
+336
在0个频道中
Get PRO
三月 '21
+280
在0个频道中
Get PRO
二月 '21
+258
在0个频道中
Get PRO
一月 '21
+262
在0个频道中
Get PRO
十二月 '20
+6 566
在0个频道中
日期
订阅者增长
提及
频道
01 七月+4
频道帖子
😋 Топ-вакансий для Go-разработчиков за неделю Go Developer — удаленно по Москве Middle Golang разработчик — от 320 000 ₽, удаленно Senior Golang Developer — до 500 000 ₽, гибрид в Москве или удаленно ➡️ Еще больше топовых вакансий — в нашем канале Go jobs 🐸 Библиотека Go-разработчика #GoWork

2
Go-разработчики, идём по барам с 2ГИС! 16 июля, 18:00, Нижний Новгород В программе три доклада, бархоппинг и разгон факапов.
Go-разработчики, идём по барам с 2ГИС! 16 июля, 18:00, Нижний Новгород В программе три доклада, бархоппинг и разгон факапов. Бархоппинг — это маршрут по барам. В каждом месте вас ждут мини-задачка и бокал чего‑то вкусного. Уже чувствуете запах крафта? Тогда скорее регистрируйтесь! В заявке обязательно поделитесь каким-нибудь фэйлом. Лучшие обсудим в финале, посмеёмся над собой и поучимся у других
1 018
3
🧑‍💻 Сравнение двух Go-структур без хардкода полей. Часть 3, как это устроено внутри В прошлой части мы договорились про тег match. Теперь посмотрим на реализацию. Она занимает около 130 строк и состоит из обычной рефлексии. Сначала разворачиваем указатели и проверяем, что перед нами действительно структуры. Функция принимает any с обеих сторон, поэтому работает с любым типом: func (m *Mapper) CompareAndMap(dto any, db any) (map[string]any, error) { dtoVal := reflect.ValueOf(dto) dbVal := reflect.ValueOf(db) if dtoVal.Kind() == reflect.Ptr { dtoVal = dtoVal.Elem() } if dbVal.Kind() == reflect.Ptr { dbVal = dbVal.Elem() } if dtoVal.Kind() != reflect.Struct || dbVal.Kind() != reflect.Struct { return nil, fmt.Errorf("both dto and db must be structs or pointers to structs") } // ... } Дальше один раз проходим по модели базы и строим карту вида имя колонки в значение поля. Так матчинг работает за O(1) на поле, а порядок полей в структурах может не совпадать: func (m *Mapper) buildColumnMap(dbVal reflect.Value) map[string]reflect.Value { columnMap := make(map[string]reflect.Value) dbType := dbVal.Type() for i := 0; i < dbType.NumField(); i++ { if column := m.parseGormColumn(dbType.Field(i).Tag.Get(dbGormTag)); column != "" { columnMap[column] = dbVal.Field(i) } } return columnMap } Имя колонки достаётся из тега GORM, который бывает грязным, например type:smallint;column:is_suspended. Режем по точке с запятой и берём кусок после column: func (m *Mapper) parseGormColumn(gormTag string) string { for _, part := range strings.Split(gormTag, ";") { if col, ok := strings.CutPrefix(part, dbColumnKey+":"); ok { return col } } return "" } Потом идём по полям DTO. Поле без тега match пропускаем, так DTO может иметь поля, которые мы намеренно не синхронизируем. Поле, чей тег указывает на несуществующую колонку, тоже пропускаем без паники: for i := 0; i < dtoVal.NumField(); i++ { dtoField := dtoVal.Field(i) dtoFieldType := dtoType.Field(i) dbGormColumnName := dtoFieldType.Tag.Get(dtoMatchTag) if dbGormColumnName == "" { continue // нет тега match, это не наша забота } dbField, ok := columnMap[dbGormColumnName] if !ok { continue // DTO ссылается на колонку, которой нет в модели базы } // ... сравниваем dtoField и dbField } Каркас готов. Осталось самое скользкое, сравнение значений с указателями и nil. Об этом в следующей части. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
1 228
4
🤔 Стоит ли Go развиваться быстрее В сабреддите r/golang разгорелось обсуждение, которое касается почти каждого, кто пишет на
🤔 Стоит ли Go развиваться быстрее В сабреддите r/golang разгорелось обсуждение, которое касается почти каждого, кто пишет на Go. Автор треда обратил внимание, что в 1.26 основная работа снова шла не в синтаксис, а в рантайм, тулчейн, планировщик и сборщик мусора. Дальше встал вопрос, стоит ли языку и дальше расти медленно, вкладываясь в производительность и инструменты, или пора добавлять больше фич, и было ли многолетнее сдерживание силой языка или тормозом для его развития. ➡️ Стабильность как главный аргумент Самая частая тема в треде это обратная совместимость. Многие участники прямо пишут, что выбрали Go именно потому, что код, написанный десять лет назад, до сих пор компилируется и работает так же, как раньше. ➡️ Маленький язык как осознанный выбор Вторая крупная тема это размер языка. Часть комментаторов прямо говорит, что любит Go именно за то, что в нём мало способов сделать одно и то же. Чем меньше в языке конструкций, тем меньше поводов для споров в код-ревью и тем легче читать чужой код, написанный по совершенно другим командам и в другое время. ➡️ Где сообщество видит реальный потенциал для роста Почти никто в треде не просит радикальных изменений синтаксиса, но запрос на улучшения всё же есть. Чаще всего звучат три темы: • Производительность и сборщик мусора, который многие сравнивают не в пользу Go с JVM. • Полноценные enum, которые называют одной из немногих фич, добавление которых не выглядит избыточным. • Доработка инструментов, в частности encoding/json/v2, который уже называют заметным шагом вперёд. 💬 Что вы думаете? Стоит ускорять разработку го и было ли многолетнее сдерживание силой языка или тормозом для его развития? 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoTalk
1 644
5
⚠️ Уже завтра стартует курс AgentOps! Мы собрали на потоке сборную из мастеров IT-рынка. Практики из BigTech научат вас контр
⚠️ Уже завтра стартует курс AgentOps! Мы собрали на потоке сборную из мастеров IT-рынка. Практики из BigTech научат вас контролировать и отлаживать ИИ-агентов, чтобы они работали предсказуемо и не сливали бюджет на API. 🔥 Заберите 3 курса по цене 1: ● При покупке VIP-тарифа (осталось 4 места) нового потока «Разработка ИИ-агентов» получаете в подарок курс «AgentOps» + ещё один любой курс Академии (например, «Математика для разработки AI», чтобы глубже освоить направление). ● Три курса обойдутся вам всего в 134.000 ₽ вместо 263.000 ₽. ● Доступна удобная беспроцентная рассрочка, платеж можно разбить на несколько комфортных частей. Хотите прокачать свое портфолио продакшн-кейсом, но пока сомневаетесь? Пройдите наш бесплатный демо-урок, чтобы протестировать формат перед покупкой. 👉 Забрать 3 курса по цене 1 и получить демо-урок
1 631
6
📎 Разбираемся, как os.ReadFile() работает с памятью Во многих туториалах по Go встречается фраза «os.ReadFile() читает весь файл в память». Мы привыкли принимать такие утверждения на веру, хотя проверить их легко. В этом посте разберём, что на самом деле происходит с памятью процесса, когда os.ReadFile() обрабатывает файл на 500 мегабайт, и почему результат измерений может удивить. ➡️ Эксперимент Берём простую программу. Она читает файл, путь к которому передан аргументом, выводит его размер и затем ждёт нажатия Enter, чтобы у нас было время посмотреть на потребление памяти процесса: package main import ( "fmt" "os" ) func main() { if len(os.Args) != 2 { fmt.Println("Please specify a path") return } b, err := os.ReadFile(os.Args[1]) if err != nil { panic(err) } fmt.Println("File has been read into memory.") fmt.Println("Size:", len(b), "bytes") fmt.Println("Press Enter to exit...") fmt.Scanln() } Сначала создаём файл на 500 мегабайт: fallocate -l 500M bigfile.txt Запускаем программу через go run: go run readingFile_bad.go bigfile.txt Пока программа стоит на паузе, смотрим на её RSS, то есть на объём физической памяти, реально занятой процессом: ps -o pid,rss,vsz,cmd -p 228159 PID RSS VSZ CMD 228159 3116 1750908 /tmp/go-build2199976311/b001/exe/readingFile_bad bigfile.txt RSS показывает примерно 3 мегабайта, хотя файл весит 500. Логично ожидать, что вся эта память должна быть занята. Разберёмся, почему этого не происходит. ❓ Почему память не занята Когда os.ReadFile() вызывается, Go выделяет на куче backing array размером с файл, в нашем случае 500 мегабайт, и операционная система копирует туда содержимое файла с диска. Переменная b []byte, которую мы получаем, это не сами данные. Это всего лишь заголовок слайса, маленькая структура размером около 24 байт на 64-битной системе, которая хранит три поля: указатель на первый байт backing array, текущую длину слайса, и полную ёмкость backing array. Сам слайс маленький, а массив с данными файла занимает реальные 500 мегабайт где-то в куче: b │ ├── pointer ───────────────┐ ├── length = 524288000 | └── capacity = 524288000 │ ▼ +-----------------------+ | 500 MB backing array | +-----------------------+ Дело в том, что в нашей программе после fmt.Scanln() переменная b больше нигде не используется. Компилятор и сборщик мусора видят, что b мертва после этой точки, поэтому backing array становится недоступным для использования и сборщик мусора может его освободить ещё до того, как мы посмотрели на RSS. Чтобы проверить эту гипотезу, раскомментируем строку с обращением к b[0] в конце программы: fmt.Println("FirstByte: ", b[0]) Теперь компилятор знает, что переменная понадобится после Scanln(), и не может позволить сборщику мусора забрать её раньше времени. Запускаем программу заново и смотрим на RSS: ps -o pid,rss,vsz,cmd -p 233685 PID RSS VSZ CMD 233685 514680 1750844 /tmp/go-build1663734078/b001/exe/readingFile_bad bigfile.txt На этот раз RSS равен 514680 килобайт, то есть около 514 мегабайт. Это уже соответствует ожиданиям. Стоит добавить ещё один нюанс. Команда go run сама по себе не запускает наш код напрямую, она сначала компилирует временный исполняемый файл во временную директорию и уже его запускает отдельным процессом. Поэтому при анализе памяти нужно смотреть не на процесс go run, а на дочерний процесс из /tmp/go-build.../exe/, именно он реально читает файл. os.ReadFile() действительно выделяет память под весь файл целиком, утверждение из туториалов верное. Но []byte, который мы получаем, это лёгкий заголовок слайса, а не сами данные, и поведение сборщика мусора влияет на то, когда именно эта память физически занята процессом. Если переменная с данными становится недостижимой раньше, чем вы успели посмотреть на RSS, может показаться, что файл вообще не загружался в память, хотя на деле это просто сборка мусора отработала раньше, чем мы ожидали. Полезный практический вывод из этого расследования простой. Если вам нужно работать с большими файлами, лучше избегать os.ReadFile() и использовать потоковое чтение через bufio.Reader или io.Copy, чтобы не держать весь файл в памяти разом. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
1 623
7
🚀 Не уверены, стоит ли переходить на зрелую ИИ-инженерию? Начните с демо-урока! Вот-вот стартует наш курс AgentOps. Если вы
🚀 Не уверены, стоит ли переходить на зрелую ИИ-инженерию? Начните с демо-урока! Вот-вот стартует наш курс AgentOps. Если вы сомневаетесь в формате, просто оставьте заявку и получите бесплатный демо-урок «AI-инструменты в разработке: как писать код быстрее с помощью ассистентов». Для тех, кто готов мощно прокачать портфолио, прямо сейчас действует предложение «3 любых курса по цене 1»: — При покупке VIP-тарифа (осталось 4 места) нового потока «ИИ-агенты» вы получаете в подарок доступ к курсу «AgentOps» + ещё один любой курс Академии на выбор — В деньгах это два топовых курса по автоматизации и контролю ИИ всего за 134.000 ₽ вместо 263.000 ₽ 🔥 А за счет третьего курса (например, можно выбрать «Математику») вы соберете мощный стек и освоите целое востребованное направление. — Платеж можно разбить на несколько частей с помощью беспроцентной рассрочки. 👉 Получить демо-урок и зафиксировать спецпредложение 3 в 1
1 820
8
📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoGiggle
📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoGiggle
1 925
9
📎 Сравнение двух Go-структур без хардкода полей. Часть 2, идея с тегами В первой части мы упёрлись в стену из почти одинаковых блоков if, которая растёт с каждым новым полем. Теперь зададим другой вопрос. Что если логика сравнения вообще не будет упоминать ни одного имени поля? У полей Go-структуры можно хранить метаданные через теги, те самые строки в обратных кавычках. GORM по тегу gorm:"column:name" знает, в какую колонку базы ложится поле. JSON по тегу json:"name" знает имя в JSON. Это просто строки, которые рантайм умеет читать обратно через рефлексию. Обе стороны сравнения уже размечены. Модель базы знает свои колонки. Нужно лишь, чтобы входящий DTO объявил, какой колонке соответствует каждое его поле. Поэтому добавляем ещё один тег: type DriverDriver struct { Name string `json:"name" match:"name"` Number string `json:"number" match:"number"` ProfilePic *string `json:"profile_pic" match:"profile_pic"` } Тег match:"name" это весь контракт. Он говорит, что это поле надо сравнивать с колонкой базы по имени name. Правило маппера получается таким. Для каждого поля DTO с тегом match найди поле базы, у которого колонка gorm равна этому тегу, сравни их, и если различаются, запиши новое значение под именем колонки. Ни одно имя поля не зашито в логику. Определения структур сами становятся конфигурацией. Добавили поле и тег, оно автоматически попало в сравнение. Удалили, оно выпало. Сам маппер при этом не меняется. Вызов выглядит как одна строка на модель: driverUpdates, err := mapper.CompareAndMap(driver.DriverDriver, dbUser) metaUpdates, err := mapper.CompareAndMap(driver.DriverMeta, dbUser) vehicleUpdates, _ := mapper.CompareAndMap(driver.Vehicle, dbVehicle) Каждый вызов возвращает map[string]any только из изменившихся колонок, и это ровно та форма, которую ждёт Updates() у GORM. Обратите внимание, второй и третий вызовы сравнивают совсем разные типы структур с разными таблицами одной и той же функцией. В следующей части разберём, как этот маппер устроен внутри. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
2 053
10
🔍 LeetCode Daily: считаем подстроки за один проход Задача дня на LeetCode — Number of Strings That Appear as Substrings in W
🔍 LeetCode Daily: считаем подстроки за один проход Задача дня на LeetCode — Number of Strings That Appear as Substrings in Word. Дан массив строк patterns и строка word. Нужно вернуть количество строк из patterns, которые входят в word как подстроки. Подстрока — это непрерывная последовательность символов внутри строки. Если patterns = ["a","abc","bc","d"] и word = "abc", ответ будет 3: входят "a", "abc" и "bc", а "d" нет. ➡️ Решение   В Go для этого есть готовый инструмент — strings.Contains. Проходим по каждому паттерну и проверяем вхождение. Если вошло, увеличиваем счётчик. import "strings"   func numOfStrings(patterns []string, word string) int { count := 0 for _, p := range patterns { if strings.Contains(word, p) { count++ } } return count } ➡️ Сложность   strings.Contains внутри использует алгоритм поиска подстроки. При длине word = n и pattern = m это O(n·m) в худшем случае. С учётом ограничений задачи в 100 символов это несущественно. ➡️ Решить 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #ReadySetGo
2 093
11
🤝 Уйти грамотно — это тоже навык В IT репутация передается быстрее, чем резюме. Бывший тимлид может ответить на звонок рекру
🤝 Уйти грамотно — это тоже навык   В IT репутация передается быстрее, чем резюме. Бывший тимлид может ответить на звонок рекрутера через год, бывший коллега — написать в личку вашему потенциальному работодателю. Один неаккуратный уход способен закрыть двери туда, где вы еще даже не пытались открыть. ➡️ Как уйти красиво 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика
2 205
12
🧑‍💻 Сравнение двух Go-структур без хардкода полей. Часть 1, проблема Почти в каждом бэкенде есть скучная рутина. Вы тянете данные из внешнего источника, у вас уже лежит их копия в своей базе, и нужно обновить только те строки, которые реально изменились. Звучит просто, но на практике это превращается в стену почти одинакового кода сравнения, который никто не хочет трогать. В этой серии разберём приём на рефлексии, который убирает эту стену целиком. Начнём с того, откуда вообще берётся боль. Допустим, приходит событие «водитель принял заказ», и вы подтягиваете свежий профиль водителя из внешнего сервиса. Он отдаёт маленькую структуру на три поля: type DriverDriver struct { Name string `json:"name"` Number string `json:"number"` ProfilePic *string `json:"profile_pic"` } А ваша модель базы выглядит куда менее мило, в ней около восьмидесяти колонок: type User struct { ID int64 `gorm:"column:id"` Name string `gorm:"column:name"` Email string `gorm:"column:email"` Number string `gorm:"column:number"` ProfilePic *string `gorm:"column:profile_pic"` // ... и ещё около восьмидесяти колонок } Перезаписывать всю строку нельзя. Нужно отправить в UPDATE только те колонки, которые отличаются, чтобы не затереть поля, о которых внешний сервис вообще ничего не знает. Первый вариант, который пишут все, выглядит так: updates := map[string]any{} if driver.Name != user.Name { updates["name"] = driver.Name } if driver.Number != user.Number { updates["number"] = user.Number // упс, опечатка, скопировали не ту сторону } if !ptrEqual(driver.ProfilePic, user.ProfilePic) { updates["profile_pic"] = driver.ProfilePic } Этот код работает, но плохо стареет. Каждое новое поле это ещё три строки. Указатели требуют отдельного хелпера. Кто-то по невнимательности пишет user.Number вместо driver.Number, и вы получаете баг, который всплывёт только когда водитель сменит телефон. Помножьте это на три разных DTO и три таблицы, и поддерживать такое становится больно. В следующей части посмотрим, как переписать сравнение так, чтобы оно вообще не упоминало ни одного имени поля. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
2 184
13
📰 Дайджест недели Автостопом по новостям — Детектор утечек горутин — uuid.NewV7() теряет случайность в браузере — gcli v3.8.
📰 Дайджест недели Автостопом по новостям — Детектор утечек горутин — uuid.NewV7() теряет случайность в браузере — gcli v3.8.0 — Добавить slog.TestHandler в стандартную библиотеку Go 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoLive
2 126
14
🤩 Фаззинг находит то, что вы не догадались проверить У обычных тестов есть слепое пятно. Они проверяют случаи, которые вы придумали, а баги живут в тех, что не придумали. Покрытие в 90% говорит, сколько строк вы прогнали, но молчит о том, сколько форм входа упустили. Зелёный прогон означает лишь, что код отработал на ваших примерах, а не что устоит против всех. ➡️ Идея в одном сдвиге Вы пишете не пример, а свойство, которое обязано держаться всегда. Машина сама ищет вход, который его ломает. Вместо «на входе X жду Y» вы описываете инвариант: func FuzzParse(f *testing.F) { f.Add("platform=5000") // сиды это ваши готовые кейсы f.Fuzz(func(t *testing.T, rule string) { team, limit, err := ParseBudgetRule(rule) if err != nil { return // ошибка на кривом входе это норма } // раз ошибки нет, результат обязан быть валидным if limit <= 0 || team == "" { t.Errorf("bad result without error for %q", rule) } }) } Частое заблуждение, что фаззер сыплет случайные байты. Современные фаззеры coverage‑guided. Стартуют с ваших сидов и мутируют их, отслеживая, какие мутации открывают новые ветки. Попал в новый путь, копает оттуда. Поэтому к багам он сходится в разы быстрее слепого рандома. Юнит‑тесты проверяют дороги, которые вы построили, фаззер ищет те, о которых вы не знали. ❓ Что ловит Класс ошибок на стыке кода и реальности. Паника на входе, которого «не бывает». parts := strings.SplitN(rule, "=", 2) limit, _ := strconv.Atoi(parts[1]) // "platform" без = и parts[1] не существует // panic: index out of range [1] with length 1 Нарушения инвариантов, когда функция вернула мусор без ошибки, что хуже явной паники. Расхождения round‑trip, когда decode(encode(x)) не равно x. Их объединяет одно. Все появляются, когда снаружи приходит то, чего код не ждал. Обычные тесты это скипают, потому что их пишут под ожидаемые сценарии, а вы рассуждаете как автор кода, который знает, как им пользоваться. Фаззеру эта рамка незнакома. Найденный падающий вход не исчезает. Go сам кладёт минимальный контрпример в testdata/fuzz/ и делает из него регрессию: go test -fuzz=FuzzParse -fuzztime=30s # ищем баг go test ./... # корпус гоняется всегда, даже без -fuzz Один раз пойманный баг вернуться уже не может. ➡️ Куда прикладывать Туда, где код разбирает или валидирует внешний вход. Парсеры, декодеры, заголовки, query, конфиги, всё, что ветвится по содержимому. Чем больше условной логики, тем больше путей фаззеру. Не стоит фаззить чистую бизнес‑логику без зависимости от входа и функции, которые ходят в сеть или базу, их фаззер дёрнет тысячи раз. Правило короткое. Зависит поведение от того, что прислал внешний мир, заведите фаззинг. ❓ Почему окупается Барьер почти нулевой. В Go фаззер встроен в тулчейн с версии 1.18, свойство пишется в считаные строки. Цена это пара минут на разработке. Альтернатива это ждать, пока вход за вас подберёт прод. Боевыми данными, под нагрузкой, в три часа ночи, в виде алерта. Прод пофаззит ваш код в любом случае, вопрос лишь в том, узнаете вы о баге от теста или от дежурного. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
2 176
15
📎 Главная проблема в микросервисах Сервис стартует чистым. Через полгода его трогать никто не хочет. Багов нет, тесты проходят, но любое изменение задевает пять файлов, а новому инженеру три дня объясняют, что где лежит. Проблема не в алгоритме и не в кэше, а в структуре. Команды копируют раскладку папок, не понимая контракт, который эта раскладка должна навязывать. Папки вместо архитектуры Спросите десять команд про структуру сервиса, и почти все опишут что‑то такое. /cmd /internal /handlers /services /repository /models /pkg Выглядит разумно и проходит ревью. Беда в том, что папки создают иллюзию границ, но не держат их. Ничто не мешает хендлеру импортировать репозиторий напрямую. Ничто не мешает модели обрасти бизнес‑логикой. Границы косметические, и код деградирует предсказуемо. Логика переползает туда, где удобнее, обычно в хендлеры, потому что там контекст. Типы репозитория протекают в API, потому что заводить отдельный DTO лень. Пакет models превращается в общий мешок структур, от которого зависят все, и поменять в нём что‑либо без каскада уже нельзя. Получается слоёная на вид архитектура, ведущая себя как монолит с накладными расходами на HTTP. В чём контракт слоёв Контракт про направление зависимостей. Они текут внутрь. Внешний слой (HTTP, gRPC, CLI) знает про слой приложения. Слой приложения знает про домен. Домен не знает ни про что. На практике это значит вот что. Хендлер принимает запрос, зовёт метод сервиса и сериализует ответ. Никакого SQL и никаких бизнес‑правил в нём нет. Сервис кодирует, что приложение делает. Он оркестрирует, валидирует, делегирует репозиторию и ничего не знает про HTTP. Репозиторий говорит с базой и возвращает доменные типы или ошибки, но не pgx.Rows. Доменные типы несут инварианты. Структура Member знает, что делает участника валидным, но не знает, как сериализовать себя в JSON. Когда контракт держится, слои тестируются независимо, реализацию репозитория можно подменить, не трогая хендлер, а gRPC добавить без дублирования логики. Анти‑паттерн god service Чаще всего контракт ломает god service. Сервис‑структура, которая всасывает всё: type MemberService struct { db *pgxpool.Pool } func (s *MemberService) CreateMember(ctx context.Context, req CreateMemberRequest) (*Member, error) { // валидация, хеш, вставка строки, welcome‑письмо, событие в Kafka } Через полгода у MemberService сорок методов, пул базы, почтовый клиент, продюсер Kafka, клиент S3 и конфиг. Любой тест требует поднять все зависимости, даже когда проверяешь одну ветку валидации. Это уже не слой, а ящик для хлама. И дробить его на MemberCreationService и MemberUpdateService бесполезно, получите ящики поменьше. Лечит инверсия зависимостей Интерфейсы в Go удовлетворяются неявно, и это недоиспользуют. Вместо *pgxpool.Pool опишите то, что сервису реально нужно: type MemberRepository interface { CreateMember(ctx context.Context, params CreateMemberParams) (*Member, error) GetMemberByID(ctx context.Context, id uuid.UUID) (*Member, error) } type EventPublisher interface { Publish(ctx context.Context, event DomainEvent) error } type MemberService struct { repo MemberRepository publisher EventPublisher } Теперь сервис ничего не знает про PostgreSQL и Kafka. Он зависит от поведения, а не от реализации. Чтобы протестировать метод, достаточно мока под интерфейс, без базы, брокера и сети. Про pgx знает реализация репозитория, про franz-go знает реализация publisher, а сервис не знает ни про то, ни про другое. В этом и есть контракт. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
2 319
16
🎮 Субботний оффтоп Кто-то собрал веб-версию Half-Life 2 и запустил её без установки и скачивания. Уровни и ресурсы грузятся
🎮 Субботний оффтоп Кто-то собрал веб-версию Half-Life 2 и запустил её без установки и скачивания. Уровни и ресурсы грузятся быстро, игра работает бодро. У многих сразу включается русский язык — Сити-17 и Рейвенхольм уже вовсю тестируют первые игроки. ➡️ Поиграть 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoLive
2 476
17
👨‍💻 Контекст неизменяемый, а память нет Правило знают все. context.WithValue ничего не меняет, а создаёт новую обёртку, поэтому контекст безопасно шарить между горутинами. Учебники забывают сказать главное. Иммутабельны обёртки, а объект в корне цепочки может переиспользоваться фреймворком через sync.Pool. Тогда горутина, пережившая запрос, держит указатель на память, которая вот‑вот достанется другому запросу. Хендлер запускает фоновую задачу и сразу возвращает ответ. Линтер ругается на отмену контекста, и со времён Go 1.21 его успокаивают одной строкой: go s.processAsync(context.WithoutCancel(ctx), id) Для net/http это правильно. Там контекст это обычная аллокация в куче. Но линтер слеп к архитектуре и не знает, что под context.Context может стоять пулящаяся структура. ❓ Где можно споткнуться В Fiber и всём на fasthttp c.Context() возвращает *fasthttp.RequestCtx. Пулящийся объект и есть ваш контекст, и после отправки ответа fasthttp сбрасывает его и кладёт обратно в пул. Gin пулит *gin.Context, поэтому передавать его в горутину нельзя (для снимка есть c.Copy()), а c.Request.Context() безопасен. Echo свой контекст пулит, но context.Context он не реализует, так что задеть гонку сложнее. Дальше воркер просыпается после I/O, зовёт ctx.Value("user_id") и читает память, которую перезаписывают данными другого пользователя. И вот что коварно. WithoutCancel снимает только отмену, а Value() по‑прежнему делегирует пулящемуся родителю. То есть фикс, который позеленил линтер, ровно и удерживает протухший указатель. ➡️ Как правильно Если задаче ничего не нужно из запроса, постройте свежий корень: ctx := context.Background() Если нужны trace‑id или user‑id, вытащите их как простые значения, пока горутина запроса ещё жива, и положите в свежий контекст: userID := c.Locals("user_id").(string) traceID := c.Locals("trace_id").(string) go func(userID, traceID string) { ctx := context.WithValue(context.Background(), userCtxKey, userID) ctx = context.WithValue(ctx, traceCtxKey, traceID) h.svc.ProcessAsync(ctx, userID) }(userID, traceID) Вся безопасность держится на том, что значения сняты до возврата из хендлера. Логику стоит вынести в хелпер вроде ctxutil.Detach(ctx). Из штатных лазеек помогают c.Copy() у Gin и c.UserContext() у Fiber. Линтер видит синтаксис, но не жизненный цикл памяти. Если горутина переживает запрос, не давайте ей контекст запроса ни в какой обёртке. Снимайте значения как простые данные, стройте свежий корень и держите -race в CI. 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoDeep
2 397
18
🎬 Где ломаются архитектуры ИИ-агентов и как этого избежать: запись урока от Proglib.Academy и cloud․ru Proglib.аcademy вмест
🎬 Где ломаются архитектуры ИИ-агентов и как этого избежать: запись урока от Proglib.Academy и cloud․ru Proglib.аcademy вместе с cloud․ru провели вебинар, где разобрали реальные боли проектирования автономных систем. Вы просили запись встречи — она уже в открытом доступе! Что внутри: — критерии выбора между одним агентом и мультиагентной системой; — разбор популярных архитектурных ошибок; — реальные ограничения современных ИИ-агентов; — практические рекомендации по проектированию агентных систем. 👉 Посмотреть запись можно тут: ● VK ● YouTube
2 363
19
🐠 Открытый фреймворк для учебного фишинга Gophish это опенсорсный инструмент для проведения фишинговых учений и обучения сот
🐠 Открытый фреймворк для учебного фишинга Gophish это опенсорсный инструмент для проведения фишинговых учений и обучения сотрудников. Он рассчитан на бизнес и пентестеров, которым нужно легально проверить, как команда реагирует на поддельные письма. Проект написан на Go. Запускать такие кампании можно только против своей организации или там, где у вас есть письменное разрешение. Без него это уже не учения. Вы описываете кампанию в веб-интерфейсе, а инструмент сам рассылает письма, поднимает посадочные страницы, отслеживает открытия и переходы и собирает всё в отчёты. Вся функциональность доступна и через REST API, поэтому кампании можно гонять из скриптов и встраивать в свои пайплайны отчётности. ➡️ Как запустить Самый быстрый путь это готовый бинарь. Под Windows, macOS и Linux есть релизы, их достаточно скачать, распаковать и запустить. Или собрать самостоятельно: git clone https://github.com/gophish/gophish.git cd gophish go build ./gophish Есть и официальный Docker образ, если не хотите ставить ничего на хост: docker run --name gophish -p 3333:3333 -p 8080:8080 gophish/gophish После старта админка поднимается на https://localhost:3333. Логин и пароль не зашиты заранее, а печатаются в лог при первом запуске. Выглядит это примерно так: level=info msg="Please login with the username admin and the password 4304d5255378177d" Gophish превращает разрозненную ручную работу по фишинговым учениям в управляемый процесс с нормальной статистикой. ➡️ Репозиторий 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #GoToProduction
2 376
20
🤔 Функция с пустым телом — что вернёт компилятор? Вот простой код: func foo() {} Тело есть. Но оно пустое. Функция ничего не делает, ничего не возвращает. Go — язык строгий. Он не любит неиспользуемые переменные и лишний импорт. Но как он относится к функции, которая существует и при этом ничего не делает? Подсказка: вспомните, как Go обрабатывает неиспользуемый код на уровне функций, а не переменных. ➡️ Правильный ответ 📍 Навигация: Вакансии • Задачи • Собесы 🐸 Библиотека Go-разработчика #ReadySetGo
2 538