当我尝试使用File参数和相同有效负载中的数组参数编辑游戏时,我无法弄清楚如何让我的JavaScript以Rails将接受的格式发送请求。
Rails控制器看起来像这样(显然简化):
class GamesController < ApplicationController
def update
@game = Game.find(params[:id])
authorize @game
respond_to do |format|
if @game.update(game_params)
format.html { render html: @game, success: "#{@game.name} was successfully updated." }
format.json { render json: @game, status: :success, location: @game }
else
format.html do
flash.now[:error] = "Unable to update game."
render :edit
end
format.json { render json: @game.errors, status: :unprocessable_entity }
end
end
end
private
def game_params
params.require(:game).permit(
:name,
:cover,
genre_ids: [],
engine_ids: []
)
end
end
所以我有这样的JavaScript:
// this.game.genres and this.game.engines come from
// elsewhere, they're both arrays of objects. These two
// lines turn them into an array of integers representing
// their IDs.
let genre_ids = Array.from(this.game.genres, genre => genre.id);
let engine_ids = Array.from(this.game.engines, engine => engine.id);
let submittableData = new FormData();
submittableData.append('game[name]', this.game.name);
submittableData.append('game[genre_ids]', genre_ids);
submittableData.append('game[engine_ids]', engine_ids);
if (this.game.cover) {
// this.game.cover is a File object
submittableData.append('game[cover]', this.game.cover, this.game.cover.name);
}
fetch("/games/4", {
method: 'PUT',
body: submittableData,
headers: {
'X-CSRF-Token': Rails.csrfToken()
},
credentials: 'same-origin'
}).then(
// success/error handling here
)
当我点击表单中的提交按钮时JavaScript运行,并且应该将数据转换为Rails的后端将接受的格式。不幸的是,我无法让它工作。
在没有要提交的图像文件的情况下,我可以使用JSON.stringify()
而不是FormData
来提交数据,如下所示:
fetch("/games/4", {
method: 'PUT',
body: JSON.stringify({ game: {
name: this.game.name,
genre_ids: genre_ids,
engine_ids: engine_ids
}}),
headers: {
'Content-Type': 'application/json',
'X-CSRF-Token': Rails.csrfToken()
},
credentials: 'same-origin'
})
这很好用。但是我在提交File对象时无法弄清楚如何使用JSON.stringify
。或者,我可以使用FormData
对象,该对象适用于简单的值,例如name
,以及File对象,但不是像数组ID那样的数组值。
只有ID数组(使用JSON.stringify
)的成功表单提交在Rails控制台中如下所示:
Parameters: {"game"=>{"name"=>"Pokémon Ruby", "engine_ids"=>[], "genre_ids"=>[13]}, "id"=>"4"}
但是,我当前的代码最终得到了更像这样的东西:
Parameters: {"game"=>{"name"=>"Pokémon Ruby", "genre_ids"=>"18,2,15", "engine_ids"=>"4,2,10"}, "id"=>"4"}
Unpermitted parameters: :genre_ids, :engine_ids
或者,如果您还在此过程中上传文件:
Parameters: {"game"=>{"name"=>"Pokémon Ruby", "genre_ids"=>"13,3", "engine_ids"=>"5", "cover"=>#<ActionDispatch::Http::UploadedFile:0x00007f9a45d11f78 @tempfile=#<Tempfile:/var/folders/2n/6l8d3x457wq9m5fpry0dltb40000gn/T/RackMultipart20190217-31684-1qmtpx2.png>, @original_filename="Screen Shot 2019-01-27 at 5.26.23 PM.png", @content_type="image/png", @headers="Content-Disposition: form-data; name=\"game[cover]\"; filename=\"Screen Shot 2019-01-27 at 5.26.23 PM.png\"\r\nContent-Type: image/png\r\n">}, "id"=>"4"}
Unpermitted parameters: :genre_ids, :engine_ids
TL; DR:我的问题是,如何使用JavaScript将此有效负载(名称字符串,ID数组以及游戏封面图像)发送给Rails?实际接受什么格式,如何实现?
Rails应用程序是开源的,如果它有帮助,you can see the repo here。提到的具体文件是app/controllers/games_controller.rb
和app/javascript/src/components/game-form.vue
,尽管我已经对这个问题进行了大量简化。
我发现我可以使用ActiveStorage's Direct Upload feature来做到这一点。
在我的JavaScript中:
// Import DirectUpload from ActiveStorage somewhere above here.
onChange(file) {
this.uploadFile(file);
},
uploadFile(file) {
const url = "/rails/active_storage/direct_uploads";
const upload = new DirectUpload(file, url);
upload.create((error, blob) => {
if (error) {
// TODO: Handle this error.
console.log(error);
} else {
this.game.coverBlob = blob.signed_id;
}
})
},
onSubmit() {
let genre_ids = Array.from(this.game.genres, genre => genre.id);
let engine_ids = Array.from(this.game.engines, engine => engine.id);
let submittableData = { game: {
name: this.game.name,
genre_ids: genre_ids,
engine_ids: engine_ids
}};
if (this.game.coverBlob) {
submittableData['game']['cover'] = this.game.coverBlob;
}
fetch(this.submitPath, {
method: this.create ? 'POST' : 'PUT',
body: JSON.stringify(submittableData),
headers: {
'Content-Type': 'application/json',
'X-CSRF-Token': Rails.csrfToken()
},
credentials: 'same-origin'
})
}
然后我想通过DirectUpload的工作方式,我可以将coverBlob
变量发送到Rails应用程序,所以它只是一个字符串。超级容易。
您可以将File
对象转换为data URL
并在JSON
中包含该字符串,请参阅processFiles
中的Upload multiple image using AJAX, PHP and jQuery函数或在JavaScript JSON.stringify()
上使用Array
并将其设置为FormData
对象的值,而不是将Array
作为值传递给FormData
。
submittableData.append('game[name]', JSON.stringify(this.game.name));
submittableData.append('game[genre_ids]', JSON.stringify(genre_ids));
submittableData.append('game[engine_ids]', JSON.stringify(engine_ids));