package main import ( "fmt" "html/template" "log" "os" "strconv" "time" "github.com/akhilrex/podgrab/controllers" "github.com/akhilrex/podgrab/db" "github.com/akhilrex/podgrab/service" "github.com/gin-contrib/location" "github.com/gin-gonic/gin" "github.com/jasonlvhit/gocron" _ "github.com/joho/godotenv/autoload" ) func main() { var err error db.DB, err = db.Init() if err != nil { fmt.Println("statuse: ", err) } else { db.Migrate() } r := gin.Default() r.Use(setupSettings()) r.Use(gin.Recovery()) r.Use(location.Default()) funcMap := template.FuncMap{ "intRange": func(start, end int) []int { n := end - start + 1 result := make([]int, n) for i := 0; i < n; i++ { result[i] = start + i } return result }, "removeStartingSlash": func(raw string) string { fmt.Println(raw) if string(raw[0]) == "/" { return raw } return "/" + raw }, "isDateNull": func(raw time.Time) bool { return raw == (time.Time{}) }, "formatDate": func(raw time.Time) string { if raw == (time.Time{}) { return "" } return raw.Format("Jan 2 2006") }, "naturalDate": func(raw time.Time) string { return service.NatualTime(time.Now(), raw) //return raw.Format("Jan 2 2006") }, "latestEpisodeDate": func(podcastItems []db.PodcastItem) string { var latest time.Time for _, item := range podcastItems { if item.PubDate.After(latest) { latest = item.PubDate } } return latest.Format("Jan 2 2006") }, "downloadedEpisodes": func(podcastItems []db.PodcastItem) int { count := 0 for _, item := range podcastItems { if item.DownloadStatus == db.Downloaded { count++ } } return count }, "downloadingEpisodes": func(podcastItems []db.PodcastItem) int { count := 0 for _, item := range podcastItems { if item.DownloadStatus == db.NotDownloaded { count++ } } return count }, "formatFileSize": func(inputSize int64) string { size := float64(inputSize) const divisor float64 = 1024 if size < divisor { return fmt.Sprintf("%.0f bytes", size) } size = size / divisor if size < divisor { return fmt.Sprintf("%.2f KB", size) } size = size / divisor if size < divisor { return fmt.Sprintf("%.2f MB", size) } size = size / divisor if size < divisor { return fmt.Sprintf("%.2f GB", size) } size = size / divisor return fmt.Sprintf("%.2f TB", size) }, "formatDuration": func(total int) string { if total <= 0 { return "" } mins := total / 60 secs := total % 60 hrs := 0 if mins >= 60 { hrs = mins / 60 mins = mins % 60 } if hrs > 0 { return fmt.Sprintf("%02d:%02d:%02d", hrs, mins, secs) } return fmt.Sprintf("%02d:%02d", mins, secs) }, } tmpl := template.Must(template.New("main").Funcs(funcMap).ParseGlob("client/*")) //r.LoadHTMLGlob("client/*") r.SetHTMLTemplate(tmpl) pass := os.Getenv("PASSWORD") var router *gin.RouterGroup if pass != "" { router = r.Group("/", gin.BasicAuth(gin.Accounts{ "podgrab": pass, })) } else { router = &r.RouterGroup } dataPath := os.Getenv("DATA") router.Static("/webassets", "./webassets") router.Static("/assets", dataPath) router.POST("/podcasts", controllers.AddPodcast) router.GET("/podcasts", controllers.GetAllPodcasts) router.GET("/podcasts/:id", controllers.GetPodcastById) router.DELETE("/podcasts/:id", controllers.DeletePodcastById) router.GET("/podcasts/:id/items", controllers.GetPodcastItemsByPodcastId) router.GET("/podcasts/:id/download", controllers.DownloadAllEpisodesByPodcastId) router.DELETE("/podcasts/:id/items", controllers.DeletePodcastEpisodesById) router.DELETE("/podcasts/:id/podcast", controllers.DeleteOnlyPodcastById) router.GET("/podcastitems", controllers.GetAllPodcastItems) router.GET("/podcastitems/:id", controllers.GetPodcastItemById) router.GET("/podcastitems/:id/image", controllers.GetPodcastItemImageById) router.GET("/podcastitems/:id/file", controllers.GetPodcastItemFileById) router.GET("/podcastitems/:id/markUnplayed", controllers.MarkPodcastItemAsUnplayed) router.GET("/podcastitems/:id/markPlayed", controllers.MarkPodcastItemAsPlayed) router.GET("/podcastitems/:id/bookmark", controllers.BookmarkPodcastItem) router.GET("/podcastitems/:id/unbookmark", controllers.UnbookmarkPodcastItem) router.PATCH("/podcastitems/:id", controllers.PatchPodcastItemById) router.GET("/podcastitems/:id/download", controllers.DownloadPodcastItem) router.GET("/podcastitems/:id/delete", controllers.DeletePodcastItem) router.GET("/tags", controllers.GetAllTags) router.GET("/tags/:id", controllers.GetTagById) router.DELETE("/tags/:id", controllers.DeleteTagById) router.POST("/tags", controllers.AddTag) router.POST("/podcasts/:id/tags/:tagId", controllers.AddTagToPodcast) router.DELETE("/podcasts/:id/tags/:tagId", controllers.RemoveTagFromPodcast) router.GET("/add", controllers.AddPage) router.GET("/search", controllers.Search) router.GET("/", controllers.HomePage) router.GET("/podcasts/:id/view", controllers.PodcastPage) router.GET("/episodes", controllers.AllEpisodesPage) router.GET("/allTags", controllers.AllTagsPage) router.GET("/settings", controllers.SettingsPage) router.POST("/settings", controllers.UpdateSetting) router.GET("/backups", controllers.BackupsPage) router.POST("/opml", controllers.UploadOpml) router.GET("/opml", controllers.GetOmpl) router.GET("/player", controllers.PlayerPage) r.GET("/ws", func(c *gin.Context) { controllers.Wshandler(c.Writer, c.Request) }) go controllers.HandleWebsocketMessages() go assetEnv() go intiCron() r.Run() // listen and serve on 0.0.0.0:8080 (for windows "localhost:8080") } func setupSettings() gin.HandlerFunc { return func(c *gin.Context) { setting := db.GetOrCreateSetting() c.Set("setting", setting) c.Writer.Header().Set("X-Clacks-Overhead", "GNU Terry Pratchett") c.Next() } } func intiCron() { checkFrequency, err := strconv.Atoi(os.Getenv("CHECK_FREQUENCY")) if err != nil { checkFrequency = 30 log.Print(err) } service.UnlockMissedJobs() //gocron.Every(uint64(checkFrequency)).Minutes().Do(service.DownloadMissingEpisodes) gocron.Every(uint64(checkFrequency)).Minutes().Do(service.RefreshEpisodes) gocron.Every(uint64(checkFrequency)).Minutes().Do(service.CheckMissingFiles) gocron.Every(uint64(checkFrequency) * 2).Minutes().Do(service.UnlockMissedJobs) gocron.Every(uint64(checkFrequency) * 3).Minutes().Do(service.UpdateAllFileSizes) gocron.Every(uint64(checkFrequency)).Minutes().Do(service.DownloadMissingImages) gocron.Every(2).Days().Do(service.CreateBackup) <-gocron.Start() } func assetEnv() { log.Println("Config Dir: ", os.Getenv("CONFIG")) log.Println("Assets Dir: ", os.Getenv("DATA")) log.Println("Check Frequency (mins): ", os.Getenv("CHECK_FREQUENCY")) }