-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathchromedpproxy.go
128 lines (112 loc) · 3.49 KB
/
chromedpproxy.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
package chromedpproxy
import (
"context"
"errors"
chromedpundetected "github.com/Davincible/chromedp-undetected"
"github.com/chromedp/cdproto/page"
"github.com/chromedp/cdproto/target"
"github.com/chromedp/chromedp"
"strings"
"sync"
)
var mutex = sync.RWMutex{}
var loaded = make(chan bool, 1)
var mainContext context.Context
var mainCancel chan bool
var totalTargets = 0
// PrepareProxy abstracts chromedp.NewExecAllocator to the use case of this package
// it accepts listen addresses for both Chrome remote debugging and frontend as configuration
// it is also a variadic function that accepts extra chromedp.ExecAllocatorOption to be passed to the chromedp.NewExecAllocator
func PrepareProxy(chromeListenAddr string, frontendListenAddr string, customOpts ...chromedp.ExecAllocatorOption) {
// ensure only exactly one context is prepared
mutex.Lock()
if mainContext != nil {
mutex.Unlock()
return
}
// split up chromeListenAddr, default host to 127.0.0.1 if not specified
chromeListenAddrSplit := strings.Split(chromeListenAddr, ":")
if chromeListenAddrSplit[0] == "" {
chromeListenAddrSplit[0] = "127.0.0.1"
}
// insert remote-debugging flags and any additional options
opts := []chromedp.ExecAllocatorOption{
chromedp.Flag("remote-debugging-address", chromeListenAddrSplit[0]),
chromedp.Flag("remote-debugging-port", chromeListenAddrSplit[1]),
}
if len(customOpts) > 0 {
opts = append(opts, customOpts...)
}
// create context and keep alive
go func() {
ctx, cancel, _ := chromedpundetected.New(chromedpundetected.NewConfig(
chromedpundetected.WithChromeFlags(opts...),
chromedpundetected.WithHeadless(),
))
defer cancel()
mainContext = ctx
loaded <- true
mutex.Unlock()
mainCancel = make(chan bool, 1)
defer close(mainCancel)
startFrontEnd(frontendListenAddr, chromeListenAddrSplit[1], mainCancel)
}()
}
// NewTab abstracts creating a new tab in the root context
// it returns a target ID or error
func NewTab(url string, customOpts ...chromedp.ContextOption) (target.ID, error) {
// if context is not prepared, create with default values
if mainContext == nil {
PrepareProxy(":9222", ":9221")
<-loaded
}
mutex.Lock()
defer mutex.Unlock()
mainContext, _ = chromedp.NewContext(mainContext, customOpts...)
err := chromedp.Run(mainContext, chromedp.Tasks{
chromedp.Navigate(url),
})
if err != nil {
return "", err
}
err = chromedp.Run(mainContext, chromedp.Tasks{
chromedp.Navigate(url),
})
if err != nil {
return "", err
}
totalTargets++
// return target ID
chromeContext := chromedp.FromContext(mainContext)
return chromeContext.Target.TargetID, nil
}
// GetTarget returns a context from a target ID
func GetTarget(id target.ID) context.Context {
mutex.RLock()
defer mutex.RUnlock()
// return context from target ID
ctx, _ := chromedp.NewContext(mainContext, chromedp.WithTargetID(id))
return ctx
}
// CloseTarget closes a target by closing the page
// if the last page has been closed, clean up everything
// it returns an error if any
func CloseTarget(id target.ID) error {
mutex.Lock()
defer mutex.Unlock()
if mainContext == nil {
return errors.New("context not prepared or already closed")
}
ctx, cancel := chromedp.NewContext(mainContext, chromedp.WithTargetID(id))
_ = chromedp.Run(ctx, page.Close())
_ = target.CloseTarget(id).Do(mainContext)
cancel()
targets, err := chromedp.Targets(mainContext)
if err != nil || len(targets) <= 0 {
loaded = make(chan bool, 1)
mainContext = nil
mainCancel <- true
totalTargets = 0
}
return nil
}