mirror of
				https://github.com/hibiken/asynqmon.git
				synced 2025-10-26 16:26:12 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			117 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			117 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| 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()
 | |
| 	router.Use(loggingMiddleware)
 | |
| 
 | |
| 	api := router.PathPrefix("/api").Subrouter()
 | |
| 	// 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")
 | |
| 	api.HandleFunc("/queues/{qname}:pause", newPauseQueueHandlerFunc(inspector)).Methods("POST")
 | |
| 	api.HandleFunc("/queues/{qname}:resume", newResumeQueueHandlerFunc(inspector)).Methods("POST")
 | |
| 
 | |
| 	// Task endpoints.
 | |
| 	api.HandleFunc("/queues/{qname}/active_tasks", newListActiveTasksHandlerFunc(inspector)).Methods("GET")
 | |
| 	api.HandleFunc("/queues/{qname}/active_tasks/{task_id}:cancel", newCancelActiveTaskHandlerFunc(inspector)).Methods("POST")
 | |
| 	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")
 | |
| 	api.HandleFunc("/queues/{qname}/scheduled_tasks:delete_all", newDeleteAllScheduledTasksHandlerFunc(inspector)).Methods("DELETE")
 | |
| 	api.HandleFunc("/queues/{qname}/scheduled_tasks/{task_key}:run", newRunTaskHandlerFunc(inspector)).Methods("POST")
 | |
| 	api.HandleFunc("/queues/{qname}/retry_tasks", newListRetryTasksHandlerFunc(inspector)).Methods("GET")
 | |
| 	api.HandleFunc("/queues/{qname}/retry_tasks/{task_key}", newDeleteTaskHandlerFunc(inspector)).Methods("DELETE")
 | |
| 	api.HandleFunc("/queues/{qname}/retry_tasks:delete_all", newDeleteAllRetryTasksHandlerFunc(inspector)).Methods("DELETE")
 | |
| 	api.HandleFunc("/queues/{qname}/retry_tasks/{task_key}:run", newRunTaskHandlerFunc(inspector)).Methods("POST")
 | |
| 	api.HandleFunc("/queues/{qname}/dead_tasks", newListDeadTasksHandlerFunc(inspector)).Methods("GET")
 | |
| 	api.HandleFunc("/queues/{qname}/dead_tasks/{task_key}", newDeleteTaskHandlerFunc(inspector)).Methods("DELETE")
 | |
| 	api.HandleFunc("/queues/{qname}/dead_tasks:delete_all", newDeleteAllDeadTasksHandlerFunc(inspector)).Methods("DELETE")
 | |
| 	api.HandleFunc("/queues/{qname}/dead_tasks:batch_delete", newBatchDeleteDeadTasksHandlerFunc(inspector)).Methods("POST")
 | |
| 	api.HandleFunc("/queues/{qname}/dead_tasks/{task_key}:run", newRunTaskHandlerFunc(inspector)).Methods("POST")
 | |
| 
 | |
| 	// Scheduler Entry endpoints.
 | |
| 	api.HandleFunc("/scheduler_entries", newListSchedulerEntriesHandlerFunc(inspector)).Methods("GET")
 | |
| 
 | |
| 	fs := &staticFileServer{staticPath: "ui/build", indexPath: "index.html"}
 | |
| 	router.PathPrefix("/").Handler(fs)
 | |
| 
 | |
| 	c := cors.New(cors.Options{
 | |
| 		AllowedMethods: []string{"GET", "POST", "DELETE"},
 | |
| 	})
 | |
| 	handler := c.Handler(router)
 | |
| 
 | |
| 	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())
 | |
| }
 |