scope/search/web/ddg.go

141 lines
3.6 KiB
Go
Raw Normal View History

/*
* Scope - A simple and minimal metasearch engine
* Copyright (C) 2021 Arsen Musayelyan
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
2021-12-08 17:24:05 +00:00
package web
import (
"net/http"
"strconv"
"strings"
"github.com/PuerkitoBio/goquery"
)
var ddgURL = urlMustParse("https://html.duckduckgo.com/html")
const uddgPrefix = "//duckduckgo.com/l/?uddg="
2021-12-08 21:18:14 +00:00
// DDG represents the DuckDuckGo search engine
2021-12-08 17:24:05 +00:00
type DDG struct {
keyword string
userAgent string
page int
doc *goquery.Document
initDone bool
baseSel *goquery.Selection
}
2021-12-08 21:18:14 +00:00
// SetKeyword sets the keyword for searching
2021-12-08 17:24:05 +00:00
func (d *DDG) SetKeyword(keyword string) {
d.keyword = keyword
}
2021-12-08 21:18:14 +00:00
// SetPage sets the page number for searching
2021-12-08 17:24:05 +00:00
func (d *DDG) SetPage(page int) {
d.page = page * 30
}
2021-12-08 21:18:14 +00:00
// SetUserAgent sets the user agent for the request
2021-12-08 17:24:05 +00:00
func (d *DDG) SetUserAgent(ua string) {
2021-12-08 21:18:14 +00:00
d.userAgent = ua
2021-12-08 17:24:05 +00:00
}
2021-12-08 21:18:14 +00:00
// Init runs requests for the DuckDuckGo search engine
2021-12-08 17:24:05 +00:00
func (d *DDG) Init() error {
2021-12-08 21:18:14 +00:00
// Copy URL so that it can be changed
2021-12-08 17:24:05 +00:00
initURL := copyURL(ddgURL)
2021-12-08 21:18:14 +00:00
// Get query parameters
2021-12-08 17:24:05 +00:00
query := initURL.Query()
2021-12-08 21:18:14 +00:00
// Set query
2021-12-08 17:24:05 +00:00
query.Set("q", d.keyword)
if d.page > 0 {
query.Set("s", strconv.Itoa(d.page))
query.Set("dc", strconv.Itoa(d.page+1))
}
2021-12-08 21:18:14 +00:00
// Update URL query
2021-12-08 17:24:05 +00:00
initURL.RawQuery = query.Encode()
2021-12-08 21:18:14 +00:00
// Create new request for modified URL
2021-12-08 17:24:05 +00:00
req, err := http.NewRequest(
http.MethodGet,
initURL.String(),
nil,
)
if err != nil {
return err
}
2021-12-08 21:18:14 +00:00
// If user agent empty, use default
2021-12-08 17:24:05 +00:00
if d.userAgent == "" {
d.userAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.169 Safari/537.36"
}
2021-12-08 21:18:14 +00:00
// Set user agent of request
2021-12-08 17:24:05 +00:00
req.Header.Set("User-Agent", d.userAgent)
2021-12-08 21:18:14 +00:00
// Perform request
2021-12-08 17:24:05 +00:00
res, err := http.DefaultClient.Do(req)
if err != nil {
return err
}
defer res.Body.Close()
2021-12-08 21:18:14 +00:00
// Create goquery document from reader
2021-12-08 17:24:05 +00:00
doc, err := goquery.NewDocumentFromReader(res.Body)
if err != nil {
return err
}
d.doc = doc
d.baseSel = doc.Find(`#links > .result`)
d.initDone = true
return nil
}
2021-12-08 21:18:14 +00:00
// Each runs eachCb with the index of each search result
2021-12-08 17:24:05 +00:00
func (d *DDG) Each(eachCb func(int) error) error {
for i := 0; i < d.baseSel.Length(); i++ {
err := eachCb(i)
if err != nil {
return err
}
}
return nil
}
2021-12-08 21:18:14 +00:00
// Title returns the title of the search result corresponding to i
2021-12-08 17:24:05 +00:00
func (d *DDG) Title(i int) (string, error) {
return strings.TrimSpace(get(d.baseSel, i).Children().First().ChildrenFiltered("h2").Text()), nil
}
2021-12-08 21:18:14 +00:00
// Link returns the link to the search result corresponding to i
2021-12-08 17:24:05 +00:00
func (d *DDG) Link(i int) (string, error) {
link := get(d.baseSel, i).Children().First().ChildrenFiltered("a").AttrOr("href", "")
if strings.HasPrefix(link, uddgPrefix) {
link = urlMustParse(link).Query().Get("uddg")
}
return link, nil
}
2021-12-08 21:18:14 +00:00
// Desc returns the description of the search result corresponding to i
2021-12-08 17:24:05 +00:00
func (d *DDG) Desc(i int) (string, error) {
return get(d.baseSel, i).Children().First().ChildrenFiltered("a").Text(), nil
}
2021-12-08 21:18:14 +00:00
// Name returns "ddg"
2021-12-08 17:24:05 +00:00
func (d *DDG) Name() string {
return "ddg"
}