无法从Sinatra将图像上传到Cloudinary

时间:2013-12-23 21:12:13

标签: ruby sinatra carrierwave ruby-datamapper cloudinary

我正在尝试从Sinatra网络应用中将图像上传到Cloudinary。一世 通过使用CarrierWave让它在本地保存图像,我知道Cloudinary和 CarrierWave可以很好地互相玩,但我已经尝试了所有东西而且没有骰子。

我已经按照文档进行了操作,但它们非常以Rails为中心。该死的Rails!

# Model - User.rb
class ImageUploader < CarrierWave::Uploader::Base
  include Cloudinary::CarrierWave

  process convert: 'png'
  version :face do
    cloudinary_transformation gravity: :face
  end
end

class User
  include DataMapper::Resource

  attr_accessor :image
  property.............
  mount_uploader :image, ImageUploader
end

# Config - app.rb

require 'carrierwave/datamapper'
require 'cloudinary'

# DataMapper code ...
# Some routes ...

post '/add' do
  user = user.create(name: params[:name])
  Cloudinary::Uploader.upload(params[:image], api_key: 'API_KEY', api_secret: 'API_SECRET', cloud_name: 'CLOUD_NAME')

# View (haml)
%head
  %script{src: "//ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.min.js"}
  %script{src: "js/cloudinary_js/jquery.ui.widget.js"}               
  %script{src: "js/cloudinary_js/jquery.iframe-transport.js"}        
  %script{src: "js/cloudinary_js/jquery.fileupload.js"}              
  %script{src: "js/cloudinary_js/jquery.cloudinary.js"}
  # snip snip ...
%form{action: "/add", method: "POST", enctype: "multipart/form-data"}
  %input{type: "file", name: "image"}
  %button Submit

我在尝试提交表单时遇到的错误是:

TypeError at /add
no implicit conversion of Hash into String

它给我的导致错误的一行是post方法中的Cloudinary上传行。

这是我的参数[:image]哈希

{:filename=>"batman_robin.jpg", :type=>"image/jpeg", :name=>"image", :tempfile=>#<Tempfile:/var/folders/rv/2w8gpstn4hb2lb__27n27ksh0000gn/T/RackMultipart20131223-36484-tgmaba>, :head=>"Content-Disposition: form-data; name=\"image\"; filename=\"batman_robin.jpg\"\r\nContent-Type: image/jpeg\r\n"}

在文档中,它提到了文件签名,然后声明CarrierWave处理这个问题。我认为我的问题在于上传表单。

我在哪里错了?任何人都知道'cl_image_upload_tag'会生成什么,以及我需要添加到文件上传标签的Cloudinary的“特殊属性”是什么?

更新:所以我想到了,我实际上可以直接传递图像:

Cloudinary::Uploader.upload(params[:image][:filename] ...

然而它现在说

No such file or directory - batman_robin.jpg

这很奇怪。我在那里硬编码了一个URL,然后上传到Cloudinary。有任何想法吗?还有关于Cloudinary jQuery上传的任何想法吗?

2 个答案:

答案 0 :(得分:1)

params[:image]是一个散列但Cloudinary::Uploader.upload期望一个字符串作为第一个参数。所有examples in the docs都会提供姓名或网址,因此请从参数传递:filename,例如

post '/add' do
  user = user.create name: params[:name]
  image_meta = params[:image]
  filename = image_meta.delete :filename
  Cloudinary::Uploader.upload filename, {api_key: 'API_KEY', api_secret: 'API_SECRET', cloud_name: 'CLOUD_NAME'}.merge(image_meta)

我还将所有这些默认设置放入settings哈希(如果它们是秘密的话,让它们从环境中读入),例如。

config do
  set :cloudinary_api, {api_key: ENV['API_KEY'],
                        api_secret: ENV['API_SECRET'],
                        cloud_name: ENV['CLOUD_NAME']}
end
然后电话变成......

Cloudinary::Uploader.upload filename, settings.cloudinary_api.merge(image_meta)

答案 1 :(得分:0)

您需要传递Tempfile中保留的实际内容,而不是文件名。因此传递Cloudinary::Uploader.upload(params[:image][:tempfile])应该有效。