| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 1 | package installer |
| 2 | |
| 3 | import ( |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 4 | "bytes" |
| 5 | "encoding/json" |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 6 | "errors" |
| 7 | "fmt" |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 8 | "io" |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 9 | "io/fs" |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 10 | "io/ioutil" |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 11 | "net" |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 12 | "net/http" |
| 13 | "os" |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 14 | "path" |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 15 | "path/filepath" |
| Giorgi Lekveishvili | 378ea88 | 2023-12-12 13:59:18 +0400 | [diff] [blame] | 16 | "sync" |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 17 | "time" |
| 18 | |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 19 | "github.com/go-git/go-billy/v5/util" |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 20 | "github.com/go-git/go-git/v5" |
| 21 | "github.com/go-git/go-git/v5/plumbing/object" |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 22 | gitssh "github.com/go-git/go-git/v5/plumbing/transport/ssh" |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 23 | "golang.org/x/crypto/ssh" |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 24 | "sigs.k8s.io/yaml" |
| Giorgi Lekveishvili | 94cda9d | 2023-07-20 10:16:09 +0400 | [diff] [blame] | 25 | |
| 26 | "github.com/giolekva/pcloud/core/installer/soft" |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 27 | ) |
| 28 | |
| 29 | type RepoIO interface { |
| Giorgi Lekveishvili | 724885f | 2023-11-29 16:18:42 +0400 | [diff] [blame] | 30 | Addr() string |
| Giorgi Lekveishvili | 5c1b06e | 2024-03-28 15:19:44 +0400 | [diff] [blame] | 31 | Pull() error |
| Giorgi Lekveishvili | 7fb28bf | 2023-06-24 19:51:16 +0400 | [diff] [blame] | 32 | ReadConfig() (Config, error) |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 33 | ReadAppConfig(path string) (AppConfig, error) |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 34 | ReadKustomization(path string) (*Kustomization, error) |
| 35 | WriteKustomization(path string, kust Kustomization) error |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 36 | ReadYaml(path string) (map[string]any, error) |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 37 | WriteYaml(path string, data any) error |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 38 | CommitAndPush(message string) error |
| Giorgi Lekveishvili | 724885f | 2023-11-29 16:18:42 +0400 | [diff] [blame] | 39 | WriteCommitAndPush(path, contents, message string) error |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 40 | Reader(path string) (io.ReadCloser, error) |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 41 | Writer(path string) (io.WriteCloser, error) |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 42 | CreateDir(path string) error |
| 43 | RemoveDir(path string) error |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 44 | InstallApp(app App, path string, values map[string]any, derived Derived) error |
| 45 | RemoveApp(path string) error |
| 46 | FindAllInstances(root string, appId string) ([]AppConfig, error) |
| 47 | FindInstance(root string, id string) (AppConfig, error) |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 48 | } |
| 49 | |
| 50 | type repoIO struct { |
| Giorgi Lekveishvili | 94cda9d | 2023-07-20 10:16:09 +0400 | [diff] [blame] | 51 | repo *soft.Repository |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 52 | signer ssh.Signer |
| Giorgi Lekveishvili | 378ea88 | 2023-12-12 13:59:18 +0400 | [diff] [blame] | 53 | l sync.Locker |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 54 | } |
| 55 | |
| Giorgi Lekveishvili | 94cda9d | 2023-07-20 10:16:09 +0400 | [diff] [blame] | 56 | func NewRepoIO(repo *soft.Repository, signer ssh.Signer) RepoIO { |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 57 | return &repoIO{ |
| 58 | repo, |
| 59 | signer, |
| Giorgi Lekveishvili | 378ea88 | 2023-12-12 13:59:18 +0400 | [diff] [blame] | 60 | &sync.Mutex{}, |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 61 | } |
| 62 | } |
| 63 | |
| Giorgi Lekveishvili | 724885f | 2023-11-29 16:18:42 +0400 | [diff] [blame] | 64 | func (r *repoIO) Addr() string { |
| Giorgi Lekveishvili | 94cda9d | 2023-07-20 10:16:09 +0400 | [diff] [blame] | 65 | return r.repo.Addr.Addr |
| 66 | } |
| 67 | |
| Giorgi Lekveishvili | 5c1b06e | 2024-03-28 15:19:44 +0400 | [diff] [blame] | 68 | func (r *repoIO) Pull() error { |
| 69 | r.l.Lock() |
| 70 | defer r.l.Unlock() |
| 71 | return r.pullWithoutLock() |
| 72 | } |
| 73 | |
| 74 | func (r *repoIO) pullWithoutLock() error { |
| 75 | wt, err := r.repo.Worktree() |
| 76 | if err != nil { |
| 77 | fmt.Printf("EEEER wt: %s\b", err) |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 78 | return nil |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 79 | } |
| Giorgi Lekveishvili | 5c1b06e | 2024-03-28 15:19:44 +0400 | [diff] [blame] | 80 | err = wt.Pull(&git.PullOptions{ |
| 81 | Auth: auth(r.signer), |
| 82 | Force: true, |
| 83 | }) |
| 84 | // TODO(gio): propagate error |
| 85 | if err != nil { |
| 86 | fmt.Printf("EEEER: %s\b", err) |
| 87 | } |
| 88 | return nil |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 89 | } |
| 90 | |
| Giorgi Lekveishvili | 7fb28bf | 2023-06-24 19:51:16 +0400 | [diff] [blame] | 91 | func (r *repoIO) ReadConfig() (Config, error) { |
| 92 | configF, err := r.Reader(configFileName) |
| 93 | if err != nil { |
| 94 | return Config{}, err |
| 95 | } |
| 96 | defer configF.Close() |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 97 | var cfg Config |
| Giorgi Lekveishvili | 57dffb3 | 2023-08-07 15:45:43 +0400 | [diff] [blame] | 98 | if err := ReadYaml(configF, &cfg); err != nil { |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 99 | return Config{}, err |
| 100 | } else { |
| 101 | return cfg, nil |
| 102 | } |
| 103 | } |
| 104 | |
| 105 | func (r *repoIO) ReadAppConfig(path string) (AppConfig, error) { |
| 106 | configF, err := r.Reader(path) |
| 107 | if err != nil { |
| 108 | return AppConfig{}, err |
| 109 | } |
| 110 | defer configF.Close() |
| 111 | var cfg AppConfig |
| Giorgi Lekveishvili | 57dffb3 | 2023-08-07 15:45:43 +0400 | [diff] [blame] | 112 | if err := ReadYaml(configF, &cfg); err != nil { |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 113 | return AppConfig{}, err |
| 114 | } else { |
| 115 | return cfg, nil |
| 116 | } |
| Giorgi Lekveishvili | 7fb28bf | 2023-06-24 19:51:16 +0400 | [diff] [blame] | 117 | } |
| 118 | |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 119 | func (r *repoIO) ReadKustomization(path string) (*Kustomization, error) { |
| 120 | inp, err := r.Reader(path) |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 121 | if err != nil { |
| 122 | return nil, err |
| 123 | } |
| 124 | defer inp.Close() |
| 125 | return ReadKustomization(inp) |
| 126 | } |
| 127 | |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 128 | func (r *repoIO) Reader(path string) (io.ReadCloser, error) { |
| 129 | wt, err := r.repo.Worktree() |
| 130 | if err != nil { |
| 131 | return nil, err |
| 132 | } |
| 133 | return wt.Filesystem.Open(path) |
| 134 | } |
| 135 | |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 136 | func (r *repoIO) Writer(path string) (io.WriteCloser, error) { |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 137 | wt, err := r.repo.Worktree() |
| 138 | if err != nil { |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 139 | return nil, err |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 140 | } |
| 141 | if err := wt.Filesystem.MkdirAll(filepath.Dir(path), fs.ModePerm); err != nil { |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 142 | return nil, err |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 143 | } |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 144 | return wt.Filesystem.Create(path) |
| 145 | } |
| 146 | |
| 147 | func (r *repoIO) WriteKustomization(path string, kust Kustomization) error { |
| 148 | out, err := r.Writer(path) |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 149 | if err != nil { |
| 150 | return err |
| 151 | } |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 152 | return kust.Write(out) |
| 153 | } |
| 154 | |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 155 | func (r *repoIO) WriteYaml(path string, data any) error { |
| 156 | out, err := r.Writer(path) |
| 157 | if err != nil { |
| 158 | return err |
| 159 | } |
| 160 | serialized, err := yaml.Marshal(data) |
| 161 | if err != nil { |
| 162 | return err |
| 163 | } |
| 164 | if _, err := out.Write(serialized); err != nil { |
| 165 | return err |
| 166 | } |
| 167 | return nil |
| 168 | } |
| 169 | |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 170 | func (r *repoIO) ReadYaml(path string) (map[string]any, error) { |
| Giorgi Lekveishvili | 1506a4f | 2023-07-11 11:49:02 +0400 | [diff] [blame] | 171 | inp, err := r.Reader(path) |
| 172 | if err != nil { |
| 173 | return nil, err |
| 174 | } |
| 175 | data := make(map[string]any) |
| Giorgi Lekveishvili | 57dffb3 | 2023-08-07 15:45:43 +0400 | [diff] [blame] | 176 | if err := ReadYaml(inp, &data); err != nil { |
| Giorgi Lekveishvili | 1506a4f | 2023-07-11 11:49:02 +0400 | [diff] [blame] | 177 | return nil, err |
| 178 | } |
| 179 | return data, err |
| 180 | } |
| 181 | |
| Giorgi Lekveishvili | 724885f | 2023-11-29 16:18:42 +0400 | [diff] [blame] | 182 | func (r *repoIO) WriteCommitAndPush(path, contents, message string) error { |
| 183 | w, err := r.Writer(path) |
| 184 | if err != nil { |
| 185 | return err |
| 186 | } |
| 187 | defer w.Close() |
| 188 | if _, err := io.WriteString(w, contents); err != nil { |
| 189 | return err |
| 190 | } |
| 191 | return r.CommitAndPush(message) |
| 192 | } |
| 193 | |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 194 | func (r *repoIO) CommitAndPush(message string) error { |
| 195 | wt, err := r.repo.Worktree() |
| 196 | if err != nil { |
| 197 | return err |
| 198 | } |
| 199 | if err := wt.AddGlob("*"); err != nil { |
| 200 | return err |
| 201 | } |
| 202 | if _, err := wt.Commit(message, &git.CommitOptions{ |
| 203 | Author: &object.Signature{ |
| 204 | Name: "pcloud-installer", |
| 205 | When: time.Now(), |
| 206 | }, |
| 207 | }); err != nil { |
| 208 | return err |
| 209 | } |
| 210 | return r.repo.Push(&git.PushOptions{ |
| Giorgi Lekveishvili | 87be4ae | 2023-06-11 23:41:09 +0400 | [diff] [blame] | 211 | RemoteName: "origin", |
| Giorgi Lekveishvili | 3550b43 | 2023-06-09 19:37:51 +0400 | [diff] [blame] | 212 | Auth: auth(r.signer), |
| 213 | }) |
| 214 | } |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 215 | |
| 216 | func (r *repoIO) CreateDir(path string) error { |
| 217 | wt, err := r.repo.Worktree() |
| 218 | if err != nil { |
| 219 | return err |
| 220 | } |
| 221 | return wt.Filesystem.MkdirAll(path, fs.ModePerm) |
| 222 | } |
| 223 | |
| 224 | func (r *repoIO) RemoveDir(path string) error { |
| 225 | wt, err := r.repo.Worktree() |
| 226 | if err != nil { |
| 227 | return err |
| 228 | } |
| 229 | err = util.RemoveAll(wt.Filesystem, path) |
| 230 | if err == nil || errors.Is(err, fs.ErrNotExist) { |
| 231 | return nil |
| 232 | } |
| 233 | return err |
| 234 | } |
| 235 | |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 236 | type Release struct { |
| Giorgi Lekveishvili | 9b52ab9 | 2024-01-05 13:12:48 +0400 | [diff] [blame] | 237 | Namespace string `json:"namespace"` |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 238 | RepoAddr string `json:"repoAddr"` |
| 239 | AppDir string `json:"appDir"` |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 240 | } |
| 241 | |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 242 | type Derived struct { |
| Giorgi Lekveishvili | 9b52ab9 | 2024-01-05 13:12:48 +0400 | [diff] [blame] | 243 | Release Release `json:"release"` |
| 244 | Global Values `json:"global"` |
| 245 | Values map[string]any `json:"input"` // TODO(gio): rename to input |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 246 | } |
| 247 | |
| 248 | type AppConfig struct { |
| 249 | Id string `json:"id"` |
| 250 | AppId string `json:"appId"` |
| 251 | Config map[string]any `json:"config"` |
| 252 | Derived Derived `json:"derived"` |
| 253 | } |
| 254 | |
| gio | 5c44e6c | 2024-04-12 16:52:59 +0400 | [diff] [blame^] | 255 | func (a AppConfig) Input(schema Schema) map[string]any { |
| 256 | ret, err := derivedToConfig(a.Derived.Values, schema) |
| 257 | if err != nil { |
| 258 | panic(err) // TODO(gio): handle |
| 259 | } |
| 260 | return ret |
| 261 | } |
| 262 | |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 263 | type allocatePortReq struct { |
| 264 | Protocol string `json:"protocol"` |
| 265 | SourcePort int `json:"sourcePort"` |
| 266 | TargetService string `json:"targetService"` |
| 267 | TargetPort int `json:"targetPort"` |
| 268 | } |
| 269 | |
| 270 | // TODO(gio): most of this logic should move to AppManager |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 271 | func (r *repoIO) InstallApp(app App, appRootDir string, values map[string]any, derived Derived) error { |
| Giorgi Lekveishvili | 378ea88 | 2023-12-12 13:59:18 +0400 | [diff] [blame] | 272 | r.l.Lock() |
| 273 | defer r.l.Unlock() |
| Giorgi Lekveishvili | 5c1b06e | 2024-03-28 15:19:44 +0400 | [diff] [blame] | 274 | if err := r.pullWithoutLock(); err != nil { |
| 275 | return err |
| 276 | } |
| Giorgi Lekveishvili | 6e81318 | 2023-06-30 13:45:30 +0400 | [diff] [blame] | 277 | if !filepath.IsAbs(appRootDir) { |
| 278 | return fmt.Errorf("Expected absolute path: %s", appRootDir) |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 279 | } |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 280 | derived.Release.RepoAddr = r.repo.Addr.FullAddress() |
| 281 | // TODO(gio): maybe client should populate this? |
| 282 | derived.Release.AppDir = appRootDir |
| 283 | rendered, err := app.Render(derived) |
| 284 | if err != nil { |
| 285 | return err |
| 286 | } |
| 287 | for _, p := range rendered.Ports { |
| 288 | var buf bytes.Buffer |
| 289 | req := allocatePortReq{ |
| 290 | Protocol: p.Protocol, |
| 291 | SourcePort: p.SourcePort, |
| 292 | TargetService: p.TargetService, |
| 293 | TargetPort: p.TargetPort, |
| 294 | } |
| 295 | fmt.Printf("%+v\n", req) |
| 296 | if err := json.NewEncoder(&buf).Encode(req); err != nil { |
| 297 | return err |
| 298 | } |
| 299 | resp, err := http.Post(p.Allocator, "application/json", &buf) |
| 300 | if err != nil { |
| 301 | return err |
| 302 | } |
| 303 | if resp.StatusCode != http.StatusOK { |
| 304 | io.Copy(os.Stdout, resp.Body) |
| 305 | return fmt.Errorf("Could not allocate port %d, status code: %d", p.SourcePort, resp.StatusCode) |
| 306 | } |
| 307 | } |
| 308 | if err := r.pullWithoutLock(); err != nil { |
| 309 | return err |
| 310 | } |
| Giorgi Lekveishvili | 6e81318 | 2023-06-30 13:45:30 +0400 | [diff] [blame] | 311 | appRootDir = filepath.Clean(appRootDir) |
| 312 | for p := appRootDir; p != "/"; { |
| 313 | parent, child := filepath.Split(p) |
| 314 | kustPath := filepath.Join(parent, "kustomization.yaml") |
| 315 | kust, err := r.ReadKustomization(kustPath) |
| 316 | if err != nil { |
| 317 | if errors.Is(err, fs.ErrNotExist) { |
| 318 | k := NewKustomization() |
| 319 | kust = &k |
| 320 | } else { |
| 321 | return err |
| 322 | } |
| 323 | } |
| 324 | kust.AddResources(child) |
| 325 | if err := r.WriteKustomization(kustPath, *kust); err != nil { |
| 326 | return err |
| 327 | } |
| 328 | p = filepath.Clean(parent) |
| 329 | } |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 330 | { |
| 331 | if err := r.RemoveDir(appRootDir); err != nil { |
| 332 | return err |
| 333 | } |
| 334 | if err := r.CreateDir(appRootDir); err != nil { |
| 335 | return err |
| 336 | } |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 337 | cfg := AppConfig{ |
| Giorgi Lekveishvili | 08af67a | 2024-01-18 08:53:05 +0400 | [diff] [blame] | 338 | AppId: app.Name(), |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 339 | Config: values, |
| 340 | Derived: derived, |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 341 | } |
| 342 | if err := r.WriteYaml(path.Join(appRootDir, configFileName), cfg); err != nil { |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 343 | return err |
| 344 | } |
| 345 | } |
| 346 | { |
| 347 | appKust := NewKustomization() |
| Giorgi Lekveishvili | 9b52ab9 | 2024-01-05 13:12:48 +0400 | [diff] [blame] | 348 | for name, contents := range rendered.Resources { |
| Giorgi Lekveishvili | 3f697b1 | 2024-01-04 00:56:06 +0400 | [diff] [blame] | 349 | appKust.AddResources(name) |
| 350 | out, err := r.Writer(path.Join(appRootDir, name)) |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 351 | if err != nil { |
| 352 | return err |
| 353 | } |
| 354 | defer out.Close() |
| Giorgi Lekveishvili | 3f697b1 | 2024-01-04 00:56:06 +0400 | [diff] [blame] | 355 | if _, err := out.Write(contents); err != nil { |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 356 | return err |
| 357 | } |
| 358 | } |
| 359 | if err := r.WriteKustomization(path.Join(appRootDir, "kustomization.yaml"), appKust); err != nil { |
| 360 | return err |
| 361 | } |
| 362 | } |
| Giorgi Lekveishvili | 7c03739 | 2024-03-11 14:40:24 +0400 | [diff] [blame] | 363 | return r.CommitAndPush(fmt.Sprintf("install: %s", app.Name())) |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 364 | } |
| 365 | |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 366 | func (r *repoIO) RemoveApp(appRootDir string) error { |
| Giorgi Lekveishvili | 378ea88 | 2023-12-12 13:59:18 +0400 | [diff] [blame] | 367 | r.l.Lock() |
| 368 | defer r.l.Unlock() |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 369 | r.RemoveDir(appRootDir) |
| 370 | parent, child := filepath.Split(appRootDir) |
| 371 | kustPath := filepath.Join(parent, "kustomization.yaml") |
| 372 | kust, err := r.ReadKustomization(kustPath) |
| 373 | if err != nil { |
| 374 | return err |
| 375 | } |
| 376 | kust.RemoveResources(child) |
| 377 | r.WriteKustomization(kustPath, *kust) |
| 378 | return r.CommitAndPush(fmt.Sprintf("uninstall: %s", child)) |
| 379 | } |
| 380 | |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 381 | func (r *repoIO) FindAllInstances(root string, name string) ([]AppConfig, error) { |
| 382 | if !filepath.IsAbs(root) { |
| 383 | return nil, fmt.Errorf("Expected absolute path: %s", root) |
| 384 | } |
| 385 | kust, err := r.ReadKustomization(filepath.Join(root, "kustomization.yaml")) |
| 386 | if err != nil { |
| 387 | return nil, err |
| 388 | } |
| 389 | ret := make([]AppConfig, 0) |
| 390 | for _, app := range kust.Resources { |
| 391 | cfg, err := r.ReadAppConfig(filepath.Join(root, app, "config.yaml")) |
| 392 | if err != nil { |
| 393 | return nil, err |
| 394 | } |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 395 | cfg.Id = app |
| 396 | if cfg.AppId == name { |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 397 | ret = append(ret, cfg) |
| 398 | } |
| 399 | } |
| 400 | return ret, nil |
| 401 | } |
| 402 | |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 403 | func (r *repoIO) FindInstance(root string, id string) (AppConfig, error) { |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 404 | if !filepath.IsAbs(root) { |
| 405 | return AppConfig{}, fmt.Errorf("Expected absolute path: %s", root) |
| 406 | } |
| 407 | kust, err := r.ReadKustomization(filepath.Join(root, "kustomization.yaml")) |
| 408 | if err != nil { |
| 409 | return AppConfig{}, err |
| 410 | } |
| 411 | for _, app := range kust.Resources { |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 412 | if app == id { |
| 413 | cfg, err := r.ReadAppConfig(filepath.Join(root, app, "config.yaml")) |
| 414 | if err != nil { |
| 415 | return AppConfig{}, err |
| 416 | } |
| 417 | cfg.Id = id |
| 418 | return cfg, nil |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 419 | } |
| 420 | } |
| 421 | return AppConfig{}, nil |
| 422 | } |
| 423 | |
| Giorgi Lekveishvili | 0ccd148 | 2023-06-21 15:02:24 +0400 | [diff] [blame] | 424 | func auth(signer ssh.Signer) *gitssh.PublicKeys { |
| 425 | return &gitssh.PublicKeys{ |
| 426 | Signer: signer, |
| 427 | HostKeyCallbackHelper: gitssh.HostKeyCallbackHelper{ |
| 428 | HostKeyCallback: func(hostname string, remote net.Addr, key ssh.PublicKey) error { |
| 429 | // TODO(giolekva): verify server public key |
| 430 | // fmt.Printf("## %s || %s -- \n", serverPubKey, ssh.MarshalAuthorizedKey(key)) |
| 431 | return nil |
| 432 | }, |
| 433 | }, |
| 434 | } |
| 435 | } |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 436 | |
| Giorgi Lekveishvili | 57dffb3 | 2023-08-07 15:45:43 +0400 | [diff] [blame] | 437 | func ReadYaml[T any](r io.Reader, o *T) error { |
| Giorgi Lekveishvili | 7695148 | 2023-06-30 23:25:09 +0400 | [diff] [blame] | 438 | if contents, err := ioutil.ReadAll(r); err != nil { |
| 439 | return err |
| 440 | } else { |
| 441 | return yaml.UnmarshalStrict(contents, o) |
| 442 | } |
| 443 | } |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 444 | |
| Giorgi Lekveishvili | 7c42760 | 2024-01-04 00:13:55 +0400 | [diff] [blame] | 445 | func deriveValues(values any, schema Schema, networks []Network) (map[string]any, error) { |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 446 | ret := make(map[string]any) |
| Giorgi Lekveishvili | b6a5806 | 2024-04-02 16:49:19 +0400 | [diff] [blame] | 447 | for k, def := range schema.Fields() { |
| 448 | // TODO(gio): validate that it is map |
| 449 | v, ok := values.(map[string]any)[k] |
| 450 | // TODO(gio): if missing use default value |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 451 | if !ok { |
| Giorgi Lekveishvili | b6a5806 | 2024-04-02 16:49:19 +0400 | [diff] [blame] | 452 | if def.Kind() == KindSSHKey { |
| 453 | key, err := NewECDSASSHKeyPair("tmp") |
| 454 | if err != nil { |
| 455 | return nil, err |
| 456 | } |
| 457 | ret[k] = map[string]string{ |
| 458 | "public": string(key.RawAuthorizedKey()), |
| 459 | "private": string(key.RawPrivateKey()), |
| 460 | } |
| 461 | } |
| 462 | continue |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 463 | } |
| Giorgi Lekveishvili | 7c42760 | 2024-01-04 00:13:55 +0400 | [diff] [blame] | 464 | switch def.Kind() { |
| 465 | case KindBoolean: |
| Giorgi Lekveishvili | 0ba5e40 | 2024-03-20 15:56:30 +0400 | [diff] [blame] | 466 | ret[k] = v |
| Giorgi Lekveishvili | 7c42760 | 2024-01-04 00:13:55 +0400 | [diff] [blame] | 467 | case KindString: |
| Giorgi Lekveishvili | a1e7790 | 2023-11-06 14:48:27 +0400 | [diff] [blame] | 468 | ret[k] = v |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 469 | case KindInt: |
| 470 | ret[k] = v |
| Giorgi Lekveishvili | 7c42760 | 2024-01-04 00:13:55 +0400 | [diff] [blame] | 471 | case KindNetwork: |
| 472 | n, err := findNetwork(networks, v.(string)) // TODO(giolekva): validate |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 473 | if err != nil { |
| 474 | return nil, err |
| 475 | } |
| Giorgi Lekveishvili | 7c42760 | 2024-01-04 00:13:55 +0400 | [diff] [blame] | 476 | ret[k] = n |
| Giorgi Lekveishvili | a09fad7 | 2024-03-21 15:24:35 +0400 | [diff] [blame] | 477 | case KindAuth: |
| 478 | r, err := deriveValues(v, AuthSchema, networks) |
| 479 | if err != nil { |
| 480 | return nil, err |
| 481 | } |
| 482 | ret[k] = r |
| Giorgi Lekveishvili | b6a5806 | 2024-04-02 16:49:19 +0400 | [diff] [blame] | 483 | case KindSSHKey: |
| 484 | r, err := deriveValues(v, SSHKeySchema, networks) |
| 485 | if err != nil { |
| 486 | return nil, err |
| 487 | } |
| 488 | ret[k] = r |
| Giorgi Lekveishvili | 7c42760 | 2024-01-04 00:13:55 +0400 | [diff] [blame] | 489 | case KindStruct: |
| 490 | r, err := deriveValues(v, def, networks) |
| 491 | if err != nil { |
| 492 | return nil, err |
| 493 | } |
| 494 | ret[k] = r |
| 495 | default: |
| 496 | return nil, fmt.Errorf("Should not reach!") |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 497 | } |
| 498 | } |
| 499 | return ret, nil |
| 500 | } |
| 501 | |
| gio | 5c44e6c | 2024-04-12 16:52:59 +0400 | [diff] [blame^] | 502 | func derivedToConfig(derived map[string]any, schema Schema) (map[string]any, error) { |
| 503 | ret := make(map[string]any) |
| 504 | for k, def := range schema.Fields() { |
| 505 | v, ok := derived[k] |
| 506 | // TODO(gio): if missing use default value |
| 507 | if !ok { |
| 508 | continue |
| 509 | } |
| 510 | switch def.Kind() { |
| 511 | case KindBoolean: |
| 512 | ret[k] = v |
| 513 | case KindString: |
| 514 | ret[k] = v |
| 515 | case KindInt: |
| 516 | ret[k] = v |
| 517 | case KindNetwork: |
| 518 | vm, ok := v.(map[string]any) |
| 519 | if !ok { |
| 520 | return nil, fmt.Errorf("expected map") |
| 521 | } |
| 522 | name, ok := vm["name"] |
| 523 | if !ok { |
| 524 | return nil, fmt.Errorf("expected network name") |
| 525 | } |
| 526 | ret[k] = name |
| 527 | case KindAuth: |
| 528 | vm, ok := v.(map[string]any) |
| 529 | if !ok { |
| 530 | return nil, fmt.Errorf("expected map") |
| 531 | } |
| 532 | r, err := derivedToConfig(vm, AuthSchema) |
| 533 | if err != nil { |
| 534 | return nil, err |
| 535 | } |
| 536 | ret[k] = r |
| 537 | case KindSSHKey: |
| 538 | vm, ok := v.(map[string]any) |
| 539 | if !ok { |
| 540 | return nil, fmt.Errorf("expected map") |
| 541 | } |
| 542 | r, err := derivedToConfig(vm, SSHKeySchema) |
| 543 | if err != nil { |
| 544 | return nil, err |
| 545 | } |
| 546 | ret[k] = r |
| 547 | case KindStruct: |
| 548 | vm, ok := v.(map[string]any) |
| 549 | if !ok { |
| 550 | return nil, fmt.Errorf("expected map") |
| 551 | } |
| 552 | r, err := derivedToConfig(vm, def) |
| 553 | if err != nil { |
| 554 | return nil, err |
| 555 | } |
| 556 | ret[k] = r |
| 557 | default: |
| 558 | return nil, fmt.Errorf("Should not reach!") |
| 559 | } |
| 560 | } |
| 561 | return ret, nil |
| 562 | } |
| 563 | |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 564 | func findNetwork(networks []Network, name string) (Network, error) { |
| 565 | for _, n := range networks { |
| 566 | if n.Name == name { |
| 567 | return n, nil |
| 568 | } |
| 569 | } |
| 570 | return Network{}, fmt.Errorf("Network not found: %s", name) |
| 571 | } |
| 572 | |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 573 | type Network struct { |
| Giorgi Lekveishvili | 9b52ab9 | 2024-01-05 13:12:48 +0400 | [diff] [blame] | 574 | Name string `json:"name,omitempty"` |
| 575 | IngressClass string `json:"ingressClass,omitempty"` |
| 576 | CertificateIssuer string `json:"certificateIssuer,omitempty"` |
| 577 | Domain string `json:"domain,omitempty"` |
| Giorgi Lekveishvili | b59b7c2 | 2024-04-03 22:17:50 +0400 | [diff] [blame] | 578 | AllocatePortAddr string `json:"allocatePortAddr,omitempty"` |
| Giorgi Lekveishvili | 4257b90 | 2023-07-07 17:08:42 +0400 | [diff] [blame] | 579 | } |