architectures


Namearchitectures JSON
Version 0.2.7 PyPI version JSON
download
home_pagehttps://github.com/jsoconno/architectures
SummaryTools for creating architecture as code using Python.
upload_time2021-08-11 01:34:49
maintainer
docs_urlNone
authorJustin O'Connor
requires_python>=3.7
license
keywords
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            - [Introduction](#introduction)
  - [Installation](#installation)
  - [Core Components](#core-components)
    - [Containers](#containers)
    - [Components](#components)
    - [Connections](#connections)
  - [Example Architecture](#example-architecture)
    - [Code](#code)
    - [Output](#output)
  - [How To](#how-to)
    - [Part 1 - Importing the Core Module](#part-1---importing-the-core-module)
    - [Part 2 - Creating a Graph](#part-2---creating-a-graph)
    - [Part 3 - Adding Nodes to a Graph](#part-3---adding-nodes-to-a-graph)
    - [Part 4 - Grouping Nodes on a Graph](#part-4---grouping-nodes-on-a-graph)
    - [Part 5 - Connecting Nodes and Clusters](#part-5---connecting-nodes-and-clusters)
    - [Part 6 - Using Themes](#part-6---using-themes)
    - [Part 7 - Using Providers](#part-7---using-providers)
  - [Tips and Tricks](#-tips-and-tricks)
    - [Using Colors](#-using-colors)
    - [Modifying Layout](#-modifying-layout)

## Introduction
[![license](https://img.shields.io/badge/license-MIT-blue.svg?)](/LICENSE)
![python version](https://img.shields.io/badge/python-3.7%2C%203.8%2C%203.9-blue?logo=python)
![pypi](https://img.shields.io/badge/pypi-pip%20install%20architectures-blue)
![downloads](https://img.shields.io/pypi/dm/architectures?color=blue)

[![codecov](https://codecov.io/gh/Jsoconno/architectures/branch/main/graph/badge.svg?token=BW4GBBD7Y5)](https://codecov.io/gh/Jsoconno/architectures)
[![CodeFactor](https://www.codefactor.io/repository/github/jsoconno/architectures/badge)](https://www.codefactor.io/repository/github/jsoconno/architectures)
![GitHub Workflow Status](https://img.shields.io/github/workflow/status/jsoconno/architectures/Publish%20to%20Codecov?label=tests)

![GitHub release (latest by date)](https://img.shields.io/github/v/release/jsoconno/architectures)
![commits](https://img.shields.io/github/commits-since/jsoconno/architectures/latest.svg)
![GitHub commit activity](https://img.shields.io/github/commit-activity/y/jsoconno/architectures)
![Lines of code](https://img.shields.io/tokei/lines/github/jsoconno/architectures)

![GitHub issues](https://img.shields.io/github/issues/jsoconno/architectures)
![GitHub closed issues](https://img.shields.io/github/issues-closed/jsoconno/architectures)
![GitHub pull requests](https://img.shields.io/github/issues-pr/jsoconno/architectures)
![GitHub closed pull requests](https://img.shields.io/github/issues-pr-closed/jsoconno/architectures)
![GitHub contributors](https://img.shields.io/github/contributors/jsoconno/architectures)

Today, almost everything is managed as code including applications, infrastructure, configurations, processes, and more.  Architectures was created to allow teams to manage architecture as code using Python.

The goal for this project is to make drawing architecture diagrams with code simple and to enable the use of version control systems and standard coding practices such as pull requests, code reviews, and approvals to eliminate the toil associated with diagram creation and maintenance.

Architectures comes with support for several providers out-of-the-box.

![aws provider](https://img.shields.io/badge/provider-Amazon%20Web%20Services-orange?logo=amazon-aws&color=232F3E)
![azure provider](https://img.shields.io/badge/provider-Azure-orange?logo=microsoft-azure&color=0089D6)
![gcp provider](https://img.shields.io/badge/provider-Google%20Cloud%20Platform-orange?logo=google-cloud&color=4285F4)
![kubernetes provider](https://img.shields.io/badge/provider-Kubernetes-orange?logo=kubernetes&color=326CE5)

## Installation
Architectures works for all version of python greater than version `3.7.x`.

It is recommended to use Graphviz version `2.41.x` or later.
### Install Mac OS Dependencies
1) Install Python:
```
brew install python
```
2) Install Graphviz:
```
brew install graphviz
```
### Install Windows Dependencies
1) Install Python:

[Manual Install](https://www.python.org/downloads/) or...
```
choco install python
```
2) Install Graphviz

[Manual Install](https://forum.graphviz.org/t/new-simplified-installation-procedure-on-windows/224) or...
```
choco install graphviz
```
### Download Architectures from PyPi
Architectures can be installed using the following command once all dependencies for your operating system are in place:
```
pip install architectures
```

## Core Components
There are several supported objects in the architectures library.  These objects primarily fall into three categories:
### Containers
- **Graphs** allow users to create a base canvas to create their diagram on
- **Clusters** allow users to group **Nodes** into a container
### Components
- **Nodes** allow users to create an object that represents a service
### Connections
- **Edges** allow users to draw a line between a pair or list of **Clusters** or **Nodes**
- **Flows** allow users to create a linear flow through a list of **Clusters** or **Nodes**

The Node object is extended with subclasses that allow for the creation of standard service components from various providers such as Azure, GCP, and AWS.  Read on for some examples.
## Example Architecture
Here is an example that shows how you can create a beautiful diagram using architectures.  This example is an Azure Event Driven Serverless Architecture in dark mode.

### Code
```
from architectures.core import *
from architectures.themes import LightMode, DarkMode

from architectures.providers.azure.ai import CognitiveServices
from architectures.providers.azure.application import FunctionApp, EventGridDomain
from architectures.providers.azure.data import AzureCosmosDb
from architectures.providers.azure.general import Computer
from architectures.providers.azure.storage import StorageAccountBlob

theme = DarkMode()

with Graph("Event Driven Serverless Architecture", theme=theme):
    computer = Computer("Single Page Web App")
    with Cluster():
        function_get_upload_url = FunctionApp("Function (get upload url)")
        function_list_images = FunctionApp("Function (list images)")
        function_resize = FunctionApp("Function (resize & recognize image)")
        function_write_metadata = FunctionApp("Function (write metadata)")
    blob_storage = StorageAccountBlob("Blob Storage (images, thumbnails)")
    event_grid = EventGridDomain("Event Grid (async events)")
    cosmos_db = AzureCosmosDb("Cosmos DB (image metadata)")
    cognitive_services = CognitiveServices("Cognitive Services (computer vision)")

    # Flow([function_get_upload_url, function_list_images, function_resize], color="green")
    Flow([computer, [function_get_upload_url, blob_storage, function_list_images]])
    Edge([function_get_upload_url, function_resize], blob_storage)
    Edge(blob_storage, function_resize, style="dashed")
    Edge(function_resize, [cognitive_services, event_grid])
    Edge(event_grid, function_write_metadata, style="dashed")
    Edge(function_write_metadata, cosmos_db)
    Edge(function_list_images, cosmos_db, constraint="false") # This constraint setting ensures the edge does not force ranking
    Edge(function_resize, function_write_metadata, color="invis") # This ensures alignment between these two function apps
```
### Output
!["Architecture"](assets/event-driven-serverless-architecture.png?v=4&s=200 "Architecture")

## How To
This step-by-step guide will walk you through the features and functionalities of architectures so you can start writing your own architecture as code.

It will first cover the basics, creating diagrams that are not so pretty, but demonstrate the functionality.  Additional imports will be layered on and introduced into the code to show how to achieve the desired end goal of simple and beautiful architecture diagrams.
### Part 1 - Importing the Core Module
The first thing you will need to do is to import the `core` module.  This module contains all of the core functionality required for creating architectures.

You can explicitly import only the objects that you need...
```
from architectures.core import Graph, Cluster, Node, Edge, Flow
```
... or import everything using `*`.
```
from architectures.core import *
```
For more information on these objects, see the [Core Components](##core-components) section above.
### Part 2 - Creating a Graph
Once you have imported the core module, you will need to use the `Graph` object to create the canvas that will be used for creating your architecture diagram.

This is the simplest graph you can make:
```
from architectures.core import Graph

with Graph():
    pass
```
Graph accepts the following optional arguments:
* `name (str)` - the title to give your diagram
* `output_file_name (str)` - the name of the output file
* `output_file_format (str)` - the type of file to output
* `theme (Theme object)` - the theme to apply
* `show (bool)` - whether or not to show the diagram when code is run

If no name is given, the diagram title and output file name will be set to a default value.  The default theme is the Graphviz default.  The default output file format is PNG.

### Part 3 - Adding Nodes to a Graph
Once a Graph is created, you will want to start adding objects to it.  One of the most basic objects you can create is a `Node`.
```
from architectures.core import Graph, Node

with Graph("My Graph"):
    Node()
```
You can give a Node a name.
```
    Node("My Node")
```
And pass attributes as keyword arguments to change the way things look.
```
    Node("My Node", style="dotted", color="grey")
```
You can add multiple Nodes to a Graph.
```
from architectures.core import Graph, Node

with Graph("My Graph"):
    Node("A")
    Node("B")
    Node("C")
    Node("D")
    Node("E")
```
Node objects accept the following optional arguments:
* `label (str)` - the label given to the Node
* `hide_node (bool)` - a flag to determine if a node is hidden
* `wrap_label_text (bool)` - a flag to determine if label text should be automatically wrapped
* `**attrs` - keyword arguments that are used to pass attributes to the Node to override defaults

If no label is given, the Node will appear without a label.  For provider classes (to be seen later), a default label is set based on the service being added to the canvas.  The default for word wrap is true.

For more information on available Node attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).
### Part 4 - Grouping Nodes on a Graph
No doubt you will want to be able to logically group Nodes on your architecture diagram.  You can do this with the `Cluster` object.

Node: Clusters must be contained by a Graph.

For more information on these objects, see the [Core Components](##core-components) section above.

Below is a code example of Nodes contained by Clusters on the Graph.

```
from architectures.core import Graph, Cluster, Node

with Graph("My Graph"):
    with Cluster("Cluster A"):
        Node("A")
        Node("B")
        Node("C")

    with Cluster("Cluster B"):
        Node("D")
        Node("E")

    Node("F")
```
The code above will generate a diagram that looks like this:

!["Architecture"](assets/step-2-1.png "Architecture")

Note that objects on the Graph are created in the reverse order that they are found in the code.  This is not always something you have to consider, but keep in mind it can have an impact on the final layout.  In particular for larger, more complex diagrams.

Cluster objects accept the following optional arguments:
* `label (str)` - the label given to the Cluster or Group
* `hide_border (bool)` - a flag that determines whether to make the cluster background and border invisible
* `**attrs` - keyword arguments that are used to pass attributes to the Cluster or Group to override defaults

If no label is given, Clusters will appear without a label.

For more information on available Node attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).

### Part 5 - Connecting Nodes and Clusters
A collection of Clusters and Nodes might not be all that useful by themselves.  To start defining relationships, you will want to draw lines between them.  This can be done using the `Edge` and `Flow` objects.

It is important to assign Clusters and Nodes to a variable so they can be referenced when drawing Edges or Flows.

The example below shows several different scenarios and builds on the previous example in Part 4.

```
with Graph("My Graph"):
    with Cluster("Cluster A") as cluster_a:
        a = Node("A")
        b = Node("B")
        c = Node("C")

    with Cluster("Cluster B") as cluster_b:
        d = Node("D")
        e = Node("E")

    f = Node("F")

    # This will connect nodes a and c
    Edge(a, c)

    # This will connect nodes d and e to cluster a
    Edge([d, e], cluster_a)

    # This will connect node f to cluster a and b
    Edge(f, [cluster_a, cluster_b])
```
!["Architecture"](assets/part5-1.png "Architecture")

You can also pass keyword arguments to modify attributes on an Edge.  For example:
```
...
Edge(a, c, dir="both", color="blue")
...
```

The Flow object is similar to the Edge object in that it can connect Nodes.  However, it allows for a list of objects so you can create an end-to-end flow on a diagram.
```
from architectures.core import Graph, Cluster, Node, Edge, Flow

with Graph("My Graph"):
    with Cluster("Cluster A") as cluster_a:
        a = Node("A")
        b = Node("B")
        c = Node("C")

    with Cluster("Cluster B") as cluster_b:
        d = Node("D")
        e = Node("E")

    f = Node("F")

    Flow([a, [b, c], cluster_b, f, d, e])
```
!["Architecture"](assets/step-3-4.png "Architecture")

As you can see, the Flow object is a powerful way to streamline your code when connecting multiple Node objects together in sequence.

Note that, as with Edges, the Flow object also accepts keyword arguments to set attributes.

Edge objects accept the following required arguments:
* `start_node (Cluster or Node object(s))` - the the source object(s) from where the Edge will originate
* `end_node (Cluster or Node object(s))` - the destination object(s) to where th Edge will terminate

Flow objects accept the follow required arguments:
* `nodes (list)` - a list of Cluster or Node objects to flow through

Both Edge and Flow objects support the following optional arguments:
* `**attrs` - keyword arguments that are used to pass attributes to the Edge to override defaults

For more information on available Edge attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).

### Part 6 - Using Themes
Although you can now draw any combinations of Clusters, Nodes, Edges, and Flows on your Graph, you are probably not too happy with the overall appearance of your diagram.  The Default theme is great for prototyping, but not what you would want to share with your team.

`Themes` allow you to style your architecture diagram so that it looks professional.  You can import any pre-defined themes from the themes module.  Currently supported Themes include:
* `Default` - the standard Graphviz settings for drawing diagrams
* `LightMode` - a generic light mode theme for architecture design
* `DarkMode` - like LightMode, but cooler
```
from architectures.themes import Default, LightMode, DarkMode
```
Here is our previous code example using the LightMode theme.
```
from architectures.core import Graph, Cluster, Node, Edge, Flow

# Import the LightMode theme
from architectures.themes import LightMode

theme = LightMode()

with Graph("My Graph", theme=theme):
    with Cluster("Cluster A") as cluster_a:
        a = Node("A")
        b = Node("B")
        c = Node("C")

    with Cluster("Cluster B") as cluster_b:
        d = Node("D")
        e = Node("E")

    f = Node("F")

    Flow([a, [b, c], cluster_b, f, d, e])
```
!["Architecture"](assets/part6-1.png "Architecture")

This does not look great right now because themes were meant to be used with provider objects that represent services (e.g. Storage, Firewall, Database, etc.).  We will see how this works in the next section.

Theme objects accept the following optional arguments:
* `graph_attr_overrides (dict)` - overrides to theme defaulted graph attributes
* `cluster_attr_overrides (dict)` - overrides to theme defaulted cluster attributes
* `node_attr_overrides (dict)` - overrides to theme defaulted node attributes
* `edge_attr_overrides (dict)` - overrides to theme defaulted edge attributes
* `color_overrides (list)` - overrides to theme defaulted color attributes that are used to determine what the background color of a Cluster will be

Here is an example:
```
...
# Set the theme and add overrides
theme = LightMode(
    graph_attr_overrides={"labeljust":"c", "nodesep":"0.5"}, 
    cluster_attr_overrides={"style":"dotted"},
    node_attr_overrides={"fontcolor":"dimgrey"}, 
    edge_attr_overrides={"color":"dimgrey"},
    color_overrides=["#EBF4FA" "#D7E9F5", "#C3DEEF", "#AFD3EA"]
)
...
```

For more information on available attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).

### Part 7 - Using Providers
Now that you know how to use the available core objects to draw a diagram, you will want to start using the available providers to create a real architecture diagram.  Providers represent things like Amazon Web Services, Google Cloud Platform, and Azure Cloud.  

Each service from these providers is represented by a class that inherits from the base Node class.  Architectures supports a variety of providers to allow you to create beautiful architecture diagrams across multiple platforms.  Each service object works exactly like the Node in the earlier examples.

Here is an example of how you can import services and themes to create a nice looking diagram.
```
# Import the base objects
from architectures.core import Graph, Cluster, Node, Edge, Flow

# Import the available themes
from architectures.themes import LightMode, DarkMode

# Import provider services
from architectures.providers.azure.networking import ApplicationGateway, LoadBalancer
from architectures.providers.azure.compute import VirtualMachineWindows
from architectures.providers.azure.data import DataLake

# Set the theme
theme = LightMode()

with Graph("Basic Architecture", theme=theme):
    with Cluster("Subscription"):
        with Cluster("Resource Group"):
            with Cluster("Virtual Network"):
                with Cluster("App Gateway Subnet") as app_gateway_subnet:
                    app_gateway = ApplicationGateway()
                with Cluster("Build Subnet") as build_subnet:
                    load_balancer = LoadBalancer()
                    vm_1 = VirtualMachineWindows()
                    vm_2 = VirtualMachineWindows()
                with Cluster("Data Tier Subnet") as data_tier_subnet:
                    data_lake = DataLake()

    Flow([app_gateway_subnet, load_balancer, [vm_1, vm_2]])
    Edge(build_subnet, data_lake)
```
!["Architecture"](assets/step-4-1.png "Architecture")

Here is what the same diagram would look like just by switching the theme to `DarkMode`.
```
...
# Set the theme
theme = DarkMode()
...
```
!["Architecture"](assets/step-5-2.png "Architecture")

## Tips and Tricks
### Using Colors
Colors can be used to change an objects background, border, fill, font color, and more.  You can use hex colors or the Graphviz default color scheme names.

Some great resources for finding colors to change the style of your theme are:
* [Google's Material Design Color Tool](https://material.io/resources/color/)
* [Color Hunt](https://colorhunt.co/)
* [Colormind](http://colormind.io/)

To see all all of the build-in colors, check out the [Graphviz Color Name Documentation](https://graphviz.org/doc/info/colors.html).

### Modifying Layout
There are several attributes that can have a big impact on the way your diagram looks.  Here are some of the major ones:
* `rankdir` - great for quickly changing the direction of objects layed out on your diagram
* `constraint` - helps to reduce some Edge constraints to get the layout you want
* `style` - allows you to change the way things look by making them dotted, dashed, or invisible
* `nodesep` - change the amount of separation between nodes which can have an impact on linen layout
* `ranksep` - small changes to this can have big effect on the output of your graph

In addition to attributes that you can modify, you can change the order of the objects in your configuration.  The underlying `dot` engine algorithm used for laying things out renders objects in the reverse order that they are found in your Graph.  Sometimes, reordering things can go a long way to cleaning things up.

MORE TO ADD HERE LATER WITH SOME EXAMPLES



            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/jsoconno/architectures",
    "name": "architectures",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": "",
    "keywords": "",
    "author": "Justin O'Connor",
    "author_email": "jsoconno@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/0a/80/6afd1d64ec567117624e74a585489055e9e398fda8d156ee2545ae20a125/architectures-0.2.7.tar.gz",
    "platform": "",
    "description": "- [Introduction](#introduction)\n  - [Installation](#installation)\n  - [Core Components](#core-components)\n    - [Containers](#containers)\n    - [Components](#components)\n    - [Connections](#connections)\n  - [Example Architecture](#example-architecture)\n    - [Code](#code)\n    - [Output](#output)\n  - [How To](#how-to)\n    - [Part 1 - Importing the Core Module](#part-1---importing-the-core-module)\n    - [Part 2 - Creating a Graph](#part-2---creating-a-graph)\n    - [Part 3 - Adding Nodes to a Graph](#part-3---adding-nodes-to-a-graph)\n    - [Part 4 - Grouping Nodes on a Graph](#part-4---grouping-nodes-on-a-graph)\n    - [Part 5 - Connecting Nodes and Clusters](#part-5---connecting-nodes-and-clusters)\n    - [Part 6 - Using Themes](#part-6---using-themes)\n    - [Part 7 - Using Providers](#part-7---using-providers)\n  - [Tips and Tricks](#-tips-and-tricks)\n    - [Using Colors](#-using-colors)\n    - [Modifying Layout](#-modifying-layout)\n\n## Introduction\n[![license](https://img.shields.io/badge/license-MIT-blue.svg?)](/LICENSE)\n![python version](https://img.shields.io/badge/python-3.7%2C%203.8%2C%203.9-blue?logo=python)\n![pypi](https://img.shields.io/badge/pypi-pip%20install%20architectures-blue)\n![downloads](https://img.shields.io/pypi/dm/architectures?color=blue)\n\n[![codecov](https://codecov.io/gh/Jsoconno/architectures/branch/main/graph/badge.svg?token=BW4GBBD7Y5)](https://codecov.io/gh/Jsoconno/architectures)\n[![CodeFactor](https://www.codefactor.io/repository/github/jsoconno/architectures/badge)](https://www.codefactor.io/repository/github/jsoconno/architectures)\n![GitHub Workflow Status](https://img.shields.io/github/workflow/status/jsoconno/architectures/Publish%20to%20Codecov?label=tests)\n\n![GitHub release (latest by date)](https://img.shields.io/github/v/release/jsoconno/architectures)\n![commits](https://img.shields.io/github/commits-since/jsoconno/architectures/latest.svg)\n![GitHub commit activity](https://img.shields.io/github/commit-activity/y/jsoconno/architectures)\n![Lines of code](https://img.shields.io/tokei/lines/github/jsoconno/architectures)\n\n![GitHub issues](https://img.shields.io/github/issues/jsoconno/architectures)\n![GitHub closed issues](https://img.shields.io/github/issues-closed/jsoconno/architectures)\n![GitHub pull requests](https://img.shields.io/github/issues-pr/jsoconno/architectures)\n![GitHub closed pull requests](https://img.shields.io/github/issues-pr-closed/jsoconno/architectures)\n![GitHub contributors](https://img.shields.io/github/contributors/jsoconno/architectures)\n\nToday, almost everything is managed as code including applications, infrastructure, configurations, processes, and more.  Architectures was created to allow teams to manage architecture as code using Python.\n\nThe goal for this project is to make drawing architecture diagrams with code simple and to enable the use of version control systems and standard coding practices such as pull requests, code reviews, and approvals to eliminate the toil associated with diagram creation and maintenance.\n\nArchitectures comes with support for several providers out-of-the-box.\n\n![aws provider](https://img.shields.io/badge/provider-Amazon%20Web%20Services-orange?logo=amazon-aws&color=232F3E)\n![azure provider](https://img.shields.io/badge/provider-Azure-orange?logo=microsoft-azure&color=0089D6)\n![gcp provider](https://img.shields.io/badge/provider-Google%20Cloud%20Platform-orange?logo=google-cloud&color=4285F4)\n![kubernetes provider](https://img.shields.io/badge/provider-Kubernetes-orange?logo=kubernetes&color=326CE5)\n\n## Installation\nArchitectures works for all version of python greater than version `3.7.x`.\n\nIt is recommended to use Graphviz version `2.41.x` or later.\n### Install Mac OS Dependencies\n1) Install Python:\n```\nbrew install python\n```\n2) Install Graphviz:\n```\nbrew install graphviz\n```\n### Install Windows Dependencies\n1) Install Python:\n\n[Manual Install](https://www.python.org/downloads/) or...\n```\nchoco install python\n```\n2) Install Graphviz\n\n[Manual Install](https://forum.graphviz.org/t/new-simplified-installation-procedure-on-windows/224) or...\n```\nchoco install graphviz\n```\n### Download Architectures from PyPi\nArchitectures can be installed using the following command once all dependencies for your operating system are in place:\n```\npip install architectures\n```\n\n## Core Components\nThere are several supported objects in the architectures library.  These objects primarily fall into three categories:\n### Containers\n- **Graphs** allow users to create a base canvas to create their diagram on\n- **Clusters** allow users to group **Nodes** into a container\n### Components\n- **Nodes** allow users to create an object that represents a service\n### Connections\n- **Edges** allow users to draw a line between a pair or list of **Clusters** or **Nodes**\n- **Flows** allow users to create a linear flow through a list of **Clusters** or **Nodes**\n\nThe Node object is extended with subclasses that allow for the creation of standard service components from various providers such as Azure, GCP, and AWS.  Read on for some examples.\n## Example Architecture\nHere is an example that shows how you can create a beautiful diagram using architectures.  This example is an Azure Event Driven Serverless Architecture in dark mode.\n\n### Code\n```\nfrom architectures.core import *\nfrom architectures.themes import LightMode, DarkMode\n\nfrom architectures.providers.azure.ai import CognitiveServices\nfrom architectures.providers.azure.application import FunctionApp, EventGridDomain\nfrom architectures.providers.azure.data import AzureCosmosDb\nfrom architectures.providers.azure.general import Computer\nfrom architectures.providers.azure.storage import StorageAccountBlob\n\ntheme = DarkMode()\n\nwith Graph(\"Event Driven Serverless Architecture\", theme=theme):\n    computer = Computer(\"Single Page Web App\")\n    with Cluster():\n        function_get_upload_url = FunctionApp(\"Function (get upload url)\")\n        function_list_images = FunctionApp(\"Function (list images)\")\n        function_resize = FunctionApp(\"Function (resize & recognize image)\")\n        function_write_metadata = FunctionApp(\"Function (write metadata)\")\n    blob_storage = StorageAccountBlob(\"Blob Storage (images, thumbnails)\")\n    event_grid = EventGridDomain(\"Event Grid (async events)\")\n    cosmos_db = AzureCosmosDb(\"Cosmos DB (image metadata)\")\n    cognitive_services = CognitiveServices(\"Cognitive Services (computer vision)\")\n\n    # Flow([function_get_upload_url, function_list_images, function_resize], color=\"green\")\n    Flow([computer, [function_get_upload_url, blob_storage, function_list_images]])\n    Edge([function_get_upload_url, function_resize], blob_storage)\n    Edge(blob_storage, function_resize, style=\"dashed\")\n    Edge(function_resize, [cognitive_services, event_grid])\n    Edge(event_grid, function_write_metadata, style=\"dashed\")\n    Edge(function_write_metadata, cosmos_db)\n    Edge(function_list_images, cosmos_db, constraint=\"false\") # This constraint setting ensures the edge does not force ranking\n    Edge(function_resize, function_write_metadata, color=\"invis\") # This ensures alignment between these two function apps\n```\n### Output\n![\"Architecture\"](assets/event-driven-serverless-architecture.png?v=4&s=200 \"Architecture\")\n\n## How To\nThis step-by-step guide will walk you through the features and functionalities of architectures so you can start writing your own architecture as code.\n\nIt will first cover the basics, creating diagrams that are not so pretty, but demonstrate the functionality.  Additional imports will be layered on and introduced into the code to show how to achieve the desired end goal of simple and beautiful architecture diagrams.\n### Part 1 - Importing the Core Module\nThe first thing you will need to do is to import the `core` module.  This module contains all of the core functionality required for creating architectures.\n\nYou can explicitly import only the objects that you need...\n```\nfrom architectures.core import Graph, Cluster, Node, Edge, Flow\n```\n... or import everything using `*`.\n```\nfrom architectures.core import *\n```\nFor more information on these objects, see the [Core Components](##core-components) section above.\n### Part 2 - Creating a Graph\nOnce you have imported the core module, you will need to use the `Graph` object to create the canvas that will be used for creating your architecture diagram.\n\nThis is the simplest graph you can make:\n```\nfrom architectures.core import Graph\n\nwith Graph():\n    pass\n```\nGraph accepts the following optional arguments:\n* `name (str)` - the title to give your diagram\n* `output_file_name (str)` - the name of the output file\n* `output_file_format (str)` - the type of file to output\n* `theme (Theme object)` - the theme to apply\n* `show (bool)` - whether or not to show the diagram when code is run\n\nIf no name is given, the diagram title and output file name will be set to a default value.  The default theme is the Graphviz default.  The default output file format is PNG.\n\n### Part 3 - Adding Nodes to a Graph\nOnce a Graph is created, you will want to start adding objects to it.  One of the most basic objects you can create is a `Node`.\n```\nfrom architectures.core import Graph, Node\n\nwith Graph(\"My Graph\"):\n    Node()\n```\nYou can give a Node a name.\n```\n    Node(\"My Node\")\n```\nAnd pass attributes as keyword arguments to change the way things look.\n```\n    Node(\"My Node\", style=\"dotted\", color=\"grey\")\n```\nYou can add multiple Nodes to a Graph.\n```\nfrom architectures.core import Graph, Node\n\nwith Graph(\"My Graph\"):\n    Node(\"A\")\n    Node(\"B\")\n    Node(\"C\")\n    Node(\"D\")\n    Node(\"E\")\n```\nNode objects accept the following optional arguments:\n* `label (str)` - the label given to the Node\n* `hide_node (bool)` - a flag to determine if a node is hidden\n* `wrap_label_text (bool)` - a flag to determine if label text should be automatically wrapped\n* `**attrs` - keyword arguments that are used to pass attributes to the Node to override defaults\n\nIf no label is given, the Node will appear without a label.  For provider classes (to be seen later), a default label is set based on the service being added to the canvas.  The default for word wrap is true.\n\nFor more information on available Node attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).\n### Part 4 - Grouping Nodes on a Graph\nNo doubt you will want to be able to logically group Nodes on your architecture diagram.  You can do this with the `Cluster` object.\n\nNode: Clusters must be contained by a Graph.\n\nFor more information on these objects, see the [Core Components](##core-components) section above.\n\nBelow is a code example of Nodes contained by Clusters on the Graph.\n\n```\nfrom architectures.core import Graph, Cluster, Node\n\nwith Graph(\"My Graph\"):\n    with Cluster(\"Cluster A\"):\n        Node(\"A\")\n        Node(\"B\")\n        Node(\"C\")\n\n    with Cluster(\"Cluster B\"):\n        Node(\"D\")\n        Node(\"E\")\n\n    Node(\"F\")\n```\nThe code above will generate a diagram that looks like this:\n\n![\"Architecture\"](assets/step-2-1.png \"Architecture\")\n\nNote that objects on the Graph are created in the reverse order that they are found in the code.  This is not always something you have to consider, but keep in mind it can have an impact on the final layout.  In particular for larger, more complex diagrams.\n\nCluster objects accept the following optional arguments:\n* `label (str)` - the label given to the Cluster or Group\n* `hide_border (bool)` - a flag that determines whether to make the cluster background and border invisible\n* `**attrs` - keyword arguments that are used to pass attributes to the Cluster or Group to override defaults\n\nIf no label is given, Clusters will appear without a label.\n\nFor more information on available Node attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).\n\n### Part 5 - Connecting Nodes and Clusters\nA collection of Clusters and Nodes might not be all that useful by themselves.  To start defining relationships, you will want to draw lines between them.  This can be done using the `Edge` and `Flow` objects.\n\nIt is important to assign Clusters and Nodes to a variable so they can be referenced when drawing Edges or Flows.\n\nThe example below shows several different scenarios and builds on the previous example in Part 4.\n\n```\nwith Graph(\"My Graph\"):\n    with Cluster(\"Cluster A\") as cluster_a:\n        a = Node(\"A\")\n        b = Node(\"B\")\n        c = Node(\"C\")\n\n    with Cluster(\"Cluster B\") as cluster_b:\n        d = Node(\"D\")\n        e = Node(\"E\")\n\n    f = Node(\"F\")\n\n    # This will connect nodes a and c\n    Edge(a, c)\n\n    # This will connect nodes d and e to cluster a\n    Edge([d, e], cluster_a)\n\n    # This will connect node f to cluster a and b\n    Edge(f, [cluster_a, cluster_b])\n```\n![\"Architecture\"](assets/part5-1.png \"Architecture\")\n\nYou can also pass keyword arguments to modify attributes on an Edge.  For example:\n```\n...\nEdge(a, c, dir=\"both\", color=\"blue\")\n...\n```\n\nThe Flow object is similar to the Edge object in that it can connect Nodes.  However, it allows for a list of objects so you can create an end-to-end flow on a diagram.\n```\nfrom architectures.core import Graph, Cluster, Node, Edge, Flow\n\nwith Graph(\"My Graph\"):\n    with Cluster(\"Cluster A\") as cluster_a:\n        a = Node(\"A\")\n        b = Node(\"B\")\n        c = Node(\"C\")\n\n    with Cluster(\"Cluster B\") as cluster_b:\n        d = Node(\"D\")\n        e = Node(\"E\")\n\n    f = Node(\"F\")\n\n    Flow([a, [b, c], cluster_b, f, d, e])\n```\n![\"Architecture\"](assets/step-3-4.png \"Architecture\")\n\nAs you can see, the Flow object is a powerful way to streamline your code when connecting multiple Node objects together in sequence.\n\nNote that, as with Edges, the Flow object also accepts keyword arguments to set attributes.\n\nEdge objects accept the following required arguments:\n* `start_node (Cluster or Node object(s))` - the the source object(s) from where the Edge will originate\n* `end_node (Cluster or Node object(s))` - the destination object(s) to where th Edge will terminate\n\nFlow objects accept the follow required arguments:\n* `nodes (list)` - a list of Cluster or Node objects to flow through\n\nBoth Edge and Flow objects support the following optional arguments:\n* `**attrs` - keyword arguments that are used to pass attributes to the Edge to override defaults\n\nFor more information on available Edge attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).\n\n### Part 6 - Using Themes\nAlthough you can now draw any combinations of Clusters, Nodes, Edges, and Flows on your Graph, you are probably not too happy with the overall appearance of your diagram.  The Default theme is great for prototyping, but not what you would want to share with your team.\n\n`Themes` allow you to style your architecture diagram so that it looks professional.  You can import any pre-defined themes from the themes module.  Currently supported Themes include:\n* `Default` - the standard Graphviz settings for drawing diagrams\n* `LightMode` - a generic light mode theme for architecture design\n* `DarkMode` - like LightMode, but cooler\n```\nfrom architectures.themes import Default, LightMode, DarkMode\n```\nHere is our previous code example using the LightMode theme.\n```\nfrom architectures.core import Graph, Cluster, Node, Edge, Flow\n\n# Import the LightMode theme\nfrom architectures.themes import LightMode\n\ntheme = LightMode()\n\nwith Graph(\"My Graph\", theme=theme):\n    with Cluster(\"Cluster A\") as cluster_a:\n        a = Node(\"A\")\n        b = Node(\"B\")\n        c = Node(\"C\")\n\n    with Cluster(\"Cluster B\") as cluster_b:\n        d = Node(\"D\")\n        e = Node(\"E\")\n\n    f = Node(\"F\")\n\n    Flow([a, [b, c], cluster_b, f, d, e])\n```\n![\"Architecture\"](assets/part6-1.png \"Architecture\")\n\nThis does not look great right now because themes were meant to be used with provider objects that represent services (e.g. Storage, Firewall, Database, etc.).  We will see how this works in the next section.\n\nTheme objects accept the following optional arguments:\n* `graph_attr_overrides (dict)` - overrides to theme defaulted graph attributes\n* `cluster_attr_overrides (dict)` - overrides to theme defaulted cluster attributes\n* `node_attr_overrides (dict)` - overrides to theme defaulted node attributes\n* `edge_attr_overrides (dict)` - overrides to theme defaulted edge attributes\n* `color_overrides (list)` - overrides to theme defaulted color attributes that are used to determine what the background color of a Cluster will be\n\nHere is an example:\n```\n...\n# Set the theme and add overrides\ntheme = LightMode(\n    graph_attr_overrides={\"labeljust\":\"c\", \"nodesep\":\"0.5\"}, \n    cluster_attr_overrides={\"style\":\"dotted\"},\n    node_attr_overrides={\"fontcolor\":\"dimgrey\"}, \n    edge_attr_overrides={\"color\":\"dimgrey\"},\n    color_overrides=[\"#EBF4FA\" \"#D7E9F5\", \"#C3DEEF\", \"#AFD3EA\"]\n)\n...\n```\n\nFor more information on available attributes, check out the [Graphviz Attributes Documentation](https://graphviz.org/doc/info/attrs.html).\n\n### Part 7 - Using Providers\nNow that you know how to use the available core objects to draw a diagram, you will want to start using the available providers to create a real architecture diagram.  Providers represent things like Amazon Web Services, Google Cloud Platform, and Azure Cloud.  \n\nEach service from these providers is represented by a class that inherits from the base Node class.  Architectures supports a variety of providers to allow you to create beautiful architecture diagrams across multiple platforms.  Each service object works exactly like the Node in the earlier examples.\n\nHere is an example of how you can import services and themes to create a nice looking diagram.\n```\n# Import the base objects\nfrom architectures.core import Graph, Cluster, Node, Edge, Flow\n\n# Import the available themes\nfrom architectures.themes import LightMode, DarkMode\n\n# Import provider services\nfrom architectures.providers.azure.networking import ApplicationGateway, LoadBalancer\nfrom architectures.providers.azure.compute import VirtualMachineWindows\nfrom architectures.providers.azure.data import DataLake\n\n# Set the theme\ntheme = LightMode()\n\nwith Graph(\"Basic Architecture\", theme=theme):\n    with Cluster(\"Subscription\"):\n        with Cluster(\"Resource Group\"):\n            with Cluster(\"Virtual Network\"):\n                with Cluster(\"App Gateway Subnet\") as app_gateway_subnet:\n                    app_gateway = ApplicationGateway()\n                with Cluster(\"Build Subnet\") as build_subnet:\n                    load_balancer = LoadBalancer()\n                    vm_1 = VirtualMachineWindows()\n                    vm_2 = VirtualMachineWindows()\n                with Cluster(\"Data Tier Subnet\") as data_tier_subnet:\n                    data_lake = DataLake()\n\n    Flow([app_gateway_subnet, load_balancer, [vm_1, vm_2]])\n    Edge(build_subnet, data_lake)\n```\n![\"Architecture\"](assets/step-4-1.png \"Architecture\")\n\nHere is what the same diagram would look like just by switching the theme to `DarkMode`.\n```\n...\n# Set the theme\ntheme = DarkMode()\n...\n```\n![\"Architecture\"](assets/step-5-2.png \"Architecture\")\n\n## Tips and Tricks\n### Using Colors\nColors can be used to change an objects background, border, fill, font color, and more.  You can use hex colors or the Graphviz default color scheme names.\n\nSome great resources for finding colors to change the style of your theme are:\n* [Google's Material Design Color Tool](https://material.io/resources/color/)\n* [Color Hunt](https://colorhunt.co/)\n* [Colormind](http://colormind.io/)\n\nTo see all all of the build-in colors, check out the [Graphviz Color Name Documentation](https://graphviz.org/doc/info/colors.html).\n\n### Modifying Layout\nThere are several attributes that can have a big impact on the way your diagram looks.  Here are some of the major ones:\n* `rankdir` - great for quickly changing the direction of objects layed out on your diagram\n* `constraint` - helps to reduce some Edge constraints to get the layout you want\n* `style` - allows you to change the way things look by making them dotted, dashed, or invisible\n* `nodesep` - change the amount of separation between nodes which can have an impact on linen layout\n* `ranksep` - small changes to this can have big effect on the output of your graph\n\nIn addition to attributes that you can modify, you can change the order of the objects in your configuration.  The underlying `dot` engine algorithm used for laying things out renders objects in the reverse order that they are found in your Graph.  Sometimes, reordering things can go a long way to cleaning things up.\n\nMORE TO ADD HERE LATER WITH SOME EXAMPLES\n\n\n",
    "bugtrack_url": null,
    "license": "",
    "summary": "Tools for creating architecture as code using Python.",
    "version": "0.2.7",
    "project_urls": {
        "Homepage": "https://github.com/jsoconno/architectures"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "d941c5526529166fb3648df38bf3379b404655062666adb2cc5ac6daad367d9e",
                "md5": "c522667a909ffb28c5311d22e4fb580c",
                "sha256": "aebb0fb90b6cd414eba48bf647fcca72fee5165f21eb385adaabcec9dd58ac90"
            },
            "downloads": -1,
            "filename": "architectures-0.2.7-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "c522667a909ffb28c5311d22e4fb580c",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 7919486,
            "upload_time": "2021-08-11T01:34:46",
            "upload_time_iso_8601": "2021-08-11T01:34:46.794515Z",
            "url": "https://files.pythonhosted.org/packages/d9/41/c5526529166fb3648df38bf3379b404655062666adb2cc5ac6daad367d9e/architectures-0.2.7-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "0a806afd1d64ec567117624e74a585489055e9e398fda8d156ee2545ae20a125",
                "md5": "e47d63a0f9bc0808d8f391b6ef8fbd14",
                "sha256": "039680fae1bfce37de6a38224aa8828faa55c000cfd5c6fc16424ef1c683cc82"
            },
            "downloads": -1,
            "filename": "architectures-0.2.7.tar.gz",
            "has_sig": false,
            "md5_digest": "e47d63a0f9bc0808d8f391b6ef8fbd14",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 7428602,
            "upload_time": "2021-08-11T01:34:49",
            "upload_time_iso_8601": "2021-08-11T01:34:49.285939Z",
            "url": "https://files.pythonhosted.org/packages/0a/80/6afd1d64ec567117624e74a585489055e9e398fda8d156ee2545ae20a125/architectures-0.2.7.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2021-08-11 01:34:49",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "jsoconno",
    "github_project": "architectures",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "architectures"
}
        
Elapsed time: 4.46224s