| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 4 | "bytes" |
| 5 | "encoding/json" |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 6 | "flag" |
| 7 | "fmt" |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 8 | "image" |
| 9 | "image/png" |
| giolekva | 1026d2d | 2021-12-19 19:09:15 +0400 | [diff] [blame^] | 10 | "strings" |
| giolekva | 52da88a | 2021-12-17 18:08:25 +0400 | [diff] [blame] | 11 | "time" |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 12 | |
| 13 | "gioui.org/app" |
| giolekva | 96202c5 | 2021-12-18 12:45:34 +0400 | [diff] [blame] | 14 | "gioui.org/font/gofont" |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 15 | "gioui.org/io/system" |
| 16 | "gioui.org/layout" |
| 17 | "gioui.org/op" |
| 18 | "gioui.org/unit" |
| giolekva | 96202c5 | 2021-12-18 12:45:34 +0400 | [diff] [blame] | 19 | "gioui.org/widget/material" |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 20 | "github.com/skip2/go-qrcode" |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 21 | ) |
| 22 | |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 23 | var vpnApiAddr = flag.String("vpn-api-addr", "", "VPN API server address") |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 24 | |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 25 | var p *processor |
| 26 | |
| 27 | type processor struct { |
| 28 | vc VPNClient |
| 29 | app App |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 30 | st Storage |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 31 | ui *UI |
| 32 | |
| 33 | inviteQrCh chan image.Image |
| 34 | inviteQrScannedCh chan []byte |
| giolekva | 52da88a | 2021-12-17 18:08:25 +0400 | [diff] [blame] | 35 | |
| 36 | onConnectCh chan interface{} |
| 37 | onDisconnectCh chan interface{} |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 38 | |
| 39 | onConfigCh chan struct{} |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 40 | } |
| 41 | |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 42 | func newProcessor() *processor { |
| giolekva | 96202c5 | 2021-12-18 12:45:34 +0400 | [diff] [blame] | 43 | th := material.NewTheme(gofont.Collection()) |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 44 | app := createApp() |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 45 | return &processor{ |
| 46 | vc: NewDirectVPNClient(*vpnApiAddr), |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 47 | app: app, |
| 48 | st: app.CreateStorage(), |
| giolekva | 96202c5 | 2021-12-18 12:45:34 +0400 | [diff] [blame] | 49 | ui: NewUI(th), |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 50 | inviteQrCh: make(chan image.Image, 1), |
| 51 | inviteQrScannedCh: make(chan []byte, 1), |
| giolekva | 52da88a | 2021-12-17 18:08:25 +0400 | [diff] [blame] | 52 | onConnectCh: make(chan interface{}, 1), |
| 53 | onDisconnectCh: make(chan interface{}, 1), |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 54 | onConfigCh: make(chan struct{}, 1), |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 55 | } |
| 56 | } |
| 57 | |
| 58 | func (p *processor) InviteQRCodeScanned(code []byte) { |
| 59 | p.inviteQrScannedCh <- code |
| 60 | } |
| 61 | |
| giolekva | 52da88a | 2021-12-17 18:08:25 +0400 | [diff] [blame] | 62 | func (p *processor) ConnectRequested(service interface{}) { |
| 63 | go func() { |
| 64 | time.Sleep(1 * time.Second) |
| 65 | p.onConnectCh <- service |
| 66 | }() |
| 67 | } |
| 68 | |
| 69 | func (p *processor) DisconnectRequested(service interface{}) { |
| 70 | p.onDisconnectCh <- service |
| 71 | } |
| 72 | |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 73 | func (p *processor) run() error { |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 74 | w := app.NewWindow( |
| 75 | app.Size(unit.Px(1500), unit.Px(1500)), |
| 76 | app.Title("PCloud"), |
| 77 | ) |
| 78 | var ops op.Ops |
| 79 | for { |
| 80 | select { |
| 81 | case e := <-w.Events(): |
| 82 | switch e := e.(type) { |
| 83 | case app.ViewEvent: |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 84 | if err := p.app.OnView(e); err != nil { |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 85 | return err |
| 86 | } else { |
| 87 | w.Invalidate() |
| 88 | } |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 89 | if config, err := p.st.Get(); err != nil { |
| 90 | return err |
| 91 | } else { |
| 92 | if config.Network != nil { |
| 93 | p.onConfigCh <- struct{}{} |
| 94 | } |
| 95 | } |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 96 | case *system.CommandEvent: |
| 97 | if e.Type == system.CommandBack { |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 98 | if p.ui.OnBack() { |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 99 | e.Cancel = true |
| 100 | w.Invalidate() |
| 101 | } |
| 102 | } |
| 103 | case system.FrameEvent: |
| 104 | gtx := layout.NewContext(&ops, e) |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 105 | events := p.ui.Layout(gtx) |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 106 | e.Frame(&ops) |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 107 | if err := p.processUIEvents(events); err != nil { |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 108 | return err |
| 109 | } |
| 110 | } |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 111 | case img := <-p.inviteQrCh: |
| 112 | p.ui.InviteQRGenerated(img) |
| 113 | w.Invalidate() |
| 114 | case code := <-p.inviteQrScannedCh: |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 115 | if err := p.JoinAndGetNetworkConfig(code); err != nil { |
| giolekva | 96202c5 | 2021-12-18 12:45:34 +0400 | [diff] [blame] | 116 | return err |
| giolekva | 52da88a | 2021-12-17 18:08:25 +0400 | [diff] [blame] | 117 | } |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 118 | case <-p.onConfigCh: |
| 119 | if err := p.app.TriggerService(); err != nil { |
| 120 | return err |
| 121 | } |
| 122 | case s := <-p.onConnectCh: |
| 123 | if err := p.app.UpdateService(s); err != nil { |
| 124 | return err |
| 125 | } |
| 126 | if config, err := p.st.Get(); err != nil { |
| 127 | return err |
| 128 | } else { |
| 129 | if err := p.app.Connect(config); err != nil { |
| 130 | return err |
| 131 | } |
| 132 | } |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 133 | } |
| 134 | } |
| 135 | return nil |
| 136 | } |
| 137 | |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 138 | func (p *processor) processUIEvents(events []UIEvent) error { |
| 139 | for _, e := range events { |
| 140 | switch e.(type) { |
| 141 | case EventGetInviteQRCode: |
| 142 | go func() { |
| 143 | if img, err := p.generateInviteQRCode(); err == nil { |
| 144 | p.inviteQrCh <- img |
| 145 | } else { |
| 146 | // TODO(giolekva): do not panic |
| 147 | panic(err) |
| 148 | } |
| 149 | }() |
| 150 | case EventScanBarcode: |
| 151 | return p.app.LaunchBarcodeScanner() |
| 152 | default: |
| 153 | return fmt.Errorf("Unhandled event: %#v", e) |
| 154 | } |
| 155 | } |
| 156 | return nil |
| 157 | } |
| 158 | |
| 159 | type qrCodeData struct { |
| 160 | VPNApiAddr string `json:"vpn_api_addr"` |
| 161 | Message []byte `json:"message"` |
| 162 | Signature []byte `json:"signature"` |
| 163 | } |
| 164 | |
| 165 | func (p *processor) generateInviteQRCode() (image.Image, error) { |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 166 | config, err := p.st.Get() |
| 167 | if err != nil { |
| 168 | return nil, err |
| 169 | } |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 170 | message := []byte("Hello PCloud") |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 171 | signature, err := p.vc.Sign(config.ApiAddr, message) |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 172 | if err != nil { |
| 173 | return nil, err |
| 174 | } |
| 175 | c := qrCodeData{ |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 176 | config.ApiAddr, |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 177 | message, |
| 178 | signature, |
| 179 | } |
| 180 | var data bytes.Buffer |
| 181 | if err := json.NewEncoder(&data).Encode(c); err != nil { |
| 182 | return nil, err |
| 183 | } |
| 184 | qr, err := qrcode.Encode(data.String(), qrcode.Medium, 1024) |
| 185 | if err != nil { |
| 186 | return nil, err |
| 187 | } |
| 188 | img, err := png.Decode(bytes.NewReader(qr)) |
| 189 | if err != nil { |
| 190 | return nil, err |
| 191 | } |
| 192 | return img, nil |
| 193 | } |
| 194 | |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 195 | func (p *processor) JoinAndGetNetworkConfig(code []byte) error { |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 196 | var invite qrCodeData |
| 197 | if err := json.NewDecoder(bytes.NewReader(code)).Decode(&invite); err != nil { |
| giolekva | 1026d2d | 2021-12-19 19:09:15 +0400 | [diff] [blame^] | 198 | return err |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 199 | } |
| giolekva | 1026d2d | 2021-12-19 19:09:15 +0400 | [diff] [blame^] | 200 | hostname, err := p.app.GetHostname() |
| 201 | if err != nil { |
| 202 | return err |
| 203 | } |
| 204 | hostname = strings.ToLower(strings.ReplaceAll(hostname, " ", "-")) |
| 205 | fmt.Printf("------ %s\n", hostname) |
| 206 | network, err := p.vc.Join(invite.VPNApiAddr, hostname, invite.Message, invite.Signature) |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 207 | if err != nil { |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 208 | return err |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 209 | } |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 210 | if err := p.st.Store(Config{invite.VPNApiAddr, network}); err != nil { |
| 211 | return err |
| giolekva | 52da88a | 2021-12-17 18:08:25 +0400 | [diff] [blame] | 212 | } |
| giolekva | 8d6a0ca | 2021-12-19 17:42:25 +0400 | [diff] [blame] | 213 | p.onConfigCh <- struct{}{} |
| 214 | return nil |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 215 | } |
| 216 | |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 217 | func main() { |
| 218 | flag.Parse() |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 219 | p = newProcessor() |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 220 | go func() { |
| giolekva | f58a769 | 2021-12-15 18:05:39 +0400 | [diff] [blame] | 221 | if err := p.run(); err != nil { |
| giolekva | 313ee2b | 2021-12-15 15:17:29 +0400 | [diff] [blame] | 222 | panic(err) |
| 223 | } |
| 224 | }() |
| 225 | app.Main() |
| 226 | } |