🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Vue.js

Vue.jsは、Webアプリケーションのインターフェースを構築するためのオープンソースJavaScriptフレームワークです。

Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

POST

POSTはHTTPプロトコルのリクエストメソッドです。ファイルをアップロードしたときや入力フォームが送信されたときなど、クライアントがデータをサーバに送る際に利用されます。

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

Amazon S3

Amazon S3 (Simple Storage Service)とはアマゾン・ウェブ・サービスが提供するオンラインストレージサービスです。

Q&A

解決済

1回答

2427閲覧

Laravelでs3を使って写真を保存できない

tenlife

総合スコア70

Vue.js

Vue.jsは、Webアプリケーションのインターフェースを構築するためのオープンソースJavaScriptフレームワークです。

Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

POST

POSTはHTTPプロトコルのリクエストメソッドです。ファイルをアップロードしたときや入力フォームが送信されたときなど、クライアントがデータをサーバに送る際に利用されます。

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

Amazon S3

Amazon S3 (Simple Storage Service)とはアマゾン・ウェブ・サービスが提供するオンラインストレージサービスです。

0グッド

0クリップ

投稿2019/12/27 03:09

前提・実現したいこと

Laravel + Vue勉強中の初心者です。

https://www.hypertextcandy.com/vue-laravel-tutorial-submit-photo/
このチュートリアルをやっています。

Postすると、500のエラーが帰ってきます。

awsのkeyとid,バケット名は一致しています。

コントローラーでの処理がおかしいのかなと。。。自分では解決出来ませんでした。

発生している問題・エラーメッセージ

POST http://127.0.0.1:8000/api/photos 500 (Internal Server Error) {message: "Can only throw objects", exception: "Symfony\Component\Debug\Exception\FatalThrowableError",…} exception: "Symfony\Component\Debug\Exception\FatalThrowableError" file: "/Users/hoge/Desktop/phps/insta/app/Http/Controllers/PhotoController.php" line: 53 message: "Can only throw objects" trace: [{function: "create", class: "App\Http\Controllers\PhotoController", type: "->"}, {,…}, {,…}, {,…},…]

該当のソースコード

コントローラー <?php namespace App\Http\Controllers; use App\Photo; use App\Http\Requests\StorePhoto; use Illuminate\Http\Request; use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\DB; use Illuminate\Support\Facades\Storage; class PhotoController extends Controller { public function __construct() { $this->middleware('auth'); } /** * 写真投稿 * @param StorePhoto $request * @return \Illuminate\Http\Response */ public function create(StorePhoto $request) { $extension = $request->photo->extension(); $photo = new Photo(); // インスタンス生成時に割り振られたランダムなID値と // 本来の拡張子を組み合わせてファイル名とする $photo->filename = $photo->id . '.' . $extension; // S3にファイルを保存する // 第三引数の'public'はファイルを公開状態で保存するため // putFileAsは指定したファイル位置のファイルのストリーミングを自動的にLaravelに管理できる // cloud() を呼んだ場合は config/filesystems.php の cloud の設定にしたがって使用されるストレージが決まります。 Storage::cloud()->putFileAs('', $request->photo, $photo->filename, 'public'); // データベースエラー時にファイル削除を行うため // トランザクションを利用する DB::beginTransaction(); try{ Auth::user()->photos()->save($photo); DB::commit(); } catch (\Exception $exception) { DB::rollBack(); // DBとの不整合を避けるためアップロードしたファイルを削除 Storage::cloud()->delete($photo->filename); throw $extension; } // リソースの新規作成なので // レスポンスコードは201(CREATED)を返却する return response($photo, 201); } }
フォームリクエスト <?php namespace App\Http\Requests; use Illuminate\Foundation\Http\FormRequest; class StorePhoto extends FormRequest { /** * Determine if the user is authorized to make this request. * * @return bool */ public function authorize() { return true; } /** * Get the validation rules that apply to the request. * * @return array */ public function rules() { return [ 'photo' => 'required|file|mimes:jpg,jpeg,png,gif' ]; } }
Photoモデル <?php namespace App; use Illuminate\Database\Eloquent\Model; class Photo extends Model { /** プライマリキーの型 */ protected $keyType = 'string'; /** IDの桁数 */ const ID_LENGTH = 12; public function __construct(array $attributes = []) { parent::__construct($attributes); if(! array_get($this->attributes, 'id')) { $this->setId(); } } /** * ランダムなID値をid属性に代入する */ private function setId() { $this->attributes['id'] = $this->getRandomId(); } /** * ランダムなID値を生成する * @return string */ private function getRandomId() { $characters = array_merge( range(0, 9), range('a', 'z'), range('A', 'Z'), ['-', '_'] ); $length = count($characters); $id = ""; for($i = 0; $i < self::ID_LENGTH; $i++) { $id = $characters[random_int(0, $length - 1)]; } return $id; } }
<?php namespace App; use Illuminate\Notifications\Notifiable; use Illuminate\Contracts\Auth\MustVerifyEmail; use Illuminate\Foundation\Auth\User as Authenticatable; class User extends Authenticatable { use Notifiable; /** * The attributes that are mass assignable. * * @var array */ protected $fillable = [ 'name', 'email', 'password', ]; /** * The attributes that should be hidden for arrays. * * @var array */ protected $hidden = [ 'password', 'remember_token', ]; /** * The attributes that should be cast to native types. * * @var array */ protected $casts = [ 'email_verified_at' => 'datetime', ]; /** * リレーションシップ - photosテーブル * @return \Illuminate\Database\Eloquent\Relations\HasMany */ public function photos() { return $this->hasMany('App\Photo'); } }
PhotoForm.vue <template> <div v-show="value" class="photo-form"> <h2 class="title">Submit a photo</h2> <div v-show="loading" class="panel"> <Loader>Sending your photo...</Loader> </div> <form v-show="! loading" class="form" @submit.prevent="submit"> <div class="errors" v-if="errors"> <ul v-if="errors.photo"> <li v-for="msg in errors.photo" :key="msg">{{ msg }}</li> </ul> </div> <input class="form__item" type="file" @change="onFileChange"> <output class="form__output" v-if="preview"> <img :src="preview" alt=""> </output> <div class="form__button"> <button type="submit" class="button button--inverse">submit</button> </div> </form> </div> </template> <script> import {CREATED, UNPROCESSABLE_ENTITY } from '../util' import Loader from './Loader.vue' export default { components: { Loader }, // バリューを受け取れるようにpropsを追加 props: { // バリュは表示/非表示の真偽値で表現するためBoolean型を使用する value: { type: Boolean, required: true } }, // beforeSend: function (xhr) { // return xhr.setRequestHeader('X-CSRF-TOKEN', "{{csrf_token()}}"); // }, data() { return { loading: false, preview: null, photo: null, errors: null } }, methods: { onFileChange(event) { // 何も選択されていなければ処理中断 if(event.target.files.length === 0) { this.reset() return false } // ファイルが画像ではなかったら処理中断 if(! event.target.files[0].type.match('image.*')) { this.reset() return false } // FileReaderクラスのインスタンスを取得 const reader = new FileReader() // FileReaderクラスのインスタンスを取得 reader.onload = e => { // previewに読み込み結果(データURL)を代入する // previewに値が入ると<output>につけたv-ifがtrueと判定される // また<output>内部の<img>のsrc属性はpreviewの値を参照しているので // 結果として画像が表示される this.preview = e.target.result } // ファイルを読み込む // 読み込まれたファイルはデータURL形式で受け取れる(上記onload参照) reader.readAsDataURL(event.target.files[0]) this.photo = event.target.files[0] }, reset() { this.preview = '', this.photo = null, // $elはコンポーネントそのものの DOM 要素を指している this.$el.querySelector('input[type="file"]').value = null }, async submit() { // 送信した時にローディングを表示させる this.loading = true const formData = new FormData() formData.append('photo', this.photo) const response = await axios.post('/api/photos', formData) this.loading = false // バリデーションエラーで帰ってきた場合422 if(response.status === UNPROCESSABLE_ENTITY) { this.errors = response.data.errors return false } this.reset() this.$emit('input', false) if(response.status !== CREATED) { this.$store.commit('error/setCode', response.status) return false } this.$store.commit('message/setContent', { content: '写真が投稿されました', timeout: 6000 }) this.$router.push(`/photos/${response.data.id}`) } } } </script>

試したこと

photoモデルにbelongsToをつけてみたりしました。(チュートリアルには無い)変わらず。

eval(\Psy\sh());を使って値が取れているのか確認してみた。

aws関係のkeyやid、バケット名を確認しましたがあっていると思います。

どなたかご教授よろしくおねがします。

補足情報(FW/ツールのバージョンなど)

Laravel 5.8.36

PHP 7.1.23

Vue 2.6.11

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

crhg

2019/12/27 04:40

PhotoControllerの例外処理のところで例外を$exceptionという変数で受けているのにthrows $extension;と変数名を間違えているので、そこでcatchされたのがどんな例外だったのかがわからなくなってしまっています。(かわりに例外でない変数をthrowしようとしているので、それがエラーとして表示されてしまっています) とりあえずそれを直してもう一度やりましょう。その例外の中身が大事な手がかりなのです。
coco_bauer

2019/12/27 08:25

僕も、"throw $extension;"を"throw $exception;"に書き直せば済む話のように思います。単なるタイプミスで、ややこしげに見えるエラーメッセージが出ただけ。
tenlife

2019/12/29 14:26

お二人とも回答ありがとうございます。おっしゃる通り変数の書き間違えでした。 変数の処理の流れを負うべきでした。 ありがとうございました!
guest

回答1

0

自己解決

お二人のおかげ解決することができました、
ありがとうございました

投稿2020/01/06 21:09

tenlife

総合スコア70

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問