什么是 supabase#
supabase 是一个替代 Firebase 的开源 BaaS 项目。其提供了各种现代应用所需的稳定、强大、易于使用和可扩展的后端功能。Supabase 基于 PostgreSQL 数据库和 RESTful API 构建,并提供身份验证、实时数据推送、存储和其他一些常见的后端服务。
不需要买服务器,只需掌握简单的 SQL 语句和数据库知识就可以创建一个后端服务。当然它的功能是非常强大的,这篇文章只介绍如何在 Next.js 中使用。
使用 supabase#
学习任务内容都需要详细的阅读官方文档
根据官方文档 Use Supabase with NextJS 这一节的内容,使用步骤分为如下几步
这里我使用 supabase 存储博客的阅读量
创建项目并建表建数据#
通过 https://app.supabase.com/projects 创建一个项目
通过 SQL 编辑器或图形界面 DataBase 创建一个表名为 Views
这里通过图形界面创建,在 Database 中使用右上角的 New table 创建 Views 表
列中的 slug 表示博客的 title,count 表示对应博客的访问量
创建 Next.js 项目#
npx create-next-app@latest --typescript
安装 supabase-js#
npm install @supabase/supabase-js
创建 supabase#
在根目录下创建 lib 文件夹,创建一个名为 supabase.ts 的文件
import { createClient } from '@supabase/supabase-js';
const supabaseUrl = process.env.NEXT_PUBLIC_SUPABASE_URL as string;
const supabaseKey = process.env.NEXT_PUBLIC_SUPABASE_KEY as string;
export const supabase = createClient(supabaseUrl, supabaseKey);
project_url 和 supabase_key 可以在项目的设置选项中找到
同样不建议将 key 暴露出去,可以存在 env.local 中
💡 tips: 在 Next.js 中使用 env.local,需要加上 NEXT_PUBLIC 前缀,否则会报错。
CRUD#
详细的命令可以通过 JavaScript 文档查询
当打开博客文章页面时,需要查询当前博客的阅读量,可以使用
fetch(`/api/views`);
在 pages/api/views 目录下创建 index.ts
import { supabase } from '@/lib/supabase';
import type { NextApiRequest, NextApiResponse } from 'next';
export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
try {
let { data } = await supabase.from('Views').select('*');
return res.status(200).json(data);
} catch (e: any) {
console.log(e);
return res.status(500).json({ message: e.message });
}
}
通过写类似于 sql 语句,可以从 Views 表中获取全部的数据
也可以通过 post 传参将 slug 传进来,查询对应的数据
同时当打开博客文章页面时,也需要对当前博客的阅读量加一,就可以用过 post 请求将 slug 参数传入
fetch(`/api/views/${slug}`, {
method: 'POST'
});
在 pages/api/views 目录下创建 [slug].ts
import { supabase } from '@/lib/supabase';
import type { NextApiRequest, NextApiResponse } from 'next';
export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
try {
const slug = req.query?.slug as string;
if (!slug) {
return res.status(400).json({ message: 'Slug is required.' });
}
const { data } = await supabase
.from('Views')
.select('count')
.eq('slug', slug);
const views = !data?.length ? 0 : Number(data[0].count);
if (req.method === 'POST') {
if (views === 0) {
await supabase.from('Views').insert({ count: views + 1, slug: slug });
return res.status(200).json({
total: views + 1
});
} else {
await supabase
.from('views')
.update({ count: views + 1 })
.eq('slug', slug);
return res.status(200).json({
total: views + 1
});
}
}
if (req.method === 'GET') {
return res.status(200).json({ total: views });
}
} catch (e: any) {
console.log(e);
return res.status(500).json({ message: e.message });
}
}
新建使用 insert,更新使用 update。这样就可以将阅读量存储到 supabase 中
其他功能#
同时 supabase 也支持 bucket 存储功能。我的项目中将 og image 存储在 supabase 中
准备后面的项目中继续使用 supabase,替代 Firebase。