2020-11-24 22:54:00 +08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/hibiken/asynq"
|
|
|
|
"github.com/rs/cors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// staticFileServer implements the http.Handler interface, so we can use it
|
|
|
|
// to respond to HTTP requests. The path to the static directory and
|
|
|
|
// path to the index file within that static directory are used to
|
|
|
|
// serve the SPA in the given static directory.
|
|
|
|
type staticFileServer struct {
|
|
|
|
staticPath string
|
|
|
|
indexPath string
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServeHTTP inspects the URL path to locate a file within the static dir
|
|
|
|
// on the SPA handler. If a file is found, it will be served. If not, the
|
|
|
|
// file located at the index path on the SPA handler will be served. This
|
|
|
|
// is suitable behavior for serving an SPA (single page application).
|
|
|
|
func (srv *staticFileServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// get the absolute path to prevent directory traversal
|
|
|
|
path, err := filepath.Abs(r.URL.Path)
|
|
|
|
if err != nil {
|
|
|
|
// if we failed to get the absolute path respond with a 400 bad request
|
|
|
|
// and stop
|
|
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// prepend the path with the path to the static directory
|
|
|
|
path = filepath.Join(srv.staticPath, path)
|
|
|
|
|
|
|
|
// check whether a file exists at the given path
|
|
|
|
_, err = os.Stat(path)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
// file does not exist, serve index.html
|
|
|
|
http.ServeFile(w, r, filepath.Join(srv.staticPath, srv.indexPath))
|
|
|
|
return
|
|
|
|
} else if err != nil {
|
|
|
|
// if we got an error (that wasn't that the file doesn't exist) stating the
|
|
|
|
// file, return a 500 internal server error and stop
|
|
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise, use http.FileServer to serve the static dir
|
|
|
|
http.FileServer(http.Dir(srv.staticPath)).ServeHTTP(w, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
const addr = "127.0.0.1:8080"
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
inspector := asynq.NewInspector(asynq.RedisClientOpt{
|
|
|
|
Addr: "localhost:6379",
|
|
|
|
})
|
|
|
|
defer inspector.Close()
|
|
|
|
|
|
|
|
router := mux.NewRouter()
|
2020-12-07 23:14:30 +08:00
|
|
|
router.Use(loggingMiddleware)
|
2020-11-24 22:54:00 +08:00
|
|
|
|
|
|
|
api := router.PathPrefix("/api").Subrouter()
|
2020-12-07 00:46:14 +08:00
|
|
|
// Queue endpoints.
|
|
|
|
api.HandleFunc("/queues", newListQueuesHandlerFunc(inspector)).Methods("GET")
|
|
|
|
api.HandleFunc("/queues/{qname}", newGetQueueHandlerFunc(inspector)).Methods("GET")
|
|
|
|
api.HandleFunc("/queues/{qname}", newDeleteQueueHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-07 22:41:16 +08:00
|
|
|
api.HandleFunc("/queues/{qname}:pause", newPauseQueueHandlerFunc(inspector)).Methods("POST")
|
|
|
|
api.HandleFunc("/queues/{qname}:resume", newResumeQueueHandlerFunc(inspector)).Methods("POST")
|
2020-12-07 00:46:14 +08:00
|
|
|
|
2020-12-28 07:45:54 +08:00
|
|
|
// Queue Historical Stats endpoint.
|
|
|
|
api.HandleFunc("/queue_stats", newListQueueStatsHandlerFunc(inspector)).Methods("GET")
|
|
|
|
|
2020-12-07 00:46:14 +08:00
|
|
|
// Task endpoints.
|
|
|
|
api.HandleFunc("/queues/{qname}/active_tasks", newListActiveTasksHandlerFunc(inspector)).Methods("GET")
|
2020-12-07 22:41:16 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/active_tasks/{task_id}:cancel", newCancelActiveTaskHandlerFunc(inspector)).Methods("POST")
|
2020-12-23 22:59:44 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/active_tasks:cancel_all", newCancelAllActiveTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-23 22:23:15 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/active_tasks:batch_cancel", newBatchCancelActiveTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-07 00:46:14 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/pending_tasks", newListPendingTasksHandlerFunc(inspector)).Methods("GET")
|
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks", newListScheduledTasksHandlerFunc(inspector)).Methods("GET")
|
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks/{task_key}", newDeleteTaskHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-07 23:22:04 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks:delete_all", newDeleteAllScheduledTasksHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-18 08:05:22 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks:batch_delete", newBatchDeleteTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-14 22:54:02 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks/{task_key}:run", newRunTaskHandlerFunc(inspector)).Methods("POST")
|
2020-12-19 22:07:23 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks:run_all", newRunAllScheduledTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-15 22:16:58 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks:batch_run", newBatchRunTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-19 22:51:46 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks/{task_key}:kill", newKillTaskHandlerFunc(inspector)).Methods("POST")
|
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks:kill_all", newKillAllScheduledTasksHandlerFunc(inspector)).Methods("POST")
|
|
|
|
api.HandleFunc("/queues/{qname}/scheduled_tasks:batch_kill", newBatchKillTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-07 00:46:14 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks", newListRetryTasksHandlerFunc(inspector)).Methods("GET")
|
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks/{task_key}", newDeleteTaskHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-07 23:22:04 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks:delete_all", newDeleteAllRetryTasksHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-18 08:05:22 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks:batch_delete", newBatchDeleteTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-14 22:54:02 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks/{task_key}:run", newRunTaskHandlerFunc(inspector)).Methods("POST")
|
2020-12-19 22:07:23 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks:run_all", newRunAllRetryTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-15 22:16:58 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks:batch_run", newBatchRunTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-19 22:51:46 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks/{task_key}:kill", newKillTaskHandlerFunc(inspector)).Methods("POST")
|
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks:kill_all", newKillAllRetryTasksHandlerFunc(inspector)).Methods("POST")
|
|
|
|
api.HandleFunc("/queues/{qname}/retry_tasks:batch_kill", newBatchKillTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-07 00:46:14 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks", newListDeadTasksHandlerFunc(inspector)).Methods("GET")
|
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks/{task_key}", newDeleteTaskHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-07 23:22:04 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks:delete_all", newDeleteAllDeadTasksHandlerFunc(inspector)).Methods("DELETE")
|
2020-12-18 08:05:22 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks:batch_delete", newBatchDeleteTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-14 22:54:02 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks/{task_key}:run", newRunTaskHandlerFunc(inspector)).Methods("POST")
|
2020-12-16 23:35:36 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks:run_all", newRunAllDeadTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-15 22:16:58 +08:00
|
|
|
api.HandleFunc("/queues/{qname}/dead_tasks:batch_run", newBatchRunTasksHandlerFunc(inspector)).Methods("POST")
|
2020-12-07 00:46:14 +08:00
|
|
|
|
|
|
|
// Scheduler Entry endpoints.
|
|
|
|
api.HandleFunc("/scheduler_entries", newListSchedulerEntriesHandlerFunc(inspector)).Methods("GET")
|
2020-12-27 02:05:19 +08:00
|
|
|
api.HandleFunc("/scheduler_entries/{entry_id}/enqueue_events", newListSchedulerEnqueueEventsHandlerFunc(inspector)).Methods("GET")
|
2020-11-24 22:54:00 +08:00
|
|
|
|
|
|
|
fs := &staticFileServer{staticPath: "ui/build", indexPath: "index.html"}
|
|
|
|
router.PathPrefix("/").Handler(fs)
|
|
|
|
|
2020-11-29 00:34:12 +08:00
|
|
|
c := cors.New(cors.Options{
|
|
|
|
AllowedMethods: []string{"GET", "POST", "DELETE"},
|
|
|
|
})
|
|
|
|
handler := c.Handler(router)
|
2020-11-24 22:54:00 +08:00
|
|
|
|
|
|
|
srv := &http.Server{
|
|
|
|
Handler: handler,
|
|
|
|
Addr: addr,
|
|
|
|
WriteTimeout: 10 * time.Second,
|
|
|
|
ReadTimeout: 10 * time.Second,
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Printf("Asynq Monitoring WebUI server is running on %s\n", addr)
|
|
|
|
log.Fatal(srv.ListenAndServe())
|
|
|
|
}
|