1*** Settings *** 2Resource ../lib/utils.robot 3Resource ../lib/connection_client.robot 4Resource ../lib/boot_utils.robot 5Library ../lib/gen_misc.py 6Library ../lib/utils.py 7Library ../lib/bmc_network_utils.py 8 9*** Variables *** 10# MAC input from user. 11${MAC_ADDRESS} ${EMPTY} 12 13 14*** Keywords *** 15 16Check And Reset MAC 17 [Documentation] Update BMC with user input MAC address. 18 [Arguments] ${mac_address}=${MAC_ADDRESS} 19 20 # Description of argument(s): 21 # mac_address The mac address (e.g. 00:01:6c:80:02:28). 22 23 ${active_channel_config}= Get Active Channel Config 24 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 25 26 Should Not Be Empty ${mac_address} 27 Open Connection And Log In 28 ${bmc_mac_addr} ${stderr} ${rc}= BMC Execute Command 29 ... cat /sys/class/net/${ethernet_interface}/address 30 IF '${mac_address.lower()}' != '${bmc_mac_addr.lower()}' 31 Set MAC Address 32 END 33 34 35Set MAC Address 36 [Documentation] Update eth0 with input MAC address. 37 [Arguments] ${mac_address}=${MAC_ADDRESS} 38 39 # Description of argument(s): 40 # mac_address The mac address (e.g. 00:01:6c:80:02:28). 41 42 ${active_channel_config}= Get Active Channel Config 43 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 44 45 Write fw_setenv ethaddr ${mac_address} 46 OBMC Reboot (off) 47 48 # Take SSH session post BMC reboot. 49 Open Connection And Log In 50 ${bmc_mac_addr} ${stderr} ${rc}= BMC Execute Command 51 ... cat /sys/class/net/${ethernet_interface}/address 52 Should Be Equal ${bmc_mac_addr} ${mac_address} ignore_case=True 53 54 55Get BMC IP Info 56 [Documentation] Get system IP address and prefix length. 57 58 59 # Get system IP address and prefix length details using "ip addr" 60 # Sample Output of "ip addr": 61 # 1: eth0: <BROADCAST,MULTIAST> mtu 1500 qdisc mq state UP qlen 1000 62 # link/ether xx:xx:xx:xx:xx:xx brd ff:ff:ff:ff:ff:ff 63 # inet xx.xx.xx.xx/24 brd xx.xx.xx.xx scope global eth0 64 65 ${active_channel_config}= Get Active Channel Config 66 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 67 ${cmd_output} ${stderr} ${rc}= BMC Execute Command 68 ... /sbin/ip addr | grep ${ethernet_interface} 69 70 # Get line having IP address details. 71 ${lines}= Get Lines Containing String ${cmd_output} inet 72 73 # List IP address details. 74 @{ip_components}= Split To Lines ${lines} 75 76 @{ip_data}= Create List 77 78 # Get all IP addresses and prefix lengths on system. 79 FOR ${ip_component} IN @{ip_components} 80 @{if_info}= Split String ${ip_component} 81 ${ip_n_prefix}= Get From List ${if_info} 1 82 Append To List ${ip_data} ${ip_n_prefix} 83 END 84 85 RETURN ${ip_data} 86 87Get BMC Route Info 88 [Documentation] Get system route info. 89 90 # Sample output of "ip route": 91 # default via xx.xx.xx.x dev eth0 92 # xx.xx.xx.0/23 dev eth0 src xx.xx.xx.xx 93 # xx.xx.xx.0/24 dev eth0 src xx.xx.xx.xx 94 95 ${cmd_output} ${stderr} ${rc}= BMC Execute Command 96 ... /sbin/ip route 97 98 RETURN ${cmd_output} 99 100Get BMC MAC Address 101 [Documentation] Get system MAC address. 102 103 # Sample output of "ip addr | grep ether": 104 # link/ether xx.xx.xx.xx.xx.xx brd ff:ff:ff:ff:ff:ff 105 106 ${active_channel_config}= Get Active Channel Config 107 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 108 109 ${cmd_output} ${stderr} ${rc}= BMC Execute Command 110 ... /sbin/ip addr | grep ${ethernet_interface} -A 1 | grep ether 111 112 # Split the line and return MAC address. 113 # Split list data: 114 # link/ether | xx:xx:xx:xx:xx:xx | brd | ff:ff:ff:ff:ff:ff 115 116 @{words}= Split String ${cmd_output} 117 118 RETURN ${words[1]} 119 120 121Get BMC MAC Address List 122 [Documentation] Get system MAC address 123 124 # Sample output of "ip addr | grep ether": 125 # link/ether xx.xx.xx.xx.xx.xx brd ff:ff:ff:ff:ff:ff 126 127 ${cmd_output} ${stderr} ${rc}= BMC Execute Command 128 ... /sbin/ip addr | grep ether 129 130 # Split the line and return MAC address. 131 # Split list data: 132 # link/ether | xx:xx:xx:xx:xx:xx | brd | ff:ff:ff:ff:ff:ff 133 # link/ether | xx:xx:xx:xx:xx:xx | brd | ff:ff:ff:ff:ff:ff 134 135 ${mac_list}= Create List 136 @{lines}= Split To Lines ${cmd_output} 137 FOR ${line} IN @{lines} 138 @{words}= Split String ${line} 139 Append To List ${mac_list} ${words[1]} 140 END 141 142 RETURN ${mac_list} 143 144Get BMC Hostname 145 [Documentation] Get BMC hostname. 146 147 # Sample output of "hostname": 148 # test_hostname 149 150 ${output} ${stderr} ${rc}= BMC Execute Command hostname 151 152 RETURN ${output} 153 154Get FW_Env MAC Address 155 [Documentation] Get FW_Env MAC address. 156 157 # Sample output of "fw_printenv | grep ethaddr" 158 # ethaddr=xx:xx:xx:xx:xx:xx:xx 159 160 ${active_channel_config}= Get Active Channel Config 161 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 162 163 ${ethernet_interface}= Set Variable If 164 ... "${ethernet_interface}"=="eth0" ethaddr eth1addr 165 166 ${cmd_output} ${stderr} ${rc}= BMC Execute Command /sbin/fw_printenv | grep ${ethernet_interface} 167 168 # Split the line and return MAC address. 169 # Split list data: 170 # ethaddr | xx:xx:xx:xx:xx:xx:xx 171 172 @{words}= Split String ${cmd_output} = 173 174 RETURN ${words[1]} 175 176 177Get List Of IP Address Via REST 178 [Documentation] Get list of IP address via REST. 179 [Arguments] @{ip_uri_list} 180 181 # Description of argument(s): 182 # ip_uri_list List of IP objects. 183 # Example: 184 # "data": [ 185 # "/xyz/openbmc_project/network/eth0/ipv4/e9767624", 186 # "/xyz/openbmc_project/network/eth0/ipv4/31f4ce8b" 187 # ], 188 189 ${ip_list}= Create List 190 191 FOR ${ip_uri} IN @{ip_uri_list} 192 ${ip_addr}= Read Attribute ${ip_uri} Address 193 Append To List ${ip_list} ${ip_addr} 194 END 195 196 RETURN @{ip_list} 197 198Delete IP And Object 199 [Documentation] Delete IP and object. 200 [Arguments] ${ip_addr} @{ip_uri_list} 201 202 # Description of argument(s): 203 # ip_addr IP address to be deleted. 204 # ip_uri_list List of IP object URIs. 205 206 # Find IP object having this IP address. 207 208 FOR ${ip_uri} IN @{ip_uri_list} 209 ${ip_addr1}= Read Attribute ${ip_uri} Address 210 IF '${ip_addr}' == '${ip_addr1}' 211 Exit For Loop 212 END 213 END 214 215 # If the given IP address is not configured, return. 216 # Otherwise, delete the IP and object. 217 218 Run Keyword And Return If '${ip_addr}' != '${ip_addr1}' 219 ... Pass Execution IP address to be deleted is not configured. 220 221 Run Keyword And Ignore Error OpenBMC Delete Request ${ip_uri} 222 223 # After any modification on network interface, BMC restarts network 224 # module, wait until it is reachable. Then wait 15 seconds for new 225 # configuration to be updated on BMC. 226 227 Wait For Host To Ping ${OPENBMC_HOST} ${NETWORK_TIMEOUT} 228 ... ${NETWORK_RETRY_TIME} 229 Sleep 15s 230 231 # Verify whether deleted IP address is removed from BMC system. 232 233 ${ip_data}= Get BMC IP Info 234 Should Not Contain Match ${ip_data} ${ip_addr}* 235 ... msg=IP address not deleted. 236 237Get First Non Pingable IP From Subnet 238 [Documentation] Find first non-pingable IP from the subnet and return it. 239 [Arguments] ${host}=${OPENBMC_HOST} 240 241 # Description of argument(s): 242 # host Any valid host name or IP address 243 # (e.g. "machine1" or "9.xx.xx.31"). 244 245 # Non-pingable IP is unused IP address in the subnet. 246 ${host_name} ${ip_addr}= Get Host Name IP ${host} 247 248 # Split IP address into network part and host part. 249 # IP address will have 4 octets xx.xx.xx.xx. 250 # Sample output after split: 251 # split_ip [xx.xx.xx, xx] 252 253 ${split_ip}= Split String From Right ${ip_addr} . 1 254 # First element in list is Network part. 255 ${network_part}= Get From List ${split_ip} 0 256 257 FOR ${octet4} IN RANGE 1 255 258 ${new_ip}= Catenate ${network_part}.${octet4} 259 ${status}= Run Keyword And Return Status Ping Host ${new_ip} 260 # If IP is non-pingable, return it. 261 Return From Keyword If '${status}' == 'False' ${new_ip} 262 END 263 264 Fail msg=No non-pingable IP could be found in subnet ${network_part}. 265 266 267Validate MAC On BMC 268 [Documentation] Validate MAC on BMC. 269 [Arguments] ${mac_addr} 270 271 # Description of argument(s): 272 # mac_addr MAC address of the BMC. 273 274 ${system_mac}= Get BMC MAC Address 275 ${mac_new_addr}= Truncate MAC Address ${system_mac} ${mac_addr} 276 277 ${status}= Compare MAC Address ${system_mac} ${mac_new_addr} 278 Should Be True ${status} 279 ... msg=MAC address ${system_mac} does not match ${mac_new_addr}. 280 281Validate MAC On FW_Env 282 [Documentation] Validate MAC on FW_Env. 283 [Arguments] ${mac_addr} 284 285 # Description of argument(s): 286 # mac_addr MAC address of the BMC. 287 288 ${fw_env_addr}= Get FW_Env MAC Address 289 ${mac_new_addr}= Truncate MAC Address ${fw_env_addr} ${mac_addr} 290 291 ${status}= Compare MAC Address ${fw_env_addr} ${mac_new_addr} 292 Should Be True ${status} 293 ... msg=MAC address ${fw_env_addr} does not match ${mac_new_addr}. 294 295Truncate MAC Address 296 [Documentation] Truncates and returns user provided MAC address. 297 [Arguments] ${sys_mac_addr} ${user_mac_addr} 298 299 # Description of argument(s): 300 # sys_mac_addr MAC address of the BMC. 301 # user_mac_addr user provided MAC address. 302 303 ${mac_byte}= Set Variable ${0} 304 @{user_mac_list}= Split String ${user_mac_addr} : 305 @{sys_mac_list}= Split String ${sys_mac_addr} : 306 ${user_new_mac_list} Create List 307 308 # Truncate extra bytes and bits from MAC address 309 FOR ${mac_item} IN @{sys_mac_list} 310 ${invalid_mac_byte} = Get Regexp Matches ${user_mac_list}[${mac_byte}] [^A-Za-z0-9]+ 311 Return From Keyword If ${invalid_mac_byte} ${user_mac_addr} 312 ${mac_int} = Convert To Integer ${user_mac_list}[${mac_byte}] 16 313 ${user_mac_len} = Get Length ${user_mac_list} 314 IF ${mac_int} >= ${256} 315 ${user_mac_byte}= Truncate MAC Bits ${user_mac_list}[${mac_byte}] 316 ELSE 317 ${user_mac_byte}= Set Variable ${user_mac_list}[${mac_byte}] 318 END 319 Append To List ${user_new_mac_list} ${user_mac_byte} 320 ${mac_byte} = Set Variable ${mac_byte + 1} 321 Exit For Loop If '${mac_byte}' == '${user_mac_len}' 322 323 END 324 ${user_new_mac_string}= Evaluate ":".join(${user_new_mac_list}) 325 RETURN ${user_new_mac_string} 326 327Truncate MAC Bits 328 [Documentation] Truncates user provided MAC address byte to bits. 329 [Arguments] ${user_mac_addr_byte} 330 331 # Description of argument(s): 332 # user_mac_addr_byte user provided MAC address byte to truncate bits 333 334 ${user_mac_addr_int}= Convert To List ${user_mac_addr_byte} 335 ${user_mac_addr_byte}= Get Slice From List ${user_mac_addr_int} 0 2 336 ${user_mac_addr_byte_string}= Evaluate "".join(${user_mac_addr_byte}) 337 RETURN ${user_mac_addr_byte_string} 338 339 340Run Build Net 341 [Documentation] Run build_net to preconfigure the ethernet interfaces. 342 343 OS Execute Command build_net help y y 344 # Run pingum to check if the "build_net" was run correctly done. 345 ${output} ${stderr} ${rc}= OS Execute Command pingum 346 Should Contain ${output} All networks ping Ok 347 348 349Configure Hostname 350 [Documentation] Configure hostname on BMC via Redfish. 351 [Arguments] ${hostname} ${status_code}=[${HTTP_OK}, ${HTTP_NO_CONTENT}] 352 353 # Description of argument(s): 354 # hostname A hostname value which is to be configured on BMC. 355 # status_codes Expected return code from patch operation 356 # (e.g. "200"). See prolog of rest_request 357 # method in redfish_plus.py for details. 358 359 ${active_channel_config}= Get Active Channel Config 360 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 361 362 ${data}= Create Dictionary HostName=${hostname} 363 Redfish.patch ${REDFISH_NW_ETH_IFACE}${ethernet_interface} body=&{data} 364 ... valid_status_codes=${status_code} 365 366 367Verify IP On BMC 368 [Documentation] Verify IP on BMC. 369 [Arguments] ${ip} 370 371 # Description of argument(s): 372 # ip IP address to be verified (e.g. "10.7.7.7"). 373 374 # Get IP address details on BMC using IP command. 375 @{ip_data}= Get BMC IP Info 376 Should Contain Match ${ip_data} ${ip}/* 377 ... msg=IP address does not exist. 378 379 380Verify Gateway On BMC 381 [Documentation] Verify gateway on BMC. 382 [Arguments] ${gateway_ip}=0.0.0.0 383 384 # Description of argument(s): 385 # gateway_ip Gateway IP address. 386 387 ${route_info}= Get BMC Route Info 388 389 # If gateway IP is empty or 0.0.0.0 it will not have route entry. 390 391 IF '${gateway_ip}' == '0.0.0.0' 392 Pass Execution Gateway IP is "0.0.0.0". 393 ELSE 394 Should Contain ${route_info} ${gateway_ip} msg=Gateway IP address not matching. 395 END 396 397 398Get BMC DNS Info 399 [Documentation] Get system DNS info. 400 401 402 # Sample output of "resolv.conf": 403 # ### Generated manually via dbus settings ### 404 # nameserver 8.8.8.8 405 406 ${cmd_output} ${stderr} ${rc}= BMC Execute Command 407 ... cat /etc/resolv.conf 408 409 RETURN ${cmd_output} 410 411 412CLI Get Nameservers 413 [Documentation] Get the nameserver IPs from /etc/resolv.conf and return as a list. 414 415 # Example of /etc/resolv.conf data: 416 # nameserver x.x.x.x 417 # nameserver y.y.y.y 418 419 ${stdout} ${stderr} ${rc}= BMC Execute Command 420 ... egrep nameserver /etc/resolv.conf | cut -f2- -d ' ' 421 ${nameservers}= Split String ${stdout} 422 423 RETURN ${nameservers} 424 425CLI Get and Verify Name Servers 426 [Documentation] Get and Verify the nameserver IPs from /etc/resolv.conf 427 ... and compare with redfish nameserver. 428 [Arguments] ${static_name_servers} 429 ... ${valid_status_codes}=${HTTP_OK} 430 431 # Description of Argument(s): 432 # static_name_servers Address for static name server 433 # valid_status_codes Expected return code from patch operation 434 # (e.g. "200"). See prolog of rest_request 435 # method in redfish_plus.py for details. 436 437 ${cli_nameservers}= CLI Get Nameservers 438 ${cmd_status}= Run Keyword And Return Status 439 ... List Should Contain Sub List ${cli_nameservers} ${static_name_servers} 440 441 IF '${valid_status_codes}' == '${HTTP_OK}' 442 Should Be True ${cmd_status} 443 ELSE 444 Should Not Be True ${cmd_status} 445 END 446 447Get Network Configuration 448 [Documentation] Get network configuration. 449 # Sample output: 450 #{ 451 # "@odata.context": "/redfish/v1/$metadata#EthernetInterface.EthernetInterface", 452 # "@odata.id": "/redfish/v1/Managers/${MANAGER_ID}/EthernetInterfaces/eth0", 453 # "@odata.type": "#EthernetInterface.v1_2_0.EthernetInterface", 454 # "Description": "Management Network Interface", 455 # "IPv4Addresses": [ 456 # { 457 # "Address": "169.254.xx.xx", 458 # "AddressOrigin": "IPv4LinkLocal", 459 # "Gateway": "0.0.0.0", 460 # "SubnetMask": "255.255.0.0" 461 # }, 462 # { 463 # "Address": "xx.xx.xx.xx", 464 # "AddressOrigin": "Static", 465 # "Gateway": "xx.xx.xx.1", 466 # "SubnetMask": "xx.xx.xx.xx" 467 # } 468 # ], 469 # "Id": "eth0", 470 # "MACAddress": "xx:xx:xx:xx:xx:xx", 471 # "Name": "Manager Ethernet Interface", 472 # "SpeedMbps": 0, 473 # "VLAN": { 474 # "VLANEnable": false, 475 # "VLANId": 0 476 # } 477 [Arguments] ${network_active_channel}=${CHANNEL_NUMBER} 478 479 # Description of argument(s): 480 # network_active_channel Ethernet channel number (eg. 1 or 2) 481 482 ${active_channel_config}= Get Active Channel Config 483 ${resp}= Redfish.Get 484 ... ${REDFISH_NW_ETH_IFACE}${active_channel_config['${network_active_channel}']['name']} 485 486 @{network_configurations}= Get From Dictionary ${resp.dict} IPv4StaticAddresses 487 RETURN @{network_configurations} 488 489 490Add IP Address 491 [Documentation] Add IP Address To BMC. 492 [Arguments] ${ip} ${subnet_mask} ${gateway} 493 ... ${valid_status_codes}=${HTTP_OK} 494 495 # Description of argument(s): 496 # ip IP address to be added (e.g. "10.7.7.7"). 497 # subnet_mask Subnet mask for the IP to be added 498 # (e.g. "255.255.0.0"). 499 # gateway Gateway for the IP to be added (e.g. "10.7.7.1"). 500 # valid_status_codes Expected return code from patch operation 501 # (e.g. "200"). See prolog of rest_request 502 # method in redfish_plus.py for details. 503 504 ${empty_dict}= Create Dictionary 505 ${ip_data}= Create Dictionary Address=${ip} 506 ... SubnetMask=${subnet_mask} Gateway=${gateway} 507 508 ${patch_list}= Create List 509 ${network_configurations}= Get Network Configuration 510 ${num_entries}= Get Length ${network_configurations} 511 512 FOR ${INDEX} IN RANGE 0 ${num_entries} 513 Append To List ${patch_list} ${empty_dict} 514 END 515 516 ${valid_status_codes}= Set Variable If '${valid_status_codes}' == '${HTTP_OK}' 517 ... ${HTTP_OK},${HTTP_NO_CONTENT} ${valid_status_codes} 518 519 # We need not check for existence of IP on BMC while adding. 520 Append To List ${patch_list} ${ip_data} 521 ${data}= Create Dictionary IPv4StaticAddresses=${patch_list} 522 523 ${active_channel_config}= Get Active Channel Config 524 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 525 526 Redfish.patch ${REDFISH_NW_ETH_IFACE}${ethernet_interface} body=&{data} 527 ... valid_status_codes=[${valid_status_codes}] 528 529 Return From Keyword If '${valid_status_codes}' != '${HTTP_OK},${HTTP_NO_CONTENT}' 530 531 # Note: Network restart takes around 15-18s after patch request processing. 532 Sleep ${NETWORK_TIMEOUT}s 533 Wait For Host To Ping ${OPENBMC_HOST} ${NETWORK_TIMEOUT} 534 535 Verify IP On BMC ${ip} 536 Validate Network Config On BMC 537 538 539Delete IP Address 540 [Documentation] Delete IP Address Of BMC. 541 [Arguments] ${ip} ${valid_status_codes}=[${HTTP_OK},${HTTP_ACCEPTED},${HTTP_NO_CONTENT}] 542 543 # Description of argument(s): 544 # ip IP address to be deleted (e.g. "10.7.7.7"). 545 # valid_status_codes Expected return code from patch operation 546 # (e.g. "200"). See prolog of rest_request 547 # method in redfish_plus.py for details. 548 549 ${empty_dict}= Create Dictionary 550 ${patch_list}= Create List 551 552 @{network_configurations}= Get Network Configuration 553 FOR ${network_configuration} IN @{network_configurations} 554 Run Keyword If '${network_configuration['Address']}' == '${ip}' 555 ... Append To List ${patch_list} ${null} 556 ... ELSE Append To List ${patch_list} ${empty_dict} 557 END 558 559 ${ip_found}= Run Keyword And Return Status List Should Contain Value 560 ... ${patch_list} ${null} msg=${ip} does not exist on BMC 561 Pass Execution If ${ip_found} == ${False} ${ip} does not exist on BMC 562 563 # Run patch command only if given IP is found on BMC 564 ${data}= Create Dictionary IPv4StaticAddresses=${patch_list} 565 566 ${active_channel_config}= Get Active Channel Config 567 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 568 569 Redfish.patch ${REDFISH_NW_ETH_IFACE}${ethernet_interface} body=&{data} 570 ... valid_status_codes=${valid_status_codes} 571 572 # Note: Network restart takes around 15-18s after patch request processing 573 Sleep ${NETWORK_TIMEOUT}s 574 Wait For Host To Ping ${OPENBMC_HOST} ${NETWORK_TIMEOUT} 575 576 ${delete_status}= Run Keyword And Return Status Verify IP On BMC ${ip} 577 IF '${valid_status_codes}' == '[${HTTP_OK},${HTTP_ACCEPTED},${HTTP_NO_CONTENT}]' 578 Should Be True '${delete_status}' == '${False}' 579 ELSE 580 Should Be True '${delete_status}' == '${True}' 581 END 582 583 Validate Network Config On BMC 584 585 586Validate Network Config On BMC 587 [Documentation] Check that network info obtained via redfish matches info 588 ... obtained via CLI. 589 590 @{network_configurations}= Get Network Configuration 591 ${ip_data}= Get BMC IP Info 592 FOR ${network_configuration} IN @{network_configurations} 593 Should Contain Match ${ip_data} ${network_configuration['Address']}/* 594 ... msg=IP address does not exist. 595 END 596 597 598Create VLAN 599 [Documentation] Create a VLAN. 600 [Arguments] ${id} ${interface}=eth0 ${expected_result}=valid 601 602 # Description of argument(s): 603 # id The VLAN ID (e.g. '53'). 604 # interface The physical interface for the VLAN(e.g. 'eth0'). 605 # expected_result Expected status of VLAN configuration. 606 607 @{data_vlan_id}= Create List ${interface} ${id} 608 ${data}= Create Dictionary data=@{data_vlan_id} 609 ${resp}= OpenBMC Post Request ${vlan_resource} data=${data} 610 ${resp.status_code}= Convert To String ${resp.status_code} 611 ${status}= Run Keyword And Return Status 612 ... Valid Value resp.status_code ["${HTTP_OK}"] 613 614 IF '${expected_result}' == 'error' 615 Should Be Equal ${status} ${False} msg=Configuration of an invalid VLAN ID Failed. 616 ELSE 617 Should Be Equal ${status} ${True} msg=Configuration of a valid VLAN ID Failed. 618 END 619 620 Sleep ${NETWORK_TIMEOUT}s 621 622 623Configure Network Settings On VLAN 624 [Documentation] Configure network settings. 625 [Arguments] ${id} ${ip_addr} ${prefix_len} ${gateway_ip}=${gateway} 626 ... ${expected_result}=valid ${interface}=eth0 627 628 # Description of argument(s): 629 # id The VLAN ID (e.g. '53'). 630 # ip_addr IP address of VLAN Interface. 631 # prefix_len Prefix length of VLAN Interface. 632 # gateway_ip Gateway IP address of VLAN Interface. 633 # expected_result Expected status of network setting configuration. 634 # interface Physical Interface on which the VLAN is defined. 635 636 @{ip_parm_list}= Create List ${network_resource} 637 ... ${ip_addr} ${prefix_len} ${gateway_ip} 638 639 ${data}= Create Dictionary data=@{ip_parm_list} 640 641 Run Keyword And Ignore Error OpenBMC Post Request 642 ... ${NETWORK_MANAGER}${interface}_${id}/action/IP data=${data} 643 644 # After any modification on network interface, BMC restarts network 645 # module, wait until it is reachable. Then wait 15 seconds for new 646 # configuration to be updated on BMC. 647 648 Wait For Host To Ping ${OPENBMC_HOST} ${NETWORK_TIMEOUT} 649 ... ${NETWORK_RETRY_TIME} 650 Sleep ${NETWORK_TIMEOUT}s 651 652 # Verify whether new IP address is populated on BMC system. 653 # It should not allow to configure invalid settings. 654 ${status}= Run Keyword And Return Status 655 ... Verify IP On BMC ${ip_addr} 656 657 IF '${expected_result}' == 'error' 658 Should Be Equal ${status} ${False} msg=Configuration of invalid IP Failed. 659 ELSE 660 Should Be Equal ${status} ${True} msg=Configuration of valid IP Failed. 661 END 662 663 664Get BMC Default Gateway 665 [Documentation] Get system default gateway. 666 667 ${route_info}= Get BMC Route Info 668 669 ${lines}= Get Lines Containing String ${route_info} default via 670 671 ${active_channel_config}= Get Active Channel Config 672 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 673 674 # Extract the corresponding default gateway for eth0 and eth1 675 ${default_gw_line}= Get Lines Containing String ${lines} ${ethernet_interface} 676 ${default_gw}= Split String ${default_gw_line} 677 678 # Example of the output 679 # default_gw: 680 # [0]: default 681 # [1]: via 682 # [2]: xx.xx.xx.1 683 # [3]: dev 684 # [4]: eth1 685 686 RETURN ${default_gw[2]} 687 688 689Validate Hostname On BMC 690 [Documentation] Verify that the hostname read via Redfish is the same as the 691 ... hostname configured on system. 692 [Arguments] ${hostname} 693 694 # Description of argument(s): 695 # hostname A hostname value which is to be compared to the hostname 696 # configured on system. 697 698 ${sys_hostname}= Get BMC Hostname 699 Should Be Equal ${sys_hostname} ${hostname} 700 ... ignore_case=True msg=Hostname does not exist. 701 702Get Channel Number For All Interface 703 [Documentation] Gets the Interface name and returns the channel number for the given interface. 704 705 ${valid_channel_number_interface_names}= Get Channel Config 706 707 ${valid_channel_number_interface_names}= Convert To Dictionary ${valid_channel_number_interface_names} 708 709 RETURN ${valid_channel_number_interface_names} 710 711Get Valid Channel Number 712 [Documentation] Get Valid Channel Number. 713 [Arguments] ${valid_channel_number_interface_names} 714 715 # Description of argument(s): 716 # valid_channel_number_interface_names Contains channel names in dict. 717 718 &{valid_channel_number_interface_name}= Create Dictionary 719 720 FOR ${key} ${values} IN &{valid_channel_number_interface_names} 721 IF '${values['is_valid']}' == 'True' 722 Set To Dictionary ${valid_channel_number_interface_name} ${key} ${values} 723 END 724 END 725 726 RETURN ${valid_channel_number_interface_name} 727 728Get Invalid Channel Number List 729 [Documentation] Get Invalid Channel and return as list. 730 731 ${available_channels}= Get Channel Number For All Interface 732 # Get the channel which medium_type as 'reserved' and append it to a list. 733 @{invalid_channel_number_list}= Create List 734 735 FOR ${channel_number} ${values} IN &{available_channels} 736 IF '${values['channel_info']['medium_type']}' == 'reserved' 737 Append To List ${invalid_channel_number_list} ${channel_number} 738 END 739 END 740 741 RETURN ${invalid_channel_number_list} 742 743 744Get Channel Number For Valid Ethernet Interface 745 [Documentation] Get channel number for all ethernet interface. 746 [Arguments] ${valid_channel_number_interface_name} 747 748 # Description of argument(s): 749 # channel_number_list Contains channel names in list. 750 751 @{channel_number_list}= Create List 752 753 FOR ${channel_number} ${values} IN &{valid_channel_number_interface_name} 754 ${usb_interface_status}= Run Keyword And Return Status Should Not Contain ${values['name']} usb 755 Continue For Loop IF ${usb_interface_status} == False 756 IF '${values['channel_info']['medium_type']}' == 'lan-802.3' 757 Append To List ${channel_number_list} ${channel_number} 758 END 759 END 760 761 RETURN ${channel_number_list} 762 763 764Get Current Channel Name List 765 [Documentation] Get Current Channel name and append it to active channel list. 766 [Arguments] ${channel_list} ${channel_config_json} 767 768 # Description of Arguments 769 # ${channel_list} - list Contains all available active channels. 770 # ${channel_config_json} - output of /usr/share/ipmi-providers/channel_config.json file. 771 772 FOR ${channel_number} ${values} IN &{channel_config_json} 773 IF '${values['name']}' == 'SELF' 774 Append To List ${channel_list} ${channel_number} 775 END 776 END 777 778 RETURN ${channel_list} 779 780 781Get Active Ethernet Channel List 782 [Documentation] Get Available channels from channel_config.json file and return as list. 783 [Arguments] ${current_channel}=${0} 784 785 # Description of Arguments 786 # ${current_channel} Current channel number. 787 788 ${valid_channel_number_interface_names}= Get Channel Number For All Interface 789 790 ${valid_channel_number_interface_name}= Get Valid Channel Number ${valid_channel_number_interface_names} 791 792 ${channel_number_list}= Get Channel Number For Valid Ethernet Interface 793 ... ${valid_channel_number_interface_name} 794 795 Return From Keyword If ${current_channel} == 0 ${channel_number_list} 796 ${channel_number_list}= Get Current Channel Name List 797 ... ${channel_number_list} ${valid_channel_number_interface_names} 798 799 RETURN ${channel_number_list} 800 801Update IP Address 802 [Documentation] Update and verify IP address of BMC. 803 [Arguments] ${ip} ${new_ip} ${netmask} ${gw_ip} 804 ... ${valid_status_codes}=[${HTTP_OK}, ${HTTP_NO_CONTENT}] 805 806 # Description of argument(s): 807 # ip IP address to be replaced (e.g. "10.7.7.7"). 808 # new_ip New IP address to be configured. 809 # netmask Netmask value. 810 # gw_ip Gateway IP address. 811 # valid_status_codes Expected return code from patch operation 812 # (e.g. "200"). See prolog of rest_request 813 # method in redfish_plus.py for details. 814 815 ${empty_dict}= Create Dictionary 816 ${patch_list}= Create List 817 ${ip_data}= Create Dictionary 818 ... Address=${new_ip} SubnetMask=${netmask} Gateway=${gw_ip} 819 820 # Find the position of IP address to be modified. 821 @{network_configurations}= Get Network Configuration 822 FOR ${network_configuration} IN @{network_configurations} 823 IF '${network_configuration['Address']}' == '${ip}' 824 Append To List ${patch_list} ${ip_data} 825 ELSE 826 Append To List ${patch_list} ${empty_dict} 827 END 828 END 829 830 # Modify the IP address only if given IP is found 831 ${ip_found}= Run Keyword And Return Status List Should Contain Value 832 ... ${patch_list} ${ip_data} msg=${ip} does not exist on BMC 833 Pass Execution If ${ip_found} == ${False} ${ip} does not exist on BMC 834 835 ${data}= Create Dictionary IPv4StaticAddresses=${patch_list} 836 837 ${active_channel_config}= Get Active Channel Config 838 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 839 840 Redfish.patch ${REDFISH_NW_ETH_IFACE}${ethernet_interface} 841 ... body=&{data} valid_status_codes=${valid_status_codes} 842 843 # Note: Network restart takes around 15-18s after patch request processing. 844 Sleep ${NETWORK_TIMEOUT}s 845 Wait For Host To Ping ${OPENBMC_HOST} ${NETWORK_TIMEOUT} 846 847 Verify IP On BMC ${new_ip} 848 Validate Network Config On BMC 849 850Get IPv4 DHCP Enabled Status 851 [Documentation] Return IPv4 DHCP enabled status from redfish URI. 852 [Arguments] ${channel_number}=${1} 853 854 # Description of argument(s): 855 # channel_number Ethernet channel number, 1 is for eth0 and 2 is for eth1 (e.g. "1"). 856 857 ${active_channel_config}= Get Active Channel Config 858 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 859 ${resp}= Redfish.Get Attribute ${REDFISH_NW_ETH_IFACE}${ethernet_interface} DHCPv4 860 Return From Keyword ${resp['DHCPEnabled']} 861 862Get DHCP IP Info 863 [Documentation] Return DHCP IP address, gateway and subnetmask from redfish URI. 864 865 ${active_channel_config}= Get Active Channel Config 866 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 867 ${resp_list}= Redfish.Get Attribute ${REDFISH_NW_ETH_IFACE}${ethernet_interface} IPv4Addresses 868 FOR ${resp} IN @{resp_list} 869 Continue For Loop If '${resp['AddressOrigin']}' != 'DHCP' 870 ${ip_addr}= Set Variable ${resp['Address']} 871 ${gateway}= Set Variable ${resp['Gateway']} 872 ${subnetmask}= Set Variable ${resp['SubnetMask']} 873 Return From Keyword ${ip_addr} ${gateway} ${subnetmask} 874 END 875 876 877DNS Test Setup Execution 878 [Documentation] Do DNS test setup execution. 879 880 Redfish.Login 881 882 ${active_channel_config}= Get Active Channel Config 883 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 884 885 ${original_nameservers}= Redfish.Get Attribute 886 ... ${REDFISH_NW_ETH_IFACE}${ethernet_interface} StaticNameServers 887 888 Rprint Vars original_nameservers 889 # Set suite variables to trigger restoration during teardown. 890 Set Suite Variable ${original_nameservers} 891 892 893Delete Static Name Servers 894 [Documentation] Delete static name servers. 895 896 DNS Test Setup Execution 897 Configure Static Name Servers static_name_servers=@{EMPTY} 898 899 # Check if all name servers deleted on BMC. 900 ${nameservers}= CLI Get Nameservers 901 Should Not Contain ${nameservers} ${original_nameservers} 902 903 DNS Test Setup Execution 904 905 Should Be Empty ${original_nameservers} 906 907 908Configure Static Name Servers 909 [Documentation] Configure DNS server on BMC. 910 [Arguments] ${static_name_servers}=${original_nameservers} 911 ... ${valid_status_codes}=${HTTP_OK} 912 913 # Description of the argument(s): 914 # static_name_servers A list of static name server IPs to be 915 # configured on the BMC. 916 # valid_status_codes Expected return code from patch operation 917 # (e.g. "200"). See prolog of rest_request 918 # method in redfish_plus.py for details. 919 920 ${active_channel_config}= Get Active Channel Config 921 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 922 923 ${type} = Evaluate type($static_name_servers).__name__ 924 ${static_name_servers}= Set Variable If '${type}'=='str' 925 ... '${static_name_servers}' ${static_name_servers} 926 927 # Currently BMC is sending 500 response code instead of 400 for invalid scenarios. 928 Redfish.Patch ${REDFISH_NW_ETH_IFACE}${ethernet_interface} 929 ... body={'StaticNameServers': ${static_name_servers}} 930 ... valid_status_codes=[${valid_status_codes}, ${HTTP_INTERNAL_SERVER_ERROR}] 931 932 # Patch operation takes 1 to 3 seconds to set new value. 933 Sleep 3s 934 935 # Check if newly added DNS server is configured on BMC. 936 CLI Get and Verify Name Servers ${static_name_servers} ${valid_status_codes} 937 938Configure MAC Settings 939 [Documentation] Configure MAC settings via Redfish. 940 [Arguments] ${mac_address} ${valid_status_code}=${HTTP_OK} 941 942 # Description of argument(s): 943 # mac_address MAC address of BMC. 944 # valid_status_code Expected response code, default is ${HTTP_OK}. 945 946 ${active_channel_config}= Get Active Channel Config 947 ${ethernet_interface}= Set Variable ${active_channel_config['${CHANNEL_NUMBER}']['name']} 948 949 Redfish.Login 950 ${payload}= Create Dictionary MACAddress=${mac_address} 951 952 Redfish.Patch ${REDFISH_NW_ETH_IFACE}${ethernet_interface} body=&{payload} 953 ... valid_status_codes=[${valid_status_code},${HTTP_NO_CONTENT},${HTTP_INTERNAL_SERVER_ERROR}] 954 955 # After any modification on network interface, BMC restarts network 956 # Note: Network restart takes around 15-18s after patch request processing. 957 Sleep ${NETWORK_TIMEOUT}s 958 959 Redfish.Get ${REDFISH_NW_ETH_IFACE}${ethernet_interface} 960 961 # Verify whether new MAC address is populated on BMC system. 962 # It should not allow to configure invalid settings. 963 ${status}= Run Keyword And Return Status 964 ... Validate MAC On BMC ${mac_address} 965 966 IF ${valid_status_code} == ${HTTP_BAD_REQUEST} 967 Should Be Equal ${status} ${False} 968 ... msg=Allowing the configuration of an invalid MAC. 969 ELSE 970 Should Be Equal ${status} ${True} 971 ... msg=Not allowing the configuration of a valid MAC. 972 END 973 974 Verify MAC Address Via FW_Env ${mac_address} ${valid_status_code} 975 976Verify MAC Address Via FW_Env 977 [Documentation] Verify MAC address on FW_Env. 978 [Arguments] ${mac_address} ${valid_status_code}=${HTTP_OK} 979 980 # Description of argument(s): 981 # mac_address MAC address of BMC. 982 # valid_status_code Expected response code, default is ${HTTP_OK}. 983 984 ${status}= Run Keyword And Return Status 985 ... Validate MAC On FW_Env ${mac_address} 986 987 IF ${valid_status_code} == ${HTTP_BAD_REQUEST} 988 Should Be Equal ${status} ${False} 989 ... msg=Allowing the configuration of an invalid MAC. 990 ELSE 991 Should Be Equal ${status} ${True} 992 ... msg=Not allowing the configuration of a valid MAC. 993 END 994