Skip to content

our-learning-path/github-management-terraform-workshop

Repository files navigation

GitHub Management Terraform Demo

This project is a Terraform project that creates resources in GitHub.

Local Environment Setup

These are the instructions for setting up the local environment and running the project on your machine.

Prerequisites

Make sure you have the following prerequisites installed on your system:

Configuration Steps

  1. Clone the Repository:

    Clone this repository to your local machine using Git:

    git clone https://github.com/erasmolpa-smart-org/github-management-terraform-demo.git
  2. Set GitHub Provider Configuration:

    Before applying the Terraform configuration, you need to set the GitHub provider configuration. Open the main.tf file and configure the GitHub provider block:

    # Configure the GitHub Provider
    provider "github" {
      owner = var.github_owner
      token = var.github_token
    }

    Replace var.github_owner and var.github_token with your GitHub owner and access token respectively. It's recommended to use environment variables or other secure methods to manage sensitive information like access tokens.

YAML Files in the config Folder

In the config folder of this project, you will find several YAML files containing input configuration for creating resources on GitHub. These YAML files are used as input for the Terraform process and define organization members, teams, repositories, and associated permissions.

  • members.yaml: Defines organization members using their GitHub usernames and assigns them specific roles.

    members:
      - username: "erasmolpa"
        role: "member"
      - username: "erasmodominguezdc"
        role: "member"
  • repositories.yaml: Defines the repositories to be created in the GitHub organization, specifying their name, description, and visibility.

    organization_repositories:
    - name: "myorganization-service-api"
        description: "Main business Service API"
        visibility: private
    - name: "myorganization-service-ui"
        description: "Front Service UI"
        visibility: private
    - name: "myorganization-service-sdk"
        description: "Backend core SDK Framework"
        visibility: public
  • teams.yaml: Defines teams in GitHub and associates a list of members with each team.

    teams:
    frontend_team:
        name: "frontend_team"
        description: "Team for frontend developers"
        privacy: "closed"
        members:
        - username: "erasmolpa"
            role: "member"
    backend_team:
        name: "backend_team"
        description: "Team for backend developers"
        privacy: "closed"
        members:
        - username: "erasmodominguezdc"
            role: "member"
        - username: "erasmolpa"
            role: "member"
  • repository_permissions.yaml: Defines access permissions to the repositories for different teams.

    repositories:
    - name: "myorganization-service-api"
        teams:
        - name: "backend_team"
            permission: "admin"
        - name: "frontend_team"
            permission: "pull"
    - name: "myorganization-service-ui"
        teams:
        - name: "frontend_team"
            permission: "admin"
    - name: "myorganization-service-sdk"
        teams:
        - name: "frontend_team"
            permission: "push"
        - name: "backend_team"
            permission: "push"

This configuration is load as terraform locals as bellow:

    locals {
    repositories_data                = yamldecode(file("${path.module}/config/repositories.yaml"))
    members_data                     = yamldecode(file("${path.module}/config/members.yaml"))
    teams_data                       = yamldecode(file("${path.module}/config/teams.yaml"))
    teams_repository_permission_data = yamldecode(file("${path.module}/config/repository_permissions.yaml"))

    team_repository_permissions = flatten([
        for repository in local.teams_repository_permission_data.repositories :
        [
        for team in repository.teams :
        {
            repository_name = repository.name
            team_name       = team.name
            permission      = team.permission
        }
        ]
    ])
    }

NOTE: Please, remember this is just a first MVP or POC. For a production readiness implementation, we will need to modify this approach

Technical Notes: Assumptions and Future Improvements

Assumptions:

  1. Manual Repository Management: The current infrastructure involves manually managing all code repositories within the GitHub organization. This manual approach has led to inconsistencies in configurations across repositories.

  2. Infrastructure as Code (IaC) Adoption: The decision to adopt an Infrastructure as Code approach aims to ensure consistency in repository configurations and streamline the management of users within the GitHub organization.

  3. Initial Implementation: The initial implementation focuses on creating repositories, managing team memberships, and assigning permissions within the GitHub organization.

  4. GitHub Organization Membership: The GitHub organization does not have a centralized identity service configured. As a result, users will be added directly to the GitHub organization.

  5. Team Structure: Developers are divided into two teams: Frontend and Backend. Each team has specific access levels to repositories based on their role.

Future Improvements:

Future improvements are based on the assumption that what is intended is to manage and provide centralized governance of a GitHub organization(s) in a large company. The implementation and strategy to follow can vary greatly depending on the distribution and size of said organization. Therefore, future improvements are a rough proposal of what could be taken into account.

  1. Centralized State Management:

    • Description: Implementing a centralized backend for storing Terraform state files will enhance collaboration and ensure consistency in infrastructure deployments.
    • Technical Approach: Use remote state management with separate state files for each resource type (e.g., repositories, teams, members).
    terraform {
    backend "s3" {
        bucket         = "smart-org-storage"
        key            = "repositories.tfstate"
        region         = "us-west-1"
        dynamodb_table = "repo-state-locked-table"
    }
    }

NOTE: Depending on the company size and the number of resources to be managed under IaC, we will need to split the states into multiple of them. For example, if we are expecting to create Thousands of repositories, will be not possible to maintain a single repository state. We can split by teams, (thousands of state files) or by any business unit. Another important aspect to be considered is the way we provide this IaC to the teams. Wrappers such as Terragrunt, can help a lot to simplify the way we manage environments and to increase the usability

  1. Modular Terraform Configuration:

    • Description: Breaking down the Terraform configuration into separate modules for each resource type (e.g., repositories, teams) will improve code organization and maintainability.
    • Technical Approach: Create individual Terraform modules for managing repositories, teams, and memberships with reusable configurations.
    module "repositories" {
    source = "git::https://github.com/erasmolpa-smart-org/github-repositories-terraform-module.git?ref=v1.0.0"
    ...
    .....
    .......
    }
    
    module "teams" {
    source = "git::https://github.com/erasmolpa-smart-org/github-teams-terraform-module.git?ref=v1.0.0"
    ....
    .....
    .......
    }
    
    module "members" {
    source = "git::https://github.com/erasmolpa-smart-org/github-members-terraform-module.git?ref=v1.0.0"
    ....
    .....
    .......
    }
  2. Governance:

    • Description: Enhancing the solution to support governance requirements, such as implementing policies for repository naming convention, defining code owners for repositories and including some security compliance settings by default.
    • Technical Approach: Define naming conventions using Terraform variables and implement code owner's configuration using GitHub's CODEOWNERS file.
    vulnerability_alerts = true
    
    security_and_analysis {
    secret_scanning {
        status = "disabled"
    }
    secret_scanning_push_protection {
        status = "disabled"
    }
    }
    
    resource "github_branch_protection" "github_repo-branch-protection" {
    for_each               = toset([for repo in local.repositories_data.organization_repositories : repo.name])
    repository_id          = github_repository.github_repo[each.key].node_id
    pattern                = "main"
    enforce_admins         = true
    allows_deletions       = false
    require_signed_commits = true
    }
    
    resource "github_repository_tag_protection" "github_repo-tag-protection" {
    for_each   = toset([for repo in local.repositories_data.organization_repositories : repo.name])
    repository = github_repository.github_repo[each.key].name
    pattern    = "v*"
    }
  3. Self-Service Provisioning:

    • Description: Implementing self-service capabilities for developers to request repository creation and manage access permissions through an automated process.
    • Technical Approach: Develop a custom web application or use GitHub Actions to automate repository creation and permission management based on user requests.
    repositories_data  = yamldecode(file("${path.module}/config/repositories.yaml"))
    
    module "repositories" {
    
    source             = "git::https://github.com/erasmolpa-smart-org/github-repositories-terraform-module.git?ref=v1.0.0"
    
    ...
    .....
    .......
    }

    and the team repositories

    repositories:
    - name: "myorganization-service-api"
        teams:
        - name: "backend_team"
            permission: "admin"
        - name: "frontend_team"
            permission: "pull"
    - name: "myorganization-service-ui"
        teams:
        - name: "frontend_team"
            permission: "admin"
    - name: "myorganization-service-sdk"
        teams:
        - name: "frontend_team"
            permission: "push"
        - name: "backend_team"
            permission: "push"
  4. Import Existing Resources:

    • Description: Adding functionality to import existing repositories, teams, and user memberships from the GitHub organization before fully implementing Terraform. This will ensure a seamless transition and preserve existing configurations.
    • Technical Approach: Use Terraform's import feature to import existing resources into Terraform state files, allowing for further management using Terraform.

workaround

Releases

No releases published

Packages

No packages published

Languages