如何在yaml文件中引用python函数?

前言

经常看到很多同学问到,如何在 yaml 文件中引用一个 python 的函数?

问题分析

大家对yaml文件还处于比较陌生的阶段,yaml 和 json 文件本质上是一样的,都是静态的文件,当然不能直接引用 python 的函数。
那这时候就有人问到了,那为什么 httprunner 框架可以在yaml文件中引用函数呢?

  • 这是因为 httprunner 框架封装过对 yaml 文件的读取了,它是先读取文件内容,正则提取到 ${} 括号里面的函数内容,再把函数的值替换过去

那么我们能不能实现这种效果呢?

  • 当然是可以的,可以参考httprunner的实现,也可以用到 python 的模板 jinja2 来实现。

使用模板可以编写出可读性更好,更容易理解和维护的代码,并且使用范围非常广泛,因此怎么使用模板主要取决于我们的想象力和创造力。
python的模板库jinja2 功能是非常强大的。

jinja2 模板库

先需要pip安装

pip install jinja2

render 函数实现

在yaml文件中,通过 {{ 函数名称() }} 来引用函数

写个 render 函数读取 yaml 文件内容

import os
import jinja2
import yaml
import random


def render(tpl_path, **kwargs):
    path, filename = os.path.split(tpl_path)
    return jinja2.Environment(loader=jinja2.FileSystemLoader(path or './')
    ).get_template(filename).render(**kwargs)

读取到的yaml文件本质上都是字符串来读取的,通过jinja2 模板来读取,会先把函数的值替换进去。最后再转成python的dict结构

import os
import jinja2
import yaml
import random
"""
作者:上海-悠悠
python QQ交流群:730246532
联系微信/QQ: 283340479
"""

def render(tpl_path, **kwargs):
    path, filename = os.path.split(tpl_path)
    return jinja2.Environment(loader=jinja2.FileSystemLoader(path or './')
    ).get_template(filename).render(**kwargs)


# yaml 文件调用以下函数
def rand_str():
    return str(random.randint(1000000, 2000000))


if __name__ == '__main__':
    r = render("aa.yml", **{"rand_str": rand_str})
    print(r)
    print(yaml.safe_load(r))

运行结果

name: yoyo
age: 22
tel: 1616350
{'name': 'yoyo', 'age': 22, 'rand_str': 1616350}

上面读取函数是写死的,我们希望能自动加载类似于debugtalk.py的文件来自动加载函数

自动加载debug.py里面的函数

写一个debug.py 文件,实现 yaml 文件里面定义的函数去替换值。

import random
"""
作者:上海-悠悠
python QQ交流群:730246532
联系微信/QQ: 283340479
"""

# yaml 文件调用以下函数
def rand_str():
    return str(random.randint(1000000, 2000000))

run.py里面定义一个函数自动读取debug.py里面的函数,生成dict 键值对格式

def all_functions():
    """加载debug.py模块"""
    debug_module = importlib.import_module("debug")
    all_function = inspect.getmembers(debug_module, inspect.isfunction)
    # print(dict(all_function))
    return dict(all_function)

函数返回 {'rand_str': <function rand_str at 0x0000017B72EA8488>}

完整的run.py文件内容

import os
import jinja2
import yaml
import importlib
import inspect
"""
作者:上海-悠悠
python QQ交流群:730246532
联系微信/QQ: 283340479
"""

def render(tpl_path, **kwargs):
    """渲染yml文件"""
    path, filename = os.path.split(tpl_path)
    return jinja2.Environment(loader=jinja2.FileSystemLoader(path or './')
    ).get_template(filename).render(**kwargs)


def all_functions():
    """加载debug.py模块"""
    debug_module = importlib.import_module("debug")
    all_function = inspect.getmembers(debug_module, inspect.isfunction)
    print(dict(all_function))
    return dict(all_function)


if __name__ == '__main__':
    r = render("aa.yml", **all_functions())
    print(r)
    print(yaml.safe_load(r))

运行结果

{'rand_str': <function rand_str at 0x000001931C248488>}
name: yoyo
age: 22
tel: 1010421
{'name': 'yoyo', 'age': 22, 'tel': 1010421}

网易云完整视频课程《pytest+yaml 框架使用与开发》https://study.163.com/course/courseMain.htm?courseId=1213419817&share=2&shareId=480000002230338
报名咨询wx:283340479 (已报名的同学学习过程中有问题,都可以协助解决)

posted @ 2022-02-17 19:19  上海-悠悠  阅读(3037)  评论(0编辑  收藏  举报