-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathchrome.go
160 lines (148 loc) · 4.42 KB
/
chrome.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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
package chromebot
import (
"context"
"errors"
"sync"
"github.com/chromedp/chromedp"
)
// Chrome represents a running Chrome browser.
type Chrome struct {
finMu sync.Mutex
fin chan struct{}
exeCancel context.CancelFunc
tabMu sync.Mutex
tabs []*Tab // tab id ascending
}
// New is an alias for NewChrome.
func New(headless bool) *Chrome {
return NewChrome(headless)
}
// NewChrome runs a new browser.
// Use `(*Chrome).Close()` to free the browser allocated by this.
func NewChrome(headless bool) *Chrome {
exeCtx, exeCancel := chromedp.NewExecAllocator(context.Background(),
func() []chromedp.ExecAllocatorOption {
if headless {
return append(chromedp.DefaultExecAllocatorOptions[:], chromedp.Headless)
}
return append(chromedp.DefaultExecAllocatorOptions[:],
chromedp.Flag("headless", false),
chromedp.Flag("hide-scrollbars", false),
chromedp.Flag("mute-audio", false),
) // windowed foreground
}()...)
tabCtx, tabCancel := chromedp.NewContext(exeCtx)
chromedp.Run(tabCtx) // ignore error
ret := &Chrome{
finMu: sync.Mutex{},
fin: make(chan struct{}),
exeCancel: exeCancel,
tabMu: sync.Mutex{},
tabs: []*Tab{newMainTab(tabCtx, tabCancel)},
}
go func() {
<-tabCtx.Done() // when the main tab is dead
defer func() { // DCL
ret.finMu.Lock()
defer ret.finMu.Unlock()
select {
case <-ret.fin:
return
default: // free
ret.exeCancel()
close(ret.fin)
}
}()
}()
return ret
}
// Close gracefully closes the browser and all its tabs.
func (c *Chrome) Close() {
defer func() { // DCL
c.finMu.Lock()
defer c.finMu.Unlock()
select {
case <-c.fin:
return
default:
c.exeCancel()
close(c.fin)
}
}()
}
// Dead returns a channel that's closed when work done on behalf of the browser should be dead.
// This notifies a user when this browser is closed.
func (c *Chrome) Dead() <-chan struct{} {
return c.fin
}
// Do runs a cdproto command on this browser.
// You might consider using `(*Tab).Do()` instead, as this function cannot run page specific methods like `Page.navigate()`.
// func (c *Chrome) Do() domain.Domain {
// ctx := func() context.Context {
// c.tabMu.Lock()
// defer c.tabMu.Unlock()
// return c.tabs[0].ctx
// }()
// return domain.Do(cdp.WithExecutor(ctx, chromedp.FromContext(ctx).Browser))
// }
// Listen adds a callback function which will be called whenever a browser event is received on this browser.
//
// Usage:
// c.Listen(func(ev interface{}) {
// switch ev := ev.(type) {
// case *runtime.EventConsoleAPICalled:
// // do something with ev
// case *runtime.EventExceptionThrown:
// // do something with ev
// }
// })
//
// Note that the function is called synchronously when handling events.
// The function should avoid blocking at all costs.
// For example, any Actions must be run via a separate goroutine.
func (c *Chrome) Listen(onBrowserEvent func(ev interface{})) error {
select {
case <-c.fin:
return errors.New("the browser is dead")
default:
}
c.tabMu.Lock()
defer c.tabMu.Unlock()
chromedp.ListenBrowser(c.tabs[0].ctx, onBrowserEvent)
return nil
}
// CountTabs returns the number of tabs created by this Chrome.
func (c *Chrome) CountTabs() int {
c.tabMu.Lock()
defer c.tabMu.Unlock()
return len(c.tabs)
}
// AddNewTab creates a new tab and add it to this Chrome.
// The function returns `nil` when the browser is dead.
// Panic if `chromedp.WithBrowserOption()` passed as an argument.
// Only tab options are accepted.
func (c *Chrome) AddNewTab(opts ...chromedp.ContextOption) (added *Tab) {
select {
case <-c.fin:
return nil
default:
}
c.tabMu.Lock()
defer c.tabMu.Unlock()
ctx, cancel := chromedp.NewContext(c.tabs[0].ctx, opts...)
retTab := newExtraTab(ctx, cancel)
c.tabs = append(c.tabs, retTab)
chromedp.Run(retTab.ctx)
return retTab
}
// Tab is a getter retrieving a tab. Returns nil upon exception.
// The `indexOfTab` is an integer within [0, N) where N is the number of tabs created by the browser.
// E.g. `(*Chrome).Tab(0)` will return the first tab of the browser. Tab(1) for the second one, Tab(2) for the third one and so on.
func (c *Chrome) Tab(indexOfTab int) *Tab {
c.tabMu.Lock()
defer c.tabMu.Unlock()
if indexOfTab >= len(c.tabs) || indexOfTab < 0 {
return nil
}
return c.tabs[indexOfTab]
}