This repository contains the Terraform code I've developed to configure my MikroTik router. The purpose of this repository is to provide a structured and repeatable way to manage and automate the setup of MikroTik routers using Infrastructure as Code (IaC) principles.
My initial idea was to use this project as a backup, start with a new router and run terraform apply
, the router would magically be ready with everything I needed. But in the end I realised that this was not feasible due to the complexity of the configuration, which required intermediate steps in Terraform and could not be applied in one step.
I decided to configure my MikroTik router with Terraform for several reasons:
-
Love for Automation: As a DevOps guy, automation is at the heart of what I do. By automating the configuration of my router, I can ensure consistency, reduce manual errors and save time on repetitive tasks.
-
Infrastructure as Code (IaC): I believe in the principles of Infrastructure as Code. Managing my network infrastructure through code allows for better version control and repeatability.
-
Skill Improvement: Working on this project is also a great way for me to improve my Terraform skills. It provides a practical, hands-on opportunity to explore advanced features.
-
Configuration Tracking: With Terraform, I can easily track changes to my router's configuration. In addition to reading changelogs, I can discover every single setting that is altered after an upgrade process, ensuring that I maintain full control and visibility over my router settings.
-
Backup: Using Terraform~provides a creative alternative to traditional backup methods for my router. By storing my router's configuration as code, I can quickly and reliably restore my settings if needed, leveraging the benefits of version control and automation.
To get started with this repository, you'll need to have the following tools installed:
-
Terraform: Ensure you have Terraform installed. You can download it from the official Terraform website.
-
Vault: You'll need HashiCorp Vault for managing secrets. Install it from the official Vault website.
-
terraform-routeros: This is the Terraform provider for MikroTik. You can find it on the Terraform Registry.
Restore the Router to the default settings but keep the Default Cofiguration.
MikroTik uses Admin as a default admin user, is a good security practice to change replace it with something not easy to predict. Currently those steps are performed manually:
-
Login to the router
-
Create a new user in the full group:
/user add name=<username> group=full
-
Logout and login with the new user
-
Detele the default admin user:
/user remove admin
I use Vault to securely store sensitive information such as the password to connect to the MikroTik API and the passwords for my WiFi networks. This ensures that the credentials are not hardcoded into the Terraform files, reducing the risk of accidental exposure.
I have built my own terraform-modules collection to streamline the deployment of a dedicated vault to store my router's secrets.
I structure the Vault different sections:
- mikrotik: username and password of the router
- wifi: my WiFi passwords, where the SSID is the key and the password is the value
- container_lego_envs: environment variables for LEGO container
-
Clone the repository:
git clone https://github.com/Schwitzd/IaC-HomeRouter.git cd IaC-HomeRouter
-
Initialize Terraform in your project directory:
terraform init
-
Create the file
dns_records.yaml
:touch dns_records.yaml
-
Create the Vault space:
cd iac_vault terraform init terraform apply -var-file=variables.tfvars
My home network is divided into four VLANs:
- VLAN 100: Home - For all general household devices.
- VLAN 200: IoT - Dedicated to Internet of Things devices like TV and smart plugs.
- VLAN 300: Server - For home servers.
- VLAN 400: Lab - Used inside my lab for experiment technologies.
- VLAN 999: Management - Router management vlan.
I have a file called _static_hosts.yaml
which contains all my static DNS records and DHCP leases. This file contains a local variable with a list of hostnames, IP addresses, DNS types and mac addresses. Here is an example of what it looks like:
static_hosts:
- hostname: "device1.home"
ip: "192.168.1.10"
type: "A"
- hostname: "device2.home"
ip: "192.168.1.11"
type: "A"
mac: "01:2A:4B:CC:1E:2A"
The _static_hosts.yaml
file is excluded in the .gitignore
to avoid exposing too much of my network (refer to the Risks section). This is the reason why is manually created after cloning the repository.
I have two WiFi networks set up:
- Home WiFi: Dedicated to the Home VLAN, operating at 5GHz.
- IoT WiFi: Dedicated to the IoT VLAN, operating at 2.4GHz.
The decision to use different frequencies is based on the typical use cases and requirements of the devices connecting to these networks. The 5GHz frequency for the Home WiFi provides higher data rates and less interference, which is ideal for devices that require more bandwidth, such as smartphones, laptops, and streaming devices. On the other hand, the 2.4GHz frequency for the IoT WiFi offers better range and penetration through walls, which is suitable for IoT devices that may be spread throughout the house and do not need high data rates. Additionally, most IoT devices nowadays only offer a 2.4GHz frequency, making this choice essential for compatibility.
A small curiosity you will discover by reading the code is that the SSIDs are suffixed with _optout_nomap
. You can understand why by reading about this decision here.
I have enabled the container feature to take advantage of the ability to run containers inside my router, I will only run network/router related containers and not other types of home containers. As stated in the official RouterOS documentation this brings security risks, I suggest you to read the red made disclaimer and understand really carefully what you are doing.
MikroTik has implemented a security mechanism that prevents the container feature from being enabled remotely or with automation, requiring you to press a physical button to acknowledge, so run this command on the terminal and restart the router:
system/device-mode/update mode=enterprise container=yes
The container
package will be installed with Terraform, but an additional manual reboot is needed.
The Mikrotik container feature has no way of keeping images up to date, so I wrote my own script mikrotik-updatecontainerimage and scheduled it to run once a week. What it does:
- Read container patameters
- Stop and delete existing container
- Create a new container with the same parameters
- Restart the container
Useless to tell you why it is important to keep images up to date!
As you may have guessed, I've decided to split my home network into different virtual LANs, primarily for security reasons, so that I can isolate devices I don't trust or can't protect as I'd like from devices I believe to be more trustworthy.
To enable the router interface in HTTPS, you can follow the video on the Mikrotik official documentation. But be aware that the router's management interface is exposed to the Internet (even if it's only accessible from Let's Encrypt IPs).
To avoid this, I took advantage of the domain I use for my website and leveraged the container functionality:
- Create a static DNS entry on RouterOS to
router.domain.tld
- Enable the container feature
- Use routeros-letsencrypt-docker to obtain a Let's Encrypt certificate using DNS challenge.
For a reason I have not yet understood Alpine Linux is not able to resolve router.domain.tld
even having set the router as DNS resolver. So in the ROUTEROS_HOST
environment variable I used the IP.
I'm using the built-in MikroTik firewall to protect each VLAN, ensuring that only authorized traffic flows through the network. By default, the firewall is configured with a rule that blocks all traffic between VLANs, creating an isolated environment for each VLAN. This setup ensures that no cross-VLAN communication occurs unless explicitly permitted by additional rules.
To manage traffic more efficiently, I'm leveraging the Address Lists feature in MikroTik. This allows me to dynamically add IP addresses to my firewall rules, making the rules more flexible and easier to manage. For example, I maintain a YAML file named _fw_addr_lists.yaml that contains all the IP addresses needed to build the rules. This file allows me to organize and update my firewall rules without modifying the main configuration directly. Here's an example of how this file is structured:
fw_addr_lists:
- list: "example-list"
address: "192.168.88.1"
In addition to address lists, I manage all my firewall rules in another YAML file, _fw_rules.yaml. This file contains the definitions for all the rules applied to the firewall, specifying which traffic is allowed or denied across the network. By keeping these rules in a YAML file, I can easily adjust the firewall settings in a centralized manner, ensuring consistency and simplicity in my network configuration.
Here’s an example of how a rule might look in the _fw_rules.yaml
file:
role14:
action: "drop"
chain: "forward"
comment: "Block all traffics between VLANs"
in_interface_list: "VLANs"
out_interface_list: "VLANs"
offline backup TBD
By publishing this repository, I'm accepting the risk of exposing my home network topology. While I've taken steps to ensure sensitive information is managed securely, sharing this code inherently comes with certain risks, such as potential exposure to network vulnerabilities.
I decided to share this repository because I believe that sharing knowledge is incredibly important. By open-sourcing my configuration, I hope to help others in the community who want to use Terraform to automate their MikroTik router setup and apply good security practices to their home network. Collaborating and learning from each other is a key aspect of the tech community.
- Disable IPSec
- Remove IPSec fw rules