-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
170 lines (137 loc) · 4.54 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
// main package that executes the code
package main
import (
"encoding/json"
"fmt"
"github.com/aws/aws-lambda-go/events"
"github.com/aws/aws-lambda-go/lambda"
"github.com/awslabs/aws-lambda-go-api-proxy/gorillamux"
"github.com/gorilla/mux"
"github.com/spf13/pflag"
"github.com/spf13/viper"
"github.com/tomasen/realip"
"log"
"os/signal"
"syscall"
"net/http"
"os"
"time"
)
// Handler is an abstraction layer for implementing ServeHTTP.
type Handler struct {
HandlerFunc func(http.ResponseWriter, *http.Request) (int, error)
}
// ServeHTTP implementation with default error response. https://golang.org/src/net/http/server.go?s=2736:2799#L75
func (fn Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
if status, err := fn.HandlerFunc(w, r); err != nil {
w.WriteHeader(status)
_ = json.NewEncoder(w).Encode(struct {
Message string `json:"message"`
}{Message: err.Error()})
log.Printf("%d %s", status, err.Error())
}
}
// lambdaInitialized is an indicator that tells if the AWS Lambda function is in the startup phase.
var lambdaInitialized = false
// Translates Gorilla Mux calls to AWS API Gateway calls
var lambdaProxy func(request events.APIGatewayProxyRequest) (events.APIGatewayProxyResponse, error)
// LambdaHandler is the callback function for the AWS Lambda function.
func LambdaHandler(req events.APIGatewayProxyRequest) (events.APIGatewayProxyResponse, error) {
if !lambdaInitialized {
// stdout and stderr are sent to AWS CloudWatch Logs
log.Print("cold start")
r := AddRoutes()
muxLambda := gorillamux.New(r)
lambdaProxy = muxLambda.Proxy
lambdaInitialized = true
}
return lambdaProxy(req)
}
// WebServerHandler sets up a local web server for handling incoming requests.
func WebServerHandler(ip string, port int) {
log.Print("web server execution start")
r := AddRoutes()
srv := &http.Server{
Addr: fmt.Sprintf("%s:%d", ip, port),
// Good practice to set timeouts to avoid Slowloris attacks.
WriteTimeout: time.Second * 30,
ReadTimeout: time.Second * 30,
IdleTimeout: time.Second * 60,
Handler: r,
}
// Manage signals for graceful exit
var gracefulStop = make(chan os.Signal)
signal.Notify(gracefulStop, syscall.SIGTERM)
signal.Notify(gracefulStop, syscall.SIGINT)
go func() {
sig := <-gracefulStop
log.Printf("caught signal: %+v", sig)
log.Print("waiting 2 seconds to finish processing")
time.Sleep(2 * time.Second)
os.Exit(0)
}()
if err := srv.ListenAndServe(); err != nil {
log.Fatal(err)
}
}
// MainHandler handles the requests coming to `/`.
func MainHandler(w http.ResponseWriter, _ *http.Request) (status int, err error) {
// Todo: Build your function
w.Header().Set("Content-Type", "application/json; charset=utf8")
_ = json.NewEncoder(w).Encode(struct {
Message string `json:"message"`
}{
Message: fmt.Sprintf("Hello %s", viper.GetString("mysetting1")),
})
return
}
// Create logs for each request
func loggingMiddleware(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Printf("[%s] %s", realip.FromRequest(r), r.RequestURI)
next.ServeHTTP(w, r)
})
}
// AddRoutes adds the routes of the different calls to GorillaMux.
func AddRoutes() (r *mux.Router) {
// Gorilla/Mux
r = mux.NewRouter()
// Todo: Add your routes
_ = r.Handle("/", Handler{HandlerFunc: MainHandler})
// Finally
r.Use(loggingMiddleware)
return
}
func main() {
// Parse command-line parameters
pflag.Bool("version", false, "Return version number and exit.")
pflag.Bool("webserver", false, "run a local web-server instead of as an AWS Lambda function")
pflag.String("config", "", "read config from this local file")
pflag.String("ip", "127.0.0.1", "IP to listen on")
pflag.Uint("port", 3000, "Port to listen on")
pflag.Parse()
_ = viper.BindPFlags(pflag.CommandLine)
// Todo: Set defaults for configuration variables
viper.SetDefault("mysetting1", "world")
// Parse configuration from optional config file
if viper.GetString("config") != "" {
viper.SetConfigFile(viper.GetString("config"))
err := viper.ReadInConfig() // Find and read the config file
if err != nil { // Handle errors reading the config file
panic(fmt.Errorf("Fatal error config file: %s \n", err))
}
}
// Enable peek at environment variables for configuration settings.
viper.AutomaticEnv()
if viper.GetBool("version") {
// Todo: Set version number
fmt.Println("v0.0.0")
return
}
if viper.GetBool("webserver") {
WebServerHandler(viper.GetString("ip"), viper.GetInt("port"))
return
}
//Lambda function on AWS
lambda.Start(LambdaHandler)
}