mirror of
https://github.com/sourcegraph/sourcegraph.git
synced 2026-02-06 15:51:43 +00:00
* first pass at watchable docker commands * extracted options to common struct * added repo root as field in options * added docker specific config and tests * merged in config overrides for linux * cleanup * Update dev/sg/internal/run/command.go Co-authored-by: William Bezuidenhout <william.bezuidenhout@sourcegraph.com> * addressed comments * gofmt --------- Co-authored-by: William Bezuidenhout <william.bezuidenhout@sourcegraph.com>
102 lines
3.0 KiB
Go
102 lines
3.0 KiB
Go
package run
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
"github.com/rjeczalik/notify"
|
|
)
|
|
|
|
// A BazelCommand is a command definition for sg run/start that uses
|
|
// bazel under the hood. It will handle restarting itself autonomously,
|
|
// as long as iBazel is running and watch that specific target.
|
|
type BazelCommand struct {
|
|
Config SGConfigCommandOptions
|
|
Target string `yaml:"target"`
|
|
// RunTarget specifies a target that should be run via `bazel run $RunTarget` instead of directly executing the binary.
|
|
RunTarget string `yaml:"runTarget"`
|
|
}
|
|
|
|
// UnmarshalYAML implements the Unmarshaler interface for BazelCommand.
|
|
// This allows us to parse the flat YAML configuration into nested struct.
|
|
func (bc *BazelCommand) UnmarshalYAML(unmarshal func(any) error) error {
|
|
// In order to not recurse infinitely (calling UnmarshalYAML over and over) we create a
|
|
// temporary type alias.
|
|
// First parse the BazelCommand specific options
|
|
type rawBazel BazelCommand
|
|
if err := unmarshal((*rawBazel)(bc)); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Then parse the common options from the same list into a nested struct
|
|
return unmarshal(&bc.Config)
|
|
}
|
|
|
|
func (bc BazelCommand) GetBinaryLocation() (string, error) {
|
|
return binaryLocation(bc.Target)
|
|
}
|
|
|
|
func (bc BazelCommand) GetConfig() SGConfigCommandOptions {
|
|
return bc.Config
|
|
}
|
|
|
|
func (bc BazelCommand) watchPaths() ([]string, error) {
|
|
// If no target is defined, there is nothing to be built and watched
|
|
if bc.Target == "" {
|
|
return nil, nil
|
|
}
|
|
// Grab the location of the binary in bazel-out.
|
|
binLocation, err := bc.GetBinaryLocation()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return []string{binLocation}, nil
|
|
|
|
}
|
|
|
|
func (bc BazelCommand) StartWatch(ctx context.Context) (<-chan struct{}, error) {
|
|
if watchPaths, err := bc.watchPaths(); err != nil {
|
|
return nil, err
|
|
} else {
|
|
// skip remove events as we don't care about files being removed, we only
|
|
// want to know when the binary has been rebuilt
|
|
return WatchPaths(ctx, watchPaths, notify.Remove)
|
|
}
|
|
}
|
|
|
|
func (bc BazelCommand) GetExecCmd(ctx context.Context) (*exec.Cmd, error) {
|
|
var cmd string
|
|
var err error
|
|
if bc.RunTarget != "" {
|
|
cmd = "bazel run " + bc.RunTarget
|
|
} else {
|
|
if cmd, err = bc.GetBinaryLocation(); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return exec.CommandContext(ctx, "bash", "-c", fmt.Sprintf("%s\n%s", bc.Config.PreCmd, cmd)), nil
|
|
}
|
|
|
|
func binaryLocation(target string) (string, error) {
|
|
// Get the output directory from Bazel, which varies depending on which OS
|
|
// we're running against.
|
|
baseOutput, err := exec.Command("bazel", "info", "output_path").Output()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
// Trim "bazel-out" because the next bazel query will include it.
|
|
outputPath := strings.TrimSuffix(strings.TrimSpace(string(baseOutput)), "bazel-out")
|
|
|
|
// Get the binary from the specific target.
|
|
bin, err := exec.Command("bazel", "cquery", target, "--output=files").Output()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
binPath := strings.TrimSpace(string(bin))
|
|
|
|
return fmt.Sprintf("%s%s", outputPath, binPath), nil
|
|
}
|