ctrlc.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. package ctrlc
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "os/signal"
  7. "strings"
  8. "syscall"
  9. "time"
  10. )
  11. const C_ICON_START = "🌟"
  12. const C_ICON_WARN = "⚡️"
  13. const C_ICON_HOT = "🔥"
  14. const C_ICON_MAG = "✨"
  15. const C_ICON_SC = "🌳"
  16. type Iface interface {
  17. Shutdown(ctx context.Context) error
  18. }
  19. type CallbackFunc func(ctx context.Context, shutdown context.CancelFunc) *[]Iface
  20. func App(f CallbackFunc) {
  21. AppWithTimeOut(8*time.Second, f)
  22. }
  23. func AppWithTimeOut(t time.Duration, f CallbackFunc) {
  24. stop := make(chan os.Signal, 1)
  25. signal.Notify(stop, syscall.SIGTERM)
  26. signal.Notify(stop, syscall.SIGINT)
  27. fmt.Printf(
  28. icon_start(UseColors())+"%s\n",
  29. cly(
  30. UseColors(),
  31. fmt.Sprintf(
  32. "Application started (timeout %d sec)",
  33. t/time.Second,
  34. ),
  35. ),
  36. )
  37. sctx, shutdown := context.WithCancel(context.Background())
  38. ifaces := f(sctx, shutdown)
  39. select {
  40. case <-sctx.Done():
  41. fmt.Printf(
  42. "\r"+icon_warn(UseColors())+"%s\n",
  43. cly(
  44. UseColors(),
  45. fmt.Sprintf(
  46. "Shutting down (application) (timeout %d sec)",
  47. t/time.Second,
  48. ),
  49. ),
  50. )
  51. case val := <-stop:
  52. switch val {
  53. case syscall.SIGINT:
  54. fmt.Printf(
  55. "\r"+icon_warn(UseColors())+"%s\n",
  56. cly(
  57. UseColors(),
  58. fmt.Sprintf(
  59. "Shutting down (interrupt) (timeout %d sec)",
  60. t/time.Second,
  61. ),
  62. ),
  63. )
  64. case syscall.SIGTERM:
  65. fmt.Printf(
  66. icon_warn(UseColors())+"%s\n",
  67. cly(
  68. UseColors(),
  69. fmt.Sprintf(
  70. "Shutting down (terminate) (timeout %d sec)",
  71. t/time.Second,
  72. ),
  73. ),
  74. )
  75. default:
  76. fmt.Printf(
  77. icon_warn(UseColors())+"%s\n",
  78. cly(
  79. UseColors(),
  80. fmt.Sprintf(
  81. "Shutting down (timeout %d sec)",
  82. t/time.Second,
  83. ),
  84. ),
  85. )
  86. }
  87. }
  88. shutdown()
  89. errors := false
  90. ctx, cancel := context.WithTimeout(context.Background(), t)
  91. for _, iface := range *ifaces {
  92. if err := iface.Shutdown(ctx); err != nil {
  93. errors = true
  94. fmt.Printf(
  95. icon_hot(UseColors())+"%s\n",
  96. clr(
  97. UseColors(),
  98. fmt.Sprintf(
  99. "Shutdown error (%T): %s",
  100. iface,
  101. err.Error(),
  102. ),
  103. ),
  104. )
  105. }
  106. }
  107. cancel()
  108. if errors {
  109. fmt.Printf(
  110. icon_mag(UseColors())+"%s\n",
  111. cly(
  112. UseColors(),
  113. fmt.Sprintf(
  114. "Application exited with errors (timeout %d sec)",
  115. t/time.Second,
  116. ),
  117. ),
  118. )
  119. os.Exit(1)
  120. } else {
  121. fmt.Printf(
  122. icon_sc(UseColors())+"%s\n",
  123. clg(
  124. UseColors(),
  125. fmt.Sprintf(
  126. "Application exited successfully (timeout %d sec)",
  127. t/time.Second,
  128. ),
  129. ),
  130. )
  131. }
  132. }
  133. func UseColors() bool {
  134. useColors := strings.Contains(
  135. fmt.Sprintf("%s", os.Args),
  136. "--color=always",
  137. )
  138. if !useColors {
  139. useColors = strings.Contains(
  140. fmt.Sprintf("%s", os.Args),
  141. "-color=always",
  142. )
  143. }
  144. if !useColors {
  145. useColors = strings.Contains(
  146. fmt.Sprintf("%s", os.Args),
  147. "color=always",
  148. )
  149. }
  150. if !useColors {
  151. useColors = strings.Contains(
  152. fmt.Sprintf("%s", os.Args),
  153. "--color always",
  154. )
  155. }
  156. if !useColors {
  157. useColors = strings.Contains(
  158. fmt.Sprintf("%s", os.Args),
  159. "-color always",
  160. )
  161. }
  162. if !useColors {
  163. useColors = strings.Contains(
  164. fmt.Sprintf("%s", os.Args),
  165. "color always",
  166. )
  167. }
  168. return !IS_WIN_PLATFORM && useColors
  169. }