ip-checker/main.go

126 lines
2.7 KiB
Go
Raw Normal View History

2024-08-26 22:01:24 +00:00
package main
import (
"fmt"
"html/template"
"log"
2024-10-19 08:04:28 +00:00
"net"
2024-08-26 22:01:24 +00:00
"net/http"
"os"
"strings"
2024-10-19 08:04:28 +00:00
"sync"
"github.com/oschwald/geoip2-golang"
)
var (
db *geoip2.Reader
tmpl *template.Template
config ServerConfig
ipCache sync.Map
2024-08-26 22:01:24 +00:00
)
type ServerConfig struct {
2024-10-19 08:04:28 +00:00
listen string
countrydb string
2024-08-26 22:01:24 +00:00
}
func main() {
2024-10-19 08:04:28 +00:00
config = ServerConfig{
listen: getEnvOr("IP_CHECKER_LISTEN", ":8080"),
countrydb: os.Getenv("IP_CHECKER_COUNTRY_DB"),
}
if config.countrydb != "" {
var err error
db, err = geoip2.Open(config.countrydb)
if err != nil {
log.Fatalf("Error opening GeoIP database: %v", err)
}
defer db.Close()
2024-08-26 22:01:24 +00:00
}
2024-10-19 08:04:28 +00:00
tmpl = template.Must(template.ParseFiles("assets/index.html"))
2024-08-26 22:01:24 +00:00
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("./assets/static"))))
2024-10-19 08:04:28 +00:00
http.HandleFunc("/{$}", handleRequest)
2024-08-26 22:01:24 +00:00
2024-10-19 04:19:03 +00:00
http.HandleFunc("/robots.txt", func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, `User-Agent: *
Disallow: /harming/humans
Disallow: /ignoring/human/orders
Disallow: /harm/to/self
`)
})
2024-08-26 22:01:24 +00:00
log.Printf("Starting server on %s", config.listen)
log.Fatal(http.ListenAndServe(config.listen, nil))
}
2024-10-19 08:04:28 +00:00
func getIPCountry(ip string) string {
if config.countrydb == "" {
return "unknown"
}
if country, found := ipCache.Load(ip); found {
return country.(string)
}
record, err := db.Country(net.ParseIP(ip))
if err != nil {
log.Print(err)
return "an error occurred"
}
country := record.Country.Names["en"]
ipCache.Store(ip, country) // Cache the result
return country
}
func handleRequest(w http.ResponseWriter, r *http.Request) {
2024-10-19 05:28:17 +00:00
sourceIP := func() string {
2024-10-19 08:04:28 +00:00
if ip := r.Header.Get("X-Forwarded-For"); ip != "" {
2024-10-19 08:18:59 +00:00
return strings.Split(ip, ",")[0]
2024-10-19 05:28:17 +00:00
}
2024-10-19 05:33:40 +00:00
return strings.Split(r.RemoteAddr, ":")[0]
2024-10-19 05:28:17 +00:00
}()
2024-08-26 22:01:24 +00:00
isHeadless := strings.HasPrefix(r.Header.Get("User-Agent"), "curl/")
log.Printf("r.URL.Path: %s, sourceIP: %s, isHeadless: %t, User-Agent: %s", r.URL.Path, sourceIP, isHeadless, r.Header.Get("User-Agent"))
2024-10-19 04:19:03 +00:00
if isHeadless {
fmt.Fprintf(w, "%s\n", sourceIP)
} else {
csp := []string{
"default-src 'none'",
"img-src 'self'",
"script-src-elem 'self'",
"style-src-elem 'self' fonts.googleapis.com",
"font-src fonts.gstatic.com",
"connect-src api.ip.sb api-v3.speedtest.cn api.ipapi.is",
2024-08-26 22:01:24 +00:00
}
2024-10-19 04:19:03 +00:00
w.Header().Set("Content-Security-Policy", strings.Join(csp, "; "))
2024-08-26 22:01:24 +00:00
2024-10-19 04:19:03 +00:00
data := map[string]string{
2024-10-19 08:04:28 +00:00
"sourceIP": sourceIP,
"sourceCountry": getIPCountry(sourceIP),
2024-10-19 04:19:03 +00:00
}
2024-08-26 22:01:24 +00:00
2024-10-19 04:19:03 +00:00
err := tmpl.Execute(w, data)
if err != nil {
http.Error(w, "Unable to load template", http.StatusInternalServerError)
log.Printf("Template execution error: %v", err)
return
}
2024-08-26 22:01:24 +00:00
}
}
func getEnvOr(key, defaultValue string) string {
if value, exists := os.LookupEnv(key); exists {
return value
}
return defaultValue
}