| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182 |
- // 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/defs"
- mKs "gitp78su.ipnodns.ru/svi/kern/v4/lev0/kspec"
- mL1 "gitp78su.ipnodns.ru/svi/kern/v4/lev1"
- )
- // kernelKeeper -- сторож системных сигналов.
- type kernelKeeper struct {
- ctx context.Context
- fnCancel func()
- wg mKs.IKernelWg
- log mKs.ILogBuf
- chSys_ chan os.Signal
- }
- var (
- kernKeep *kernelKeeper
- block sync.Mutex
- qName = defs.NewStreamName("kernel_keeper")
- )
- // GetKernelKeeper -- возвращает новый сторож системных сигналов.
- func GetKernelKeeper(ctx context.Context, fnCancel func(),
- wg mKs.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
- _ = mKs.IKernelKeeper(sf)
- return sf
- }
- // Log -- возвращает лог сторожа системных сигналов.
- func (sf *kernelKeeper) Log() mKs.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")
- }
|