mirror of
https://github.com/taigrr/shorturl
synced 2025-01-18 04:03:16 -08:00
* Changed URL format to shortuuid * Fixed drone CI failure * Moved database to Bitcask and some minor changes * Moved database to Bitcask and some minor changes * Delete coverage.txt
397 lines
9.0 KiB
Go
397 lines
9.0 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"html/template"
|
|
"log"
|
|
"net/http"
|
|
"net/url"
|
|
|
|
// Logging
|
|
|
|
"github.com/unrolled/logger"
|
|
|
|
// Stats/Metrics
|
|
"github.com/rcrowley/go-metrics"
|
|
"github.com/rcrowley/go-metrics/exp"
|
|
"github.com/thoas/stats"
|
|
|
|
rice "github.com/GeertJohan/go.rice"
|
|
"github.com/julienschmidt/httprouter"
|
|
)
|
|
|
|
// Counters ...
|
|
type Counters struct {
|
|
r metrics.Registry
|
|
}
|
|
|
|
func NewCounters() *Counters {
|
|
counters := &Counters{
|
|
r: metrics.NewRegistry(),
|
|
}
|
|
return counters
|
|
}
|
|
|
|
func (c *Counters) Inc(name string) {
|
|
metrics.GetOrRegisterCounter(name, c.r).Inc(1)
|
|
}
|
|
|
|
func (c *Counters) Dec(name string) {
|
|
metrics.GetOrRegisterCounter(name, c.r).Dec(1)
|
|
}
|
|
|
|
func (c *Counters) IncBy(name string, n int64) {
|
|
metrics.GetOrRegisterCounter(name, c.r).Inc(n)
|
|
}
|
|
|
|
func (c *Counters) DecBy(name string, n int64) {
|
|
metrics.GetOrRegisterCounter(name, c.r).Dec(n)
|
|
}
|
|
|
|
// Server ...
|
|
type Server struct {
|
|
bind string
|
|
env *Env
|
|
templates *Templates
|
|
router *httprouter.Router
|
|
|
|
// Logger
|
|
logger *logger.Logger
|
|
|
|
// Stats/Metrics
|
|
counters *Counters
|
|
stats *stats.Stats
|
|
}
|
|
|
|
func (s *Server) render(name string, w http.ResponseWriter, ctx interface{}) {
|
|
buf, err := s.templates.Exec(name, ctx)
|
|
if err != nil {
|
|
log.Printf("error rendering template %s: %s", name, err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
}
|
|
|
|
_, err = buf.WriteTo(w)
|
|
if err != nil {
|
|
log.Printf("error writing template buffer: %s", err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
}
|
|
}
|
|
|
|
type IndexContext struct {
|
|
URLList []*URL
|
|
}
|
|
|
|
// IndexHandler ...
|
|
func (s *Server) IndexHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, _ httprouter.Params) {
|
|
s.counters.Inc("n_index")
|
|
|
|
var urlList []*URL
|
|
for id := range s.env.db.Keys() {
|
|
u, err := get(s.env.db, string(id))
|
|
if err != nil {
|
|
log.Printf("error querying urls index: %s", err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
urlList = append(urlList, &u)
|
|
}
|
|
|
|
ctx := &IndexContext{
|
|
URLList: urlList,
|
|
}
|
|
|
|
s.render("index", w, ctx)
|
|
}
|
|
}
|
|
|
|
// ShortenHandler ...
|
|
func (s *Server) ShortenHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
s.counters.Inc("n_shorten")
|
|
|
|
u, err := s.env.NewURL(r.FormValue("url"))
|
|
if err != nil {
|
|
log.Printf("error creating new url: %s", err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
redirectURL, err := url.Parse(fmt.Sprintf("./u/%s", u.ID))
|
|
if err != nil {
|
|
log.Printf("error parsing redirect url ./u/%s: %s", u.ID, err)
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
}
|
|
|
|
http.Redirect(w, r, redirectURL.String(), http.StatusFound)
|
|
}
|
|
}
|
|
|
|
// ViewHandler ...
|
|
func (s *Server) ViewHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
|
|
s.counters.Inc("n_view")
|
|
|
|
id := p.ByName("id")
|
|
if id == "" {
|
|
http.Error(w, "Bad Request", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
// Check if entry exists
|
|
if !s.env.db.Has([]byte(id)) {
|
|
http.Error(w, "Not Found", http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
// Retrieve URL
|
|
u, err := get(s.env.db, id)
|
|
if err != nil {
|
|
log.Printf("error looking up %s for redirect: %s", id, err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
baseURL, err := url.Parse(s.env.cfg.baseURL)
|
|
if err != nil {
|
|
log.Printf("error parsing config.baseURL: %s", s.env.cfg.baseURL)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
}
|
|
|
|
redirectURL, err := url.Parse(fmt.Sprintf("./r/%s", id))
|
|
if err != nil {
|
|
log.Printf("error parsing redirect url ./r/%s: %s", id, err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
}
|
|
|
|
fullURL := baseURL.ResolveReference(redirectURL)
|
|
|
|
s.render(
|
|
"view", w,
|
|
struct {
|
|
ID string
|
|
URL string
|
|
Ori string
|
|
}{
|
|
ID: id,
|
|
URL: fullURL.String(),
|
|
Ori: u.URL,
|
|
},
|
|
)
|
|
}
|
|
}
|
|
|
|
// RedirectHandler ...
|
|
func (s *Server) RedirectHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
s.counters.Inc("n_redirect")
|
|
|
|
id := p.ByName("id")
|
|
if id == "" {
|
|
http.Error(w, "Bad Request", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
// Check if entry exists
|
|
if !s.env.db.Has([]byte(id)) {
|
|
http.Error(w, "Not Found", http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
// Retrieve URL
|
|
u, err := get(s.env.db, id)
|
|
if err != nil {
|
|
log.Printf("error looking up %s for redirect: %s", id, err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
http.Redirect(w, r, u.URL, http.StatusFound)
|
|
}
|
|
}
|
|
|
|
// StatsHandler ...
|
|
func (s *Server) StatsHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
bs, err := json.Marshal(s.stats.Data())
|
|
if err != nil {
|
|
log.Printf("error marshalling stats: %s", err)
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
}
|
|
w.Write(bs)
|
|
}
|
|
}
|
|
|
|
// EditHandler ...
|
|
func (s *Server) EditHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
id := p.ByName("id")
|
|
if id == "" {
|
|
http.Error(w, "Bad Request", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
// Check if entry exists
|
|
if !s.env.db.Has([]byte(id)) {
|
|
http.Error(w, "Not Found", http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
// Retrieve URL
|
|
u, err := get(s.env.db, id)
|
|
if err != nil {
|
|
log.Printf("error looking up %s for redirect: %s", id, err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
s.render(
|
|
"edit", w,
|
|
struct {
|
|
ID string
|
|
URL string
|
|
}{
|
|
ID: u.ID,
|
|
URL: string(u.URL),
|
|
},
|
|
)
|
|
}
|
|
}
|
|
|
|
// UpdateHandler ...
|
|
func (s *Server) UpdateHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
var u URL
|
|
|
|
id := p.ByName("id")
|
|
target := r.FormValue("url")
|
|
newID := r.FormValue("id")
|
|
if id == "" || target == "" {
|
|
http.Error(w, "Bad Request", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
// Check if entry exists
|
|
if !s.env.db.Has([]byte(id)) {
|
|
http.Error(w, "Not Found", http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
err := u.update(s.env.db, target)
|
|
if err != nil {
|
|
log.Printf("error updating %s error: %v", id, err)
|
|
http.Error(w, "Internal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
redirectURL := fmt.Sprintf("/u/%s", newID)
|
|
|
|
http.Redirect(w, r, redirectURL, http.StatusFound)
|
|
}
|
|
}
|
|
|
|
// ListenAndServe ...
|
|
func (s *Server) ListenAndServe() {
|
|
log.Fatal(
|
|
http.ListenAndServe(
|
|
s.bind,
|
|
s.logger.Handler(
|
|
s.stats.Handler(s.router),
|
|
),
|
|
),
|
|
)
|
|
}
|
|
|
|
// DeleteHandler ...
|
|
func (s *Server) DeleteHandler() httprouter.Handle {
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
|
|
id := p.ByName("id")
|
|
if id == "" {
|
|
http.Error(w, "Bad Request", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
err := del(s.env.db, id)
|
|
if err != nil {
|
|
log.Printf("error delete id: %s: %v", id, err)
|
|
http.Error(w, "Iternal Error", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
http.Redirect(w, r, "/", http.StatusFound)
|
|
}
|
|
}
|
|
|
|
func (s *Server) initRoutes() {
|
|
s.router.Handler("GET", "/debug/metrics", exp.ExpHandler(s.counters.r))
|
|
s.router.GET("/debug/stats", s.StatsHandler())
|
|
|
|
s.router.ServeFiles(
|
|
"/css/*filepath",
|
|
rice.MustFindBox("static/css").HTTPBox(),
|
|
)
|
|
|
|
s.router.ServeFiles(
|
|
"/js/*filepath",
|
|
rice.MustFindBox("static/js").HTTPBox(),
|
|
)
|
|
|
|
s.router.GET("/", s.IndexHandler())
|
|
s.router.POST("/", s.ShortenHandler())
|
|
s.router.GET("/u/:id", s.ViewHandler())
|
|
s.router.GET("/r/:id", s.RedirectHandler())
|
|
s.router.GET("/e/:id", s.EditHandler())
|
|
s.router.POST("/e/:id", s.UpdateHandler())
|
|
s.router.GET("/d/:id", s.DeleteHandler())
|
|
}
|
|
|
|
// NewServer ...
|
|
func (env *Env) NewServer(bind string) *Server {
|
|
server := &Server{
|
|
bind: bind,
|
|
env: env,
|
|
router: httprouter.New(),
|
|
templates: NewTemplates("base"),
|
|
|
|
// Logger
|
|
logger: logger.New(logger.Options{
|
|
Prefix: "shorturl",
|
|
RemoteAddressHeaders: []string{"X-Forwarded-For"},
|
|
OutputFlags: log.LstdFlags,
|
|
}),
|
|
|
|
// Stats/Metrics
|
|
counters: NewCounters(),
|
|
stats: stats.New(),
|
|
}
|
|
|
|
// Templates
|
|
box := rice.MustFindBox("templates")
|
|
|
|
indexTemplate := template.New("index")
|
|
template.Must(indexTemplate.Parse(box.MustString("index.html")))
|
|
template.Must(indexTemplate.Parse(box.MustString("base.html")))
|
|
|
|
viewTemplate := template.New("view")
|
|
template.Must(viewTemplate.Parse(box.MustString("view.html")))
|
|
template.Must(viewTemplate.Parse(box.MustString("base.html")))
|
|
|
|
editTemplate := template.New("edit")
|
|
template.Must(editTemplate.Parse(box.MustString("edit.html")))
|
|
template.Must(editTemplate.Parse(box.MustString("base.html")))
|
|
|
|
server.templates.Add("index", indexTemplate)
|
|
server.templates.Add("view", viewTemplate)
|
|
server.templates.Add("edit", editTemplate)
|
|
|
|
server.initRoutes()
|
|
|
|
return server
|
|
}
|