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.
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.
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 aspecification
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 codedocs
: These track problems with the documentation (i.e. missing or incomplete)maintenance
: These tracks problems, update and migration for dependencies / third-party toolsrefactoring
: These tracks internal improvement with no direct impact on the productneed review
: this status must be set when you feel confident with your submissionin progress
: some important change has been requested on your submission, so you can toggle fromneed review
toin 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
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
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.
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
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
.
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"' \;
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"' \;
In this section we'll explain the main packages if you want to contribute/read/understand how InfraMap works.
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.
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.