blob: f55344d4b0ce63d3351d5cc17f5cdb551869e6e7 [file] [log] [blame]
package tasks
import (
"fmt"
"log"
"path/filepath"
"github.com/giolekva/pcloud/core/installer"
"github.com/giolekva/pcloud/core/installer/io"
"github.com/giolekva/pcloud/core/installer/soft"
)
func SetupConfigRepoTask(env installer.EnvConfig, st *state) Task {
ret := newSequentialParentTask(
"Configure Git repository",
true,
newSequentialParentTask(
"Start up Git server",
false,
NewCreateConfigRepoTask(env, st),
CreateGitClientTask(env, st),
),
NewInitConfigRepoTask(env, st),
NewActivateEnvTask(env, st),
newSequentialParentTask(
"Create initial commit",
false,
CreateRepoClient(env, st),
CommitEnvironmentConfiguration(env, st),
ConfigureFirstAccount(env, st),
),
)
ret.beforeStart = func() {
st.infoListener("dodo is driven by GitOps, changes are committed to the repository before updating an environment. This unlocks functionalities such as: rolling back to old working state, migrating dodo to new infrastructure (for example from Cloud to on-prem).")
}
return ret
}
func NewCreateConfigRepoTask(env installer.EnvConfig, st *state) Task {
t := newLeafTask("Install Git server", func() error {
appsRepo := installer.NewInMemoryAppRepository(installer.CreateAllApps())
app, err := installer.FindInfraApp(appsRepo, "config-repo")
if err != nil {
return err
}
adminKeys, err := installer.NewSSHKeyPair(fmt.Sprintf("%s-config-repo-admin-keys", env.Id))
if err != nil {
return err
}
st.ssAdminKeys = adminKeys
keys, err := installer.NewSSHKeyPair(fmt.Sprintf("%s-config-repo-keys", env.Id))
if err != nil {
return err
}
appDir := filepath.Join("/environments", env.Id, "config-repo")
_, err = st.infraAppManager.Install(app, appDir, env.Id, map[string]any{
"privateKey": string(keys.RawPrivateKey()),
"publicKey": string(keys.RawAuthorizedKey()),
"adminKey": string(adminKeys.RawAuthorizedKey()),
})
return err
})
return &t
}
func CreateGitClientTask(env installer.EnvConfig, st *state) Task {
t := newLeafTask("Wait git server to come up", func() error {
ssClient, err := st.repoClient.Get(
fmt.Sprintf("soft-serve.%s.svc.cluster.local:%d", env.Id, 22),
st.ssAdminKeys.RawPrivateKey(),
log.Default())
if err != nil {
return err
}
if err := ssClient.AddPublicKey("admin", env.AdminPublicKey); err != nil {
return err
}
// // TODO(gio): defer?
// // TODO(gio): remove at the end of final task cleanup
// if err := ssClient.RemovePublicKey("admin", string(ssAdminKeys.RawAuthorizedKey())); err != nil {
// t.callDoneListeners(err)
// return
// }
st.ssClient = ssClient
return nil
})
return &t
}
func NewInitConfigRepoTask(env installer.EnvConfig, st *state) Task {
t := newLeafTask("Configure access control lists", func() error {
st.fluxUserName = fmt.Sprintf("flux-%s", env.Id)
keys, err := installer.NewSSHKeyPair(st.fluxUserName)
if err != nil {
return err
}
st.keys = keys
if err := st.ssClient.AddRepository("config"); err != nil {
return err
}
repoIO, err := st.ssClient.GetRepo("config")
if err != nil {
return err
}
if err := repoIO.Do(func(r soft.RepoFS) (string, error) {
w, err := r.Writer("README.md")
if err != nil {
return "", err
}
defer w.Close()
if _, err := fmt.Fprintf(w, "# %s PCloud environment", env.Id); err != nil {
return "", err
}
if err := soft.WriteYaml(r, "kustomization.yaml", io.NewKustomization()); err != nil {
return "", err
}
return "init", nil
}); err != nil {
return err
}
if err := st.ssClient.AddUser(st.fluxUserName, keys.AuthorizedKey()); err != nil {
return err
}
if err := st.ssClient.AddReadOnlyCollaborator("config", st.fluxUserName); err != nil {
return err
}
return nil
})
return &t
}