refactor(k3s): streamline inventory and primary server IP handling

Signed-off-by: Tuan-Dat Tran <tuan-dat.tran@tudattr.dev>
This commit is contained in:
Tuan-Dat Tran
2025-07-13 00:40:48 +02:00
parent dac0d88d60
commit f1b0cfad2c
21 changed files with 185 additions and 170 deletions

View File

@@ -1,17 +1,20 @@
- name: Set up Agents - name: Set up Agents
hosts: k3s_nodes hosts: k3s
gather_facts: yes gather_facts: true
vars:
k3s_primary_server_ip: "{{ groups['k3s_server'] | map('extract', hostvars, 'ansible_host') | list | first }}"
pre_tasks: pre_tasks:
- name: Get K3s token from the first server - name: Get K3s token from the first server
when: host.ip == k3s.server.ips[0] and inventory_hostname in groups["k3s_server"] when: host.ip == k3s_primary_server_ip and inventory_hostname in groups["k3s_server"]
slurp: slurp:
src: /var/lib/rancher/k3s/server/node-token src: /var/lib/rancher/k3s/server/node-token
register: k3s_token register: k3s_token
become: true become: true
- name: Set fact on k3s.server.ips[0] - name: Set fact on k3s_primary_server_ip
when: host.ip == k3s.server.ips[0] and inventory_hostname in groups["k3s_server"] when: host.ip == k3s_primary_server_ip and inventory_hostname in groups["k3s_server"]
set_fact: k3s_token="{{ k3s_token['content'] | b64decode | trim }}" set_fact:
k3s_token: "{{ k3s_token['content'] | b64decode | trim }}"
roles: roles:
- role: common - role: common
@@ -20,7 +23,7 @@
- common - common
- role: k3s_agent - role: k3s_agent
when: inventory_hostname in groups["k3s_agent"] when: inventory_hostname in groups["k3s_agent"]
k3s_token: "{{ hostvars[(hostvars | dict2items | map(attribute='value') | map('dict2items') | map('selectattr', 'key', 'match', 'host') | map('selectattr', 'value.ip', 'match', k3s.server.ips[0] ) | select() | first | items2dict).host.hostname].k3s_token }}" k3s_token: "{{ hostvars[(hostvars | dict2items | map(attribute='value') | map('dict2items') | map('selectattr', 'key', 'match', 'host') | map('selectattr', 'value.ip', 'match', k3s_primary_server_ip ) | select() | first | items2dict).host.hostname].k3s_token }}"
tags: tags:
- k3s_agent - k3s_agent
- role: node_exporter - role: node_exporter

View File

@@ -0,0 +1,17 @@
---
- name: Set up Servers
hosts: k3s
gather_facts: true
roles:
- role: common
tags:
- common
when: inventory_hostname in groups["k3s_loadbalancer"]
- role: k3s_loadbalancer
tags:
- k3s_loadbalancer
when: inventory_hostname in groups["k3s_loadbalancer"]
# - role: node_exporter
# tags:
# - node_exporter
# when: inventory_hostname in groups["k3s_loadbalancer"]

View File

@@ -1,7 +1,7 @@
--- ---
- name: Set up Servers - name: Set up Servers
hosts: k3s_server hosts: k3s
gather_facts: yes gather_facts: true
roles: roles:
- role: common - role: common
tags: tags:
@@ -9,6 +9,7 @@
- role: k3s_server - role: k3s_server
tags: tags:
- k3s_server - k3s_server
when: inventory_hostname in groups["k3s_server"]
- role: node_exporter - role: node_exporter
tags: tags:
- node_exporter - node_exporter

View File

@@ -1,17 +1,20 @@
- name: Set up storage - name: Set up storage
hosts: k3s_nodes hosts: k3s_nodes
gather_facts: true gather_facts: true
vars:
k3s_primary_server_ip: "{{ groups['k3s_server'] | map('extract', hostvars, 'ansible_host') | list | first }}"
pre_tasks: pre_tasks:
- name: Get K3s token from the first server - name: Get K3s token from the first server
when: host.ip == k3s.server.ips[0] and inventory_hostname in groups["k3s_server"] when: host.ip == k3s_primary_server_ip and inventory_hostname in groups["k3s_server"]
slurp: slurp:
src: /var/lib/rancher/k3s/server/node-token src: /var/lib/rancher/k3s/server/node-token
register: k3s_token register: k3s_token
become: true become: true
- name: Set fact on k3s.server.ips[0] - name: Set fact on k3s_primary_server_ip
when: host.ip == k3s.server.ips[0] and inventory_hostname in groups["k3s_server"] when: host.ip == k3s_primary_server_ip and inventory_hostname in groups["k3s_server"]
set_fact: k3s_token="{{ k3s_token['content'] | b64decode | trim }}" set_fact:
k3s_token: "{{ k3s_token['content'] | b64decode | trim }}"
roles: roles:
- role: common - role: common
@@ -20,7 +23,7 @@
- common - common
- role: k3s_storage - role: k3s_storage
when: inventory_hostname in groups["k3s_storage"] when: inventory_hostname in groups["k3s_storage"]
k3s_token: "{{ hostvars[(hostvars | dict2items | map(attribute='value') | map('dict2items') | map('selectattr', 'key', 'match', 'host') | map('selectattr', 'value.ip', 'match', k3s.server.ips[0] ) | select() | first | items2dict).host.hostname].k3s_token }}" k3s_token: "{{ hostvars[(hostvars | dict2items | map(attribute='value') | map('dict2items') | map('selectattr', 'key', 'match', 'host') | map('selectattr', 'value.ip', 'match', k3s_primary_server_ip ) | select() | first | items2dict).host.hostname].k3s_token }}"
tags: tags:
- k3s_storage - k3s_storage
- role: node_exporter - role: node_exporter

View File

@@ -1,14 +0,0 @@
---
- name: Set up Servers
hosts: loadbalancer
gather_facts: yes
roles:
- role: common
tags:
- common
- role: loadbalancer
tags:
- loadbalancer
- role: node_exporter
tags:
- node_exporter

View File

@@ -16,6 +16,6 @@
ansible.builtin.command: | ansible.builtin.command: |
/tmp/k3s_install.sh /tmp/k3s_install.sh
environment: environment:
K3S_URL: "https://{{ k3s.loadbalancer.ip }}:{{ k3s.loadbalancer.default_port }}" K3S_URL: "https://{{ hostvars['k3s-loadbalancer'].ansible_default_ipv4.address }}:{{ k3s.loadbalancer.default_port }}"
K3S_TOKEN: "{{ k3s_token }}" K3S_TOKEN: "{{ k3s_token }}"
become: true become: true

View File

@@ -2,7 +2,7 @@
- name: Template the nginx config file with dynamic upstreams - name: Template the nginx config file with dynamic upstreams
ansible.builtin.template: ansible.builtin.template:
src: templates/nginx.conf.j2 src: templates/nginx.conf.j2
dest: "{{ nginx_config_path }}" dest: "{{ k3s_loadbalancer_nginx_config_path }}"
owner: root owner: root
group: root group: root
mode: "0644" mode: "0644"
@@ -10,7 +10,7 @@
notify: notify:
- Restart nginx - Restart nginx
vars: vars:
k3s_server_ips: "{{ k3s.server.ips }}" k3s_server_ips: "{{ groups['k3s_server'] | map('extract', hostvars, 'ansible_default_ipv4') | map(attribute='address') | unique | list }}"
- name: Enable nginx - name: Enable nginx
ansible.builtin.systemd: ansible.builtin.systemd:

View File

@@ -0,0 +1,87 @@
include /etc/nginx/modules-enabled/*.conf;
events {}
stream {
# TCP Load Balancing for the K3s API
upstream k3s_servers {
{% for ip in k3s_server_ips %}
server {{ ip }}:{{ k3s.loadbalancer.default_port }};
{% endfor %}
}
server {
listen {{k3s.loadbalancer.default_port}};
proxy_pass k3s_servers;
}
upstream dns_servers {
{% for ip in k3s_server_ips %}
server {{ ip }}:53;
{% endfor %}
}
server {
listen 53 udp;
proxy_pass dns_servers;
}
}
# http {
# upstream k3s_servers_http {
# least_conn;
# {% for ip in k3s_server_ips %}
# server {{ ip }}:80;
# {% endfor %}
# }
#
# upstream k3s_servers_https {
# least_conn;
# {% for ip in k3s_server_ips %}
# server {{ ip }}:443;
# {% endfor %}
# }
#
# server {
# listen 80;
#
# location / {
# proxy_pass http://k3s_servers_http;
# proxy_set_header Host $http_host;
# proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
# proxy_set_header X-Forwarded-Proto http;
# }
# }
#
# server {
# listen 443 ssl;
#
# server_name staging.k3s.seyshiro.de *.staging.k3s.seyshiro.de;
#
# ssl_certificate /etc/nginx/ssl/staging_tls.crt;
# ssl_certificate_key /etc/nginx/ssl/staging_tls.key;
#
# location / {
# proxy_pass https://k3s_servers_https;
# proxy_set_header Host $host;
# proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
# proxy_set_header X-Forwarded-Proto https;
# }
# }
#
# server {
# listen 443 ssl;
#
# server_name k3s.seyshiro.de *.k3s.seyshiro.de;
#
# ssl_certificate /etc/nginx/ssl/production_tls.crt;
# ssl_certificate_key /etc/nginx/ssl/production_tls.key;
#
# location / {
# proxy_pass https://k3s_servers_https;
# proxy_set_header Host $host;
# proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
# proxy_set_header X-Forwarded-Proto https;
# }
# }
# }

View File

@@ -0,0 +1 @@
k3s_loadbalancer_nginx_config_path: "/etc/nginx/nginx.conf"

View File

@@ -16,7 +16,7 @@
ansible.builtin.command: | ansible.builtin.command: |
/tmp/k3s_install.sh server \ /tmp/k3s_install.sh server \
--node-taint CriticalAddonsOnly=true:NoExecute \ --node-taint CriticalAddonsOnly=true:NoExecute \
--tls-san {{ k3s.loadbalancer.ip }} --tls-san {{ hostvars['k3s-loadbalancer'].ansible_default_ipv4.address }}
become: true become: true
async: 300 async: 300
poll: 0 poll: 0

View File

@@ -18,6 +18,6 @@
--node-taint storage=true:NoExecute \ --node-taint storage=true:NoExecute \
--node-label longhorn=true --node-label longhorn=true
environment: environment:
K3S_URL: "https://{{ k3s.loadbalancer.ip }}:{{ k3s.loadbalancer.default_port }}" K3S_URL: "https://{{ hostvars['k3s-loadbalancer'].ansible_default_ipv4.address }}:{{ k3s.loadbalancer.default_port }}"
K3S_TOKEN: "{{ k3s_token }}" K3S_TOKEN: "{{ k3s_token }}"
become: true become: true

View File

@@ -1,89 +0,0 @@
include /etc/nginx/modules-enabled/*.conf;
events {}
stream {
# TCP Load Balancing for the K3s API
upstream k3s_servers {
{% for ip in k3s_server_ips %}
server {{ ip }}:{{k3s.loadbalancer.default_port}};
{% endfor %}
}
server {
listen {{k3s.loadbalancer.default_port}};
proxy_pass k3s_servers;
}
upstream dns_servers {
{% for ip in k3s_server_ips %}
server {{ ip }}:53;
{% endfor %}
}
server {
listen 53 udp;
proxy_pass dns_servers;
}
}
http {
upstream k3s_servers_http {
least_conn;
{% for ip in k3s_server_ips %}
server {{ ip }}:80;
{% endfor %}
}
upstream k3s_servers_https {
least_conn;
{% for ip in k3s_server_ips %}
server {{ ip }}:443;
{% endfor %}
}
server {
listen 80;
location / {
proxy_pass http://k3s_servers_http;
proxy_set_header Host $http_host;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto http;
}
}
server {
listen 443 ssl;
server_name staging.k3s.seyshiro.de *.staging.k3s.seyshiro.de;
ssl_certificate /etc/nginx/ssl/staging_tls.crt;
ssl_certificate_key /etc/nginx/ssl/staging_tls.key;
location / {
proxy_pass https://k3s_servers_https;
proxy_set_header Host $host;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto https;
}
}
server {
listen 443 ssl;
server_name k3s.seyshiro.de *.k3s.seyshiro.de;
ssl_certificate /etc/nginx/ssl/production_tls.crt;
ssl_certificate_key /etc/nginx/ssl/production_tls.key;
location / {
proxy_pass https://k3s_servers_https;
proxy_set_header Host $host;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto https;
}
}
}

View File

@@ -1 +0,0 @@
nginx_config_path: "/etc/nginx/nginx.conf"

View File

@@ -78,11 +78,11 @@
ProxyJump {{ vm.node }} ProxyJump {{ vm.node }}
StrictHostKeyChecking no StrictHostKeyChecking no
- name: Add VM to homelab_vms group in production.ini # - name: Add VM to homelab_vms group in production.ini
ansible.builtin.lineinfile: # ansible.builtin.lineinfile:
path: "{{ inventory_file }}" # path: "{{ inventory_file }}"
line: "{{ vm.name }}" # line: "{{ vm.name }}"
insertafter: '^\[vms\]' # insertafter: '^\[vms\]'
create: true # create: true
state: present # state: present
delegate_to: localhost # delegate_to: localhost

View File

@@ -1,18 +1,7 @@
k3s: k3s:
net: "192.168.20.0/24"
server: server:
ips: ips: []
- 192.168.20.21
- 192.168.20.24
- 192.168.20.30
loadbalancer: loadbalancer:
ip: 192.168.20.22
default_port: 6443 default_port: 6443
db:
ip: 192.168.20.23
default_port: "5432"
agent: agent:
ips: ips: []
- 192.168.20.25
- 192.168.20.26
- 192.168.20.27

View File

@@ -1,23 +1,34 @@
$ANSIBLE_VAULT;1.1;AES256 $ANSIBLE_VAULT;1.1;AES256
31643231626635633436363136386537616133326538323239663963346332383961396132316662 64336139336538333337376465316164383766643666336666643166333134636338323562303364
3938393638646562306634333932666663363363353264620a613833666634383061343565613364 6235613337366634613532373933396230666137373562650a643633306165643331643464633762
38343537333930303563613839303265373339616463626133646365643630313339633765333231 35336433626161393735353133343739353738653061613733393135313061643663616665316463
6236306463616565350a626235666164303737646338363232336363336539656439316462643332 6238376435633435650a306636303934383739656439383632313964356434353536373961646531
36346530306266616465643766333864356264386435383633356534663438376335643630613230 35303533666633346363663936366535613039356164383362393736306338613236373138663731
64313663613332666534623433653539653234646661636230616134353336663631313661333661 65666635353734353261333332393962636664653332313062336239313834653536363539306630
32666632363765613934353536343339306632666238626330663938313030633362316661656432 61316431313631643637616434376334323232306232363936613139373762613862653938373461
64393863356336343261663935373530346162323665303632646531613530393432393332663963 34366363643337326439633963303430613935323866343764326639663531303931396235643231
62303663613766613830383735643839353039663631333231343036636537643237643932656162 36346463653866653137653931303439326433366231303530316632613033333761326536326335
32396632316263646637653562386438613930313331653261373363386134663835313762646136 30343233333232333434303562396166386133313633323732636532376539633336613532633765
37623237636464613736353237313666656234303534623961666230393530386435393734376639 66656663353964316364636236623133306533656465303833346563376461396639626262333133
39366636623132326230396635376136383634306664336332663535366230653632613935383135 33663966393030653762636164653534363338613536636432663938393033313933323830336538
31383232386633666263666439306631373663613930623762343635376261316136656539323631 61663865353466393836333539636466613137396430636566303135326565383764373831336532
64393062623461383733316231633335303535363763633737373933656563623234353930323262 66626332383065643636663638616337316136623131333630613861353730646339366239633861
33623463613638306630653639646230396539383065336166643935346435363534353836626262 31343133346138343637373039633930653731396537323438623237393436303063623862663965
62393262646262323433373561303132336564353062396331623264333464346534626633333561 65353332393331623933323138633231363539323834333631643337613863643737306363323135
30636462646664656532393139636331376534643234663566633862373263306365643336343039 61353663643563393539373839643462616339333762353962653065653134653063336466343431
64343236303139626164656139613438623030633735666130346335626530373636666534616233 61313262616631343265386530653431356632616230633032363165656666333662636339306539
65626166386537333162393962666461613266366261316339356665643765376666393965613835 37646634353961346165356565313038303333303564333862323766366238366434643562306262
65376561383865336165343662333236653537666563613730666461633233356166653964333164 38656532333339643335386130356637353434393037636530363233393162663330663566663962
35646264653062396330373135363338346138353136626661643531323961316231356262363966 34343333383631343330663962343639633464353961343933653764643666626631346434366365
3636356230396130663531353437653034396534313863336134 37303433626330346630353064613766303634386238636230346531663038653865393939663732
37613461313738313766306663653264616563633966316362356539373239663464386430636464
61373864313064626133623332643139336163643465376234373530666630656361616236336130
37623962393237623135656534613839363831613165356563333039366462306230636432653636
64333633393532313635323830333432666134373630666561626231666433303132663939633965
61373137633865323564343661623039616331323164396133343165656263383865383861616262
64636230336130356364333964336335656664303334326537303033613331353038353666646463
63363631613238633831666136363833363964356432373434643131653531666166666233613861
30306435306563303333343364333065616438383331383437353234323633393733653965313165
30643539663330356630363833643136643265623966636466336539353738373136616265393265
36613564653634313438666334313636653435336263393635656138343534336232346332356264
33366232613832643862386532663264353735393033303864356230333864363366

View File

@@ -78,3 +78,14 @@ vms:
ciuser: "{{ user }}" ciuser: "{{ user }}"
sshkeys: "{{ pubkey }}" sshkeys: "{{ pubkey }}"
disk_size: 64 # in Gb disk_size: 64 # in Gb
- name: "k3s-loadbalancer"
node: "naruto01"
vmid: 150
cores: 1
memory: 2048 # in MiB
net:
net0: "virtio,bridge=vmbr0,firewall=1"
boot_image: "{{ proxmox_cloud_init_images.debian.name }}"
ciuser: "{{ user }}"
sshkeys: "{{ pubkey }}"
disk_size: 32 # in Gb

View File

@@ -4,7 +4,6 @@
k3s_server k3s_server
k3s_agent k3s_agent
k3s_storage k3s_storage
k3s_storage
k3s_loadbalancer k3s_loadbalancer
[k3s_server] [k3s_server]
@@ -13,9 +12,6 @@ k3s-server10
[k3s_agent] [k3s_agent]
k3s-agent[10:12] k3s-agent[10:12]
[k3s_storage]
k3s-longhorn[10:12]
[k3s_loadbalancer] [k3s_loadbalancer]
k3s-loadbalancer k3s-loadbalancer