簡體   English   中英

讀取 python dict 或 yaml 並作為帶有參數和對象的 python 函數調用

[英]Read python dict or yaml and call as python function with arguments and objects

閱讀下面的python dict或其等效的yaml並生成等效的python函數調用

mydict = {'RouteAdd': {'route_config': {'RouteConfig': {'table_name': 'my table', 'app_id': 'my app', 'nexthops': [{'NexthopInfo': {'nexthop_index': 2, 'nexthop_address': {'GatewayAddress': {'ethernet_mac': 'my mac', 'nexthop_ip': 'my ip'}}, 'if_name': 'my interface'}}]}}}}

它的 yaml(為了可讀性):

RouteAdd:
  route_config:
    RouteConfig:
      app_id: "my app"
      nexthops:
      - NexthopInfo:
          if_name: "my interface"
          nexthop_address:
            GatewayAddress:
              ethernet_mac: "my mac"
              nexthop_ip: "my ip"
          nexthop_index: 2
      table_name: "my table"

我想閱讀上述類型的 yaml 或 python dict 並調用如下:

RouteAdd(route_config=Routeconfig(app_id="my app",nexthops=[NexthopInfo(if_name="my interface",nexthop_address=GatewayAddress(ethernet_mac="my mac",nexthop_ip="my ip"),nexthop_index=2)],table_name="my table"))

基本上替代層次結構是一個對象。 我粘貼的是一個小剪輯。 尋找一個遞歸函數,它通過讀取 yaml 或 python dict 並將其轉換為上述格式,以便我可以調用和執行該函數。 任何幫助深表感謝。 謝謝

嘗試這個:

def call_dict(d):
    k, v = list(d.items())[0]  # ('RouteAdd', {route_config: ...})
    kwargs = {}
    for k2, v2 in v.items():
        if isinstance(v2, dict):
            kwargs[k2] = call_dict(v2)
        elif isinstance(v2, list):
            kwargs[k2] = [(call_dict(v3) if isinstance(v3, dict) else v3) for v3 in v2]
        else:
            kwargs[k2] = v2
    return globals()[k](**kwargs)

測試:

def test1(t_arg=None, t_arg2=None):
    return t_arg + sum(t_arg2)

def test2(t_arg=None):
    return t_arg


res = test1(t_arg=1, t_arg2=[test2(t_arg=2), test2(t_arg=3)])
print(res)  # 6


test_dict = {
    'test1': {
        't_arg': 1,
        't_arg2': [
            {'test2': {'t_arg': 2}},
            {'test2': {'t_arg': 3}},
        ]
    }
}

res = call_dict(test_dict)
print(res)  # 6

更新:

作為代碼字符串:

def str_of_code(d):
    k, v = list(d.items())[0]
    kwargs = {}
    for k2, v2 in v.items():
        if isinstance(v2, dict):
            kwargs[k2] = str_of_code(v2)
        elif isinstance(v2, list):
            kwargs[k2] = '[{}]'.format(', '.join(
                (str_of_code(v3) if isinstance(v3, dict) else repr(v3)) for v3 in v2)
            )
        else:
            kwargs[k2] = repr(v2)
    return '{}({})'.format(k, ', '.join('{}={}'.format(*i) for i in kwargs.items()))


test_dict = {
    'test1': {
        't_arg': 1,
        't_arg2': [
            {'test2': {'t_arg': 2}},
            {'test2': {'t_arg': 3}},
        ]
    }
}

res = str_of_code(test_dict)
print(res)  # test1(t_arg=1, t_arg2=[test2(t_arg=2), test2(t_arg=3)])

嘗試了論壇中建議的所有可用方法,但不知何故沒有解決我正在尋找的解決方案。 因此,作為初學者,通過以下非正統的查找和替換方式解決了它。 如果有人有更好的解決方案,請發布,我想使用它。

api_map_complete = {api: {'config': {'AddressConfig': {'unit': 0, 'port_name': "my_name", 'address': "my address", 'family': 2}}}} 

def dict_to_obj(mystr,flag):
    global api_string
    if re.search(':\(',mystr):    
        if flag % 2 == 0:    
            api_string=mystr.replace(":(","(",1)
            flag=flag+1
        else:
            api_string=mystr.replace(":("," = (",1)
            flag=flag+1
        dict_to_obj(api_string,flag)
    else:    
        mystr=mystr.replace(":"," = ")
        mystr=mystr.replace(",",", ")
        api_string=mystr    

for combo in api_map_complete:    

    api_name=combo.keys()[0]
    for k,v in combo.iteritems():
        api_string=str(v)
        api_string=api_string.replace("{","(")
        api_string=api_string.replace("}",")")
        api_string=api_string.replace(" ","")
        api_string=api_string.replace(":'",":\"")
        api_string=api_string.replace("',","\",")
        api_string=api_string.replace("')","\")")
        api_string=api_string.replace("'","")
        dict_to_obj(api_string,1)
    #print api_string
    api_obj=api_name+api_string
    print api_obj

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM