From b90d272f6f82f33ee1f1c9579f68df6ae6d3684d Mon Sep 17 00:00:00 2001 From: Victor Palma Date: Sat, 16 Nov 2024 12:47:08 -0600 Subject: [PATCH] Add blog post: Getting Started with Pulumi and OpenStack Flex --- docs/blog/.authors.yml | 7 + ...-started-with-pulumi-and-openstack-flex.md | 376 ++++++++++++++++++ .../posts/assets/images/2024-11-15/pulumi.png | Bin 0 -> 8529 bytes 3 files changed, 383 insertions(+) create mode 100644 docs/blog/posts/2024-11-15-getting-started-with-pulumi-and-openstack-flex.md create mode 100644 docs/blog/posts/assets/images/2024-11-15/pulumi.png diff --git a/docs/blog/.authors.yml b/docs/blog/.authors.yml index d33b328..f7b81ad 100644 --- a/docs/blog/.authors.yml +++ b/docs/blog/.authors.yml @@ -48,3 +48,10 @@ authors: github: sulochan twitter: sulo linkedin: sulochan + devx: + name: Victor Palma + description: Husban, Dad, Friend, Leader, Coffee Lover. + avatar: https://github.com/devx.png + github: devx + twitter: devx + linkedin: victorpalma diff --git a/docs/blog/posts/2024-11-15-getting-started-with-pulumi-and-openstack-flex.md b/docs/blog/posts/2024-11-15-getting-started-with-pulumi-and-openstack-flex.md new file mode 100644 index 0000000..a9beb52 --- /dev/null +++ b/docs/blog/posts/2024-11-15-getting-started-with-pulumi-and-openstack-flex.md @@ -0,0 +1,376 @@ +--- +title: Getting Started with Pulumi and OpenStack Flex +date: 2024-11-08 +authors: + - devx +description: > + Getting Started with Pulumi and OpenStack Flex +categories: + - Kubernetes + - DevOps + - Pulumi + - Automation +--- + +# Getting Started with Pulumi and OpenStack + +![pulumi](assets/images/2024-11-15/pulumi.png){ align=left : style="max-width:125px" } + +Pulumi is an open-source infrastructure-as-code (IaC) platform that enables you to define, deploy, and manage cloud infrastructure using familiar programming languages like Python, JavaScript, TypeScript, Go, and C#. By leveraging your existing coding skills and knowledge, Pulumi allows you to build, deploy, and manage infrastructure on any cloud provider, including AWS, Azure, Google Cloud, Kubernetes, and OpenStack. Unlike traditional tools that rely on YAML files or domain-specific languages, Pulumi offers a modern approach by utilizing general-purpose programming languages for greater flexibility and expressiveness. This means you can use standard programming constructs—such as loops, conditionals, and functions—to create complex infrastructure deployments efficiently. + + + + + +Infrastructure as Code (IaC) has revolutionized the way we manage and deploy cloud resources. Pulumi, a modern IaC tool, allows you to define cloud infrastructure using familiar programming languages. When combined with OpenStack, an open-source cloud computing platform, you gain unparalleled flexibility and control over your cloud environments. In this guide, we'll walk you through getting started with Pulumi and OpenStack. We will do so by creating an secure entry point (bastion server) and exposing securely by only allowing access to the server via ssh. + +## Prerequisites + - An OpenStack account with appropriate permissions. + - Basic knowledge of programming (we'll use Python in this guide). + - Python installed on your machine (version 3.6 or higher). + - Git installed (optional but recommended). + +## What is Pulumi? +Pulumi is an open-source IaC tool that enables you to define and manage cloud resources using general-purpose programming languages like Python, JavaScript, TypeScript, Go, and C#. Unlike traditional templating languages or domain-specific languages (DSLs), Pulumi leverages the full power of these languages, including loops, conditions, and functions. + +## Setting Up Your Environment +Before we dive in, ensure your environment is ready: + + - OpenStack Credentials: Obtain your OpenStack authentication credentials (e.g., `auth_url`, `project_name`, `username`, `password`, `region_name`). + - Install Python: If you haven't installed Python yet, download it from the [official website](https://www.python.org/) and follow the installation instructions. + - Install Git: While optional, Git is useful for version control. Download it from the [official website](https://git-scm.com/). + +## Installing Pulumi +Pulumi provides a straightforward installation process you can follow the instructions bellow or you can go to the [official website](https://www.pulumi.com/docs/iac/download-install/): + +### MacOS +The easiest way is to install via brew. For more information visit the [official website](). +``` +brew install pulumi/tap/pulumi +``` + +### For linux utilize the Installer Script +Run the following command in your terminal or command prompt: + +```bash +curl -fsSL https://get.pulumi.com | sh +``` +This script downloads and installs the Pulumi CLI to ~/.pulumi/bin, adding it to your PATH. Don't forget to add `~/.pulumi/bin` to your PATH. + +### Verifying the Installation +Check that Pulumi is installed correctly: +```bash +pulumi version +``` +You should see the Pulumi version number printed out. + +## Configuring Pulumi for OpenStack +Pulumi interacts with OpenStack through environment variables or a configuration file. If you have a working `clouds.yaml` config file then you can add it to your project as shown in step 3 in the Deploying resources to OpenStack Flex section. + + +If you do not have a working clouds config file. We'll use environment variables for simplicity. + +### Setting Environment Variables +Set the following environment variables with your OpenStack credentials: +```bash +export OS_AUTH_URL="https://your-openstack-auth-url" +export OS_PROJECT_NAME="your-project-name" +export OS_USERNAME="your-username" +export OS_PASSWORD="your-password" +export OS_REGION_NAME="your-region-name" +``` +Replace the placeholders with your actual credentials. + +#### Optional Variables +You might also need to set additional variables like OS_USER_DOMAIN_NAME or OS_PROJECT_DOMAIN_NAME depending on your OpenStack setup. + +## Creating Your First Pulumi Project +Now, let's create a Pulumi project to deploy resources to OpenStack. + +### Step A: Initialize Pulumi's self-managed backend +For this demo we will *not* be using pulumi cloud. Instead we will use a self-managed backend. To keep things simple we will use our local filesystem. For more information on [managing backends and state visit the official site](https://www.pulumi.com/docs/iac/concepts/state-and-backends/). +```bash +pulumi login --local +``` +You will see Logged into `` as `` (file://~). This will result in the storing of all stacks created in a JSON format in the directory `~/.pulumi`. + +### Step B: Create a New Directory +Create and navigate to a new directory for your project: +```bash +mkdir pulumi-openstack-flex-demo +cd pulumi-openstack-flex-demo +``` + +### Step C: Initialize the Pulumi Project +Run the Pulumi new command to bootstrap a new project utilizing all the python defaults. It will also ask you setup a passphrase to protect `config/secrets`. +```bash +pulumi new python -y +``` + +This command does the following: + - Creates a new Pulumi project using the OpenStack Python template. + - Installs necessary dependencies. + - Prompts you for a project name, description, and stack name. +** Depending on your system you might be asked to install the python venv module. + +### Step D: Review the Generated Files +The template generates several files: + + - `Pulumi.yaml`: The project metadata. + - `__main__.py`: The main program where you'll define your infrastructure. + - `requirements.txt`: Python dependencies. + - `Pulumi.dev.yaml`: The Projects stack that was automatically created for you. + - `venv`: This directory has a virtual python environment to help faciliate development. + + +!!! note + + Pulumi stacks are isolated instances of your infrastructure configurations that enable you to manage multiple environments—such as development, staging, and production—within the same project. By having many stacks for the same project, you can deploy the same infrastructure code with different settings or parameters tailored to each environment, ensuring consistent and efficient infrastructure management across all stages. + +## Deploying Resources to OpenStack Flex +Let's modify `__main__.py` to deploy a simple resource, such as an OpenStack instance. + +#### Step 0: Active the venv that was created +Activate the virtual environment + +Fish Shell: +```bash +source venv/bin/activate.fish + +``` +Bash, sh, zsh Shells: +```bash +source venv/bin/activate +``` + +Depending on your shell configuration it might show you that you are now utilizing a virtual environment. You can also verify by doing the following: +```bash +which python +``` +This will return something like +```bash +/home/debian/pulumi-openstack-flex-demo/venv/bin/python +``` + +#### Step 1: Install the OpenStack Provider +Ensure the Pulumi OpenStack provider is installed: +```bash + +pip install pulumi-openstack +``` +#### Step 2: Update requirements.txt +Add pulumi-openstack to your requirements.txt: +```bash +pulumi +pulumi-openstack +``` +Run `pip install -r requirements.txt` to install dependencies. + +#### Step 3: Update your Pulumi.yaml to utilize your OpenStack flex `clouds.yaml` +Edit your `Pulumi.yaml` file and add the following: +```yaml + openstack:cloud: + value: +``` +Make sure you replace with your cloud value. In my config my cloud is called rxt. +The final file should look something like: +``` +name: pulumi-openstack-flex-demo +description: A minimal Python Pulumi program +runtime: + name: python + options: + toolchain: pip + virtualenv: venv +config: + pulumi:tags: + value: + pulumi:template: Python + openstack:cloud: + value: rxt +``` + +#### Step 4: Define an OpenStack Instance +Edit __main__.py and add the following code: +```python +"""A Python Pulumi program""" + +import pulumi_openstack as openstack +import pulumi + +config = pulumi.Config() + +demo_name = "openstack-demo" + +server_flavor = config.require("bastion_server_flavor") +bastion_srv_name = "bastion01-" + demo_name + +# Get External Network +ext_net_name = config.require("ext_net_name") +ext_net = openstack.networking.get_network(name=ext_net_name) + +ssh_public_key = config.require("ssh_public_key") + +# Create a key pair for SSH access +keypair = openstack.compute.Keypair( + "keypair", + name=demo_name + "-key", + public_key=ssh_public_key, +) + +# Create a Private network +private_network = openstack.networking.Network( + demo_name + "-private-net", + name=demo_name + "-private-net", + admin_state_up=True, +) + +# Create a subnet within the newly created private network +private_subnet = openstack.networking.Subnet( + demo_name + "-private-subnet", + name=demo_name + "-private-subnet", + network_id=private_network.id, + cidr="192.168.0.0/24", + ip_version=4, + dns_nameservers=["8.8.8.8"], + # gateway_ip=private_subnet_gateway, +) + + +# Create a router to connect the private network to the public network +router = openstack.networking.Router( + demo_name + "-router", + admin_state_up=True, + external_network_id=ext_net.id, +) + +router_interface = openstack.networking.RouterInterface( + demo_name + "-router_interface", + router_id=router.id, + subnet_id=private_subnet.id, +) + + +# Create a security group allowing load balancer port +sec_group = openstack.networking.SecGroup( + demo_name + "-sec_group", + name=demo_name + "-sec_group", + description="Security group for " + demo_name + " control plane", + tags=[demo_name], +) + +# Allow Load Balancer Ingress port 50000 +openstack.networking.SecGroupRule( + demo_name + "-allow_22_port", + security_group_id=sec_group.id, + direction="ingress", + ethertype="IPv4", + protocol="tcp", + port_range_min=22, + port_range_max=22, + remote_ip_prefix="0.0.0.0/0", +) + +bastion_port = openstack.networking.Port( + bastion_srv_name, + name=bastion_srv_name, + network_id=private_network.id, + fixed_ips=[{"subnet_id": private_subnet.id}], + security_group_ids=[sec_group.id], +) + +bastion = openstack.compute.Instance( + bastion_srv_name, + name=bastion_srv_name, + flavor_name=server_flavor, + image_id="727958e9-d037-45d1-9716-ea7ac322fe02", + key_pair=keypair.name, + networks=[{"port": bastion_port.id}], + user_data =f"""#!/bin/bash + apt-get update + """, +) + +bastion_fip = openstack.networking.FloatingIp( + bastion_srv_name, + pool=ext_net.name, + port_id=bastion_port.id, + ) + +pulumi.export("bastion_ip", bastion_fip.address) +``` + +#### Step 5: Deploy the Stack +Run the following commands: + +Set your public key for the current `dev` stack: +```bash +pulumi config set ssh_public_key $(cat ~/.ssh/id_ed25519.pub) +pulumi config set ext_net_name PUBLICNET +``` +!!! note + + Replace with your public ssh key and your External network Name if it's something different. + +Preview the Changes: +```bash +pulumi preview +``` +This command shows you what resources will be created and it should look something like following: +```bash +Enter your passphrase to unlock config/secrets + (set PULUMI_CONFIG_PASSPHRASE or PULUMI_CONFIG_PASSPHRASE_FILE to remember): +Enter your passphrase to unlock config/secrets +Previewing update (dev): + Type Name Plan + + pulumi:pulumi:Stack getting-started-dev create + + ├─ openstack:networking:SecGroupRule openstack-demo-allow_22_port create + + ├─ openstack:networking:Subnet openstack-demo-private-subnet create + + ├─ openstack:compute:Keypair keypair create + + ├─ openstack:networking:Network openstack-demo-private-net create + + ├─ openstack:networking:SecGroup openstack-demo-sec_group create + + ├─ openstack:networking:Router openstack-demo-router create + + ├─ openstack:networking:Port bastion01-openstack-demo create + + ├─ openstack:networking:FloatingIp bastion01-openstack-demo create + + ├─ openstack:networking:RouterInterface openstack-demo-router_interface create + + └─ openstack:compute:Instance bastion01-openstack-demo create + +Outputs: + bastion_ip: output + +Resources: + + 11 to create +``` + +Deploy the Stack: +```bash +pulumi up +``` +Pulumi will display the proposed changes and prompt for confirmation. Type yes to proceed. + + +#### Step 6: Verify the Deployment +After deployment, Pulumi will output the instance_ip. You can SSH into your new instance: +```bash +ssh ubuntu@ +``` +Replace with the IP address provided in the output section: +``` +Outputs: + bastion_ip : "XX.XXX.XXX.XXX". +``` + + +## Conclusion +Congratulations! You've successfully deployed an OpenStack instance using Pulumi. This guide covered the basics of setting up Pulumi with OpenStack, but there's much more to explore. You can define more complex infrastructures, manage configurations, and leverage Pulumi's state management for robust deployments. The best part is that you can use the programing language of your choosing. + +## Next Steps + - Explore Pulumi Stacks: Learn how to manage different environments (e.g., dev, prod) using Pulumi stacks. + - Pulumi Configuration: Use pulumi config to manage configuration variables securely. + - CI/CD Integration: Integrate Pulumi deployments into your continuous integration and deployment pipelines. + +## Resources + - [Pulumi OpenStack Provider Documentation](https://www.pulumi.com/registry/packages/openstack/) + - [OpenStack Documentation](https://docs.openstack.org/) + - [Pulumi Official Website](https://www.pulumi.com/) + diff --git a/docs/blog/posts/assets/images/2024-11-15/pulumi.png b/docs/blog/posts/assets/images/2024-11-15/pulumi.png new file mode 100644 index 0000000000000000000000000000000000000000..614b68f4ec55ffe8fb27daee6dc629b69b51bf78 GIT binary patch literal 8529 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