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()
dataPath := os.Getenv("DATA")
r.Static("/webassets", "./webassets")
r.Static("/assets", dataPath)
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
}
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"))
}