forked from electricdusk/rushlink
		
	Refactor database login into a separate module
This commit is contained in:
		
							parent
							
								
									8b87cd0f8a
								
							
						
					
					
						commit
						0cfad96b68
					
				@ -5,6 +5,7 @@ import (
 | 
			
		||||
	"log"
 | 
			
		||||
 | 
			
		||||
	"gitea.hashru.nl/dsprenkels/rushlink"
 | 
			
		||||
	"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
@ -17,14 +18,16 @@ var (
 | 
			
		||||
func main() {
 | 
			
		||||
	flag.Parse()
 | 
			
		||||
 | 
			
		||||
	if err := rushlink.OpenDB(*databasePath); err != nil {
 | 
			
		||||
	database, err := db.OpenDB(*databasePath)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatalln(err)
 | 
			
		||||
	}
 | 
			
		||||
	defer rushlink.CloseDB()
 | 
			
		||||
	if err := rushlink.OpenFileStore(*fileStorePath); err != nil {
 | 
			
		||||
	defer database.Close()
 | 
			
		||||
	filestore, err := db.OpenFileStore(*fileStorePath)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatalln(err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	go rushlink.StartMetricsServer(*metricsListen)
 | 
			
		||||
	rushlink.StartMainServer(*httpListen)
 | 
			
		||||
	go rushlink.StartMetricsServer(*metricsListen, database)
 | 
			
		||||
	rushlink.StartMainServer(*httpListen, database, filestore)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										135
									
								
								db.go
									
									
									
									
									
								
							
							
						
						
									
										135
									
								
								db.go
									
									
									
									
									
								
							@ -1,135 +0,0 @@
 | 
			
		||||
package rushlink
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"log"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
	bolt "go.etcd.io/bbolt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var DB *bolt.DB
 | 
			
		||||
 | 
			
		||||
// The current database version
 | 
			
		||||
//
 | 
			
		||||
// If we alter the database format, we bump this number and write a new
 | 
			
		||||
// database migration in migrateDatabase().
 | 
			
		||||
const CURRENT_MIGRATE_VERSION = 2
 | 
			
		||||
 | 
			
		||||
// Bucket storing everything that is not a bulk value. This includes stuff like
 | 
			
		||||
// the database version, secret site-wide keys.
 | 
			
		||||
const BUCKET_CONF = "conf"
 | 
			
		||||
 | 
			
		||||
// The main bucket for paste values and URL redirects
 | 
			
		||||
const BUCKET_PASTES = "pastes"
 | 
			
		||||
 | 
			
		||||
// The main bucket for file uploads
 | 
			
		||||
const BUCKET_FILE_UPLOAD = "fileUpload"
 | 
			
		||||
 | 
			
		||||
// This value stores the current migration version. If this value is less than
 | 
			
		||||
// CURRENT_MIGRATE_VERSION, the database has to be migrated.
 | 
			
		||||
const KEY_MIGRATE_VERSION = "migrate_version"
 | 
			
		||||
 | 
			
		||||
// Open the bolt database
 | 
			
		||||
func OpenDB(path string) error {
 | 
			
		||||
	if path == "" {
 | 
			
		||||
		return errors.New("database not set")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var err error
 | 
			
		||||
	DB, err = bolt.Open(path, 0666, &bolt.Options{Timeout: 1 * time.Second})
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.Wrapf(err, "failed to open database at '%v'", path)
 | 
			
		||||
	}
 | 
			
		||||
	return DB.Update(migrateDatabase)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Close the bolt database
 | 
			
		||||
func CloseDB() error {
 | 
			
		||||
	if DB == nil {
 | 
			
		||||
		panic("no open database")
 | 
			
		||||
	}
 | 
			
		||||
	return DB.Close()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Initialize and migrate the database to the current version
 | 
			
		||||
func migrateDatabase(tx *bolt.Tx) error {
 | 
			
		||||
	dbVersion, err := dbVersion(tx)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Migrate the database to version 1
 | 
			
		||||
	if dbVersion < 1 {
 | 
			
		||||
		log.Println("migrating database to version 1")
 | 
			
		||||
		// Create conf bucket
 | 
			
		||||
		_, err := tx.CreateBucket([]byte(BUCKET_CONF))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		// Create paste bucket
 | 
			
		||||
		_, err = tx.CreateBucket([]byte(BUCKET_PASTES))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		// Update the version number
 | 
			
		||||
		if err := setDBVersion(tx, 1); err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if dbVersion < 2 {
 | 
			
		||||
		log.Println("migrating database to version 2")
 | 
			
		||||
		// Create fileUpload bucket
 | 
			
		||||
		_, err := tx.CreateBucket([]byte(BUCKET_FILE_UPLOAD))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		// Update the version number
 | 
			
		||||
		if err := setDBVersion(tx, 2); err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Get the current migrate version from the database
 | 
			
		||||
func dbVersion(tx *bolt.Tx) (int, error) {
 | 
			
		||||
	conf := tx.Bucket([]byte(BUCKET_CONF))
 | 
			
		||||
	if conf == nil {
 | 
			
		||||
		return 0, nil
 | 
			
		||||
	}
 | 
			
		||||
	dbVersionBytes := conf.Get([]byte(KEY_MIGRATE_VERSION))
 | 
			
		||||
	if dbVersionBytes == nil {
 | 
			
		||||
		return 0, nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Version was already stored
 | 
			
		||||
	var dbVersion int
 | 
			
		||||
	if err := Unmarshal(dbVersionBytes, &dbVersion); err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
	if dbVersion == 0 {
 | 
			
		||||
		return 0, fmt.Errorf("database version is invalid (%v)", dbVersion)
 | 
			
		||||
	}
 | 
			
		||||
	if dbVersion > CURRENT_MIGRATE_VERSION {
 | 
			
		||||
		return 0, fmt.Errorf("database version is too recent (%v > %v)", dbVersion, CURRENT_MIGRATE_VERSION)
 | 
			
		||||
	}
 | 
			
		||||
	return dbVersion, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Update the current migrate version in the database
 | 
			
		||||
func setDBVersion(tx *bolt.Tx, version int) error {
 | 
			
		||||
	conf, err := tx.CreateBucketIfNotExists([]byte(BUCKET_CONF))
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	versionBytes, err := Marshal(version)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	return conf.Put([]byte(KEY_MIGRATE_VERSION), versionBytes)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										169
									
								
								fileupload.go
									
									
									
									
									
								
							
							
						
						
									
										169
									
								
								fileupload.go
									
									
									
									
									
								
							@ -1,33 +1,22 @@
 | 
			
		||||
package rushlink
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/hex"
 | 
			
		||||
	"hash/crc32"
 | 
			
		||||
	"io"
 | 
			
		||||
	"log"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"os"
 | 
			
		||||
	"path"
 | 
			
		||||
 | 
			
		||||
	"github.com/google/uuid"
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
	bolt "go.etcd.io/bbolt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Use the Castagnoli checksum because of the acceleration on Intel CPUs
 | 
			
		||||
var checksumTable = crc32.MakeTable(crc32.Castagnoli)
 | 
			
		||||
 | 
			
		||||
// Where to store the uploaded files
 | 
			
		||||
var fileStoreDir = ""
 | 
			
		||||
 | 
			
		||||
// Custom HTTP filesystem handler
 | 
			
		||||
type fileUploadFileSystem struct {
 | 
			
		||||
// FileUploadFileSystem is a HTTP filesystem handler
 | 
			
		||||
type FileUploadFileSystem struct {
 | 
			
		||||
	fs http.FileSystem
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Open opens file
 | 
			
		||||
func (fs fileUploadFileSystem) Open(path string) (http.File, error) {
 | 
			
		||||
// Open opens a file
 | 
			
		||||
func (fs FileUploadFileSystem) Open(path string) (http.File, error) {
 | 
			
		||||
	log.Println(path)
 | 
			
		||||
	file, err := fs.fs.Open(path)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
@ -42,153 +31,3 @@ func (fs fileUploadFileSystem) Open(path string) (http.File, error) {
 | 
			
		||||
	}
 | 
			
		||||
	return file, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type fileUploadState int
 | 
			
		||||
 | 
			
		||||
type fileUpload struct {
 | 
			
		||||
	State       fileUploadState
 | 
			
		||||
	ID          uuid.UUID
 | 
			
		||||
	FileName    string
 | 
			
		||||
	ContentType string
 | 
			
		||||
	Checksum    uint32
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	dirMode  os.FileMode = 0750
 | 
			
		||||
	fileMode os.FileMode = 0640
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	fileUploadStateUndef   fileUploadState = 0
 | 
			
		||||
	fileUploadStatePresent                 = 1
 | 
			
		||||
	fileUploadStateDeleted                 = 2
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func (t fileUploadState) String() string {
 | 
			
		||||
	switch t {
 | 
			
		||||
	case fileUploadStateUndef:
 | 
			
		||||
		return "unknown"
 | 
			
		||||
	case fileUploadStatePresent:
 | 
			
		||||
		return "present"
 | 
			
		||||
	case fileUploadStateDeleted:
 | 
			
		||||
		return "deleted"
 | 
			
		||||
	default:
 | 
			
		||||
		return "invalid"
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func OpenFileStore(path string) error {
 | 
			
		||||
	if path == "" {
 | 
			
		||||
		return errors.New("file-store not set")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Try to create the file store directory if it does not yet exist
 | 
			
		||||
	if err := os.MkdirAll(path, dirMode); err != nil {
 | 
			
		||||
		return errors.Wrap(err, "creating file store directory")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	fileStoreDir = path[:]
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newFileUpload(tx *bolt.Tx, r io.Reader, fileName string, contentType string) (*fileUpload, error) {
 | 
			
		||||
	id, err := uuid.NewRandom()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "generating UUID")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	filePath := fileStorePath(id, fileName)
 | 
			
		||||
	if err := os.Mkdir(path.Dir(filePath), dirMode); err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "creating file dir")
 | 
			
		||||
	}
 | 
			
		||||
	file, err := os.OpenFile(filePath, os.O_WRONLY|os.O_CREATE, fileMode)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "opening file")
 | 
			
		||||
	}
 | 
			
		||||
	defer file.Close()
 | 
			
		||||
 | 
			
		||||
	hash := crc32.New(checksumTable)
 | 
			
		||||
	tee := io.TeeReader(r, hash)
 | 
			
		||||
	_, err = io.Copy(file, tee)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "writing to file")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	fu := &fileUpload{
 | 
			
		||||
		State:       fileUploadStatePresent,
 | 
			
		||||
		ID:          id,
 | 
			
		||||
		FileName:    fileName,
 | 
			
		||||
		ContentType: contentType,
 | 
			
		||||
		Checksum:    hash.Sum32(),
 | 
			
		||||
	}
 | 
			
		||||
	if err := fu.save(tx); err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
	return fu, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func getFileUpload(tx *bolt.Tx, id uuid.UUID) (*fileUpload, error) {
 | 
			
		||||
	bucket := tx.Bucket([]byte(BUCKET_FILE_UPLOAD))
 | 
			
		||||
	if bucket == nil {
 | 
			
		||||
		return nil, errors.Errorf("bucket %v does not exist", BUCKET_FILE_UPLOAD)
 | 
			
		||||
	}
 | 
			
		||||
	storedBytes := bucket.Get(id[:])
 | 
			
		||||
	if storedBytes == nil {
 | 
			
		||||
		return nil, nil
 | 
			
		||||
	}
 | 
			
		||||
	fu := &fileUpload{}
 | 
			
		||||
	err := Unmarshal(storedBytes, fu)
 | 
			
		||||
	return fu, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (fu *fileUpload) save(tx *bolt.Tx) error {
 | 
			
		||||
	bucket := tx.Bucket([]byte(BUCKET_FILE_UPLOAD))
 | 
			
		||||
	if bucket == nil {
 | 
			
		||||
		return errors.Errorf("bucket %v does not exist", BUCKET_FILE_UPLOAD)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	buf, err := Marshal(fu)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.Wrap(err, "encoding for database failed")
 | 
			
		||||
	}
 | 
			
		||||
	if err := bucket.Put(fu.ID[:], buf); err != nil {
 | 
			
		||||
		return errors.Wrap(err, "database transaction failed")
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (fu *fileUpload) delete(tx *bolt.Tx) error {
 | 
			
		||||
	// Remove the file in the backend
 | 
			
		||||
	filePath := fileStorePath(fu.ID, fu.FileName)
 | 
			
		||||
	if err := os.Remove(filePath); err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Update the file in the server
 | 
			
		||||
	if err := (&fileUpload{
 | 
			
		||||
		ID:    fu.ID,
 | 
			
		||||
		State: fileUploadStateDeleted,
 | 
			
		||||
	}).save(tx); err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Cleanup the parent directory
 | 
			
		||||
	wrap := "deletion succeeded, but removing the file directory has failed"
 | 
			
		||||
	return errors.Wrap(os.Remove(path.Dir(filePath)), wrap)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (fu *fileUpload) url() *url.URL {
 | 
			
		||||
	rawurl := "/uploads/" + hex.EncodeToString(fu.ID[:]) + "/" + fu.FileName
 | 
			
		||||
	urlParse, err := url.Parse(rawurl)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		panic("could not construct /uploads/ url")
 | 
			
		||||
	}
 | 
			
		||||
	return urlParse
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func fileStorePath(id uuid.UUID, fileName string) string {
 | 
			
		||||
	if fileStoreDir == "" {
 | 
			
		||||
		panic("fileStoreDir called while the file store path has not been set")
 | 
			
		||||
	}
 | 
			
		||||
	return path.Join(fileStoreDir, hex.EncodeToString(id[:]), fileName)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										122
									
								
								handlers.go
									
									
									
									
									
								
							
							
						
						
									
										122
									
								
								handlers.go
									
									
									
									
									
								
							@ -2,7 +2,6 @@ package rushlink
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"crypto/subtle"
 | 
			
		||||
	"encoding/base64"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"io"
 | 
			
		||||
	"log"
 | 
			
		||||
@ -12,6 +11,7 @@ import (
 | 
			
		||||
	"os"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
 | 
			
		||||
	"github.com/google/uuid"
 | 
			
		||||
	"github.com/gorilla/mux"
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
@ -26,32 +26,25 @@ const (
 | 
			
		||||
	viewShowMeta
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const CookieDeleteToken = "owner_token"
 | 
			
		||||
const cookieDeleteToken = "owner_token"
 | 
			
		||||
 | 
			
		||||
// These keys are designated reserved, and will not be randomly chosen
 | 
			
		||||
var ReservedPasteKeys = []string{"xd42", "example"}
 | 
			
		||||
 | 
			
		||||
// Base64 encoding and decoding
 | 
			
		||||
var base64Alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_"
 | 
			
		||||
var base64Encoder = base64.RawURLEncoding.WithPadding(base64.NoPadding)
 | 
			
		||||
 | 
			
		||||
func indexGetHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
func (rl *rushlink) indexGetHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	render(w, r, "index", map[string]interface{}{})
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func uploadFileGetHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
func (rl *rushlink) uploadFileGetHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	vars := mux.Vars(r)
 | 
			
		||||
	id := vars["id"]
 | 
			
		||||
 | 
			
		||||
	var fu *fileUpload
 | 
			
		||||
	var fu *db.FileUpload
 | 
			
		||||
	var badID bool
 | 
			
		||||
	if err := DB.View(func(tx *bolt.Tx) error {
 | 
			
		||||
	if err := rl.db.Bolt.View(func(tx *bolt.Tx) error {
 | 
			
		||||
		fuID, err := uuid.Parse(id)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			badID = true
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		fu, err = getFileUpload(tx, fuID)
 | 
			
		||||
		fu, err = db.GetFileUpload(tx, fuID)
 | 
			
		||||
		return err
 | 
			
		||||
	}); err != nil {
 | 
			
		||||
		if badID {
 | 
			
		||||
@ -62,7 +55,7 @@ func uploadFileGetHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	filePath := fileStorePath(fu.ID, fu.FileName)
 | 
			
		||||
	filePath := rl.fs.FilePath(fu.ID, fu.FileName)
 | 
			
		||||
	file, err := os.Open(filePath)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		if os.IsNotExist(err) {
 | 
			
		||||
@ -77,35 +70,35 @@ func uploadFileGetHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	io.Copy(w, file)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func viewPasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	viewPasteHandlerInner(w, r, 0)
 | 
			
		||||
func (rl *rushlink) viewPasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	rl.viewPasteHandlerInner(w, r, 0)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func viewPasteHandlerNoRedirect(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	viewPasteHandlerInner(w, r, viewNoRedirect)
 | 
			
		||||
func (rl *rushlink) viewPasteHandlerNoRedirect(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	rl.viewPasteHandlerInner(w, r, viewNoRedirect)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func viewPasteHandlerMeta(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	viewPasteHandlerInner(w, r, viewShowMeta)
 | 
			
		||||
func (rl *rushlink) viewPasteHandlerMeta(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	rl.viewPasteHandlerInner(w, r, viewShowMeta)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func viewPasteHandlerInner(w http.ResponseWriter, r *http.Request, flags viewPaste) {
 | 
			
		||||
func (rl *rushlink) viewPasteHandlerInner(w http.ResponseWriter, r *http.Request, flags viewPaste) {
 | 
			
		||||
	vars := mux.Vars(r)
 | 
			
		||||
	key := vars["key"]
 | 
			
		||||
	var p *paste
 | 
			
		||||
	var p *db.Paste
 | 
			
		||||
	var fuID *uuid.UUID
 | 
			
		||||
	var fu *fileUpload
 | 
			
		||||
	if err := DB.View(func(tx *bolt.Tx) error {
 | 
			
		||||
	var fu *db.FileUpload
 | 
			
		||||
	if err := rl.db.Bolt.View(func(tx *bolt.Tx) error {
 | 
			
		||||
		var err error
 | 
			
		||||
		p, err = getPaste(tx, key)
 | 
			
		||||
		p, err = db.GetPaste(tx, key)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		if p != nil && p.Type == pasteTypeFileUpload {
 | 
			
		||||
		if p != nil && p.Type == db.PasteTypeFileUpload {
 | 
			
		||||
			var id uuid.UUID
 | 
			
		||||
			copy(id[:], p.Content)
 | 
			
		||||
			fuID = &id
 | 
			
		||||
			fu, err = getFileUpload(tx, id)
 | 
			
		||||
			fu, err = db.GetFileUpload(tx, id)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				return err
 | 
			
		||||
			}
 | 
			
		||||
@ -146,17 +139,17 @@ func viewPasteHandlerInner(w http.ResponseWriter, r *http.Request, flags viewPas
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	switch p.State {
 | 
			
		||||
	case pasteStatePresent:
 | 
			
		||||
	case db.PasteStatePresent:
 | 
			
		||||
		var location string
 | 
			
		||||
		switch p.Type {
 | 
			
		||||
		case pasteTypeFileUpload:
 | 
			
		||||
		case db.PasteTypeFileUpload:
 | 
			
		||||
			if fu == nil {
 | 
			
		||||
				panic(fmt.Sprintf("file for id %v does not exist in database\n", fuID))
 | 
			
		||||
			}
 | 
			
		||||
			location = fu.url().String()
 | 
			
		||||
			location = fu.URL().String()
 | 
			
		||||
			break
 | 
			
		||||
		case pasteTypeRedirect:
 | 
			
		||||
			location = p.redirectURL().String()
 | 
			
		||||
		case db.PasteTypeRedirect:
 | 
			
		||||
			location = p.RedirectURL().String()
 | 
			
		||||
			break
 | 
			
		||||
		default:
 | 
			
		||||
			panic("paste type unsupported")
 | 
			
		||||
@ -165,17 +158,17 @@ func viewPasteHandlerInner(w http.ResponseWriter, r *http.Request, flags viewPas
 | 
			
		||||
			http.Redirect(w, r, location, http.StatusSeeOther)
 | 
			
		||||
		}
 | 
			
		||||
		fmt.Fprint(w, location)
 | 
			
		||||
	case pasteStateDeleted:
 | 
			
		||||
	case db.PasteStateDeleted:
 | 
			
		||||
		renderError(w, r, http.StatusGone, "paste has been deleted\n")
 | 
			
		||||
	default:
 | 
			
		||||
		panic(errors.Errorf("invalid paste.State (%v) for key '%v'", p.State, p.Key))
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newPasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
func (rl *rushlink) newPasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	file, fileHeader, err := r.FormFile("file")
 | 
			
		||||
	if err == nil {
 | 
			
		||||
		newFileUploadPasteHandler(w, r, file, *fileHeader)
 | 
			
		||||
		rl.newFileUploadPasteHandler(w, r, file, *fileHeader)
 | 
			
		||||
		return
 | 
			
		||||
	} else if err == http.ErrMissingFile {
 | 
			
		||||
		// Fallthrough
 | 
			
		||||
@ -187,23 +180,26 @@ func newPasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
 | 
			
		||||
	shorten := r.FormValue("shorten")
 | 
			
		||||
	if shorten != "" {
 | 
			
		||||
		newRedirectPasteHandler(w, r, shorten)
 | 
			
		||||
		rl.newRedirectPasteHandler(w, r, shorten)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	renderError(w, r, http.StatusBadRequest, "no 'file' and no 'shorten' fields given in form\n")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newFileUploadPasteHandler(w http.ResponseWriter, r *http.Request, file multipart.File, header multipart.FileHeader) {
 | 
			
		||||
	var fu *fileUpload
 | 
			
		||||
	var paste *paste
 | 
			
		||||
	if err := DB.Update(func(tx *bolt.Tx) error {
 | 
			
		||||
func (rl *rushlink) newFileUploadPasteHandler(w http.ResponseWriter, r *http.Request, file multipart.File, header multipart.FileHeader) {
 | 
			
		||||
	var fu *db.FileUpload
 | 
			
		||||
	var paste *db.Paste
 | 
			
		||||
	if err := rl.db.Bolt.Update(func(tx *bolt.Tx) error {
 | 
			
		||||
		var err error
 | 
			
		||||
		// Create the fileUpload in the database
 | 
			
		||||
		fu, err = newFileUpload(tx, file, header.Filename, header.Header.Get("Content-Type"))
 | 
			
		||||
		fu, err = db.NewFileUpload(rl.fs, file, header.Filename, header.Header.Get("Content-Type"))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			panic(errors.Wrap(err, "creating fileUpload"))
 | 
			
		||||
		}
 | 
			
		||||
		if err := fu.Save(tx); err != nil {
 | 
			
		||||
			panic(errors.Wrap(err, "saving fileUpload in db"))
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		paste, err = shortenFileUploadID(tx, fu.ID)
 | 
			
		||||
		return err
 | 
			
		||||
@ -214,7 +210,7 @@ func newFileUploadPasteHandler(w http.ResponseWriter, r *http.Request, file mult
 | 
			
		||||
	render(w, r, "newFileUploadPasteSuccess", data)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newPasteHandlerURLEncoded(w http.ResponseWriter, r *http.Request, next http.HandlerFunc) {
 | 
			
		||||
func (rl *rushlink) newPasteHandlerURLEncoded(w http.ResponseWriter, r *http.Request, next http.HandlerFunc) {
 | 
			
		||||
	if err := r.ParseForm(); err != nil {
 | 
			
		||||
		next(w, r)
 | 
			
		||||
		return
 | 
			
		||||
@ -224,10 +220,10 @@ func newPasteHandlerURLEncoded(w http.ResponseWriter, r *http.Request, next http
 | 
			
		||||
		renderError(w, r, http.StatusBadRequest, "no 'shorten' param given\n")
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
	newRedirectPasteHandler(w, r, shorten)
 | 
			
		||||
	rl.newRedirectPasteHandler(w, r, shorten)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newRedirectPasteHandler(w http.ResponseWriter, r *http.Request, rawurl string) {
 | 
			
		||||
func (rl *rushlink) newRedirectPasteHandler(w http.ResponseWriter, r *http.Request, rawurl string) {
 | 
			
		||||
	userURL, err := url.ParseRequestURI(rawurl)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		msg := fmt.Sprintf("invalid url (%v): %v", err, rawurl)
 | 
			
		||||
@ -243,8 +239,8 @@ func newRedirectPasteHandler(w http.ResponseWriter, r *http.Request, rawurl stri
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var paste *paste
 | 
			
		||||
	if err := DB.Update(func(tx *bolt.Tx) error {
 | 
			
		||||
	var paste *db.Paste
 | 
			
		||||
	if err := rl.db.Bolt.Update(func(tx *bolt.Tx) error {
 | 
			
		||||
		var err error
 | 
			
		||||
		paste, err = shortenURL(tx, userURL)
 | 
			
		||||
		return err
 | 
			
		||||
@ -256,7 +252,7 @@ func newRedirectPasteHandler(w http.ResponseWriter, r *http.Request, rawurl stri
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Delete a URL from the database
 | 
			
		||||
func deletePasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
func (rl *rushlink) deletePasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	vars := mux.Vars(r)
 | 
			
		||||
	key := vars["key"]
 | 
			
		||||
 | 
			
		||||
@ -267,14 +263,14 @@ func deletePasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var errorCode int
 | 
			
		||||
	var paste paste
 | 
			
		||||
	if err := DB.Update(func(tx *bolt.Tx) error {
 | 
			
		||||
		p, err := getPaste(tx, key)
 | 
			
		||||
	var paste db.Paste
 | 
			
		||||
	if err := rl.db.Bolt.Update(func(tx *bolt.Tx) error {
 | 
			
		||||
		p, err := db.GetPaste(tx, key)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			errorCode = http.StatusNotFound
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		if p.State == pasteStateDeleted {
 | 
			
		||||
		if p.State == db.PasteStateDeleted {
 | 
			
		||||
			errorCode = http.StatusGone
 | 
			
		||||
			return errors.New("already deleted")
 | 
			
		||||
		}
 | 
			
		||||
@ -282,7 +278,7 @@ func deletePasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
			errorCode = http.StatusForbidden
 | 
			
		||||
			return errors.New("invalid delete token")
 | 
			
		||||
		}
 | 
			
		||||
		if err := p.delete(tx); err != nil {
 | 
			
		||||
		if err := p.Delete(tx, rl.fs); err != nil {
 | 
			
		||||
			errorCode = http.StatusInternalServerError
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
@ -302,41 +298,41 @@ func deletePasteHandler(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
//
 | 
			
		||||
// Returns the new paste key if the fileUpload was successfully added to the
 | 
			
		||||
// database
 | 
			
		||||
func shortenFileUploadID(tx *bolt.Tx, id uuid.UUID) (*paste, error) {
 | 
			
		||||
	return shorten(tx, pasteTypeFileUpload, id[:])
 | 
			
		||||
func shortenFileUploadID(tx *bolt.Tx, id uuid.UUID) (*db.Paste, error) {
 | 
			
		||||
	return shorten(tx, db.PasteTypeFileUpload, id[:])
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Add a new URL to the database
 | 
			
		||||
//
 | 
			
		||||
// Returns the new paste key if the url was successfully shortened
 | 
			
		||||
func shortenURL(tx *bolt.Tx, userURL *url.URL) (*paste, error) {
 | 
			
		||||
	return shorten(tx, pasteTypeRedirect, []byte(userURL.String()))
 | 
			
		||||
func shortenURL(tx *bolt.Tx, userURL *url.URL) (*db.Paste, error) {
 | 
			
		||||
	return shorten(tx, db.PasteTypeRedirect, []byte(userURL.String()))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Add a paste (of any kind) to the database with arbitrary content.
 | 
			
		||||
func shorten(tx *bolt.Tx, ty pasteType, content []byte) (*paste, error) {
 | 
			
		||||
func shorten(tx *bolt.Tx, ty db.PasteType, content []byte) (*db.Paste, error) {
 | 
			
		||||
	// Generate the paste key
 | 
			
		||||
	pasteKey, err := generatePasteKey(tx)
 | 
			
		||||
	pasteKey, err := db.GeneratePasteKey(tx)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "generating paste key")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Also generate a deleteToken
 | 
			
		||||
	deleteToken, err := generateDeleteToken()
 | 
			
		||||
	deleteToken, err := db.GenerateDeleteToken()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "generating delete token")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Store the new key
 | 
			
		||||
	p := paste{
 | 
			
		||||
	p := db.Paste{
 | 
			
		||||
		Type:        ty,
 | 
			
		||||
		State:       pasteStatePresent,
 | 
			
		||||
		State:       db.PasteStatePresent,
 | 
			
		||||
		Content:     content,
 | 
			
		||||
		Key:         pasteKey,
 | 
			
		||||
		DeleteToken: deleteToken,
 | 
			
		||||
		TimeCreated: time.Now().UTC(),
 | 
			
		||||
	}
 | 
			
		||||
	if err := p.save(tx); err != nil {
 | 
			
		||||
	if err := p.Save(tx); err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
	return &p, nil
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										147
									
								
								internal/db/db.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								internal/db/db.go
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,147 @@
 | 
			
		||||
package db
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"log"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
 | 
			
		||||
	bolt "go.etcd.io/bbolt"
 | 
			
		||||
 | 
			
		||||
	gobmarsh "gitea.hashru.nl/dsprenkels/rushlink/pkg/gobmarsh"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Database is the main rushlink database type.
 | 
			
		||||
//
 | 
			
		||||
// Open a database using DB.Open() and close it in the end using DB.Close().
 | 
			
		||||
// Only one instance of DB should exist in a program at any moment.
 | 
			
		||||
type Database struct {
 | 
			
		||||
	Bolt *bolt.DB
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// CurrentMigrateVersion holds the current "migrate version".
 | 
			
		||||
//
 | 
			
		||||
// If we alter the database format, we bump this number and write a new
 | 
			
		||||
// database migration in migrate().
 | 
			
		||||
const CurrentMigrateVersion = 2
 | 
			
		||||
 | 
			
		||||
// BucketConf holds the name for the "configuration" bucket.
 | 
			
		||||
//
 | 
			
		||||
// This bucket holds the database version, secret site-wide keys, etc.
 | 
			
		||||
const BucketConf = "conf"
 | 
			
		||||
 | 
			
		||||
// BucketPastes holds the name for the pastes bucket.
 | 
			
		||||
const BucketPastes = "pastes"
 | 
			
		||||
 | 
			
		||||
// BucketFileUpload holds the name for the file-upload bucket.
 | 
			
		||||
const BucketFileUpload = "fileUpload"
 | 
			
		||||
 | 
			
		||||
// KeyMigrateVersion stores the current migration version. If this value is less than
 | 
			
		||||
// CurrentMigrateVersion, the database has to be migrated.
 | 
			
		||||
const KeyMigrateVersion = "migrate_version"
 | 
			
		||||
 | 
			
		||||
// OpenDB opens a database file located at path.
 | 
			
		||||
func OpenDB(path string) (*Database, error) {
 | 
			
		||||
	if path == "" {
 | 
			
		||||
		return nil, errors.New("database not set")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	db, err := bolt.Open(path, 0666, &bolt.Options{Timeout: 1 * time.Second})
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrapf(err, "failed to open database at '%v'", path)
 | 
			
		||||
	}
 | 
			
		||||
	if err := db.Update(migrate); err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
	return &Database{db}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Close the bolt database
 | 
			
		||||
func (db *Database) Close() error {
 | 
			
		||||
	if db == nil {
 | 
			
		||||
		panic("no open database")
 | 
			
		||||
	}
 | 
			
		||||
	return db.Close()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Initialize and migrate the database to the current version
 | 
			
		||||
func migrate(tx *bolt.Tx) error {
 | 
			
		||||
	dbVersion, err := dbVersion(tx)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Migrate the database to version 1
 | 
			
		||||
	if dbVersion < 1 {
 | 
			
		||||
		log.Println("migrating database to version 1")
 | 
			
		||||
		// Create conf bucket
 | 
			
		||||
		_, err := tx.CreateBucket([]byte(BucketConf))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		// Create paste bucket
 | 
			
		||||
		_, err = tx.CreateBucket([]byte(BucketPastes))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		// Update the version number
 | 
			
		||||
		if err := setDBVersion(tx, 1); err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if dbVersion < 2 {
 | 
			
		||||
		log.Println("migrating database to version 2")
 | 
			
		||||
		// Create fileUpload bucket
 | 
			
		||||
		_, err := tx.CreateBucket([]byte(BucketFileUpload))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		// Update the version number
 | 
			
		||||
		if err := setDBVersion(tx, 2); err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Get the current migrate version from the database
 | 
			
		||||
func dbVersion(tx *bolt.Tx) (int, error) {
 | 
			
		||||
	conf := tx.Bucket([]byte(BucketConf))
 | 
			
		||||
	if conf == nil {
 | 
			
		||||
		return 0, nil
 | 
			
		||||
	}
 | 
			
		||||
	dbVersionBytes := conf.Get([]byte(KeyMigrateVersion))
 | 
			
		||||
	if dbVersionBytes == nil {
 | 
			
		||||
		return 0, nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Version was already stored
 | 
			
		||||
	var dbVersion int
 | 
			
		||||
	if err := gobmarsh.Unmarshal(dbVersionBytes, &dbVersion); err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
	if dbVersion == 0 {
 | 
			
		||||
		return 0, fmt.Errorf("database version is invalid (%v)", dbVersion)
 | 
			
		||||
	}
 | 
			
		||||
	if dbVersion > CurrentMigrateVersion {
 | 
			
		||||
		return 0, fmt.Errorf("database version is too recent (%v > %v)", dbVersion, CurrentMigrateVersion)
 | 
			
		||||
	}
 | 
			
		||||
	return dbVersion, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Update the current migrate version in the database
 | 
			
		||||
func setDBVersion(tx *bolt.Tx, version int) error {
 | 
			
		||||
	conf, err := tx.CreateBucketIfNotExists([]byte(BucketConf))
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	versionBytes, err := gobmarsh.Marshal(version)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	return conf.Put([]byte(KeyMigrateVersion), versionBytes)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										184
									
								
								internal/db/fileupload.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								internal/db/fileupload.go
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,184 @@
 | 
			
		||||
package db
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/hex"
 | 
			
		||||
	"hash/crc32"
 | 
			
		||||
	"io"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"os"
 | 
			
		||||
	"path"
 | 
			
		||||
 | 
			
		||||
	"github.com/google/uuid"
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
	bolt "go.etcd.io/bbolt"
 | 
			
		||||
 | 
			
		||||
	gobmarsh "gitea.hashru.nl/dsprenkels/rushlink/pkg/gobmarsh"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Use the Castagnoli checksum because of the acceleration on Intel CPUs
 | 
			
		||||
var checksumTable = crc32.MakeTable(crc32.Castagnoli)
 | 
			
		||||
 | 
			
		||||
// FileStore holds the path to a file storage location.
 | 
			
		||||
type FileStore struct {
 | 
			
		||||
	path string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// FileUploadState determines the current state of a FileUpload object.
 | 
			
		||||
type FileUploadState int
 | 
			
		||||
 | 
			
		||||
// FileUpload models an uploaded file.
 | 
			
		||||
type FileUpload struct {
 | 
			
		||||
	State       FileUploadState
 | 
			
		||||
	ID          uuid.UUID
 | 
			
		||||
	FileName    string
 | 
			
		||||
	ContentType string
 | 
			
		||||
	Checksum    uint32
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	dirMode  os.FileMode = 0750
 | 
			
		||||
	fileMode os.FileMode = 0640
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	// FileUploadStateUndef is an undefined FileUpload.
 | 
			
		||||
	FileUploadStateUndef FileUploadState = 0
 | 
			
		||||
	// FileUploadStatePresent denotes the normal (existing) state.
 | 
			
		||||
	FileUploadStatePresent FileUploadState = 1
 | 
			
		||||
	// FileUploadStateDeleted denotes a deleted state.
 | 
			
		||||
	FileUploadStateDeleted FileUploadState = 2
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func (t FileUploadState) String() string {
 | 
			
		||||
	switch t {
 | 
			
		||||
	case FileUploadStateUndef:
 | 
			
		||||
		return "unknown"
 | 
			
		||||
	case FileUploadStatePresent:
 | 
			
		||||
		return "present"
 | 
			
		||||
	case FileUploadStateDeleted:
 | 
			
		||||
		return "deleted"
 | 
			
		||||
	default:
 | 
			
		||||
		return "invalid"
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// OpenFileStore opens the file storage at path.
 | 
			
		||||
func OpenFileStore(path string) (*FileStore, error) {
 | 
			
		||||
	if path == "" {
 | 
			
		||||
		return nil, errors.New("file-store not set")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Try to create the file store directory if it does not yet exist
 | 
			
		||||
	if err := os.MkdirAll(path, dirMode); err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "creating file store directory")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return &FileStore{path[:]}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// NewFileUpload creates a new FileUpload object.
 | 
			
		||||
func NewFileUpload(fs *FileStore, r io.Reader, fileName string, contentType string) (*FileUpload, error) {
 | 
			
		||||
	id, err := uuid.NewRandom()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "generating UUID")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	filePath := fs.FilePath(id, fileName)
 | 
			
		||||
	if err := os.Mkdir(path.Dir(filePath), dirMode); err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "creating file dir")
 | 
			
		||||
	}
 | 
			
		||||
	file, err := os.OpenFile(filePath, os.O_WRONLY|os.O_CREATE, fileMode)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "opening file")
 | 
			
		||||
	}
 | 
			
		||||
	defer file.Close()
 | 
			
		||||
 | 
			
		||||
	hash := crc32.New(checksumTable)
 | 
			
		||||
	tee := io.TeeReader(r, hash)
 | 
			
		||||
	_, err = io.Copy(file, tee)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.Wrap(err, "writing to file")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	fu := &FileUpload{
 | 
			
		||||
		State:       FileUploadStatePresent,
 | 
			
		||||
		ID:          id,
 | 
			
		||||
		FileName:    fileName,
 | 
			
		||||
		ContentType: contentType,
 | 
			
		||||
		Checksum:    hash.Sum32(),
 | 
			
		||||
	}
 | 
			
		||||
	return fu, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (fs *FileStore) Path() string {
 | 
			
		||||
	return fs.path
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (fs *FileStore) FilePath(id uuid.UUID, fileName string) string {
 | 
			
		||||
	if fs.path == "" {
 | 
			
		||||
		panic("fileStoreDir called while the file store path has not been set")
 | 
			
		||||
	}
 | 
			
		||||
	return path.Join(fs.path, hex.EncodeToString(id[:]), fileName)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func GetFileUpload(tx *bolt.Tx, id uuid.UUID) (*FileUpload, error) {
 | 
			
		||||
	bucket := tx.Bucket([]byte(BucketFileUpload))
 | 
			
		||||
	if bucket == nil {
 | 
			
		||||
		return nil, errors.Errorf("bucket %v does not exist", BucketFileUpload)
 | 
			
		||||
	}
 | 
			
		||||
	storedBytes := bucket.Get(id[:])
 | 
			
		||||
	if storedBytes == nil {
 | 
			
		||||
		return nil, nil
 | 
			
		||||
	}
 | 
			
		||||
	fu := &FileUpload{}
 | 
			
		||||
	err := gobmarsh.Unmarshal(storedBytes, fu)
 | 
			
		||||
	return fu, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Save saves a FileUpload in the database.
 | 
			
		||||
func (fu *FileUpload) Save(tx *bolt.Tx) error {
 | 
			
		||||
	bucket := tx.Bucket([]byte(BucketFileUpload))
 | 
			
		||||
	if bucket == nil {
 | 
			
		||||
		return errors.Errorf("bucket %v does not exist", BucketFileUpload)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	buf, err := gobmarsh.Marshal(fu)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.Wrap(err, "encoding for database failed")
 | 
			
		||||
	}
 | 
			
		||||
	if err := bucket.Put(fu.ID[:], buf); err != nil {
 | 
			
		||||
		return errors.Wrap(err, "database transaction failed")
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Delete deletes a FileUpload from the database.
 | 
			
		||||
func (fu *FileUpload) Delete(tx *bolt.Tx, fs *FileStore) error {
 | 
			
		||||
	// Remove the file in the backend
 | 
			
		||||
	filePath := fs.FilePath(fu.ID, fu.FileName)
 | 
			
		||||
	if err := os.Remove(filePath); err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Update the file in the server
 | 
			
		||||
	if err := (&FileUpload{
 | 
			
		||||
		ID:    fu.ID,
 | 
			
		||||
		State: FileUploadStateDeleted,
 | 
			
		||||
	}).Save(tx); err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Cleanup the parent directory
 | 
			
		||||
	wrap := "deletion succeeded, but removing the file directory has failed"
 | 
			
		||||
	return errors.Wrap(os.Remove(path.Dir(filePath)), wrap)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// URL returns the URL for the FileUpload.
 | 
			
		||||
func (fu *FileUpload) URL() *url.URL {
 | 
			
		||||
	rawurl := "/uploads/" + hex.EncodeToString(fu.ID[:]) + "/" + fu.FileName
 | 
			
		||||
	urlParse, err := url.Parse(rawurl)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		panic("could not construct /uploads/ url")
 | 
			
		||||
	}
 | 
			
		||||
	return urlParse
 | 
			
		||||
}
 | 
			
		||||
@ -1,97 +1,106 @@
 | 
			
		||||
package rushlink
 | 
			
		||||
package db
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"crypto/rand"
 | 
			
		||||
	"encoding/base64"
 | 
			
		||||
	"encoding/hex"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	gobmarsh "gitea.hashru.nl/dsprenkels/rushlink/pkg/gobmarsh"
 | 
			
		||||
	"github.com/google/uuid"
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
	bolt "go.etcd.io/bbolt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type pasteType int
 | 
			
		||||
type pasteState int
 | 
			
		||||
type PasteType int
 | 
			
		||||
type PasteState int
 | 
			
		||||
 | 
			
		||||
type paste struct {
 | 
			
		||||
	Type        pasteType
 | 
			
		||||
	State       pasteState
 | 
			
		||||
type Paste struct {
 | 
			
		||||
	Type        PasteType
 | 
			
		||||
	State       PasteState
 | 
			
		||||
	Content     []byte
 | 
			
		||||
	Key         string
 | 
			
		||||
	DeleteToken string
 | 
			
		||||
	TimeCreated time.Time
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Note: we use iota here. That means removals of pasteType* are not allowed,
 | 
			
		||||
// ReservedPasteKeys keys are designated reserved, and will not be randomly chosen
 | 
			
		||||
var ReservedPasteKeys = []string{"xd42", "example"}
 | 
			
		||||
 | 
			
		||||
// Note: we use iota here. That means removals of PasteType* are not allowed,
 | 
			
		||||
// because this changes the value of the constant. Please add the comment
 | 
			
		||||
// "// deprecated" if you want to remove the constant. Additions are only
 | 
			
		||||
// allowed at the bottom of this block, for the same reason.
 | 
			
		||||
const (
 | 
			
		||||
	pasteTypeUndef pasteType = iota
 | 
			
		||||
	pasteTypePaste
 | 
			
		||||
	pasteTypeRedirect
 | 
			
		||||
	pasteTypeFileUpload
 | 
			
		||||
	PasteTypeUndef PasteType = iota
 | 
			
		||||
	PasteTypePaste
 | 
			
		||||
	PasteTypeRedirect
 | 
			
		||||
	PasteTypeFileUpload
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Note: we use iota here. See the comment above pasteType*
 | 
			
		||||
// Note: we use iota here. See the comment above PasteType*
 | 
			
		||||
const (
 | 
			
		||||
	pasteStateUndef pasteState = iota
 | 
			
		||||
	pasteStatePresent
 | 
			
		||||
	pasteStateDeleted
 | 
			
		||||
	PasteStateUndef PasteState = iota
 | 
			
		||||
	PasteStatePresent
 | 
			
		||||
	PasteStateDeleted
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func (t pasteType) String() string {
 | 
			
		||||
// Base64 encoding and decoding
 | 
			
		||||
var base64Alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_"
 | 
			
		||||
var base64Encoder = base64.RawURLEncoding.WithPadding(base64.NoPadding)
 | 
			
		||||
 | 
			
		||||
func (t PasteType) String() string {
 | 
			
		||||
	switch t {
 | 
			
		||||
	case pasteTypeUndef:
 | 
			
		||||
	case PasteTypeUndef:
 | 
			
		||||
		return "unknown"
 | 
			
		||||
	case pasteTypePaste:
 | 
			
		||||
	case PasteTypePaste:
 | 
			
		||||
		return "paste"
 | 
			
		||||
	case pasteTypeRedirect:
 | 
			
		||||
	case PasteTypeRedirect:
 | 
			
		||||
		return "redirect"
 | 
			
		||||
	case pasteTypeFileUpload:
 | 
			
		||||
	case PasteTypeFileUpload:
 | 
			
		||||
		return "file"
 | 
			
		||||
	default:
 | 
			
		||||
		return "invalid"
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (t pasteState) String() string {
 | 
			
		||||
func (t PasteState) String() string {
 | 
			
		||||
	switch t {
 | 
			
		||||
	case pasteStateUndef:
 | 
			
		||||
	case PasteStateUndef:
 | 
			
		||||
		return "unknown"
 | 
			
		||||
	case pasteStatePresent:
 | 
			
		||||
	case PasteStatePresent:
 | 
			
		||||
		return "present"
 | 
			
		||||
	case pasteStateDeleted:
 | 
			
		||||
	case PasteStateDeleted:
 | 
			
		||||
		return "deleted"
 | 
			
		||||
	default:
 | 
			
		||||
		return "invalid"
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Retrieve a paste from the database
 | 
			
		||||
func getPaste(tx *bolt.Tx, key string) (*paste, error) {
 | 
			
		||||
	pastesBucket := tx.Bucket([]byte(BUCKET_PASTES))
 | 
			
		||||
// GetPaste retrieves a paste from the database.
 | 
			
		||||
func GetPaste(tx *bolt.Tx, key string) (*Paste, error) {
 | 
			
		||||
	pastesBucket := tx.Bucket([]byte(BucketPastes))
 | 
			
		||||
	if pastesBucket == nil {
 | 
			
		||||
		return nil, errors.Errorf("bucket %v does not exist", BUCKET_PASTES)
 | 
			
		||||
		return nil, errors.Errorf("bucket %v does not exist", BucketPastes)
 | 
			
		||||
	}
 | 
			
		||||
	storedBytes := pastesBucket.Get([]byte(key))
 | 
			
		||||
	if storedBytes == nil {
 | 
			
		||||
		return nil, nil
 | 
			
		||||
	}
 | 
			
		||||
	p := &paste{}
 | 
			
		||||
	err := Unmarshal(storedBytes, p)
 | 
			
		||||
	p := &Paste{}
 | 
			
		||||
	err := gobmarsh.Unmarshal(storedBytes, p)
 | 
			
		||||
	return p, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (p *paste) save(tx *bolt.Tx) error {
 | 
			
		||||
	bucket := tx.Bucket([]byte(BUCKET_PASTES))
 | 
			
		||||
func (p *Paste) Save(tx *bolt.Tx) error {
 | 
			
		||||
	bucket := tx.Bucket([]byte(BucketPastes))
 | 
			
		||||
	if bucket == nil {
 | 
			
		||||
		return errors.Errorf("bucket %v does not exist", BUCKET_PASTES)
 | 
			
		||||
		return errors.Errorf("bucket %v does not exist", BucketPastes)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	buf, err := Marshal(p)
 | 
			
		||||
	buf, err := gobmarsh.Marshal(p)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.Wrap(err, "encoding for database failed")
 | 
			
		||||
	}
 | 
			
		||||
@ -101,39 +110,39 @@ func (p *paste) save(tx *bolt.Tx) error {
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (p *paste) delete(tx *bolt.Tx) error {
 | 
			
		||||
func (p *Paste) Delete(tx *bolt.Tx, fs *FileStore) error {
 | 
			
		||||
	// Remove the (maybe) attached file
 | 
			
		||||
	if p.Type == pasteTypeFileUpload {
 | 
			
		||||
	if p.Type == PasteTypeFileUpload {
 | 
			
		||||
		fuID, err := uuid.FromBytes(p.Content)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return errors.Wrap(err, "failed to parse uuid")
 | 
			
		||||
		}
 | 
			
		||||
		fu, err := getFileUpload(tx, fuID)
 | 
			
		||||
		fu, err := GetFileUpload(tx, fuID)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return errors.Wrap(err, "failed to find file in database")
 | 
			
		||||
		}
 | 
			
		||||
		if err := fu.delete(tx); err != nil {
 | 
			
		||||
		if err := fu.Delete(tx, fs); err != nil {
 | 
			
		||||
			return errors.Wrap(err, "failed to remove file")
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Replace the old paste with a new empty paste
 | 
			
		||||
	p.Type = pasteTypeUndef
 | 
			
		||||
	p.State = pasteStateDeleted
 | 
			
		||||
	p.Type = PasteTypeUndef
 | 
			
		||||
	p.State = PasteStateDeleted
 | 
			
		||||
	p.Content = []byte{}
 | 
			
		||||
	if err := p.save(tx); err != nil {
 | 
			
		||||
	if err := p.Save(tx); err != nil {
 | 
			
		||||
		return errors.Wrap(err, "failed to delete paste in database")
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Get the URL from this paste.
 | 
			
		||||
// RedirectURL returns the URL from this paste.
 | 
			
		||||
//
 | 
			
		||||
// This function assumes that the paste is valid.  If the paste struct is
 | 
			
		||||
// corrupted in some way, this function will panic.
 | 
			
		||||
func (p *paste) redirectURL() *url.URL {
 | 
			
		||||
	if p.Type != pasteTypeRedirect {
 | 
			
		||||
		panic("expected p.Type to be pasteTypeRedirect")
 | 
			
		||||
func (p *Paste) RedirectURL() *url.URL {
 | 
			
		||||
	if p.Type != PasteTypeRedirect {
 | 
			
		||||
		panic("expected p.Type to be PasteTypeRedirect")
 | 
			
		||||
	}
 | 
			
		||||
	rawurl := string(p.Content)
 | 
			
		||||
	urlParse, err := url.Parse(rawurl)
 | 
			
		||||
@ -143,12 +152,13 @@ func (p *paste) redirectURL() *url.URL {
 | 
			
		||||
	return urlParse
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Generate a key until it is not in the database, this occurs in O(log N),
 | 
			
		||||
// where N is the amount of keys stored in the url-shorten database.
 | 
			
		||||
func generatePasteKey(tx *bolt.Tx) (string, error) {
 | 
			
		||||
	pastesBucket := tx.Bucket([]byte(BUCKET_PASTES))
 | 
			
		||||
// GeneratePasteKey generates a key until it is not in the database, the
 | 
			
		||||
// running time of this function is in O(log N), where N is the amount of
 | 
			
		||||
// keys stored in the url-shorten database.
 | 
			
		||||
func GeneratePasteKey(tx *bolt.Tx) (string, error) {
 | 
			
		||||
	pastesBucket := tx.Bucket([]byte(BucketPastes))
 | 
			
		||||
	if pastesBucket == nil {
 | 
			
		||||
		return "", errors.Errorf("bucket %v does not exist", BUCKET_PASTES)
 | 
			
		||||
		return "", errors.Errorf("bucket %v does not exist", BucketPastes)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	epoch := 0
 | 
			
		||||
@ -216,7 +226,7 @@ func generatePasteKeyInner(epoch int) (string, error) {
 | 
			
		||||
	return string(urlKey), nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func generateDeleteToken() (string, error) {
 | 
			
		||||
func GenerateDeleteToken() (string, error) {
 | 
			
		||||
	var deleteToken [16]byte
 | 
			
		||||
	_, err := rand.Read(deleteToken[:])
 | 
			
		||||
	if err != nil {
 | 
			
		||||
@ -5,6 +5,8 @@ import (
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
 | 
			
		||||
 | 
			
		||||
	"github.com/gorilla/mux"
 | 
			
		||||
	"github.com/pkg/errors"
 | 
			
		||||
	"github.com/prometheus/client_golang/prometheus"
 | 
			
		||||
@ -13,7 +15,7 @@ import (
 | 
			
		||||
	bolt "go.etcd.io/bbolt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func StartMetricsServer(addr string) {
 | 
			
		||||
func StartMetricsServer(addr string, db *db.Database) {
 | 
			
		||||
	var (
 | 
			
		||||
		_ = promauto.NewGaugeFunc(prometheus.GaugeOpts{
 | 
			
		||||
			Namespace: "rushlink",
 | 
			
		||||
@ -22,7 +24,7 @@ func StartMetricsServer(addr string) {
 | 
			
		||||
			Help:      "The current amount of pastes in the database.",
 | 
			
		||||
		}, func() float64 {
 | 
			
		||||
			var metric float64
 | 
			
		||||
			if err := DB.View(func(tx *bolt.Tx) error {
 | 
			
		||||
			if err := db.Bolt.View(func(tx *bolt.Tx) error {
 | 
			
		||||
				bucket := tx.Bucket([]byte("pastes"))
 | 
			
		||||
				if bucket == nil {
 | 
			
		||||
					return errors.New("bucket 'pastes' could not be found")
 | 
			
		||||
 | 
			
		||||
@ -7,6 +7,7 @@ import (
 | 
			
		||||
	"encoding/gob"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Marshal serializes the value in v to a byte buffer.
 | 
			
		||||
func Marshal(v interface{}) ([]byte, error) {
 | 
			
		||||
	b := new(bytes.Buffer)
 | 
			
		||||
	err := gob.NewEncoder(b).Encode(v)
 | 
			
		||||
@ -16,6 +17,7 @@ func Marshal(v interface{}) ([]byte, error) {
 | 
			
		||||
	return b.Bytes(), nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Unmarshal deserializes the data in data into the object in v.
 | 
			
		||||
func Unmarshal(data []byte, v interface{}) error {
 | 
			
		||||
	b := bytes.NewBuffer(data)
 | 
			
		||||
	return gob.NewDecoder(b).Decode(v)
 | 
			
		||||
							
								
								
									
										29
									
								
								router.go
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								router.go
									
									
									
									
									
								
							@ -7,9 +7,15 @@ import (
 | 
			
		||||
	"runtime/debug"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"gitea.hashru.nl/dsprenkels/rushlink/internal/db"
 | 
			
		||||
	"github.com/gorilla/mux"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type rushlink struct {
 | 
			
		||||
	db *db.Database
 | 
			
		||||
	fs *db.FileStore
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func recoveryMiddleware(next http.Handler) http.Handler {
 | 
			
		||||
	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | 
			
		||||
		defer func() {
 | 
			
		||||
@ -31,18 +37,23 @@ func recoveryMiddleware(next http.Handler) http.Handler {
 | 
			
		||||
	})
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func StartMainServer(addr string) {
 | 
			
		||||
func StartMainServer(addr string, db *db.Database, fs *db.FileStore) {
 | 
			
		||||
	rl := rushlink{
 | 
			
		||||
		db: db,
 | 
			
		||||
		fs: fs,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Initialize Gorilla router
 | 
			
		||||
	router := mux.NewRouter()
 | 
			
		||||
	router.Use(recoveryMiddleware)
 | 
			
		||||
	router.HandleFunc("/", indexGetHandler).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/", newPasteHandler).Methods("POST")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}", viewPasteHandler).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/nr", viewPasteHandlerNoRedirect).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/meta", viewPasteHandlerMeta).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}", deletePasteHandler).Methods("DELETE")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/delete", deletePasteHandler).Methods("POST")
 | 
			
		||||
	router.HandleFunc("/uploads/{id:[A-Za-z0-9-_]+}/{filename:.+}", uploadFileGetHandler).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/", rl.indexGetHandler).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/", rl.newPasteHandler).Methods("POST")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}", rl.viewPasteHandler).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/nr", rl.viewPasteHandlerNoRedirect).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/meta", rl.viewPasteHandlerMeta).Methods("GET")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}", rl.deletePasteHandler).Methods("DELETE")
 | 
			
		||||
	router.HandleFunc("/{key:[A-Za-z0-9-_]{4,}}/delete", rl.deletePasteHandler).Methods("POST")
 | 
			
		||||
	router.HandleFunc("/uploads/{id:[A-Za-z0-9-_]+}/{filename:.+}", rl.uploadFileGetHandler).Methods("GET")
 | 
			
		||||
 | 
			
		||||
	srv := &http.Server{
 | 
			
		||||
		Handler:      router,
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user