hardfiles/main.go

279 lines
6.1 KiB
Go
Raw Normal View History

2023-09-30 23:06:22 +00:00
package main
import (
"io"
"math/rand"
"net/http"
"os"
"strconv"
"time"
"github.com/BurntSushi/toml"
"github.com/gabriel-vasile/mimetype"
"github.com/gorilla/mux"
"github.com/landlock-lsm/go-landlock/landlock"
"github.com/rs/zerolog"
"github.com/rs/zerolog/log"
2023-11-01 01:19:21 +00:00
bolt "go.etcd.io/bbolt"
2023-09-30 23:06:22 +00:00
)
var (
db *bolt.DB
conf Config
)
type Config struct {
2023-12-10 05:32:53 +00:00
Webroot string `toml:"webroot"`
LPort string `toml:"lport"`
VHost string `toml:"vhost"`
DBFile string `toml:"dbfile"`
FileLen int `toml:"filelen"`
FileFolder string `toml:"folder"`
FileExpirySeconds int `toml:"fileexpiry"`
2023-09-30 23:06:22 +00:00
}
func LoadConf() {
if _, err := toml.DecodeFile("config.toml", &conf); err != nil {
log.Fatal().Err(err).Msg("unable to parse config.toml")
}
}
2023-11-01 13:11:34 +00:00
func Shred(path string) error {
2023-11-01 01:19:21 +00:00
fileinfo, err := os.Stat(path)
if err != nil {
return err
}
size := fileinfo.Size()
2023-11-01 13:11:34 +00:00
err = Scramble(path, size)
2023-11-01 01:19:21 +00:00
if err != nil {
return err
}
2023-11-01 13:11:34 +00:00
err = Zeros(path, size)
2023-11-01 01:19:21 +00:00
if err != nil {
return err
}
err = os.Remove(path)
if err != nil {
return err
}
return nil
}
2023-11-01 13:11:34 +00:00
func Scramble(path string, size int64) error {
2023-11-01 01:19:21 +00:00
var i int64
for i = 0; i < 7; i++ { // 7 iterations
file, err := os.OpenFile(path, os.O_RDWR, 0)
if err != nil {
return err
}
2023-11-01 13:11:34 +00:00
defer file.Close()
2023-11-01 01:19:21 +00:00
offset, err := file.Seek(0, 0)
if err != nil {
return err
}
buff := make([]byte, size)
rand.Read(buff)
file.WriteAt(buff, offset)
file.Close()
}
return nil
}
2023-11-01 13:11:34 +00:00
func Zeros(path string, size int64) error {
2023-11-01 01:19:21 +00:00
file, err := os.OpenFile(path, os.O_RDWR, 0)
if err != nil {
return err
}
2023-11-01 13:11:34 +00:00
defer file.Close()
2023-11-01 01:19:21 +00:00
offset, err := file.Seek(0, 0)
if err != nil {
return err
}
buff := make([]byte, size)
file.WriteAt(buff, offset)
return nil
}
2023-09-30 23:06:22 +00:00
func NameGen() string {
const chars = "abcdefghjkmnpqrstuvwxyzABCDEFGHJKLMNPQRSTUVWXYZ123456789"
ll := len(chars)
b := make([]byte, conf.FileLen)
rand.Read(b) // generates len(b) random bytes
for i := int64(0); i < int64(conf.FileLen); i++ {
b[i] = chars[int(b[i])%ll]
}
return string(b)
}
func CheckFile(name string) bool { // false if doesn't exist, true if exists
tfd, err := os.Open(conf.FileFolder + "/" + name)
if err != nil {
return false
}
tfd.Close()
return true
}
func UploadHandler(w http.ResponseWriter, r *http.Request) {
// expiry sanitize
2023-12-10 05:32:53 +00:00
twentyfour := int64(conf.FileExpirySeconds)
2023-09-30 23:06:22 +00:00
file, _, err := r.FormFile("file")
if err != nil {
w.WriteHeader(http.StatusBadRequest)
return
}
defer file.Close()
mtype, err := mimetype.DetectReader(file)
if err != nil {
w.Write([]byte("error detecting the mime type of your file\n"))
return
}
file.Seek(0, 0)
// generate + check name
var name string
for {
id := NameGen()
name = id + mtype.Extension()
if !CheckFile(name) {
break
}
}
err = db.Update(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte("expiry"))
err := b.Put([]byte(name), []byte(strconv.FormatInt(time.Now().Unix()+twentyfour, 10)))
return err
})
if err != nil {
log.Error().Err(err).Msg("Failed to put expiry")
}
log.Info().Int64("expiry", twentyfour).Msg("Writing new file")
f, err := os.OpenFile(conf.FileFolder+"/"+name, os.O_WRONLY|os.O_CREATE, 0644)
if err != nil {
log.Error().Err(err).Msg("Error opening a file for write")
w.WriteHeader(http.StatusInternalServerError) // change to json
return
}
defer f.Close()
io.Copy(f, file)
w.Write([]byte("https://" + conf.VHost + "/uploads/" + name))
}
func Cull() {
for {
removed := 0
db.Update(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte("expiry"))
c := b.Cursor()
for k, v := c.First(); k != nil; k, v = c.Next() {
eol, err := strconv.ParseInt(string(v), 10, 64)
if err != nil {
log.Error().Err(err).Bytes("k", k).Bytes("v", v).Msg("expiration time could not be parsed")
continue
}
if time.Now().After(time.Unix(eol, 0)) {
2023-11-01 13:11:34 +00:00
if err := Shred(conf.FileFolder + "/" + string(k)); err != nil {
2023-11-01 01:19:21 +00:00
log.Error().Err(err).Msg("shredding failed")
} else {
removed += 1
}
2023-09-30 23:06:22 +00:00
c.Delete()
}
}
return nil
})
if removed >= 1 {
2023-11-01 13:11:34 +00:00
log.Info().Int("amount", removed).Msg("shredded")
2023-09-30 23:06:22 +00:00
}
time.Sleep(5 * time.Second)
}
}
func main() {
log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr})
LoadConf()
err := landlock.V2.BestEffort().RestrictPaths(
2023-11-01 13:11:34 +00:00
landlock.RWDirs(conf.FileFolder),
2023-11-01 01:19:21 +00:00
landlock.RWDirs(conf.Webroot),
2023-09-30 23:06:22 +00:00
landlock.RWFiles(conf.DBFile),
)
if err != nil {
log.Warn().Err(err).Msg("could not landlock")
}
_, err = os.Open("/etc/passwd")
if err == nil {
log.Warn().Msg("landlock failed, could open /etc/passwd")
} else {
log.Info().Err(err).Msg("Landlocked")
}
db, err = bolt.Open(conf.DBFile, 0600, nil)
if err != nil {
log.Fatal().Err(err).Msg("unable to open database file")
}
db.Update(func(tx *bolt.Tx) error {
_, err := tx.CreateBucketIfNotExists([]byte("expiry"))
if err != nil {
log.Fatal().Err(err).Msg("error creating expiry bucket")
return err
}
return nil
})
r := mux.NewRouter()
r.HandleFunc("/", UploadHandler).Methods("POST")
2023-11-01 13:11:34 +00:00
r.HandleFunc("/uploads/{name}", func(w http.ResponseWriter, r *http.Request) {
2023-09-30 23:06:22 +00:00
vars := mux.Vars(r)
if !CheckFile(vars["name"]) {
w.WriteHeader(http.StatusNotFound)
} else {
http.ServeFile(w, r, conf.FileFolder+"/"+vars["name"])
}
}).Methods("GET")
r.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, conf.Webroot+"/index.html")
}).Methods("GET")
2023-11-01 13:11:34 +00:00
r.HandleFunc("/index.html", func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, conf.Webroot+"/index.html")
}).Methods("GET")
r.HandleFunc("/fist.ico", func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, conf.Webroot+"/fist.ico")
}).Methods("GET")
2023-11-16 02:43:18 +00:00
r.HandleFunc("/header.png", func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, conf.Webroot+"/header.png")
}).Methods("GET")
2023-09-30 23:06:22 +00:00
http.Handle("/", r)
go Cull()
serv := &http.Server{
2023-11-01 13:11:34 +00:00
Addr: ":" + conf.LPort,
Handler: r,
ErrorLog: nil,
2023-09-30 23:06:22 +00:00
IdleTimeout: 20 * time.Second,
}
2023-11-01 13:11:34 +00:00
log.Info().Err(err).Msg("listening on port " + conf.LPort + "...")
2023-09-30 23:06:22 +00:00
if err := serv.ListenAndServe(); err != nil {
log.Fatal().Err(err).Msg("error starting server")
}
db.Close()
}