【Hexo】鱼塘朋友圈部署全攻略

一、介绍

image-20241022151735279

初始项目:hexo-circle-of-friends

改进版:Friend-Circle-Lite

  • 轻量化:对比原版友链朋友圈的功能,该友圈功能简洁,去掉了设置和fastAPI的臃肿,仅保留关键内容。
  • 无数据库:因为内容较少,我采用json直接存储文章信息,减少数据库操作,提升action运行效率。
  • 部署简单:原版友链朋友圈由于功能多,导致部署较为麻烦,本方案仅需简单的部署action即可使用,vercel
  • 用于部署前端静态页面和实时获取最新内容。
  • 文件占用:对比原版4MB的bundle.js文件大小,本项目仅需要5.50KB的fclite.min.js文件即可轻量的展示到前端。
  • 前端分离: 将前后端分离,前端文件放在page分支,后端文件放在主分支

二、实操

方案一

后端采用最新的 Friend-Circle-Lite 项目进行部署,效果如下:

image-20241023002012733

前端

1、友圈json生成

新增友链后记得重新生成友圈json文件

在博客根目录创建link.js,复制下面的代码(原项目地址说明文档没有定义blacklist会执行出错)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
const YML = require('yamljs')
const fs = require('fs')

// 定义黑名单
const blacklist = []; // 替换为实际要过滤的名字

let friends = [],
data_f = YML.parse(fs.readFileSync('source/_data/link.yml').toString().replace(/(?<=rss:)\s*\n/g, ' ""\n'));

data_f.forEach((entry, index) => {
let lastIndex = 2;
if (index < lastIndex) {
const filteredLinkList = entry.link_list.filter(linkItem => !blacklist.includes(linkItem.name));
friends = friends.concat(filteredLinkList);
}
});

// 根据规定的格式构建 JSON 数据
const friendData = {
friends: friends.map(item => {
return [item.name, item.link, item.avatar];
})
};

// 将 JSON 对象转换为字符串
const friendJSON = JSON.stringify(friendData, null, 2);

// 写入 friend.json 文件
fs.writeFileSync('./source/friend.json', friendJSON);

console.log('friend.json 文件已生成。');

在博客根目录下运行

1
node link.js

生成friend.json文件,可以把这个文件移动到一个你喜欢的目录,比如我放到了/api/friends目录下

2、创建新页面

创建一个新的页面,尽量不要和主题默认朋友圈页面页面相同,比如安知鱼主题默认页面为fcircle

此处我直接用之前创建好的页面flink

1
hexo new page flink

页面 Front-matter配置如下

1
2
3
4
5
6
7
8
---
title: 時光的鱼塘
type: flink
aside: false
top_img: false
comments: false
date: 2024-05-08 15:23:31
---

继续粘贴以下内容:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<div id="friend-circle-lite-root"></div>
<script>
if (typeof UserConfig === 'undefined') {
var UserConfig = {
// 填写你的fc Lite地址
private_api_url: 'https://friends.shiguangdev.cn/',
// 点击加载更多时,一次最多加载几篇文章,默认20
page_turning_number: 20,
// 头像加载失败时,默认头像地址
error_img: 'https://i.p-i.vip/30/20240815-66bced9226a36.webp',
}
}
</script>
<link rel="stylesheet" href="https://fastly.jsdelivr.net/gh/willow-god/Friend-Circle-Lite/main/fclite.min.css">
<script src="https://fastly.jsdelivr.net/gh/willow-god/Friend-Circle-Lite/main/fclite.min.js"></script>

其中private_api_url是要修改后端服务地址,这里可以先不填注意尾部带/,不要遗漏。

执行命令三连部署一次,因为后面要用到friend.json文件

1
hexo cl && hexo g && hexo d

后端

这里我采用自部署使用方法

如果你有一台境内服务器,你也可以通过以下操作将其部署到你的服务器上,操作如下:

1、前置工作

确保你的服务器有定时任务 crontab 功能包,一般是linux自带,如果你没有宝塔等可以管理定时任务的面板工具,可能需要你自行了解定时工具并导入,本教程提供了简单的介绍。

2、克隆仓库

可以在本地下载好上传到云服务器

1
2
git clone https://github.com/willow-god/Friend-Circle-Lite.git
cd Friend-Circle-Lite

3、修改conf.yaml配置文件

设置spider_settings.json_url为你friend.json文件访问地址。

例如此处我的是:https://blog.shiguangdev.cn/api/friends/friend.json

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 爬虫相关配置
# 解释:使用request实现友链文章爬取,并放置到根目录的all.json下
# enable: 是否启用爬虫
# json_url: 请填写对应格式json的地址,仅支持网络地址
# article_count: 请填写每个博客需要获取的最大文章数量
# marge_result: 是否合并多个json文件,若为true则会合并指定网络地址和本地地址的json文件
# enable: 是否启用合并功能,该功能提供与自部署的友链合并功能,可以解决服务器部分国外网站无法访问的问题
# marge_json_path: 请填写网络地址的json文件,用于合并,不带空格!!!
spider_settings:
enable: true
json_url: "https://blog.shiguangdev.cn/api/friends/friend.json"
article_count: 3
merge_result:
enable: false
merge_json_url: "https://fc.liushen.fun"

另外可以修改article_count来限制每个友链最多获取文章数量,避免友链朋友圈展示的文章过于集中。

其他配置项不需要的话可以都改为false

4、安装依赖

下载服务相关包,其中 requirements-server.txt 是部署API服务所用包, requirements.txt 是抓取服务所用包,请均下载一遍。

1
2
pip install -r ./requirements.txt
pip install -r ./server/requirements-server.txt

5、部署服务

进入目录路径后直接运行deploy.sh脚本启动API服务:

1
2
chmod +x ./deploy.sh
./deploy.sh

我这里执行./deploy.sh时报了个错误

image-20241023003212633

通常是由于脚本文件中包含了 Windows 样式的行尾符(CRLF),而 Linux/Unix 系统期望的是 Unix 样式的行尾符(LF)。这种情况下,^M 字符表示的是回车符(CR),它在 Linux 系统上是不可见的,并且会导致解释器无法正确识别。可以使用支持多种换行符格式的文本编辑器,如 VSCode、Sublime Text 或 Notepad++,打开文件并将其保存为 Unix 格式(LF),然后重新运行即可

其中的注释应该是较为详细的,如果部署成功你可以使用以下命令进行测试,如果获取到了首页html内容则成功:

1
curl 127.0.0.1:1223

或者可以直接浏览器访问,将IP替换为服务器公网IP即可

image-20241023003721685

这个端口号可以修改,在server.py最后一行修改数字即可,如果你想删除该API服务,可以使用ps找到对应进程并使用Kill命令杀死进程:

1
2
ps aux | grep python
kill -9 [这里填写上面查询结果中对应的进程号]

为了后面方便,可以再给当前服务配置反向代理,例如我代理到域名friends.shiguangdev.cn

我这里直接借助1Panel创建反向代理并配置域名证书了。

image-20241023003918868

最后修改页面中的private_api_url为你的后端服务域名或公网ip即可。

本地直接访问http://localhost:4000/flink/进行预览

6、文章定时抓取

可以借助1Panel获取宝塔面板创建定时任务,实现文章定时抓取

image-20241025111547975

7、配置服务开机自启动

systemd 是现代 Linux 发行版中常用的初始化系统。你可以创建一个自定义的 systemd 服务来启动你的脚本。

  1. 创建一个新的服务文件:

    1
    sudo nano /etc/systemd/system/friend-circle-lite.service
  2. 在文件中添加以下内容:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    [Unit]
    Description=Friend Circle Lite Deployment Script
    After=network.target

    [Service]
    ExecStart=/home/Friend-Circle-Lite/deploy.sh
    WorkingDirectory=/home/Friend-Circle-Lite
    User=your-username
    Restart=always

    [Install]
    WantedBy=multi-user.target

    WorkingDirectoryExecStart替换为工作目录和启动脚本所在目录,将 your-username 替换为实际的用户名。

  3. Ctrl + X,然后按 Y 保存文件并退出编辑器。

  4. 重新加载 systemd 配置:

    1
    sudo systemctl daemon-reload
  5. 启用服务以在开机时启动:

    1
    sudo systemctl enable friend-circle-lite.service
  6. 启动服务:

    1
    sudo systemctl start friend-circle-lite.service

方案二

样式一:

image-20241219144316112

样式二:

image-20241219125933165

后端

后端建议采用Server部署,采用Docker部署无法修改默认配置路径地址(试了很多次,都采用的默认配置)

首先请确保你的服务器安装好docker和git,如果未安装可参考如何安装docker如何安装git

1、克隆仓库

clone项目仓库,地址:https://github.com/Rock-Candy-Tea/hexo-circle-of-friends

2、修改配置

修改配置文件中的友链地址:hexo_circle_of_friends/fc_settings.yaml

注意,theme根据你实际用到的主题尽心配置,我使用的安知鱼主题,采用默认配置common2

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
# 友链页地址
# 参数说明:
# link:必填,在这里填写你的友链页面地址
# theme:必填,友链页的获取策略。需要指定该页面的主题,可选参数如下(这些是目前支持的主题):
# - common1: 通用主题1,请参考:https://fcircle-doc.yyyzyyyz.cn/#/developmentdoc?id=友链页适配
# - common2: 通用主题2,请参考:https://fcircle-doc.yyyzyyyz.cn/#/developmentdoc?id=友链页适配
# - butterfly:butterfly主题
# - fluid:fluid主题
# - matery:matery主题
# - nexmoe:nexmoe主题
# - stun:stun主题
# - sakura: sakura主题
# - volantis:volantis主题
# - Yun:Yun主题
# - stellar:stellar主题
# 支持配置多个友链页面并指定不同主题策略,每个用{}分隔,它们会被同时爬取,数据保存在一起。
LINK: [
{ link: "https://blog.shiguangdev.cn/link/", theme: "common2" }, # 友链页地址1,修改为你的友链页地址
# { link: "https://blog.class1v1.com/link/", theme: "common2" }, # 友链页地址2
# { link: "https://immmmm.com/about/", theme: "common1" }, # 友链页地址3
# ...
]

修改每个友链最多获取文章数量,避免友链朋友圈展示的文章过于集中。

1
2
3
# 每个友链最多获取几篇文章,此值越大,则抓取的文章上限越多,相应地运行速度也会降低,反之亦然
# 请设置一个正整数
MAX_POSTS_NUM: 3

3、安装依赖

1
2
pip install -r ./requirements.txt
pip install -r ./hexo_circle_of_friends/requirements.txt

4、部署服务

然后运行位于项目根目录的部署脚本:

1
python3 deploy.py

选择docker--->部署,选择1、Server进行部署,等待运行完毕即可。

尝试访问API:

1
curl 127.0.0.1:8000/all

出现数据即为部署成功。

image-20241219103808311

接下来,开放服务器的对应端口,就可以通过IP:端口或者域名:端口访问到API,前端需要的就是这个地址。

直接访问这个地址是没有数据的

image-20241023010406124

访问/all可以查看所有数据

image-20241023010549059

访问/randompost随机获取一篇文章数据

image-20241023010606106

使用如下命令查看日志

1
cat /tmp/crawler.log

最后配置下反向代理

image-20241023010742385

5、配置服务开机自启动

为了在启动脚本中处理需要两次确认的步骤,你可以使用 Expect 脚本来自动化输入。Expect 是一个用于自动化交互式应用程序的工具,可以模拟用户输入。

首先,确保系统上安装了 Expect

1
sudo apt-get install expect

然后,创建一个 Expect 脚本来处理确认步骤。创建一个名为 start_hexo_friends.expExpect 脚本:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
#!/usr/bin/expect -f

# 启动 Python 脚本
spawn /usr/bin/python3 /home/hexo-circle-of-friends/deploy.py

# 第一次输入:选择部署方式
expect "欢迎使用部署工具,选择部署方式:" { send "1\r" }

# 第二次输入:选择部署操作
expect "请选择:" { send "1\r" }

# 第三次输入:指定 API 服务端口(直接回车选择默认值)
expect "指定api服务端口,按回车不输入则默认为8000:" { send "\r" }

# 第四次输入:等待部署完成并退出
expect "已部署!" { send "q\r" }

# 等待脚本结束
# expect eof

# 保持脚本运行
# interact

while {1} {
sleep 60
}

将上述脚本保存为 /home/hexo-circle-of-friends/start_hexo_friends.exp,并赋予执行权限:

1
chmod +x /home/hexo-circle-of-friends/start_hexo_friends.exp

接下来,修改你的 systemd 服务单元文件,使用 Expect 脚本来启动服务:

1
sudo nano /etc/systemd/system/hexo-circle-of-friends.service

在打开的文件中,输入以下内容:

1
2
3
4
5
6
7
8
9
10
11
12
13
[Unit]
Description=Hexo Circle of Friends Service
After=network.target

[Service]
ExecStart=/usr/bin/expect /home/hexo-circle-of-friends/start_hexo_friends.exp
WorkingDirectory=/home/hexo-circle-of-friends
Restart=on-failure
User=your-username
Group=your-username

[Install]
WantedBy=multi-user.target

替换ExecStart中的启动文件路径,将 your-usernameyour-groupname 替换为你实际使用的用户名和组名。如果你不确定,可以使用 whoami 命令查看当前用户名,使用 id -gn 查看当前用户的组名。

Ctrl + X,然后按 Y 保存文件并退出编辑器。

运行以下命令以重新加载 systemd 配置,使新的服务文件生效:

1
sudo systemctl daemon-reload

启用服务以确保它在开机时自动启动:

1
sudo systemctl enable hexo-circle-of-friends.service

立即启动服务:

1
sudo systemctl start hexo-circle-of-friends.service

调试 Expect 脚本:

如果服务仍然无法正常运行,可以手动运行 Expect 脚本并启用调试模式,查看详细的匹配过程:

1
expect -d /home/hexo-circle-of-friends/start_hexo_friends.exp

根据调试输出,检查 expect 语句是否与 deploy.py 的实际输出完全匹配。如果提示信息有变化,可以调整 expect 语句中的字符串。

验证服务状态和日志:

使用以下命令查看服务状态和日志,确认是否正常运行:

1
2
sudo systemctl status hexo-circle-of-friends.service
sudo journalctl -u hexo-circle-of-friends.service -e

脚本说明

  1. Expect 脚本

    为了避免服务进入重启循环,我们需要调整 Expect 脚本和 systemd 服务配置,确保服务在完成部署后不会退出。在 Expect 脚本中,使用 sleepwhile true 保持脚本运行,而不是依赖 interact。将 while {1} { sleep 60 } 添加到脚本末尾,使脚本在完成部署后保持运行。

  2. systemd 服务文件

Restart 配置改为 Restart=on-failure,避免服务在正常退出后重启

前端

样式一:适配安知鱼主题

由于我使用的是安知鱼主题,所以我这里直接参考安知鱼的教程了。

1、创建朋友圈页面

在 Hexo 博客根目录 [blog]下打开终端,输入

1
hexo new page fcircle

打开[blog]\source\fcircle\index.md,添加一行type: "fcircle":

1
2
3
4
5
6
7
8
---
title: 朋友圈
date: 2022-11-21 17:06:17
comments: false
aside: false
top_img: false
type: "fcircle"
---
2、生成js文件

https://npm.elemecdn.com/anzhiyu-blog-static@1.0.0/js/friends_vue/index.js保存到一个js文件,将中的 url替换为您的后端 url

image-20241022143006243

将js文件放到博客某个目录或者上传到代码托管平台,此处我直接放到/api/friends/friends_front.js

也可以下载前端项目:hexo-circle-of-friends-front到本地,修改代码中src/utils/config.ts的 url 变量路径,改为你自己的,然后执行npm run build构建后将dist文件夹中的js粘贴到你的js文件中

image-20241023012226802

3、启用朋友圈功能

主题配置文件中开启menu中友链和朋友圈的注释,导航栏朋友圈,注意缩进!!!

1
2
3
menu:
友链:
朋友圈: /fcircle/ || anzhiyu-icon-artstation

主题配置文件中开启friends_vue.enable

1
2
3
4
5
6
7
# 朋友圈配置
friends_vue:
enable: true
vue_js: /api/friends/friends_front.js
apiurl: https://fcircle.shiguangdev.cn/
top_tips: 🌟🌟欢迎光临時光的鱼塘🌟🌟
top_background: https://img.shiguangdev.cn/i/2024/10/21/6715cffbdb61a.webp
参数 备选值/类型 解释
enable boolean 【必须】是否启用
vue_js url 【必须】朋友圈前端构建后的 url
apiurl string 【必须】朋友圈后端 url
top_background url 【可选】朋友圈顶部背景图
4、访问并测试

执行三连命令

1
hexo cl && hexo g && hexo s 

访问域名下的/fcircle即可看到效果

image-20241219144316112

样式二

1、创建朋友圈页面

在 Hexo 博客根目录 [blog]下打开终端,输入

安知鱼主题不能与fcircle名称相同,例如我这里为flink

1
hexo new page flink

打开[blog]\source\flink\index.md,添加一行type: "flink":

1
2
3
4
5
6
7
8
---
title: 朋友圈
type: flink
comments: false
aside: false
top_img: false
date: 2024-05-08 15:23:31
---
2、添加样式

[blog]\source\flink\index.md下面粘贴代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<div id="hexo-circle-of-friends-root"></div>
<script>
let UserConfig = {
// 填写你的api地址
private_api_url: 'https://fcircle.shiguangdev.cn/',
// 初始加载几篇文章
page_init_number: 20,
// 点击加载更多时,一次最多加载几篇文章,默认10
page_turning_number: 10,
// 头像加载失败时,默认头像地址
error_img: 'https://sdn.geekzu.org/avatar/57d8260dfb55501c37dde588e7c3852c',
// 进入页面时第一次的排序规则
sort_rule: 'created',
// 本地文章缓存数据过期时间(天)
expire_days: 1,
}
</script>
<script type="text/javascript" src="https://npm.elemecdn.com/fcircle-theme-yyyz@1.0.13/dist/fcircle.min.js"></script>

修改private_api_url为你的后端api地址,例如:https://fcircle.shiguangdev.cn/

本地访问:http://localhost:4000/flink/进行预览,效果如下:

image-20241023033353845

3、修改公共库请求地址

这里有个小问题,点击切换公共库时会调用后端开发者的接口,但是目前已经失效了,很影响体验感。

image-20241219130141021

请求如下:

image-20241219130231152

我们把调用地址改成自己的接口,把原来的js下载到本地

例如此处我放到: \blog\source\api\friends\fcircle.min.js

1
<script type="text/javascript" src="https://npm.elemecdn.com/fcircle-theme-yyyz@1.0.13/dist/fcircle.min.js"></script>

搜索并替换下面两个地址

1
2
3
private_api_url:"http://127.0.0.1:8000/"

public_api_url:"https://fcircle-pub.rct.cool/"

如下:

image-20241219130705336

然后把js路径改成我们自己的

1
<script type="text/javascript" src="http://localhost:4000/api/friends/fcircle.min.js"></script>
4、修改菜单地址

最后别忘了修改菜单地址

image-20241219131711194

控制面板配置

新版前端在顶部右下角卡片新增管理面板:

image-20241022145436138

点击即可进入。第一次部署成功后,输入第一个密码的同时设置密码。

再次修改配置

image-20241022161817532

前端改密码方式

友链朋友圈项目的前端第一次登录时即设置密码,如果想要改的话则需要自行删除数据库 authsecret 表格。

进入服务器 hexo-circle-of-friends 根目录,找到 data.db 文件

1
2
3
4
5
6
7
8
9
sqlite3
sqlite> .open data.db # 打开数据库
sqlite> .tables
# auth friends posts secret
sqlite> DROP TABLE auth;
sqlite> DROP TABLE secret;
sqlite> .tables
# friends posts
sqlite> .exit

参考

鱼塘朋友圈部署前端方案

鱼塘朋友圈后端部署方案

安知鱼主题朋友圈页面配置

Friend-Circle-Lite:轻量友链朋友圈

友链朋友圈

林木木前端部署方案