From 60843869892bd34d86b322f3b1c78116f496ba3b Mon Sep 17 00:00:00 2001 From: nite Date: Mon, 6 Oct 2025 07:58:31 +1100 Subject: [PATCH] refactor(bilinovel): Migrate browser automation from Chromedp to Playwright This commit replaces the `chromedp` library with `playwright-go` for browser automation within the Bilinovel downloader. Changes include: * Updated `Bilinovel` struct to manage Playwright browser, context, and page instances. * Rewrote `initBrowser` and `Close` methods to use Playwright's API for browser lifecycle management. * Refactored `processContentWithChromedp` to `processContentWithPlaywright`, adapting the logic to use Playwright's page evaluation capabilities. * Removed unused `context` and `time` imports. * Added HTML cleanup in `getChapterByPage` to remove `class` attributes from images and `data-k` attributes from all elements, improving content consistency. --- .vscode/launch.json | 4 +- downloader/bilinovel/bilinovel.go | 233 +++++++++++++----------------- go.mod | 12 +- go.sum | 48 +++--- main.go | 12 ++ 5 files changed, 140 insertions(+), 169 deletions(-) diff --git a/.vscode/launch.json b/.vscode/launch.json index 4de0887..76707b0 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -10,7 +10,9 @@ "args": [ "download", "-n", - "3095" + "2321", + "-v", + "162759" ] } ] diff --git a/downloader/bilinovel/bilinovel.go b/downloader/bilinovel/bilinovel.go index 2d8f211..fc8b95e 100644 --- a/downloader/bilinovel/bilinovel.go +++ b/downloader/bilinovel/bilinovel.go @@ -4,7 +4,6 @@ import ( "bilinovel-downloader/model" "bilinovel-downloader/utils" "bytes" - "context" "crypto/sha256" _ "embed" "fmt" @@ -16,12 +15,10 @@ import ( "regexp" "strconv" "strings" - "time" "github.com/PuerkitoBio/goquery" mapper "github.com/bestnite/font-mapper" - "github.com/chromedp/cdproto/network" - "github.com/chromedp/chromedp" + "github.com/playwright-community/playwright-go" ) //go:embed read.ttf @@ -36,10 +33,9 @@ type Bilinovel struct { restyClient *utils.RestyClient // 浏览器实例复用 - allocCtx context.Context - allocCancel context.CancelFunc - browserCtx context.Context - browserCancel context.CancelFunc + browser playwright.Browser + browserContext playwright.BrowserContext + page playwright.Page } func New() (*Bilinovel, error) { @@ -74,46 +70,41 @@ func (b *Bilinovel) GetExtraFiles() []model.ExtraFile { // initBrowser 初始化浏览器实例 func (b *Bilinovel) initBrowser() error { - // 创建chromedp选项 - opts := append(chromedp.DefaultExecAllocatorOptions[:], - chromedp.Flag("headless", true), - chromedp.Flag("disable-gpu", true), - chromedp.Flag("disable-dev-shm-usage", true), - chromedp.Flag("disable-extensions", true), - chromedp.Flag("no-sandbox", true), - chromedp.Flag("disable-background-timer-throttling", true), - chromedp.Flag("disable-backgrounding-occluded-windows", true), - chromedp.Flag("disable-renderer-backgrounding", true), - ) - - var err error - b.allocCtx, b.allocCancel = chromedp.NewExecAllocator(context.Background(), opts...) - b.browserCtx, b.browserCancel = chromedp.NewContext(b.allocCtx) - - // 预热浏览器 - 导航到空白页 - err = chromedp.Run(b.browserCtx, chromedp.Navigate("about:blank")) + pw, err := playwright.Run() if err != nil { - b.closeBrowser() - return fmt.Errorf("failed to initialize browser: %v", err) + return fmt.Errorf("could not start playwright: %w", err) + } + b.browser, err = pw.Chromium.Launch(playwright.BrowserTypeLaunchOptions{ + Headless: playwright.Bool(false), + }) + if err != nil { + return fmt.Errorf("could not launch browser: %w", err) + } + + b.browserContext, err = b.browser.NewContext() + if err != nil { + return fmt.Errorf("could not create browser context: %w", err) + } + + b.page, err = b.browserContext.NewPage() + if err != nil { + return fmt.Errorf("could not create page: %w", err) } log.Println("Browser initialized successfully") return nil } -// closeBrowser 关闭浏览器实例 -func (b *Bilinovel) closeBrowser() { - if b.browserCancel != nil { - b.browserCancel() - } - if b.allocCancel != nil { - b.allocCancel() - } -} - -// Close 关闭下载器时清理资源 +// Close 清理资源 func (b *Bilinovel) Close() error { - b.closeBrowser() + if b.browser != nil { + if err := b.browser.Close(); err != nil { + log.Printf("could not close browser: %v", err) + } + b.browser = nil + b.browserContext = nil + b.page = nil + } return nil } @@ -354,7 +345,7 @@ func (b *Bilinovel) getChapterByPage(chapter *model.Chapter, page int) (bool, er html := resp.Body() // 解决乱序问题 - resortedHtml, err := b.processContentWithChromedp(string(html)) + resortedHtml, err := b.processContentWithPlaywright(string(html)) if err != nil { return false, fmt.Errorf("failed to process html: %w", err) } @@ -402,6 +393,7 @@ func (b *Bilinovel) getChapterByPage(chapter *model.Chapter, page int) (bool, er imageFilename := fmt.Sprintf("%x%s", string(imageHash[:]), path.Ext(imgUrl)) s.SetAttr("src", imageFilename) s.SetAttr("alt", imgUrl) + s.RemoveAttr("class") img, err := b.getImg(imgUrl) if err != nil { return @@ -416,6 +408,19 @@ func (b *Bilinovel) getChapterByPage(chapter *model.Chapter, page int) (bool, er }) } + doc.Find("*").Each(func(i int, s *goquery.Selection) { + if len(s.Nodes) > 0 && len(s.Nodes[0].Attr) > 0 { + // 遍历元素的所有属性 + for _, attr := range s.Nodes[0].Attr { + // 3. 检查属性名是否以 "data-k" 开头,且属性值是否为空 + if strings.HasPrefix(attr.Key, "data-k") { + // 4. 如果满足条件,就移除这个属性 + s.RemoveAttr(attr.Key) + } + } + } + }) + htmlStr, err := content.Html() if err != nil { return false, fmt.Errorf("failed to get html: %v", err) @@ -439,8 +444,8 @@ func (b *Bilinovel) getImg(url string) ([]byte, error) { return resp.Body(), nil } -// processContentWithChromedp 使用复用的浏览器实例处理内容 -func (b *Bilinovel) processContentWithChromedp(htmlContent string) (string, error) { +// processContentWithPlaywright 使用复用的浏览器实例处理内容 +func (b *Bilinovel) processContentWithPlaywright(htmlContent string) (string, error) { tempFile, err := os.CreateTemp("", "bilinovel-temp-*.html") if err != nil { return "", fmt.Errorf("failed to create temp file: %w", err) @@ -454,103 +459,63 @@ func (b *Bilinovel) processContentWithChromedp(htmlContent string) (string, erro tempFile.Close() tempFilePath := tempFile.Name() - // 为当前任务创建子上下文 - ctx, cancel := context.WithTimeout(b.browserCtx, 30*time.Second) - defer cancel() + _, err = b.page.ExpectResponse(func(url string) bool { + return strings.Contains(url, "chapterlog.js") + }, func() error { + _, err = b.page.Goto("file://" + filepath.ToSlash(tempFilePath)) + if err != nil { + return fmt.Errorf("could not navigate to file: %w", err) + } + return nil + }, playwright.PageExpectResponseOptions{ + Timeout: playwright.Float(5000), + }) + if err != nil { + return "", fmt.Errorf("failed to wait for network request finish") + } - var processedHTML string + err = b.page.Locator("#acontent").WaitFor(playwright.LocatorWaitForOptions{ + State: playwright.WaitForSelectorStateVisible, + }) + if err != nil { + return "", fmt.Errorf("could not wait for #acontent: %w", err) + } - // 设置网络事件监听 - networkEventChan := make(chan bool, 1) - var requestID string - - // 执行处理任务 - err = chromedp.Run(ctx, - network.Enable(), - - // 设置网络事件监听器 - chromedp.ActionFunc(func(ctx context.Context) error { - chromedp.ListenTarget(ctx, func(ev interface{}) { - switch ev := ev.(type) { - case *network.EventRequestWillBeSent: - if strings.Contains(ev.Request.URL, "chapterlog.js") { - requestID = ev.RequestID.String() - } - case *network.EventLoadingFinished: - if ev.RequestID.String() == requestID && requestID != "" { - select { - case networkEventChan <- true: - default: - } - } + // 遍历所有 #acontent 的子元素, 通过 window.getComputedStyle().display 检测是否是 none, 如果是 none 则从页面删除这个元素 + result, err := b.page.Evaluate(` + (function() { + const acontent = document.getElementById('acontent'); + if (!acontent) { + return 'acontent element not found'; + } + + let removedCount = 0; + const elements = acontent.querySelectorAll('*'); + + // 从后往前遍历,避免删除元素时影响索引 + for (let i = elements.length - 1; i >= 0; i--) { + const element = elements[i]; + const computedStyle = window.getComputedStyle(element); + + if (computedStyle.display === 'none') { + element.remove(); + removedCount++; } - }) - return nil - }), - - // 导航到本地文件 - chromedp.Navigate("file://"+filepath.ToSlash(tempFilePath)), - - // 等待页面加载完成 - chromedp.WaitVisible(`#acontent`, chromedp.ByID), - - // 等待外部脚本加载或超时 - chromedp.ActionFunc(func(ctx context.Context) error { - select { - case <-networkEventChan: - log.Println("External script loaded successfully") - case <-time.After(10 * time.Second): - log.Println("Timeout waiting for external script, continuing anyway") - case <-ctx.Done(): - return ctx.Err() } - return nil - }), - - // 遍历所有 #acontent 的子元素, 通过 window.getComputedStyle().display 检测是否是 none, 如果是 none 则从页面删除这个元素 - chromedp.ActionFunc(func(ctx context.Context) error { - // 执行JavaScript来移除display:none的元素 - var result string - err := chromedp.Evaluate(` - (function() { - const acontent = document.getElementById('acontent'); - if (!acontent) { - return 'acontent element not found'; - } - - let removedCount = 0; - const elements = acontent.querySelectorAll('*'); - - // 从后往前遍历,避免删除元素时影响索引 - for (let i = elements.length - 1; i >= 0; i--) { - const element = elements[i]; - const computedStyle = window.getComputedStyle(element); - - if (computedStyle.display === 'none') { - element.remove(); - removedCount++; - } - } - - return 'Removed ' + removedCount + ' hidden elements'; - })() - `, &result).Do(ctx) - - if err != nil { - log.Printf("Failed to remove hidden elements: %v", err) - return err - } - - log.Printf("Hidden elements removal result: %s", result) - return nil - }), - - // 获取页面的HTML代码 - chromedp.OuterHTML("html", &processedHTML, chromedp.ByQuery), - ) + + return 'Removed ' + removedCount + ' hidden elements'; + })() + `) if err != nil { - return "", fmt.Errorf("chromedp execution failed: %w", err) + return "", fmt.Errorf("failed to remove hidden elements: %w", err) + } + + log.Printf("Hidden elements removal result: %s", result) + + processedHTML, err := b.page.Content() + if err != nil { + return "", fmt.Errorf("could not get page content: %w", err) } return processedHTML, nil diff --git a/go.mod b/go.mod index e459659..2c3495a 100644 --- a/go.mod +++ b/go.mod @@ -6,24 +6,20 @@ require ( github.com/PuerkitoBio/goquery v1.10.3 github.com/a-h/templ v0.3.943 github.com/bestnite/font-mapper v0.0.0-20250823155658-56c76d820267 - github.com/chromedp/cdproto v0.0.0-20250803210736-d308e07a266d - github.com/chromedp/chromedp v0.14.1 github.com/go-resty/resty/v2 v2.16.5 github.com/google/uuid v1.6.0 + github.com/playwright-community/playwright-go v0.5200.1 github.com/spf13/cobra v1.9.1 ) require ( github.com/andybalholm/cascadia v1.3.3 // indirect - github.com/chromedp/sysutil v1.1.0 // indirect - github.com/go-json-experiment/json v0.0.0-20250813233538-9b1f9ea2e11b // indirect - github.com/gobwas/httphead v0.1.0 // indirect - github.com/gobwas/pool v0.2.1 // indirect - github.com/gobwas/ws v1.4.0 // indirect + github.com/deckarep/golang-set/v2 v2.8.0 // indirect + github.com/go-jose/go-jose/v3 v3.0.4 // indirect + github.com/go-stack/stack v1.8.1 // indirect github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 // indirect github.com/inconshreveable/mousetrap v1.1.0 // indirect github.com/spf13/pflag v1.0.7 // indirect golang.org/x/image v0.30.0 // indirect golang.org/x/net v0.43.0 // indirect - golang.org/x/sys v0.35.0 // indirect ) diff --git a/go.sum b/go.sum index 52a57e7..c98ffac 100644 --- a/go.sum +++ b/go.sum @@ -1,49 +1,48 @@ github.com/PuerkitoBio/goquery v1.10.3 h1:pFYcNSqHxBD06Fpj/KsbStFRsgRATgnf3LeXiUkhzPo= github.com/PuerkitoBio/goquery v1.10.3/go.mod h1:tMUX0zDMHXYlAQk6p35XxQMqMweEKB7iK7iLNd4RH4Y= -github.com/a-h/templ v0.3.906 h1:ZUThc8Q9n04UATaCwaG60pB1AqbulLmYEAMnWV63svg= -github.com/a-h/templ v0.3.906/go.mod h1:FFAu4dI//ESmEN7PQkJ7E7QfnSEMdcnu7QrAY8Dn334= github.com/a-h/templ v0.3.943 h1:o+mT/4yqhZ33F3ootBiHwaY4HM5EVaOJfIshvd5UNTY= github.com/a-h/templ v0.3.943/go.mod h1:oCZcnKRf5jjsGpf2yELzQfodLphd2mwecwG4Crk5HBo= github.com/andybalholm/cascadia v1.3.3 h1:AG2YHrzJIm4BZ19iwJ/DAua6Btl3IwJX+VI4kktS1LM= github.com/andybalholm/cascadia v1.3.3/go.mod h1:xNd9bqTn98Ln4DwST8/nG+H0yuB8Hmgu1YHNnWw0GeA= github.com/bestnite/font-mapper v0.0.0-20250823155658-56c76d820267 h1:nmUTJV2u/0XmVjQ++VIy/Hu+MtxdpQvOevvcSZtUATA= github.com/bestnite/font-mapper v0.0.0-20250823155658-56c76d820267/go.mod h1:cfB1e9YhoI/QWrXPp3h6QVAKU6iCI2ifbjRPHP3xf/0= -github.com/chromedp/cdproto v0.0.0-20250803210736-d308e07a266d h1:ZtA1sedVbEW7EW80Iz2GR3Ye6PwbJAJXjv7D74xG6HU= -github.com/chromedp/cdproto v0.0.0-20250803210736-d308e07a266d/go.mod h1:NItd7aLkcfOA/dcMXvl8p1u+lQqioRMq/SqDp71Pb/k= -github.com/chromedp/chromedp v0.14.1 h1:0uAbnxewy/Q+Bg7oafVePE/6EXEho9hnaC38f+TTENg= -github.com/chromedp/chromedp v0.14.1/go.mod h1:rHzAv60xDE7VNy/MYtTUrYreSc0ujt2O1/C3bzctYBo= -github.com/chromedp/sysutil v1.1.0 h1:PUFNv5EcprjqXZD9nJb9b/c9ibAbxiYo4exNWZyipwM= -github.com/chromedp/sysutil v1.1.0/go.mod h1:WiThHUdltqCNKGc4gaU50XgYjwjYIhKWoHGPTUfWTJ8= github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g= -github.com/go-json-experiment/json v0.0.0-20250813233538-9b1f9ea2e11b h1:6Q4zRHXS/YLOl9Ng1b1OOOBWMidAQZR3Gel0UKPC/KU= -github.com/go-json-experiment/json v0.0.0-20250813233538-9b1f9ea2e11b/go.mod h1:TiCD2a1pcmjd7YnhGH0f/zKNcCD06B029pHhzV23c2M= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/deckarep/golang-set/v2 v2.8.0 h1:swm0rlPCmdWn9mESxKOjWk8hXSqoxOp+ZlfuyaAdFlQ= +github.com/deckarep/golang-set/v2 v2.8.0/go.mod h1:VAky9rY/yGXJOLEDv3OMci+7wtDpOF4IN+y82NBOac4= +github.com/go-jose/go-jose/v3 v3.0.4 h1:Wp5HA7bLQcKnf6YYao/4kpRpVMp/yf6+pJKV8WFSaNY= +github.com/go-jose/go-jose/v3 v3.0.4/go.mod h1:5b+7YgP7ZICgJDBdfjZaIt+H/9L9T/YQrVfLAMboGkQ= github.com/go-resty/resty/v2 v2.16.5 h1:hBKqmWrr7uRc3euHVqmh1HTHcKn99Smr7o5spptdhTM= github.com/go-resty/resty/v2 v2.16.5/go.mod h1:hkJtXbA2iKHzJheXYvQ8snQES5ZLGKMwQ07xAwp/fiA= -github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU= -github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM= -github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og= -github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= -github.com/gobwas/ws v1.4.0 h1:CTaoG1tojrh4ucGPcoJFiAQUAsEWekEWvLy7GsVNqGs= -github.com/gobwas/ws v1.4.0/go.mod h1:G3gNqMNtPppf5XUz7O4shetPpcZ1VJ7zt18dlUeakrc= +github.com/go-stack/stack v1.8.1 h1:ntEHSVwIt7PNXNpgPmVfMrNhLtgjlmnZha2kOpuRiDw= +github.com/go-stack/stack v1.8.1/go.mod h1:dcoOX6HbPZSZptuspn9bctJ+N/CnF5gGygcUP3XYfe4= github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 h1:DACJavvAHhabrF08vX0COfcOBJRhZ8lUbR+ZWIs0Y5g= github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGwJL78qG/PmXZO1EjYhfJinVAhrmmHX6Z8B9k= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= -github.com/ledongthuc/pdf v0.0.0-20220302134840-0c2507a12d80 h1:6Yzfa6GP0rIo/kULo2bwGEkFvCePZ3qHDDTC3/J9Swo= -github.com/ledongthuc/pdf v0.0.0-20220302134840-0c2507a12d80/go.mod h1:imJHygn/1yfhB7XSJJKlFZKl/J+dCPAknuiaGOshXAs= -github.com/orisano/pixelmatch v0.0.0-20220722002657-fb0b55479cde h1:x0TT0RDC7UhAVbbWWBzr41ElhJx5tXPWkIHA2HWPRuw= -github.com/orisano/pixelmatch v0.0.0-20220722002657-fb0b55479cde/go.mod h1:nZgzbfBr3hhjoZnS66nKrHmduYNpc34ny7RK4z5/HM0= +github.com/mitchellh/go-ps v1.0.0 h1:i6ampVEEF4wQFF+bkYfwYgY+F/uYJDktmvLPf7qIgjc= +github.com/mitchellh/go-ps v1.0.0/go.mod h1:J4lOc8z8yJs6vUwklHw2XEIiT4z4C40KtWVN3nvg8Pg= +github.com/playwright-community/playwright-go v0.5200.1 h1:Sm2oOuhqt0M5Y4kUi/Qh9w4cyyi3ZIWTBeGKImc2UVo= +github.com/playwright-community/playwright-go v0.5200.1/go.mod h1:UnnyQZaqUOO5ywAZu60+N4EiWReUqX1MQBBA3Oofvf8= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo= github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0= -github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o= github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/pflag v1.0.7 h1:vN6T9TfwStFPFM5XzjsvmzZkLuaLX+HS+0SeFLRgU6M= github.com/spf13/pflag v1.0.7/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= +github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= @@ -67,8 +66,6 @@ golang.org/x/net v0.15.0/go.mod h1:idbUs1IY1+zTqbi8yxTbhexhEEk5ur9LInksu6HrEpk= golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4= -golang.org/x/net v0.39.0 h1:ZCu7HMWDxpXpaiKdhzIfaltL9Lp31x/3fCP11bc6/fY= -golang.org/x/net v0.39.0/go.mod h1:X7NRbYVEA+ewNkCNyJ513WmMdQ3BineSwVtN2zD/d+E= golang.org/x/net v0.43.0 h1:lat02VYK2j4aLzMzecihNvTlJNQUq316m2Mr9rnM6YE= golang.org/x/net v0.43.0/go.mod h1:vhO1fvI4dGsIjh73sWfUVjj3N7CA9WkKJNQm2svM6Jg= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -84,14 +81,11 @@ golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/sys v0.28.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= -golang.org/x/sys v0.35.0 h1:vz1N37gP5bs89s7He8XuIYXpyY0+QlsKmzipCbUtyxI= -golang.org/x/sys v0.35.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= golang.org/x/telemetry v0.0.0-20240228155512-f48c80bd79b2/go.mod h1:TeRTkGYfJXctD9OcfyVLyj2J3IxLnKwHJR8f4D8a3YE= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= @@ -120,4 +114,6 @@ golang.org/x/tools v0.13.0/go.mod h1:HvlwmtVNQAhOuCjW7xxvovg8wbNq7LwfXh/k7wXUl58 golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/main.go b/main.go index f628dfe..cb02fe7 100644 --- a/main.go +++ b/main.go @@ -2,8 +2,20 @@ package main import ( "bilinovel-downloader/cmd" + "io" + "log" + + "github.com/playwright-community/playwright-go" ) func main() { + log.Println("Installing playwright") + err := playwright.Install(&playwright.RunOptions{ + Browsers: []string{"chromium"}, + Stdout: io.Discard, + }) + if err != nil { + log.Panicf("failed to install playwright") + } _ = cmd.RootCmd.Execute() }