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