Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/jklaiber/cisco_restconf_example

Cisco Restconf Automation Example
https://github.com/jklaiber/cisco_restconf_example

Last synced: about 1 month ago
JSON representation

Cisco Restconf Automation Example

Awesome Lists containing this project

README

        

# Cisco Restconf Example
## What the tool accomplish
The application automatically configures features on Cisco devices.
Following configuration changes are supported:
- Hostname
- Interface
- OSPF
- BGP

Currently there is only Cisco Native support.
The features are tested on Cisco CSR1000v with IOS XE 16.06.01.

## Requirements
```bash
$ python3 –m venv path/to/desired/dir
$ source activate /bin/activate
$ pip3 install -r requirements.txt
```

## Work with it
1. Open the `device_infos.yaml` file
2. Edit the configuration to meet your needs
3. You can add additional bgp neighbors or ospf areas and also interfaces
4. When you need an additional router then you can also do that (please consider the yaml syntax)
5. Run & enjoy it.

You can find an example of `device_infos.yaml` at the end of this document.

**Run**: `python3 configuration.py`

## Code Explanation

### configuration.py

**main - Starting Point**
Load the devices and execute a HTTP PATCH request for each device.
Analyze response_code and decide if configuration change was successfully or not.
Successfull Responses are 200, 201 & 204 (more information: https://www.cisco.com/c/en/us/td/docs/ios-xml/ios/prog/configuration/166/b_166_programmability_cg/restconf_prog_int.html)
```python
def main():
devices = load_devices()
for device in devices:
logger.info
logger.info(f'Getting information for device {device["hostname"]}')
response_code = patch_configuration(device)
if( int(response_code/100) == 2): #Because successfull Response Messages are 200, 201 and 204 more info:
logger.info(f'SUCCESS: Ended device {device["hostname"]} with response code {response_code} \n')
else:
logger.warning(f'ERROR: Ended device {device["hostname"]} with response code {response_code} \n')
```

**load_devices()**
Read the input of the configuration file ```device_infos.yaml``` and return a list of dicts. Each dictionary includes the devices information.

Returns:
- List[dict]: A List of dictionaries which includes the config infos about each device

```python
def load_devices() -> List[dict]:
with open('device_infos.yaml', 'r') as host_file:
hosts = yaml.load(host_file.read(), Loader=yaml.FullLoader)
return hosts
```
**get_hostname()**
Request the hostname of a given host with a HTTP GET request and return it.

Returns:
- str: The text of the response

```python
def get_hostname(host: dict) -> str:
response = restconf_helpers.RestconfRequestHelper().get(
url=f'https://{host["connection_address"]}/restconf/data/Cisco-IOS-XE-native:native/hostname/',
username=host['username'],
password=host['password'])
return response

```
**patch_configuration()**
- Rendering the given host dict (which includes the configuration information) into a correct XML format.
- Render XML into one single lined string (because Cisco devices only support this format if writting to the native RESTAPI path)
- Configure the device with sending the information via HTTP PATCH request to the REST API.
- Calling the REST API the HTTP respone will be returned.
- If Error Code 409 is returned, the function will be executed again( maximal amount of retries are defined in MAX_RETRIES constant). If the max threshold is reached, an error message will be added to the logger, and configuration of this device will be skipped.

Parameters:
- host: Dictionary with device information which should be rendered
- retries: the current amount of retries

Returns:
- int: The response code

```python
def patch_configuration(host: dict, retries = 0) -> int:
rendering_data = rendering.RenderJinjaTemplate().rendering(host)
rendered_xml_data = rendering.XmlParser().parseXml(rendering_data)

response_code = restconf_helpers.RestconfRequestHelper().patch(
url=f'https://{host["connection_address"]}/restconf/data/Cisco-IOS-XE-native:native/',
username=host['username'],
password=host['password'],
data=rendered_xml_data)
if(ERROR_CODE_409 == response_code and retries <= MAX_RETRIES):
patch_configuration(host, retries+1)
elif(retries > MAX_RETRIES):
logger.error('409 Conflict status-line. The error-tag value object already exists is returned if used for other methods or resource types.\n 409 Client Error: Conflict')
return response_code
```

### rendering.py
Class for rendering the given jinja2 template and the XML.
Rendering XML is essential because Cisco devices only
support one single line string and no correct formatted XML.

**rendering()**
Rendering function for jinja2 which creates an own environment. In the environment open the template folder and render the template with the given infromation.

Parameters:
- config_data: Dictionary with device information which should be rendered

Returns:
- str: the rendered data

```python
def rendering(self, config_data: dict) ->str:
env = Environment(loader = FileSystemLoader('./template'),
trim_blocks=True,
lstrip_blocks=True
)
template = env.get_template('default_template.xml')
rendereddata = template.render(config_data)
return rendereddata
```
**parseXml()**
Parse a correct formatted XML file into a single line string.

Parameters:
- input_xml: XML, which should be parsed

Returns:
- str: the single line string

```python
def parseXml(self, input_xml: str) -> str:
_dom = dom.parseString(input_xml)
output_xml = ''.join([line.strip() for line in _dom.toxml().splitlines()])
_dom.unlink()
return output_xml
```
### restconf_helpers.py
Class for getting information from the device and patch configuration onto the device.

**get()**
Executes a get request to the specified url and adds RESTCONF specific headers.
Raises an exception if the request fails.

Parameters:
- url: url which should be requested
- username: the username for the authentication
- password: the password for the authentication
- restconf_format: which restconf headers should be set. (default RestconfFormat.XML)
- headers: which additional headers should be set (default None)
- kwargs: additional parameters for the request

Returns:
- str: The text of the response

```python
def get(self, url: str, username: str, password: str,
restconf_format: Optional[RestconfFormat] = RestconfFormat.XML,
headers: Optional[Dict[str, str]] = None,
**kwargs: Dict[Any, Any]) -> str:

logger.debug(f'GET request to {url}')
request_headers = self.get_headers(restconf_format, headers)
response = requests.request(method='GET', auth=(username, password),
headers=request_headers,
url=url,
verify=False,
**kwargs)
logger.debug(f'Got response from {url} with code {response.status_code} and content \n {response.text}')
response.raise_for_status()
return response.text
```

**patch()**
Executes a patch to the specified url and updates the configuration of the device.
Raises an exception if the request fail

Parameters:
- url: url which should be requested
- username: the username for the authentication
- password: the password for the authentication
- restconf_format: which restconf headers should be set. (default RestconfFormat.XML)
- headers: which additional headers should be set (default None)
- kwargs: additional parameters for the reques

Returns:
- int: The response code

```python
def patch(self, url: str, username: str, password: str,
data: str,
restconf_format: Optional[RestconfFormat] = RestconfFormat.XML,
headers: Optional[Dict[str, str]] = None,
**kwargs: Dict[Any, Any]) -> int:

logger.debug(f'PATCH request to {url}')
request_headers = self.get_headers(restconf_format, headers)
response = requests.request(method='PATCH',
auth=(username, password),
headers=request_headers,
data=data,
url=url,
verify=False,
**kwargs)
if not response.text:
logger.debug(f'Got response from {url} with code {response.status_code}')
else:
logger.debug(f'Got response from {url} with code {response.status_code} and content \n {response.text}')
response.raise_for_status()
return response.status_code
```

**get_headers()**
Adds restconf specific headers to a dict

Parameters:
- restconf: which restconf headers should be set
- headers: which additional headers should be set

Returns:
- dict: The header information

```python
def get_headers(self, format: RestconfFormat, headers: Optional[Dict[str, str]]) -> Dict[str, str]:


restconf_headers = self.headers_json if format == RestconfFormat.JSON else self.headers_xml
if headers and isinstance(headers, dict):
return dict(headers, **restconf_headers)
return dict(restconf_headers)
```

# Contributors
- Julian Klaiber ()
- Severin Dellsperger ()

# Motivation behind the solution
We decided to use a HTTP PATCH method in our application, which has following advantages:
- The application design is uncomplicated & clear. Adjustments in the configuration have to be made in just one file. It's also possible to configure several hosts. So every configuration can be made by simply adding or removing information to one single file. Therefore our solution is well structured and can be fast adoppted to new devices and functions.
- The use of the PATCH method creates as few overhead as possible. Only changes in the device configuration are updated. It also simplifies the whole application code, just one method has to be called to do one or more changes in the configuration of the device. This advantage was also noticeable during testing, writing the entire configuration took longer than just changing individual sub-configurations.
- RFC 8040 mentions following problem: If multiple update request or too big requests are send to an RESTCONF interface, the device respond with a HTTP status 409 changes couldn't be written. Our solution only needs to use one PATCH request per device, so our script should bypass the mentioned problem. If it should occur anyways, a retry is executed automatically.

# Example
This example shows how you can configure two different devices.
You can take this example and change it like you need it:
```
- hostname: RT-6
username: python
password: cisco
connection_address: 10.3.255.106
interfaces:
- name: 1
ip: 10.3.255.106
mask: 255.255.255.0
loopback: false
- name: 0
ip: 6.6.6.6
mask: 255.255.255.255
loopback: true
- name: 1
ip: 192.168.6.1
mask: 255.255.255.0
loopback: true
ospf:
process_id: 1
areas:
- area_id: 0
networks:
- 6.6.6.6: 0.0.0.0
- 192.168.6.0: 0.0.0.255
- 10.3.255.0: 0.0.0.255
router_id: 6.6.6.6
passive_interfaces:
- lo1
bgp:
as_number: 6
networks:
- 192.168.6.0: 255.255.255.0
neighbors:
- address: 20.20.20.20
remote_as: 20
multihop_count: 2
update_source:
loopback: true
id: 0
- address: 7.7.7.7
remote_as: 7
multihop_count: 3
update_source:
loopback: true
id: 0
- hostname: RT-7
username: python
password: cisco
connection_address: 10.3.255.107
interfaces:
- name: 1
ip: 10.3.255.107
mask: 255.255.255.0
loopback: false
- name: 0
ip: 7.7.7.7
mask: 255.255.255.255
loopback: true
- name: 1
ip: 192.168.7.1
mask: 255.255.255.0
loopback: true
ospf:
process_id: 1
areas:
- area_id: 0
networks:
- 7.7.7.7: 0.0.0.0
- 192.168.7.0: 0.0.0.255
- 10.3.255.0: 0.0.0.255
router_id: 7.7.7.7
passive_interfaces:
- lo1
bgp:
as_number: 7
networks:
- 192.168.7.0: 255.255.255.0
neighbors:
- address: 6.6.6.6
remote_as: 6
multihop_count: 3
update_source:
loopback: true
id: 0
- address: 20.20.20.20
remote_as: 20
multihop_count: 2
update_source:
loopback: true
id: 0
```