123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 |
- package operation
- import (
- "github.com/seaweedfs/seaweedfs/weed/pb"
- "io"
- "mime"
- "net/url"
- "os"
- "path"
- "strconv"
- "strings"
- "google.golang.org/grpc"
- "github.com/seaweedfs/seaweedfs/weed/glog"
- "github.com/seaweedfs/seaweedfs/weed/security"
- )
- type FilePart struct {
- Reader io.Reader
- FileName string
- FileSize int64
- MimeType string
- ModTime int64 //in seconds
- Replication string
- Collection string
- DataCenter string
- Ttl string
- DiskType string
- Server string //this comes from assign result
- Fid string //this comes from assign result, but customizable
- Fsync bool
- }
- type SubmitResult struct {
- FileName string `json:"fileName,omitempty"`
- FileUrl string `json:"url,omitempty"`
- Fid string `json:"fid,omitempty"`
- Size uint32 `json:"size,omitempty"`
- Error string `json:"error,omitempty"`
- }
- type GetMasterFn func() pb.ServerAddress
- func SubmitFiles(masterFn GetMasterFn, grpcDialOption grpc.DialOption, files []FilePart, replication string, collection string, dataCenter string, ttl string, diskType string, maxMB int, usePublicUrl bool) ([]SubmitResult, error) {
- results := make([]SubmitResult, len(files))
- for index, file := range files {
- results[index].FileName = file.FileName
- }
- ar := &VolumeAssignRequest{
- Count: uint64(len(files)),
- Replication: replication,
- Collection: collection,
- DataCenter: dataCenter,
- Ttl: ttl,
- DiskType: diskType,
- }
- ret, err := Assign(masterFn, grpcDialOption, ar)
- if err != nil {
- for index := range files {
- results[index].Error = err.Error()
- }
- return results, err
- }
- for index, file := range files {
- file.Fid = ret.Fid
- if index > 0 {
- file.Fid = file.Fid + "_" + strconv.Itoa(index)
- }
- file.Server = ret.Url
- if usePublicUrl {
- file.Server = ret.PublicUrl
- }
- file.Replication = replication
- file.Collection = collection
- file.DataCenter = dataCenter
- file.Ttl = ttl
- file.DiskType = diskType
- results[index].Size, err = file.Upload(maxMB, masterFn, usePublicUrl, ret.Auth, grpcDialOption)
- if err != nil {
- results[index].Error = err.Error()
- }
- results[index].Fid = file.Fid
- results[index].FileUrl = ret.PublicUrl + "/" + file.Fid
- }
- return results, nil
- }
- func NewFileParts(fullPathFilenames []string) (ret []FilePart, err error) {
- ret = make([]FilePart, len(fullPathFilenames))
- for index, file := range fullPathFilenames {
- if ret[index], err = newFilePart(file); err != nil {
- return
- }
- }
- return
- }
- func newFilePart(fullPathFilename string) (ret FilePart, err error) {
- fh, openErr := os.Open(fullPathFilename)
- if openErr != nil {
- glog.V(0).Info("Failed to open file: ", fullPathFilename)
- return ret, openErr
- }
- ret.Reader = fh
- fi, fiErr := fh.Stat()
- if fiErr != nil {
- glog.V(0).Info("Failed to stat file:", fullPathFilename)
- return ret, fiErr
- }
- ret.ModTime = fi.ModTime().UTC().Unix()
- ret.FileSize = fi.Size()
- ext := strings.ToLower(path.Ext(fullPathFilename))
- ret.FileName = fi.Name()
- if ext != "" {
- ret.MimeType = mime.TypeByExtension(ext)
- }
- return ret, nil
- }
- func (fi FilePart) Upload(maxMB int, masterFn GetMasterFn, usePublicUrl bool, jwt security.EncodedJwt, grpcDialOption grpc.DialOption) (retSize uint32, err error) {
- fileUrl := "http://" + fi.Server + "/" + fi.Fid
- if fi.ModTime != 0 {
- fileUrl += "?ts=" + strconv.Itoa(int(fi.ModTime))
- }
- if fi.Fsync {
- fileUrl += "?fsync=true"
- }
- if closer, ok := fi.Reader.(io.Closer); ok {
- defer closer.Close()
- }
- baseName := path.Base(fi.FileName)
- if maxMB > 0 && fi.FileSize > int64(maxMB*1024*1024) {
- chunkSize := int64(maxMB * 1024 * 1024)
- chunks := fi.FileSize/chunkSize + 1
- cm := ChunkManifest{
- Name: baseName,
- Size: fi.FileSize,
- Mime: fi.MimeType,
- Chunks: make([]*ChunkInfo, 0, chunks),
- }
- var ret *AssignResult
- var id string
- if fi.DataCenter != "" {
- ar := &VolumeAssignRequest{
- Count: uint64(chunks),
- Replication: fi.Replication,
- Collection: fi.Collection,
- Ttl: fi.Ttl,
- DiskType: fi.DiskType,
- }
- ret, err = Assign(masterFn, grpcDialOption, ar)
- if err != nil {
- return
- }
- }
- for i := int64(0); i < chunks; i++ {
- if fi.DataCenter == "" {
- ar := &VolumeAssignRequest{
- Count: 1,
- Replication: fi.Replication,
- Collection: fi.Collection,
- Ttl: fi.Ttl,
- DiskType: fi.DiskType,
- }
- ret, err = Assign(masterFn, grpcDialOption, ar)
- if err != nil {
- // delete all uploaded chunks
- cm.DeleteChunks(masterFn, usePublicUrl, grpcDialOption)
- return
- }
- id = ret.Fid
- } else {
- id = ret.Fid
- if i > 0 {
- id += "_" + strconv.FormatInt(i, 10)
- }
- }
- fileUrl := "http://" + ret.Url + "/" + id
- if usePublicUrl {
- fileUrl = "http://" + ret.PublicUrl + "/" + id
- }
- count, e := upload_one_chunk(
- baseName+"-"+strconv.FormatInt(i+1, 10),
- io.LimitReader(fi.Reader, chunkSize),
- masterFn, fileUrl,
- ret.Auth)
- if e != nil {
- // delete all uploaded chunks
- cm.DeleteChunks(masterFn, usePublicUrl, grpcDialOption)
- return 0, e
- }
- cm.Chunks = append(cm.Chunks,
- &ChunkInfo{
- Offset: i * chunkSize,
- Size: int64(count),
- Fid: id,
- },
- )
- retSize += count
- }
- err = upload_chunked_file_manifest(fileUrl, &cm, jwt)
- if err != nil {
- // delete all uploaded chunks
- cm.DeleteChunks(masterFn, usePublicUrl, grpcDialOption)
- }
- } else {
- uploadOption := &UploadOption{
- UploadUrl: fileUrl,
- Filename: baseName,
- Cipher: false,
- IsInputCompressed: false,
- MimeType: fi.MimeType,
- PairMap: nil,
- Jwt: jwt,
- }
- ret, e, _ := Upload(fi.Reader, uploadOption)
- if e != nil {
- return 0, e
- }
- return ret.Size, e
- }
- return
- }
- func upload_one_chunk(filename string, reader io.Reader, masterFn GetMasterFn,
- fileUrl string, jwt security.EncodedJwt,
- ) (size uint32, e error) {
- glog.V(4).Info("Uploading part ", filename, " to ", fileUrl, "...")
- uploadOption := &UploadOption{
- UploadUrl: fileUrl,
- Filename: filename,
- Cipher: false,
- IsInputCompressed: false,
- MimeType: "",
- PairMap: nil,
- Jwt: jwt,
- }
- uploadResult, uploadError, _ := Upload(reader, uploadOption)
- if uploadError != nil {
- return 0, uploadError
- }
- return uploadResult.Size, nil
- }
- func upload_chunked_file_manifest(fileUrl string, manifest *ChunkManifest, jwt security.EncodedJwt) error {
- buf, e := manifest.Marshal()
- if e != nil {
- return e
- }
- glog.V(4).Info("Uploading chunks manifest ", manifest.Name, " to ", fileUrl, "...")
- u, _ := url.Parse(fileUrl)
- q := u.Query()
- q.Set("cm", "true")
- u.RawQuery = q.Encode()
- uploadOption := &UploadOption{
- UploadUrl: u.String(),
- Filename: manifest.Name,
- Cipher: false,
- IsInputCompressed: false,
- MimeType: "application/json",
- PairMap: nil,
- Jwt: jwt,
- }
- _, e = UploadData(buf, uploadOption)
- return e
- }
|