質問をすることでしか得られない、回答やアドバイスがある。

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

新規登録して質問してみよう
ただいま回答率
85.48%
Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby on Rails 6

Ruby on Rails 6は、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Q&A

解決済

2回答

1944閲覧

【Rails】「投稿」に対して中間テーブルを介して「カテゴリー」を付与したい

退会済みユーザー

退会済みユーザー

総合スコア0

Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby on Rails 6

Ruby on Rails 6は、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

0グッド

0クリップ

投稿2022/03/10 08:58

編集2022/03/10 12:19

はじめまして。
RailsとNuxtでメディア系のWebアプリを作成しています。
noteやZennのようなイメージで、登録したユーザーが記事を投稿できるようなサービスです。

投稿機能にセレクトボックスで「カテゴリー」を選択する機能を追加しようとしているのですが、投稿のコントローラーをどのように設定すればいいのか分からず、困っています。

投稿機能のフォームは

  1. タイトルを入力し(input type="text")
  2. 本文も入力、(textarea)
  3. カテゴリーを選択し(select)
  4. [送信]ボタンで投稿

といった入力の流れになります。

テーブルは以下の3つを用意しています。

1. 投稿:
postsテーブル

2. 投稿とカテゴリーの中間テーブル:
post_category_mapsテーブル

3. カテゴリー:
categoriesテーブル

postsテーブル

idtitlebody
1タイトル1本文1
2タイトル2本文2
3タイトル3本文3

post_category_mapsテーブル

idpost_idcategory_id
111
222
331

categoriesテーブル

idname
1カテゴリー1
2カテゴリー2

※このcategoriesテーブルの中身は固定で、ユーザーが新たに登録することはできません。投稿のカテゴリーは記事の投稿時にセレクトボックス・既存のカテゴリー(categoriesテーブルにあらかじめ保存されているデータ)の中から選ぶだけというイメージです。

モデルは以下のものを作成し、アソシエーションを設定しています。

Postモデル

ruby

1class Post < ApplicationRecord 2 3 has_one :post_category_map 4 dependent: :destroy 5 has_one :category, 6 through: :post_category_map 7 8end

PostCategoryMapモデル

ruby

1class PostCategoryMap < ApplicationRecord 2 3 belongs_to :post 4 belongs_to :category 5 6 validates :post_id, 7 presence: true 8 9 validates :category_id, 10 presence: true 11 12end

Categoryモデル

ruby

1class Category < ApplicationRecord 2 3 has_many :post_category_maps, 4 dependent: :destroy, 5 foreign_key: 'category_id' 6 has_many :posts, 7 through: :post_category_maps 8 9end

投稿のコントローラーは以下のようにしています。

postsコントローラー

ruby

1module Api 2 module V1 3 class PostsController < ApplicationController 4 5 def index 6 @post = Post.all 7 render json: posts 8 end 9 10 def new 11 @post = Post.new 12 end 13 14 def create 15 @post = Post.new(post_params) 16 if @post.save 17 render json: @post, status: :created 18 else 19 render json: @post.errors, status: :unprocessable_entity 20 end 21 end 22 23 private 24 25 def post_params 26 params.require(:post).permit( 27 :user_id, 28 :title, 29 :body, 30 :category_id 31 ) 32 end 33 34 end 35 end 36end

実現したいこと

ユーザーが新たな記事を投稿した際、

  1. postsテーブルに「title」と「body」を保存
  2. post_category_mapsテーブルに「post_id」と「category_id」の1セット(関連付け)を保存

したいのですが、現状、投稿しても「title」と「body」は保存されるものの、「post_id」と「category_id」の関連付けは保存されずエラーとなってしまいます。

エラー内容

Completed 500 Internal Server Error ActiveModel::UnknownAttributeError (unknown attribute 'category_id' for Post.):

postsコントローラーファイルをどのように修正すれば「投稿」と「カテゴリー」の関連付けがpost_category_mapsテーブルに保存できるようになるでしょうか?
お教えいただければ幸いです。

追記

newのviewファイルの内容は以下になります。

vue

1<template> 2 <v-form 3 ref="form" 4 v-model="isValid" 5 @submit.prevent="post" 6 > 7 <form-input-post-title 8 :title.sync="params.post.title" 9 /> 10 <form-input-post-body 11 :body.sync="params.post.body" 12 /> 13 <form-input-post-category 14 :category_id.sync="params.post.category_id" 15 /> 16 <v-btn 17 type="submit" 18 :disabled="!isValid || loading" 19 :loading="loading" 20 value="送信" 21 > 22 送信 23 </v-btn> 24 </v-form> 25</template> 26 27<script> 28export default { 29 name: 'PagesNew', 30 layout: 'new', 31 data ({ $store }) { 32 return { 33 isValid: false, 34 loading: false, 35 params: { 36 post: { 37 user_id: this.$store.state.user.current.id, 38 title: '', 39 body: '', 40 category_id: 0 41 } 42 } 43 } 44 }, 45 methods: { 46 async post () { 47 this.loading = true 48 setTimeout(() => { 49 this.loading = false 50 }, 1500) 51 if (this.isValid) { 52 await this.$axios.$post('/api/v1/posts', this.params) 53 } 54 this.loading = false 55 } 56 } 57} 58</script>

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

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

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

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

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

winterboum

2022/03/10 11:33

カテゴリーは複数設定できるのですか? 一つですか?
winterboum

2022/03/10 11:33

new の viewも載せてください
退会済みユーザー

退会済みユーザー

2022/03/10 12:05

カテゴリーは一つだけ設定可能です。 viewファイルも追記いたしました。
guest

回答2

0

カテゴリーが一つだとすると、中間tableを使う必要はありません。使ってもできるけどDB構造もプログラムもちょっと複雑になってしまいます。
has_one :category, through: :post_category_mapbelongs_to :category にして、Post に category_id を追加すれば、おそらくいまのままでカテゴリーがsaveされます。
Category の has_many :posts, through: :post_category_mapshas_many :posts に。

なお、
Railsの風習として、中間tableを使うなら PostCategoryMap ではなく PostCategory が普通です。辞書順に CategoryPost というのが薦められた時代がありましたが、今はどうなのかな。

投稿2022/03/10 12:51

winterboum

総合スコア23329

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

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

退会済みユーザー

退会済みユーザー

2022/03/10 13:17

ご回答いただきありがとうございます。 >> カテゴリーが一つだとすると、中間tableを使う必要はありません。使ってもできるけどDB構造もプログラムもちょっと複雑になってしまいます。 今回のPost以外にも同内容のカテゴリーを当てたい投稿機能が別にありまして...。そのため中間テーブルを使った実装にしたいという思いがあります。 >> has_one :category, through: :post_category_map を belongs_to :category にして、Post に category_id を追加すれば、おそらくいまのままでカテゴリーがsaveされます。 Category の has_many :posts, through: :post_category_mapsもhas_many :posts に。 こちら試してみましたが質問文と同じエラーが出てダメでした...。 >> なお、 Railsの風習として、中間tableを使うなら PostCategoryMap ではなく PostCategory が普通です。辞書順に CategoryPost というのが薦められた時代がありましたが、今はどうなのかな。 そうなんですね!今回の機能の実装ができたら名前の方も修正したいと思います。
退会済みユーザー

退会済みユーザー

2022/03/10 13:28

>> カテゴリーが一つだとすると、中間tableを使う必要はありません。使ってもできるけどDB構造もプログラムもちょっと複雑になってしまいます。 すみません、こちらは単純にpostsテーブルにcategoryカラムを追加すればよいというお話ですね。失礼いたしました。
winterboum

2022/03/10 23:06

>今回のPost以外にも同内容のカテゴリーを当てたい投稿機能が別にありまして...。 こちらも、カテゴリーが一つしか付けられないなら同じく belongs_to :category です。 複数つけるのでしたら 「Rails 関連を同時に登録する」 あたりで検索してみて。 なお、これで出てくるのは 多くは 1:多 のものなので、 Post と Category ではなく、Post と PostCategoryMap として読み替えてください。
退会済みユーザー

退会済みユーザー

2022/03/12 03:59

引き続きご回答いただきありがとうございます。 後に複数のタグ付けの機能も作る予定なので、その予行演習として「カテゴリーは一つしか付けられないがあえて中間テーブルを使った実装をしたい」といった考えもありました。 ただ、winterboumさんのおっしゃるようにDB構造が複雑かつ余分なテーブルが存在することになるのも確かなので、今回は中間テーブルを使わずpostsテーブルにカテゴリー用の外部キーのカラムを追加することで実装を済ませました。 >> 複数つけるのでしたら 「Rails 関連を同時に登録する」 あたりで検索してみて。 なお、これで出てくるのは 多くは 1:多 のものなので、 Post と Category ではなく、Post と PostCategoryMap として読み替えてください。 ご教示いただきありがとうございます。少しそちらのワードで検索してみましたが、いくつか有用そうな情報を見つけることができました。複数タグ付け機能の実装時などに参考にしてみたいと思います。
guest

0

ベストアンサー

当初実現したかったこととは異なりますが、ひとまずwinterboumさんのおっしゃる通りpostsテーブルにcategoryカラムを追加することで解決としました。のちのち複数のタグ付け機能も作成する予定なので、その時にまた中間テーブルを用いたデータ登録の実装にチャレンジしたいと思います。

投稿2022/03/12 04:01

退会済みユーザー

退会済みユーザー

総合スコア0

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問