push-to-dokku/services.go

134 lines
2.7 KiB
Go
Raw Normal View History

package main
import (
"bytes"
"fmt"
"log"
"os"
2024-07-13 01:21:00 +00:00
"os/exec"
"strings"
"text/template"
)
type Services struct {
cfg config
}
func NewServices(cfg config) *Services {
return &Services{cfg: cfg}
}
func (s *Services) AddPrivateKey() error {
log.Println("Adding private key")
2024-07-13 01:21:00 +00:00
if err := os.MkdirAll(os.ExpandEnv("${HOME}/.ssh"), 0700); err != nil {
return err
}
2024-07-13 01:13:40 +00:00
if err := writeFile(`${HOME}/.ssh/id_rsa`, s.cfg.PrivateKey, 0400); err != nil {
return err
}
return nil
}
func (s *Services) ConfigureSSH() error {
log.Println("Configuring SSH")
var bfr bytes.Buffer
if err := sshConfigTemplate.Execute(&bfr, struct {
Cfg config
Home string
}{
Cfg: s.cfg,
Home: os.Getenv("HOME"),
}); err != nil {
return err
}
2024-07-13 01:13:40 +00:00
if err := writeFile(`${HOME}/.ssh/config`, bfr.String(), 0400); err != nil {
return err
}
return nil
}
func (s *Services) PushRepository() error {
sshUrl := fmt.Sprintf(`dokku@%v`, s.cfg.HostName)
remoteUrl := fmt.Sprintf(`ssh://%v:22/%v`, sshUrl, s.cfg.AppName)
log.Println("Testing Dokku SSH connection")
2024-07-13 01:21:00 +00:00
if err := runCmd("ssh", sshUrl, "version"); err != nil {
return err
}
log.Println("SSH connection good. Pushing repository")
2024-07-13 01:21:00 +00:00
if err := runCmd("git", "remote", "add", "dokku", remoteUrl); err != nil {
return err
}
sourceBranch := s.cfg.LocalBranch
if sourceBranch == "" {
// Get the current ref
ref, err := runCmdExpectingOutput("git", "rev-parse", "HEAD")
if err != nil {
return err
}
sourceBranch = strings.TrimSpace(ref)
}
remoteBranch := s.cfg.RemoteBranch
if remoteBranch == "" {
remoteBranch = "main"
}
branchArg := sourceBranch
if sourceBranch != remoteBranch {
branchArg = fmt.Sprintf("%v:%v", sourceBranch, remoteBranch)
}
if err := runCmd("git", "push", "dokku", branchArg); err != nil {
return err
}
return nil
}
2024-07-13 01:21:00 +00:00
func runCmd(cmd ...string) error {
log.Printf(" .. [exec] %v", strings.Join(cmd, " "))
e := exec.Command(cmd[0], cmd[1:]...)
e.Stderr = os.Stderr
e.Stdout = os.Stdout
return e.Run()
}
func runCmdExpectingOutput(cmd ...string) (string, error) {
log.Printf(" .. [exec] %v", strings.Join(cmd, " "))
e := exec.Command(cmd[0], cmd[1:]...)
e.Stderr = os.Stderr
res, err := e.Output()
if err != nil {
return "", err
}
return string(res), nil
}
2024-07-13 01:13:40 +00:00
func writeFile(path string, content string, mode os.FileMode) error {
fullPath := os.ExpandEnv(path)
2024-07-13 01:21:00 +00:00
log.Printf(" .. [file] %v (%v bytes)", fullPath, len(content))
2024-07-13 01:13:40 +00:00
return os.WriteFile(fullPath, []byte(content), mode)
}
var (
sshConfigTemplate = template.Must(template.New("ssh-config").Parse(`
Host *
StrictHostKeyChecking no
2024-07-13 01:31:33 +00:00
Host {{.Cfg.HostName}}
HostName {{.Cfg.HostName}}
User dokku
IdentityFile {{.Home}}/.ssh/id_rsa
`))
)