Skip to content

Commit

Permalink
example/resources: add tag use cases and more examples
Browse files Browse the repository at this point in the history
Change-Id: Ib138c69e6736d9ba9f38773108632d75057a9e25
  • Loading branch information
noredistribution committed Aug 25, 2020
1 parent 4676a0f commit f6cf009
Show file tree
Hide file tree
Showing 22 changed files with 270 additions and 22 deletions.
14 changes: 7 additions & 7 deletions examples/resources/event/get_events.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,9 +27,9 @@

import grpc

import arista.event.v1.event_pb2 as models
import arista.event.v1.services.gen_pb2 as messages
import arista.event.v1.services.gen_pb2_grpc as service
# import the events models and services
from arista.event.v1 import models
from arista.event.v1 import services

RPC_TIMEOUT = 30 # in seconds
SEVERITIES = ["INFO", "WARNING", "ERROR", "CRITICAL"]
Expand All @@ -46,7 +46,7 @@ def main(args):
channelCreds = grpc.ssl_channel_credentials()
connCreds = grpc.composite_channel_credentials(channelCreds, callCreds)

get_all_req = messages.EventStreamRequest()
get_all_req = services.EventStreamRequest()

if args.end and not args.start:
raise ValueError("--start must be specified when --end is specified")
Expand Down Expand Up @@ -78,13 +78,13 @@ def main(args):
print(f"selecting events that match the filter {get_all_req}")

with grpc.secure_channel(args.server, connCreds) as channel:
event_stub = service.EventServiceStub(channel)
event_ack_stub = service.EventAnnotationConfigServiceStub(channel)
event_stub = services.EventServiceStub(channel)
event_ack_stub = services.EventAnnotationConfigServiceStub(channel)
for resp in event_stub.GetAll(get_all_req, timeout=RPC_TIMEOUT):
print(f"{resp}")
if args.ack and not resp.value.ack.ack.value:
print("acking event")
req = messages.EventAnnotationConfigSetRequest(
req = services.EventAnnotationConfigSetRequest(
value=models.EventAnnotationConfig(
key=resp.value.key,
)
Expand Down
91 changes: 91 additions & 0 deletions examples/resources/event/sub_events.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
# Copyright (c) 2020 Arista Networks, Inc.
# Use of this source code is governed by the Apache License 2.0
# that can be found in the COPYING file.

# Subscribing to CVP events
#
# Examples:
# 1) Subscribe to all events
# python sub_events.py --server 10.83.12.79:8443 --token-file token.txt \
# --cert-file cvp.crt
# 2) Subscribe to only DEVICE_INTF_ERR_SMART events
# python sub_events.py --server 10.83.12.79:8443 --token-file token.txt \
# --cert-file cvp.crt --event-type DEVICE_INTF_ERR_SMART
# 3) Subscribe to events with INFO severity
# python sub_events.py --server 10.83.12.79:8443 --token-file token.txt \
# --cert-file cvp.crt --severity INFO

import argparse

import grpc

# import the events models and services
from arista.event.v1 import models
from arista.event.v1 import services

RPC_TIMEOUT = 300 # in seconds
SEVERITIES = ["INFO", "WARNING", "ERROR", "CRITICAL"]


def main(args):
# read the file containing a session token to authenticate with
token = args.token_file.read().strip()
# create the header object for the token
callCreds = grpc.access_token_call_credentials(token)

# if using a self-signed certificate (should be provided as arg)
if args.cert_file:
cert = args.cert_file.read()
channelCreds = grpc.ssl_channel_credentials(root_certificates=cert)
else:
# otherwise default to checking against CAs
channelCreds = grpc.ssl_channel_credentials()
connCreds = grpc.composite_channel_credentials(channelCreds, callCreds)
# create a stream request
subscribe = services.EventStreamRequest()

# create a filter model
event_filter = models.Event()

if args.event_type:
event_filter.event_type.value = args.event_type

if args.severity:
# enum with val 0 is always unset
event_filter.severity = SEVERITIES.index(args.severity) + 1
subscribe.partial_eq_filter.append(event_filter)
# initialize a connection to the server using our connection settings (auth + TLS)
with grpc.secure_channel(args.server, connCreds) as channel:
event_stub = services.EventServiceStub(channel)
for resp in event_stub.Subscribe(subscribe, timeout=RPC_TIMEOUT):
print(resp.value)
print("\n")


if __name__ == '__main__':
ds = ("Subscribe to CVP events. "
"Examples:\n"
"1) Subscribe to all events:\n"
"\tpython sub_events.py\n"
"2) Subscribe to only DEVICE_INTF_ERR_SMART events\n"
"\tpython sub_events.py --event-type "
"DEVICE_INTF_ERR_SMART"
"3) Subscribe to events with INFO severity:\n"
"\tpython sub_events.py --severity INFO")
parser = argparse.ArgumentParser(
description=ds,
formatter_class=argparse.RawDescriptionHelpFormatter)
parser.add_argument(
'--server',
required=True,
help="CloudVision server to connect to in <host>:<port> format")
parser.add_argument("--event-type", help="get events of this type only")
parser.add_argument("--severity",
help="get events of this severity only",
choices=SEVERITIES)
parser.add_argument("--token-file", required=True,
type=argparse.FileType('r'), help="file with access token")
parser.add_argument("--cert-file", type=argparse.FileType('rb'),
help="certificate to use as root CA")
args = parser.parse_args()
main(args)
157 changes: 157 additions & 0 deletions examples/resources/tag/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,157 @@
# Uses Cases

## Example 1) Making servers visible on Topology

As of 2020.2.0 the only supported LLDP Chassis ID TLV is subtype 4 (MAC Address). Due to this some 3rd party devices, like ESXI servers, which might advertise UUID or ifName instead of the Chassis MAC in the TLV, those devices will not be drawn on Topology, even though their LLDP states are seen in EOS. To workaround this we can tag the interfaces on the switches where these servers are connected with `lldp_chassis`, `lldp_mode` and `lldp_hostname` tags, thus forcing the values for those states.

Consider the following setup: **tacsrv34 (ens34) -- (et47) co590 -- ro224**

`co590` is receiving below LLDP information from tacsrv34:

```
co590...23:26:30(config)#show lldp nei et47 det
Interface Ethernet47 detected 1 LLDP neighbors:
Neighbor "ens34"/0050.5697.2953, age 29 seconds
Discovered 0:45:05 ago; Last changed 0:45:05 ago
- Chassis ID type: Interface name (6)
Chassis ID : "ens34"
- Port ID type: MAC address(3)
Port ID : 0050.5697.2953
- Time To Live: 120 seconds
- Port Description: "Interface 3 as ens34"
- System Name: "tacsrv34"
- System Description: "Linux tacsrv34 3.19.8-100.fc20.x86_64 #1 SMP Tue May 12 17:08:50 UTC 2015 x86_64"
- System Capabilities : Station Only
Enabled Capabilities: Station Only
- Management Address Subtype: IPv4
Management Address : 10.37.11.34
Interface Number Subtype : ifIndex (2)
Interface Number : 3
OID String :
```

As you can see the `Chassis ID TLV` has subtype 6, which is `Interface name` and has the value of `ens34` (the interface name of the linux server).

If you check topology you'll see something like below (notice that the server is not shown):

![esxi-lldp-topo-fix-fig1-before](./images/esxi-lldp-topo-fix-fig1-before.png)

On the Device page you'll see the server showing up in the LLDP table

![esxi-lldp-topo-fix-fig2-lldp-table](./images/esxi-lldp-topo-fix-fig2-lldp-table.png)

### Steps

1\. Create the tags

Create the `lldp_chassis` tag and assign the MAC address of the server as the value:
`python3 create_tag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --tag_name "lldp_chassis" --tag_value "00:50:56:97:29:53"`

If it doesn't exist already, create the `lldp_mode: default` tag as well:
`python3 create_tag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --tag_name "lldp_mode" --tag_value "default"`

(Optional) Create the `lldp_hostname` tag:

`python3 create_tag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --tag_name "lldp_hostname" --tag_value "tacsr34"`

2\. Assign the tags to the interface:
`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id JPE14252458 --interface_id Ethernet47 --tag_name "lldp_mode" --tag_value "default"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id JPE14252458 --interface_id Ethernet47 --tag_name "lldp_chassis" --tag_value "00:50:56:97:29:53"`

(Optional) `python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id JPE14252458 --interface_id Ethernet47 --tag_name "lldp_hostname" --tag_value "tacsr34"`

After this you should see the user tags and system tags added on the UI:

![esxi-lldp-topo-fix-fig3-systags](./images/esxi-lldp-topo-fix-fig3-systags.png)

And finally the server should show up in Topology:

![esxi-lldp-topo-fix-fig6-after](./images/esxi-lldp-topo-fix-fig6-after.png)

## Example 2) Devices showing up multiple times in Topology

Some devices advertise the Interface MAC instead of the Chassis MAC in the LLDP Chassis ID TLV and since those are unique addresses, each link will be detected as a different device.

In the below setup, `ar-dub-01` is connected to `n9k-dub-01` and `n9k-dub-02` on 8 interfaces (4 each). Without tagging all interfaces to use one MAC for the same neighbor Topology looks like below:

![ra-tag-g1-topo-before](./images/ra-tag-g1-topo-before.png)

### Steps

1\. To workaround this we can create the tags which will be in the format of `tag_name: tag_value`. In our case we will need two tags:
- `lldp_chassis` mapped to the chassis MAC (in my case I just used the first interface's MAC for all interfaces)
- `lldp_mode` with the value of `default`

To create the tag we can use `create_tag.py` using the following syntax:

`python3 create_tag.py --server {cvpIP}:8443 --token-file {tokenFile} --cert-file {certFile} --tag_name {value} --tag_value {value}`

e.g.:
`python3 create_tag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --tag_name "lldp_chassis" --tag_value "50:08:00:12:00:08"`

At this point you should see that there's an unassigned tag. Go to:

- **Provisioning**
- **Device Tags**
- click on the **Interface** tab
- click on **Edit Tags**

![ra-tag-g4-mg-assigned-tags](./images/ra-tag-g4-mg-assigned-tags.png)

2\. We now have to assign the tag we create to all the interfaces for the same neighbor. Before doing that you can check if there's a tag already assigned on the interfaces. Go to:
- **Provisioning**
- **Device Tags**
- click on the **Interface** tab
- select the interfaces of the device and check the user and system tags. As you can see there's no `lldp_mode` or `lldp_chassis` tag assigned:

![ra-tag-g2-no-assigned-tags](./images/ra-tag-g2-no-assigned-tags.png)

For assigning the tag we can use `itag.py` with the following syntax:
`python3 itag.py --server {cvpIP}:8443 --token-file {tokenFile} --cert-file {certFile} --device_id {deviceSN} --interface_id {interface} --tag_name {value} --tag_value {value}`

e.g.:
`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet1 --tag_name "lldp_chassis" --tag_value "50:08:00:12:00:08"`



If everything went smoothly, you should see the `lldp_chassis` tag applied on Ethernet1:

![ra-tag-g7-intf-tag-assigned](./images/ra-tag-g7-intf-tag-assigned.png)

3\. Now repeate the tag assignment for all interfaces (note that the tag will not have to be created again)

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet2 --tag_name "lldp_chassis" --tag_value "50:08:00:12:00:08"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet3 --tag_name "lldp_chassis" --tag_value "50:08:00:12:00:08"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet4 --tag_name "lldp_chassis" --tag_value "50:08:00:12:00:08"`

4\. We'll have to do the same for `lldp_mode: default`

`python3 create_tag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --tag_name "lldp_mode" --tag_value "default"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet4 --tag_name "lldp_mode" --tag_value "default"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet3 --tag_name "lldp_mode" --tag_value "default"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet2 --tag_name "lldp_mode" --tag_value "default"`

`python3 itag.py --server 10.83.12.79:8443 --token-file token.txt --cert-file cvp.crt --device_id 4B0C688DBE67D1AC9572445DCDB552F9 --interface_id Ethernet1 --tag_name "lldp_mode" --tag_value "default"`

5\. Repeat for the second box

After this you should see the tags assigned to all interfaces

![ra-tag-g7-all-intf-sys-tag-assigned](./images/ra-tag-g7-all-intf-sys-tag-assigned.png)
![ra-tag-g8-all-user-tags](./images/ra-tag-g8-all-user-tags.png)

6\. Checking topology you should now only see 2 neighbors:

![ra-tag-topo-after-final](./images/ra-tag-topo-after-final.png)





10 changes: 5 additions & 5 deletions examples/resources/tag/create_tag.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,9 @@

import grpc

import arista.tag.v1.tag_pb2 as models
import arista.tag.v1.services.gen_pb2 as messages
import arista.tag.v1.services.gen_pb2_grpc as service
# import the tags models and services
from arista.tag.v1 import models
from arista.tag.v1 import services
from google.protobuf import wrappers_pb2 as wrappers

RPC_TIMEOUT = 30 # in seconds
Expand Down Expand Up @@ -42,9 +42,9 @@ def main(args):
tag_name = args.tag_name
# initialize a connection to the server using our connection settings (auth + TLS)
with grpc.secure_channel(args.server, connCreds) as channel:
tag_stub = service.InterfaceTagConfigServiceStub(channel)
tag_stub = services.InterfaceTagConfigServiceStub(channel)

req = messages.InterfaceTagConfigSetRequest(
req = services.InterfaceTagConfigSetRequest(
value=models.InterfaceTagConfig(
key=models.TagKey(
label=wrappers.StringValue(value=tag_name),
Expand Down
10 changes: 5 additions & 5 deletions examples/resources/tag/get_tags.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,9 +32,9 @@

import grpc

import arista.tag.v1.tag_pb2 as models
import arista.tag.v1.services.gen_pb2 as messages
import arista.tag.v1.services.gen_pb2_grpc as service
# import the tags models and services
from arista.tag.v1 import models
from arista.tag.v1 import services

RPC_TIMEOUT = 30 # in seconds

Expand All @@ -56,7 +56,7 @@ def main(args):
connCreds = grpc.composite_channel_credentials(channelCreds, callCreds)

# create a stream request
get_all_req = messages.InterfaceTagAssignmentConfigStreamRequest()
get_all_req = services.InterfaceTagAssignmentConfigStreamRequest()

tag_filter = models.InterfaceTagAssignmentConfig()

Expand All @@ -74,7 +74,7 @@ def main(args):

# initialize a connection to the server using our connection settings (auth + TLS)
with grpc.secure_channel(args.server, connCreds) as channel:
tag_stub = service.InterfaceTagAssignmentConfigServiceStub(channel)
tag_stub = services.InterfaceTagAssignmentConfigServiceStub(channel)
print("Printing all tag assignments based on the filters:")
for resp in tag_stub.GetAll(get_all_req, timeout=RPC_TIMEOUT):
print(resp.value)
Expand Down
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
10 changes: 5 additions & 5 deletions examples/resources/tag/itag.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,9 +18,9 @@

import grpc

import arista.tag.v1.tag_pb2 as models
import arista.tag.v1.services.gen_pb2 as messages
import arista.tag.v1.services.gen_pb2_grpc as service
# import the tags models and services
from arista.tag.v1 import models
from arista.tag.v1 import services
from google.protobuf import wrappers_pb2 as wrappers

RPC_TIMEOUT = 30 # in seconds
Expand Down Expand Up @@ -55,8 +55,8 @@ def main(args):

# initialize a connection to the server using our connection settings (auth + TLS)
with grpc.secure_channel(args.server, connCreds) as channel:
tag_stub = service.InterfaceTagAssignmentConfigServiceStub(channel)
req = messages.InterfaceTagAssignmentConfigSetRequest(
tag_stub = services.InterfaceTagAssignmentConfigServiceStub(channel)
req = services.InterfaceTagAssignmentConfigSetRequest(
value=models.InterfaceTagAssignmentConfig(
key=models.InterfaceTagAssignmentKey(
label=wrappers.StringValue(value=tag_name),
Expand Down

0 comments on commit f6cf009

Please sign in to comment.