Grupttor is two phase graceful application interruptor(that's why grupttor). It supports handle interrupt, notify application parts when you need to wait for finish and then stop the whole application, or do whatever else you want to do.
Grupttor was released in initial phase and you can look at the Roadmap for next steps and features
Grupttor has several phases:
- when you initialize, it switch to init state and wait for configuration(hook add, etc.)
- you run startAndWait() method in new goroutine, you have to, because wihout it you will not be able to run your own code :-), then it switch to WAITING state and start wait channel for interrupt signal
- when it receive interrupt signal, it switch state to INTERRUPTING and run interruptHandler which you specify when you init grupttor
- the interrupt handle do your bussiness and after that you run Stop method on the grupttor to init stop phase - everything its in your hand, so you can do anything you want
- when it receive stop signal, it just switch state to stop and run your stop handler, when you exit application or send kill signal whatever you need
Just run:
go get -u github.com/PragGoLabs/grupttor
// init interruptor
interruptor := grupttor.NewGrupttor(
handlers.NewWrapHandler(
func(interrupter *grupttor.Grupttor) {
// implement interrupt handler
interrupter.Stop()
},
func(interrupter *grupttor.Grupttor) {
// and stop handler
},
),
// add hooks
[]grupttor.Hook{},
)
// if you need add later phase
// or add hooks manually
interruptor.AddHook(hooks.NewSystemInterruptHook([]os.Signal{syscall.SIGKILL, syscall.SIGTERM}))
// and start, it will wait for signals
go interruptor.StartAndWait()
Grupttor supports hook system, which you allow to specify logic which will run the interrupt phase.
By default there are 2 hooks:
-
SystemInterruptHook
- it handles system interruption calls, like sigterm, sigkill, sigabrt, etc.
- you can specify the signals you want to handle
interruptor.AddHook( hooks.NewSystemInterruptHook( []os.Signal{ syscall.SIGKILL, syscall.SIGTERM, } ) )
-
TimedInterruptHook
- its one shot timed interrupt action
- you specify how long it will wait to shot interrupt task
- if you need restart you app every hour for example and you run it in docker with restart always config
interruptor.AddHook(hooks.NewTimedInterruptHook( 10*time.Second, ))
You can implement your own hooks, just use the Hook interface and pass out.
Handlers are used for handling the signals, interrupt, stop.
Now is there only WrapHandler(), which will wrap the two handler function - for keeping simple interface, when you need simple handler.
I'll do a lot of handlers in future, for http, rabbitmq, etc. For details look at there: Handlers
package main
import (
"fmt"
"github.com/PragGoLabs/grupttor"
"github.com/PragGoLabs/grupttor/hooks"
"os"
"syscall"
"time"
)
func main() {
interruptor := grupttor.NewGrupttor(
handlers.NewWrapHandler(
func(interrupter *grupttor.Grupttor) {
fmt.Println("Received interrupt signal")
time.Sleep(4*time.Second)
interrupter.Stop()
},
func(interrupter *grupttor.Grupttor) {
// exit application
os.Exit(0)
},
),
[]grupttor.Hook{},
)
interruptor.AddHook(hooks.NewSystemInterruptHook([]os.Signal{syscall.SIGKILL, syscall.SIGTERM}))
go interruptor.StartAndWait()
for {
time.Sleep(2*time.Second)
fmt.Println("Running loop and waiting for kill or sigterm")
}
}
- Phase deploy the initial version
- Add support for passing the interrupt handler in concrete context not in init phase
- Add support for multiple interrupt handlers
- Add more interrupters
- Fork it
- Clone (
git clone https://github.com/your_username/grupttor && cd grupttor
) - Create your feature branch (
git checkout -b my-new-feature
) - Do changes, Commit, Push
- Create new pull request
- Thanks in advance :-)
See LICENSE.txt