| package installer |
| |
| import ( |
| "archive/tar" |
| "compress/gzip" |
| "embed" |
| "encoding/json" |
| "fmt" |
| htemplate "html/template" |
| "io" |
| "log" |
| "net/http" |
| "strings" |
| "text/template" |
| |
| "github.com/Masterminds/sprig/v3" |
| "github.com/go-git/go-billy/v5" |
| "sigs.k8s.io/yaml" |
| ) |
| |
| //go:embed values-tmpl |
| var valuesTmpls embed.FS |
| |
| type Named interface { |
| Nam() string |
| } |
| |
| type appConfig struct { |
| Name string `json:"name"` |
| Version string `json:"version"` |
| Description string `json:"description"` |
| Namespaces []string `json:"namespaces"` |
| Icon htemplate.HTML `json:"icon"` |
| } |
| |
| type App struct { |
| Name string |
| Namespaces []string |
| Templates []*template.Template |
| Schema string |
| Readme *template.Template |
| } |
| |
| func (a App) ConfigSchema() map[string]any { |
| ret := make(map[string]any) |
| if err := json.NewDecoder(strings.NewReader(a.Schema)).Decode(&ret); err != nil { |
| panic(err) // TODO(giolekva): prevalidate |
| } |
| return ret |
| } |
| |
| type StoreApp struct { |
| App |
| Icon htemplate.HTML |
| ShortDescription string |
| } |
| |
| func (a App) Nam() string { |
| return a.Name |
| } |
| |
| func (a StoreApp) Nam() string { |
| return a.Name |
| } |
| |
| type AppRepository[A Named] interface { |
| GetAll() ([]A, error) |
| Find(name string) (*A, error) |
| } |
| |
| type InMemoryAppRepository[A Named] struct { |
| apps []A |
| } |
| |
| func NewInMemoryAppRepository[A Named](apps []A) InMemoryAppRepository[A] { |
| return InMemoryAppRepository[A]{ |
| apps, |
| } |
| } |
| |
| func (r InMemoryAppRepository[A]) Find(name string) (*A, error) { |
| for _, a := range r.apps { |
| if a.Nam() == name { |
| return &a, nil |
| } |
| } |
| return nil, fmt.Errorf("Application not found: %s", name) |
| } |
| |
| func (r InMemoryAppRepository[A]) GetAll() ([]A, error) { |
| return r.apps, nil |
| } |
| |
| func CreateAllApps() []App { |
| tmpls, err := template.New("root").Funcs(template.FuncMap(sprig.FuncMap())).ParseFS(valuesTmpls, "values-tmpl/*") |
| if err != nil { |
| log.Fatal(err) |
| } |
| ret := []App{ |
| CreateAppIngressPrivate(valuesTmpls, tmpls), |
| CreateCertificateIssuerPublic(valuesTmpls, tmpls), |
| CreateCertificateIssuerPrivate(valuesTmpls, tmpls), |
| CreateAppCoreAuth(valuesTmpls, tmpls), |
| CreateAppHeadscale(valuesTmpls, tmpls), |
| CreateAppHeadscaleUser(valuesTmpls, tmpls), |
| CreateMetallbIPAddressPool(valuesTmpls, tmpls), |
| CreateEnvManager(valuesTmpls, tmpls), |
| CreateWelcome(valuesTmpls, tmpls), |
| CreateAppManager(valuesTmpls, tmpls), |
| CreateIngressPublic(valuesTmpls, tmpls), |
| CreateCertManager(valuesTmpls, tmpls), |
| CreateCertManagerWebhookGandi(valuesTmpls, tmpls), |
| CreateCSIDriverSMB(valuesTmpls, tmpls), |
| CreateResourceRendererController(valuesTmpls, tmpls), |
| CreateHeadscaleController(valuesTmpls, tmpls), |
| CreateDNSZoneManager(valuesTmpls, tmpls), |
| } |
| for _, a := range CreateStoreApps() { |
| ret = append(ret, a.App) |
| } |
| return ret |
| } |
| |
| func CreateStoreApps() []StoreApp { |
| tmpls, err := template.New("root").Funcs(template.FuncMap(sprig.FuncMap())).ParseFS(valuesTmpls, "values-tmpl/*") |
| if err != nil { |
| log.Fatal(err) |
| } |
| return []StoreApp{ |
| CreateAppVaultwarden(valuesTmpls, tmpls), |
| CreateAppMatrix(valuesTmpls, tmpls), |
| CreateAppPihole(valuesTmpls, tmpls), |
| CreateAppMaddy(valuesTmpls, tmpls), |
| CreateAppQBittorrent(valuesTmpls, tmpls), |
| CreateAppJellyfin(valuesTmpls, tmpls), |
| CreateAppSoftServe(valuesTmpls, tmpls), |
| CreateAppRpuppy(valuesTmpls, tmpls), |
| } |
| } |
| |
| // TODO(gio): service account needs permission to create/update secret |
| func CreateAppIngressPrivate(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/private-network.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "private-network", |
| []string{"ingress-private"}, // TODO(gio): rename to private network |
| []*template.Template{ |
| tmpls.Lookup("ingress-private.yaml"), |
| tmpls.Lookup("tailscale-proxy.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("private-network.md"), |
| } |
| } |
| |
| func CreateCertificateIssuerPrivate(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/certificate-issuer-private.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "certificate-issuer-private", |
| []string{}, |
| []*template.Template{ |
| tmpls.Lookup("certificate-issuer-private.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("certificate-issuer-private.md"), |
| } |
| } |
| |
| func CreateCertificateIssuerPublic(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/certificate-issuer-public.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "certificate-issuer-public", |
| []string{}, |
| []*template.Template{ |
| tmpls.Lookup("certificate-issuer-public.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("certificate-issuer-public.md"), |
| } |
| } |
| |
| func CreateAppCoreAuth(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/core-auth.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "core-auth", |
| []string{"core-auth"}, |
| []*template.Template{ |
| tmpls.Lookup("core-auth-storage.yaml"), |
| tmpls.Lookup("core-auth.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("core-auth.md"), |
| } |
| } |
| |
| func CreateAppVaultwarden(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/vaultwarden.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App: App{ |
| "vaultwarden", |
| []string{"app-vaultwarden"}, |
| []*template.Template{ |
| tmpls.Lookup("vaultwarden.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("vaultwarden.md"), |
| }, |
| Icon: `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 48 48"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" d="M35.38 25.63V9.37H24v28.87a34.93 34.93 0 0 0 5.41-3.48q6-4.66 6-9.14Zm4.87-19.5v19.5A11.58 11.58 0 0 1 39.4 30a16.22 16.22 0 0 1-2.11 3.81a23.52 23.52 0 0 1-3 3.24a34.87 34.87 0 0 1-3.22 2.62c-1 .69-2 1.35-3.07 2s-1.82 1-2.27 1.26l-1.08.51a1.53 1.53 0 0 1-1.32 0l-1.08-.51c-.45-.22-1.21-.64-2.27-1.26s-2.09-1.27-3.07-2A34.87 34.87 0 0 1 13.7 37a23.52 23.52 0 0 1-3-3.24A16.22 16.22 0 0 1 8.6 30a11.58 11.58 0 0 1-.85-4.32V6.13A1.64 1.64 0 0 1 9.38 4.5h29.24a1.64 1.64 0 0 1 1.63 1.63Z"/></svg>`, |
| ShortDescription: "Open source implementation of Bitwarden password manager. Can be used with official client applications.", |
| } |
| } |
| |
| func CreateAppMatrix(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/matrix.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "matrix", |
| []string{"app-matrix"}, |
| []*template.Template{ |
| tmpls.Lookup("matrix-storage.yaml"), |
| tmpls.Lookup("matrix.yaml"), |
| }, |
| string(schema), |
| nil, |
| }, |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 24 24"><path fill="currentColor" d="M.632.55v22.9H2.28V24H0V0h2.28v.55zm7.043 7.26v1.157h.033a3.312 3.312 0 0 1 1.117-1.024c.433-.245.936-.365 1.5-.365c.54 0 1.033.107 1.481.314c.448.208.785.582 1.02 1.108c.254-.374.6-.706 1.034-.992c.434-.287.95-.43 1.546-.43c.453 0 .872.056 1.26.167c.388.11.716.286.993.53c.276.245.489.559.646.951c.152.392.23.863.23 1.417v5.728h-2.349V11.52c0-.286-.01-.559-.032-.812a1.755 1.755 0 0 0-.18-.66a1.106 1.106 0 0 0-.438-.448c-.194-.11-.457-.166-.785-.166c-.332 0-.6.064-.803.189a1.38 1.38 0 0 0-.48.499a1.946 1.946 0 0 0-.231.696a5.56 5.56 0 0 0-.06.785v4.768h-2.35v-4.8c0-.254-.004-.503-.018-.752a2.074 2.074 0 0 0-.143-.688a1.052 1.052 0 0 0-.415-.503c-.194-.125-.476-.19-.854-.19c-.111 0-.259.024-.439.074c-.18.051-.36.143-.53.282a1.637 1.637 0 0 0-.439.595c-.12.259-.18.6-.18 1.02v4.966H5.46V7.81zm15.693 15.64V.55H21.72V0H24v24h-2.28v-.55z"/></svg>`, |
| "An open network for secure, decentralised communication", |
| } |
| } |
| |
| func CreateAppPihole(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/pihole.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "pihole", |
| []string{"app-pihole"}, |
| []*template.Template{ |
| tmpls.Lookup("pihole.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("pihole.md"), |
| }, |
| // "simple-icons:pihole", |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 24 24"><path fill="currentColor" d="M4.344 0c.238 4.792 3.256 7.056 6.252 7.376c.165-1.692-4.319-5.6-4.319-5.6c-.008-.011.009-.025.019-.014c0 0 4.648 4.01 5.423 5.645c2.762-.15 5.196-1.947 5-4.912c0 0-4.12-.613-5 4.618C11.48 2.753 8.993 0 4.344 0zM12 7.682v.002a3.68 3.68 0 0 0-2.591 1.077L4.94 13.227a3.683 3.683 0 0 0-.86 1.356a3.31 3.31 0 0 0-.237 1.255A3.681 3.681 0 0 0 4.92 18.45l4.464 4.466a3.69 3.69 0 0 0 2.251 1.06l.002.001c.093.01.187.015.28.017l-.1-.008c.06.003.117.009.177.009l-.077-.001L12 24l-.004-.005a3.68 3.68 0 0 0 2.61-1.077l4.469-4.465a3.683 3.683 0 0 0 1.006-1.888l.012-.063a3.682 3.682 0 0 0 .057-.541l.003-.061c0-.017.003-.05.004-.06h-.002a3.683 3.683 0 0 0-1.077-2.607l-4.466-4.468a3.694 3.694 0 0 0-1.564-.927l-.07-.02a3.43 3.43 0 0 0-.946-.133L12 7.682zm3.165 3.357c.023 1.748-1.33 3.078-1.33 4.806c.164 2.227 1.733 3.207 3.266 3.146c-.035.003-.068.007-.104.009c-1.847.135-3.209-1.326-5.002-1.326c-2.23.164-3.21 1.736-3.147 3.27l-.008-.104c-.133-1.847 1.328-3.21 1.328-5.002c-.173-2.32-1.867-3.284-3.46-3.132c.1-.011.203-.021.31-.027c1.847-.133 3.209 1.328 5.002 1.328c2.082-.155 3.074-1.536 3.145-2.968zM4.344 0c.238 4.792 3.256 7.056 6.252 7.376c.165-1.692-4.319-5.6-4.319-5.6c-.008-.011.009-.025.019-.014c0 0 4.648 4.01 5.423 5.645c2.762-.15 5.196-1.947 5-4.912c0 0-4.12-.613-5 4.618C11.48 2.753 8.993 0 4.344 0zM12 7.682v.002a3.68 3.68 0 0 0-2.591 1.077L4.94 13.227a3.683 3.683 0 0 0-.86 1.356a3.31 3.31 0 0 0-.237 1.255A3.681 3.681 0 0 0 4.92 18.45l4.464 4.466a3.69 3.69 0 0 0 2.251 1.06l.002.001c.093.01.187.015.28.017l-.1-.008c.06.003.117.009.177.009l-.077-.001L12 24l-.004-.005a3.68 3.68 0 0 0 2.61-1.077l4.469-4.465a3.683 3.683 0 0 0 1.006-1.888l.012-.063a3.682 3.682 0 0 0 .057-.541l.003-.061c0-.017.003-.05.004-.06h-.002a3.683 3.683 0 0 0-1.077-2.607l-4.466-4.468a3.694 3.694 0 0 0-1.564-.927l-.07-.02a3.43 3.43 0 0 0-.946-.133L12 7.682zm3.165 3.357c.023 1.748-1.33 3.078-1.33 4.806c.164 2.227 1.733 3.207 3.266 3.146c-.035.003-.068.007-.104.009c-1.847.135-3.209-1.326-5.002-1.326c-2.23.164-3.21 1.736-3.147 3.27l-.008-.104c-.133-1.847 1.328-3.21 1.328-5.002c-.173-2.32-1.867-3.284-3.46-3.132c.1-.011.203-.021.31-.027c1.847-.133 3.209 1.328 5.002 1.328c2.082-.155 3.074-1.536 3.145-2.968z"/></svg>`, |
| "Pi-hole is a Linux network-level advertisement and Internet tracker blocking application which acts as a DNS sinkhole and optionally a DHCP server, intended for use on a private network.", |
| } |
| } |
| |
| func CreateAppMaddy(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/maddy.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "maddy", |
| []string{"app-maddy"}, |
| []*template.Template{ |
| tmpls.Lookup("maddy.yaml"), |
| }, |
| string(schema), |
| nil, |
| }, |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 48 48"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" d="M9.5 13c13.687 13.574 14.825 13.09 29 0"/><rect width="37" height="31" x="5.5" y="8.5" fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" rx="2"/></svg>`, |
| "SMPT/IMAP server to communicate via email.", |
| } |
| } |
| |
| func CreateAppQBittorrent(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/qbittorrent.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "qbittorrent", |
| []string{"app-qbittorrent"}, |
| []*template.Template{ |
| tmpls.Lookup("qbittorrent.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("qbittorrent.md"), |
| }, |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 48 48"><circle cx="24" cy="24" r="21.5" fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"/><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" d="M26.651 22.364a5.034 5.034 0 0 1 5.035-5.035h0a5.034 5.034 0 0 1 5.034 5.035v3.272a5.034 5.034 0 0 1-5.034 5.035h0a5.034 5.034 0 0 1-5.035-5.035m0 5.035V10.533m-5.302 15.103a5.034 5.034 0 0 1-5.035 5.035h0a5.034 5.034 0 0 1-5.034-5.035v-3.272a5.034 5.034 0 0 1 5.034-5.035h0a5.034 5.034 0 0 1 5.035 5.035m0-5.035v20.138"/></svg>`, |
| "qBittorrent is a cross-platform free and open-source BitTorrent client written in native C++. It relies on Boost, Qt 6 toolkit and the libtorrent-rasterbar library, with an optional search engine written in Python.", |
| } |
| } |
| |
| func CreateAppJellyfin(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/jellyfin.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "jellyfin", |
| []string{"app-jellyfin"}, |
| []*template.Template{ |
| tmpls.Lookup("jellyfin.yaml"), |
| }, |
| string(schema), |
| nil, |
| }, |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 48 48"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" d="M24 20c-1.62 0-6.85 9.48-6.06 11.08s11.33 1.59 12.12 0S25.63 20 24 20Z"/><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" d="M24 5.5c-4.89 0-20.66 28.58-18.25 33.4s34.13 4.77 36.51 0S28.9 5.5 24 5.5Zm12 29.21c-1.56 3.13-22.35 3.17-23.93 0S20.8 12.83 24 12.83s13.52 18.76 12 21.88Z"/></svg>`, |
| "Jellyfin is a free and open-source media server and suite of multimedia applications designed to organize, manage, and share digital media files to networked devices.", |
| } |
| } |
| |
| func CreateAppRpuppy(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/rpuppy.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "rpuppy", |
| []string{"app-rpuppy"}, |
| []*template.Template{ |
| tmpls.Lookup("rpuppy.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("rpuppy.md"), |
| }, |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 256 256"><path fill="currentColor" d="M100 140a8 8 0 1 1-8-8a8 8 0 0 1 8 8Zm64 8a8 8 0 1 0-8-8a8 8 0 0 0 8 8Zm64.94-9.11a12.12 12.12 0 0 1-5 1.11a11.83 11.83 0 0 1-9.35-4.62l-2.59-3.29V184a36 36 0 0 1-36 36H80a36 36 0 0 1-36-36v-51.91l-2.53 3.27A11.88 11.88 0 0 1 32.1 140a12.08 12.08 0 0 1-5-1.11a11.82 11.82 0 0 1-6.84-13.14l16.42-88a12 12 0 0 1 14.7-9.43h.16L104.58 44h46.84l53.08-15.6h.16a12 12 0 0 1 14.7 9.43l16.42 88a11.81 11.81 0 0 1-6.84 13.06ZM97.25 50.18L49.34 36.1a4.18 4.18 0 0 0-.92-.1a4 4 0 0 0-3.92 3.26l-16.42 88a4 4 0 0 0 7.08 3.22ZM204 121.75L150 52h-44l-54 69.75V184a28 28 0 0 0 28 28h44v-18.34l-14.83-14.83a4 4 0 0 1 5.66-5.66L128 186.34l13.17-13.17a4 4 0 0 1 5.66 5.66L132 193.66V212h44a28 28 0 0 0 28-28Zm23.92 5.48l-16.42-88a4 4 0 0 0-4.84-3.16l-47.91 14.11l62.11 80.28a4 4 0 0 0 7.06-3.23Z"/></svg>`, |
| "Delights users with randomly generate puppy pictures. Can be configured to be reachable only from private network or publicly.", |
| } |
| } |
| |
| func CreateAppSoftServe(fs embed.FS, tmpls *template.Template) StoreApp { |
| schema, err := fs.ReadFile("values-tmpl/soft-serve.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return StoreApp{ |
| App{ |
| "soft-serve", |
| []string{"app-soft-serve"}, |
| []*template.Template{ |
| tmpls.Lookup("soft-serve.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("soft-serve.md"), |
| }, |
| `<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 48 48"><g fill="none" stroke="currentColor" stroke-linecap="round" stroke-width="4"><path stroke-linejoin="round" d="M15.34 22.5L21 37l3 6l3-6l5.66-14.5"/><path d="M19 32h10"/><path stroke-linejoin="round" d="M24 3c-6 0-8 6-8 6s-6 2-6 7s5 7 5 7s3.5-2 9-2s9 2 9 2s5-2 5-7s-6-7-6-7s-2-6-8-6Z"/></g></svg>`, |
| "A tasty, self-hostable Git server for the command line. 🍦", |
| } |
| } |
| |
| func CreateAppHeadscale(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/headscale.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "headscale", |
| []string{"app-headscale"}, |
| []*template.Template{ |
| tmpls.Lookup("headscale.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("headscale.md"), |
| } |
| } |
| |
| func CreateAppHeadscaleUser(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/headscale-user.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "headscale-user", |
| []string{"app-headscale"}, |
| []*template.Template{ |
| tmpls.Lookup("headscale-user.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("headscale-user.md"), |
| } |
| } |
| |
| func CreateMetallbIPAddressPool(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/metallb-ipaddresspool.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "metallb-ipaddresspool", |
| []string{"metallb-ipaddresspool"}, |
| []*template.Template{ |
| tmpls.Lookup("metallb-ipaddresspool.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("metallb-ipaddresspool.md"), |
| } |
| } |
| |
| func CreateEnvManager(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/env-manager.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "env-manager", |
| []string{"env-manager"}, |
| []*template.Template{ |
| tmpls.Lookup("env-manager.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("env-manager.md"), |
| } |
| } |
| |
| func CreateWelcome(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/welcome.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "welcome", |
| []string{"app-welcome"}, |
| []*template.Template{ |
| tmpls.Lookup("welcome.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("welcome.md"), |
| } |
| } |
| |
| func CreateAppManager(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/appmanager.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "app-manager", |
| []string{"core-appmanager"}, |
| []*template.Template{ |
| tmpls.Lookup("appmanager.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("appmanager.md"), |
| } |
| } |
| |
| func CreateIngressPublic(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/ingress-public.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "ingress-public", |
| []string{"ingress-public"}, |
| []*template.Template{ |
| tmpls.Lookup("ingress-public.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("ingress-public.md"), |
| } |
| } |
| |
| func CreateCertManager(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/cert-manager.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "cert-manager", |
| []string{"cert-manager"}, |
| []*template.Template{ |
| tmpls.Lookup("cert-manager.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("cert-manager.md"), |
| } |
| } |
| |
| func CreateCertManagerWebhookGandi(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/cert-manager-webhook-pcloud.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "cert-manager-webhook-pcloud", |
| []string{}, |
| []*template.Template{ |
| tmpls.Lookup("cert-manager-webhook-pcloud.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("cert-manager-webhook-pcloud.md"), |
| } |
| } |
| |
| func CreateCSIDriverSMB(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/csi-driver-smb.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "csi-driver-smb", |
| []string{"csi-driver-smb"}, |
| []*template.Template{ |
| tmpls.Lookup("csi-driver-smb.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("csi-driver-smb.md"), |
| } |
| } |
| |
| func CreateResourceRendererController(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/resource-renderer-controller.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "resource-renderer-controller", |
| []string{"rr-controller"}, |
| []*template.Template{ |
| tmpls.Lookup("resource-renderer-controller.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("resource-renderer-controller.md"), |
| } |
| } |
| |
| func CreateHeadscaleController(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/headscale-controller.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "headscale-controller", |
| []string{"headscale-controller"}, |
| []*template.Template{ |
| tmpls.Lookup("headscale-controller.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("headscale-controller.md"), |
| } |
| } |
| |
| func CreateDNSZoneManager(fs embed.FS, tmpls *template.Template) App { |
| schema, err := fs.ReadFile("values-tmpl/dns-zone-controller.jsonschema") |
| if err != nil { |
| panic(err) |
| } |
| return App{ |
| "dns-zone-manager", |
| []string{"dns-zone-manager"}, |
| []*template.Template{ |
| tmpls.Lookup("dns-zone-storage.yaml"), |
| tmpls.Lookup("coredns.yaml"), |
| tmpls.Lookup("dns-zone-controller.yaml"), |
| }, |
| string(schema), |
| tmpls.Lookup("dns-zone-controller.md"), |
| } |
| } |
| |
| type httpAppRepository struct { |
| apps []StoreApp |
| } |
| |
| type appVersion struct { |
| Version string `json:"version"` |
| Urls []string `json:"urls"` |
| } |
| |
| type allAppsResp struct { |
| ApiVersion string `json:"apiVersion"` |
| Entries map[string][]appVersion `json:"entries"` |
| } |
| |
| func FetchAppsFromHTTPRepository(addr string, fs billy.Filesystem) error { |
| resp, err := http.Get(addr) |
| if err != nil { |
| return err |
| } |
| b, err := io.ReadAll(resp.Body) |
| if err != nil { |
| return err |
| } |
| var apps allAppsResp |
| if err := yaml.Unmarshal(b, &apps); err != nil { |
| return err |
| } |
| for name, conf := range apps.Entries { |
| for _, version := range conf { |
| resp, err := http.Get(version.Urls[0]) |
| if err != nil { |
| return err |
| } |
| nameVersion := fmt.Sprintf("%s-%s", name, version.Version) |
| if err := fs.MkdirAll(nameVersion, 0700); err != nil { |
| return err |
| } |
| sub, err := fs.Chroot(nameVersion) |
| if err != nil { |
| return err |
| } |
| if err := extractApp(resp.Body, sub); err != nil { |
| return err |
| } |
| } |
| } |
| return nil |
| } |
| |
| func extractApp(archive io.Reader, fs billy.Filesystem) error { |
| uncompressed, err := gzip.NewReader(archive) |
| if err != nil { |
| return err |
| } |
| tarReader := tar.NewReader(uncompressed) |
| for true { |
| header, err := tarReader.Next() |
| if err == io.EOF { |
| break |
| } |
| if err != nil { |
| return err |
| } |
| switch header.Typeflag { |
| case tar.TypeDir: |
| if err := fs.MkdirAll(header.Name, 0755); err != nil { |
| return err |
| } |
| case tar.TypeReg: |
| out, err := fs.Create(header.Name) |
| if err != nil { |
| return err |
| } |
| defer out.Close() |
| if _, err := io.Copy(out, tarReader); err != nil { |
| return err |
| } |
| default: |
| return fmt.Errorf("Uknown type: %s", header.Name) |
| } |
| } |
| return nil |
| } |
| |
| type fsAppRepository struct { |
| InMemoryAppRepository[StoreApp] |
| fs billy.Filesystem |
| } |
| |
| func NewFSAppRepository(fs billy.Filesystem) (AppRepository[StoreApp], error) { |
| all, err := fs.ReadDir(".") |
| if err != nil { |
| return nil, err |
| } |
| apps := make([]StoreApp, 0) |
| for _, e := range all { |
| if !e.IsDir() { |
| continue |
| } |
| appFS, err := fs.Chroot(e.Name()) |
| if err != nil { |
| return nil, err |
| } |
| app, err := loadApp(appFS) |
| if err != nil { |
| log.Printf("Ignoring directory %s: %s", e.Name(), err) |
| continue |
| } |
| apps = append(apps, app) |
| } |
| return &fsAppRepository{ |
| NewInMemoryAppRepository[StoreApp](apps), |
| fs, |
| }, nil |
| } |
| |
| func loadApp(fs billy.Filesystem) (StoreApp, error) { |
| cfg, err := fs.Open("Chart.yaml") |
| if err != nil { |
| return StoreApp{}, err |
| } |
| defer cfg.Close() |
| b, err := io.ReadAll(cfg) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| var appCfg appConfig |
| if err := yaml.Unmarshal(b, &appCfg); err != nil { |
| return StoreApp{}, err |
| } |
| rb, err := fs.Open("README.md") |
| if err != nil { |
| return StoreApp{}, err |
| } |
| defer rb.Close() |
| readme, err := io.ReadAll(rb) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| readmeTmpl, err := template.New("README.md").Parse(string(readme)) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| sb, err := fs.Open("schema.json") |
| if err != nil { |
| return StoreApp{}, err |
| } |
| defer sb.Close() |
| schema, err := io.ReadAll(sb) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| tFiles, err := fs.ReadDir("templates") |
| if err != nil { |
| return StoreApp{}, err |
| } |
| tmpls := make([]*template.Template, 0) |
| for _, t := range tFiles { |
| if !strings.HasSuffix(t.Name(), ".yaml") { |
| continue |
| } |
| inp, err := fs.Open(fs.Join("templates", t.Name())) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| b, err := io.ReadAll(inp) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| tmpl, err := template.New(t.Name()).Parse(string(b)) |
| if err != nil { |
| return StoreApp{}, err |
| } |
| tmpls = append(tmpls, tmpl) |
| } |
| return StoreApp{ |
| App: App{ |
| Name: appCfg.Name, |
| Readme: readmeTmpl, |
| Schema: string(schema), |
| Namespaces: appCfg.Namespaces, |
| Templates: tmpls, |
| }, |
| ShortDescription: appCfg.Description, |
| Icon: appCfg.Icon, |
| }, nil |
| } |