-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathplugin.go
127 lines (102 loc) · 2.78 KB
/
plugin.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
package pocketbase_plugin_ngrok
import (
"context"
"fmt"
"log"
"strings"
"github.com/fatih/color"
"github.com/pocketbase/pocketbase/core"
"golang.ngrok.com/ngrok"
"golang.ngrok.com/ngrok/config"
)
// Options defines optional struct to customize the default plugin behavior.
type Options struct {
// Ctx is a context that will be used to start ngrok tunnel.
Ctx context.Context
// Enabled defines if ngrok tunnel should be started.
Enabled bool
// AuthToken is your ngrok auth token. You can get it from https://dashboard.ngrok.com/auth
AuthToken string
// AfterSetup is a callback function that will be called after ngrok tunnel is started.
AfterSetup func(url string) error
}
type Plugin struct {
// app is a Pocketbase application instance.
app core.App
// options is a plugin options.
options *Options
}
// Validate plugin options. Return error if some option is invalid.
func (p *Plugin) Validate() error {
if p.options == nil {
return fmt.Errorf("options is required")
}
if p.options.Ctx == nil {
return fmt.Errorf("context is required")
}
if p.app == nil {
return fmt.Errorf("app is required")
}
if p.options.Enabled && p.options.AuthToken == "" {
return fmt.Errorf("AuthToken is required when ngrok is enabled")
}
return nil
}
func (p *Plugin) exposeNgrok(e *core.ServeEvent) error {
if p.options.Enabled {
tun, err := ngrok.Listen(
p.options.Ctx,
config.HTTPEndpoint(config.WithHTTPHandler(e.Router)),
ngrok.WithAuthtoken(p.options.AuthToken),
)
if err != nil {
return err
}
if p.options.AfterSetup != nil {
if afterErr := p.options.AfterSetup(tun.URL()); afterErr != nil {
return afterErr
}
}
date := new(strings.Builder)
log.New(date, "", log.LstdFlags).Print()
bold := color.New(color.Bold).Add(color.FgGreen)
bold.Printf(
"%s Ngrok tunnel started at %s\n",
strings.TrimSpace(date.String()),
color.CyanString("%s", tun.URL()),
)
regular := color.New()
regular.Printf(" ➜ REST API: %s\n", color.CyanString("%s/api/", tun.URL()))
regular.Printf(" ➜ Admin UI: %s\n", color.CyanString("%s/_/", tun.URL()))
}
return nil
}
// MustRegister is a helper function that registers plugin and panics if error occurred.
func MustRegister(app core.App, options *Options) *Plugin {
if p, err := Register(app, options); err != nil {
panic(err)
} else {
return p
}
}
// Register registers plugin.
func Register(app core.App, options *Options) (*Plugin, error) {
p := &Plugin{app: app}
// Set default options
if options != nil {
p.options = options
} else {
p.options = &Options{}
}
// Validate options
if err := p.Validate(); err != nil {
return p, err
}
app.OnBeforeServe().Add(func(e *core.ServeEvent) error {
if err := p.exposeNgrok(e); err != nil {
return err
}
return nil
})
return p, nil
}