123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409 |
- package oss
- import (
- "bytes"
- "errors"
- "fmt"
- "hash/crc32"
- "hash/crc64"
- "net/http"
- "os"
- "os/exec"
- "runtime"
- "strconv"
- "strings"
- "time"
- )
- // userAgent gets user agent
- // It has the SDK version information, OS information and GO version
- func userAgent() string {
- sys := getSysInfo()
- return fmt.Sprintf("aliyun-sdk-go/%s (%s/%s/%s;%s)", Version, sys.name,
- sys.release, sys.machine, runtime.Version())
- }
- type sysInfo struct {
- name string // OS name such as windows/Linux
- release string // OS version 2.6.32-220.23.2.ali1089.el5.x86_64 etc
- machine string // CPU type amd64/x86_64
- }
- // getSysInfo gets system info
- // gets the OS information and CPU type
- func getSysInfo() sysInfo {
- name := runtime.GOOS
- release := "-"
- machine := runtime.GOARCH
- if out, err := exec.Command("uname", "-s").CombinedOutput(); err == nil {
- name = string(bytes.TrimSpace(out))
- }
- if out, err := exec.Command("uname", "-r").CombinedOutput(); err == nil {
- release = string(bytes.TrimSpace(out))
- }
- if out, err := exec.Command("uname", "-m").CombinedOutput(); err == nil {
- machine = string(bytes.TrimSpace(out))
- }
- return sysInfo{name: name, release: release, machine: machine}
- }
- // unpackedRange
- type unpackedRange struct {
- hasStart bool // Flag indicates if the start point is specified
- hasEnd bool // Flag indicates if the end point is specified
- start int64 // Start point
- end int64 // End point
- }
- // invalidRangeError returns invalid range error
- func invalidRangeError(r string) error {
- return fmt.Errorf("InvalidRange %s", r)
- }
- // parseRange parse various styles of range such as bytes=M-N
- func parseRange(normalizedRange string) (*unpackedRange, error) {
- var err error
- hasStart := false
- hasEnd := false
- var start int64
- var end int64
- // Bytes==M-N or ranges=M-N
- nrSlice := strings.Split(normalizedRange, "=")
- if len(nrSlice) != 2 || nrSlice[0] != "bytes" {
- return nil, invalidRangeError(normalizedRange)
- }
- // Bytes=M-N,X-Y
- rSlice := strings.Split(nrSlice[1], ",")
- rStr := rSlice[0]
- if strings.HasSuffix(rStr, "-") { // M-
- startStr := rStr[:len(rStr)-1]
- start, err = strconv.ParseInt(startStr, 10, 64)
- if err != nil {
- return nil, invalidRangeError(normalizedRange)
- }
- hasStart = true
- } else if strings.HasPrefix(rStr, "-") { // -N
- len := rStr[1:]
- end, err = strconv.ParseInt(len, 10, 64)
- if err != nil {
- return nil, invalidRangeError(normalizedRange)
- }
- if end == 0 { // -0
- return nil, invalidRangeError(normalizedRange)
- }
- hasEnd = true
- } else { // M-N
- valSlice := strings.Split(rStr, "-")
- if len(valSlice) != 2 {
- return nil, invalidRangeError(normalizedRange)
- }
- start, err = strconv.ParseInt(valSlice[0], 10, 64)
- if err != nil {
- return nil, invalidRangeError(normalizedRange)
- }
- hasStart = true
- end, err = strconv.ParseInt(valSlice[1], 10, 64)
- if err != nil {
- return nil, invalidRangeError(normalizedRange)
- }
- hasEnd = true
- }
- return &unpackedRange{hasStart, hasEnd, start, end}, nil
- }
- // adjustRange returns adjusted range, adjust the range according to the length of the file
- func adjustRange(ur *unpackedRange, size int64) (start, end int64) {
- if ur == nil {
- return 0, size
- }
- if ur.hasStart && ur.hasEnd {
- start = ur.start
- end = ur.end + 1
- if ur.start < 0 || ur.start >= size || ur.end > size || ur.start > ur.end {
- start = 0
- end = size
- }
- } else if ur.hasStart {
- start = ur.start
- end = size
- if ur.start < 0 || ur.start >= size {
- start = 0
- }
- } else if ur.hasEnd {
- start = size - ur.end
- end = size
- if ur.end < 0 || ur.end > size {
- start = 0
- end = size
- }
- }
- return
- }
- // GetNowSec returns Unix time, the number of seconds elapsed since January 1, 1970 UTC.
- // gets the current time in Unix time, in seconds.
- func GetNowSec() int64 {
- return time.Now().Unix()
- }
- // GetNowNanoSec returns t as a Unix time, the number of nanoseconds elapsed
- // since January 1, 1970 UTC. The result is undefined if the Unix time
- // in nanoseconds cannot be represented by an int64. Note that this
- // means the result of calling UnixNano on the zero Time is undefined.
- // gets the current time in Unix time, in nanoseconds.
- func GetNowNanoSec() int64 {
- return time.Now().UnixNano()
- }
- // GetNowGMT gets the current time in GMT format.
- func GetNowGMT() string {
- return time.Now().UTC().Format(http.TimeFormat)
- }
- // FileChunk is the file chunk definition
- type FileChunk struct {
- Number int // Chunk number
- Offset int64 // Chunk offset
- Size int64 // Chunk size.
- }
- // SplitFileByPartNum splits big file into parts by the num of parts.
- // Split the file with specified parts count, returns the split result when error is nil.
- func SplitFileByPartNum(fileName string, chunkNum int) ([]FileChunk, error) {
- if chunkNum <= 0 || chunkNum > 10000 {
- return nil, errors.New("chunkNum invalid")
- }
- file, err := os.Open(fileName)
- if err != nil {
- return nil, err
- }
- defer file.Close()
- stat, err := file.Stat()
- if err != nil {
- return nil, err
- }
- if int64(chunkNum) > stat.Size() {
- return nil, errors.New("oss: chunkNum invalid")
- }
- var chunks []FileChunk
- var chunk = FileChunk{}
- var chunkN = (int64)(chunkNum)
- for i := int64(0); i < chunkN; i++ {
- chunk.Number = int(i + 1)
- chunk.Offset = i * (stat.Size() / chunkN)
- if i == chunkN-1 {
- chunk.Size = stat.Size()/chunkN + stat.Size()%chunkN
- } else {
- chunk.Size = stat.Size() / chunkN
- }
- chunks = append(chunks, chunk)
- }
- return chunks, nil
- }
- // SplitFileByPartSize splits big file into parts by the size of parts.
- // Splits the file by the part size. Returns the FileChunk when error is nil.
- func SplitFileByPartSize(fileName string, chunkSize int64) ([]FileChunk, error) {
- if chunkSize <= 0 {
- return nil, errors.New("chunkSize invalid")
- }
- file, err := os.Open(fileName)
- if err != nil {
- return nil, err
- }
- defer file.Close()
- stat, err := file.Stat()
- if err != nil {
- return nil, err
- }
- var chunkN = stat.Size() / chunkSize
- if chunkN >= 10000 {
- return nil, errors.New("Too many parts, please increase part size")
- }
- var chunks []FileChunk
- var chunk = FileChunk{}
- for i := int64(0); i < chunkN; i++ {
- chunk.Number = int(i + 1)
- chunk.Offset = i * chunkSize
- chunk.Size = chunkSize
- chunks = append(chunks, chunk)
- }
- if stat.Size()%chunkSize > 0 {
- chunk.Number = len(chunks) + 1
- chunk.Offset = int64(len(chunks)) * chunkSize
- chunk.Size = stat.Size() % chunkSize
- chunks = append(chunks, chunk)
- }
- return chunks, nil
- }
- // GetPartEnd calculates the end position
- func GetPartEnd(begin int64, total int64, per int64) int64 {
- if begin+per > total {
- return total - 1
- }
- return begin + per - 1
- }
- // crcTable returns the table constructed from the specified polynomial
- var crcTable = func() *crc64.Table {
- return crc64.MakeTable(crc64.ECMA)
- }
- // crcTable returns the table constructed from the specified polynomial
- var crc32Table = func() *crc32.Table {
- return crc32.MakeTable(crc32.IEEE)
- }
- // choiceTransferPartOption choices valid option supported by Uploadpart or DownloadPart
- func ChoiceTransferPartOption(options []Option) []Option {
- var outOption []Option
- listener, _ := findOption(options, progressListener, nil)
- if listener != nil {
- outOption = append(outOption, Progress(listener.(ProgressListener)))
- }
- payer, _ := findOption(options, HTTPHeaderOssRequester, nil)
- if payer != nil {
- outOption = append(outOption, RequestPayer(PayerType(payer.(string))))
- }
- versionId, _ := findOption(options, "versionId", nil)
- if versionId != nil {
- outOption = append(outOption, VersionId(versionId.(string)))
- }
- trafficLimit, _ := findOption(options, HTTPHeaderOssTrafficLimit, nil)
- if trafficLimit != nil {
- speed, _ := strconv.ParseInt(trafficLimit.(string), 10, 64)
- outOption = append(outOption, TrafficLimitHeader(speed))
- }
- respHeader, _ := findOption(options, responseHeader, nil)
- if respHeader != nil {
- outOption = append(outOption, GetResponseHeader(respHeader.(*http.Header)))
- }
- return outOption
- }
- // ChoiceCompletePartOption choices valid option supported by CompleteMulitiPart
- func ChoiceCompletePartOption(options []Option) []Option {
- var outOption []Option
- listener, _ := findOption(options, progressListener, nil)
- if listener != nil {
- outOption = append(outOption, Progress(listener.(ProgressListener)))
- }
- payer, _ := findOption(options, HTTPHeaderOssRequester, nil)
- if payer != nil {
- outOption = append(outOption, RequestPayer(PayerType(payer.(string))))
- }
- acl, _ := findOption(options, HTTPHeaderOssObjectACL, nil)
- if acl != nil {
- outOption = append(outOption, ObjectACL(ACLType(acl.(string))))
- }
- callback, _ := findOption(options, HTTPHeaderOssCallback, nil)
- if callback != nil {
- outOption = append(outOption, Callback(callback.(string)))
- }
- callbackVar, _ := findOption(options, HTTPHeaderOssCallbackVar, nil)
- if callbackVar != nil {
- outOption = append(outOption, CallbackVar(callbackVar.(string)))
- }
- respHeader, _ := findOption(options, responseHeader, nil)
- if respHeader != nil {
- outOption = append(outOption, GetResponseHeader(respHeader.(*http.Header)))
- }
- forbidOverWrite, _ := findOption(options, HTTPHeaderOssForbidOverWrite, nil)
- if forbidOverWrite != nil {
- if forbidOverWrite.(string) == "true" {
- outOption = append(outOption, ForbidOverWrite(true))
- } else {
- outOption = append(outOption, ForbidOverWrite(false))
- }
- }
- return outOption
- }
- // ChoiceAbortPartOption choices valid option supported by AbortMultipartUpload
- func ChoiceAbortPartOption(options []Option) []Option {
- var outOption []Option
- payer, _ := findOption(options, HTTPHeaderOssRequester, nil)
- if payer != nil {
- outOption = append(outOption, RequestPayer(PayerType(payer.(string))))
- }
- respHeader, _ := findOption(options, responseHeader, nil)
- if respHeader != nil {
- outOption = append(outOption, GetResponseHeader(respHeader.(*http.Header)))
- }
- return outOption
- }
- // ChoiceHeadObjectOption choices valid option supported by HeadObject
- func ChoiceHeadObjectOption(options []Option) []Option {
- var outOption []Option
- // not select HTTPHeaderRange to get whole object length
- payer, _ := findOption(options, HTTPHeaderOssRequester, nil)
- if payer != nil {
- outOption = append(outOption, RequestPayer(PayerType(payer.(string))))
- }
- versionId, _ := findOption(options, "versionId", nil)
- if versionId != nil {
- outOption = append(outOption, VersionId(versionId.(string)))
- }
- respHeader, _ := findOption(options, responseHeader, nil)
- if respHeader != nil {
- outOption = append(outOption, GetResponseHeader(respHeader.(*http.Header)))
- }
- return outOption
- }
- func CheckBucketName(bucketName string) error {
- nameLen := len(bucketName)
- if nameLen < 3 || nameLen > 63 {
- return fmt.Errorf("bucket name %s len is between [3-63],now is %d", bucketName, nameLen)
- }
- for _, v := range bucketName {
- if !(('a' <= v && v <= 'z') || ('0' <= v && v <= '9') || v == '-') {
- return fmt.Errorf("bucket name %s can only include lowercase letters, numbers, and -", bucketName)
- }
- }
- if bucketName[0] == '-' || bucketName[nameLen-1] == '-' {
- return fmt.Errorf("bucket name %s must start and end with a lowercase letter or number", bucketName)
- }
- return nil
- }
|