forked from electricdusk/rushlink
Use a more standardized project layout
In #2, mrngm made the point that we should move to a more standardized project structure. This commit does exactly that. The new project structure is based on the repository listed at <https://github.com/golang-standards/project-layout>. Fixes #2.
This commit is contained in:
parent
471a9c69db
commit
544c093c35
@ -8,7 +8,10 @@ import (
|
|||||||
|
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
bolt "go.etcd.io/bbolt"
|
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/internal/handlers"
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/internal/metrics"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ParsedArguments struct {
|
type ParsedArguments struct {
|
||||||
@ -16,7 +19,6 @@ type ParsedArguments struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var appConfig ParsedArguments
|
var appConfig ParsedArguments
|
||||||
var db *bolt.DB
|
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
// Parse the arguments and construct the ParsedArguments
|
// Parse the arguments and construct the ParsedArguments
|
||||||
@ -26,29 +28,18 @@ func main() {
|
|||||||
}
|
}
|
||||||
appConfig = *appConfigRef
|
appConfig = *appConfigRef
|
||||||
|
|
||||||
// Open the bolt database
|
db.Init(appConfig.databaseName)
|
||||||
db, err = bolt.Open(appConfig.databaseName, 0666, &bolt.Options{Timeout: 1 * time.Second})
|
|
||||||
if err != nil {
|
|
||||||
decoratedErr := errors.Wrapf(err, "failed to open database at '%v'", appConfig.databaseName)
|
|
||||||
log.Fatal(decoratedErr)
|
|
||||||
}
|
|
||||||
defer db.Close()
|
|
||||||
err = db.Update(migrateDatabase)
|
|
||||||
if err != nil {
|
|
||||||
decoratedErr := errors.Wrapf(err, "failed to migrate database")
|
|
||||||
log.Fatal(decoratedErr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Export prometheus metrics
|
// Export prometheus metrics
|
||||||
go startMetricsServer()
|
go metrics.StartMetricsServer()
|
||||||
|
|
||||||
// Initialize Gorilla router
|
// Initialize Gorilla router
|
||||||
router := mux.NewRouter()
|
router := mux.NewRouter()
|
||||||
router.HandleFunc("/", indexGetHandler).Methods("GET")
|
router.HandleFunc("/", handlers.IndexGetHandler).Methods("GET")
|
||||||
router.HandleFunc("/", indexPostHandler).Methods("POST")
|
router.HandleFunc("/", handlers.IndexPostHandler).Methods("POST")
|
||||||
router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}", pasteGetHandler).Methods("GET")
|
router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}", handlers.PasteGetHandler).Methods("GET")
|
||||||
router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/nr", pasteGetHandlerNoRedirect).Methods("GET")
|
router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/nr", handlers.PasteGetHandlerNoRedirect).Methods("GET")
|
||||||
router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/meta", pasteGetHandlerMeta).Methods("GET")
|
router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/meta", handlers.PasteGetHandlerMeta).Methods("GET")
|
||||||
|
|
||||||
// Start the server
|
// Start the server
|
||||||
srv := &http.Server{
|
srv := &http.Server{
|
@ -1,54 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
"net/http"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
bolt "go.etcd.io/bbolt"
|
|
||||||
|
|
||||||
"github.com/gorilla/mux"
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
"github.com/prometheus/client_golang/prometheus"
|
|
||||||
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
||||||
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
METRICS_ADDR = "127.0.0.1:58614"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
shortenUrlsTotal = promauto.NewGaugeFunc(prometheus.GaugeOpts{
|
|
||||||
Namespace: "rushlink",
|
|
||||||
Subsystem: "shorten",
|
|
||||||
Name: "urls_total",
|
|
||||||
Help: "The current amount of shortened urls in the database.",
|
|
||||||
}, func() float64 {
|
|
||||||
var metric float64
|
|
||||||
if err := db.View(func(tx *bolt.Tx) error {
|
|
||||||
bucket := tx.Bucket([]byte("shorten"))
|
|
||||||
if bucket == nil {
|
|
||||||
return errors.New("bucket 'shorten' could not be found")
|
|
||||||
}
|
|
||||||
metric = float64(bucket.Stats().KeyN)
|
|
||||||
return nil
|
|
||||||
}); err != nil {
|
|
||||||
log.Printf("error: %v", errors.Wrap(err, "fetching shorten_urls_total metric"))
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
return metric
|
|
||||||
})
|
|
||||||
)
|
|
||||||
|
|
||||||
func startMetricsServer() {
|
|
||||||
router := mux.NewRouter()
|
|
||||||
router.Handle("/metrics", promhttp.Handler()).Methods("GET")
|
|
||||||
srv := &http.Server{
|
|
||||||
Handler: router,
|
|
||||||
Addr: METRICS_ADDR,
|
|
||||||
WriteTimeout: 15 * time.Second,
|
|
||||||
ReadTimeout: 15 * time.Second,
|
|
||||||
}
|
|
||||||
log.Fatal(srv.ListenAndServe())
|
|
||||||
}
|
|
1
go.mod
1
go.mod
@ -3,6 +3,7 @@ module gitea.hashru.nl/dsprenkels/rushlink
|
|||||||
go 1.12
|
go 1.12
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
github.com/go-bindata/go-bindata v3.1.2+incompatible // indirect
|
||||||
github.com/gorilla/mux v1.7.3
|
github.com/gorilla/mux v1.7.3
|
||||||
github.com/pkg/errors v0.8.1
|
github.com/pkg/errors v0.8.1
|
||||||
github.com/prometheus/client_golang v1.1.0
|
github.com/prometheus/client_golang v1.1.0
|
||||||
|
1
go.sum
1
go.sum
@ -6,6 +6,7 @@ github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
|
|||||||
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
|
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/go-bindata/go-bindata v3.1.2+incompatible/go.mod h1:xK8Dsgwmeed+BBsSy2XTopBn/8uK2HWuGSnA11C3Joo=
|
||||||
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
|
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
|
||||||
github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
|
github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
|
||||||
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
||||||
|
@ -1,13 +1,18 @@
|
|||||||
package main
|
package db
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
"time"
|
||||||
|
|
||||||
"gitea.hashru.nl/dsprenkels/rushlink/gobmarsh"
|
"github.com/pkg/errors"
|
||||||
bolt "go.etcd.io/bbolt"
|
bolt "go.etcd.io/bbolt"
|
||||||
|
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/pkg/gobmarsh"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var DB *bolt.DB
|
||||||
|
|
||||||
// The current database version
|
// The current database version
|
||||||
//
|
//
|
||||||
// If we alter the database format, we bump this number and write a new
|
// If we alter the database format, we bump this number and write a new
|
||||||
@ -25,6 +30,16 @@ const BUCKET_PASTES = "pastes"
|
|||||||
// CURRENT_MIGRATE_VERSION, the database has to be migrated.
|
// CURRENT_MIGRATE_VERSION, the database has to be migrated.
|
||||||
const KEY_MIGRATE_VERSION = "migrate_version"
|
const KEY_MIGRATE_VERSION = "migrate_version"
|
||||||
|
|
||||||
|
// Open the bolt database
|
||||||
|
func Init(name string) error {
|
||||||
|
var err error
|
||||||
|
DB, err = bolt.Open(name, 0666, &bolt.Options{Timeout: 1 * time.Second})
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrapf(err, "failed to open database at '%v'", name)
|
||||||
|
}
|
||||||
|
return DB.Update(migrateDatabase)
|
||||||
|
}
|
||||||
|
|
||||||
// Initialize and migrate the database to the current version
|
// Initialize and migrate the database to the current version
|
||||||
func migrateDatabase(tx *bolt.Tx) error {
|
func migrateDatabase(tx *bolt.Tx) error {
|
||||||
dbVersion, err := dbVersion(tx)
|
dbVersion, err := dbVersion(tx)
|
@ -1,8 +1,8 @@
|
|||||||
package main
|
package handlers
|
||||||
|
|
||||||
//go:generate go get github.com/go-bindata/go-bindata
|
//go:generate go get github.com/go-bindata/go-bindata
|
||||||
//go:generate go get -u github.com/go-bindata/go-bindata/...
|
//go:generate go get -u github.com/go-bindata/go-bindata/...
|
||||||
//go:generate go-bindata -pkg $GOPACKAGE text/
|
//go:generate go-bindata -pkg $GOPACKAGE -prefix ../../assets ../../assets/...
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
@ -19,10 +19,12 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
"unicode"
|
"unicode"
|
||||||
|
|
||||||
"gitea.hashru.nl/dsprenkels/rushlink/gobmarsh"
|
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
bolt "go.etcd.io/bbolt"
|
bolt "go.etcd.io/bbolt"
|
||||||
|
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/pkg/gobmarsh"
|
||||||
)
|
)
|
||||||
|
|
||||||
type PasteType int
|
type PasteType int
|
||||||
@ -82,13 +84,13 @@ func (t PasteState) String() (string, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func indexGetHandler(w http.ResponseWriter, r *http.Request) {
|
func IndexGetHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
if err := indexTemplate.Execute(w, nil); err != nil {
|
if err := indexTemplate.Execute(w, nil); err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func indexPostHandler(w http.ResponseWriter, r *http.Request) {
|
func IndexPostHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
if err := r.ParseMultipartForm(50 * 1000 * 1000); err != nil {
|
if err := r.ParseMultipartForm(50 * 1000 * 1000); err != nil {
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
fmt.Fprintf(w, "Internal server error: %v\n", err)
|
fmt.Fprintf(w, "Internal server error: %v\n", err)
|
||||||
@ -115,18 +117,18 @@ func indexPostHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
shortenPostHandler(w, r)
|
ShortenPostHandler(w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
func pasteGetHandler(w http.ResponseWriter, r *http.Request) {
|
func PasteGetHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
pasteGetHandlerInner(w, r, false, false)
|
pasteGetHandlerInner(w, r, false, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
func pasteGetHandlerNoRedirect(w http.ResponseWriter, r *http.Request) {
|
func PasteGetHandlerNoRedirect(w http.ResponseWriter, r *http.Request) {
|
||||||
pasteGetHandlerInner(w, r, true, false)
|
pasteGetHandlerInner(w, r, true, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
func pasteGetHandlerMeta(w http.ResponseWriter, r *http.Request) {
|
func PasteGetHandlerMeta(w http.ResponseWriter, r *http.Request) {
|
||||||
pasteGetHandlerInner(w, r, false, true)
|
pasteGetHandlerInner(w, r, false, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -134,7 +136,7 @@ func pasteGetHandlerInner(w http.ResponseWriter, r *http.Request, noRedirect, sh
|
|||||||
vars := mux.Vars(r)
|
vars := mux.Vars(r)
|
||||||
key := vars["key"]
|
key := vars["key"]
|
||||||
var storedPaste *StoredPaste
|
var storedPaste *StoredPaste
|
||||||
if err := db.View(func(tx *bolt.Tx) error {
|
if err := db.DB.View(func(tx *bolt.Tx) error {
|
||||||
var err error
|
var err error
|
||||||
storedPaste, err = getURL(tx, []byte(key))
|
storedPaste, err = getURL(tx, []byte(key))
|
||||||
return err
|
return err
|
||||||
@ -204,7 +206,7 @@ func pasteGetHandlerInner(w http.ResponseWriter, r *http.Request, noRedirect, sh
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func shortenPostHandler(w http.ResponseWriter, r *http.Request) {
|
func ShortenPostHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
rawurl := r.PostForm.Get("shorten")
|
rawurl := r.PostForm.Get("shorten")
|
||||||
userURL, err := url.ParseRequestURI(rawurl)
|
userURL, err := url.ParseRequestURI(rawurl)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -224,7 +226,7 @@ func shortenPostHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var storedPaste *StoredPaste
|
var storedPaste *StoredPaste
|
||||||
if err := db.Update(func(tx *bolt.Tx) error {
|
if err := db.DB.Update(func(tx *bolt.Tx) error {
|
||||||
ownerKey, ok := getOwnerTokenFromRequest(r)
|
ownerKey, ok := getOwnerTokenFromRequest(r)
|
||||||
if ok == false {
|
if ok == false {
|
||||||
// Owner key not supplied or invalid, generate a new one
|
// Owner key not supplied or invalid, generate a new one
|
||||||
@ -262,9 +264,9 @@ func shortenPostHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
// Retrieve a URL from the database
|
// Retrieve a URL from the database
|
||||||
func getURL(tx *bolt.Tx, key []byte) (*StoredPaste, error) {
|
func getURL(tx *bolt.Tx, key []byte) (*StoredPaste, error) {
|
||||||
shortenBucket := tx.Bucket([]byte(BUCKET_PASTES))
|
shortenBucket := tx.Bucket([]byte(db.BUCKET_PASTES))
|
||||||
if shortenBucket == nil {
|
if shortenBucket == nil {
|
||||||
return nil, fmt.Errorf("bucket %v does not exist", BUCKET_PASTES)
|
return nil, fmt.Errorf("bucket %v does not exist", db.BUCKET_PASTES)
|
||||||
}
|
}
|
||||||
storedBytes := shortenBucket.Get(key)
|
storedBytes := shortenBucket.Get(key)
|
||||||
if storedBytes == nil {
|
if storedBytes == nil {
|
||||||
@ -279,9 +281,9 @@ func getURL(tx *bolt.Tx, key []byte) (*StoredPaste, error) {
|
|||||||
//
|
//
|
||||||
// Returns the new ID if the url was successfully shortened
|
// Returns the new ID if the url was successfully shortened
|
||||||
func shortenURL(tx *bolt.Tx, userURL *url.URL, ownerKey [16]byte) (*StoredPaste, error) {
|
func shortenURL(tx *bolt.Tx, userURL *url.URL, ownerKey [16]byte) (*StoredPaste, error) {
|
||||||
shortenBucket := tx.Bucket([]byte(BUCKET_PASTES))
|
shortenBucket := tx.Bucket([]byte(db.BUCKET_PASTES))
|
||||||
if shortenBucket == nil {
|
if shortenBucket == nil {
|
||||||
return nil, fmt.Errorf("bucket %v does not exist", BUCKET_PASTES)
|
return nil, fmt.Errorf("bucket %v does not exist", db.BUCKET_PASTES)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate a key until it is not in the database, this occurs in O(log N),
|
// Generate a key until it is not in the database, this occurs in O(log N),
|
55
internal/metrics/metrics.go
Normal file
55
internal/metrics/metrics.go
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
||||||
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
||||||
|
bolt "go.etcd.io/bbolt"
|
||||||
|
|
||||||
|
"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
METRICS_ADDR = "127.0.0.1:58614"
|
||||||
|
)
|
||||||
|
|
||||||
|
func StartMetricsServer() {
|
||||||
|
var (
|
||||||
|
_ = promauto.NewGaugeFunc(prometheus.GaugeOpts{
|
||||||
|
Namespace: "rushlink",
|
||||||
|
Subsystem: "shorten",
|
||||||
|
Name: "urls_total",
|
||||||
|
Help: "The current amount of shortened urls in the database.",
|
||||||
|
}, func() float64 {
|
||||||
|
var metric float64
|
||||||
|
if err := db.DB.View(func(tx *bolt.Tx) error {
|
||||||
|
bucket := tx.Bucket([]byte("shorten"))
|
||||||
|
if bucket == nil {
|
||||||
|
return errors.New("bucket 'shorten' could not be found")
|
||||||
|
}
|
||||||
|
metric = float64(bucket.Stats().KeyN)
|
||||||
|
return nil
|
||||||
|
}); err != nil {
|
||||||
|
log.Printf("error: %v", errors.Wrap(err, "fetching shorten_urls_total metric"))
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return metric
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
router := mux.NewRouter()
|
||||||
|
router.Handle("/metrics", promhttp.Handler()).Methods("GET")
|
||||||
|
srv := &http.Server{
|
||||||
|
Handler: router,
|
||||||
|
Addr: METRICS_ADDR,
|
||||||
|
WriteTimeout: 15 * time.Second,
|
||||||
|
ReadTimeout: 15 * time.Second,
|
||||||
|
}
|
||||||
|
log.Fatal(srv.ListenAndServe())
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user