atomic-bomb-engine


Nameatomic-bomb-engine JSON
Version 0.38.0 PyPI version JSON
download
home_pageNone
Summary使用rust开发的高性能python压测工具
upload_time2024-05-07 07:41:37
maintainerNone
docs_urlNone
authorNone
requires_pythonNone
licenseMIT
keywords rust python binding
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # atomic-bomb-engine-py
#### [atomic-bomb-engine](https://github.com/we-lsp/atomic-bomb-engine)的python包装实现

<img src="img/atomic-bomb-engine-logo.png" width="350px" height="350px" alt="logo">


## 前端仓库
#### [atomic-bomb-engine-front](https://github.com/GiantAxeWhy/atomic-bomb-engine-front)

## 使用条件:
- python版本 >= 3.8
- windows(x86), linux(x86), mac

## 使用方法:
- ### 准备开始
通过pip安装 (0.5.0版本之前)
```shell
pip install atomic-bomb-engine-py
```
在python中引用时注意,需要引用atomic_bomb_engine, 而不是atomic_bomb_engine_py
<br/> 为了避免混淆,0.5.0版本之后,pip更换了包名,更改为atomic-bomb-engine,
```shell
pip install atomic-bomb-engine
```
在python中导入
```python
import atomic_bomb_engine
```
异步使用的时候,还需要引用asyncio
```python
import asyncio
```
- ### 开始压测
  - ~~单接口压测~~ (功能与多接口压测重叠,已废除)

  - 多接口压测

多接口压测可以使用batch_async方法进行操作,函数签名和解释如下
 ```python
async def batch_async(
        test_duration_secs: int,
        concurrent_requests: int,
        api_endpoints:List[Dict],
        step_option:Dict[str, int]|None=None,
        setup_options:List[Dict[str, Any]]|None=None,
        verbose:bool=False,
        should_prevent:bool=False,
        assert_channel_buffer_size:int=1024,
        timeout_secs=0,
        cookie_store_enable=True
) ->Dict:
  """
      批量压测
      :param test_duration_secs: 测试持续时间
      :param concurrent_requests: 并发数
      :param api_endpoints: 接口信息
      :param step_option: 阶梯加压选项
      :param setup_options: 初始化选项
      :param verbose: 打印详细信息
      :param should_prevent: 是否禁用睡眠
      :param assert_channel_buffer_size: 断言队列buffer大小
      :param timeout_secs: http超时时间
      :param cookie_store_enable: 是否为客户端启用持久性cookie存储。
  """
 ```

使用assert_option方法可以返回断言选项字典
```python
assert_options=[
atomic_bomb_engine.assert_option("$.code", 429),
atomic_bomb_engine.assert_option("$.code", 200)
])
```
jsonpath如果不会用的话,建议去[jsonpath](https://jsonpath.com/)学习

使用step_option方法可以返回阶梯加压选项字典
```python
def step_option(increase_step: int, increase_interval: int) -> Dict[str, int]:
    """
    生成step option
    :param increase_step: 阶梯步长
    :param increase_interval: 阶梯间隔
    """
```

同样的本包中也包含了一个对api_endpoint的包装:endpoint方法,方便调用,endpoint中的assert_options中也可以套用assert_option方法
 ```python
async def run_batch():
  result = await atomic_bomb_engine.batch_async(
    # 测试持续时间
    test_duration_secs=60,
    # 并发量
    concurrent_requests=200,
    # 阶梯设置(每5秒增加30个并发)
    step_option=atomic_bomb_engine.step_option(increase_step=30, increase_interval=5),
    # 接口超时时间
    timeout_secs=10,
    # 是否开启客户端启用持久性cookie存储
    cookie_store_enable=True,
    # 全局初始化
    setup_options=[
      atomic_bomb_engine.setup_option(
        name="初始化-1",
        url="http://localhost:8080/setup",
        method="get",
        jsonpath_extract=[
          atomic_bomb_engine.jsonpath_extract_option(key="test-msg", jsonpath="$.msg"),
          atomic_bomb_engine.jsonpath_extract_option(key="test-code", jsonpath="$.code"),
        ]
      )],
    # 是否开启详细日志
    verbose=False,
    # 被压接口设置
    api_endpoints=[
      atomic_bomb_engine.endpoint(
        # 接口任务命名
        name="test-1",
        # 针对每个接口初始化
        setup_options=[
          atomic_bomb_engine.setup_option(
            name="api-初始化-1",
            url="http://localhost:8080/api_setup",
            method="get",
            jsonpath_extract=[
              atomic_bomb_engine.jsonpath_extract_option(key="api-test-msg-1", jsonpath="$.msg"),
              atomic_bomb_engine.jsonpath_extract_option(key="api-test-code-1", jsonpath="$.code"),
            ]
          )
        ],
        # 被压接口url
        url="http://localhost:8080/direct",
        # 请求方式
        method="POST",
        # 权重
        weight=1,
        # 发送json请求
        json={"name": "{{api-test-msg-1}}", "number": 1},
        # 断言选项
        assert_options=[
          atomic_bomb_engine.assert_option(jsonpath="$.number", reference_object=1),
        ],
        # 思考时间选项(在最大和最小之间随机,单位毫秒)
        think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),
      ),
    ])
  print(result)
  return result
 ```
    
监听时可以使用BatchListenIter生成器
```python
async def listen_batch():
    iterator = atomic_bomb_engine.BatchListenIter()
    for message in iterator:
        if message:
            print(message)
        else:
            await asyncio.sleep(0.3)
```
压测+同时监听
```python 
async def main():
    await asyncio.gather(
        run_batch(),
        listen_batch(),
    )


if __name__ == "__main__":
    asyncio.run(main())
```

# 压测时使用ui界面监控

0.5.0版本后,添加了ui页面,支持批量压测方法
<br/>导入
```python
from atomic_bomb_engine import server
```
使用
```python
import asyncio

import atomic_bomb_engine
from atomic_bomb_engine import server


@server.ui(port=8000)
async def run_batch():
  result = await atomic_bomb_engine.batch_async(
    # 测试持续时间
    test_duration_secs=60,
    # 并发量
    concurrent_requests=200,
    # 阶梯设置(每5秒增加30个并发)
    step_option=atomic_bomb_engine.step_option(increase_step=30, increase_interval=5),
    # 接口超时时间
    timeout_secs=10,
    # 是否开启客户端启用持久性cookie存储
    cookie_store_enable=True,
    # 全局初始化
    setup_options=[
      atomic_bomb_engine.setup_option(
        name="初始化-1",
        url="http://localhost:8080/setup",
        method="get",
        jsonpath_extract=[
          atomic_bomb_engine.jsonpath_extract_option(key="test-msg", jsonpath="$.msg"),
          atomic_bomb_engine.jsonpath_extract_option(key="test-code", jsonpath="$.code"),
        ]
      )],
    # 是否开启详细日志
    verbose=False,
    # 被压接口设置
    api_endpoints=[
      atomic_bomb_engine.endpoint(
        # 接口任务命名
        name="test-1",
        # 针对每个接口初始化
        setup_options=[
          atomic_bomb_engine.setup_option(
            name="api-初始化-1",
            url="http://localhost:8080/api_setup",
            method="get",
            jsonpath_extract=[
              atomic_bomb_engine.jsonpath_extract_option(key="api-test-msg-1", jsonpath="$.msg"),
              atomic_bomb_engine.jsonpath_extract_option(key="api-test-code-1", jsonpath="$.code"),
            ]
          )
        ],
        # 被压接口url
        url="http://localhost:8080/direct",
        # 请求方式
        method="POST",
        # 权重
        weight=1,
        # 发送json请求
        json={"name": "{{api-test-msg-1}}", "number": 1},
        # 断言选项
        assert_options=[
          atomic_bomb_engine.assert_option(jsonpath="$.number", reference_object=1),
        ],
        # 思考时间选项(在最大和最小之间随机,单位毫秒)
        think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),
      ),
    ])
  print(result)
  return result


if __name__ == '__main__':
    asyncio.run(run_batch())
```

使用server.ui装饰器,可以给批量压测方法启动一个简单的web服务器,不需要再手动监听BatchListenIter生成器

## 内部架构图
![architecture.png](img/architecture.png)

## [0.19.0] - 2024-04-16
### Added
- 增加了初始化和参数模版功能
```python
setup_options=[
  atomic_bomb_engine.setup_option(
    name="初始化-1",
    url="http://localhost:8080/setup",
    method="get",
    jsonpath_extract=[
      atomic_bomb_engine.jsonpath_extract_option(key="test-msg", jsonpath="$.msg"),
      atomic_bomb_engine.jsonpath_extract_option(key="test-code", jsonpath="$.code"),
    ]
  )]
```
上述实例展示了如何在初始化的时候调用某个接口,并且通过jsonpath将数据提取出来,保存在全局变量test-msg和test-code中
提取完全局变量后,就可以在后续的api_endpoints中使用
```python
api_endpoints=[
  atomic_bomb_engine.endpoint(
    # 接口任务命名
    name="test-1",
    # 针对每个接口初始化
    setup_options=[
      atomic_bomb_engine.setup_option(
        name="api-初始化-1",
        url="http://localhost:8080/api_setup",
        method="get",
        jsonpath_extract=[
          atomic_bomb_engine.jsonpath_extract_option(key="api-test-msg-1", jsonpath="$.msg"),
          atomic_bomb_engine.jsonpath_extract_option(key="api-test-code-1", jsonpath="$.code"),
        ]
      )
    ],
    # 被压接口url
    url="http://localhost:8080/direct",
    # 请求方式
    method="POST",
    # 权重
    weight=1,
    # 发送json请求
    json={"name": "{{api-test-msg-1}}", "number": 1},
    # 断言选项
    assert_options=[
      atomic_bomb_engine.assert_option(jsonpath="$.number", reference_object=1),
    ],
    # 思考时间选项(在最大和最小之间随机,单位毫秒)
    think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),
  ),
]
```
上述实例展示了如何在请求中使用全局变量,使用双大括号即可使用

### Fixed
- 修复了如果http状态码错误时,不会记录
- 修复了json反序列化的问题

## [0.20.0] - 2024-04-17
### Added
断言更改为异步生产消费,提升性能

## bug和需求
- 如果发现了bug,把复现步骤一起写到Issus中哈
- 如果有需求也可以在Issues中讨论
- 本程序是本人业余时间开发,不太准备保证时效性,但是如果有时间,一定第一时间回复和修改bug

## [0.22.0] - 2024-04-18
### Added
前端进行了性能优化

## [0.24.0] - 2024-04-22
### Added
异步断言使用了补偿消息,保证消息的一致性

## [0.25.0] - 2024-04-23
### Added
在endpoints中增加思考时间,模拟用户行为
```python
think_time_option(min_millis=200, max_millis=300)
```
  - min_millis:最小思考时间(毫秒)
  - max_millis:最大思考时间(毫秒)

使用时在endpoint中增加think_time_option参数

```python
api_endpoints=[
  atomic_bomb_engine.endpoint(
    name="test-1",
    url="http://localhost:8080/a",
    method="POST",
    weight=1,
    timeout_secs=10,
    json={"name": "{{test-msg}}", "number": "{{test-code}}"},
    think_time_option=atomic_bomb_engine.think_time_option(min_millis=200, max_millis=300),
  ),
]
```

## [0.26.0] - 2024-04-24
### Added
- 增加endpoint中的setup,在并发中可以做接口断言
- 增加有关联条件下的cookie自动管理功能
```python
atomic_bomb_engine.endpoint(
  # 接口任务命名
  name="test-1",
  # 针对每个接口初始化
  setup_options=[
    atomic_bomb_engine.setup_option(
      name="api-初始化-1",
      url="http://localhost:8080/api_setup",
      method="get",
      jsonpath_extract=[
        atomic_bomb_engine.jsonpath_extract_option(key="api-test-msg-1", jsonpath="$.msg"),
        atomic_bomb_engine.jsonpath_extract_option(key="api-test-code-1", jsonpath="$.code"),
      ]
    )
  ],
  # 被压接口url
  url="http://localhost:8080/direct",
  # 请求方式
  method="POST",
  # 权重
  weight=1,
  # 发送json请求
  json={"name": "{{api-test-msg-1}}", "number": 1},
  # 断言选项
  assert_options=[
    atomic_bomb_engine.assert_option(jsonpath="$.number", reference_object=1),
  ],
  # 思考时间选项(在最大和最小之间随机,单位毫秒)
  think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),
)
```
- 参数cookie_store_enable控制是否自动管理cookie,前置条件的cookie会带入到最终的压测接口中
- 在endpoint中使用setup_options可以传入多个接口,并且提取参数
- 提取到的参数如果和全局的setup的key冲突,会覆盖全局提取到的参数
- 接口中提取的参数只能在本线程(v-user)中使用
- ⚠️ 使用时注意:setup_options是顺序执行的,没有并发,但是相当于添加了think time

## [0.28.0] - 2024-04-25
### Added
- 将持久化cookie添加到全局选项中
- 复用http client
- 选择性开启断言任务
- 接口初始化时出现错误等待后重试##

## [0.29.0] - 2024-04-25
### Added
- 优化并发逻辑
- 前端更改为web worker发送心跳

## [0.38.0] - 2024-05-7
### Added
- 增加附件上传功能
  - 在初始化和每个接口中增加了multipart_options参数用于附件上传
  - 增加multipart_option方法封装附件参数
    - form_key: form表单的key
    - path: 附件路径
    - file_name: 附件名
    - mime: 附件类型 (类型可以参考[这里](https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Basics_of_HTTP/MIME_types/Common_types))
```python
api_endpoints=[
            atomic_bomb_engine.endpoint(
                name="test-file",
                url="http://127.0.0.1:8888/upload",
                method="post",
                weight=100,
                multipart_options=[atomic_bomb_engine.multipart_option(form_key="file", path="./ui.py", file_name="ui.py", mime="text/plain")],
                assert_options=[
                    atomic_bomb_engine.assert_option(jsonpath="$.message", reference_object="File uploaded successfully!"),
                ],
                think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),
            ),]
```

## bug和需求
- 如果发现了bug,把复现步骤一起写到Issus中哈
- 如果有需求也可以在Issues中讨论
- 本程序是本人业余时间开发,不太准备保证时效性,但是如果有时间,一定第一时间回复和修改bug

## TODO
- [x] 前端展示页面 ✅
- [x] 接口关联 ✅
- [x] 每个接口可以配置思考时间 ✅
- [x] 增加form支持 ✅
- [ ] 增加代理支持
- [x] 增加附件支持 ✅
- [ ] 断言支持不等于等更多表达方式

## 联系方式
- 邮箱:[qyzhg@qyzhg.com](mailto:qyzhg@qyzhg.com)
- 微信:qy-zhg

## 👏🏻👏🏻👏🏻欢迎加群交流
![img.png](img/img.png)


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "atomic-bomb-engine",
    "maintainer": null,
    "docs_url": null,
    "requires_python": null,
    "maintainer_email": null,
    "keywords": "rust, python, binding",
    "author": null,
    "author_email": "qyzhg <qyzhg@qyzhg.com>",
    "download_url": null,
    "platform": null,
    "description": "# atomic-bomb-engine-py\n#### [atomic-bomb-engine](https://github.com/we-lsp/atomic-bomb-engine)\u7684python\u5305\u88c5\u5b9e\u73b0\n\n<img src=\"img/atomic-bomb-engine-logo.png\" width=\"350px\" height=\"350px\" alt=\"logo\">\n\n\n## \u524d\u7aef\u4ed3\u5e93\n#### [atomic-bomb-engine-front](https://github.com/GiantAxeWhy/atomic-bomb-engine-front)\n\n## \u4f7f\u7528\u6761\u4ef6\uff1a\n- python\u7248\u672c >= 3.8\n- windows(x86), linux(x86), mac\n\n## \u4f7f\u7528\u65b9\u6cd5\uff1a\n- ### \u51c6\u5907\u5f00\u59cb\n\u901a\u8fc7pip\u5b89\u88c5 \uff080.5.0\u7248\u672c\u4e4b\u524d\uff09\n```shell\npip install atomic-bomb-engine-py\n```\n\u5728python\u4e2d\u5f15\u7528\u65f6\u6ce8\u610f\uff0c\u9700\u8981\u5f15\u7528atomic_bomb_engine, \u800c\u4e0d\u662fatomic_bomb_engine_py\n<br/> \u4e3a\u4e86\u907f\u514d\u6df7\u6dc6\uff0c0.5.0\u7248\u672c\u4e4b\u540e\uff0cpip\u66f4\u6362\u4e86\u5305\u540d\uff0c\u66f4\u6539\u4e3aatomic-bomb-engine\uff0c\n```shell\npip install atomic-bomb-engine\n```\n\u5728python\u4e2d\u5bfc\u5165\n```python\nimport atomic_bomb_engine\n```\n\u5f02\u6b65\u4f7f\u7528\u7684\u65f6\u5019\uff0c\u8fd8\u9700\u8981\u5f15\u7528asyncio\n```python\nimport asyncio\n```\n- ### \u5f00\u59cb\u538b\u6d4b\n  - ~~\u5355\u63a5\u53e3\u538b\u6d4b~~ \uff08\u529f\u80fd\u4e0e\u591a\u63a5\u53e3\u538b\u6d4b\u91cd\u53e0\uff0c\u5df2\u5e9f\u9664\uff09\n\n  - \u591a\u63a5\u53e3\u538b\u6d4b\n\n\u591a\u63a5\u53e3\u538b\u6d4b\u53ef\u4ee5\u4f7f\u7528batch_async\u65b9\u6cd5\u8fdb\u884c\u64cd\u4f5c\uff0c\u51fd\u6570\u7b7e\u540d\u548c\u89e3\u91ca\u5982\u4e0b\n ```python\nasync def batch_async(\n        test_duration_secs: int,\n        concurrent_requests: int,\n        api_endpoints:List[Dict],\n        step_option:Dict[str, int]|None=None,\n        setup_options:List[Dict[str, Any]]|None=None,\n        verbose:bool=False,\n        should_prevent:bool=False,\n        assert_channel_buffer_size:int=1024,\n        timeout_secs=0,\n        cookie_store_enable=True\n) ->Dict:\n  \"\"\"\n      \u6279\u91cf\u538b\u6d4b\n      :param test_duration_secs: \u6d4b\u8bd5\u6301\u7eed\u65f6\u95f4\n      :param concurrent_requests: \u5e76\u53d1\u6570\n      :param api_endpoints: \u63a5\u53e3\u4fe1\u606f\n      :param step_option: \u9636\u68af\u52a0\u538b\u9009\u9879\n      :param setup_options: \u521d\u59cb\u5316\u9009\u9879\n      :param verbose: \u6253\u5370\u8be6\u7ec6\u4fe1\u606f\n      :param should_prevent: \u662f\u5426\u7981\u7528\u7761\u7720\n      :param assert_channel_buffer_size: \u65ad\u8a00\u961f\u5217buffer\u5927\u5c0f\n      :param timeout_secs: http\u8d85\u65f6\u65f6\u95f4\n      :param cookie_store_enable: \u662f\u5426\u4e3a\u5ba2\u6237\u7aef\u542f\u7528\u6301\u4e45\u6027cookie\u5b58\u50a8\u3002\n  \"\"\"\n ```\n\n\u4f7f\u7528assert_option\u65b9\u6cd5\u53ef\u4ee5\u8fd4\u56de\u65ad\u8a00\u9009\u9879\u5b57\u5178\n```python\nassert_options=[\natomic_bomb_engine.assert_option(\"$.code\", 429),\natomic_bomb_engine.assert_option(\"$.code\", 200)\n])\n```\njsonpath\u5982\u679c\u4e0d\u4f1a\u7528\u7684\u8bdd\uff0c\u5efa\u8bae\u53bb[jsonpath](https://jsonpath.com/)\u5b66\u4e60\n\n\u4f7f\u7528step_option\u65b9\u6cd5\u53ef\u4ee5\u8fd4\u56de\u9636\u68af\u52a0\u538b\u9009\u9879\u5b57\u5178\n```python\ndef step_option(increase_step: int, increase_interval: int) -> Dict[str, int]:\n    \"\"\"\n    \u751f\u6210step option\n    :param increase_step: \u9636\u68af\u6b65\u957f\n    :param increase_interval: \u9636\u68af\u95f4\u9694\n    \"\"\"\n```\n\n\u540c\u6837\u7684\u672c\u5305\u4e2d\u4e5f\u5305\u542b\u4e86\u4e00\u4e2a\u5bf9api_endpoint\u7684\u5305\u88c5\uff1aendpoint\u65b9\u6cd5\uff0c\u65b9\u4fbf\u8c03\u7528\uff0cendpoint\u4e2d\u7684assert_options\u4e2d\u4e5f\u53ef\u4ee5\u5957\u7528assert_option\u65b9\u6cd5\n ```python\nasync def run_batch():\n  result = await atomic_bomb_engine.batch_async(\n    # \u6d4b\u8bd5\u6301\u7eed\u65f6\u95f4\n    test_duration_secs=60,\n    # \u5e76\u53d1\u91cf\n    concurrent_requests=200,\n    # \u9636\u68af\u8bbe\u7f6e\uff08\u6bcf5\u79d2\u589e\u52a030\u4e2a\u5e76\u53d1\uff09\n    step_option=atomic_bomb_engine.step_option(increase_step=30, increase_interval=5),\n    # \u63a5\u53e3\u8d85\u65f6\u65f6\u95f4\n    timeout_secs=10,\n    # \u662f\u5426\u5f00\u542f\u5ba2\u6237\u7aef\u542f\u7528\u6301\u4e45\u6027cookie\u5b58\u50a8\n    cookie_store_enable=True,\n    # \u5168\u5c40\u521d\u59cb\u5316\n    setup_options=[\n      atomic_bomb_engine.setup_option(\n        name=\"\u521d\u59cb\u5316-1\",\n        url=\"http://localhost:8080/setup\",\n        method=\"get\",\n        jsonpath_extract=[\n          atomic_bomb_engine.jsonpath_extract_option(key=\"test-msg\", jsonpath=\"$.msg\"),\n          atomic_bomb_engine.jsonpath_extract_option(key=\"test-code\", jsonpath=\"$.code\"),\n        ]\n      )],\n    # \u662f\u5426\u5f00\u542f\u8be6\u7ec6\u65e5\u5fd7\n    verbose=False,\n    # \u88ab\u538b\u63a5\u53e3\u8bbe\u7f6e\n    api_endpoints=[\n      atomic_bomb_engine.endpoint(\n        # \u63a5\u53e3\u4efb\u52a1\u547d\u540d\n        name=\"test-1\",\n        # \u9488\u5bf9\u6bcf\u4e2a\u63a5\u53e3\u521d\u59cb\u5316\n        setup_options=[\n          atomic_bomb_engine.setup_option(\n            name=\"api-\u521d\u59cb\u5316-1\",\n            url=\"http://localhost:8080/api_setup\",\n            method=\"get\",\n            jsonpath_extract=[\n              atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-msg-1\", jsonpath=\"$.msg\"),\n              atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-code-1\", jsonpath=\"$.code\"),\n            ]\n          )\n        ],\n        # \u88ab\u538b\u63a5\u53e3url\n        url=\"http://localhost:8080/direct\",\n        # \u8bf7\u6c42\u65b9\u5f0f\n        method=\"POST\",\n        # \u6743\u91cd\n        weight=1,\n        # \u53d1\u9001json\u8bf7\u6c42\n        json={\"name\": \"{{api-test-msg-1}}\", \"number\": 1},\n        # \u65ad\u8a00\u9009\u9879\n        assert_options=[\n          atomic_bomb_engine.assert_option(jsonpath=\"$.number\", reference_object=1),\n        ],\n        # \u601d\u8003\u65f6\u95f4\u9009\u9879\uff08\u5728\u6700\u5927\u548c\u6700\u5c0f\u4e4b\u95f4\u968f\u673a\uff0c\u5355\u4f4d\u6beb\u79d2\uff09\n        think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),\n      ),\n    ])\n  print(result)\n  return result\n ```\n    \n\u76d1\u542c\u65f6\u53ef\u4ee5\u4f7f\u7528BatchListenIter\u751f\u6210\u5668\n```python\nasync def listen_batch():\n    iterator = atomic_bomb_engine.BatchListenIter()\n    for message in iterator:\n        if message:\n            print(message)\n        else:\n            await asyncio.sleep(0.3)\n```\n\u538b\u6d4b+\u540c\u65f6\u76d1\u542c\n```python \nasync def main():\n    await asyncio.gather(\n        run_batch(),\n        listen_batch(),\n    )\n\n\nif __name__ == \"__main__\":\n    asyncio.run(main())\n```\n\n# \u538b\u6d4b\u65f6\u4f7f\u7528ui\u754c\u9762\u76d1\u63a7\n\n0.5.0\u7248\u672c\u540e\uff0c\u6dfb\u52a0\u4e86ui\u9875\u9762\uff0c\u652f\u6301\u6279\u91cf\u538b\u6d4b\u65b9\u6cd5\n<br/>\u5bfc\u5165\n```python\nfrom atomic_bomb_engine import server\n```\n\u4f7f\u7528\n```python\nimport asyncio\n\nimport atomic_bomb_engine\nfrom atomic_bomb_engine import server\n\n\n@server.ui(port=8000)\nasync def run_batch():\n  result = await atomic_bomb_engine.batch_async(\n    # \u6d4b\u8bd5\u6301\u7eed\u65f6\u95f4\n    test_duration_secs=60,\n    # \u5e76\u53d1\u91cf\n    concurrent_requests=200,\n    # \u9636\u68af\u8bbe\u7f6e\uff08\u6bcf5\u79d2\u589e\u52a030\u4e2a\u5e76\u53d1\uff09\n    step_option=atomic_bomb_engine.step_option(increase_step=30, increase_interval=5),\n    # \u63a5\u53e3\u8d85\u65f6\u65f6\u95f4\n    timeout_secs=10,\n    # \u662f\u5426\u5f00\u542f\u5ba2\u6237\u7aef\u542f\u7528\u6301\u4e45\u6027cookie\u5b58\u50a8\n    cookie_store_enable=True,\n    # \u5168\u5c40\u521d\u59cb\u5316\n    setup_options=[\n      atomic_bomb_engine.setup_option(\n        name=\"\u521d\u59cb\u5316-1\",\n        url=\"http://localhost:8080/setup\",\n        method=\"get\",\n        jsonpath_extract=[\n          atomic_bomb_engine.jsonpath_extract_option(key=\"test-msg\", jsonpath=\"$.msg\"),\n          atomic_bomb_engine.jsonpath_extract_option(key=\"test-code\", jsonpath=\"$.code\"),\n        ]\n      )],\n    # \u662f\u5426\u5f00\u542f\u8be6\u7ec6\u65e5\u5fd7\n    verbose=False,\n    # \u88ab\u538b\u63a5\u53e3\u8bbe\u7f6e\n    api_endpoints=[\n      atomic_bomb_engine.endpoint(\n        # \u63a5\u53e3\u4efb\u52a1\u547d\u540d\n        name=\"test-1\",\n        # \u9488\u5bf9\u6bcf\u4e2a\u63a5\u53e3\u521d\u59cb\u5316\n        setup_options=[\n          atomic_bomb_engine.setup_option(\n            name=\"api-\u521d\u59cb\u5316-1\",\n            url=\"http://localhost:8080/api_setup\",\n            method=\"get\",\n            jsonpath_extract=[\n              atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-msg-1\", jsonpath=\"$.msg\"),\n              atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-code-1\", jsonpath=\"$.code\"),\n            ]\n          )\n        ],\n        # \u88ab\u538b\u63a5\u53e3url\n        url=\"http://localhost:8080/direct\",\n        # \u8bf7\u6c42\u65b9\u5f0f\n        method=\"POST\",\n        # \u6743\u91cd\n        weight=1,\n        # \u53d1\u9001json\u8bf7\u6c42\n        json={\"name\": \"{{api-test-msg-1}}\", \"number\": 1},\n        # \u65ad\u8a00\u9009\u9879\n        assert_options=[\n          atomic_bomb_engine.assert_option(jsonpath=\"$.number\", reference_object=1),\n        ],\n        # \u601d\u8003\u65f6\u95f4\u9009\u9879\uff08\u5728\u6700\u5927\u548c\u6700\u5c0f\u4e4b\u95f4\u968f\u673a\uff0c\u5355\u4f4d\u6beb\u79d2\uff09\n        think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),\n      ),\n    ])\n  print(result)\n  return result\n\n\nif __name__ == '__main__':\n    asyncio.run(run_batch())\n```\n\n\u4f7f\u7528server.ui\u88c5\u9970\u5668\uff0c\u53ef\u4ee5\u7ed9\u6279\u91cf\u538b\u6d4b\u65b9\u6cd5\u542f\u52a8\u4e00\u4e2a\u7b80\u5355\u7684web\u670d\u52a1\u5668\uff0c\u4e0d\u9700\u8981\u518d\u624b\u52a8\u76d1\u542cBatchListenIter\u751f\u6210\u5668\n\n## \u5185\u90e8\u67b6\u6784\u56fe\n![architecture.png](img/architecture.png)\n\n## [0.19.0] - 2024-04-16\n### Added\n- \u589e\u52a0\u4e86\u521d\u59cb\u5316\u548c\u53c2\u6570\u6a21\u7248\u529f\u80fd\n```python\nsetup_options=[\n  atomic_bomb_engine.setup_option(\n    name=\"\u521d\u59cb\u5316-1\",\n    url=\"http://localhost:8080/setup\",\n    method=\"get\",\n    jsonpath_extract=[\n      atomic_bomb_engine.jsonpath_extract_option(key=\"test-msg\", jsonpath=\"$.msg\"),\n      atomic_bomb_engine.jsonpath_extract_option(key=\"test-code\", jsonpath=\"$.code\"),\n    ]\n  )]\n```\n\u4e0a\u8ff0\u5b9e\u4f8b\u5c55\u793a\u4e86\u5982\u4f55\u5728\u521d\u59cb\u5316\u7684\u65f6\u5019\u8c03\u7528\u67d0\u4e2a\u63a5\u53e3\uff0c\u5e76\u4e14\u901a\u8fc7jsonpath\u5c06\u6570\u636e\u63d0\u53d6\u51fa\u6765\uff0c\u4fdd\u5b58\u5728\u5168\u5c40\u53d8\u91cftest-msg\u548ctest-code\u4e2d\n\u63d0\u53d6\u5b8c\u5168\u5c40\u53d8\u91cf\u540e\uff0c\u5c31\u53ef\u4ee5\u5728\u540e\u7eed\u7684api_endpoints\u4e2d\u4f7f\u7528\n```python\napi_endpoints=[\n  atomic_bomb_engine.endpoint(\n    # \u63a5\u53e3\u4efb\u52a1\u547d\u540d\n    name=\"test-1\",\n    # \u9488\u5bf9\u6bcf\u4e2a\u63a5\u53e3\u521d\u59cb\u5316\n    setup_options=[\n      atomic_bomb_engine.setup_option(\n        name=\"api-\u521d\u59cb\u5316-1\",\n        url=\"http://localhost:8080/api_setup\",\n        method=\"get\",\n        jsonpath_extract=[\n          atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-msg-1\", jsonpath=\"$.msg\"),\n          atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-code-1\", jsonpath=\"$.code\"),\n        ]\n      )\n    ],\n    # \u88ab\u538b\u63a5\u53e3url\n    url=\"http://localhost:8080/direct\",\n    # \u8bf7\u6c42\u65b9\u5f0f\n    method=\"POST\",\n    # \u6743\u91cd\n    weight=1,\n    # \u53d1\u9001json\u8bf7\u6c42\n    json={\"name\": \"{{api-test-msg-1}}\", \"number\": 1},\n    # \u65ad\u8a00\u9009\u9879\n    assert_options=[\n      atomic_bomb_engine.assert_option(jsonpath=\"$.number\", reference_object=1),\n    ],\n    # \u601d\u8003\u65f6\u95f4\u9009\u9879\uff08\u5728\u6700\u5927\u548c\u6700\u5c0f\u4e4b\u95f4\u968f\u673a\uff0c\u5355\u4f4d\u6beb\u79d2\uff09\n    think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),\n  ),\n]\n```\n\u4e0a\u8ff0\u5b9e\u4f8b\u5c55\u793a\u4e86\u5982\u4f55\u5728\u8bf7\u6c42\u4e2d\u4f7f\u7528\u5168\u5c40\u53d8\u91cf\uff0c\u4f7f\u7528\u53cc\u5927\u62ec\u53f7\u5373\u53ef\u4f7f\u7528\n\n### Fixed\n- \u4fee\u590d\u4e86\u5982\u679chttp\u72b6\u6001\u7801\u9519\u8bef\u65f6\uff0c\u4e0d\u4f1a\u8bb0\u5f55\n- \u4fee\u590d\u4e86json\u53cd\u5e8f\u5217\u5316\u7684\u95ee\u9898\n\n## [0.20.0] - 2024-04-17\n### Added\n\u65ad\u8a00\u66f4\u6539\u4e3a\u5f02\u6b65\u751f\u4ea7\u6d88\u8d39\uff0c\u63d0\u5347\u6027\u80fd\n\n## bug\u548c\u9700\u6c42\n- \u5982\u679c\u53d1\u73b0\u4e86bug\uff0c\u628a\u590d\u73b0\u6b65\u9aa4\u4e00\u8d77\u5199\u5230Issus\u4e2d\u54c8\n- \u5982\u679c\u6709\u9700\u6c42\u4e5f\u53ef\u4ee5\u5728Issues\u4e2d\u8ba8\u8bba\n- \u672c\u7a0b\u5e8f\u662f\u672c\u4eba\u4e1a\u4f59\u65f6\u95f4\u5f00\u53d1\uff0c\u4e0d\u592a\u51c6\u5907\u4fdd\u8bc1\u65f6\u6548\u6027\uff0c\u4f46\u662f\u5982\u679c\u6709\u65f6\u95f4\uff0c\u4e00\u5b9a\u7b2c\u4e00\u65f6\u95f4\u56de\u590d\u548c\u4fee\u6539bug\n\n## [0.22.0] - 2024-04-18\n### Added\n\u524d\u7aef\u8fdb\u884c\u4e86\u6027\u80fd\u4f18\u5316\n\n## [0.24.0] - 2024-04-22\n### Added\n\u5f02\u6b65\u65ad\u8a00\u4f7f\u7528\u4e86\u8865\u507f\u6d88\u606f\uff0c\u4fdd\u8bc1\u6d88\u606f\u7684\u4e00\u81f4\u6027\n\n## [0.25.0] - 2024-04-23\n### Added\n\u5728endpoints\u4e2d\u589e\u52a0\u601d\u8003\u65f6\u95f4,\u6a21\u62df\u7528\u6237\u884c\u4e3a\n```python\nthink_time_option(min_millis=200, max_millis=300)\n```\n  - min_millis:\u6700\u5c0f\u601d\u8003\u65f6\u95f4(\u6beb\u79d2)\n  - max_millis:\u6700\u5927\u601d\u8003\u65f6\u95f4(\u6beb\u79d2)\n\n\u4f7f\u7528\u65f6\u5728endpoint\u4e2d\u589e\u52a0think_time_option\u53c2\u6570\n\n```python\napi_endpoints=[\n  atomic_bomb_engine.endpoint(\n    name=\"test-1\",\n    url=\"http://localhost:8080/a\",\n    method=\"POST\",\n    weight=1,\n    timeout_secs=10,\n    json={\"name\": \"{{test-msg}}\", \"number\": \"{{test-code}}\"},\n    think_time_option=atomic_bomb_engine.think_time_option(min_millis=200, max_millis=300),\n  ),\n]\n```\n\n## [0.26.0] - 2024-04-24\n### Added\n- \u589e\u52a0endpoint\u4e2d\u7684setup\uff0c\u5728\u5e76\u53d1\u4e2d\u53ef\u4ee5\u505a\u63a5\u53e3\u65ad\u8a00\n- \u589e\u52a0\u6709\u5173\u8054\u6761\u4ef6\u4e0b\u7684cookie\u81ea\u52a8\u7ba1\u7406\u529f\u80fd\n```python\natomic_bomb_engine.endpoint(\n  # \u63a5\u53e3\u4efb\u52a1\u547d\u540d\n  name=\"test-1\",\n  # \u9488\u5bf9\u6bcf\u4e2a\u63a5\u53e3\u521d\u59cb\u5316\n  setup_options=[\n    atomic_bomb_engine.setup_option(\n      name=\"api-\u521d\u59cb\u5316-1\",\n      url=\"http://localhost:8080/api_setup\",\n      method=\"get\",\n      jsonpath_extract=[\n        atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-msg-1\", jsonpath=\"$.msg\"),\n        atomic_bomb_engine.jsonpath_extract_option(key=\"api-test-code-1\", jsonpath=\"$.code\"),\n      ]\n    )\n  ],\n  # \u88ab\u538b\u63a5\u53e3url\n  url=\"http://localhost:8080/direct\",\n  # \u8bf7\u6c42\u65b9\u5f0f\n  method=\"POST\",\n  # \u6743\u91cd\n  weight=1,\n  # \u53d1\u9001json\u8bf7\u6c42\n  json={\"name\": \"{{api-test-msg-1}}\", \"number\": 1},\n  # \u65ad\u8a00\u9009\u9879\n  assert_options=[\n    atomic_bomb_engine.assert_option(jsonpath=\"$.number\", reference_object=1),\n  ],\n  # \u601d\u8003\u65f6\u95f4\u9009\u9879\uff08\u5728\u6700\u5927\u548c\u6700\u5c0f\u4e4b\u95f4\u968f\u673a\uff0c\u5355\u4f4d\u6beb\u79d2\uff09\n  think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),\n)\n```\n- \u53c2\u6570cookie_store_enable\u63a7\u5236\u662f\u5426\u81ea\u52a8\u7ba1\u7406cookie\uff0c\u524d\u7f6e\u6761\u4ef6\u7684cookie\u4f1a\u5e26\u5165\u5230\u6700\u7ec8\u7684\u538b\u6d4b\u63a5\u53e3\u4e2d\n- \u5728endpoint\u4e2d\u4f7f\u7528setup_options\u53ef\u4ee5\u4f20\u5165\u591a\u4e2a\u63a5\u53e3\uff0c\u5e76\u4e14\u63d0\u53d6\u53c2\u6570\n- \u63d0\u53d6\u5230\u7684\u53c2\u6570\u5982\u679c\u548c\u5168\u5c40\u7684setup\u7684key\u51b2\u7a81\uff0c\u4f1a\u8986\u76d6\u5168\u5c40\u63d0\u53d6\u5230\u7684\u53c2\u6570\n- \u63a5\u53e3\u4e2d\u63d0\u53d6\u7684\u53c2\u6570\u53ea\u80fd\u5728\u672c\u7ebf\u7a0b\uff08v-user\uff09\u4e2d\u4f7f\u7528\n- \u26a0\ufe0f \u4f7f\u7528\u65f6\u6ce8\u610f:setup_options\u662f\u987a\u5e8f\u6267\u884c\u7684\uff0c\u6ca1\u6709\u5e76\u53d1\uff0c\u4f46\u662f\u76f8\u5f53\u4e8e\u6dfb\u52a0\u4e86think time\n\n## [0.28.0] - 2024-04-25\n### Added\n- \u5c06\u6301\u4e45\u5316cookie\u6dfb\u52a0\u5230\u5168\u5c40\u9009\u9879\u4e2d\n- \u590d\u7528http client\n- \u9009\u62e9\u6027\u5f00\u542f\u65ad\u8a00\u4efb\u52a1\n- \u63a5\u53e3\u521d\u59cb\u5316\u65f6\u51fa\u73b0\u9519\u8bef\u7b49\u5f85\u540e\u91cd\u8bd5##\n\n## [0.29.0] - 2024-04-25\n### Added\n- \u4f18\u5316\u5e76\u53d1\u903b\u8f91\n- \u524d\u7aef\u66f4\u6539\u4e3aweb worker\u53d1\u9001\u5fc3\u8df3\n\n## [0.38.0] - 2024-05-7\n### Added\n- \u589e\u52a0\u9644\u4ef6\u4e0a\u4f20\u529f\u80fd\n  - \u5728\u521d\u59cb\u5316\u548c\u6bcf\u4e2a\u63a5\u53e3\u4e2d\u589e\u52a0\u4e86multipart_options\u53c2\u6570\u7528\u4e8e\u9644\u4ef6\u4e0a\u4f20\n  - \u589e\u52a0multipart_option\u65b9\u6cd5\u5c01\u88c5\u9644\u4ef6\u53c2\u6570\n    - form_key: form\u8868\u5355\u7684key\n    - path: \u9644\u4ef6\u8def\u5f84\n    - file_name: \u9644\u4ef6\u540d\n    - mime: \u9644\u4ef6\u7c7b\u578b (\u7c7b\u578b\u53ef\u4ee5\u53c2\u8003[\u8fd9\u91cc](https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Basics_of_HTTP/MIME_types/Common_types))\n```python\napi_endpoints=[\n            atomic_bomb_engine.endpoint(\n                name=\"test-file\",\n                url=\"http://127.0.0.1:8888/upload\",\n                method=\"post\",\n                weight=100,\n                multipart_options=[atomic_bomb_engine.multipart_option(form_key=\"file\", path=\"./ui.py\", file_name=\"ui.py\", mime=\"text/plain\")],\n                assert_options=[\n                    atomic_bomb_engine.assert_option(jsonpath=\"$.message\", reference_object=\"File uploaded successfully!\"),\n                ],\n                think_time_option=atomic_bomb_engine.think_time_option(min_millis=500, max_millis=1200),\n            ),]\n```\n\n## bug\u548c\u9700\u6c42\n- \u5982\u679c\u53d1\u73b0\u4e86bug\uff0c\u628a\u590d\u73b0\u6b65\u9aa4\u4e00\u8d77\u5199\u5230Issus\u4e2d\u54c8\n- \u5982\u679c\u6709\u9700\u6c42\u4e5f\u53ef\u4ee5\u5728Issues\u4e2d\u8ba8\u8bba\n- \u672c\u7a0b\u5e8f\u662f\u672c\u4eba\u4e1a\u4f59\u65f6\u95f4\u5f00\u53d1\uff0c\u4e0d\u592a\u51c6\u5907\u4fdd\u8bc1\u65f6\u6548\u6027\uff0c\u4f46\u662f\u5982\u679c\u6709\u65f6\u95f4\uff0c\u4e00\u5b9a\u7b2c\u4e00\u65f6\u95f4\u56de\u590d\u548c\u4fee\u6539bug\n\n## TODO\n- [x] \u524d\u7aef\u5c55\u793a\u9875\u9762 \u2705\n- [x] \u63a5\u53e3\u5173\u8054 \u2705\n- [x] \u6bcf\u4e2a\u63a5\u53e3\u53ef\u4ee5\u914d\u7f6e\u601d\u8003\u65f6\u95f4 \u2705\n- [x] \u589e\u52a0form\u652f\u6301 \u2705\n- [ ] \u589e\u52a0\u4ee3\u7406\u652f\u6301\n- [x] \u589e\u52a0\u9644\u4ef6\u652f\u6301 \u2705\n- [ ] \u65ad\u8a00\u652f\u6301\u4e0d\u7b49\u4e8e\u7b49\u66f4\u591a\u8868\u8fbe\u65b9\u5f0f\n\n## \u8054\u7cfb\u65b9\u5f0f\n- \u90ae\u7bb1:[qyzhg@qyzhg.com](mailto:qyzhg@qyzhg.com)\n- \u5fae\u4fe1:qy-zhg\n\n## \ud83d\udc4f\ud83c\udffb\ud83d\udc4f\ud83c\udffb\ud83d\udc4f\ud83c\udffb\u6b22\u8fce\u52a0\u7fa4\u4ea4\u6d41\n![img.png](img/img.png)\n\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "\u4f7f\u7528rust\u5f00\u53d1\u7684\u9ad8\u6027\u80fdpython\u538b\u6d4b\u5de5\u5177",
    "version": "0.38.0",
    "project_urls": null,
    "split_keywords": [
        "rust",
        " python",
        " binding"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "ee85302065079c1dc5513dd1e7d91c3799ea704c55375acb7da57b917c2ab876",
                "md5": "c2fb80f9e6adfc6467e277c86d93be70",
                "sha256": "e6abcbb22459c0a1f7db350caf45180a90312ccc0d1f493b486e8c7dd6818b7e"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp310-cp310-macosx_10_12_x86_64.whl",
            "has_sig": false,
            "md5_digest": "c2fb80f9e6adfc6467e277c86d93be70",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 3780204,
            "upload_time": "2024-05-07T07:41:37",
            "upload_time_iso_8601": "2024-05-07T07:41:37.343392Z",
            "url": "https://files.pythonhosted.org/packages/ee/85/302065079c1dc5513dd1e7d91c3799ea704c55375acb7da57b917c2ab876/atomic_bomb_engine-0.38.0-cp310-cp310-macosx_10_12_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "e97fe2c060eb8f54430e4b789c7a6e7ca9a8fd2f134cd49a4390a3753166dcb7",
                "md5": "bd4b3252fc82576a5ebac8a890e3946e",
                "sha256": "84516f1c5e2146f63c77a508d6cab63961761a7294ced2ef68dc23136c1801f1"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp310-cp310-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "bd4b3252fc82576a5ebac8a890e3946e",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 3708795,
            "upload_time": "2024-05-07T07:37:30",
            "upload_time_iso_8601": "2024-05-07T07:37:30.592192Z",
            "url": "https://files.pythonhosted.org/packages/e9/7f/e2c060eb8f54430e4b789c7a6e7ca9a8fd2f134cd49a4390a3753166dcb7/atomic_bomb_engine-0.38.0-cp310-cp310-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "a337090f6db08d70a03bc0883837a890d0ba2e6878e957f7975fa3daf3398044",
                "md5": "510c2fc1ab981b5c9bd258d8be751cbb",
                "sha256": "3ce35b6911079dba8b7d45ebc93c110651ce1a18eaf12a0f304f4f12dd22e2fe"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp310-cp310-manylinux_2_34_x86_64.whl",
            "has_sig": false,
            "md5_digest": "510c2fc1ab981b5c9bd258d8be751cbb",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 6121343,
            "upload_time": "2024-05-07T07:35:47",
            "upload_time_iso_8601": "2024-05-07T07:35:47.217124Z",
            "url": "https://files.pythonhosted.org/packages/a3/37/090f6db08d70a03bc0883837a890d0ba2e6878e957f7975fa3daf3398044/atomic_bomb_engine-0.38.0-cp310-cp310-manylinux_2_34_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "16d2eb2cabdb8f9a097fd5740c45929dd3035c410d291cff88feb0e0d9209984",
                "md5": "d547c2327c01e804be47d37d41d8c18c",
                "sha256": "48bcf3d3c75cd0276879ca1c7b7c1823d02d591ae0e186801947c7628251067c"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp310-cp310-musllinux_1_2_x86_64.whl",
            "has_sig": false,
            "md5_digest": "d547c2327c01e804be47d37d41d8c18c",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 4504871,
            "upload_time": "2024-05-07T07:38:41",
            "upload_time_iso_8601": "2024-05-07T07:38:41.369015Z",
            "url": "https://files.pythonhosted.org/packages/16/d2/eb2cabdb8f9a097fd5740c45929dd3035c410d291cff88feb0e0d9209984/atomic_bomb_engine-0.38.0-cp310-cp310-musllinux_1_2_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "7d9c4e83f5e4b5c80e16971acc6bccb241845d69698bd1f5ab7d780c4f9488df",
                "md5": "92e552625e930574dffd61fda455ba8c",
                "sha256": "397baef5304d196c34f4712c75a6f3cad10a499e1e3374110f33d71edf42db8f"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp310-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "92e552625e930574dffd61fda455ba8c",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 3482395,
            "upload_time": "2024-05-07T07:38:34",
            "upload_time_iso_8601": "2024-05-07T07:38:34.855987Z",
            "url": "https://files.pythonhosted.org/packages/7d/9c/4e83f5e4b5c80e16971acc6bccb241845d69698bd1f5ab7d780c4f9488df/atomic_bomb_engine-0.38.0-cp310-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "78e7130e90addf19633e95d316906cbba2960d1489d662b7132ab1f9655e06e7",
                "md5": "6523f95c4d2ed2bc30448fba4e620461",
                "sha256": "68403aa9ef22f47ad2131579e0842a9cb739b760bab201ca93eb6284e50b199f"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp311-cp311-macosx_10_12_x86_64.whl",
            "has_sig": false,
            "md5_digest": "6523f95c4d2ed2bc30448fba4e620461",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 3779879,
            "upload_time": "2024-05-07T07:44:29",
            "upload_time_iso_8601": "2024-05-07T07:44:29.124427Z",
            "url": "https://files.pythonhosted.org/packages/78/e7/130e90addf19633e95d316906cbba2960d1489d662b7132ab1f9655e06e7/atomic_bomb_engine-0.38.0-cp311-cp311-macosx_10_12_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "7bc25150d6add71a0ebfdfc94ad21c61ff10a6e1bea92672f9f6fc19c5b28912",
                "md5": "f2da7b06ffbb7e0200ac39c5fd463b70",
                "sha256": "37d380795d7c72949fa1058f563353d2fcad3bda305a650575501cc68fcceeb8"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp311-cp311-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "f2da7b06ffbb7e0200ac39c5fd463b70",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 3708356,
            "upload_time": "2024-05-07T07:40:22",
            "upload_time_iso_8601": "2024-05-07T07:40:22.054629Z",
            "url": "https://files.pythonhosted.org/packages/7b/c2/5150d6add71a0ebfdfc94ad21c61ff10a6e1bea92672f9f6fc19c5b28912/atomic_bomb_engine-0.38.0-cp311-cp311-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "e608fe96aff6d7196c98eb3e3138bbe65ea5d6cd5e4f728fd1f412d90f279680",
                "md5": "3be26d81f1b275e42438ea28bc5386a0",
                "sha256": "9a4d22dbc0935358f3df96b00ce3a85264178dc2f3b9800c4cd4c0f3f57d35f8"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp311-cp311-manylinux_2_34_x86_64.whl",
            "has_sig": false,
            "md5_digest": "3be26d81f1b275e42438ea28bc5386a0",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 6120090,
            "upload_time": "2024-05-07T07:35:39",
            "upload_time_iso_8601": "2024-05-07T07:35:39.905843Z",
            "url": "https://files.pythonhosted.org/packages/e6/08/fe96aff6d7196c98eb3e3138bbe65ea5d6cd5e4f728fd1f412d90f279680/atomic_bomb_engine-0.38.0-cp311-cp311-manylinux_2_34_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "2b4e0dcb16d0cdea11c1eff26bd8efe022791dcac77aadf7504eb1c3a65a6c24",
                "md5": "5a9ede978263aae6118ebaad9867f6ec",
                "sha256": "7010d281a986f26086701335ac25efdabc175a0aa3771be6303baac0fff223ab"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp311-cp311-musllinux_1_2_x86_64.whl",
            "has_sig": false,
            "md5_digest": "5a9ede978263aae6118ebaad9867f6ec",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 4504012,
            "upload_time": "2024-05-07T07:38:50",
            "upload_time_iso_8601": "2024-05-07T07:38:50.717792Z",
            "url": "https://files.pythonhosted.org/packages/2b/4e/0dcb16d0cdea11c1eff26bd8efe022791dcac77aadf7504eb1c3a65a6c24/atomic_bomb_engine-0.38.0-cp311-cp311-musllinux_1_2_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "e7d09aeed9f5cfcbb03c492d26756905462a46ce765f6f5f0cdb2d00ed9874c6",
                "md5": "fdfeccbe6d391cc0cf5e0abd161055b3",
                "sha256": "4f1c4c1069c145567ff9e1346a9c20b3812f197eca41a717f245a14c6c9da3a2"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp311-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "fdfeccbe6d391cc0cf5e0abd161055b3",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 3481819,
            "upload_time": "2024-05-07T07:38:28",
            "upload_time_iso_8601": "2024-05-07T07:38:28.796094Z",
            "url": "https://files.pythonhosted.org/packages/e7/d0/9aeed9f5cfcbb03c492d26756905462a46ce765f6f5f0cdb2d00ed9874c6/atomic_bomb_engine-0.38.0-cp311-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "303d97bb9ac99fe15aaf2320a7bf4dc33b11ea66402a7c8bf52dd880ba1f9a16",
                "md5": "60df6cd0abee8b2f5f360bf1aca1c670",
                "sha256": "c6a394d0fda75c6bbf290c2afa3e7b1d3100b994d51c7ec9f7534c2b29befa58"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp312-cp312-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "60df6cd0abee8b2f5f360bf1aca1c670",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 3710443,
            "upload_time": "2024-05-07T07:40:31",
            "upload_time_iso_8601": "2024-05-07T07:40:31.924815Z",
            "url": "https://files.pythonhosted.org/packages/30/3d/97bb9ac99fe15aaf2320a7bf4dc33b11ea66402a7c8bf52dd880ba1f9a16/atomic_bomb_engine-0.38.0-cp312-cp312-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "02c3edf85917b92fd43006d1fd71fb4d828af842d587f254e801e951ae6594b8",
                "md5": "019a34fdb0e4d135a6cbf7a9a19505e1",
                "sha256": "f89f58d770868eaaadc506920e4b9a07c5ecf2738de2a93ddc47be0e9af801f3"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp312-cp312-manylinux_2_34_x86_64.whl",
            "has_sig": false,
            "md5_digest": "019a34fdb0e4d135a6cbf7a9a19505e1",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 6124019,
            "upload_time": "2024-05-07T07:35:42",
            "upload_time_iso_8601": "2024-05-07T07:35:42.280389Z",
            "url": "https://files.pythonhosted.org/packages/02/c3/edf85917b92fd43006d1fd71fb4d828af842d587f254e801e951ae6594b8/atomic_bomb_engine-0.38.0-cp312-cp312-manylinux_2_34_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "963836ea4903463c0619557a4dbf90a8a23bbd7c133fe7abbd09a5f43917c575",
                "md5": "3259b43823eecc584abae0bff73924c9",
                "sha256": "93c2bde4cbb7040252474aa78c583b74eb84cb1c5c10ead8ee1fc96091ade72e"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp312-cp312-musllinux_1_2_x86_64.whl",
            "has_sig": false,
            "md5_digest": "3259b43823eecc584abae0bff73924c9",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 4505188,
            "upload_time": "2024-05-07T07:39:07",
            "upload_time_iso_8601": "2024-05-07T07:39:07.550993Z",
            "url": "https://files.pythonhosted.org/packages/96/38/36ea4903463c0619557a4dbf90a8a23bbd7c133fe7abbd09a5f43917c575/atomic_bomb_engine-0.38.0-cp312-cp312-musllinux_1_2_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "1c1396283904dc477c5a866b7631cbe6d991dd7976dd751ab435555a1fe065d0",
                "md5": "c55086885ed7cd2ff5ae6362c720a066",
                "sha256": "08dcc215bcd6e54225f037a16033875988582841b6f337ec25b9816cd943b551"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp312-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "c55086885ed7cd2ff5ae6362c720a066",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 3482262,
            "upload_time": "2024-05-07T07:38:17",
            "upload_time_iso_8601": "2024-05-07T07:38:17.911466Z",
            "url": "https://files.pythonhosted.org/packages/1c/13/96283904dc477c5a866b7631cbe6d991dd7976dd751ab435555a1fe065d0/atomic_bomb_engine-0.38.0-cp312-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "0adc6116dafdd2a983c523b4f197cb219e68efc077f48d660efb7eb011a48200",
                "md5": "4ef9eae6e634f8b37bf1ed4d00e448d9",
                "sha256": "1906046cbaaa644113e60f5b681570fd3ab6a9cf3df6f1892c5a50e9538b9710"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp38-cp38-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "4ef9eae6e634f8b37bf1ed4d00e448d9",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": null,
            "size": 3709053,
            "upload_time": "2024-05-07T07:37:07",
            "upload_time_iso_8601": "2024-05-07T07:37:07.599855Z",
            "url": "https://files.pythonhosted.org/packages/0a/dc/6116dafdd2a983c523b4f197cb219e68efc077f48d660efb7eb011a48200/atomic_bomb_engine-0.38.0-cp38-cp38-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "3f4eb5bb61c3a8115004f5fb176b87aefa8871397eebb8cf16324d0450d8f7cd",
                "md5": "288d416e38bb5e644d089035967efd5e",
                "sha256": "2a0088e48da04bb99ff97b1c2e3d456419bb3c7e5461fa6ff8835cb1acac4017"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp38-cp38-manylinux_2_34_x86_64.whl",
            "has_sig": false,
            "md5_digest": "288d416e38bb5e644d089035967efd5e",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": null,
            "size": 6122007,
            "upload_time": "2024-05-07T07:35:32",
            "upload_time_iso_8601": "2024-05-07T07:35:32.773398Z",
            "url": "https://files.pythonhosted.org/packages/3f/4e/b5bb61c3a8115004f5fb176b87aefa8871397eebb8cf16324d0450d8f7cd/atomic_bomb_engine-0.38.0-cp38-cp38-manylinux_2_34_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "ebe42420022aad4d2a078ea214866a567fbbb53b5d5b635504b9ab2d9280cb57",
                "md5": "bc4daed9d36a6033da1708058dd89a7a",
                "sha256": "87b6540b903e15965600423c966feb056ecb29e1d9ba8198024e743caae2e526"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp38-cp38-musllinux_1_2_x86_64.whl",
            "has_sig": false,
            "md5_digest": "bc4daed9d36a6033da1708058dd89a7a",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": null,
            "size": 4505554,
            "upload_time": "2024-05-07T07:38:30",
            "upload_time_iso_8601": "2024-05-07T07:38:30.823301Z",
            "url": "https://files.pythonhosted.org/packages/eb/e4/2420022aad4d2a078ea214866a567fbbb53b5d5b635504b9ab2d9280cb57/atomic_bomb_engine-0.38.0-cp38-cp38-musllinux_1_2_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "1760b4d10f8c84c409174223427503fd5b240999e3f719f3551c34d3aa4f09f4",
                "md5": "c67dd9ba7daab68cc69a8bc9d82f16ce",
                "sha256": "402d92384ac3550a496b559c35cde8e6c5a988474d9e128570cbfbe75e0f7d3b"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp38-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "c67dd9ba7daab68cc69a8bc9d82f16ce",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": null,
            "size": 3482961,
            "upload_time": "2024-05-07T07:38:00",
            "upload_time_iso_8601": "2024-05-07T07:38:00.755477Z",
            "url": "https://files.pythonhosted.org/packages/17/60/b4d10f8c84c409174223427503fd5b240999e3f719f3551c34d3aa4f09f4/atomic_bomb_engine-0.38.0-cp38-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "c3dbb44a9771e98a8c064bce82dfaf6fcf4dd90891f8617a3842142c4095b03f",
                "md5": "7eea43f2801f7289fcc3716a6f383c2c",
                "sha256": "23777d16cc0ab3eaf7b68949ae75c94fce9676dcec9767b5b750a99cd641fcb8"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp39-cp39-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "7eea43f2801f7289fcc3716a6f383c2c",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": null,
            "size": 3709035,
            "upload_time": "2024-05-07T07:37:46",
            "upload_time_iso_8601": "2024-05-07T07:37:46.429215Z",
            "url": "https://files.pythonhosted.org/packages/c3/db/b44a9771e98a8c064bce82dfaf6fcf4dd90891f8617a3842142c4095b03f/atomic_bomb_engine-0.38.0-cp39-cp39-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "fa7dabf43479007a1587adce3c35a7cb0985d2742631d674526838fdbf81d528",
                "md5": "dec5d5e5247af8e768d7dd05ec643c30",
                "sha256": "8dee2b9442e3d3b93c422eeac3ece632c7ca27dc36c4c5984b5dacb53d3cfcb0"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp39-cp39-manylinux_2_34_x86_64.whl",
            "has_sig": false,
            "md5_digest": "dec5d5e5247af8e768d7dd05ec643c30",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": null,
            "size": 6121662,
            "upload_time": "2024-05-07T07:35:33",
            "upload_time_iso_8601": "2024-05-07T07:35:33.401885Z",
            "url": "https://files.pythonhosted.org/packages/fa/7d/abf43479007a1587adce3c35a7cb0985d2742631d674526838fdbf81d528/atomic_bomb_engine-0.38.0-cp39-cp39-manylinux_2_34_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "730cfaab7e545f3b94afe5bacf94d96e7954f88215963177ff7bf81ea75b87d3",
                "md5": "fd5934abd5c82e18a4a7cabbe1dc0d2d",
                "sha256": "f95ee7b6886e3d733b00aeb0b7ebc86a836e9e21ad5485db02247ae950fd8d63"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp39-cp39-musllinux_1_2_x86_64.whl",
            "has_sig": false,
            "md5_digest": "fd5934abd5c82e18a4a7cabbe1dc0d2d",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": null,
            "size": 4505329,
            "upload_time": "2024-05-07T07:38:56",
            "upload_time_iso_8601": "2024-05-07T07:38:56.748815Z",
            "url": "https://files.pythonhosted.org/packages/73/0c/faab7e545f3b94afe5bacf94d96e7954f88215963177ff7bf81ea75b87d3/atomic_bomb_engine-0.38.0-cp39-cp39-musllinux_1_2_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8b67b1490a84811271141cbcd8c7024608ee0ab0661cf757ae3baf0f0ff1f2ba",
                "md5": "6f0ad9ae96c564ec6f6a9294747ef995",
                "sha256": "e2e7aff3b0d151108e16dd34397efb0741568616df0732220b2d255e1192bdcc"
            },
            "downloads": -1,
            "filename": "atomic_bomb_engine-0.38.0-cp39-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "6f0ad9ae96c564ec6f6a9294747ef995",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": null,
            "size": 3483269,
            "upload_time": "2024-05-07T07:38:01",
            "upload_time_iso_8601": "2024-05-07T07:38:01.079706Z",
            "url": "https://files.pythonhosted.org/packages/8b/67/b1490a84811271141cbcd8c7024608ee0ab0661cf757ae3baf0f0ff1f2ba/atomic_bomb_engine-0.38.0-cp39-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-05-07 07:41:37",
    "github": false,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "lcname": "atomic-bomb-engine"
}
        
Elapsed time: 0.26596s