├── lib
└── compat.go
├── internal
├── handler
│ ├── static
│ │ ├── index.html
│ │ ├── logo.svg
│ │ └── app.js
│ └── handler.go
├── x
│ └── join.go
└── server
│ ├── log.go
│ ├── udp_files.go
│ └── server.go
├── .gitignore
├── .github
├── Dockerfile
├── goreleaser.yml
└── workflows
│ └── ci.yml
├── go.mod
├── main.go
├── LICENSE
├── README.md
└── go.sum
/lib/compat.go:
--------------------------------------------------------------------------------
1 | package uploader
2 |
3 | import (
4 | "net/http"
5 |
6 | "github.com/jpillora/uploader/internal/handler"
7 | )
8 |
9 | type Config = handler.Config
10 |
11 | func New(c Config) http.Handler {
12 | h, err := handler.New(c)
13 | if err != nil {
14 | panic(err)
15 | }
16 | return h
17 | }
18 |
--------------------------------------------------------------------------------
/internal/handler/static/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Uploader
5 |
6 |
7 |
8 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Compiled Object files, Static and Dynamic libs (Shared Objects)
2 | *.o
3 | *.a
4 | *.so
5 |
6 | # Folders
7 | _obj
8 | _test
9 |
10 | # Architecture specific extensions/prefixes
11 | *.[568vq]
12 | [568vq].out
13 |
14 | *.cgo1.go
15 | *.cgo2.c
16 | _cgo_defun.c
17 | _cgo_gotypes.go
18 | _cgo_export.*
19 |
20 | _testmain.go
21 |
22 | *.exe
23 | *.test
24 | *.prof
25 |
--------------------------------------------------------------------------------
/.github/Dockerfile:
--------------------------------------------------------------------------------
1 | # build stage
2 | FROM golang:alpine AS build
3 | RUN apk update && apk add git
4 | ADD . /src
5 | WORKDIR /src
6 | ENV CGO_ENABLED 0
7 | RUN go build \
8 | -trimpath \
9 | -ldflags "-s -w -X main.version=$(git describe --abbrev=0 --tags)" \
10 | -o /tmp/bin
11 | # run stage
12 | FROM scratch
13 | COPY --from=build /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/
14 | WORKDIR /app
15 | COPY --from=build /tmp/bin /app/bin
16 | ENTRYPOINT ["/app/bin"]
--------------------------------------------------------------------------------
/go.mod:
--------------------------------------------------------------------------------
1 | module github.com/jpillora/uploader
2 |
3 | go 1.21
4 |
5 | require (
6 | github.com/NYTimes/gziphandler v1.1.1
7 | github.com/jpillora/opts v1.2.3
8 | github.com/jpillora/requestlog v1.0.0
9 | github.com/jpillora/sizestr v1.0.0
10 | golang.org/x/sync v0.5.0
11 | )
12 |
13 | require (
14 | github.com/andrew-d/go-termutil v0.0.0-20150726205930-009166a695a2 // indirect
15 | github.com/hashicorp/errwrap v1.0.0 // indirect
16 | github.com/hashicorp/go-multierror v1.0.0 // indirect
17 | github.com/jpillora/ansi v1.0.3 // indirect
18 | github.com/jpillora/ipfilter v1.2.9
19 | github.com/phuslu/iploc v1.0.20230201 // indirect
20 | github.com/posener/complete v1.2.2-0.20190308074557-af07aa5181b3 // indirect
21 | github.com/tomasen/realip v0.0.0-20180522021738-f0c99a92ddce // indirect
22 | )
23 |
--------------------------------------------------------------------------------
/internal/x/join.go:
--------------------------------------------------------------------------------
1 | package x
2 |
3 | import (
4 | "fmt"
5 | "os"
6 | "path/filepath"
7 | "strings"
8 | )
9 |
10 | func Join(overwrite bool, parts ...string) string {
11 | if overwrite {
12 | return filepath.Join(parts...)
13 | }
14 | return JoinUnique(parts...)
15 | }
16 |
17 | func JoinUnique(parts ...string) string {
18 | if len(parts) == 0 {
19 | return ""
20 | }
21 | path := filepath.Join(parts...)
22 | count := 1
23 | for {
24 | if _, err := os.Stat(path); os.IsNotExist(err) {
25 | break
26 | }
27 | ext := filepath.Ext(path)
28 | prefix := strings.TrimSuffix(path, ext)
29 | if count > 1 {
30 | prefix = strings.TrimSuffix(prefix, fmt.Sprintf("-%d", count))
31 | }
32 | count++
33 | path = fmt.Sprintf("%s-%d%s", prefix, count, ext)
34 | }
35 | return path
36 | }
37 |
--------------------------------------------------------------------------------
/.github/goreleaser.yml:
--------------------------------------------------------------------------------
1 | # test this file with
2 | # goreleaser --skip-publish --rm-dist --config goreleaser.yml
3 | builds:
4 | - env:
5 | - CGO_ENABLED=0
6 | flags:
7 | - -trimpath
8 | ldflags:
9 | - -s -w -X main.version={{.Version}}
10 | goos:
11 | - linux
12 | - darwin
13 | - windows
14 | - openbsd
15 | goarch:
16 | - '386'
17 | - amd64
18 | - arm
19 | - arm64
20 | goarm:
21 | - '6'
22 | - '7'
23 | nfpms:
24 | - maintainer: "https://github.com/{{ .Env.GITHUB_USER }}"
25 | formats:
26 | - deb
27 | - rpm
28 | - apk
29 | archives:
30 | - format: gz
31 | files:
32 | - none*
33 | rlcp: true
34 | changelog:
35 | sort: asc
36 | filters:
37 | exclude:
38 | - "^docs:"
39 | - "^test:"
40 |
--------------------------------------------------------------------------------
/main.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "log"
5 | "time"
6 |
7 | "github.com/jpillora/opts"
8 | "github.com/jpillora/uploader/internal/server"
9 | )
10 |
11 | var version = "0.0.0"
12 |
13 | func main() {
14 | config := server.Config{
15 | Port: 3000,
16 | UDPClose: 2 * time.Second,
17 | }
18 | opts.New(&config).
19 | Name("uploader").
20 | Summary("note: udp creates files using a stream of packets. udp packets are not authenticated,\n" +
21 | "so it's highly recommended that you set an allowed-ip range to prevent misuse.\n" +
22 | "udp packets are all appended to a file called 'md5(:).bin'.\n" +
23 | "udp streams are considered closed after --udp-close and the file will be closed.").
24 | Repo("github.com/jpillora/uploader").
25 | Version(version).
26 | Parse()
27 | h := server.New(config)
28 | if err := h.Start(); err != nil {
29 | log.Fatal(err)
30 | }
31 | }
32 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2015 Jaime Pillora
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
23 |
--------------------------------------------------------------------------------
/internal/server/log.go:
--------------------------------------------------------------------------------
1 | package server
2 |
3 | import (
4 | "context"
5 | "fmt"
6 | "log/slog"
7 | "strings"
8 | "time"
9 | )
10 |
11 | type shandler struct {
12 | group string
13 | verbose bool
14 | attrs []slog.Attr
15 | }
16 |
17 | func (h *shandler) Enabled(_ context.Context, l slog.Level) bool {
18 | if h.verbose {
19 | return true
20 | }
21 | return l >= slog.LevelInfo
22 | }
23 |
24 | func (h *shandler) WithAttrs(attrs []slog.Attr) slog.Handler {
25 | h2 := *h
26 | h2.attrs = append(h.attrs, attrs...)
27 | return &h2
28 | }
29 |
30 | func (h *shandler) WithGroup(name string) slog.Handler {
31 | h2 := *h
32 | if h2.group == "" {
33 | h2.group = name
34 | } else {
35 | h2.group += "." + name
36 | }
37 | return &h2
38 | }
39 |
40 | func (h *shandler) Handle(ctx context.Context, r slog.Record) error {
41 | sb := strings.Builder{}
42 | sb.WriteString(r.Time.Format(time.RFC3339))
43 |
44 | sb.WriteRune(' ')
45 | sb.WriteString(r.Level.String())
46 |
47 | if h.group != "" {
48 | sb.WriteRune(' ')
49 | sb.WriteString(h.group)
50 | }
51 |
52 | sb.WriteRune(' ')
53 | sb.WriteString(r.Message)
54 |
55 | add := func(attr slog.Attr) bool {
56 | sb.WriteRune(' ')
57 | sb.WriteString(attr.Key)
58 | sb.WriteRune('=')
59 | sb.WriteString(attr.Value.String())
60 | return true
61 | }
62 | r.Attrs(add)
63 | for _, attr := range h.attrs {
64 | add(attr)
65 | }
66 | fmt.Println(sb.String())
67 | return nil
68 | }
69 |
--------------------------------------------------------------------------------
/internal/handler/static/logo.svg:
--------------------------------------------------------------------------------
1 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/internal/handler/handler.go:
--------------------------------------------------------------------------------
1 | package handler
2 |
3 | import (
4 | "embed"
5 | "fmt"
6 | "io"
7 | "log/slog"
8 | "net/http"
9 | "os"
10 |
11 | "github.com/NYTimes/gziphandler"
12 | "github.com/jpillora/sizestr"
13 | "github.com/jpillora/uploader/internal/x"
14 | )
15 |
16 | //go:embed static/*
17 | var content embed.FS
18 |
19 | type Config struct {
20 | Dir string
21 | Overwrite bool
22 | Auth string
23 | Logger *slog.Logger
24 | }
25 |
26 | func New(config Config) (http.Handler, error) {
27 |
28 | log := config.Logger
29 | if log == nil {
30 | log = slog.Default()
31 | }
32 | if config.Dir == "" {
33 | config.Dir = os.TempDir()
34 | }
35 | if info, err := os.Stat(config.Dir); err != nil || !info.IsDir() {
36 | return nil, fmt.Errorf("invalid directory: %s", config.Dir)
37 | }
38 |
39 | var contentHandler = gziphandler.GzipHandler(
40 | http.FileServer(http.FS(content)),
41 | )
42 |
43 | uploadID := 1
44 | return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
45 | defer r.Body.Close()
46 |
47 | l := log.WithGroup(fmt.Sprintf("f%d", uploadID))
48 | if config.Auth != "" {
49 | u, p, _ := r.BasicAuth()
50 | if config.Auth != u+":"+p {
51 | w.WriteHeader(http.StatusUnauthorized)
52 | w.Write([]byte("Access Denied"))
53 | return
54 | }
55 | }
56 |
57 | if r.Method == "GET" {
58 | r.URL.Path = "/static" + r.URL.Path
59 | contentHandler.ServeHTTP(w, r)
60 | return
61 | }
62 |
63 | if r.Method != "POST" {
64 | w.WriteHeader(400)
65 | fmt.Fprint(w, "Expecting POST")
66 | return
67 | }
68 |
69 | multi, err := r.MultipartReader()
70 | if err != nil {
71 | w.WriteHeader(400)
72 | fmt.Fprint(w, "Expecting multipart form")
73 | return
74 | }
75 |
76 | part, err := multi.NextPart()
77 | if err != nil {
78 | w.WriteHeader(400)
79 | fmt.Fprintf(w, "Expecting multipart form (%s)", err)
80 | return
81 | }
82 | defer part.Close()
83 |
84 | if part.FormName() != "file" {
85 | w.WriteHeader(400)
86 | fmt.Fprint(w, "Expecting multipart entry 'file'")
87 | return
88 | }
89 |
90 | filename := part.FileName()
91 | if filename == "" {
92 | filename = "file"
93 | }
94 |
95 | path := x.Join(config.Overwrite, config.Dir, filename)
96 | file, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY, 0600)
97 | if err != nil {
98 | l.Error("open-file", "err", err)
99 | w.WriteHeader(500)
100 | return
101 | }
102 | defer file.Close()
103 |
104 | l.Info("receiving", "path", path)
105 | if n, err := io.Copy(file, part); err != nil {
106 | l.Error("receive-copy", "err", err)
107 | } else {
108 | l.Info("received", "size", sizestr.ToString(n))
109 | }
110 | uploadID++
111 | }), nil
112 | }
113 |
--------------------------------------------------------------------------------
/internal/server/udp_files.go:
--------------------------------------------------------------------------------
1 | package server
2 |
3 | import (
4 | "log/slog"
5 | "os"
6 | "sync"
7 | "time"
8 |
9 | "github.com/jpillora/sizestr"
10 | "github.com/jpillora/uploader/internal/x"
11 | )
12 |
13 | type udpFiles struct {
14 | log *slog.Logger
15 | config Config
16 | mut sync.Mutex
17 | files map[string]*udpFile
18 | sweeping bool
19 | }
20 |
21 | func (cs *udpFiles) upsert(id string) (*udpFile, error) {
22 | cs.mut.Lock()
23 | defer cs.mut.Unlock()
24 | if cs.files == nil {
25 | cs.files = map[string]*udpFile{}
26 | }
27 | f, ok := cs.files[id]
28 | if !ok {
29 | // open file
30 | path := x.Join(cs.config.Overwrite, cs.config.Dir, id+".bin")
31 | file, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY, 0600)
32 | if err != nil {
33 | return nil, err
34 | }
35 | // prepare new file
36 | f = &udpFile{
37 | log: cs.log.WithGroup(id),
38 | id: id,
39 | file: file,
40 | path: path,
41 | }
42 | cs.log.Info("receiving", "path", path)
43 | cs.files[id] = f
44 | }
45 | return f, nil
46 | }
47 |
48 | func (cs *udpFiles) enqueueSweep() {
49 | cs.mut.Lock()
50 | defer cs.mut.Unlock()
51 | if cs.sweeping {
52 | return
53 | }
54 | cs.sweeping = true
55 | time.AfterFunc(cs.config.UDPClose, cs.sweep)
56 | cs.log.Debug("enque-sweep")
57 | }
58 |
59 | func (cs *udpFiles) sweep() {
60 | cs.mut.Lock()
61 | defer cs.mut.Unlock()
62 | closed := 0
63 | for id, c := range cs.files {
64 | if c.timeout() {
65 | c.close()
66 | delete(cs.files, id)
67 | closed++
68 | }
69 | }
70 | cs.log.Debug("swept", "closed", closed, "remaining", len(cs.files))
71 | cs.sweeping = false
72 | }
73 |
74 | func (cs *udpFiles) del(id string) {
75 | cs.mut.Lock()
76 | defer cs.mut.Unlock()
77 | if cs.files == nil {
78 | return
79 | }
80 | c, ok := cs.files[id]
81 | if ok {
82 | c.del()
83 | }
84 | delete(cs.files, id)
85 | cs.log.Info("delete", "id", id)
86 | }
87 |
88 | type udpFile struct {
89 | log *slog.Logger
90 | id string
91 | path string
92 | size int64
93 | mut sync.Mutex
94 | file *os.File
95 | mtime time.Time
96 | }
97 |
98 | func (f *udpFile) write(p []byte) (n int, err error) {
99 | f.mut.Lock()
100 | defer f.mut.Unlock()
101 | n, err = f.file.Write(p)
102 | f.size += int64(n)
103 | return n, err
104 | }
105 |
106 | func (f *udpFile) timeout() bool {
107 | f.mut.Lock()
108 | defer f.mut.Unlock()
109 | return time.Since(f.mtime) > 5*time.Second
110 | }
111 |
112 | func (f *udpFile) close() {
113 | f.mut.Lock()
114 | defer f.mut.Unlock()
115 | if f.file != nil {
116 | f.log.Info("received", "size", sizestr.ToString(f.size))
117 | f.file.Close()
118 | f.file = nil
119 | }
120 | }
121 |
122 | func (f *udpFile) del() {
123 | f.close()
124 | os.Remove(f.path)
125 | f.log.Info("deleted")
126 | }
127 |
--------------------------------------------------------------------------------
/.github/workflows/ci.yml:
--------------------------------------------------------------------------------
1 | name: CI
2 | on:
3 | pull_request: {}
4 | push: {}
5 | permissions: write-all
6 | jobs:
7 | # ================
8 | # BUILD AND TEST JOB
9 | # ================
10 | test:
11 | name: Build & Test
12 | strategy:
13 | matrix:
14 | go-version: ["1.21"]
15 | platform: [ubuntu-latest]
16 | runs-on: ${{ matrix.platform }}
17 | steps:
18 | - name: Checkout
19 | uses: actions/checkout@v3
20 | with:
21 | fetch-depth: 0
22 | - name: Set up Go
23 | uses: actions/setup-go@v3
24 | with:
25 | go-version: ${{ matrix.go-version }}
26 | check-latest: true
27 | - name: Build
28 | run: go build -v -o /dev/null .
29 | - name: Test
30 | run: go test -v ./...
31 | # ================
32 | # RELEASE BINARIES (on push "v*" tag)
33 | # ================
34 | release_binaries:
35 | name: Release Binaries
36 | needs: test
37 | if: startsWith(github.ref, 'refs/tags/v')
38 | runs-on: ubuntu-latest
39 | steps:
40 | - name: Check out code
41 | uses: actions/checkout@v3
42 | - name: goreleaser
43 | uses: docker://goreleaser/goreleaser:latest
44 | env:
45 | GITHUB_USER: ${{ github.repository_owner }}
46 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
47 | with:
48 | args: release --config .github/goreleaser.yml
49 | # ================
50 | # RELEASE DOCKER IMAGES (on push "v*" tag)
51 | # ================
52 | release_docker:
53 | name: Release Docker Images
54 | needs: test
55 | if: startsWith(github.ref, 'refs/tags/v')
56 | runs-on: ubuntu-latest
57 | steps:
58 | - name: Check out code
59 | uses: actions/checkout@v3
60 | - name: Set up QEMU
61 | uses: docker/setup-qemu-action@v2
62 | - name: Set up Docker Buildx
63 | uses: docker/setup-buildx-action@v2
64 | - name: Login to GitHub Container Registry
65 | uses: docker/login-action@v2
66 | with:
67 | registry: ghcr.io
68 | username: ${{ github.actor }}
69 | password: ${{ secrets.GITHUB_TOKEN }}
70 | - name: Docker meta
71 | id: meta
72 | uses: docker/metadata-action@v4
73 | with:
74 | images: ghcr.io/${{ github.repository }}
75 | tags: |
76 | type=semver,pattern={{version}}
77 | type=semver,pattern={{major}}.{{minor}}
78 | type=semver,pattern={{major}}
79 | - name: Build and push
80 | uses: docker/build-push-action@v3
81 | with:
82 | file: .github/Dockerfile
83 | platforms: linux/amd64,linux/arm64,linux/ppc64le,linux/386,linux/arm/v7,linux/arm/v6
84 | push: true
85 | tags: ${{ steps.meta.outputs.tags }}
86 | labels: ${{ steps.meta.outputs.labels }}
87 | cache-from: type=gha
88 | cache-to: type=gha,mode=max
89 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # uploader
2 |
3 | A small server to receive files over HTTP and UDP, with an embedded web UI:
4 |
5 |
6 |
7 | ### Features
8 |
9 | * Simple
10 | * Single binary; Tiny Docker image
11 | * Classic write-only "drop box"; No file reads ever
12 | * Renames files, not overwrite, by default
13 |
14 | ### Install
15 |
16 | **Binaries**
17 |
18 | See [the latest release](https://github.com/jpillora/uploader/releases/latest) or download it with this one-liner: `curl https://i.jpillora.com/uploader! | bash`
19 |
20 | **Docker**
21 |
22 | ``` sh
23 | $ docker run --rm -it -p 3000:3000 -v /tmp:/tmp ghcr.io/jpillora/uploader
24 | ```
25 |
26 | **Source**
27 |
28 | ``` sh
29 | $ go install -v github.com/jpillora/uploader@latest
30 | ```
31 |
32 | ### Usage
33 |
34 | ```sh
35 | # server
36 | $ uploader
37 | 2023/03/26 21:55:58 listening on 3000...
38 | 2023/03/26 21:55:58 saving files to: /tmp
39 |
40 | ...
41 |
42 | # client (cli)
43 | $ curl -F file=@my-file.txt localhost:3000
44 | # client (browser)
45 | # OPEN BROWSER, DRAG AND DROP FILE
46 | ...
47 |
48 | # server
49 | 2015/06/24 01:10:59 #0001 receiving my-file.txt
50 | 2015/06/24 01:10:59 #0001 received 53B
51 | ```
52 |
53 | ### CLI Usage
54 |
55 | ```
56 | Usage: uploader [options]
57 |
58 | note: udp creates files using a stream of packets. udp packets are not authenticated,
59 | so it's highly recommended that you set an allowed-ip range to prevent misuse.
60 | udp packets are all appended to a file called 'md5(:).bin'.
61 | udp streams are considered closed after --udp-close and the file will be closed.
62 |
63 | Options:
64 | --dir, -d output directory (defaults to tmp)
65 | --overwrite, -o duplicates are overwritten (auto-renames files by default)
66 | --auth, -a require basic auth 'username:password' on http connections
67 | --port, -p tcp listening port (default 3000)
68 | --udp-port, -u udp listening port (default disabled)
69 | --udp-close close udp file after timeout (default 2s)
70 | --no-log, -n disable http request logging
71 | --allowed-ip, -i allowed ip range (allows multiple)
72 | --verbose, -v enable verbose logging
73 | --version display version
74 | --help, -h display help
75 |
76 | Version:
77 | 0.0.0
78 |
79 | Read more:
80 | github.com/jpillora/uploader
81 | ```
82 |
83 | #### MIT License
84 |
85 | Copyright © 2023 Jaime Pillora <dev@jpillora.com>
86 |
87 | Permission is hereby granted, free of charge, to any person obtaining
88 | a copy of this software and associated documentation files (the
89 | 'Software'), to deal in the Software without restriction, including
90 | without limitation the rights to use, copy, modify, merge, publish,
91 | distribute, sublicense, and/or sell copies of the Software, and to
92 | permit persons to whom the Software is furnished to do so, subject to
93 | the following conditions:
94 |
95 | The above copyright notice and this permission notice shall be
96 | included in all copies or substantial portions of the Software.
97 |
98 | THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
99 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
100 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
101 | IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
102 | CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
103 | TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
104 | SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
105 |
--------------------------------------------------------------------------------
/go.sum:
--------------------------------------------------------------------------------
1 | github.com/NYTimes/gziphandler v1.1.1 h1:ZUDjpQae29j0ryrS0u/B8HZfJBtBQHjqw2rQ2cqUQ3I=
2 | github.com/NYTimes/gziphandler v1.1.1/go.mod h1:n/CVRwUEOgIxrgPvAQhUUr9oeUtvrhMomdKFjzJNB0c=
3 | github.com/andrew-d/go-termutil v0.0.0-20150726205930-009166a695a2 h1:axBiC50cNZOs7ygH5BgQp4N+aYrZ2DNpWZ1KG3VOSOM=
4 | github.com/andrew-d/go-termutil v0.0.0-20150726205930-009166a695a2/go.mod h1:jnzFpU88PccN/tPPhCpnNU8mZphvKxYM9lLNkd8e+os=
5 | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
6 | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
7 | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
8 | github.com/hashicorp/errwrap v1.0.0 h1:hLrqtEDnRye3+sgx6z4qVLNuviH3MR5aQ0ykNJa/UYA=
9 | github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
10 | github.com/hashicorp/go-multierror v1.0.0 h1:iVjPR7a6H0tWELX5NxNe7bYopibicUzc7uPribsnS6o=
11 | github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk=
12 | github.com/jpillora/ansi v1.0.3 h1:nn4Jzti0EmRfDxm7JtEs5LzCbNwd5sv+0aE+LdS9/ZQ=
13 | github.com/jpillora/ansi v1.0.3/go.mod h1:D2tT+6uzJvN1nBVQILYWkIdq7zG+b5gcFN5WI/VyjMY=
14 | github.com/jpillora/ipfilter v1.2.9 h1:vjjcI1JpxZ6HvIj1MZfomhrfzXW/67QNdE449ZZfon8=
15 | github.com/jpillora/ipfilter v1.2.9/go.mod h1:QUYQLXQU0myCdxZVbYBZ5+An/qtSB2m1OBRiwqTa9pk=
16 | github.com/jpillora/opts v1.2.3 h1:Q0YuOM7y0BlunHJ7laR1TUxkUA7xW8A2rciuZ70xs8g=
17 | github.com/jpillora/opts v1.2.3/go.mod h1:7p7X/vlpKZmtaDFYKs956EujFqA6aCrOkcCaS6UBcR4=
18 | github.com/jpillora/requestlog v1.0.0 h1:bg++eJ74T7DYL3DlIpiwknrtfdUA9oP/M4fL+PpqnyA=
19 | github.com/jpillora/requestlog v1.0.0/go.mod h1:HTWQb7QfDc2jtHnWe2XEIEeJB7gJPnVdpNn52HXPvy8=
20 | github.com/jpillora/sizestr v1.0.0 h1:4tr0FLxs1Mtq3TnsLDV+GYUWG7Q26a6s+tV5Zfw2ygw=
21 | github.com/jpillora/sizestr v1.0.0/go.mod h1:bUhLv4ctkknatr6gR42qPxirmd5+ds1u7mzD+MZ33f0=
22 | github.com/phuslu/iploc v1.0.20230201 h1:AMhy7j8z0N5iI0jaqh514KTDEB7wVdQJ4Y4DJPCvKBU=
23 | github.com/phuslu/iploc v1.0.20230201/go.mod h1:gsgExGWldwv1AEzZm+Ki9/vGfyjkL33pbSr9HGpt2Xg=
24 | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
25 | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
26 | github.com/posener/complete v1.2.2-0.20190308074557-af07aa5181b3 h1:GqpA1/5oN1NgsxoSA4RH0YWTaqvUlQNeOpHXD/JRbOQ=
27 | github.com/posener/complete v1.2.2-0.20190308074557-af07aa5181b3/go.mod h1:6gapUrK/U1TAN7ciCoNRIdVC5sbdBTUh1DKN0g6uH7E=
28 | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
29 | github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
30 | github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
31 | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
32 | github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
33 | github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
34 | github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk=
35 | github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
36 | github.com/tomasen/realip v0.0.0-20180522021738-f0c99a92ddce h1:fb190+cK2Xz/dvi9Hv8eCYJYvIGUTN2/KLq1pT6CjEc=
37 | github.com/tomasen/realip v0.0.0-20180522021738-f0c99a92ddce/go.mod h1:o8v6yHRoik09Xen7gje4m9ERNah1d1PPsVq1VEx9vE4=
38 | golang.org/x/sync v0.5.0 h1:60k92dhOjHxJkrqnwsfl8KuaHbn/5dl0lUPUklKo3qE=
39 | golang.org/x/sync v0.5.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
40 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
41 | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
42 | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
43 | gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
44 |
--------------------------------------------------------------------------------
/internal/server/server.go:
--------------------------------------------------------------------------------
1 | package server
2 |
3 | import (
4 | "crypto/md5"
5 | "encoding/hex"
6 | "fmt"
7 | "log/slog"
8 | "net"
9 | "net/http"
10 | "os"
11 | "time"
12 |
13 | "github.com/jpillora/ipfilter"
14 | "github.com/jpillora/requestlog"
15 | "github.com/jpillora/uploader/internal/handler"
16 | "golang.org/x/sync/errgroup"
17 | )
18 |
19 | type Config struct {
20 | Dir string `help:"output directory (defaults to tmp)"`
21 | Overwrite bool `help:"duplicates are overwritten (auto-renames files by default)"`
22 | Auth string `help:"require basic auth 'username:password' on http connections"`
23 | Port int `help:"tcp listening port"`
24 | UDPPort int `help:"udp listening port (default disabled)"`
25 | UDPClose time.Duration `help:"close udp file after timeout"`
26 | NoLog bool `help:"disable http request logging"`
27 | AllowedIP []string `opts:"short=i, help=allowed ip range"`
28 | Verbose bool `help:"enable verbose logging"`
29 | }
30 |
31 | type Server struct {
32 | Config
33 | filter *ipfilter.IPFilter
34 | log *slog.Logger
35 | }
36 |
37 | func New(c Config) *Server {
38 | if c.Dir == "" {
39 | c.Dir = os.TempDir()
40 | }
41 |
42 | log := slog.New(&shandler{
43 | verbose: c.Verbose,
44 | }).WithGroup("uploader")
45 |
46 | return &Server{
47 | Config: c,
48 | filter: nil,
49 | log: log,
50 | }
51 | }
52 |
53 | func (s *Server) Start() error {
54 | if len(s.AllowedIP) > 0 {
55 | s.log.Info("enable filter", "allowed-ips", s.AllowedIP)
56 | s.filter = ipfilter.New(ipfilter.Options{
57 | AllowedIPs: s.AllowedIP,
58 | BlockByDefault: true,
59 | })
60 | }
61 |
62 | if info, err := os.Stat(s.Dir); err != nil || !info.IsDir() {
63 | return fmt.Errorf("invalid directory: %s", s.Dir)
64 | }
65 | s.log.Info("output directory", "path", s.Dir)
66 |
67 | eg := errgroup.Group{}
68 | if s.UDPPort != 0 {
69 | eg.Go(s.startUDP)
70 | }
71 | eg.Go(s.startHTTP)
72 | return eg.Wait()
73 | }
74 |
75 | func (s *Server) startHTTP() error {
76 | log := s.log.WithGroup("http")
77 | // for compatibility with the old version, we keep the handler config separate
78 | h, err := handler.New(handler.Config{
79 | Dir: s.Dir,
80 | Overwrite: s.Overwrite,
81 | Auth: s.Auth,
82 | Logger: log,
83 | })
84 | if err != nil {
85 | return err
86 | }
87 | if s.filter != nil {
88 | h = s.filter.Wrap(h)
89 | }
90 | if !s.NoLog {
91 | // TODO: add slog support to requestlog
92 | h = requestlog.Wrap(h)
93 | }
94 | addr := fmt.Sprintf("0.0.0.0:%d", s.Port)
95 | l, err := net.Listen("tcp", addr)
96 | if err != nil {
97 | return err
98 | }
99 | log.Info("listening on tcp", "addr", addr)
100 | return http.Serve(l, h)
101 | }
102 |
103 | func (s *Server) startUDP() error {
104 | log := s.log.WithGroup("udp")
105 | addr := fmt.Sprintf("0.0.0.0:%d", s.UDPPort)
106 | uaddr, err := net.ResolveUDPAddr("udp", addr)
107 | if err != nil {
108 | return err
109 | }
110 | conn, err := net.ListenUDP("udp", uaddr)
111 | if err != nil {
112 | return fmt.Errorf("udp: %w", err)
113 | }
114 | log.Info("listening on udp", "addr", addr)
115 | // track udp files
116 | files := &udpFiles{
117 | log: log,
118 | config: s.Config,
119 | }
120 | // read all udp packets
121 | // TODO: writer-pool for each file to allow concurrent writes
122 | buff := make([]byte, 32*1024)
123 | for {
124 | n, addr, err := conn.ReadFromUDP(buff)
125 | if err != nil {
126 | log.Warn("read", "err", err)
127 | continue
128 | }
129 | // ip check
130 | if s.filter != nil && !s.filter.NetAllowed(addr.IP) {
131 | log.Debug("blocked", "addr", addr)
132 | continue
133 | }
134 | // prepare id
135 | hash := md5.Sum([]byte(addr.String()))
136 | id := hex.EncodeToString(hash[:])[0:8]
137 | // prepare file
138 | f, err := files.upsert(id)
139 | if err != nil {
140 | log.Warn("get-err", "id", id, "err", err)
141 | continue
142 | }
143 | // write to file
144 | b := buff[:n]
145 | if _, err := f.write(b); err != nil {
146 | log.Warn("write-err", "id", id, "err", err)
147 | files.del(id)
148 | }
149 | // sweep all files soon
150 | files.enqueueSweep()
151 | }
152 | }
153 |
--------------------------------------------------------------------------------
/internal/handler/static/app.js:
--------------------------------------------------------------------------------
1 | const { createApp } = Vue;
2 |
3 | const gh =
4 | "";
5 |
6 | const app = createApp({
7 | data() {
8 | return {
9 | dragging: false,
10 | queue: [],
11 | handled: [],
12 | };
13 | },
14 | mounted() {
15 | window.APP = this;
16 | },
17 | methods: {
18 | drop(e) {
19 | this.upload(e.dataTransfer.files);
20 | },
21 | choose(e) {
22 | this.upload(e.target.files);
23 | },
24 | upload(files) {
25 | const jobs = Array.from(files).map(file => ({
26 | file,
27 | progress: 0,
28 | status: null,
29 | message: "",
30 | error: false,
31 | }));
32 | this.queue.push(...jobs);
33 | this.dequeue();
34 | },
35 | dequeue() {
36 | const job = this.queue.shift();
37 | if (!job) {
38 | return;
39 | }
40 | const xhr = new XMLHttpRequest();
41 | xhr.open("POST", "/", true);
42 | xhr.upload.addEventListener("progress", e => {
43 | if (e.lengthComputable) {
44 | job.progress = Math.round((e.loaded / e.total) * 100);
45 | }
46 | });
47 | xhr.addEventListener("error", () => {
48 | job.status = -1;
49 | job.error = true;
50 | this.dequeue();
51 | });
52 | xhr.addEventListener("load", () => {
53 | job.status = xhr.status;
54 | job.message = xhr.responseText;
55 | this.dequeue();
56 | });
57 | const fd = new FormData();
58 | fd.append("file", job.file);
59 | xhr.send(fd);
60 | this.handled.push(job);
61 | },
62 | },
63 |
64 | template: /*html*/ `
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |

74 |
75 |
Upload files
76 |
77 |
Drag and drop
78 |
or
79 |
Click to select
80 |
81 |
82 |
90 |
{{ job.file.name }}
96 |
{{ job.message }}
97 |
{{ job.progress }}%
109 |
110 |
111 |
112 |
113 |
114 |
115 |
116 |
134 | `,
135 | });
136 |
137 | app.mount("#app");
138 |
--------------------------------------------------------------------------------