contented/Server.go

224 lines
5.3 KiB
Go
Raw Normal View History

2017-10-06 07:02:57 +00:00
package contented
import (
"embed"
2017-10-06 07:02:57 +00:00
"encoding/json"
2023-05-19 07:13:55 +00:00
"fmt"
"io/fs"
2017-10-06 07:02:57 +00:00
"log"
"net/http"
"os"
2017-11-18 00:11:39 +00:00
"regexp"
2017-10-06 07:02:57 +00:00
"strings"
2017-10-08 03:34:11 +00:00
"time"
2017-10-06 07:02:57 +00:00
2023-05-19 07:13:55 +00:00
"github.com/minio/minio-go/v7"
"github.com/minio/minio-go/v7/pkg/credentials"
2017-10-08 02:06:46 +00:00
"github.com/mxk/go-flowrate/flowrate"
bolt "go.etcd.io/bbolt"
2017-10-06 07:02:57 +00:00
)
//go:embed static
var staticAssets embed.FS
2017-10-08 03:30:52 +00:00
var SERVER_HEADER string = `contented/0.0.0-dev`
2017-10-06 07:02:57 +00:00
const (
DEFAULT_MAX_CONCURRENT_THUMBS = 16
DEFAULT_MAX_THUMBSIZE = 20 * 1024 * 1024 // 20 MiB
ALBUM_MIMETYPE = `contented/album`
2023-05-19 07:13:55 +00:00
STORAGE_LOCAL int = 0
STORAGE_S3 int = 1
)
type ServerPublicProperties struct {
AppTitle string
MaxUploadBytes int64
CanonicalBaseURL string
2017-10-06 07:02:57 +00:00
}
type ServerOptions struct {
2023-05-19 07:13:55 +00:00
StorageType int // STORAGE_xx
DataDirectory string
DataS3Options struct {
Hostname string
AccessKey string
SecretKey string
Bucket string
Prefix string
}
DBPath string
DiskFilesWorldReadable bool
BandwidthLimit int64
TrustXForwardedFor bool
EnableHomepage bool
EnableUpload bool
MaxConcurrentThumbs int
MaxThumbSizeBytes int64
ServerPublicProperties
}
func (this *ServerOptions) FileMode() os.FileMode {
if this.DiskFilesWorldReadable {
return 0644
} else {
return 0600
}
}
2017-10-06 07:02:57 +00:00
type Server struct {
opts ServerOptions
2023-05-19 07:13:55 +00:00
s3client *minio.Client
2017-10-06 07:02:57 +00:00
db *bolt.DB
2017-10-08 03:34:11 +00:00
startTime time.Time
thumbnailSem chan struct{}
2017-10-06 07:02:57 +00:00
metadataBucket []byte
staticDir fs.FS // interface
2017-10-06 07:02:57 +00:00
}
func NewServer(opts *ServerOptions) (*Server, error) {
s := &Server{
opts: *opts,
metadataBucket: []byte(`METADATA`),
2017-10-08 03:34:11 +00:00
startTime: time.Now(),
2017-10-06 07:02:57 +00:00
}
if s.opts.MaxConcurrentThumbs <= 0 {
s.opts.MaxConcurrentThumbs = DEFAULT_MAX_CONCURRENT_THUMBS // default
log.Printf("Allowing %d concurrent thumbnails", s.opts.MaxConcurrentThumbs)
}
if s.opts.MaxThumbSizeBytes <= 0 {
s.opts.MaxThumbSizeBytes = DEFAULT_MAX_THUMBSIZE
log.Printf("Allowing thumbnails for files up to %d byte(s)", s.opts.MaxThumbSizeBytes)
}
s.staticDir, _ = fs.Sub(staticAssets, `static`) // can't fail
2023-05-19 07:13:55 +00:00
// Maybe open s3 connection
if s.opts.StorageType == STORAGE_S3 {
cl, err := minio.New(opts.DataS3Options.Hostname, &minio.Options{
Creds: credentials.NewStaticV4(opts.DataS3Options.AccessKey, opts.DataS3Options.SecretKey, ""),
Secure: true,
})
if err != nil {
return nil, fmt.Errorf("Connecting to S3 host: %w", err)
}
s.s3client = cl
}
// "fill" the thumbnailer semaphore
s.thumbnailSem = make(chan struct{}, s.opts.MaxConcurrentThumbs)
for i := 0; i < s.opts.MaxConcurrentThumbs; i += 1 {
s.thumbnailSem <- struct{}{}
}
2017-10-06 07:02:57 +00:00
b, err := bolt.Open(opts.DBPath, 0644, bolt.DefaultOptions)
if err != nil {
return nil, err
}
err = b.Update(func(tx *bolt.Tx) error {
_, err := tx.CreateBucketIfNotExists(s.metadataBucket)
return err
})
if err != nil {
return nil, err
}
s.db = b
return s, nil
}
func (this *Server) serveJsonObject(w http.ResponseWriter, o interface{}) {
jb, err := json.Marshal(o)
2017-10-06 07:02:57 +00:00
if err != nil {
log.Println(err.Error())
http.Error(w, "Internal error", 500)
return
}
w.Header().Set(`Content-Type`, `application/json`)
w.WriteHeader(200)
w.Write(jb)
}
func (this *Server) handleAbout(w http.ResponseWriter) {
this.serveJsonObject(w, this.opts.ServerPublicProperties)
}
2017-10-15 06:09:14 +00:00
func (this *Server) remoteIP(r *http.Request) string {
if this.opts.TrustXForwardedFor {
if xff := r.Header.Get("X-Forwarded-For"); len(xff) > 0 {
return xff
}
}
return strings.TrimRight(strings.TrimRight(r.RemoteAddr, "0123456789"), ":")
}
const (
downloadUrlPrefix = `/get/`
metadataUrlPrefix = `/info/`
2017-11-18 00:48:34 +00:00
previewUrlPrefix = `/p/`
)
2017-11-18 00:11:39 +00:00
var rxThumbUrl = regexp.MustCompile(`^/thumb/(.)/(.*)$`)
2017-10-06 07:02:57 +00:00
func (this *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
w.Header().Set(`Server`, SERVER_HEADER)
2017-10-08 04:06:24 +00:00
w.Header().Set(`Access-Control-Allow-Origin`, `*`) // Blanket allow CORS
2017-10-06 07:02:57 +00:00
if this.opts.MaxUploadBytes > 0 {
r.Body = http.MaxBytesReader(w, r.Body, this.opts.MaxUploadBytes)
}
2017-10-08 02:06:46 +00:00
if this.opts.BandwidthLimit > 0 {
r.Body = flowrate.NewReader(r.Body, this.opts.BandwidthLimit)
}
2017-10-06 07:02:57 +00:00
//
if r.Method == "GET" && strings.HasPrefix(r.URL.Path, downloadUrlPrefix) {
this.handleView(w, r, r.URL.Path[len(downloadUrlPrefix):])
} else if r.Method == "GET" && strings.HasPrefix(r.URL.Path, metadataUrlPrefix) {
this.handleInformation(w, r.URL.Path[len(metadataUrlPrefix):])
2017-10-06 07:02:57 +00:00
2017-11-18 00:48:34 +00:00
} else if r.Method == "GET" && strings.HasPrefix(r.URL.Path, previewUrlPrefix) {
2023-05-19 07:13:31 +00:00
this.handlePreview(r.Context(), w, r.URL.Path[len(previewUrlPrefix):])
2017-11-18 00:48:34 +00:00
2017-11-18 00:11:39 +00:00
} else if r.Method == "GET" && rxThumbUrl.MatchString(r.URL.Path) {
parts := rxThumbUrl.FindStringSubmatch(r.URL.Path)
this.handleThumb(w, r, parts[1][0], parts[2])
} else if r.Method == "GET" && r.URL.Path == `/about` {
this.handleAbout(w)
2017-10-06 07:02:57 +00:00
} else if r.Method == "POST" && r.URL.Path == `/upload` {
2017-10-08 01:10:15 +00:00
this.handleUpload(w, r)
2017-10-06 07:02:57 +00:00
2017-10-08 03:42:06 +00:00
} else if r.Method == "OPTIONS" {
2017-10-08 04:06:24 +00:00
// Blanket allow (headers already set)
2017-10-08 03:42:06 +00:00
w.WriteHeader(200)
} else if r.Method == "GET" {
// Conditionally block homepage access
if !this.opts.EnableHomepage && (r.URL.Path == `/index.html` || r.URL.Path == `/`) {
http.Error(w, "Not found", 404)
return
}
2017-10-08 03:54:26 +00:00
// Serve static html/css/js assets
// http.FileServer transparently redirects index.html->/ internally
http.FileServer(http.FS(this.staticDir)).ServeHTTP(w, r)
2017-10-06 07:02:57 +00:00
2017-10-08 03:34:11 +00:00
} else {
http.Error(w, "Not found", 404)
2017-10-06 07:02:57 +00:00
}
}