|
7 | 7 | "fmt"
|
8 | 8 | "io"
|
9 | 9 | "log"
|
| 10 | + "math" |
10 | 11 | "net"
|
11 | 12 | "net/http"
|
12 | 13 | "os"
|
@@ -109,6 +110,8 @@ type Application struct {
|
109 | 110 | // Hosts field is available after `Run` or `NewHost`.
|
110 | 111 | Hosts []*host.Supervisor
|
111 | 112 | hostConfigurators []host.Configurator
|
| 113 | + runError error |
| 114 | + runErrorMu sync.RWMutex |
112 | 115 | }
|
113 | 116 |
|
114 | 117 | // New creates and returns a fresh empty iris *Application instance.
|
@@ -627,6 +630,10 @@ func (app *Application) Shutdown(ctx stdContext.Context) error {
|
627 | 630 | app.mu.Lock()
|
628 | 631 | defer app.mu.Unlock()
|
629 | 632 |
|
| 633 | + defer func() { |
| 634 | + app.setRunError(ErrServerClosed) // make sure to set the error so any .Wait calls return. |
| 635 | + }() |
| 636 | + |
630 | 637 | for i, su := range app.Hosts {
|
631 | 638 | app.logger.Debugf("Host[%d]: Shutdown now", i)
|
632 | 639 | if err := su.Shutdown(ctx); err != nil {
|
@@ -1006,7 +1013,8 @@ var (
|
1006 | 1013 | // on the TCP network address "host:port" which
|
1007 | 1014 | // handles requests on incoming connections.
|
1008 | 1015 | //
|
1009 |
| -// Listen always returns a non-nil error. |
| 1016 | +// Listen always returns a non-nil error except |
| 1017 | +// when NonBlocking option is being passed, so the error goes to the Wait method. |
1010 | 1018 | // Ignore specific errors by using an `iris.WithoutServerError(iris.ErrServerClosed)`
|
1011 | 1019 | // as a second input argument.
|
1012 | 1020 | //
|
@@ -1048,15 +1056,132 @@ func (app *Application) Run(serve Runner, withOrWithout ...Configurator) error {
|
1048 | 1056 | app.logger.Debugf("Application: running using %d host(s)", len(app.Hosts)+1 /* +1 the current */)
|
1049 | 1057 | }
|
1050 | 1058 |
|
1051 |
| - // this will block until an error(unless supervisor's DeferFlow called from a Task). |
| 1059 | + if app.config.NonBlocking { |
| 1060 | + go func() { |
| 1061 | + err := app.serve(serve) |
| 1062 | + if err != nil { |
| 1063 | + app.setRunError(err) |
| 1064 | + } |
| 1065 | + }() |
| 1066 | + |
| 1067 | + return nil |
| 1068 | + } |
| 1069 | + |
| 1070 | + // this will block until an error(unless supervisor's DeferFlow called from a Task) |
| 1071 | + // or NonBlocking was passed (see above). |
| 1072 | + return app.serve(serve) |
| 1073 | +} |
| 1074 | + |
| 1075 | +func (app *Application) serve(serve Runner) error { |
1052 | 1076 | err := serve(app)
|
1053 | 1077 | if err != nil {
|
1054 | 1078 | app.logger.Error(err)
|
1055 | 1079 | }
|
| 1080 | + return err |
| 1081 | +} |
1056 | 1082 |
|
| 1083 | +func (app *Application) setRunError(err error) { |
| 1084 | + app.runErrorMu.Lock() |
| 1085 | + app.runError = err |
| 1086 | + app.runErrorMu.Unlock() |
| 1087 | +} |
| 1088 | + |
| 1089 | +func (app *Application) getRunError() error { |
| 1090 | + app.runErrorMu.RLock() |
| 1091 | + err := app.runError |
| 1092 | + app.runErrorMu.RUnlock() |
1057 | 1093 | return err
|
1058 | 1094 | }
|
1059 | 1095 |
|
| 1096 | +// Wait blocks the main goroutine until the server application is up and running. |
| 1097 | +// Useful only when `Run` is called with `iris.NonBlocking()` option. |
| 1098 | +func (app *Application) Wait(ctx stdContext.Context) error { |
| 1099 | + if !app.config.NonBlocking { |
| 1100 | + return nil |
| 1101 | + } |
| 1102 | + |
| 1103 | + // First check if there is an error already from the app.Run. |
| 1104 | + if err := app.getRunError(); err != nil { |
| 1105 | + return err |
| 1106 | + } |
| 1107 | + |
| 1108 | + // Set the base for exponential backoff. |
| 1109 | + base := 2.0 |
| 1110 | + |
| 1111 | + // Get the maximum number of retries by context or force to 7 retries. |
| 1112 | + var maxRetries int |
| 1113 | + // Get the deadline of the context. |
| 1114 | + if deadline, ok := ctx.Deadline(); ok { |
| 1115 | + now := time.Now() |
| 1116 | + timeout := deadline.Sub(now) |
| 1117 | + |
| 1118 | + maxRetries = getMaxRetries(timeout, base) |
| 1119 | + } else { |
| 1120 | + maxRetries = 7 // 256 seconds max. |
| 1121 | + } |
| 1122 | + |
| 1123 | + // Set the initial retry interval. |
| 1124 | + retryInterval := time.Second |
| 1125 | + |
| 1126 | + return app.tryConnect(ctx, maxRetries, retryInterval, base) |
| 1127 | +} |
| 1128 | + |
| 1129 | +// getMaxRetries calculates the maximum number of retries from the retry interval and the base. |
| 1130 | +func getMaxRetries(retryInterval time.Duration, base float64) int { |
| 1131 | + // Convert the retry interval to seconds. |
| 1132 | + seconds := retryInterval.Seconds() |
| 1133 | + // Apply the inverse formula. |
| 1134 | + retries := math.Log(seconds)/math.Log(base) - 1 |
| 1135 | + return int(math.Round(retries)) |
| 1136 | +} |
| 1137 | + |
| 1138 | +// tryConnect tries to connect to the server with the given context and retry parameters. |
| 1139 | +func (app *Application) tryConnect(ctx stdContext.Context, maxRetries int, retryInterval time.Duration, base float64) error { |
| 1140 | + address := app.config.GetVHost() // Get this server's listening address. |
| 1141 | + |
| 1142 | + // Try to connect to the server in a loop. |
| 1143 | + for i := 0; i < maxRetries; i++ { |
| 1144 | + // Check the context before each attempt. |
| 1145 | + select { |
| 1146 | + case <-ctx.Done(): |
| 1147 | + // Context is canceled, return the context error. |
| 1148 | + return ctx.Err() |
| 1149 | + default: |
| 1150 | + // Context is not canceled, proceed with the attempt. |
| 1151 | + conn, err := net.Dial("tcp", address) |
| 1152 | + if err == nil { |
| 1153 | + // Connection successful, close the connection and return nil. |
| 1154 | + conn.Close() |
| 1155 | + return nil // exit. |
| 1156 | + } // ignore error. |
| 1157 | + |
| 1158 | + // Connection failed, wait for the retry interval and try again. |
| 1159 | + time.Sleep(retryInterval) |
| 1160 | + // After each failed attempt, check the server Run's error again. |
| 1161 | + if err := app.getRunError(); err != nil { |
| 1162 | + return err |
| 1163 | + } |
| 1164 | + |
| 1165 | + // Increase the retry interval by the base raised to the power of the number of attempts. |
| 1166 | + /* |
| 1167 | + 0 2 seconds |
| 1168 | + 1 4 seconds |
| 1169 | + 2 8 seconds |
| 1170 | + 3 ~16 seconds |
| 1171 | + 4 ~32 seconds |
| 1172 | + 5 ~64 seconds |
| 1173 | + 6 ~128 seconds |
| 1174 | + 7 ~256 seconds |
| 1175 | + 8 ~512 seconds |
| 1176 | + ... |
| 1177 | + */ |
| 1178 | + retryInterval = time.Duration(math.Pow(base, float64(i+1))) * time.Second |
| 1179 | + } |
| 1180 | + } |
| 1181 | + // All attempts failed, return an error. |
| 1182 | + return fmt.Errorf("failed to connect to the server after %d retries", maxRetries) |
| 1183 | +} |
| 1184 | + |
1060 | 1185 | // https://ngrok.com/docs
|
1061 | 1186 | func (app *Application) tryStartTunneling() {
|
1062 | 1187 | if len(app.config.Tunneling.Tunnels) == 0 {
|
|
0 commit comments