如何在 Supabase 中使用单个 API 调用插入多个表

How to insert in multiple tables with a single API call in Supabase

这是我的 tables 的简化结构 - 2 个主要 tables,一个关系 table。
为此处理插入 API 的最佳方法是什么? 如果我只有一个 Client 和 Supabase:

- First API call to insert book and get ID
- Second API call to insert genre and get ID
- Third API call to insert book-genre relation

这是我能想到的,但是 3 API 调用似乎是错误的。
有没有一种方法可以通过我的客户 的单个 API 调用插入这 3 个 table,就像我可以调用的单个 postgres 函数一样?
请与 API 分享一个通用示例,谢谢!

你有什么理由需要一次调用就可以做到这一点吗?我从你的结构中假设你不会为你创建的每本书创建一个新的 genre,所以大多数时候,你只是插入一个 book 记录和一个 book_gen_rel记录。在现实世界中,您可能会有多种类型的书籍,因此最终您将更改函数以在一次调用中处理插入一本书和多种类型。

也就是说,也有两种方法可以解决这个问题。您可以从客户端发出多个 API 调用(这样做确实没有问题——这很常见)。其次,如果您创建一个 PostgreSQL 函数并使用 .rpc().

调用它,您可以在一次调用中完成所有操作

仅使用客户端调用在每个 table:

中插入记录的示例
    const { data: genre_data, error: genre_error } = await supabase
      .from('genre')
      .insert([
        { name: 'Technology' }
      ]);
    const genre_id = genre_data[0].id;  
    const { data: book_data, error: book_error } = await supabase
      .from('book')
      .insert([
        { name: 'The Joys of PostgreSQL' }
      ]);
    const book_id = book_data[0].id;
    const { data: book_genre_rel_data, error: book_genre_rel_error } = await supabase
      .from('book_genre_rel_data')
      .insert([
        { book_id, genre_id }
      ]);

这是一次插入 3 table 的单个 SQL 语句:

WITH genre AS (
  insert into genre (name) values ('horror') returning id
),
book AS (
  insert into book (name) values ('my scary book') returning id
)
insert into book_genre_rel (genre_id, book_id) 
  select genre.id, book.id from genre, book

现在这是一个 PostgreSQL 函数,可以在一个函数调用中完成所有事情:

CREATE OR REPLACE FUNCTION public.insert_book_and_genre(book_name text, genre_name text)
    RETURNS void language SQL AS
$$
  WITH genre AS (
    insert into genre (name) values (genre_name) returning id
  ),
  book AS (
    insert into book (name) values (book_name) returning id
  )
  insert into book_genre_rel (genre_id, book_id) 
    select genre.id, book.id from genre, book
$$

这里有一个例子来测试它:

select insert_book_and_genre('how to win friends by writing good sql', 'self-help')

现在,如果您已经创建了该函数(在 Supabase 查询编辑器中),那么您可以像这样从客户端调用它:

const { data, error } = await supabase
  .rpc('insert_book_and_genre', {book_name: 'how I became a millionaire at age 3', genre_name: 'lifestyle'})

同样,我不推荐这种方法,至少不推荐 genre 部分。您应该首先插入您的流派(它们可能不会改变)并将其简化为仅插入一条 book 和一条 book_genre_rel 记录。