// package kernel_keeper -- сторож системных сигналов. package kernel_keeper import ( "context" "os" "os/signal" "sync" "syscall" mL0 "gitp78su.ipnodns.ru/svi/kern/v4/lev0" "gitp78su.ipnodns.ru/svi/kern/v4/lev0/alias" mKt "gitp78su.ipnodns.ru/svi/kern/v4/lev0/ktypes" mL1 "gitp78su.ipnodns.ru/svi/kern/v4/lev1" ) // kernelKeeper -- сторож системных сигналов. type kernelKeeper struct { ctx context.Context fnCancel func() wg mKt.IKernelWg log mKt.ILogBuf chSys_ chan os.Signal } var ( kernKeep *kernelKeeper block sync.Mutex qName = alias.NewAStreamName("kernel_keeper") ) // GetKernelKeeper -- возвращает новый сторож системных сигналов. func GetKernelKeeper(ctx context.Context, fnCancel func(), wg mKt.IKernelWg) *kernelKeeper { block.Lock() defer block.Unlock() if kernKeep != nil { kernKeep.log.Debug("GetKernelKeeper()") return kernKeep } mL0.Hassert(ctx != nil, "NewKernelCtx(): ctx==nil") mL0.Hassert(wg != nil, "NewKernelCtx(): IKernelWg==nil") mL0.Hassert(fnCancel != nil, "NewKernelCtx(): fnCancel==nil") sf := &kernelKeeper{ ctx: ctx, fnCancel: fnCancel, wg: wg, log: mL1.NewLogBuf(mL1.OptIsTerm(true), mL1.OptPrefix("kernelKeeper")), chSys_: make(chan os.Signal, 2), } sf.log.Debug("GetKernelKeeper(): first run") sf.wg.Add(qName) go sf.run(sf.chSys_) kernKeep = sf _ = mKt.IKernelKeeper(sf) return sf } // Log -- возвращает лог сторожа системных сигналов. func (sf *kernelKeeper) Log() mKt.ILogBuf { return sf.log } // Работает в отдельном потоке и ждёт сигналов прерываний работы. func (sf *kernelKeeper) run(chSys chan os.Signal) { sf.log.Debug("run()") // Регистрируем сигналы SIGINT (Ctrl+C) и SIGTERM (завершение процесса) // syscall.SIGHUP: Сигнал, отправляемый при закрытии терминала. // syscall.SIGQUIT: Сигнал, отправляемый при нажатии **Ctrl+**. signal.Notify(chSys, syscall.SIGINT, syscall.SIGTERM, syscall.SIGHUP, syscall.SIGQUIT) select { case sig := <-chSys: // системный сигнал sf.log.Debug("run(): system signal, sig=%v\n", sig) sf.fnCancel() case <-sf.ctx.Done(): // сигнал от приложения sf.log.Debug("run(): cancel app context, err=\n\t%v\n", sf.ctx.Err()) } sf.wg.Done(qName) sf.log.Debug("run(): end") }