Skip to content

Latest commit

 

History

History
170 lines (108 loc) · 8.73 KB

CONTRIBUTING.md

File metadata and controls

170 lines (108 loc) · 8.73 KB

Contributing Guidelines

Cycloid Team is glad to see you contributing to this project ! In this document, we will provide you some guidelines in order to help get your contribution accepted.

Reporting an issue

Issues

When you find a bug in InfraMap, it should be reported using GitHub issues. Please define key information like your Operating System (OS), InfraMap origin (docker or from source) and finally the version you are using.

Issue Types

There are 6 types of labels, they can be used for issues or PRs:

  • enhancement: These track specific feature requests and ideas until they are completed. They can evolve from a specification or they can be submitted individually depending on the size.
  • specification: These track issues with a detailed description, this is like a proposal.
  • bug: These track bugs with the code
  • docs: These track problems with the documentation (i.e. missing or incomplete)
  • maintenance: These tracks problems, update and migration for dependencies / third-party tools
  • refactoring: These tracks internal improvement with no direct impact on the product
  • need review: this status must be set when you feel confident with your submission
  • in progress: some important change has been requested on your submission, so you can toggle from need review to in progress
  • under discussion: it's time to take a break, think about this submission and try to figure out how we can implement this or this

Submit a contribution

Setup your git repository

If you want to contribute to an existing issue, you can start by forking this repository, then clone your fork on your machine.

$ git clone https://github.com/<your-username>/inframap.git
$ cd inframap

In order to stay updated with the upstream, it's highly recommended to add cycloidio/inframap as a remote upstream.

$ git remote add upstream https://github.com/cycloidio/inframap.git

Do not forget to frequently update your fork with the upstream.

$ git fetch upstream --prune
$ git rebase upstream/master

Play with the codebase

Build from sources

Since InfraMap is a Go project, Go must be installed and configured on your machine (really ?). We currently support Go1.13+ and go modules as dependency manager. You can simply pull all necessaries dependencies by running an initial.

$ make build

This basically builds inframap with the current sources.

Add a new Provider

To add a new Provider you need to follow the provider.Provider interface and create it on the provider/{provider_name}, also add it to the provider.Type list and then run make generate. Basically you can also check any of the other providers that we already have.

To add a new test we recommend to run the inframap prune --canonicals [FILE] and add the test to the generate/testdata/ and add a new test to the generate/graph_test.go

Add a new Printer

To add a new Printer just follow the printer.Printer interface and add it to printer/{printer_name} and printer/type.go and run make generate. After that it'll be directly added to the list of available --printer.

Add new icons

Icons follow the mapping defined in tfdocs. Instead of .svg we use .png and files are located under the assets/{provider} directory with a size of 72x72. To add new icons, or regenerate the ones we have run one of the commands below (depending on the provider) and then copy the result from resize/ to the corresponding directory on the assets/{provider}. Once the new icons at the right place, run go run scripts/assets/main.go --dry (add --branch=your-branch if you did also update tfdocs and it's not yet merged) to see if anything could be removed. If so then run it without the --dry to clean the icons we do not use so we do not import extra stuff, it'll also display what's missing. Then, as a final step run make generate-icons and that's it!

If we need to regenerate some icons from Providers, I'll leave here the commands we used for them so the process is easier:

AWS

The images provided for AWS on SVG do not map the same way in PNG, so we did convert the SVG ones to PNG with inkscape

You need to be in the first level directory, rename all the & for _and_ and the for _ and then you can run the command

find SVG_Light/ -type d -exec mkdir resize/{} \; && find . -name '*.svg' -type f -exec inkscape {} -w 72 -h 72 --export-filename resize/{}.png \; && find ./resize/ -type f -name '*.svg.png' -exec sh -c 'f="{}"; mv -- "$f" "${f%.svg.png}.png"' \;

Google

Images are available in PNG format at this link: https://cloud.google.com/icons. Spaces are renamed to _. Only the resizing is required.

$ find assets/icons/google -name '*.png' -type f -exec convert {} -resize 72x72 {} \;

OpenStack

They do not provide any PNG so we convert the SVG to PNG.

You need to be in the directory in which the images are defined and have a resize/ dir.

find . -name '*-gray.svg' -type f -exec inkscape {} -w 72 -h 72 --export-filename resize/{}.png \; && find ./resize/ -type f -name '*.svg.png' -exec sh -c 'f="{}"; mv -- "$f" "${f%.svg.png}.png"' \;

Code Architecture

In this section we'll explain the main packages if you want to contribute/read/understand how InfraMap works.

Providers

The main idea behind InfraMap is to abstract each specific Terraform provider logic into a different implementation using a common interface: provider.Provider. All this implementations are inside the provider/ package.

Not all the methods are needed to be implemented and that's why we have the provider/nop_provider.go which has an empty implementation of the provider.Provider interface.

All the providers that we have implemented are defined on provider/{provider-name} ({provider-name} is the name it has on Terraform) for example provider/aws/ which holds the implementation of the AWS provider.

For more information on the provider interface we recommend to read the interface itself.

Generate and Prune

The main entry point starts with the main.go which actually calls the cmd/. The logic of each subcommand: generate and prune is separated on it's own packages

Prune

For the Prune what we do is to use the Provider.UsedAttributes as a white list of the attributes that are actually needed, drop the rest and output the pruned version of the input.

Generate

For the Generate it's a bit more complex. We want to have a common structure to operate in order to be independent of the input type (HCL or TFState), this structure is a graph.Graph.

To get the Graph from TFState we use the dependencies (or depends_on if it is older than Terraform 0.13) to generate the edges between resources so then we can have the first Graph in which each resource is a node and we have edges between them.

To get the Graph from HCL we use the interpolations, which is basically the references between resources on HCL

resource "aws_db_instance" "application" {
  identifier = "rds"

  vpc_security_group_ids = [aws_security_group.rds.id]

  tags = {
    Name = "name"
    type = "master"
    role = "rds"
  }
}

This means that aws_db_instance.application -> aws_db_instance.rds.id will have an edge between them.

This is not always enough so we have also have a Provider.PreProcess which from the JSON config (common JSON representation of the HCL or TFState configuration) can return specific edges to create between nodes. An example can be found for the Google provider https://github.com/cycloidio/inframap/blob/074e1f9bc98fb99ede7d55404b16b4d9ff9cb1e1/provider/google/google.go#L167

Once we have the common Graph, we start doing the operations:

  • Using the ResourceInOut we fix the direction of the edges, some of the first edges are wrong in terms of direction, so we fix them first. We do that by checking the config and asking the provider which are the right directions and fix them.
  • Remove the nodes that are edges on the specific provider (ex: aws_security_group is an edge on AWS). We do that by checking which is the shortest connections between two nodes (provider nodes) on the Graph and then merge all the nodes in between. The merging works by merging each node one by one and reassigning all the edges to the new node with the right direction. To keep the right direction at the end of the merge, and successive merges, we keep a weigh that is used to know how many merges have been done and in which direction (read the code for more information).

After all this we clean the end graph by removing not connected/merged edges because they are just hanging on the Graph.

When all this operations are done, we finally print the Graph using the printer/ package.