| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 4 | "bytes" |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 5 | "database/sql" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 6 | "encoding/json" |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 7 | "fmt" |
| 8 | "io" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 9 | "log" |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 10 | "net" |
| 11 | "net/http" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 12 | "os" |
| 13 | |
| 14 | "github.com/giolekva/pcloud/core/installer" |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 15 | dodoapp "github.com/giolekva/pcloud/core/installer/server/dodo-app" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 16 | "github.com/giolekva/pcloud/core/installer/soft" |
| gio | 43b0f42 | 2024-08-21 10:40:13 +0400 | [diff] [blame] | 17 | "github.com/giolekva/pcloud/core/installer/tasks" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 18 | |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 19 | _ "github.com/ncruces/go-sqlite3" |
| 20 | _ "github.com/ncruces/go-sqlite3/driver" |
| 21 | _ "github.com/ncruces/go-sqlite3/embed" |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 22 | "github.com/spf13/cobra" |
| 23 | ) |
| 24 | |
| 25 | var dodoAppFlags struct { |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 26 | external bool |
| gio | cb34ad2 | 2024-07-11 08:01:13 +0400 | [diff] [blame] | 27 | port int |
| 28 | apiPort int |
| 29 | sshKey string |
| 30 | repoAddr string |
| 31 | self string |
| gio | c81a847 | 2024-09-24 13:06:19 +0200 | [diff] [blame] | 32 | selfPublic string |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 33 | repoPublicAddr string |
| gio | cb34ad2 | 2024-07-11 08:01:13 +0400 | [diff] [blame] | 34 | namespace string |
| 35 | envAppManagerAddr string |
| 36 | envConfig string |
| gio | cb34ad2 | 2024-07-11 08:01:13 +0400 | [diff] [blame] | 37 | gitRepoPublicKey string |
| 38 | db string |
| gio | 23bdc1b | 2024-07-11 16:07:47 +0400 | [diff] [blame] | 39 | networks []string |
| gio | cafd4e6 | 2024-07-31 10:53:40 +0400 | [diff] [blame] | 40 | fetchUsersAddr string |
| gio | 36b23b3 | 2024-08-25 12:20:54 +0400 | [diff] [blame] | 41 | headscaleAPIAddr string |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 42 | } |
| 43 | |
| 44 | func dodoAppCmd() *cobra.Command { |
| 45 | cmd := &cobra.Command{ |
| 46 | Use: "dodo-app", |
| 47 | RunE: dodoAppCmdRun, |
| 48 | } |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 49 | cmd.Flags().BoolVar( |
| 50 | &dodoAppFlags.external, |
| 51 | "external", |
| 52 | false, |
| 53 | "", |
| 54 | ) |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 55 | cmd.Flags().IntVar( |
| 56 | &dodoAppFlags.port, |
| 57 | "port", |
| 58 | 8080, |
| 59 | "", |
| 60 | ) |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 61 | cmd.Flags().IntVar( |
| 62 | &dodoAppFlags.apiPort, |
| 63 | "api-port", |
| 64 | 8081, |
| 65 | "", |
| 66 | ) |
| 67 | cmd.Flags().StringVar( |
| 68 | &dodoAppFlags.db, |
| 69 | "db", |
| 70 | "", |
| 71 | "", |
| 72 | ) |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 73 | cmd.Flags().StringVar( |
| 74 | &dodoAppFlags.repoAddr, |
| 75 | "repo-addr", |
| 76 | "", |
| 77 | "", |
| 78 | ) |
| 79 | cmd.Flags().StringVar( |
| 80 | &dodoAppFlags.sshKey, |
| 81 | "ssh-key", |
| 82 | "", |
| 83 | "", |
| 84 | ) |
| 85 | cmd.Flags().StringVar( |
| 86 | &dodoAppFlags.self, |
| 87 | "self", |
| 88 | "", |
| 89 | "", |
| 90 | ) |
| 91 | cmd.Flags().StringVar( |
| gio | c81a847 | 2024-09-24 13:06:19 +0200 | [diff] [blame] | 92 | &dodoAppFlags.selfPublic, |
| 93 | "self-public", |
| 94 | "", |
| 95 | "", |
| 96 | ) |
| 97 | cmd.Flags().StringVar( |
| gio | cafd4e6 | 2024-07-31 10:53:40 +0400 | [diff] [blame] | 98 | &dodoAppFlags.fetchUsersAddr, |
| 99 | "fetch-users-addr", |
| 100 | "", |
| 101 | "", |
| 102 | ) |
| 103 | cmd.Flags().StringVar( |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 104 | &dodoAppFlags.repoPublicAddr, |
| 105 | "repo-public-addr", |
| 106 | "", |
| 107 | "", |
| 108 | ) |
| 109 | cmd.Flags().StringVar( |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 110 | &dodoAppFlags.namespace, |
| 111 | "namespace", |
| 112 | "", |
| 113 | "", |
| 114 | ) |
| 115 | cmd.Flags().StringVar( |
| gio | cb34ad2 | 2024-07-11 08:01:13 +0400 | [diff] [blame] | 116 | &dodoAppFlags.envAppManagerAddr, |
| 117 | "env-app-manager-addr", |
| 118 | "", |
| 119 | "", |
| 120 | ) |
| 121 | cmd.Flags().StringVar( |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 122 | &dodoAppFlags.envConfig, |
| 123 | "env-config", |
| 124 | "", |
| 125 | "", |
| 126 | ) |
| gio | 266c04f | 2024-07-03 14:18:45 +0400 | [diff] [blame] | 127 | cmd.Flags().StringVar( |
| gio | 266c04f | 2024-07-03 14:18:45 +0400 | [diff] [blame] | 128 | &dodoAppFlags.gitRepoPublicKey, |
| 129 | "git-repo-public-key", |
| 130 | "", |
| 131 | "", |
| 132 | ) |
| gio | 23bdc1b | 2024-07-11 16:07:47 +0400 | [diff] [blame] | 133 | cmd.Flags().StringSliceVar( |
| 134 | &dodoAppFlags.networks, |
| 135 | "networks", |
| 136 | []string{}, |
| 137 | "", |
| 138 | ) |
| gio | 36b23b3 | 2024-08-25 12:20:54 +0400 | [diff] [blame] | 139 | cmd.Flags().StringVar( |
| 140 | &dodoAppFlags.headscaleAPIAddr, |
| 141 | "headscale-api-addr", |
| 142 | "", |
| 143 | "", |
| 144 | ) |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 145 | return cmd |
| 146 | } |
| 147 | |
| 148 | func dodoAppCmdRun(cmd *cobra.Command, args []string) error { |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 149 | sshKey, err := os.ReadFile(dodoAppFlags.sshKey) |
| 150 | if err != nil { |
| 151 | return err |
| 152 | } |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 153 | envConfig, err := os.Open(dodoAppFlags.envConfig) |
| 154 | if err != nil { |
| 155 | return err |
| 156 | } |
| 157 | defer envConfig.Close() |
| 158 | var env installer.EnvConfig |
| 159 | if err := json.NewDecoder(envConfig).Decode(&env); err != nil { |
| 160 | return err |
| 161 | } |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 162 | cg := soft.RealClientGetter{} |
| gio | 0710c6b | 2024-11-04 19:08:30 +0400 | [diff] [blame] | 163 | log := log.Default() |
| 164 | log.SetOutput(io.Discard) |
| 165 | softClient, err := cg.Get(dodoAppFlags.repoAddr, sshKey, log) |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 166 | if err != nil { |
| 167 | return err |
| 168 | } |
| gio | f884341 | 2024-05-22 16:38:05 +0400 | [diff] [blame] | 169 | jc, err := newJobCreator() |
| 170 | if err != nil { |
| 171 | return err |
| 172 | } |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 173 | nsc, err := newNSCreator() |
| 174 | if err != nil { |
| gio | 266c04f | 2024-07-03 14:18:45 +0400 | [diff] [blame] | 175 | return err |
| 176 | } |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 177 | if ok, err := softClient.RepoExists(dodoapp.ConfigRepoName); err != nil { |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 178 | return err |
| 179 | } else if !ok { |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 180 | if err := softClient.AddRepository(dodoapp.ConfigRepoName); err != nil { |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 181 | return err |
| 182 | } |
| 183 | } |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 184 | configRepo, err := softClient.GetRepo(dodoapp.ConfigRepoName) |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 185 | if err != nil { |
| 186 | return err |
| 187 | } |
| 188 | db, err := sql.Open("sqlite3", dodoAppFlags.db) |
| 189 | if err != nil { |
| 190 | return err |
| 191 | } |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 192 | st, err := dodoapp.NewStore(configRepo, db) |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 193 | if err != nil { |
| 194 | return err |
| 195 | } |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 196 | var nf dodoapp.NetworkFilter |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 197 | if len(dodoAppFlags.networks) == 0 { |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 198 | nf = dodoapp.NewNoNetworkFilter() |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 199 | } else { |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 200 | nf = dodoapp.NewAllowListFilter(dodoAppFlags.networks) |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 201 | } |
| 202 | if dodoAppFlags.external { |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 203 | nf = dodoapp.NewCombinedFilter(dodoapp.NewNetworkFilterByOwner(st), nf) |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 204 | } |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 205 | var ug dodoapp.UserGetter |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 206 | if dodoAppFlags.external { |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 207 | ug = dodoapp.NewExternalUserGetter() |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 208 | } else { |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 209 | ug = dodoapp.NewInternalUserGetter() |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 210 | } |
| gio | 43b0f42 | 2024-08-21 10:40:13 +0400 | [diff] [blame] | 211 | reconciler := &tasks.SequentialReconciler{ |
| 212 | []tasks.Reconciler{ |
| 213 | &tasks.SourceGitReconciler{}, |
| 214 | // &tasks.KustomizationReconciler{}, |
| 215 | }, |
| 216 | } |
| gio | 36b23b3 | 2024-08-25 12:20:54 +0400 | [diff] [blame] | 217 | vpnKeyGen := installer.NewHeadscaleAPIClient(dodoAppFlags.headscaleAPIAddr) |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 218 | cnc := &proxyConfigurator{dodoAppFlags.envAppManagerAddr} |
| gio | 5994628 | 2024-10-07 12:55:51 +0400 | [diff] [blame] | 219 | s, err := dodoapp.NewServer( |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 220 | st, |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 221 | nf, |
| 222 | ug, |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 223 | dodoAppFlags.port, |
| gio | a60f0de | 2024-07-08 10:49:48 +0400 | [diff] [blame] | 224 | dodoAppFlags.apiPort, |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 225 | dodoAppFlags.self, |
| gio | c81a847 | 2024-09-24 13:06:19 +0200 | [diff] [blame] | 226 | dodoAppFlags.selfPublic, |
| gio | 11617ac | 2024-07-15 16:09:04 +0400 | [diff] [blame] | 227 | dodoAppFlags.repoPublicAddr, |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 228 | string(sshKey), |
| 229 | dodoAppFlags.gitRepoPublicKey, |
| 230 | softClient, |
| 231 | dodoAppFlags.namespace, |
| gio | cb34ad2 | 2024-07-11 08:01:13 +0400 | [diff] [blame] | 232 | dodoAppFlags.envAppManagerAddr, |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 233 | nsc, |
| 234 | jc, |
| gio | 36b23b3 | 2024-08-25 12:20:54 +0400 | [diff] [blame] | 235 | vpnKeyGen, |
| gio | f6ad298 | 2024-08-23 17:42:49 +0400 | [diff] [blame] | 236 | cnc, |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 237 | env, |
| gio | cafd4e6 | 2024-07-31 10:53:40 +0400 | [diff] [blame] | 238 | dodoAppFlags.external, |
| 239 | dodoAppFlags.fetchUsersAddr, |
| gio | 43b0f42 | 2024-08-21 10:40:13 +0400 | [diff] [blame] | 240 | reconciler, |
| gio | 3305976 | 2024-07-05 13:19:07 +0400 | [diff] [blame] | 241 | ) |
| gio | 9d66f32 | 2024-07-06 13:45:10 +0400 | [diff] [blame] | 242 | if err != nil { |
| 243 | return err |
| 244 | } |
| gio | 0eaf271 | 2024-04-14 13:08:46 +0400 | [diff] [blame] | 245 | return s.Start() |
| 246 | } |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 247 | |
| gio | 721c004 | 2025-04-03 11:56:36 +0400 | [diff] [blame] | 248 | // TODO(gio): this should be removed, all of app installtions including dodo app |
| 249 | // must be done directly by main app manager. |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 250 | type proxyConfigurator struct { |
| 251 | apiAddr string |
| 252 | } |
| 253 | |
| 254 | func (pc *proxyConfigurator) AddCluster(name string, ingressIP net.IP) error { |
| 255 | return fmt.Errorf("NOT IMPLEMENTED") |
| 256 | } |
| 257 | |
| 258 | func (pc *proxyConfigurator) RemoveCluster(name string, ingressIP net.IP) error { |
| 259 | return fmt.Errorf("NOT IMPLEMENTED") |
| 260 | } |
| 261 | |
| 262 | type proxyPair struct { |
| 263 | From string `json:"from"` |
| 264 | To string `json:"to"` |
| 265 | } |
| 266 | |
| gio | 721c004 | 2025-04-03 11:56:36 +0400 | [diff] [blame] | 267 | func (pc *proxyConfigurator) AddIngressProxy(src, dst string) error { |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 268 | var buf bytes.Buffer |
| 269 | if err := json.NewEncoder(&buf).Encode(proxyPair{src, dst}); err != nil { |
| 270 | return err |
| 271 | } |
| 272 | resp, err := http.Post(fmt.Sprintf("%s/api/proxy/add", pc.apiAddr), "application/json", &buf) |
| 273 | if err != nil { |
| 274 | return err |
| 275 | } |
| 276 | if resp.StatusCode != http.StatusOK { |
| 277 | var buf bytes.Buffer |
| 278 | io.Copy(&buf, resp.Body) |
| 279 | return fmt.Errorf(buf.String()) |
| 280 | } |
| 281 | return nil |
| 282 | } |
| 283 | |
| gio | 721c004 | 2025-04-03 11:56:36 +0400 | [diff] [blame] | 284 | func (pc *proxyConfigurator) AddProxy(src int, dst string, protocol installer.Protocol) (string, error) { |
| 285 | // TODO(gio): implement |
| 286 | return "", fmt.Errorf("NOT IMPLEMENTED") |
| 287 | } |
| 288 | |
| 289 | func (pc *proxyConfigurator) RemoveProxy(src int, dst string, protocol installer.Protocol) error { |
| 290 | // TODO(gio): implement |
| 291 | return fmt.Errorf("NOT IMPLEMENTED") |
| 292 | } |
| 293 | |
| 294 | func (pc *proxyConfigurator) RemoveIngressProxy(src, dst string) error { |
| gio | f15b9da | 2024-09-19 06:59:16 +0400 | [diff] [blame] | 295 | var buf bytes.Buffer |
| 296 | if err := json.NewEncoder(&buf).Encode(proxyPair{src, dst}); err != nil { |
| 297 | return err |
| 298 | } |
| 299 | resp, err := http.Post(fmt.Sprintf("%s/api/proxy/remove", pc.apiAddr), "application/json", &buf) |
| 300 | if err != nil { |
| 301 | return err |
| 302 | } |
| 303 | if resp.StatusCode != http.StatusOK { |
| 304 | var buf bytes.Buffer |
| 305 | io.Copy(&buf, resp.Body) |
| 306 | return fmt.Errorf(buf.String()) |
| 307 | } |
| 308 | return nil |
| 309 | } |