Signal plays an important role in communication with POSIX system.

For example, what if our code been killed, interrupted, hung on? Generally, the operation system (I am talking about BSD, Unix-like, POSIX system, same in the follow but will omitted), will not really "KILL" it, it will just send a kill signal with id to this process, the this process will process this signal later.

Here is a table to describe the id to default action.

Num Name Default Action Description
1 SIGHUP terminate process terminal line hangup
2 SIGINT terminate process interrupt program
3 SIGQUIT create core image quit program
4 SIGILL create core image illegal instruction
5 SIGTRAP create core image trace trap
6 SIGABRT create core image abort program (formerly SIGIOT)
7 SIGEMT create core image emulate instruction executed
8 SIGFPE create core image floating-point exception
9 SIGKILL terminate process kill program
10 SIGBUS create core image bus error
11 SIGSEGV create core image segmentation violation
12 SIGSYS create core image non-existent system call invoked
13 SIGPIPE terminate process write on a pipe with no reader
14 SIGALRM terminate process real-time timer expired
15 SIGTERM terminate process software termination signal
16 SIGURG discard signal urgent condition present on socket
17 SIGSTOP stop process stop (cannot be caught or ignored)
18 SIGTSTP stop process stop signal generated from keyboard
19 SIGCONT discard signal continue after stop
20 SIGCHLD discard signal child status has changed
21 SIGTTIN stop process background read attempted from control terminal
22 SIGTTOU stop process background write attempted to control terminal
23 SIGIO discard signal I/O is possible on a descriptor (see fcntl(2))
24 SIGXCPU terminate process cpu time limit exceeded (see setrlimit(2))
25 SIGXFSZ terminate process file size limit exceeded (see setrlimit(2))
26 SIGVTALRM terminate process virtual time alarm (see setitimer(2))
27 SIGPROF terminate process profiling timer alarm (see setitimer(2))
28 SIGWINCH discard signal Window size change
29 SIGINFO discard signal status request from keyboard
30 SIGUSR1 terminate process User defined signal 1
31 SIGUSR2 terminate process User defined signal 2

You can also find this table using

$ man 3 signal

According to this table, we can easily find the default action of '9' is 'terminate process'

If we execute a command like this

$ kill -9 <process-id>

We are not killed this process, we just send a signal to this process, and this process have suicide later in processing this signal.

However, if we are required to process some important action, and a interrupting action comes, what should we do?

The best way, as far as I am concerned, is roll back if finishing will take some time, or save current status to make sure we could continue it once it restarted.

A lot of code provide this signal proces.

In (https://golang.org/), it provides signal as well, which is implemented using channel.

However, what if we have more than one thing to process in different part? A solution could be like this.

We could create an individual service to replace the default action, and then other code could just register/unregister functions by name (which is used to avoid duplicate processing).

Here is the code of my implementation. Please don't hesitate to give me any feedback if it is possible.

A simple usage could be as follow:

package main

import (
    "fmt"
    "github.com/argcv/sigr"
    "time"
)

func main() {
    // register a new name
    // this is used to
    name1 := sigr.RegisterOnStopFuncAutoName(func() {
        fmt.Println("Hello, World! #1")
    })
    name2 := sigr.RegisterOnStopFuncAutoName(func() {
        fmt.Println("Hello, World! #2")
    })
    sigr.RegisterOnStopFunc("customized name", func() {
        fmt.Println("Hello, World! #3")
    })
    fmt.Println("name1:", name1, "name2:", name2)
    sigr.RegisterOnStopFunc(name1, func() {
        fmt.Println("Hello, World! #1 #overridden")
    })
    sigr.UnregisterOnStopFunc(name2)
    // default: true
    sigr.SetQuitDirectyl(true)
    // get verbose log
    sigr.VerboseLog()
    // without verbose log, your printing is still work here
    //sigr.NoLog()
    // you may type ctrl+C to excute the functions (name1 and "customized name")
    time.Sleep(time.Duration(20 * time.Second))
}

The output seems like follow:

$ go run example.go
name1: __auto_1 name2: __auto_2
^C2017/10/26 19:06:17.804737 sigr.go:74: sig: [interrupt] Processing...
2017/10/26 19:06:17.804935 sigr.go:78: Processing task [__auto_1]
Hello, World! #1 #overridden
2017/10/26 19:06:17.804949 sigr.go:78: Processing task [customized name]
Hello, World! #3
2017/10/26 19:06:17.805069 sigr.go:85: sig: [interrupt] Processed, quitting directly
signal: interrupt
Categories: Code

Yu

Ideals are like the stars: we never reach them, but like the mariners of the sea, we chart our course by them.

Leave a Reply

Your email address will not be published. Required fields are marked *