1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
16 "git.arvados.org/arvados.git/sdk/go/arvados"
19 // Don't trust "passenger-config" (or "bundle install") to handle
20 // concurrent installs.
21 var passengerInstallMutex sync.Mutex
23 var railsEnv = []string{
24 "ARVADOS_RAILS_LOG_TO_STDOUT=1",
25 "ARVADOS_CONFIG_NOLEGACY=1", // don't load database.yml from source tree
28 // Install a Rails application's dependencies, including phusion
30 type installPassenger struct {
32 depends []supervisedTask
35 func (runner installPassenger) String() string {
36 return "installPassenger:" + runner.src
39 func (runner installPassenger) Run(ctx context.Context, fail func(error), super *Supervisor) error {
40 if super.ClusterType == "production" {
41 // passenger has already been installed via package
44 err := super.wait(ctx, runner.depends...)
49 passengerInstallMutex.Lock()
50 defer passengerInstallMutex.Unlock()
53 err = super.RunProgram(ctx, runner.src, &buf, nil, "gem", "list", "--details", "bundler")
57 for _, version := range []string{"1.16.6", "1.17.3", "2.0.2"} {
58 if !strings.Contains(buf.String(), "("+version+")") {
59 err = super.RunProgram(ctx, runner.src, nil, nil, "gem", "install", "--user", "--conservative", "--no-document", "bundler:1.16.6", "bundler:1.17.3", "bundler:2.0.2")
66 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "install", "--jobs", "4", "--path", filepath.Join(os.Getenv("HOME"), ".gem"))
70 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "exec", "passenger-config", "build-native-support")
74 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "exec", "passenger-config", "install-standalone-runtime")
78 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "exec", "passenger-config", "validate-install")
79 if err != nil && !strings.Contains(err.Error(), "exit status 2") {
80 // Exit code 2 indicates there were warnings (like
81 // "other passenger installations have been detected",
82 // which we can't expect to avoid) but no errors.
83 // Other non-zero exit codes (1, 9) indicate errors.
89 type runPassenger struct {
90 src string // path to app in source tree
91 varlibdir string // path to app (relative to /var/lib/arvados) in OS package
93 depends []supervisedTask
96 func (runner runPassenger) String() string {
97 return "runPassenger:" + runner.src
100 func (runner runPassenger) Run(ctx context.Context, fail func(error), super *Supervisor) error {
101 err := super.wait(ctx, runner.depends...)
105 port, err := internalPort(runner.svc)
107 return fmt.Errorf("bug: no internalPort for %q: %v (%#v)", runner, err, runner.svc)
110 if super.ClusterType == "production" {
111 appdir = "/var/lib/arvados/" + runner.varlibdir
116 if lvl, ok := map[string]string{
124 }[super.cluster.SystemLogs.LogLevel]; ok {
127 super.waitShutdown.Add(1)
129 defer super.waitShutdown.Done()
132 "passenger", "start",
134 "--log-level", loglevel,
135 "--no-friendly-error-pages",
136 "--disable-anonymous-telemetry",
137 "--disable-security-update-check",
138 "--no-compile-runtime",
139 "--no-install-runtime",
140 "--pid-file", filepath.Join(super.wwwtempdir, "passenger."+strings.Replace(appdir, "/", "_", -1)+".pid"),
142 if super.ClusterType == "production" {
143 cmdline = append([]string{"sudo", "-u", "www-data", "-E", "HOME=/var/www", "PATH=/var/lib/arvados/bin:" + os.Getenv("PATH"), "/var/lib/arvados/bin/bundle"}, cmdline[1:]...)
145 // This would be desirable in the production
146 // case too, but it fails with sudo because
147 // /dev/stderr is a symlink to a pty owned by
148 // root: "nginx: [emerg] open() "/dev/stderr"
149 // failed (13: Permission denied)"
150 cmdline = append(cmdline, "--log-file", "/dev/stderr")
152 env := append([]string{"TMPDIR=" + super.wwwtempdir}, railsEnv...)
153 err = super.RunProgram(ctx, appdir, nil, env, cmdline[0], cmdline[1:]...)