-
Notifications
You must be signed in to change notification settings - Fork 159
/
runner.go
85 lines (71 loc) · 2.14 KB
/
runner.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
package empire
import (
"io"
"golang.org/x/net/context"
)
const (
// GenericProcessName is the process name for `emp run` processes not defined in the procfile.
GenericProcessName = "run"
)
// RunRecorder is a function that returns an io.Writer that will be written to
// to record Stdout and Stdin of interactive runs.
type RunRecorder func() (io.Writer, error)
type runnerService struct {
*Empire
}
func (r *runnerService) Run(ctx context.Context, opts RunOpts) error {
release, err := releasesFind(r.db, ReleasesQuery{App: opts.App})
if err != nil {
return err
}
procName := opts.Command[0]
var proc Process
// First, let's check if the command we're running matches a defined
// process in the Procfile/Formation. If it does, we'll replace the
// command, with the one in the procfile and expand it's arguments.
//
// For example, given a procfile like this:
//
// psql:
// command: ./bin/psql
//
// Calling `emp run psql DATABASE_URL` will expand the command to
// `./bin/psql DATABASE_URL`.
if cmd, ok := release.Formation[procName]; ok {
proc = cmd
proc.Command = append(cmd.Command, opts.Command[1:]...)
proc.NoService = false
} else {
// If we've set the flag to only allow `emp run` on commands
// defined in the procfile, return an error since the command is
// not defined in the procfile.
if r.AllowedCommands == AllowCommandProcfile {
return commandNotInFormation(Command{procName}, release.Formation)
}
// This is an unnamed command, fallback to a generic proc name.
procName = GenericProcessName
proc.Command = opts.Command
proc.SetConstraints(DefaultConstraints)
}
proc.Quantity = 1
// Set the size of the process.
if opts.Constraints != nil {
proc.SetConstraints(*opts.Constraints)
}
release.Formation = Formation{procName: proc}
a, err := newSchedulerApp(release)
if err != nil {
return err
}
for _, p := range a.Processes {
p.Stdin = opts.Stdin
p.Stdout = opts.Stdout
p.Stderr = opts.Stderr
p.Labels["empire.user"] = opts.User.Name
// Add additional environment variables to the process.
for k, v := range opts.Env {
p.Env[k] = v
}
}
return r.Scheduler.Run(ctx, a)
}