Signals

Сигналы (Signals)|Signals #

// Иногда нам хотелось бы, чтобы наши программы на Go
// интеллектуально обрабатывали [сигналы Unix](http://en.wikipedia.org/wiki/Unix_signal).
// Например, мы можем захотеть, чтобы сервер корректно
// завершил работу при получении `SIGTERM`, или инструмент
// командной строки остановил обработку ввода, если он
// получил `SIGINT`. Вот как обрабатывать сигналы в Go
// с каналами.

package main

import (
	"fmt"
	"os"
	"os/signal"
	"syscall"
)

func main() {

	// Уведомление о выходе сигнала работает путем
	// отправки значений `os.Signal` в канал. Мы создадим
	// канал для получения этих уведомлений (мы также
	// создадим канал, чтобы уведомить нас, когда программа
	// может выйти).
	sigs := make(chan os.Signal, 1)
	done := make(chan bool, 1)

	// `signal.Notify` регистрирует данный канал для
	// получения уведомлений об указанных сигналах.
	signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)

	// Эта горутина выполняет блокировку приема сигналов.
	// Когда она получит его, то распечатает его, а
	// затем уведомит программу, что она может быть завершена.
	go func() {
		sig := <-sigs
		fmt.Println()
		fmt.Println(sig)
		done <- true
	}()

	// Программа будет ждать здесь, пока не получит
	// ожидаемый сигнал (как указано в приведенной
	// выше процедуре, отправляющей значение в `done`),
	// и затем завершится.
	fmt.Println("awaiting signal")
	<-done
	fmt.Println("exiting")
}
# Когда мы запустим эту программу, она заблокирует
# ожидание сигнала. Набрав `ctrl-C` (который терминал
# показывает как `^C`), мы можем послать сигнал `SIGINT`,
# в результате чего программа напечатает `interrupt`
# и затем выйдет.
$ go run signals.go
awaiting signal
^C
interrupt
exiting