summaryrefslogtreecommitdiff
path: root/main.go
blob: 2c15ed641eb93ae59bd949827752dc4fe0943e83 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
package main

import (
	"context"
	"flag"
	"io/fs"
	"log"
	"log/slog"
	"net/http"
	"os"
	"os/signal"
	"path/filepath"
	"time"

	"go.neonxp.ru/mux"
	"go.neonxp.ru/mux/middleware"
	"golang.org/x/sync/errgroup"
)

var (
	host     string
	listen   string
	path     string
	htmlPath string
	ttl      time.Duration
)

func init() {
	flag.StringVar(&host, "host", "https://nixshare.ru/", "host of nixshare")
	flag.StringVar(&listen, "listen", ":8000", "port to listen")
	flag.StringVar(&htmlPath, "html_path", "./html", "html directory")
	flag.StringVar(&path, "path", "./storage", "storage directory")
	flag.DurationVar(&ttl, "ttl", 24*time.Hour, "time to delete uploaded file")
}

func main() {
	ctx, cancel := signal.NotifyContext(context.Background(), os.Interrupt, os.Kill)
	defer cancel()

	flag.Parse()

	r := http.NewServeMux()

	r.Handle("POST /upload", http.HandlerFunc(uploadHandler))
	r.Handle("/", http.FileServer(http.Dir("html")))
	r.Handle("/s/", http.StripPrefix("/s/", http.FileServer(http.Dir(path))))

	srv := http.Server{
		Addr: listen,
		Handler: mux.Use(r,
			middleware.Recover(slog.Default()),
			middleware.RequestID,
			middleware.Logger(slog.Default()),
		),
	}

	eg, egCtx := errgroup.WithContext(ctx)

	eg.Go(func() error {
		<-egCtx.Done()
		srv.Close()

		return nil
	})

	eg.Go(func() error {
		if err := srv.ListenAndServe(); err != http.ErrServerClosed {
			return err
		}

		return nil
	})

	eg.Go(func() error {
		timer := time.NewTimer(1 * time.Minute)
		defer timer.Stop()
		for {
			select {
			case <-timer.C:
				now := time.Now()
				err := filepath.Walk(path, func(p string, info fs.FileInfo, err error) error {
					if info == nil {
						return nil
					}
					if info.IsDir() {
						return nil
					}
					if now.Sub(info.ModTime()).Seconds() > ttl.Seconds() {
						log.Println("delete old file:", p)
						if err := os.Remove(p); err != nil {
							return err
						}
					}
					return nil
				})
				if err != nil {
					log.Println(err)
				}
			case <-egCtx.Done():
				return nil
			}
		}
	})

	if err := eg.Wait(); err != nil {
		log.Fatal(err)
	}
}