2021-08-25 22:12:19 +02:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2024-10-13 15:01:38 +02:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
|
2024-10-26 14:56:17 +02:00
|
|
|
"github.com/semaphoreui/semaphore/api"
|
|
|
|
"github.com/semaphoreui/semaphore/api/sockets"
|
|
|
|
"github.com/semaphoreui/semaphore/db"
|
|
|
|
"github.com/semaphoreui/semaphore/db/factory"
|
|
|
|
"github.com/semaphoreui/semaphore/services/schedules"
|
|
|
|
"github.com/semaphoreui/semaphore/services/tasks"
|
|
|
|
"github.com/semaphoreui/semaphore/util"
|
2021-08-25 22:36:04 +02:00
|
|
|
"github.com/gorilla/context"
|
|
|
|
"github.com/gorilla/handlers"
|
2024-04-20 15:06:19 +02:00
|
|
|
log "github.com/sirupsen/logrus"
|
2021-08-25 22:12:19 +02:00
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
2024-10-13 15:01:38 +02:00
|
|
|
var persistentFlags struct {
|
|
|
|
configPath string
|
|
|
|
noConfig bool
|
|
|
|
logLevel string
|
|
|
|
}
|
2021-08-25 22:12:19 +02:00
|
|
|
|
|
|
|
var rootCmd = &cobra.Command{
|
|
|
|
Use: "semaphore",
|
2024-06-12 21:53:00 +02:00
|
|
|
Short: "Semaphore UI is a beautiful web UI for Ansible",
|
|
|
|
Long: `Semaphore UI is a beautiful web UI for Ansible.
|
2024-10-26 14:56:17 +02:00
|
|
|
Source code is available at https://github.com/semaphoreui/semaphore.
|
|
|
|
Complete documentation is available at https://semaphoreui.com.`,
|
2021-08-25 22:12:19 +02:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
2022-01-21 17:27:48 +01:00
|
|
|
_ = cmd.Help()
|
|
|
|
os.Exit(0)
|
2021-08-25 22:12:19 +02:00
|
|
|
},
|
2024-10-13 15:10:54 +02:00
|
|
|
PersistentPreRun: func(cmd *cobra.Command, args []string) {
|
|
|
|
if persistentFlags.logLevel == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
lvl, err := log.ParseLevel(persistentFlags.logLevel)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.SetLevel(lvl)
|
|
|
|
},
|
2021-08-25 22:12:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func Execute() {
|
2024-10-13 15:01:38 +02:00
|
|
|
rootCmd.PersistentFlags().StringVar(&persistentFlags.logLevel, "log-level", "", "Log level: DEBUG, INFO, WARN, ERROR, FATAL, PANIC")
|
|
|
|
rootCmd.PersistentFlags().StringVar(&persistentFlags.configPath, "config", "", "Configuration file path")
|
|
|
|
rootCmd.PersistentFlags().BoolVar(&persistentFlags.noConfig, "no-config", false, "Don't use configuration file")
|
2021-08-25 22:12:19 +02:00
|
|
|
if err := rootCmd.Execute(); err != nil {
|
2024-09-25 21:28:22 +02:00
|
|
|
_, _ = fmt.Fprintln(os.Stderr, err)
|
2021-08-25 22:12:19 +02:00
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-25 22:36:04 +02:00
|
|
|
func runService() {
|
2022-11-09 17:30:35 +01:00
|
|
|
store := createStore("root")
|
2022-01-29 19:00:21 +01:00
|
|
|
taskPool := tasks.CreateTaskPool(store)
|
|
|
|
schedulePool := schedules.CreateSchedulePool(store, &taskPool)
|
2021-09-06 14:01:38 +02:00
|
|
|
|
|
|
|
defer schedulePool.Destroy()
|
2021-08-28 18:24:54 +02:00
|
|
|
|
2022-11-19 21:20:00 +01:00
|
|
|
util.Config.PrintDbInfo()
|
|
|
|
|
2023-09-20 10:22:26 +02:00
|
|
|
port := util.Config.Port
|
|
|
|
|
|
|
|
if !strings.HasPrefix(port, ":") {
|
|
|
|
port = ":" + port
|
|
|
|
}
|
|
|
|
|
2021-08-28 13:44:41 +02:00
|
|
|
fmt.Printf("Tmp Path (projects home) %v\n", util.Config.TmpPath)
|
2024-04-20 15:06:19 +02:00
|
|
|
fmt.Printf("Semaphore %v\n", util.Version())
|
2021-08-25 22:36:04 +02:00
|
|
|
fmt.Printf("Interface %v\n", util.Config.Interface)
|
|
|
|
fmt.Printf("Port %v\n", util.Config.Port)
|
|
|
|
|
|
|
|
go sockets.StartWS()
|
2021-09-06 14:01:38 +02:00
|
|
|
go schedulePool.Run()
|
2022-01-29 19:00:21 +01:00
|
|
|
go taskPool.Run()
|
2021-08-25 22:36:04 +02:00
|
|
|
|
|
|
|
route := api.Route()
|
|
|
|
|
|
|
|
route.Use(func(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
context.Set(r, "store", store)
|
2021-09-06 17:45:43 +02:00
|
|
|
context.Set(r, "schedule_pool", schedulePool)
|
2022-01-29 19:00:21 +01:00
|
|
|
context.Set(r, "task_pool", &taskPool)
|
2021-08-25 22:36:04 +02:00
|
|
|
next.ServeHTTP(w, r)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
var router http.Handler = route
|
|
|
|
|
|
|
|
router = handlers.ProxyHeaders(router)
|
|
|
|
http.Handle("/", router)
|
|
|
|
|
|
|
|
fmt.Println("Server is running")
|
|
|
|
|
2022-11-18 23:23:30 +01:00
|
|
|
if store.PermanentConnection() {
|
2022-11-09 17:30:35 +01:00
|
|
|
defer store.Close("root")
|
|
|
|
} else {
|
|
|
|
store.Close("root")
|
|
|
|
}
|
|
|
|
|
2023-09-20 10:22:26 +02:00
|
|
|
err := http.ListenAndServe(util.Config.Interface+port, cropTrailingSlashMiddleware(router))
|
2021-08-25 22:36:04 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-09 17:30:35 +01:00
|
|
|
func createStore(token string) db.Store {
|
2024-10-13 15:01:38 +02:00
|
|
|
util.ConfigInit(persistentFlags.configPath, persistentFlags.noConfig)
|
2021-08-25 22:12:19 +02:00
|
|
|
|
|
|
|
store := factory.CreateStore()
|
|
|
|
|
2022-11-18 23:23:30 +01:00
|
|
|
store.Connect(token)
|
|
|
|
|
2022-01-23 14:21:27 +01:00
|
|
|
err := db.Migrate(store)
|
2021-12-18 14:16:34 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-07-07 19:12:21 +02:00
|
|
|
err = db.FillConfigFromDB(store)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2024-07-10 14:35:21 +02:00
|
|
|
util.LookupDefaultApps()
|
2024-07-09 10:57:33 +02:00
|
|
|
|
2021-08-25 22:12:19 +02:00
|
|
|
return store
|
2021-12-18 14:16:34 +01:00
|
|
|
}
|