前言
最开始使用的 gitee 作为个人图床,但总觉得不踏实,gitee 毕竟是公开的仓库,而且还是国内的服务
这两天考虑部署 easyimage 个人图床的时候,使用 picgo + web-uploader ,发现图片无法正常上传
于是瞅了一下 picgo-plugin-gitee 插件的源码参考,不曾想上面赫然写着,图床这个在几年前就被 gitee 废掉了

还有一个重要的原因是自己手贱,本来想用 notepad++ 打开一个文本文件,结果给整到图床上去了
picgo 这个右键菜单 ”Upload pictures with PicGo” 也太便利了,竟然没对文件类型进行过滤就直接给上传了
文件虽然是删除了,也不是什么私密文件,但是 log 还在,万一哪天不小心把一些重要文件给上传就不好了


图床部署
开源的图床用的比较多的就是兰空和easyimage了,这里以easyiamge进行图床部署
官方地址 https://github.com/icret/EasyImages2.0
代码下载后,放置于 phpstudy 的 WWW 目录下,新增一个站点指定目录就可以了
安装比较简单,不需要配置数据库什么的,打开站点首页,然后下一步基本就可以了
初始化一个管理员的账号密码

然后刷新首页,输入账号密码登录就可以了,登录后页面大致如下
可以在 “设置” 中进行更精细的配置,如显示权限,上传权限等

设置
如果使用接口进行图片上传,需要一个 api 地址,和一个授权的 token ,在 “设置” 中的 “API设置” 中进行设置

python代码上传图片
python 的参考代码如下,可以通过剪切板上传,或者读取文件上传
import io
import requests
from PIL import ImageGrab
image_path = "./docs/images/225906016.png"
token = "1c17b11693cb5ec63859b091c5b9c1b2"
url = "http://192.168.10.200/api/index.php"
image = ImageGrab.grabclipboard()
if image:
    img_byte_arr = io.BytesIO()
    image.save(img_byte_arr, format='PNG')
    img_byte_arr.seek(0)
    files = {'image': ('screenshot.png', img_byte_arr, 'image/png')}
    data = {'token': token}
    print("upload image use clipboard...")
else:
    files = {'image': open(image_path, 'rb')}
    data = {'token': token}
    print("upload image use local image file...")
response = requests.post(url, files=files, data=data)
if response.status_code == 200:
    print("Upload successful, text:", response.text)
else:
    print(f"Upload failed with status code {response.status_code}.")
PicGo 配置
安装插件以及配置
picgo 支持大部分的图床上传,但是官方并不直接支持 easyimage 图床
picgo 以提供插件支持的方式,方便为各种图床定制开发相应的上传插件
easyimage 官方推荐使用 web-uploader 插件,另外还有名为 “easyimage” 的插件
在 “插件设置” 中搜索,或者在 github 中搜索,将代码下载到本地,然后再 “导入本地插件” 也可以

安装完插件后,输入配置信息,web-uploader 稍微有点不同,这里以 easyimage 插件进行说明
就配置 api 地址以及 token 就可以了

插件问题分析
上面两个插件尝试了,发现都不能正常上传,估计是我环境的问题
分析的时候,easyimage 后端无法解析到 token ,网上搜了下也没有人提到这类问题
以 windows 为例,插件安装在以下目录中
C:\Users\Administrator\AppData\Roaming\picgo\node_modules

实际上核心的文件只有一个 picgo-plugin-easyimage\dist\index.js
尝试了好多次,也无法使用 PicGo 通过 easyimage 图床插件进行图片上传
只好将插件代码修改下,改为纯手动构建的请求包,然后就可以正常上传图片了
修改后的完整代码如下,有需要可以直接替换原文件的代码,然后重启 PicGo
const UPLOADER = "easyimage";
module.exports = (ctx) => {
    const config = (ctx) => {
        let userConfig = ctx.getConfig("picBed." + UPLOADER);
        if (!userConfig) {
            userConfig = {};
        }
        return [
            {
                name: "server",
                type: "input",
                default: userConfig.server,
                required: true,
                message: "示例: http://10.20.30.19:8070/api/index.php",
                alias: "API调用地址",
            },
            {
                name: "token",
                type: "input",
                default: userConfig.token,
                required: true,
                message: "认证 token 信息",
                alias: "调用Token",
            },
        ];
    };
    // 上传图片
    const uploader = {
        config,
        handle: async (ctx) => {
            let userConfig = ctx.getConfig("picBed." + UPLOADER);
            if (!userConfig) {
                throw new Error("Can't find uploader config");
            }
            const imgList = ctx.output;
            for (let i in imgList) {
                const img = imgList[i];
                const { base64Image, fileName } = img;
                let { buffer } = img;
                if (!buffer && base64Image) {
                    buffer = Buffer.from(img.base64Image, "base64");
                }
                // 随机生成边界
                const boundary = '----WebKitFormBoundary' + Math.random().toString(36).substr(2, 10);
                // 构造 multipart/form-data 内容
                let reqBodyParts = [];
                reqBodyParts.push(Buffer.from(`--${boundary}\r\n`));
                reqBodyParts.push(Buffer.from(`Content-Disposition: form-data; name="token"\r\n\r\n`));
                reqBodyParts.push(Buffer.from(`${userConfig.token}\r\n`));
                reqBodyParts.push(Buffer.from(`--${boundary}\r\n`));
                reqBodyParts.push(Buffer.from(`Content-Disposition: form-data; name="image"; filename="${fileName}"\r\n`));
                reqBodyParts.push(Buffer.from(`Content-Type: image/png\r\n\r\n`));
                reqBodyParts.push(buffer);
                 reqBodyParts.push(Buffer.from(`\r\n`));
                // 计算 Content-Length
                reqBodyParts.push(Buffer.from(`--${boundary}--\r\n`));
                const reqBody = Buffer.concat(reqBodyParts);
                const requestConfig = {
                    url: userConfig.server,
                    method: "POST",
                    headers: { 
                        "Content-Type": `multipart/form-data; boundary=${boundary}`,
                        "User-Agent": "PicGo-easyimage",
                        "Content-Length": reqBody.length,
                    },
                    body: reqBody,
                };
                let body = await ctx.Request.request(requestConfig);
                body = JSON.parse(body);
                const { url: imgUrl, message } = body;
                if (imgUrl) {
                    img.imgUrl = imgUrl;
                }
                else {
                    ctx.emit("notification", {
                        title: "上传失败",
                        body: message,
                    });
                }
            }
            return ctx;
        },
    };
    const register = () => {
        ctx.helper.uploader.register(UPLOADER, uploader);
    };
    return {
        register,
        config,
        uploader: UPLOADER,
    };
};