使用Alamofire将文件上传到Amazon S3

4
我正在使用Alamofire 4.0创建一个请求,直接将文件上传到S3亚马逊。
我正在使用GetCloudApp的服务。在我的请求Router.shared.prepareForUploadItem从api"https://my.cl.ly/v3/items之后,我像这样检索json
{
  "slug": "1h132K0z2n3G",
  "name": "Image.png",
  "url": "http://f.cl.ly",
  "uploads_remaining": 1,
  "max_upload_size": 26214400,
  "s3": {
    "AWSAccessKeyId": "AKIAJP2C6U543KJIE2GA",
    "key": "items/353u2B053p0H0D1O3w1b/${filename}",
    "policy": "eyJleHBpcmF0aW9uIjoiMjAxNS0xMS0xMlQxMjo0MTozOFoiLCJjb25kaXRpb25zIjpbeyJidWNrZXQiOiJjbG91ZGFwcC5jb3BwZXIuaW8ifSxbInN0YXJ0cy13aXRoIiwiJGtleSIsIml0ZW1zLzM1M3UyQjA1M3AwSDBEMU8zdzFiLyJdLHsic3VjY2Vzc19hY3Rpb25fcmVkaXJlY3QiOiJodHRwOi8vbXkuY2wuZGV2L3YzL2l0ZW1zLzFoMTMySzB6Mm4zRy9zMyJ9LHsiYWNsIjoicHVibGljLXJlYWQifSxbImNvbnRlbnQtbGVuZ3RoLXJhbmdlIiwwLDI2MjE0NDAwXV19",
    "signature": "wqSVl9+fvkvtIzGfakNF+drqN0s=",
    "success_action_redirect": "http://api.cl.ly/v3/items/1h132K0z2n3G/s3",
    "acl": "public-read"
  }
}

我将这些keyvalue作为我的参数:

"AWSAccessKeyId": "AKIAJP2C6U543KJIE2GA",
"key": "items/353u2B053p0H0D1O3w1b/${filename}",
"policy": "eyJleHBpcmF0aW9uIjoiMjAxNS0xMS0xMlQxMjo0MTozOFoiLCJjb25kaXRpb25zIjpbeyJidWNrZXQiOiJjbG91ZGFwcC5jb3BwZXIuaW8ifSxbInN0YXJ0cy13aXRoIiwiJGtleSIsIml0ZW1zLzM1M3UyQjA1M3AwSDBEMU8zdzFiLyJdLHsic3VjY2Vzc19hY3Rpb25fcmVkaXJlY3QiOiJodHRwOi8vbXkuY2wuZGV2L3YzL2l0ZW1zLzFoMTMySzB6Mm4zRy9zMyJ9LHsiYWNsIjoicHVibGljLXJlYWQifSxbImNvbnRlbnQtbGVuZ3RoLXJhbmdlIiwwLDI2MjE0NDAwXV19",
"signature": "wqSVl9+fvkvtIzGfakNF+drqN0s=",
"success_action_redirect": "http://api.cl.ly/v3/items/1h132K0z2n3G/s3",
"acl": "public-read"

然后我使用以下代码创建上传文件的请求,使用 s3 字典:

 Alamofire.upload(multipartFormData: { (multipartForm) in

  for (key, value) in parameter {

    let valueData = value.data(using: .utf8, allowLossyConversion: false)

    guard let newData = valueData else{
      return
    }

    multipartForm.append(newData, withName: key)

    print("\(key) - \(value)")
  }

  multipartForm.append(data, withName: "file")

}, to: s3.url, method : .post) { (encodingResult) in

  switch encodingResult {
  case .success(let upload, _, _):
    upload.responseJSON { response in
      debugPrint(response)
    }
  case .failure(let encodingError):
    print(encodingError)
  }
}

最后,我像这样从服务器检索响应,我知道这全部都与身份验证有关,但我真的不知道正确的做法。
status code: 401, headers {
"Cache-Control" = "no-cache";
Connection = "keep-alive";
"Content-Length" = 28;
"Content-Type" = "text/plain; charset=utf-8";
Date = "Sun, 08 Jan 2017 17:37:17 GMT";
Server = "nginx/1.4.6 (Ubuntu)";
Status = "401 Unauthorized";
"Www-Authenticate" = "Digest realm=\"Application\", qop=\"auth\", algorithm=MD5, nonce=\"MTQ4Mzg5NzAzNzpiYmEwN2MzZDljM2RjNDMyMGE1NzI2ODQ5MjhjMWVkNQ==\", opaque=\"9eb56ccb2e8b017ae42bdb4739690863\"";
"X-Request-Id" = 7bd77581bf677aee1b1abba21b3ad097;
"X-Runtime" = "0.004697";
"X-UA-Compatible" = "IE=Edge,chrome=1";

我尝试了很多方法,但仍然没有运气,无法使其工作。我真的需要帮助。提前感谢。


你是自己创建签名还是依赖于Alamofire? - donkon
嗨,CloudApp服务支持签名。我找出了我的问题。请查看下面的答案。谢谢你的提问 :) - sunsunai
1个回答

2

我找出了这个问题的原因,因为在向CloudApp上传新文件之前,需要使用CloudApp的摘要认证对我的上传请求进行身份验证。

下面是我完成的代码:

func uploadFileToS3(_ s3: S3, data : Data, fileName : String, mimeType : String) {

let params = s3.s3Parameter

Alamofire.upload(multipartFormData: { (multipartForm) in

  for (key, value) in params{

    multipartForm.append(value.data(using: .utf8)!, withName: key)

  }

  multipartForm.append(data, withName: "file", fileName: fileName, mimeType: mimeType)

}, to: s3.url, method : .post, headers:["Accept":"application/json"]) { (encodingResult) in

  guard let userInfor = UserDefaults.standard.dictionary(forKey: "UserInformation") else {
    return
  }

  switch encodingResult {
  case .success(let upload, _, _):
    upload.authenticate(user: userInfor["email"]! as! String, password: userInfor["password"]! as! String)

    self.progressbar.isHidden = false
    upload.uploadProgress(closure: { (progress) in

      self.progressbar.progress = Float(progress.fractionCompleted)

    })

    upload.responseJSON{ response in

      self.progressbar.isHidden = true

      let result = JSON(response.result.value!)
      let file = CloudFile(fileInformation: result)

      self.files.append(file)

      let indexPath = IndexPath(row: 0, section: 0)

      DispatchQueue.main.async {
          self.filesTableView.insertRows(at: [indexPath], with: .automatic)
      }

    }
  case .failure(let encodingError):
    print(encodingError)
  }
}

注意到encodingResult结果完成处理程序中的upload.authenticate(user: userInfor["email"]! as! String, password: userInfor["password"]! as! String),这是您需要验证上传请求的地方。 Alamofire将处理其余部分。

希望这可以帮助您。谢谢


@sunsunai,您有关于认证部分的任何指令吗? - Victor 'Chris' Cabral
@Victor'Chris'Cabral 你好,很抱歉我没有经常查看我的stackoverflow评论哈哈,顺便感谢你的提问。实际上,我不理解你问题中的指令是什么意思?你是指关于认证部分的知识吗?如果是的话,我必须说我是自学成才,没有任何基础知识。 - sunsunai
谢谢!如果有人需要的话,可以按照这个答案更新Alamofire的语法 https://dev59.com/pWkMtIcB2Jgan1znTTSt#69273951此外,您可以像这样设置Content-Type:multipartForm.append(Data("image/jpeg".utf8), withName: "Content-Type") - Liviu

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接