Golang io.copy twice on the request body

后端 未结 4 656
一向
一向 2021-02-03 11:33

I am building a blob storage system and i picked Go as the programming language. I create a stream to do a multipart file upload from client to the blob server.

The stre

4条回答
  •  清酒与你
    2021-02-03 12:22

    You can't do that directly but you can write a wrapper that does the hashing on io.Copy

    // this works for either a reader or writer, 
    //  but if you use both in the same time the hash will be wrong.
    type Hasher struct {
        io.Writer
        io.Reader
        hash.Hash
        Size uint64
    }
    
    func (h *Hasher) Write(p []byte) (n int, err error) {
        n, err = h.Writer.Write(p)
        h.Hash.Write(p)
        h.Size += uint64(n)
        return
    }
    
    func (h *Hasher) Read(p []byte) (n int, err error) {
        n, err = h.Reader.Read(p)
        h.Hash.Write(p[:n]) //on error n is gonna be 0 so this is still safe.
        return
    }
    
    func (h *Hasher) Sum() string {
        return hex.EncodeToString(h.Hash.Sum(nil))
    }
    
    func (h *UploadHandle) Read() (io.Reader, string, int64, error) {
        var b bytes.Buffer
    
        hashedReader := &Hasher{Reader: h.Contents, Hash: sha1.New()}
        n, err := io.Copy(&b, hashedReader)
    
        if err != nil {
            return nil, "", 0, err
        }
    
        return &b, hashedReader.Sum(), n, nil
    }
    

    // updated version based on @Dustin's comment since I complete forgot io.TeeReader existed.

    func (h *UploadHandle) Read() (io.Reader, string, int64, error) {
        var b bytes.Buffer
    
        hash := sha1.New()
        n, err := io.Copy(&b, io.TeeReader(h.Contents, hash))
    
        if err != nil {
            return nil, "", 0, err
        }
    
        return &b, hex.EncodeToString(hash.Sum(nil)), n, nil
    }
    

提交回复
热议问题