真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

PythonWeb開發(fā):教你如何解放路由管理-創(chuàng)新互聯(lián)

1. 痛點

隨著業(yè)務(wù)的飛速發(fā)展,API接口越來越多,路由管理文件從幾十號變成幾百上千行,且每次上新服務(wù),需要在修改路由文件代碼,帶來一定的風(fēng)險。

創(chuàng)新互聯(lián)服務(wù)項目包括歷城網(wǎng)站建設(shè)、歷城網(wǎng)站制作、歷城網(wǎng)頁制作以及歷城網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,歷城網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到歷城省份的部分城市,未來相信會繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!

Python Web開發(fā): 教你如何解放路由管理

2. 解決方案

  • 既然路由文件隨著業(yè)務(wù)的擴(kuò)展越來越龐大,那就去掉路由文件。
  • 制定對應(yīng)規(guī)則,路由通過API文件名根據(jù)一定的規(guī)則對應(yīng)類名,然后自動導(dǎo)入對應(yīng)實現(xiàn)類,注冊到Web框架中。

    2.1 制定規(guī)則

    下面這套規(guī)則只是其中一種方案,可以針對項目情況制定對應(yīng)的規(guī)則,然后實現(xiàn)相關(guān)代碼,但是整體思路基本一樣。

    1. 代碼目錄結(jié)構(gòu),列一下簡單的項目文件目錄,下面以flask框架為例:
      Python Web開發(fā): 教你如何解放路由管理
      app.py是啟動文件。
      resources是API接口代碼文件夾。
      services是為API接口服務(wù)的函數(shù)封裝文件夾。
      如果項目還有依賴文件,也可以單獨再建其他文件夾。

    2. 項目的API接口代碼均放在resources文件夾下,且此文件夾只能寫接口API服務(wù)代碼。

    3. 接口名稱命名以_連接單詞,而對應(yīng)文件里的類名文件名稱的單詞,不過換成是駝峰寫法。

    4. 類的導(dǎo)入則通過文件名對應(yīng)到類名,實現(xiàn)自動映射注冊到web框架中。

    規(guī)則舉例如下:
    如上圖,resources下有一個hello_world接口,還有一個ab項目文件夾,ab下面還有一個hello_world_python接口以及子項目文件夾testab, testab下面也有一個hello_world_python.

    • 接口文件的文件名命名規(guī)范:
      文件名命名均為小寫,多個單詞之間使用'_'隔開,比如hello_world.py 命名正確,helloWorld.py命名錯誤。

    • 接口文件里的接口類Class命名是以文件名字轉(zhuǎn)為駝峰格式,且首字母大寫。比如hello_world.py 對應(yīng)的接口類是 HelloWorld
      舉例: hello_world.py
      Python Web開發(fā): 教你如何解放路由管理
      hello_world_python.py
      Python Web開發(fā): 教你如何解放路由管理
    1. 路由入口文件會自動映射,映射規(guī)則為:
      前綴 / 項目文件夾[...] / 文件名

      其中 前綴為整個項目的路由前綴,可以定義,也可以不定義,比如api-ab項目,可以定義整個項目的路由前綴為 ab/
      resource下面項目文件夾如果有,則會自動拼接,如果沒有,則不會讀取。
      舉例
      前綴為空,上圖resources中的三個接口對應(yīng)的路由為:

      hello_world.py ==>  /hello_world
      ab/hello_world_python.py ==> /ab/hello_world_python
      ab/testab/hello_world_python.py ==> /ab/testab/hello _world_python

      前綴為ab/,上圖resources中的三個接口對應(yīng)的路由為:

      hello_world.py ==> ab/hello_world
      ab/hello_world_python.py ==> ab/ab/hello_world_python
      ab/testab/hello_world_python.py ==> ab/ab/testab/hello_world_python
    2. 關(guān)于resources里目錄結(jié)構(gòu),代碼里是可以允許N層,但建議不要超過3層, 不易管理。

2.2 代碼實現(xiàn)

python很多框架的啟動和路由管理都很類似,所以這套規(guī)則適合很多框架,測試過程中有包括flask, tornado, sanic, japronto。 以前年代久遠(yuǎn)的web.py也是支持的。

完整代碼地址:
https://github.com/CrystalSkyZ/PyAutoApiRoute

  1. 實現(xiàn)下劃線命名 轉(zhuǎn) 駝峰命名 函數(shù),代碼演示:

    def underline_to_hump(underline_str):
    '''
    下劃線形式字符串轉(zhuǎn)成駝峰形式,首字母大寫
    '''
    sub = re.sub(r'(_\w)', lambda x: x.group(1)[1].upper(), underline_str)
    if len(sub) > 1:
        return sub[0].upper() + sub[1:]
    return sub
  2. 實現(xiàn)根據(jù)字符串導(dǎo)入模塊函數(shù), 代碼演示:

    • 通過python內(nèi)置函數(shù)__import__函數(shù)實現(xiàn)加載類

          def import_object(name):
          """Imports an object by name.
          import_object('x') is equivalent to 'import x'.
          import_object('x.y.z') is equivalent to 'from x.y import z'.
          """
          if not isinstance(name, str):
                  name = name.encode('utf-8')
          if name.count('.') == 0:
                  return __import__(name, None, None)
      
          parts = name.split('.')
          obj = __import__('.'.join(parts[:-1]), None, None, [parts[-1]], 0)
          try:
                  return getattr(obj, parts[-1])
          except AttributeError:
                  raise ImportError("No module named %s" % parts[-1])
    • 通過importlib模塊實現(xiàn)
      importlib.import_module(name)

    上面2種方法都可以,github上代碼里2種方法都有測試。

  3. 檢索resources文件夾,生成路由映射,并導(dǎo)入對應(yīng)實現(xiàn)類, 代碼演示如下:

    def route(route_file_path,
          resources_name="resources",
          route_prefix="",
          existing_route=None):
    
    route_list = []
    
        def get_route_tuple(file_name, route_pre, resource_module_name):
            """
            :param file_name: API file name
            :param route_pre: route prefix
            :param resource_module_name: resource module
            """
            nonlocal route_list
            nonlocal existing_route
            route_endpoint = file_name.split(".py")[0]
            #module = importlib.import_module('{}.{}'.format(
            #    resource_module_name, route_endpoint))
            module = import_object('{}.{}'.format(
                resource_module_name, route_endpoint))
            route_class = underline_to_hump(route_endpoint)
            real_route_endpoint = r'/{}{}'.format(route_pre, route_endpoint)
            if existing_route and isinstance(existing_route, dict):
                if real_route_endpoint in existing_route:
                    real_route_endpoint = existing_route[real_route_endpoint]
            route_list.append((real_route_endpoint, getattr(module, route_class)))
    
        def check_file_right(file_name):
            if file_name.startswith("_"):
                return False
            if not file_name.endswith(".py"):
                return False
            if file_name.startswith("."):
                return False
            return True
    
        def recursive_find_route(route_path, sub_resource, route_pre=""):
            nonlocal route_prefix
            nonlocal resources_name
            file_list = os.listdir(route_path)
            if config.DEBUG:
                print("FileList:", file_list)
            for file_item in file_list:
                if file_item.startswith("_"):
                    continue
                if file_item.startswith("."):
                    continue
                if os.path.isdir(route_path + "/{}".format(file_item)):
                    recursive_find_route(route_path + "/{}".format(file_item), sub_resource + ".{}".format(file_item), "{}{}/".format(route_pre, file_item))
                    continue
                if not check_file_right(file_item):
                    continue
                get_route_tuple(file_item, route_prefix + route_pre, sub_resource)
    
    recursive_find_route(route_file_path, resources_name)
    if config.DEBUG:
        print("RouteList:", route_list)
    
    return route_list
    • get_route_tuple函數(shù)作用是通過字符串導(dǎo)入類,并將路由和類以元組的方式添加到數(shù)組中。
    • check_file_right函數(shù)作用是過濾文件夾中不合法的文件。
    • recursive_find_route函數(shù)采用遞歸查找resources中的文件。
    • existing_route參數(shù)是將已經(jīng)線上存在的路由替換新規(guī)則生成的路由,這樣舊項目也是可以優(yōu)化使用這套規(guī)則。

3. 應(yīng)用到項目中

以flask框架為例,其余框架請看github中的代碼演示。
app.py 中代碼

   app = Flask(__name__)
   api = Api(app)
   # APi route and processing functions
   exist_route = {"/flask/hello_world": "/hello_world"}
   route_path = "./resources"
   route_list = route(
       route_path,
       resources_name="resources",
       route_prefix="flask/",
       existing_route=exist_route)

   for item in route_list:
       api.add_resource(item[1], item[0])

   if __name__ == "__main__":
       app.run(host="0.0.0.0", port=int(parse_args.port), debug=config.DEBUG)

運行app.py之后,路由打印如下:

   RouteList: [
  ('/hello_world', ),
  ('/flask/ab/testab/hello_world_python_test', \
            ), 
  ('/flask/ab/hello_world_python', )
  ]

元組第一個元素則是路由,第二個元素是對應(yīng)的實現(xiàn)類。


總結(jié):
至此,通過制定一定規(guī)則,解放路由管理文件方案完成。 歡迎各位一起討論其余比較好的方案,更多方案討論可以關(guān)注微信公眾號: 天澄的技術(shù)筆記 。
Python Web開發(fā): 教你如何解放路由管理

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)cdcxhl.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。


分享名稱:PythonWeb開發(fā):教你如何解放路由管理-創(chuàng)新互聯(lián)
URL標(biāo)題:http://weahome.cn/article/ceiooj.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部