added libreY/X fix for pirate bay
This commit is contained in:
parent
8e5f8c8a10
commit
7d1d2cba67
7 changed files with 325 additions and 16 deletions
137
get-searchxng.go
Normal file
137
get-searchxng.go
Normal file
|
@ -0,0 +1,137 @@
|
|||
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)
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue