mirror of https://github.com/Xhofe/alist
				
				
				
			
		
			
				
	
	
		
			105 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			105 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Go
		
	
	
| package common
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 
 | |
| 	"github.com/alist-org/alist/v3/internal/model"
 | |
| 	"github.com/alist-org/alist/v3/internal/net"
 | |
| 	"github.com/alist-org/alist/v3/internal/stream"
 | |
| 	"github.com/alist-org/alist/v3/pkg/http_range"
 | |
| 	"github.com/alist-org/alist/v3/pkg/utils"
 | |
| 	log "github.com/sirupsen/logrus"
 | |
| )
 | |
| 
 | |
| func Proxy(w http.ResponseWriter, r *http.Request, link *model.Link, file model.Obj) error {
 | |
| 	if link.MFile != nil {
 | |
| 		defer link.MFile.Close()
 | |
| 		attachFileName(w, file)
 | |
| 		contentType := link.Header.Get("Content-Type")
 | |
| 		if contentType != "" {
 | |
| 			w.Header().Set("Content-Type", contentType)
 | |
| 		}
 | |
| 		http.ServeContent(w, r, file.GetName(), file.ModTime(), link.MFile)
 | |
| 		return nil
 | |
| 	} else if link.RangeReadCloser != nil {
 | |
| 		attachFileName(w, file)
 | |
| 		net.ServeHTTP(w, r, file.GetName(), file.ModTime(), file.GetSize(), link.RangeReadCloser.RangeRead)
 | |
| 		defer func() {
 | |
| 			_ = link.RangeReadCloser.Close()
 | |
| 		}()
 | |
| 		return nil
 | |
| 	} else if link.Concurrency != 0 || link.PartSize != 0 {
 | |
| 		attachFileName(w, file)
 | |
| 		size := file.GetSize()
 | |
| 		//var finalClosers model.Closers
 | |
| 		finalClosers := utils.EmptyClosers()
 | |
| 		header := net.ProcessHeader(r.Header, link.Header)
 | |
| 		rangeReader := func(ctx context.Context, httpRange http_range.Range) (io.ReadCloser, error) {
 | |
| 			down := net.NewDownloader(func(d *net.Downloader) {
 | |
| 				d.Concurrency = link.Concurrency
 | |
| 				d.PartSize = link.PartSize
 | |
| 			})
 | |
| 			req := &net.HttpRequestParams{
 | |
| 				URL:       link.URL,
 | |
| 				Range:     httpRange,
 | |
| 				Size:      size,
 | |
| 				HeaderRef: header,
 | |
| 			}
 | |
| 			rc, err := down.Download(ctx, req)
 | |
| 			finalClosers.Add(rc)
 | |
| 			return rc, err
 | |
| 		}
 | |
| 		net.ServeHTTP(w, r, file.GetName(), file.ModTime(), file.GetSize(), rangeReader)
 | |
| 		defer finalClosers.Close()
 | |
| 		return nil
 | |
| 	} else {
 | |
| 		//transparent proxy
 | |
| 		header := net.ProcessHeader(r.Header, link.Header)
 | |
| 		res, err := net.RequestHttp(context.Background(), r.Method, header, link.URL)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		defer res.Body.Close()
 | |
| 
 | |
| 		for h, v := range res.Header {
 | |
| 			w.Header()[h] = v
 | |
| 		}
 | |
| 		w.WriteHeader(res.StatusCode)
 | |
| 		if r.Method == http.MethodHead {
 | |
| 			return nil
 | |
| 		}
 | |
| 		_, err = io.Copy(w, res.Body)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| }
 | |
| func attachFileName(w http.ResponseWriter, file model.Obj) {
 | |
| 	fileName := file.GetName()
 | |
| 	w.Header().Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"; filename*=UTF-8''%s`, fileName, url.PathEscape(fileName)))
 | |
| 	w.Header().Set("Content-Type", utils.GetMimeType(fileName))
 | |
| }
 | |
| 
 | |
| var NoProxyRange = &model.RangeReadCloser{}
 | |
| 
 | |
| func ProxyRange(link *model.Link, size int64) {
 | |
| 	if link.MFile != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	if link.RangeReadCloser == nil {
 | |
| 		var rrc, err = stream.GetRangeReadCloserFromLink(size, link)
 | |
| 		if err != nil {
 | |
| 			log.Warnf("ProxyRange error: %s", err)
 | |
| 			return
 | |
| 		}
 | |
| 		link.RangeReadCloser = rrc
 | |
| 	} else if link.RangeReadCloser == NoProxyRange {
 | |
| 		link.RangeReadCloser = nil
 | |
| 	}
 | |
| }
 |