接著昨天的內容,今天直接實戰。因正規表示式非常強大也很複雜,本次實驗中超出上次所學範圍的部分,請自行查閱相關文檔,本教程僅幫助運維工程師快速上手,如果不理解也可以當作工具直接替換使用。
回顧Telnet、SSH自動巡檢腳本裡面的正規表示式:DeviceName = re.search('.*(?=#show run)',telreply).group(),telreply就是巡檢的內容,那麼打開日誌開頭就能看見:
R1>enable Password: R1#terminal length 0 R1#show run Building configuration...
很顯然,這個正則表達式是要匹配在'#show'之前的任意字符(除換行符),那當然就是R1了。
再做一個複雜點的實驗,截取多一點的配置片段,例如下面的配置,要求找到這台設備每個介面的IP位址:
############################ ! interface FastEthernet0/0 ip address 192.168.1.200 255.255.255.0 duplex half ! interface GigabitEthernet1/0 no ip address shutdown negotiation auto ! interface Serial2/0 ip address 192.168.2.200 255.255.255.0 ! ############################
把它賦值在log中:(真實環境是直接讀取log日誌)
log = ''' #這裡是三引號,之後回車把上面的配置貼進來,之後回車三引號,回車。 (三引號可以貼上多行字符串)。
這時查看一下log:
>>> log '\n!\ninterface FastEthernet0/0\n ip address 192.168.1.200 255.255.255.0\n duplex half\n!\ninterface GigabitEthernet1/0\n no ip address\n shutdown\n negotiation auto\n!\ninterface Serial2/0\n ip address 192.168.2.200 255.255.255.0\n!\n' >>>
可以發現字符串都在一行,換行實際就是'n',如果使用print(log),那麼顯示的效果就是上面那樣有換行的了。
還可以發現Cisco設備的設定都是用感嘆號(!)隔開的,每個介面的開頭都是'interface',IP位址前面都是'ip address'。所以不難想像,正規表示式中有'interface(.*)n',這個是符合了某個介面配置的首行。配置結尾是'!',所以在'!'之前的表達式需要'(?=!)'。中間配置的行數不定,能確定的是至少有一行配置,並且會有換行符,即'(.*n)+?)'。這裡的'+?'表示非貪婪的匹配前面的表達式(.*n)一次或多次。圓括號可以達到捕獲效果,但如果想要關掉捕獲效果,在左括號後面跟隨一個?:就可以了。這次實驗是要匹配所有字符串,所以使用的是're.findall()'。不多解釋了,直接上程式碼:
ljds = re.findall('interface(.*)\n((?:.*\n)+?)(?=!)',log) >>> ljds [(' FastEthernet0/0', ' ip address 192.168.1.200 255.255.255.0\n duplex half\n'), (' GigabitEthernet1/0', ' no ip address\n shutdown\n negotiation auto\n'), (' Serial2/0', ' ip address 192.168.2.200 255.255.255.0\n')]
結果是把每個介面的配置分成了介面名稱+配置的元組,嵌套在了列表裡。
接下來寫個For循環把介面和IP再抓出來:
運作效果:
運作效果:試驗完成!達到了取得每個介面IP位址的目的。