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 // Install a Rails application's dependencies, including phusion
25 type installPassenger struct {
27 depends []supervisedTask
30 func (runner installPassenger) String() string {
31 return "installPassenger:" + runner.src
34 func (runner installPassenger) Run(ctx context.Context, fail func(error), super *Supervisor) error {
35 err := super.wait(ctx, runner.depends...)
40 passengerInstallMutex.Lock()
41 defer passengerInstallMutex.Unlock()
44 err = super.RunProgram(ctx, runner.src, &buf, nil, "gem", "list", "--details", "bundler")
48 for _, version := range []string{"1.11.0", "1.17.3", "2.0.2"} {
49 if !strings.Contains(buf.String(), "("+version+")") {
50 err = super.RunProgram(ctx, runner.src, nil, nil, "gem", "install", "--user", "bundler:1.11", "bundler:1.17.3", "bundler:2.0.2")
57 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "install", "--jobs", "4", "--path", filepath.Join(os.Getenv("HOME"), ".gem"))
61 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "exec", "passenger-config", "build-native-support")
65 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "exec", "passenger-config", "install-standalone-runtime")
69 err = super.RunProgram(ctx, runner.src, nil, nil, "bundle", "exec", "passenger-config", "validate-install")
70 if err != nil && !strings.Contains(err.Error(), "exit status 2") {
71 // Exit code 2 indicates there were warnings (like
72 // "other passenger installations have been detected",
73 // which we can't expect to avoid) but no errors.
74 // Other non-zero exit codes (1, 9) indicate errors.
80 type runPassenger struct {
83 depends []supervisedTask
86 func (runner runPassenger) String() string {
87 return "runPassenger:" + runner.src
90 func (runner runPassenger) Run(ctx context.Context, fail func(error), super *Supervisor) error {
91 err := super.wait(ctx, runner.depends...)
95 port, err := internalPort(runner.svc)
97 return fmt.Errorf("bug: no internalPort for %q: %v (%#v)", runner, err, runner.svc)
100 if lvl, ok := map[string]string{
108 }[super.cluster.SystemLogs.LogLevel]; ok {
111 super.waitShutdown.Add(1)
113 defer super.waitShutdown.Done()
114 err = super.RunProgram(ctx, runner.src, nil, []string{"ARVADOS_RAILS_LOG_TO_STDOUT=1"}, "bundle", "exec",
115 "passenger", "start",
117 "--log-file", "/dev/stderr",
118 "--log-level", loglevel,
119 "--no-friendly-error-pages",
120 "--pid-file", filepath.Join(super.tempdir, "passenger."+strings.Replace(runner.src, "/", "_", -1)+".pid"))