Mirror

Volodymyr Tkach cba3a90ba9 Update LICENSE 2 years ago
.github 3f4650a930 Create go.yml 2 years ago
ctrlc 5754790f0f Don't show "Shutdown error" msg on standard error 2 years ago
.gitignore 513344e4b9 First version, examples, test object 5 years ago
LICENSE cba3a90ba9 Update LICENSE 2 years ago
Makefile f5032a69e7 Update Makefile 2 years ago
README.md 283d0ab0de Update README.md 2 years ago
go.mod c3dab62598 Update Go version 2 years ago
main.go 0194082704 Optimization, error helper 2 years ago
testobject.go e6b74e2926 Rename variable 2 years ago

README.md

golang-ctrlc

Wrapper with context for safe application closing by signals like ctrl+c or kill

How to use

go get github.com/vladimirok5959/golang-ctrlc
package main

import (
	"context"
	"fmt"
	"net/http"
	"time"

	"github.com/vladimirok5959/golang-ctrlc/ctrlc"
)

func main() {
	MyAppFunc := func(ctx context.Context, shutdown context.CancelFunc) *[]ctrlc.Iface {
		// Some custom logic
		// With goroutine inside
		test := Run()

		// err1 := fmt.Errorf("Startup error 1")
		// err2 := fmt.Errorf("Startup error 2")
		// return ctrlc.MakeError(shutdown, ctrlc.AppError(err1), ctrlc.AppError(err2))

		// Http web server
		mux := http.NewServeMux()
		mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
			fmt.Printf("New web request (%s)!\n", r.URL.Path)

			// Do something hard inside (12 seconds)
			for i := 0; i < 12000; i++ {
				select {
				case <-ctx.Done():
					// Interrupt request by server
					fmt.Printf("[BY SERVER] OK, I will cancel (%s)!\n", r.URL.Path)
					return
				case <-r.Context().Done():
					// Interrupt request by client
					fmt.Printf("[BY CLIENT] OK, I will cancel (%s)!\n", r.URL.Path)
					return
				default:
					// Main some logic
					// Some very long logic, just for example
					time.Sleep(1 * time.Millisecond)
				}
			}

			fmt.Printf("After 12 seconds!\n")
			w.Header().Set("Content-Type", "text/html")
			if _, err := w.Write([]byte(`<div>After 12 seconds!</div>`)); err != nil {
				fmt.Printf("%s\n", err.Error())
				return
			}
			if _, err := w.Write([]byte(`<div>` + r.URL.Path + `</div>`)); err != nil {
				fmt.Printf("%s\n", err.Error())
				return
			}
		})
		srv := &http.Server{Addr: "127.0.0.1:8080", Handler: mux}
		go func() {
			fmt.Printf("Starting web server: http://127.0.0.1:8080/\n")
			if err := srv.ListenAndServe(); err != nil {
				if err != http.ErrServerClosed {
					fmt.Printf("Web server startup error: %s\n", err.Error())
					// Application can't working without http web server
					// Call cancel context func on error
					shutdown()
					return
				}
			}
		}()

		return &[]ctrlc.Iface{test, srv}
	}

	// Run application
	ctrlc.App(MyAppFunc)
}

Explanation

type Iface interface {
	Shutdown(ctx context.Context) error
}

type CallbackFunc func(ctx context.Context, shutdown context.CancelFunc) *[]Iface

func App(f CallbackFunc)
func AppWithTimeOut(t time.Duration, f CallbackFunc)

t in App function 8 seconds is a default timeout for Shutdown function, if Shutdown function will be not closed, context will be canceled aftet this value and will print error to console. f - is function for main application code with cancel context inside, ctx will be triggered when application got one of close/terminate signals and we can terminate application by calling shutdown function

Callback function f in App must return pointer to array of interfaces which must contain/implement Shutdown(ctx context.Context) error function for correct shutdown