检查Chrome收藏夹链接是否有效

检查Chrome浏览器标签,通过Chrome导出的收藏夹文件,挨个使用http get请求检查url是否有效

package main

import (
	"bufio"
	"errors"
	"flag"
	"fmt"
	"io"
	"net"
	"net/http"
	"net/url"
	"os"
	"strings"
	"sync"
	"time"
)

func main() {
	path := flag.String("f", "", "bookmark path")
	proxy := flag.String("p", "", "proxy address")
	flag.Parse()

	err := checkChromeBookmark(*path, *proxy)
	if err != nil {
		fmt.Println(err)
	}
}

func checkChromeBookmark(path, proxy string) error {
	fw, err := os.Open(path)
	if err != nil {
		return err
	}
	//goland:noinspection GoUnhandledErrorResult
	defer fw.Close()

	const to = time.Second * 5 // 指定超时时间
	var (
		r    = bufio.NewScanner(fw)
		data []string
		dc   = &http.Client{Timeout: to}
		pc   *http.Client
		wg   sync.WaitGroup
	)
	// 解析代理url成功,才设置代理对象
	if up, err := url.Parse(proxy); err == nil {
		pc = &http.Client{
			Timeout: to,
			Transport: &http.Transport{
				Proxy: func(*http.Request) (*url.URL, error) {
					return up, nil
				},
			},
		}
	}

	for r.Scan() {
		line := r.Text()
		level := strings.IndexFunc(line, func(r rune) bool { return r != ' ' })
		if level > 0 && level%4 == 0 {
			level = level/4 - 1 // 每4个空格算一级,这里计算dir属于哪一级

			if si := strings.Index(line, `<DT><H3 ADD_DATE="`); si > 0 { // 目录
				si = strings.LastIndex(line, `">`) + 2
				if ei := len(line) - 5; level >= len(data) {
					data = append(data, line[si:ei]) // 目录多一级则追加
				} else {
					data[level] = line[si:ei] // 否则赋值同级已有目录
				}
			} else if si = strings.Index(line, `<DT><A HREF="`); si > 0 { // 链接
				ul := line[si+13 : strings.Index(line, ` ADD_DATE="`)-1]
				info := fmt.Sprintf("\n\"%s\", %s, %s:\n\t",
					strings.Join(data[:level], `" -> "`),
					line[strings.LastIndex(line, `">`)+2:len(line)-4],
					ul)

				wg.Add(1)
				go func() { // 启动协程同时执行检测逻辑
					defer wg.Done()

					err = checkUrl(dc, ul)
					if err != nil && pc != nil {
						// 只有当直连超时才尝试代理连接
						if e, ok := err.(net.Error); ok && e.Timeout() {
							err = checkUrl(pc, ul)
							if err == nil {
								err = errors.New("direct connection failed Proxy success")
							}
						}
					}
					if err != nil {
						fmt.Println(info, err)
					}
				}()
			}
		}
	}
	wg.Wait()
	return r.Err()
}

func checkUrl(c *http.Client, u string) error {
	req, err := http.NewRequest(http.MethodGet, u, nil)
	if err != nil {
		return err
	}
	req.Header.Set("user-agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36")
	req.Header.Set("accept", "*/*")

	resp, err := c.Do(req)
	if err != nil {
		return err
	}
	//goland:noinspection GoUnhandledErrorResult
	defer resp.Body.Close()

	n, err := io.Copy(io.Discard, resp.Body)
	if err != nil {
		return err
	}

	if resp.StatusCode != http.StatusOK {
		return fmt.Errorf("code %d", resp.StatusCode)
	}
	if n <= 0 {
		return fmt.Errorf("n <= 0")
	}
	return nil
}
posted @ 2023-02-02 15:02  janbar  阅读(88)  评论(0编辑  收藏  举报