• 使用fastapi和pulumi搭建基于Azure云的IAC Restful API服务 — 对外发布


    图片

    图片

    前言

    在IAC(即Infrastructure As Code,基础设施即代码)领域,Terraform 是一个老牌工具,使用HCL(HashiCorp Configuration Language)语言来编写配置文件。它支持几乎所有主流的云提供商,如AWS、Azure、GCP等,通过如下几个关键步骤管理基础设施:编写配置文件、初始化环境、生成执行计划、应用计划以及查看状态。

    Pulumi 则是一个较新的IAC工具,支持使用多种常见的编程语言(如TypeScript、Python、Go、C#)来编写基础设施代码,而不仅限于声明式语法。这带来了许多灵活性和优势:

    1. 使用常用编程语言:Pulumi允许使用者使用熟悉的编程语言编写基础设施配置,这样可以使用语言本身的功能,如条件语句、循环、函数和模块化,提高代码的可读性和重用性。

    2. 更好的集成:Pulumi在与现有开发工具链(如CI/CD管道、测试框架等)集成方面表现出色,因此可以使用流行的编程生态系统中的库和工具。这对DevOps团队来说非常有吸引力。

    3. 灵活的状态管理:虽然Terraform也有状态文件用于记录资源的现实状态,Pulumi进一步简化了状态管理,允许使用不同的存储后端,包括云存储和Pulumi自己的服务。

    4. 更高的可扩展性:Pulumi支持JavaScript、TypeScript、Python、Go和C#等多种语言,这让它能够适应更广泛的需求和团队技术栈,可以更灵活地处理复杂的基础设施场景。

    本文使用Python语言, 以pulumi作为后端IAC工具,服务端采用FastAPI框架提供Restful能力,来实现基于Azure云的IAC Restful API服务

    架构设计

    图片

    架构说明:

    • 开发人员调用http接口来进行IAC操作

    • FastAPI服务端分为2个部分,一个gateway负责对外提供restful api服务,一个backend端负责解析前端数据,调用pulumi模块生成IAC代码

    • 任务信息会存入mysql数据库,起到任务追溯的作用以及作为资源cmdb的数据源

    • IAC操作完成后,会将此次操作变更同步进入资源CMDB

    Pulumi模块说明

    与经典的命令行模式执行pulumi来进行资源创建/变更不同,在本架构中使用Pulumi Automation来实现IAC能力。Pulumi Automation提供了一种编程接口,使得基础设施管理过程可以完全自动化和自定义。

    Pulumi模块支持哪些资源

    该模块涉及:

    Azure资源模块

    Resource GroupAks Cluster(ManagedCluster)Aks Nodepool(AgentPool)Availability SetContainer RegistryData FactoryDatabricks Access ConnectorDatabricks WorkspaceKey VaultLoad BalancerMysql Flexible ServerMysql Single ServerNetwork InterfaceNetwork Security GroupPrivate ZoneRecovery Service VaultRedisRole AssignmentRoute TableStorage AccountSubnetUser Assigned IdentityVirtual Machine

    目前我们的pulumi模块现已支持以上azure资源

    感兴趣的可以联系我们,提供相关代码模块

    主要特点

    1. 集成灵活性:利用Pulumi Automation,可以更轻松地将基础设施管理操作嵌入到现有的应用程序或平台中。这对于需要与其他系统进行复杂交互的场景尤为适用,例如动态响应用户请求或事件驱动的基础设施变更。

    2. 复用性和模块化:通过将Pulumi的命令行操作封装在代码中,Pulumi Automation允许我们创建更具复用性和模块化的基础设施管理代码库。这对于大型团队和复杂项目尤其有价值,因为代码更易于维护和共享。

    3. 状态和并行处理:使用Pulumi Automation,可以更细致地管理资源状态和并行操作。我们可以根据具体情况编写代码来处理不同的资源状态,以及根据依赖关系有序地创建或销毁资源,这可以显著提升执行效率和可靠性。

    4. 错误处理和重试机制:通过编写自定义的错误处理和重试逻辑,Pulumi Automation可以更健壮地应对基础设施操作中的各种可能错误。我们可以在代码中捕获异常并进行适当的重试或回滚动作,而不必依赖于外部脚本或手动操作。

    依赖模块

    需求python 3.7+, 执行pip install -r requirements.txt来安装依赖模块

    requirements.txt

    pulumi>=3.0.0,<4.0.0pulumi-azure-native>=1.0.0,<2.0.0fastapi[all]jsonify>=0.5,<0.6pydantic>=1.9.1,<1.9.2pymysqlpyparsingpulumi_azure

    实验步骤

    前置条件

    Python 

    需求python 3.7+, 执行pip install -r requirements.txt来安装依赖模块 

    Service Principal

    服务主体必须对该订阅具有完全访问权限

    pulumi环境变量

    必须配置PULUMI_CONFIG_PASSPHRASE供pulumi连接到您的stack,若您使用非本地文件作为backend,则还需配置相应的backend所需字段

    mysql数据库(可选)

    用以记录资源创建任务信息

    资源CMDB(可选)

    根据自身需要,可配置一个资源CMDB,存放云资源实时参数配置与状态,以供实际使用

    流程说明

    图片

    • FastAPI gateway将异步调用backend方法,传递前端参数给到后端方法

    • FastAPI后端将参数入库,并调用pulumi模块,生成IAC代码

    • Pulumi模块内部使用Azure API,进行IAC操作

    以下步骤为可选步骤

    • 资源信息同步至cmdb

    • FastAPI接口提供资源信息查询

    步骤一、创建fastapi应用服务端

    使用以下代码创建fastapi app(即对外restful api服务)

    app.py

    1. @auto_deploy_api.post("/create_resource")
    2. async def create_res(VERABLES_DICT: Item, backgroundTasks: BackgroundTasks):
    3. VERABLES = VERABLES_DICT.VERABLES
    4. try:
    5. # 启动异步任务,因为创建资源耗时较长,http请求不应也无法一直保持
    6. backgroundTasks.add_task(create_resource, VERABLES, id)
    7. return {"message": "Task running at backend", "id": id}
    8. except auto.StackAlreadyExistsError:
    9. return HTTPException(status_code=409,detail="application already exists")
    10. except Exception as e:
    11. return Response(status_code=500, content=repr(e))

    步骤二、集成fastapi与pulumi模块

    将pulumi模块与fastapi服务集成在同一目录下,文件架构如下:

    图片

    app.py:fastapi 应用入口文件,定义restful接口

    auto_deploy_inline.py:调用pulumi模块,执行IAC操作

    pulumi_resources/IAC_resource:存放pulumi模块

    create_resource 

    1. def create_resource(VERABLES, task_id):
    2. project_name = VERABLES["GLOBAL_VARS"]["PROJECT_NAME"]
    3. app_name = VERABLES["GLOBAL_VARS"]["APPLICATION_NAME"]
    4. env = VERABLES["GLOBAL_VARS"]["ENV"]
    5. stack_name = "%s-%s-%s-stack" % (project_name, app_name, env)
    6. try:
    7. # 此方法处理前端传入的参数,调用pulumi模块,进行资源创建,此处仅以网络资源为例
    8. def resource_init():
    9. if "RESOURCE_GROUP" not in VERABLES.keys():
    10. raise Exception("Resource group is mandatory")
    11. rg = ResourceGroup(VERABLES["RESOURCE_GROUP"]["NAME"], location)
    12. if "NETWORK" in VERABLES.keys():
    13. if "ROUTE_TABLE" in VERABLES["NETWORK"].keys():
    14. for rt_ele in VERABLES["NETWORK"]["ROUTE_TABLE"]:
    15. RouteTable(rt_ele["name"], location, None, **rt_ele)
    16. if "NETWORK_SECURITY_GROUP" in VERABLES["NETWORK"].keys():
    17. for nsg_ele in VERABLES["NETWORK"]["NETWORK_SECURITY_GROUP"]:
    18. NetworkSecurityGroup(nsg_ele["name"], location, None, **nsg_ele)
    19. resource_stack = auto.create_or_select_stack(stack_name=stack_name,
    20. project_name=app_name,
    21. program=resource_init)
    22. update_resource_creation_task_record(data_dict)
    23. print("Start preview...")
    24. resource_up_res = resource_stack.preview(on_output=print)
    25. print("Preview OK...")
    26. time.sleep(1)
    27. print("Start up...")
    28. resource_up_res = resource_stack.up(on_output=print)
    29. print("Up OK...")
    30. data_dict["status"] = "successed"
    31. update_resource_creation_task_record(data_dict)
    32. return (resource_up_res.stdout)
    33. except Exception as e:
    34. data_dict["status"] = "failed"
    35. data_dict["output"] = repr(e)
    36. update_resource_creation_task_record(data_dict)

    步骤三、启动fastapi应用

    1. cd $YOURPATH/resource_deploy/auto_deploy_fastapi
    2. uvicorn app:auto_deploy_api --host 0.0.0.0 --port 8000

    步骤四、Postman调用接口创建资源

    接下来,我们就可以通过http请求的方式,调用pulumi的IAC能力,操作Azure云的资源了

    图片

    示例payload

    payload

    1. {
    2. 'VERABLES': {
    3. 'GLOBAL_VARS': {
    4. 'PROJECT_NAME': 'auto',
    5. 'APPLICATION_NAME': 'auto',
    6. 'ENV': 'dev',
    7. 'LOCATION': 'chinanorth3',
    8. 'SUBSCRIPTION_ID': '***********'
    9. },
    10. 'RESOURCE_GROUP': {
    11. 'NAME': 'test-pulumi-rg',
    12. 'TAGS': {
    13. 'ApplicationID': 'ITPlatform',
    14. 'ApplicationName': 'ITPlatform'
    15. }
    16. },
    17. 'NETWORK': {
    18. 'ROUTE_TABLE': [{
    19. 'name': 'test-rt01',
    20. 'resource_group_name': 'test-network-rg01',
    21. 'routes': [{
    22. 'name': 'rule1',
    23. 'address_prefix': '0.0.0.0/0',
    24. 'next_hop_type': 'VirtualAppliance',
    25. 'next_hop_ip_address': '10.20.128.140'
    26. }]
    27. }],
    28. 'NETWORK_SECURITY_GROUP': [{
    29. 'name': 'test-nsg01',
    30. 'resource_group_name': 'test-network-rg01',
    31. 'rules': [{
    32. 'access': 'Allow',
    33. 'destination_address_prefix': '*',
    34. 'destination_port_range': '*',
    35. 'direction': 'Inbound',
    36. 'name': 'rule1',
    37. 'priority': 1001,
    38. 'protocol': '*',
    39. 'source_address_prefixes': ['10.20.2.0/24'],
    40. 'source_port_range': '*'
    41. }]
    42. }]
    43. }
    44. }

    附言

    用类似的方法,可以创建删除资源的接口与查看任务状态的接口,示例如下:

    app.py

    1. @auto_deploy_api.delete("/delete_resource")
    2. async def delete_res(VERABLES_DICT: Item, backgroundTasks: BackgroundTasks):
    3. VERABLES = VERABLES_DICT.VERABLES
    4. try:
    5. # 启动异步任务,因为删除资源耗时较长,http请求不应也无法一直保持
    6. backgroundTasks.add_task(delete_resource, VERABLES)
    7. return {"message": "Will delete"}
    8. except auto.StackNotFoundError:
    9. return HTTPException(status_code=404,detail="application not found")
    10. except Exception as e:
    11. return Response(status_code=500, content=repr(e))
    12. @auto_deploy_api.post("/stack_status")
    13. async def delete_res(VERABLES_DICT: Item):
    14. VERABLES = VERABLES_DICT.VERABLES
    15. try:
    16. data = get_stack_status(VERABLES["ID"])
    17. return data
    18. except auto.StackNotFoundError:
    19. return HTTPException(status_code=404,detail="id not found")
    20. except Exception as e:
    21. return Response(status_code=500, content=repr(e))

    auto_deploy_inline.py

    1. def delete_resource(VERABLES):
    2. stack_name = VERABLES["GLOBAL_VARS"]["STACK_NAME"]
    3. project_name = VERABLES["GLOBAL_VARS"]["PROJECT_NAME"]
    4. client_id, client_secret, client_tenant = get_azure_sp()
    5. def resource_init():
    6. pass
    7. resource_stack = auto.create_or_select_stack(stack_name=stack_name,
    8. project_name=app_name,
    9. program=resource_init)
    10. resource_stack.refresh(on_output=print)
    11. resource_destroy_res = resource_stack.destroy(on_output=print)
    12. return (resource_destroy_res.stdout)
    13. def get_stack_status(id):
    14. data = get_task_record(id)
    15. if len(data) == 0:
    16. raise auto.StackNotFoundError
    17. data_dict = {"id": id, "stack_name": data[0][1], "status": data[0][3],
    18. "payload": data[0][2], "output": data[0][4]}
    19. return data_dict

    调用方式:

    图片

    图片

    另外,若配置了资源CMDB,可在FastAPI Gateway上配置资源信息查询接口,示例如下:

    图片

  • 相关阅读:
    WPF创建自定义控件编译通过但是找不到资源
    【网络篇】如何在服务器之间建立互信
    真正的测试 =“半个产品+半个开发”?
    vscode搭建c/c++环境
    【数字图像处理】图像的几何变换
    JVM下篇(四、JVM运行时参数)
    2021 CCPC桂林G E D【二分、区间贪心、有向最小环】
    12.LoadRunner,基于html录制和基于url录制
    在windows下CorelDraw中VBE的调用原理?
    python语言字符串练习题
  • 原文地址:https://blog.csdn.net/weixin_42196283/article/details/139835264