大家好~我是米洛

我正在从0到1打造一个开源的接口测试平台, 也在编写一套与之对应的教程,希望大家多多支持。

欢迎关注我的公众号米洛的测开日记,获取最新文章教程!

回顾

上一节我们编写了在线执行测试计划功能,并稍微改了下报告页面。那其实我们之前的内容都是有很多在里面的。

比如http请求只支持了json和form,没有支持文件上传的请求,甚至有一些crud的功能都没有太完善。

不过不要紧,我的想法还是先创造,再完善。当然也不是盲目创造,也得提前预判好后面的走向。

为什么要用oss

pity里面oss打算用在2个地方,第一个就是一些静态资源图片。比方说项目图片用户头像

另一个地方就是上文说的,测试文件上传接口的时候,我们需要测试文件。这些文件怎么来,怎么管理?都得借助oss来完成。

目前以我熟悉的oss为例,打算支持以下几种oss:

  • 阿里云oss ✔
  • 腾讯云cos(无账号) 💊
  • 七牛云(免费,我能给展示demo)
  • gitee(免费)
    其中阿里云和腾讯云由于都是付费产品,博主还是买不起的。所以可能需要有对应的测试账号,在此感谢小右(Mini-Right)的帮助,给了我一个阿里云的测试账号,保证了crud能正常进行。
    如果需要我实现其他oss客户端,请带上对应的测试账号私信我哈。
    测试网站的数据估计会用gitee或者七牛云

说明

关于文件管理这块,由于时间关系,我暂时不会落一张表与oss数据进行关联,主要目的是为了节省时间

表关联可以作为二期工程。

编写oss基类

新建app.middleware.oss.oss_file.py

由于咱们支持多种oss客户端,所以包装好一个抽象类(类似go的interface)等着各个oss客户端去实现之。

  1. from abc import ABC, abstractmethod
  2. from typing import ByteString
  3. class OssFile(ABC):
  4. @abstractmethod
  5. def create_file(self, filepath: str, content: ByteString):
  6. pass
  7. @abstractmethod
  8. def update_file(self, filepath: str, content: ByteString):
  9. pass
  10. @abstractmethod
  11. def delete_file(self, filepath: str):
  12. pass
  13. @abstractmethod
  14. def list_file(self):
  15. pass
  16. @abstractmethod
  17. def download_file(self, filepath):
  18. pass

抽象类没有具体的实现,只有方法的定义,目的是为了限制子类的方法,即必须实现基类中定义的方法。

总结了一下,必须有增删改查下载5种方法。

编写AliyunOss实现

新建app.middleware.oss.aliyun.py

  1. from typing import ByteString
  2. import oss2
  3. from app.middleware.oss.files import OssFile
  4. class AliyunOss(OssFile):
  5. def __init__(self, access_key_id: str, access_key_secret: str, endpoint: str, bucket: str):
  6. auth = oss2.Auth(access_key_id=access_key_id,
  7. access_key_secret=access_key_secret)
  8. # auth = oss2.AnonymousAuth()
  9. self.bucket = oss2.Bucket(auth, endpoint, bucket)
  10. def create_file(self, filepath: str, content: ByteString):
  11. self.bucket.put_object(filepath, content)
  12. def update_file(self, filepath: str, content: ByteString):
  13. self.bucket.put_object(filepath, content)
  14. def delete_file(self, filepath: str):
  15. self.bucket.delete_object(filepath)
  16. def list_file(self):
  17. ans = []
  18. for obj in oss2.ObjectIteratorV2(self.bucket):
  19. ans.append(dict(key=obj.key, last_modified=obj.last_modified,
  20. size=obj.size, owner=obj.owner))
  21. return ans
  22. def download_file(self, filepath):
  23. if not self.bucket.object_exists(filepath):
  24. raise Exception(f"oss文件: {filepath}不存在")
  25. return self.bucket.get_object(filepath)

AliyunOss继承了OssFile,构造方法获取阿里云的身份信息,并验证。最后读取bucket,这bucket我理解的是一块区域,你的文件都存储在这块区域里面,我们就叫他F盘吧。

其他的方法很简单,基本上是调用对应的api,去做crud操作。oss没有文件夹的概念,都是统一用路径来存储文件地址的,比如:

woody/github.txt

这个文件路径可以理解为,woody目录下的github.txt文件。

编写获取客户端方法

app.middleware.oss._\_init__.py

  1. from app.core.configuration import SystemConfiguration
  2. from app.middleware.oss.aliyun import AliyunOss
  3. from app.middleware.oss.files import OssFile
  4. class OssClient(object):
  5. _client = None
  6. @classmethod
  7. def get_oss_client(cls) -> OssFile:
  8. """
  9. 通过oss配置拿到oss客户端
  10. :return:
  11. """
  12. if OssClient._client is None:
  13. cfg = SystemConfiguration.get_config()
  14. oss_config = cfg.get("oss")
  15. access_key_id = oss_config.get("access_key_id")
  16. access_key_secret = oss_config.get("access_key_secret")
  17. bucket = oss_config.get("bucket")
  18. endpoint = oss_config.get("endpoint")
  19. if oss_config is None:
  20. raise Exception("服务器未配置oss信息, 请在configuration.json中添加")
  21. if oss_config.get("type").lower() == "aliyun":
  22. return AliyunOss(access_key_id, access_key_secret, endpoint, bucket)
  23. raise Exception("不支持的oss类型")
  24. return OssClient._client

我们在configuration.json配置oss信息,接着每次都从OssClient获取客户端即可。

测试平台系列(90) 编写oss客户端 - 图1

编写后端接口

  1. from fastapi import APIRouter, File, UploadFile
  2. from app.handler.fatcory import PityResponse
  3. from app.middleware.oss import OssClient
  4. router = APIRouter(prefix="/oss")
  5. @router.post("/upload")
  6. async def create_oss_file(filepath: str, file: UploadFile = File(...)):
  7. try:
  8. file_content = await file.read()
  9. # 获取oss客户端
  10. client = OssClient.get_oss_client()
  11. client.create_file(filepath, file_content)
  12. return PityResponse.success()
  13. except Exception as e:
  14. return PityResponse.failed(f"上传失败: {e}")
  15. @router.get("/list")
  16. async def list_oss_file():
  17. try:
  18. client = OssClient.get_oss_client()
  19. files = client.list_file()
  20. return PityResponse.success(files)
  21. except Exception as e:
  22. return PityResponse.failed(f"获取失败: {e}")
  23. @router.get("/delete")
  24. async def delete_oss_file(filepath: str):
  25. try:
  26. client = OssClient.get_oss_client()
  27. client.delete_file(filepath)
  28. return PityResponse.success()
  29. except Exception as e:
  30. return PityResponse.failed(f"删除失败: {e}")
  31. @router.post("/update")
  32. async def update_oss_file(filepath: str, file: UploadFile = File(...)):
  33. """
  34. 更新oss文件,路径不能变化
  35. :param filepath:
  36. :param file:
  37. :return:
  38. """
  39. try:
  40. client = OssClient.get_oss_client()
  41. file_content = await file.read()
  42. client.update_file(filepath, file_content)
  43. return PityResponse.success()
  44. except Exception as e:
  45. return PityResponse.failed(f"删除失败: {e}")

方法很简单,文件路径在url参数里边,文件的话,利用fastapi里面的File和UploadFile皆可获取到上传的文件。

注意,必须先安装python-multipart库配合文件上传

测试一下

成功读取到了oss的文件信息,但下载接口

测试平台系列(90) 编写oss客户端 - 图2

好像忘记下载文件相关内容了,那么参考我的上一篇FsatApi下载文件的文章吧。

FastApi下载文件

或者直接去github查看源码~

今天的内容就介绍到这里了,下一节卷oss的用途。