通过 Nginx 建立自己的图床后,之前的 Picgo 无法使用,导致在文档中插入图片十分不便,本文记录自己搭建 Python 后端服务来为自己的图床适配 Picgo 的方法。
背景
- 已经成功搭建 Nginx 图床
- PicGo 上传图像十分方便,可以将图像数据放在数据包中向指定服务器发送
- 此时我们有服务器,有工具,只差中间接洽的桥梁
- Python 自带简易后端 flask,可以方便地获取数据包并返回我们需要的数据
实现要点
不同的使用者可能有不同的需求,我提供自己的实现思路,供大家参考。
调试
- 链路没有打通前调试十分重要,具体方法为:
- 在服务器安装 Python
- 本地 VScode 远程调试服务器代码
- 运行 flask 服务,本地使用 Picgo 发送数据包进行调试
思路
- 服务器上使用 Python 开启 flask 监听特定端口
- 服务器开通防火墙暴露端口
- 本地 Picgo 向服务器地址发送数据包
- flask 接收、解析数据,按照需求处理业务逻辑,返回文件 url
- 将该服务设置为服务器开机启动服务,之后便再也不用操心无人接管 Picgo 数据包
实现方法
PicGo 端配置
- 建议使用 Picgo 的自定义上传插件
- 配置
自定义 Web 图床设置
API 地址为服务器地址,后面
/
后可以接一段字符作为输入参数,我将其作为存放文件的子文件夹 post 参数名为文件所在参数名 其余建议不要填,不然会上传失败,不知道原因
服务器端配置
- 引入库
from flask import Flask, request
import io
from PIL import Image
import mtutils as mt
import numpy as np
- 配置信息
key_para = 'file' # Picgo 映射字段
save_root = 'path-to-your-file' # 数据存储文件夹
host = 'https://1.1.1.1' # flask 监听IP,建议使用 0.0.0.0
default_name = 'imgbed' # 默认二级文件夹名称
log_file_path = '/usr/local/imgbed/log.log' # 日志存放路径
port = '6789' # flask 监听端口
- 初始化 flask app
app = Flask(__name__)
app.last_file = None
# 日志对象
app.logger = mt.log_init(log_file_path)
- 核心数据解析于业务逻辑
def specific_path(save_to):
# 获取数据包文件
file = request.files[key_para]
file_content = file.read()
# 存放文件的文件夹
save_dir = mt.OS_join(save_root, save_to)
mt.dir_check(save_dir)
# 此处我将非 jpg 图像压缩为 jpg 图保存,其余文件类型直接保存
# check if a jpg Image file
if file.content_type[:5].lower() == 'image' and file.content_type[-4:].lower() not in ['/jpg', 'jpeg']:
# is image
image_obj = Image.open(io.BytesIO(file_content))
# to jpg
file_name = str(mt.Path(file.filename).with_suffix('.jpg'))
save_path = mt.OS_join(save_dir, file_name)
image_save_path = mt.Path(save_path)
image_save_path.parent.mkdir(parents=True, exist_ok=True)
# 保存 jpg 图像
image_obj.save(str(image_save_path))
# 记录日志
app.logger(f'Transfer and save image to {save_path}.')
else:
file_name = file.filename
save_path = mt.OS_join(save_dir, file_name)
obj = open(save_path, mode='wb')
# 保存文件
obj.write(file_content)
# 记录日志
app.logger(f'Save file to {save_path}.')
app.last_file = save_path
back_link = mt.OS_join(host, 'HexoFiles', save_to, file_name)
return back_link
- 之前使用 PicGo 传图时经常遇到想要删除刚刚上传图像的需求,此处加了相关功能
if '__del__' in save_to:
res_str = 'nothing happened.'
if app.last_file is not None:
mt.remove_file(app.last_file)
res_str = f"File deleted {app.last_file}"
app.logger(res_str)
app.last_file = None
return res_str
在 API 链接里
/
后的字符串中如包含__del__
则删除刚刚上传的图像,返回操作结果
- 如果没有配置 API 后的文件夹作为参数,使用默认二级文件夹
@app.route("/", methods=['GET','POST'])
def default_save_path():
default_path = default_name
return specific_path(default_path)
- 运行服务
if __name__ == '__main__':
app.logger("**************** Sever Start *******************")
app.run('0.0.0.0', port=port)
pass
- 完整代码
from flask import Flask, request
import io
from PIL import Image
import mtutils as mt
import numpy as np
key_para = 'file'
save_root = 'path-to-your-file'
host = 'https://1.1.1.1'
default_name = 'imgbed'
log_file_path = '/usr/local/imgbed/log.log'
port = '6789'
app = Flask(__name__)
app.last_file = None
app.logger = mt.log_init(log_file_path)
@app.route("/<save_to>", methods=['GET','POST'])
def specific_path(save_to):
if '__del__' in save_to:
res_str = 'nothing happened.'
if app.last_file is not None:
mt.remove_file(app.last_file)
res_str = f"File deleted {app.last_file}"
app.logger(res_str)
app.last_file = None
return res_str
file = request.files[key_para]
file_content = file.read()
save_dir = mt.OS_join(save_root, save_to)
mt.dir_check(save_dir)
# check if a jpg Image file
if file.content_type[:5].lower() == 'image' and file.content_type[-4:].lower() not in ['/jpg', 'jpeg']:
# is image
image_obj = Image.open(io.BytesIO(file_content))
# to jpg
file_name = str(mt.Path(file.filename).with_suffix('.jpg'))
save_path = mt.OS_join(save_dir, file_name)
image_save_path = mt.Path(save_path)
image_save_path.parent.mkdir(parents=True, exist_ok=True)
image_obj.save(str(image_save_path))
app.logger(f'Transfer and save image to {save_path}.')
else:
file_name = file.filename
save_path = mt.OS_join(save_dir, file_name)
obj = open(save_path, mode='wb')
obj.write(file_content)
app.logger(f'Save file to {save_path}.')
app.last_file = save_path
back_link = mt.OS_join(host, 'HexoFiles', save_to, file_name)
return back_link
@app.route("/", methods=['GET','POST'])
def default_save_path():
default_path = default_name
return specific_path(default_path)
if __name__ == '__main__':
app.logger("**************** Sever Start *******************")
app.run('0.0.0.0', port=port)
pass
设置开机运行服务
- 开机启动服务
[Unit]
Description = Service for uploading file to target dir
After = network.target
[Service]
ExecStart = path-to-your-python main.py
WorkingDirectory = /usr/local/imgbed/
StandardOutput = inherit
StandardError = inherit
Restart = always
User = lighthouse
[Install]
WantedBy=multi-user.target
参考资料
- https://github.com/PicGo/Awesome-PicGo