前回は、特定の条件に合致したデータを抽出しました。
data:image/s3,"s3://crabby-images/1b309/1b309678f67572ebf16c5165bbe38427cd17d332" alt="graphql-input"
【GraphQL】特定の条件に合致したデータを抽出する
今回は、データを追加、削除、更新します。
GraphQL では、データを追加したり、削除、更新したりすることをミューテーションと言います。
コードは、前回のコードを使用します。
データを追加する
まずは、本のデータを追加します。
schema.js のdypeDefs内に、ミューテーションの方を作ります。
本を追加するミューテーションを作りたいので、名前をaddBookとします。
js
type Mutation {
addBook
}
addBookにBookを返す様にします。
js
type Mutation {
addBook: Book!
}
addBookの内容をinputで設定します。
設定内容は、bookのデータにします。
js
input AddBookInput {
id: Int!
title: String!
author: String!
categoryId: String!
isRead: Boolean!
}
addBookの中にAddBookInputを指定します。
js
type Mutation {
addBook(input: AddBookInput!): Book!
}
スキーマが完成したので、次はリゾルバーを作成します。
resolvers フォルダの中に、Mutation.js を作成します。
Mutationをエクスポートする様にします。
スキーマで設定した、addBookを追加します。
第二引数にinput、第三引数にbooksを指定します。
js
exports.Mutation = {
addBook: (parent, { input }, { books }) => {},
};
input から、『id』『title』『author』『categoryId』『isRead』を指定します。
また、newBookを作成します。
inputの内容を、newBookに記述します。
js
exports.Mutation = {
addBook: (parent, { input }, { books }) => {
const { id, title, author, categoryId, isRead } = input;
const newBook = {
id,
title,
author,
categoryId,
isRead,
};
},
};
newBookをbooksデータに追加します。
newBooksを返します。
js
exports.Mutation = {
addBook: (parent, { input }, { books }) => {
const { id, title, author, categoryId, isRead } = input;
const newBook = {
id,
title,
author,
categoryId,
isRead,
};
books.push(newBook);
return newBook;
},
};
リゾルバーが完成したので、index.js のサーバー内にMutationを追加します。
js
const server = new ApolloServer({
typeDefs,
resolvers: {
Query,
Mutation,
Category,
Book,
},
context: {
books,
categories,
},
});
今のところ、全ての books データを取得すると、エラーになります。
全てのデータを取得できるように、resolvers の Query.js で、id の指定がある場合のみ、filter を使う様にします。
js
books: (parent, { filter }, { db }) => {
let filteredBooks = db.books;
if (filter) {
if (filter.isRead === true) {
filteredBooks = filteredBooks.filter((book) => {
return book.isRead;
});
}
}
return filteredBooks;
},
data:image/s3,"s3://crabby-images/de5b1/de5b18eb386e3362cfdedde2581a4b725483b032" alt="image2"
books データの一覧が取得できました。
では、データが追加できるか、ブラウザで確認してみます。
GraphQL でデータを抽出するときは、queryを使いました。
今回は、mutationを使います。
graphql
mutation AddBook($input: AddBookInput!) {
addBook(input: $input) {
id
title
author
}
}
data:image/s3,"s3://crabby-images/e320f/e320f649ae8cbfb3504d99a2eda5b9b53a77b3d5" alt="image3"
variables に追加のデータを指定します。
data:image/s3,"s3://crabby-images/19c28/19c28edac7458f352886792420a3db038b324953" alt="image4"
AddBook ボタンをクリックすると、
data:image/s3,"s3://crabby-images/15cea/15ceaf58862f5bdf82cc2715efcddb10e557fa93" alt="image5"
データが追加されました。
books データ一覧を取得でして、追加されているか確認してみましょう。
data:image/s3,"s3://crabby-images/6ef62/6ef628910b068c9a12d0d39b11ae8de41d7f2788" alt="image6"
無事、データが追加されました。
データを削除する
次は、books データを削除します。
ここで作成したデータでは、GraphQL で削除が反映されないので、データをオブジェクト内で設定する様にします。
index.js のbooksとcategoriesをdbでまとめます。
js
const db = { books, categories };
context内もdbへ修正します。
js
const server = new ApolloServer({
typeDefs,
resolvers: {
Query,
Mutation,
Category,
Book,
},
context: {
db,
},
});
contextの部分を、全てdbに置き換えましょう。
resolvers/Query.js
js
exports.Query = {
books: (parent, { filter }, { db }) => {
let filteredBooks = db.books;
if (filter) {
if (filter.isRead === true) {
filteredBooks = filteredBooks.filter((book) => {
return book.isRead;
});
}
}
return filteredBooks;
},
book: (parent, { id }, { db }) => {
const book = db.books.find((book) => book.id === id);
if (!book) return null;
return book;
},
categories: (parent, args, { db }) => db.categories,
category: (parent, { id }, { db }) => {
const category = db.categories.find((category) => category.id === id);
if (!category) return null;
return category;
},
};
resolvers/Category.js
js
exports.Category = {
books: ({ id }, args, { db }) => {
return db.books.filter((book) => book.categoryId === id);
},
};
resolvers/Book.js
js
exports.Book = {
category: ({ categoryId }, args, { db }) => {
return db.categories.find((category) => category.id === categoryId);
},
};
resolvers/Mutation.js
js
exports.Mutation = {
addBook: (parent, { input }, { db }) => {
const { id, title, author, categoryId, isRead } = input;
const newBook = {
id,
title,
author,
categoryId,
isRead,
};
db.books.push(newBook);
return newBook;
},
};
一度、ブラウザで確認します。
data:image/s3,"s3://crabby-images/de5b1/de5b18eb386e3362cfdedde2581a4b725483b032" alt="image7"
data:image/s3,"s3://crabby-images/de5b1/de5b18eb386e3362cfdedde2581a4b725483b032" alt="image8"
うまく機能しています。
schema.js のMutationにdeleteBookを作成します。
指定はidを指定し、成功したかどうかを知りたいので、Booleanを返す様にします。
js
type Mutation {
addBook(input: AddBookInput!): Book!
deleteBook(id: Int!): Boolean!
}
resolvers フォルダの Mutation.js を開きます。
Mutation 内にdeleteBookを作成します。
js
exports.Mutation = {
addBook: (parent, { input }, { db }) => {
const { id, title, author, categoryId, isRead } = input;
const newBook = {
id,
title,
author,
categoryId,
isRead,
};
db.books.push(newBook);
return newBook;
},
deleteBook: (parent, { id }, { db }) => {},
};
特定のデータを抽出するために、filterを設定します。
また、成功したらtrueを返す様にしましょう。
js
deleteBook: (parent, { id }, { db }) => {
db.books = db.books.filter((book) => book.id !== id);
return true;
},
では、ブラウザで確認します。
今回は、id が『2』のデータを削除します。
data:image/s3,"s3://crabby-images/31e70/31e701db36f02c264b858c8cedcd1521ae7cacd0" alt="image9"
『DeleteBook』をクリックすると、
data:image/s3,"s3://crabby-images/1fed5/1fed5a81ba0e84b12873f465e02f992172fd51b7" alt="image10"
trueが返ってきました。
全ての books データを取得すると、
data:image/s3,"s3://crabby-images/0a5dc/0a5dcb6d2b5f1bcbeae8b773582d02ff411bd268" alt="image11"
id が『2』のデータを削除することができました。
データを更新する
最後に、books データを更新します。
schema.js のMutationにupdateBookを作成します。
js
type Mutation {
addBook(input: AddBookInput!): Book!
deleteBook(id: Int!): Boolean!
updateBook
}
変更内容を指定したいので、inputを作成します。
js
input UpdateBookInput {
id: Int
title: String
author: String
categoryId: String
isRead: Boolean
}
updateBook にidとinputを指定します。
また、addBookと同様に、Bookを返す様にします。
js
type Mutation {
addBook(input: AddBookInput!): Book!
deleteBook(id: Int!): Boolean!
updateBook(id: Int!, input: UpdateBookInput!): Book!
}
これでスキーマが完成しました。
次は、理ゾルバーを作成します。
resolvers フォルダの Mutation.js を開きます。
Mutationの中に、updateBookを作成します。
js
exports.Mutation = {
addBook: (parent, { input }, { db }) => {
const { id, title, author, categoryId, isRead } = input;
const newBook = {
id,
title,
author,
categoryId,
isRead,
};
db.books.push(newBook);
return newBook;
},
deleteBook: (parent, { id }, { db }) => {
db.books = db.books.filter((book) => book.id !== id);
return true;
},
updateBook: (parent, { id, input }, { db }) => {},
};
index で何番目のデータを指定するかを設定します。
js
updateBook: (parent, { id, input }, { db }) => {
const index = db.books.findIndex((book) => book.id === id);
},
指定したデータに、変更する内容と、それ以外の内容を指定します。
また、その内容を返します。
js
updateBook: (parent, { id, input }, { db }) => {
const index = db.books.findIndex((book) => book.id === id);
const updateData = (db.books[index] = {
...db.books[index],
...input,
});
return updateData;
},
では、ブラウザで確認します。
今回は、id が『3』の author を変更します。
data:image/s3,"s3://crabby-images/60a69/60a698a4e96604f7d922e7306cae3cc89f493b57" alt="image12"
実行してみると、
data:image/s3,"s3://crabby-images/2d778/2d778345bf674650ee75ff057a96076cef37a286" alt="image13"
変更後の author が返ってきました。
data:image/s3,"s3://crabby-images/fabff/fabff243b78535e6c86a7053b1e6616209f4dca0" alt="image14"
データも反映されています。
これで、データの追加、削除、更新ができました。
全文は、以下のコードです。