You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

217 lines
4.8 KiB
Go

package main
import (
"context"
"fmt"
"log"
"os"
"os/signal"
"syscall"
"time"
"tzgyn/kon-redirect/db"
"github.com/gofiber/fiber/v2"
"github.com/gofiber/fiber/v2/middleware/limiter"
"github.com/gofiber/fiber/v2/middleware/logger"
"github.com/jackc/pgx/v5"
"github.com/jackc/pgx/v5/pgtype"
"github.com/joho/godotenv"
"github.com/mileusna/useragent"
"github.com/oschwald/geoip2-golang"
"github.com/patrickmn/go-cache"
"github.com/robfig/cron/v3"
"github.com/ua-parser/uap-go/uaparser"
)
func main() {
err := godotenv.Load()
if err != nil {
fmt.Println("No .env found")
}
parser, err := uaparser.New("./regexes.yaml")
if err != nil {
log.Fatal(err)
}
fmt.Println("Initializing GeoLite2 DB...")
_, err = os.Stat("./data/GeoLite2-City.mmdb")
if err != nil {
fmt.Println("GeoLite2 DB Not Found...")
err := downloadAndExtractDB()
if err != nil {
log.Fatal(err)
}
}
geodb, err := geoip2.Open("./data/GeoLite2-City.mmdb")
if err != nil {
log.Fatal("Failed to initialize GeoLite2 DB")
}
fmt.Println("Finished initializing GeoLite2 DB")
fmt.Println("Initializing postgres DB and cache...")
ctx := context.Background()
c := cache.New(1*time.Hour, 5*time.Minute)
c.Set("key", "value", cache.DefaultExpiration)
dbUrl := os.Getenv("DATABASE_URL")
if len(dbUrl) == 0 {
log.Fatal("DATABASE_URL not found")
}
conn, err := pgx.Connect(ctx, dbUrl)
if err != nil {
log.Fatal(err)
}
queries := db.New(conn)
fmt.Println("Finished initializing postgres DB and cache")
cache_client := cache.New(1*time.Hour, 5*time.Minute)
app := fiber.New()
app.Use(logger.New())
app.Use(limiter.New(limiter.Config{
Max: 20,
Expiration: 1 * time.Minute,
}))
app.Static("/", "./public")
fallbackurl := os.Getenv("FALLBACK_URL")
if len(fallbackurl) == 0 {
fallbackurl = "https://app.kon.sh"
}
appurl := os.Getenv("APP_URL")
if len(appurl) == 0 {
appurl = "kon.sh"
}
app.Get("/", func(c *fiber.Ctx) error {
return c.Redirect(fallbackurl)
})
app.Get("/:code", func(c *fiber.Ctx) error {
code := c.Params("code")
domain := c.Hostname()
uastring := c.GetReqHeaders()["User-Agent"][0]
ua := useragent.Parse(uastring)
client := parser.Parse(uastring)
redirecturl := ""
var shortenerId int32
if domain == appurl {
shortener, err := queries.GetShortener(ctx, code)
shortenerId = shortener.ID
if err != nil {
return c.Redirect(fallbackurl)
}
if ua.OS == "iOS" && shortener.Ios && len(shortener.IosLink.String) != 0 {
redirecturl = shortener.IosLink.String
} else if ua.OS == "Android" && shortener.Android && len(shortener.AndroidLink.String) != 0 {
redirecturl = shortener.AndroidLink.String
} else {
redirecturl = shortener.Link
}
} else {
shortener, err := queries.GetShortenerWithDomain(ctx, db.GetShortenerWithDomainParams{
Code: code,
CustomDomain: pgtype.Text{String: domain, Valid: true},
})
shortenerId = shortener.ID
if err != nil {
return c.Redirect(fallbackurl)
}
if ua.OS == "iOS" && shortener.Ios && len(shortener.IosLink.String) != 0 {
redirecturl = shortener.IosLink.String
} else if ua.OS == "Android" && shortener.Android && len(shortener.AndroidLink.String) != 0 {
redirecturl = shortener.AndroidLink.String
} else {
redirecturl = shortener.Link
}
}
ip := c.IPs()[0]
_, found := cache_client.Get(ip + "_" + string(shortenerId))
if found {
return c.Redirect(redirecturl)
}
cache_client.Set(ip+"_"+string(shortenerId), true, cache.DefaultExpiration)
record, err := getCity(geodb, ip)
if err != nil {
return c.Redirect(redirecturl)
}
devicetype := ""
if ua.Mobile {
devicetype = "mobile"
} else if ua.Tablet {
devicetype = "tablet"
} else if ua.Desktop {
devicetype = "desktop"
}
err = queries.CreateVisitor(ctx, db.CreateVisitorParams{
ShortenerID: shortenerId,
DeviceType: devicetype,
DeviceVendor: client.Device.Brand,
Browser: client.UserAgent.Family,
Os: client.Os.Family,
Country: record.Country.Names["en"],
CountryCode: record.Country.IsoCode,
City: record.City.Names["en"],
})
if err != nil {
return c.Redirect(redirecturl)
}
return c.Redirect(redirecturl)
})
cron := cron.New()
cron.AddFunc("@weekly", func() {
fmt.Println("Updating GeoLite2 DB...")
err := downloadAndExtractDB()
if err != nil {
log.Fatal(err)
}
fmt.Println("Finished updating GeoLite2 DB")
})
cron.Start()
done := make(chan os.Signal, 1)
signal.Notify(done, syscall.SIGINT, syscall.SIGTERM, os.Interrupt)
go func() {
<-done
fmt.Println("Gracefully shutting down...")
_ = app.Shutdown()
}()
if err := app.Listen(":3000"); err != nil {
log.Panic(err)
}
fmt.Println("Running cleanup tasks...")
fmt.Println("Stopping cronjobs...")
cron.Stop()
geodb.Close()
conn.Close(ctx)
}