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) // }