Search/get-searchxng.go
2024-11-20 14:57:55 +01:00

137 lines
3.8 KiB
Go

package main
import (
"encoding/json"
"fmt"
"log"
"net/http"
"strings"
"sync"
"time"
)
const SEARX_SPACE_URL = "https://searx.space/data/instances.json"
type SearXInstance struct {
URL string `json:"url"`
Status int `json:"status_code"`
SSLGrade string `json:"grade"`
}
type SearXInstanceMetadata struct {
Timestamp int64 `json:"timestamp"`
}
type SearXInstanceResponse struct {
Metadata SearXInstanceMetadata `json:"metadata"`
Instances map[string]map[string]interface{} `json:"instances"`
}
var searxInstances []SearXInstance
var searxInstanceLastFetched time.Time
var searxInstanceFetchLock sync.Mutex
var backupInstances = []SearXInstance{
{URL: "https://searx.ox2.fr/", Status: 200, SSLGrade: "A+"},
{URL: "https://search.datura.network/", Status: 200, SSLGrade: "A+"},
{URL: "https://searx.foss.family/", Status: 200, SSLGrade: "A+"},
// Add more backup instances as needed
}
func fetchSearXInstances() error {
resp, err := http.Get(SEARX_SPACE_URL)
if err != nil {
return fmt.Errorf("error fetching SearX instances: %w", err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
return fmt.Errorf("unexpected status code: %d", resp.StatusCode)
}
var instanceResp SearXInstanceResponse
if err := json.NewDecoder(resp.Body).Decode(&instanceResp); err != nil {
return fmt.Errorf("error decoding SearX instance response: %w", err)
}
searxInstances = make([]SearXInstance, 0)
for url, instanceData := range instanceResp.Instances {
if httpData, ok := instanceData["http"].(map[string]interface{}); ok {
if status, ok := httpData["status_code"].(float64); ok && int(status) == 200 {
if grade, ok := httpData["grade"].(string); ok && grade != "" {
searxInstances = append(searxInstances, SearXInstance{
URL: url,
Status: int(status),
SSLGrade: grade,
})
}
}
}
}
log.Printf("Fetched %d SearX instances: %+v", len(searxInstances), searxInstances)
searxInstanceLastFetched = time.Now()
return nil
}
func getRandomSearXInstance() (SearXInstance, error) {
searxInstanceFetchLock.Lock()
defer searxInstanceFetchLock.Unlock()
if searxInstances == nil || time.Since(searxInstanceLastFetched) > 24*time.Hour {
if err := fetchSearXInstances(); err != nil {
log.Printf("Error fetching instances, using backup instances: %v", err)
searxInstances = backupInstances
}
}
if len(searxInstances) == 0 {
return SearXInstance{}, fmt.Errorf("no available SearX instances")
}
for _, instance := range searxInstances {
if isInstanceValid(instance) {
log.Printf("Selected SearX instance: %+v", instance)
return instance, nil
}
}
return SearXInstance{}, fmt.Errorf("no valid SearX instances found")
}
func isInstanceValid(instance SearXInstance) bool {
searchURL := fmt.Sprintf("%s/search?q=test&categories=general&language=en&safe_search=1&page=1&format=json", strings.TrimRight(instance.URL, "/"))
client := &http.Client{
Timeout: 10 * time.Second,
}
req, err := http.NewRequest("GET", searchURL, nil)
if err != nil {
log.Printf("Instance validation failed for URL: %s, Error: %v", searchURL, err)
return false
}
req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36")
resp, err := client.Do(req)
if err != nil {
log.Printf("Instance validation failed for URL: %s, Error: %v", searchURL, err)
return false
}
defer resp.Body.Close()
if resp.StatusCode == http.StatusOK {
return true
} else {
log.Printf("Instance validation failed for URL: %s, StatusCode: %d", searchURL, resp.StatusCode)
return false
}
}
// func main() {
// instance, err := getRandomSearXInstance()
// if err != nil {
// log.Fatalf("Failed to get a SearX instance: %v", err)
// }
// fmt.Printf("Selected SearX instance: %s\n", instance.URL)
// }