1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
18 "git.arvados.org/arvados.git/lib/boot"
19 "git.arvados.org/arvados.git/lib/cloud/cloudtest"
20 "git.arvados.org/arvados.git/lib/cmd"
21 "git.arvados.org/arvados.git/lib/config"
22 "git.arvados.org/arvados.git/lib/controller"
23 "git.arvados.org/arvados.git/lib/crunchrun"
24 "git.arvados.org/arvados.git/lib/dispatchcloud"
25 "git.arvados.org/arvados.git/lib/install"
26 "git.arvados.org/arvados.git/lib/lsf"
27 "git.arvados.org/arvados.git/lib/recovercollection"
28 "git.arvados.org/arvados.git/lib/service"
29 "git.arvados.org/arvados.git/sdk/go/arvados"
30 "git.arvados.org/arvados.git/sdk/go/health"
31 "git.arvados.org/arvados.git/services/githttpd"
32 keepbalance "git.arvados.org/arvados.git/services/keep-balance"
33 keepweb "git.arvados.org/arvados.git/services/keep-web"
34 "git.arvados.org/arvados.git/services/keepproxy"
35 "git.arvados.org/arvados.git/services/keepstore"
36 "git.arvados.org/arvados.git/services/ws"
37 "github.com/prometheus/client_golang/prometheus"
41 handler = cmd.Multi(map[string]cmd.Handler{
42 "version": cmd.Version,
43 "-version": cmd.Version,
44 "--version": cmd.Version,
47 "check": health.CheckCommand,
48 "cloudtest": cloudtest.Command,
49 "config-check": config.CheckCommand,
50 "config-defaults": config.DumpDefaultsCommand,
51 "config-dump": config.DumpCommand,
52 "controller": controller.Command,
53 "crunch-run": crunchrun.Command,
54 "dispatch-cloud": dispatchcloud.Command,
55 "dispatch-lsf": lsf.DispatchCommand,
56 "git-httpd": githttpd.Command,
57 "health": healthCommand,
58 "install": install.Command,
59 "init": install.InitCommand,
60 "keep-balance": keepbalance.Command,
61 "keep-web": keepweb.Command,
62 "keepproxy": keepproxy.Command,
63 "keepstore": keepstore.Command,
64 "recover-collection": recovercollection.Command,
65 "workbench2": wb2command{},
71 os.Exit(handler.RunCommand(os.Args[0], os.Args[1:], os.Stdin, os.Stdout, os.Stderr))
74 type wb2command struct{}
76 func (wb2command) RunCommand(prog string, args []string, stdin io.Reader, stdout, stderr io.Writer) int {
78 fmt.Fprintf(stderr, "usage: %s api-host listen-addr app-dir\n", prog)
81 configJSON, err := json.Marshal(map[string]string{"API_HOST": args[0]})
83 fmt.Fprintf(stderr, "json.Marshal: %s\n", err)
86 servefs := http.FileServer(http.Dir(args[2]))
87 mux := http.NewServeMux()
88 mux.Handle("/", http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
89 for _, ent := range strings.Split(req.URL.Path, "/") {
91 http.Error(w, "invalid URL path", http.StatusBadRequest)
95 fnm := filepath.Join(args[2], filepath.FromSlash(path.Clean("/"+req.URL.Path)))
96 if _, err := os.Stat(fnm); os.IsNotExist(err) {
99 servefs.ServeHTTP(w, req)
101 mux.HandleFunc("/config.json", func(w http.ResponseWriter, _ *http.Request) {
104 mux.HandleFunc("/_health/ping", func(w http.ResponseWriter, _ *http.Request) {
105 io.WriteString(w, `{"health":"OK"}`)
107 err = http.ListenAndServe(args[1], mux)
109 fmt.Fprintln(stderr, err.Error())
115 var healthCommand cmd.Handler = service.Command(arvados.ServiceNameHealth, func(ctx context.Context, cluster *arvados.Cluster, _ string, reg *prometheus.Registry) service.Handler {
116 mClockSkew := prometheus.NewGauge(prometheus.GaugeOpts{
117 Namespace: "arvados",
119 Name: "clock_skew_seconds",
120 Help: "Clock skew observed in most recent health check",
122 reg.MustRegister(mClockSkew)
123 return &health.Aggregator{
125 MetricClockSkew: mClockSkew,