亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

Echo web 框架綁定 FormFile

Echo web 框架綁定 FormFile

Go
慕田峪7331174 2022-06-06 14:51:27
我正在將 Echo 框架用于接受表單數據的 post 端點。我使用 Struct 作為綁定模型來提取表單數據。我的綁定模型和上傳處理程序代碼如下所示。 type FormModel struct {    ID string                `form:"ID"`    FirstName string                `form:"FirstName"`    File      *multipart.FileHeader `form:"myFileName"`}func (cs *handler) uploadForm(c echo.Context) error {s := new(FormModel)if err := c.Bind(s); err != nil {    return nil}fileHandler, err := c.FormFile("myFileName")我可以通過綁定獲得 ID 和 FirstName 等表單值。但我無法在綁定期間獲取文件。我必須使用fileHandler, err := c.FormFile("myFileName")來獲取文件。有沒有辦法在綁定模型中獲取文件信息?
查看完整描述

1 回答

?
幕布斯7119047

TA貢獻1794條經驗 獲得超8個贊

echo 默認不支持綁定 multipart.Form.File 數據,需要重新綁定實現接口。


為 echo Bind 封裝了一層額外的 bind FormFile。如果結構指針類型的屬性類型為*multipart.FileHeader 或[]*multipart.FileHeader,則該屬性將通過反射設置為FormFile 的值。


我大概實現了這個功能。我沒有使用過echo,也沒有測試過,但想法是正確的。


最后更新:添加示例并修復代碼以構建。感謝@vicTROLLA 指出typeMultipartFileHeader類型定義錯誤。


package main


import (

    "bytes"

    "fmt"

    "github.com/labstack/echo"

    "io/ioutil"

    "mime/multipart"

    "net/http"

    "reflect"

    "strings"

    "time"

)


var (

    typeMultipartFileHeader      = reflect.TypeOf((*multipart.FileHeader)(nil))

    typeMultipartSliceFileHeader = reflect.TypeOf(([]*multipart.FileHeader)(nil))

)


type File struct {

    File  *multipart.FileHeader   `form:"file"`

    Files []*multipart.FileHeader `form:"files"`

}


func main() {

    app := echo.New()

    // warp bind suppet bind FormFile

    app.Binder = NewBindFile(app.Binder)

    app.Any("/file", func(ctx echo.Context) error {

        var file File

        err := ctx.Bind(&file)

        if err != nil {

            return err

        }


        readfile := func(file *multipart.FileHeader) {

            f, err := file.Open()

            if err != nil {

                fmt.Printf("open %s error: %s\n", file.Filename, err.Error())

                return

            }

            body, err := ioutil.ReadAll(f)

            fmt.Printf("read file %s error: %v body: %s\n", file.Filename, err, body)

        }


        readfile(file.File)

        for _, file := range file.Files {

            readfile(file)

        }

        return err

    })


    go func() {

        time.Sleep(200 * time.Millisecond)

        buf := bytes.NewBuffer(nil)

        w := multipart.NewWriter(buf)

        part, _ := w.CreateFormFile("file", "file")

        part.Write([]byte("this one file"))

        part, _ = w.CreateFormFile("files", "files")

        part.Write([]byte("fils part 1"))

        part, _ = w.CreateFormFile("files", "files")

        part.Write([]byte("fils part 2"))

        part, _ = w.CreateFormFile("files", "files")

        part.Write([]byte("fils part 3"))

        part, _ = w.CreateFormFile("files", "files")

        part.Write([]byte("fils part 4"))

        w.Close()

        http.Post("http://localhost:1323/file", w.FormDataContentType(), buf)

    }()


    app.Start(":1323")

}


type BindFunc func(interface{}, echo.Context) error


func (fn BindFunc) Bind(i interface{}, ctx echo.Context) error {

    return fn(i, ctx)

}


func NewBindFile(b echo.Binder) echo.Binder {

    return BindFunc(func(i interface{}, ctx echo.Context) error {

        err := b.Bind(i, ctx)

        if err == nil {

            ctype := ctx.Request().Header.Get(echo.HeaderContentType)

            // if bind form

            if strings.HasPrefix(ctype, echo.MIMEApplicationForm) || strings.HasPrefix(ctype, echo.MIMEMultipartForm) {

                // get form files

                var form *multipart.Form

                form, err = ctx.MultipartForm()

                if err == nil {

                    err = EchoBindFile(i, ctx, form.File)

                }

            }

        }

        return err

    })

}


func EchoBindFile(i interface{}, ctx echo.Context, files map[string][]*multipart.FileHeader) error {

    iValue := reflect.Indirect(reflect.ValueOf(i))

    // check bind type is struct pointer

    if iValue.Kind() != reflect.Struct {

        return fmt.Errorf("BindFile input not is struct pointer, indirect type is %s", iValue.Type().String())

    }


    iType := iValue.Type()

    for i := 0; i < iType.NumField(); i++ {

        fType := iType.Field(i)

        // check canset field

        fValue := iValue.Field(i)

        if !fValue.CanSet() {

            continue

        }

        // revc type must *multipart.FileHeader or []*multipart.FileHeader

        switch fType.Type {

        case typeMultipartFileHeader:

            file := getFiles(files, fType.Name, fType.Tag.Get("form"))

            if len(file) > 0 {

                fValue.Set(reflect.ValueOf(file[0]))

            }

        case typeMultipartSliceFileHeader:

            file := getFiles(files, fType.Name, fType.Tag.Get("form"))

            if len(file) > 0 {

                fValue.Set(reflect.ValueOf(file))

            }

        }

    }

    return nil

}


func getFiles(files map[string][]*multipart.FileHeader, names ...string) []*multipart.FileHeader {

    for _, name := range names {

        file, ok := files[name]

        if ok {

            return file

        }

    }

    return nil

}


查看完整回答
反對 回復 2022-06-06
  • 1 回答
  • 0 關注
  • 249 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號