| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| 4 | "encoding/json" |
| 5 | "flag" |
| 6 | "fmt" |
| gio | 5e4d1a7 | 2024-10-09 15:25:29 +0400 | [diff] [blame] | 7 | "log" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 8 | "net" |
| 9 | "os" |
| gio | 37fba25 | 2025-07-03 14:02:04 +0400 | [diff] [blame] | 10 | "os/signal" |
| gio | 24d6e9a | 2025-06-24 15:02:29 +0400 | [diff] [blame] | 11 | "path/filepath" |
| gio | a421b06 | 2025-04-21 09:45:04 +0400 | [diff] [blame] | 12 | "strings" |
| gio | 37fba25 | 2025-07-03 14:02:04 +0400 | [diff] [blame] | 13 | "syscall" |
| gio | 5be6f78 | 2025-07-07 17:42:00 +0400 | [diff] [blame] | 14 | "time" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 15 | |
| 16 | "golang.org/x/crypto/ssh" |
| 17 | |
| 18 | "github.com/go-git/go-billy/v5/osfs" |
| 19 | "github.com/go-git/go-git/v5" |
| gio | 2b1157a | 2024-10-24 08:45:07 +0400 | [diff] [blame] | 20 | "github.com/go-git/go-git/v5/plumbing" |
| gio | 24d6e9a | 2025-06-24 15:02:29 +0400 | [diff] [blame] | 21 | "github.com/go-git/go-git/v5/plumbing/cache" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 22 | gitssh "github.com/go-git/go-git/v5/plumbing/transport/ssh" |
| gio | 24d6e9a | 2025-06-24 15:02:29 +0400 | [diff] [blame] | 23 | "github.com/go-git/go-git/v5/storage/filesystem" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 24 | ) |
| 25 | |
| 26 | var port = flag.Int("port", 3000, "Port to listen on") |
| gio | 266c04f | 2024-07-03 14:18:45 +0400 | [diff] [blame] | 27 | var appId = flag.String("app-id", "", "Application ID") |
| gio | b87415c | 2025-05-08 22:32:11 +0400 | [diff] [blame] | 28 | var service = flag.String("service", "", "Service name") |
| gio | e65d9a9 | 2025-06-19 09:02:32 +0400 | [diff] [blame] | 29 | var agentMode = flag.Bool("agent-mode", false, "Sketch agent mode") |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 30 | var repoAddr = flag.String("repo-addr", "", "Git repository address") |
| gio | 2b1157a | 2024-10-24 08:45:07 +0400 | [diff] [blame] | 31 | var branch = flag.String("branch", "", "Name of the branch to process") |
| gio | fc441e3 | 2024-11-11 16:26:14 +0400 | [diff] [blame] | 32 | var rootDir = flag.String("root-dir", "/", "Path to the app code") |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 33 | var sshKey = flag.String("ssh-key", "", "Private SSH key to access Git repository") |
| 34 | var appDir = flag.String("app-dir", "", "Path to store application repository locally") |
| 35 | var runCfg = flag.String("run-cfg", "", "Run configuration") |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 36 | var managerAddr = flag.String("manager-addr", "", "Address of the manager") |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 37 | |
| 38 | type Command struct { |
| 39 | Bin string `json:"bin"` |
| 40 | Args []string `json:"args"` |
| gio | 1364e43 | 2024-06-29 11:39:18 +0400 | [diff] [blame] | 41 | Env []string `json:"env"` |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 42 | } |
| 43 | |
| gio | 9635ccb | 2025-05-22 08:33:38 +0400 | [diff] [blame] | 44 | type Commit struct { |
| 45 | Hash string `json:"hash"` |
| 46 | Message string `json:"message"` |
| 47 | } |
| 48 | |
| 49 | func CloneRepositoryBranch(addr, branch, rootDir string, signer ssh.Signer, path string) (*Commit, error) { |
| gio | 2b1157a | 2024-10-24 08:45:07 +0400 | [diff] [blame] | 50 | ref := fmt.Sprintf("refs/heads/%s", branch) |
| gio | fc441e3 | 2024-11-11 16:26:14 +0400 | [diff] [blame] | 51 | opts := &git.CloneOptions{ |
| 52 | URL: addr, |
| 53 | RemoteName: "origin", |
| 54 | ReferenceName: plumbing.ReferenceName(ref), |
| 55 | SingleBranch: true, |
| 56 | Depth: 1, |
| 57 | InsecureSkipTLS: true, |
| 58 | Progress: os.Stdout, |
| 59 | } |
| 60 | if signer != nil { |
| 61 | opts.Auth = &gitssh.PublicKeys{ |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 62 | User: "git", |
| 63 | Signer: signer, |
| 64 | HostKeyCallbackHelper: gitssh.HostKeyCallbackHelper{ |
| 65 | HostKeyCallback: func(hostname string, remote net.Addr, key ssh.PublicKey) error { |
| 66 | // TODO(giolekva): verify server public key |
| 67 | fmt.Printf("--- %+v\n", ssh.MarshalAuthorizedKey(key)) |
| 68 | return nil |
| 69 | }, |
| 70 | }, |
| gio | fc441e3 | 2024-11-11 16:26:14 +0400 | [diff] [blame] | 71 | } |
| 72 | } |
| gio | 24d6e9a | 2025-06-24 15:02:29 +0400 | [diff] [blame] | 73 | repo, err := git.Clone( |
| 74 | filesystem.NewStorage( |
| 75 | osfs.New(filepath.Join(path, ".git"), osfs.WithBoundOS()), |
| 76 | cache.NewObjectLRUDefault(), |
| 77 | ), |
| 78 | osfs.New(path, osfs.WithBoundOS()), |
| 79 | opts, |
| 80 | ) |
| gio | f5ffedb | 2024-06-19 14:14:43 +0400 | [diff] [blame] | 81 | if err != nil { |
| gio | 9635ccb | 2025-05-22 08:33:38 +0400 | [diff] [blame] | 82 | return nil, err |
| gio | f5ffedb | 2024-06-19 14:14:43 +0400 | [diff] [blame] | 83 | } |
| gio | 9635ccb | 2025-05-22 08:33:38 +0400 | [diff] [blame] | 84 | head, err := repo.Head() |
| gio | f5ffedb | 2024-06-19 14:14:43 +0400 | [diff] [blame] | 85 | if err != nil { |
| gio | 9635ccb | 2025-05-22 08:33:38 +0400 | [diff] [blame] | 86 | return nil, err |
| gio | f5ffedb | 2024-06-19 14:14:43 +0400 | [diff] [blame] | 87 | } |
| gio | 9635ccb | 2025-05-22 08:33:38 +0400 | [diff] [blame] | 88 | commit, err := repo.CommitObject(head.Hash()) |
| 89 | if err != nil { |
| 90 | return nil, err |
| 91 | } |
| 92 | return &Commit{ |
| 93 | Hash: head.Hash().String(), |
| 94 | Message: commit.Message, |
| 95 | }, nil |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 96 | } |
| 97 | |
| gio | 5be6f78 | 2025-07-07 17:42:00 +0400 | [diff] [blame] | 98 | func readRunConfiguration(p string) []Command { |
| 99 | r, err := os.Open(p) |
| 100 | if err != nil { |
| 101 | panic(err) |
| 102 | } |
| 103 | defer r.Close() |
| 104 | var cmds []Command |
| 105 | if err := json.NewDecoder(r).Decode(&cmds); err != nil { |
| 106 | log.Fatal(err) |
| 107 | } |
| 108 | return cmds |
| 109 | } |
| 110 | |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 111 | func main() { |
| 112 | flag.Parse() |
| 113 | self, ok := os.LookupEnv("SELF_IP") |
| 114 | if !ok { |
| 115 | panic("no SELF_IP") |
| 116 | } |
| gio | 5155c1a | 2025-05-21 15:36:21 +0400 | [diff] [blame] | 117 | id, ok := os.LookupEnv("SELF_ID") |
| 118 | if !ok { |
| 119 | panic("no SELF_ID") |
| 120 | } |
| gio | fc441e3 | 2024-11-11 16:26:14 +0400 | [diff] [blame] | 121 | var signer ssh.Signer |
| gio | a421b06 | 2025-04-21 09:45:04 +0400 | [diff] [blame] | 122 | // TODO(gio): revisit this logic |
| 123 | if *sshKey != "" && !(strings.HasPrefix(*repoAddr, "http://") || strings.HasPrefix(*repoAddr, "https://")) { |
| gio | fc441e3 | 2024-11-11 16:26:14 +0400 | [diff] [blame] | 124 | key, err := os.ReadFile(*sshKey) |
| 125 | if err != nil { |
| 126 | panic(err) |
| 127 | } |
| 128 | signer, err = ssh.ParsePrivateKey(key) |
| 129 | if err != nil { |
| 130 | panic(err) |
| 131 | } |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 132 | } |
| gio | e65d9a9 | 2025-06-19 09:02:32 +0400 | [diff] [blame] | 133 | if !*agentMode { |
| 134 | if err := os.Mkdir(*appDir, os.ModePerm); err != nil { |
| 135 | panic(err) |
| 136 | } |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 137 | } |
| gio | 5be6f78 | 2025-07-07 17:42:00 +0400 | [diff] [blame] | 138 | cmds := readRunConfiguration(*runCfg) |
| gio | e65d9a9 | 2025-06-19 09:02:32 +0400 | [diff] [blame] | 139 | s := NewServer(*agentMode, *port, *appId, *service, id, *repoAddr, *branch, *rootDir, signer, *appDir, cmds, self, *managerAddr) |
| gio | 37fba25 | 2025-07-03 14:02:04 +0400 | [diff] [blame] | 140 | go func() { |
| 141 | if err := s.Start(); err != nil { |
| 142 | log.Fatal(err) |
| 143 | } else { |
| 144 | log.Println("Done") |
| 145 | os.Exit(0) |
| 146 | } |
| 147 | }() |
| gio | 5be6f78 | 2025-07-07 17:42:00 +0400 | [diff] [blame] | 148 | go func() { |
| 149 | for { |
| 150 | time.Sleep(30 * time.Second) |
| 151 | newCmds := readRunConfiguration(*runCfg) |
| 152 | if commandsChanged(cmds, newCmds) { |
| 153 | s.UpdateRunCommands(newCmds) |
| 154 | cmds = newCmds |
| 155 | } |
| 156 | } |
| 157 | }() |
| gio | 37fba25 | 2025-07-03 14:02:04 +0400 | [diff] [blame] | 158 | sigChan := make(chan os.Signal, 1) |
| 159 | signal.Notify(sigChan, syscall.SIGINT, syscall.SIGTERM) |
| 160 | <-sigChan |
| 161 | s.Stop() |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 162 | } |
| gio | 5be6f78 | 2025-07-07 17:42:00 +0400 | [diff] [blame] | 163 | |
| 164 | func commandsChanged(a, b []Command) bool { |
| 165 | if len(a) != len(b) { |
| 166 | return true |
| 167 | } |
| 168 | for i, x := range a { |
| 169 | y := b[i] |
| 170 | if x.Bin != y.Bin { |
| 171 | return true |
| 172 | } |
| 173 | if len(x.Args) != len(y.Args) { |
| 174 | return true |
| 175 | } |
| 176 | for j, k := range x.Args { |
| 177 | l := y.Args[j] |
| 178 | if k != l { |
| 179 | return true |
| 180 | } |
| 181 | } |
| 182 | if !*agentMode { |
| 183 | if len(x.Env) != len(y.Env) { |
| 184 | return true |
| 185 | } |
| 186 | for j, k := range x.Env { |
| 187 | l := y.Env[j] |
| 188 | if k != l { |
| 189 | return true |
| 190 | } |
| 191 | } |
| 192 | } |
| 193 | } |
| 194 | return false |
| 195 | } |