`
for i := range posts {
err := json.Unmarshal(posts[i], &p)
if err != nil {
log.Println("Error unmarshal json into", t, err, posts[i])
post := `
Error decoding data. Possible file corruption.
`
b.Write([]byte(post))
continue
}
post := adminPostListItem(p, t, status)
b.Write(post)
}
case "pending":
// get _pending posts of type t from the db
posts = db.Query(t+"_pending", opts)
html += `
`
html = html + b.String() + script + btn
adminView, err := Admin([]byte(html))
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
return
}
res.Header().Set("Content-Type", "text/html")
res.Write(adminView)
}
// adminPostListItem is a helper to create the li containing a post.
// p is the asserted post as an Editable, t is the Type of the post.
// specifier is passed to append a name to a namespace like _pending
func adminPostListItem(e editor.Editable, typeName, status string) []byte {
s, ok := e.(editor.Sortable)
if !ok {
log.Println("Content type", typeName, "doesn't implement editor.Sortable")
post := `
Error retreiving data. Your data type doesn't implement necessary interfaces.
`
return []byte(post)
}
// use sort to get other info to display in admin UI post list
tsTime := time.Unix(int64(s.Time()/1000), 0)
upTime := time.Unix(int64(s.Touch()/1000), 0)
updatedTime := upTime.Format("01/02/06 03:04 PM")
publishTime := tsTime.Format("01/02/06")
cid := fmt.Sprintf("%d", s.ItemID())
switch status {
case "public", "":
status = ""
default:
status = "_" + status
}
post := `
`
return []byte(post)
}
func approvePostHandler(res http.ResponseWriter, req *http.Request) {
if req.Method != http.MethodPost {
res.WriteHeader(http.StatusMethodNotAllowed)
errView, err := Error405()
if err != nil {
return
}
res.Write(errView)
return
}
err := req.ParseMultipartForm(1024 * 1024 * 4) // maxMemory 4MB
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
t := req.FormValue("type")
if strings.Contains(t, "_") {
t = strings.Split(t, "_")[0]
}
post := content.Types[t]()
// run hooks
hook, ok := post.(content.Hookable)
if !ok {
log.Println("Type", t, "does not implement content.Hookable or embed content.Item.")
res.WriteHeader(http.StatusBadRequest)
errView, err := Error400()
if err != nil {
return
}
res.Write(errView)
return
}
// check if we have a Mergeable
m, ok := post.(api.Mergeable)
if !ok {
log.Println("Content type", t, "must implement api.Mergable before it can bee approved.")
res.WriteHeader(http.StatusBadRequest)
errView, err := Error400()
if err != nil {
return
}
res.Write(errView)
return
}
dec := schema.NewDecoder()
dec.IgnoreUnknownKeys(true)
dec.SetAliasTag("json")
err = dec.Decode(post, req.Form)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
err = hook.BeforeApprove(req)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
// call its Approve method
err = m.Approve(req)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
err = hook.AfterApprove(req)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
err = hook.BeforeSave(req)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
// Store the content in the bucket t
id, err := db.SetContent(t+":-1", req.Form)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
err = hook.AfterSave(req)
if err != nil {
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
// redirect to the new approved content's editor
redir := req.URL.Scheme + req.URL.Host + strings.TrimSuffix(req.URL.Path, "/approve")
redir += fmt.Sprintf("?type=%s&id=%d", t, id)
http.Redirect(res, req, redir, http.StatusFound)
}
func editHandler(res http.ResponseWriter, req *http.Request) {
switch req.Method {
case http.MethodGet:
q := req.URL.Query()
i := q.Get("id")
t := q.Get("type")
status := q.Get("status")
contentType, ok := content.Types[t]
if !ok {
fmt.Fprintf(res, content.ErrTypeNotRegistered, t)
return
}
post := contentType()
if i != "" {
if status == "pending" {
t = t + "_pending"
}
data, err := db.Content(t + ":" + i)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
if len(data) < 1 || data == nil {
fmt.Println(string(data))
res.WriteHeader(http.StatusNotFound)
errView, err := Error404()
if err != nil {
return
}
res.Write(errView)
return
}
err = json.Unmarshal(data, post)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
} else {
s, ok := post.(content.Identifiable)
if !ok {
log.Println("Content type", t, "doesn't implement editor.Identifiable")
return
}
s.SetItemID(-1)
}
m, err := manager.Manage(post.(editor.Editable), t)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
adminView, err := Admin(m)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
return
}
res.Header().Set("Content-Type", "text/html")
res.Write(adminView)
case http.MethodPost:
err := req.ParseMultipartForm(1024 * 1024 * 4) // maxMemory 4MB
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusBadRequest)
errView, err := Error405()
if err != nil {
return
}
res.Write(errView)
return
}
cid := req.FormValue("id")
t := req.FormValue("type")
ts := req.FormValue("timestamp")
up := req.FormValue("updated")
// create a timestamp if one was not set
if ts == "" {
ts := fmt.Sprintf("%d", time.Now().Unix()*1000)
req.PostForm.Set("timestamp", ts)
}
if up == "" {
req.PostForm.Set("updated", ts)
}
urlPaths, err := upload.StoreFiles(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
for name, urlPath := range urlPaths {
req.PostForm.Add(name, urlPath)
}
// check for any multi-value fields (ex. checkbox fields)
// and correctly format for db storage. Essentially, we need
// fieldX.0: value1, fieldX.1: value2 => fieldX: []string{value1, value2}
var discardKeys []string
for k, v := range req.PostForm {
if strings.Contains(k, ".") {
key := strings.Split(k, ".")[0]
if req.PostForm.Get(key) == "" {
req.PostForm.Set(key, v[0])
discardKeys = append(discardKeys, k)
} else {
req.PostForm.Add(key, v[0])
}
}
}
for _, discardKey := range discardKeys {
req.PostForm.Del(discardKey)
}
if strings.Contains(t, "_") {
t = strings.Split(t, "_")[0]
}
p, ok := content.Types[t]
if !ok {
log.Println("Type", t, "is not a content type. Cannot edit or save.")
res.WriteHeader(http.StatusBadRequest)
errView, err := Error400()
if err != nil {
return
}
res.Write(errView)
return
}
post := p()
hook, ok := post.(content.Hookable)
if !ok {
log.Println("Type", t, "does not implement content.Hookable or embed content.Item.")
res.WriteHeader(http.StatusBadRequest)
errView, err := Error400()
if err != nil {
return
}
res.Write(errView)
return
}
err = hook.BeforeSave(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
id, err := db.SetContent(t+":"+cid, req.PostForm)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
err = hook.AfterSave(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
scheme := req.URL.Scheme
host := req.URL.Host
path := req.URL.Path
sid := fmt.Sprintf("%d", id)
redir := scheme + host + path + "?type=" + t + "&id=" + sid
if req.URL.Query().Get("status") == "pending" {
redir += "&status=pending"
}
http.Redirect(res, req, redir, http.StatusFound)
default:
res.WriteHeader(http.StatusMethodNotAllowed)
}
}
func deleteHandler(res http.ResponseWriter, req *http.Request) {
if req.Method != http.MethodPost {
res.WriteHeader(http.StatusMethodNotAllowed)
return
}
err := req.ParseMultipartForm(1024 * 1024 * 4) // maxMemory 4MB
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
return
}
id := req.FormValue("id")
t := req.FormValue("type")
ct := t
if id == "" || t == "" {
res.WriteHeader(http.StatusBadRequest)
return
}
// catch specifier suffix from delete form value
if strings.Contains(t, "_") {
spec := strings.Split(t, "_")
ct = spec[0]
}
p, ok := content.Types[ct]
if !ok {
log.Println("Type", t, "does not implement content.Hookable or embed content.Item.")
res.WriteHeader(http.StatusBadRequest)
errView, err := Error400()
if err != nil {
return
}
res.Write(errView)
return
}
post := p()
hook, ok := post.(content.Hookable)
if !ok {
log.Println("Type", t, "does not implement content.Hookable or embed content.Item.")
res.WriteHeader(http.StatusBadRequest)
errView, err := Error400()
if err != nil {
return
}
res.Write(errView)
return
}
reject := req.URL.Query().Get("reject")
if reject == "true" {
err = hook.BeforeReject(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
}
err = hook.BeforeDelete(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
err = db.DeleteContent(t + ":" + id)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
return
}
err = hook.AfterDelete(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
if reject == "true" {
err = hook.AfterReject(req)
if err != nil {
log.Println(err)
res.WriteHeader(http.StatusInternalServerError)
errView, err := Error500()
if err != nil {
return
}
res.Write(errView)
return
}
}
redir := strings.TrimSuffix(req.URL.Scheme+req.URL.Host+req.URL.Path, "/edit/delete")
redir = redir + "/posts?type=" + ct
http.Redirect(res, req, redir, http.StatusFound)
}
func editUploadHandler(res http.ResponseWriter, req *http.Request) {
if req.Method != http.MethodPost {
res.WriteHeader(http.StatusMethodNotAllowed)
return
}
urlPaths, err := upload.StoreFiles(req)
if err != nil {
log.Println("Couldn't store file uploads.", err)
res.WriteHeader(http.StatusInternalServerError)
return
}
res.Header().Set("Content-Type", "application/json")
res.Write([]byte(`{"data": [{"url": "` + urlPaths["file"] + `"}]}`))
}
func searchHandler(res http.ResponseWriter, req *http.Request) {
q := req.URL.Query()
t := q.Get("type")
search := q.Get("q")
if t == "" || search == "" {
http.Redirect(res, req, req.URL.Scheme+req.URL.Host+"/admin", http.StatusFound)
return
}
posts := db.ContentAll(t)
b := &bytes.Buffer{}
p := content.Types[t]().(editor.Editable)
html := `
` + t + ` Results
`
for i := range posts {
// skip posts that don't have any matching search criteria
match := strings.ToLower(search)
all := strings.ToLower(string(posts[i]))
if !strings.Contains(all, match) {
continue
}
err := json.Unmarshal(posts[i], &p)
if err != nil {
log.Println("Error unmarshal search result json into", t, err, posts[i])
post := `