package frontend import ( "encoding/json" "fmt" "html/template" "io" "log" "net/http" "os" "path/filepath" "strconv" "strings" "time" "github.com/google/uuid" b "streifling.com/jason/cpolis/cmd/backend" ) const ( EditMode = iota PreviewMode ) type EditorHTMLData struct { Selected map[int64]bool Content string Action string ActionTitle string ActionButton string HTMLContent template.HTML Article *b.Article Tags []*b.Tag } func copyFile(src, dst string) error { srcFile, err := os.Open(src) if err != nil { return fmt.Errorf("error opening source file: %v", err) } defer srcFile.Close() dstFile, err := os.Create(dst) if err != nil { return fmt.Errorf("error opening destination file: %v", err) } defer dstFile.Close() _, err = io.Copy(dstFile, srcFile) if err != nil { return fmt.Errorf("error copying file: %v", err) } return dstFile.Sync() } func WriteArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } data := new(EditorHTMLData) if session.Values["article"] == nil { data = &EditorHTMLData{Article: new(b.Article)} } else { data = session.Values["article"].(*EditorHTMLData) } // data.Mode = EditMode data.Tags, err = db.GetTagList() if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Action = "submit" tmpl, err := template.ParseFiles(c.WebDir + "/templates/editor.html") template.Must(tmpl, err).ExecuteTemplate(w, "page-content", data) } } func SubmitArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } session.Values["article"] = nil if err = session.Save(r, w); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } article := &b.Article{ Title: r.PostFormValue("article-title"), Description: r.PostFormValue("article-description"), Published: false, Rejected: false, AuthorID: session.Values["id"].(int64), IsInIssue: r.PostFormValue("issue") == "on", AutoGenerated: false, } article.ID, err = db.AddArticle(article) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } articleAbsName := fmt.Sprint(c.ArticleDir, "/", article.ID, ".md") if err = os.WriteFile(articleAbsName, []byte(r.PostFormValue("article-content")), 0644); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } article.Link = fmt.Sprint(c.Domain, "/article/serve/", article.ID) if err = db.UpdateAttributes(&b.Attribute{Table: "articles", ID: article.ID, AttName: "link", Value: article.Link}); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } r.ParseForm() tags := make([]int64, 0) for _, tag := range r.Form["tags"] { tagID, err := strconv.ParseInt(tag, 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tags = append(tags, tagID) } if err = db.WriteArticleTags(article.ID, tags); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/hub.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", session.Values["role"]) } } func ResubmitArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } title := r.PostFormValue("article-title") description := r.PostFormValue("article-description") content := r.PostFormValue("article-content") link := fmt.Sprint(c.ArticleDir, "/", id, ".md") if err = os.WriteFile(link, []byte(content), 0644); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.UpdateAttributes( &b.Attribute{Table: "articles", ID: id, AttName: "title", Value: title}, &b.Attribute{Table: "articles", ID: id, AttName: "description", Value: description}, &b.Attribute{Table: "articles", ID: id, AttName: "rejected", Value: false}, &b.Attribute{Table: "articles", ID: id, AttName: "is_in_issue", Value: r.PostFormValue("issue") == "on"}, ); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } r.ParseForm() tags := make([]int64, 0) for _, tag := range r.Form["tags"] { tagID, err := strconv.ParseInt(tag, 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tags = append(tags, tagID) } if err = db.UpdateArticleTags(id, tags); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/hub.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", session.Values["role"]) } } func ShowUnpublishedUnrejectedAndPublishedRejectedArticles(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } rejectedArticles, err := db.GetCertainArticles("rejected", true) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } articles := make([]*b.Article, 0) for _, article := range rejectedArticles { if article.Published { articles = append(articles, article) } } unpublishedArticles, err := db.GetCertainArticles("published", false) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } for _, article := range unpublishedArticles { if !article.Rejected { articles = append(articles, article) } } tmpl, err := template.ParseFiles(c.WebDir + "/templates/unpublished-articles.html") template.Must(tmpl, err).ExecuteTemplate(w, "page-content", articles) } } func ShowRejectedArticles(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } data := new(struct { MyIDs map[int64]bool RejectedArticles []*b.Article }) data.RejectedArticles, err = db.GetCertainArticles("rejected", true) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.MyIDs = make(map[int64]bool) for _, article := range data.RejectedArticles { if article.AuthorID == session.Values["id"].(int64) { data.MyIDs[article.ID] = true } } tmpl, err := template.ParseFiles(c.WebDir + "/templates/rejected-articles.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", data) } } func ReviewRejectedArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data := new(EditorHTMLData) data.Article, err = db.GetArticle(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } articleAbsName := fmt.Sprint(c.ArticleDir, "/", data.Article.ID, ".md") content, err := os.ReadFile(articleAbsName) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Content = string(content) data.Tags, err = db.GetTagList() if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } selectedTags, err := db.GetArticleTags(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Selected = make(map[int64]bool) for _, tag := range selectedTags { data.Selected[tag.ID] = true } data.Action = fmt.Sprint("resubmit/", data.Article.ID) tmpl, err := template.ParseFiles(c.WebDir + "/templates/editor.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", data) } } func PublishArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusBadRequest) return } article, err := db.GetArticle(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.AddArticleToCurrentIssue(article.ID); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.UpdateAttributes( &b.Attribute{Table: "articles", ID: id, AttName: "published", Value: true}, &b.Attribute{Table: "articles", ID: id, AttName: "rejected", Value: false}, &b.Attribute{Table: "articles", ID: id, AttName: "created", Value: time.Now().Format("2006-01-02 15:04:05")}, ); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if article.EditedID != 0 { oldArticle, err := db.GetArticle(article.EditedID) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.DeleteArticle(oldArticle.ID); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = os.Remove(fmt.Sprint(c.ArticleDir, "/", oldArticle.ID, ".md")); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.UpdateAttributes( &b.Attribute{Table: "articles", ID: id, AttName: "link", Value: fmt.Sprint(c.Domain, "/article/serve/", article.ID)}, &b.Attribute{Table: "articles", ID: id, AttName: "edited_id", Value: 0}, ); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } } feed, err := b.GenerateRSS(c, db) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = b.SaveRSS(c.RSSFile, feed); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/hub.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", session.Values["role"]) } } func RejectArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.UpdateAttributes(&b.Attribute{Table: "articles", ID: id, AttName: "rejected", Value: true}); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/hub.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", session.Values["role"]) } } func ShowCurrentArticles(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } articles, err := db.GetCurrentIssueArticles() if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/current-articles.html") template.Must(tmpl, err).ExecuteTemplate(w, "page-content", articles) } } func UploadArticleImage(c *b.Config, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } file, header, err := r.FormFile("article-image") if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusBadRequest) return } defer file.Close() nameStrings := strings.Split(header.Filename, ".") extension := "." + nameStrings[len(nameStrings)-1] filename := fmt.Sprint(uuid.New(), extension) absFilepath, err := filepath.Abs(fmt.Sprint(c.PicsDir, "/", filename)) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } img, err := os.Create(absFilepath) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } defer img.Close() if _, err = io.Copy(img, file); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } url := fmt.Sprint(c.Domain, "/image/serve/", filename) w.Header().Set("Content-Type", "application/json") json.NewEncoder(w).Encode(url) } } func ShowPublishedArticles(c *b.Config, db *b.DB, s *b.CookieStore, action string) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } data := new(struct { Action string Articles []*b.Article }) data.Action = action publishedArticles, err := db.GetCertainArticles("published", true) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } for _, article := range publishedArticles { if !article.AutoGenerated { data.Articles = append(data.Articles, article) } } tmpl, err := template.ParseFiles(c.WebDir + "/templates/published-articles.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", data) } } func ReviewArticle(c *b.Config, db *b.DB, s *b.CookieStore, action, title, button string) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } article, err := db.GetArticle(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data := &EditorHTMLData{ Article: &b.Article{ ID: id, IsInIssue: article.IsInIssue, }, Action: action, ActionTitle: title, ActionButton: button, } data.Article.Title, err = b.ConvertToPlain(article.Title) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Article.Description, err = b.ConvertToPlain(article.Description) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } articleAbsName := fmt.Sprint(c.ArticleDir, "/", article.ID, ".md") content, err := os.ReadFile(articleAbsName) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Content, err = b.ConvertToHTML(string(content)) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.HTMLContent = template.HTML(data.Content) data.Tags, err = db.GetArticleTags(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/review-article.html") template.Must(tmpl, err).ExecuteTemplate(w, "page-content", data) } } func DeleteArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.DeleteArticle(id); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = os.Remove(fmt.Sprint(c.ArticleDir, "/", id, ".md")); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } feed, err := b.GenerateRSS(c, db) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = b.SaveRSS(c.RSSFile, feed); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/hub.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", session.Values["role"].(int)) } } func AllowEditArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { session, err := getSession(w, r, c, s) if err != nil { return } oldID, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } fmt.Println(oldID) oldArticle, err := db.GetArticle(oldID) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } newArticle := oldArticle newArticle.Published = false newArticle.Rejected = true newArticle.EditedID = oldArticle.ID newID, err := db.AddArticle(newArticle) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = db.UpdateAttributes(&b.Attribute{Table: "articles", ID: oldID, AttName: "edited_id", Value: newID}); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } if err = copyFile(fmt.Sprint(c.ArticleDir, "/", oldID, ".md"), fmt.Sprint(c.ArticleDir, "/", newID, ".md")); err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } tmpl, err := template.ParseFiles(c.WebDir + "/templates/hub.html") tmpl = template.Must(tmpl, err) tmpl.ExecuteTemplate(w, "page-content", session.Values["role"].(int)) } } func EditArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { if _, err := getSession(w, r, c, s); err != nil { return } id, err := strconv.ParseInt(r.PathValue("id"), 10, 64) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data := new(EditorHTMLData) data.Article, err = db.GetArticle(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } content, err := os.ReadFile(fmt.Sprint(c.ArticleDir, "/", data.Article.ID, ".md")) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Content = string(content) data.Tags, err = db.GetArticleTags(data.Article.ID) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } selectedTags, err := db.GetArticleTags(id) if err != nil { log.Println(err) http.Error(w, err.Error(), http.StatusInternalServerError) return } data.Selected = make(map[int64]bool) for _, tag := range selectedTags { data.Selected[tag.ID] = true } data.Action = fmt.Sprint("save/", data.Article.ID) tmpl, err := template.ParseFiles(c.WebDir + "/templates/editor.html") template.Must(tmpl, err).ExecuteTemplate(w, "page-content", data) } }