234 lines
4.8 KiB
Go
234 lines
4.8 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
|
|
log "github.com/Sirupsen/logrus"
|
|
haikunator "github.com/atrox/haikunatorgo"
|
|
"github.com/gin-contrib/sessions"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/msoedov/hacker-slides/files"
|
|
)
|
|
|
|
const sessionHeader = "slide-session"
|
|
|
|
func Header(c *gin.Context, key string) string {
|
|
if values, _ := c.Request.Header[key]; len(values) > 0 {
|
|
return values[0]
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func BasicAuth() gin.HandlerFunc {
|
|
realm := "Authorization Required"
|
|
realm = "Basic realm=" + strconv.Quote(realm)
|
|
user := os.Getenv("USER")
|
|
password := os.Getenv("PASSWORD")
|
|
enabled := isEnabled(user, password)
|
|
if enabled {
|
|
log.Warn("Auth mode enabled")
|
|
log.Warn(fmt.Sprintf("Visit http://%s:%s@0.0.0.0:8080", user, password))
|
|
}
|
|
return func(c *gin.Context) {
|
|
header := Header(c, "Authorization")
|
|
if enabled && header != authorizationHeader(user, password) {
|
|
// Credentials doesn't match, we return 401 and abort handlers chain.
|
|
c.Header("WWW-Authenticate", realm)
|
|
c.AbortWithStatus(401)
|
|
return
|
|
}
|
|
c.Next()
|
|
}
|
|
}
|
|
|
|
func isEnabled(user, password string) bool {
|
|
switch {
|
|
case user == "":
|
|
return false
|
|
case password == "":
|
|
return false
|
|
default:
|
|
return true
|
|
}
|
|
}
|
|
|
|
func authorizationHeader(user, password string) string {
|
|
base := user + ":" + password
|
|
return "Basic " + base64.StdEncoding.EncodeToString([]byte(base))
|
|
}
|
|
|
|
func NewApp() *gin.Engine {
|
|
|
|
r := gin.Default()
|
|
|
|
store := sessions.NewCookieStore([]byte("secret"))
|
|
r.Use(sessions.Sessions(sessionHeader, store))
|
|
r.Use(BasicAuth())
|
|
|
|
r.LoadHTMLGlob("templates/*.tmpl")
|
|
r.Static("/static", "./static")
|
|
|
|
r.GET("/", func(c *gin.Context) {
|
|
|
|
latest := files.LatestFileIn("slides")
|
|
log.WithFields(log.Fields{
|
|
"name": latest,
|
|
}).Info("Restoring latest point")
|
|
|
|
var path string
|
|
if latest == "" {
|
|
haikunator := haikunator.New()
|
|
haikunator.TokenLength = 0
|
|
name := haikunator.Haikunate()
|
|
path = fmt.Sprintf("slides/%s.md", name)
|
|
} else {
|
|
name := latest
|
|
path = fmt.Sprintf("slides/%s", name)
|
|
|
|
}
|
|
|
|
log.WithFields(log.Fields{
|
|
"path": path,
|
|
}).Info("A new session")
|
|
session := sessions.Default(c)
|
|
session.Set("name", path)
|
|
session.Save()
|
|
|
|
c.HTML(200, "index.tmpl", gin.H{
|
|
"pubTo": path,
|
|
})
|
|
})
|
|
|
|
mustHaveSession := func(c *gin.Context) (string, error) {
|
|
session := sessions.Default(c)
|
|
val := session.Get("name")
|
|
emptySession := errors.New("Emtpy session")
|
|
if val == nil {
|
|
c.String(400, "No context")
|
|
return "", emptySession
|
|
}
|
|
log.WithFields(log.Fields{
|
|
"path": val,
|
|
}).Info("Got session")
|
|
path, ok := val.(string)
|
|
if !ok {
|
|
c.String(400, "No context")
|
|
return "", emptySession
|
|
}
|
|
return path, nil
|
|
}
|
|
|
|
r.GET("/slides.md", func(c *gin.Context) {
|
|
path, err := mustHaveSession(c)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if _, err := os.Stat(path); err != nil {
|
|
// copy sample markdown file to the path
|
|
body, err := ioutil.ReadFile("initial-slides.md")
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
ioutil.WriteFile(path, body, 0644)
|
|
c.String(200, string(body))
|
|
return
|
|
}
|
|
|
|
body, err := ioutil.ReadFile(path)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
c.String(200, string(body))
|
|
})
|
|
|
|
r.PUT("/slides.md", func(c *gin.Context) {
|
|
path, err := mustHaveSession(c)
|
|
if err != nil {
|
|
return
|
|
}
|
|
body, _ := ioutil.ReadAll(c.Request.Body)
|
|
ioutil.WriteFile(path, body, 0644)
|
|
log.WithFields(log.Fields{
|
|
"size": len(body),
|
|
"file": path,
|
|
}).Info("Wrote to file")
|
|
c.String(200, "")
|
|
})
|
|
|
|
r.GET("/stash", func(c *gin.Context) {
|
|
files, err := ioutil.ReadDir("slides")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
var stash []string
|
|
for _, file := range files {
|
|
stash = append(stash, file.Name())
|
|
}
|
|
c.HTML(200, "stash.tmpl", gin.H{
|
|
"stash": stash,
|
|
})
|
|
})
|
|
|
|
r.GET("/stash/edit/:name", func(c *gin.Context) {
|
|
|
|
name := c.Param("name")
|
|
log.WithFields(log.Fields{
|
|
"name": name,
|
|
}).Info("Restore session?")
|
|
|
|
if strings.HasSuffix(name, ".md") {
|
|
name = name[0 : len(name)-3]
|
|
}
|
|
path := fmt.Sprintf("slides/%s.md", name)
|
|
session := sessions.Default(c)
|
|
session.Set("name", path)
|
|
session.Save()
|
|
|
|
c.HTML(200, "index.tmpl", gin.H{
|
|
"pubTo": path,
|
|
})
|
|
})
|
|
|
|
r.GET("/published/slides/:name", func(c *gin.Context) {
|
|
|
|
name := c.Param("name")
|
|
log.WithFields(log.Fields{
|
|
"name": name,
|
|
}).Info("Published")
|
|
|
|
if strings.HasSuffix(name, ".md") {
|
|
name = name[0 : len(name)-3]
|
|
}
|
|
path := fmt.Sprintf("slides/%s.md", name)
|
|
session := sessions.Default(c)
|
|
session.Set("name", path)
|
|
session.Save()
|
|
c.HTML(200, "slides.tmpl", gin.H{
|
|
"pubTo": path,
|
|
})
|
|
})
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
func main() {
|
|
r := NewApp()
|
|
port := "8080"
|
|
if len(os.Args) > 1 {
|
|
port = os.Args[1]
|
|
} else {
|
|
envPort := os.Getenv("PORT")
|
|
if len(envPort) > 0 {
|
|
port = envPort
|
|
}
|
|
}
|
|
log.Info("Started http://0.0.0.0:8080")
|
|
r.Run(fmt.Sprintf(":%s", port))
|
|
}
|