b72a30b6737f164fd6a1e694aa6601b2eaa077eb
[icn.git] / deploy / metal3-vm / 03_launch_mgmt_cluster.sh
1 #!/usr/bin/env bash
2 set -eux -o pipefail
3
4 # shellcheck disable=SC1091
5 source lib/logging.sh
6 # shellcheck disable=SC1091
7 source lib/common.sh
8
9 eval "$(go env)"
10 export GOPATH
11 DEPLOYDIR="$(dirname "$PWD")"
12 BMODIR=$DEPLOYDIR/metal3/scripts/bmo
13
14 # Environment variables
15 # M3PATH : Path to clone the metal3 dev env repo
16 # BMOPATH : Path to clone the baremetal operator repo
17 #
18 # BMOREPO : Baremetal operator repository URL
19 # BMOBRANCH : Baremetal operator repository branch to checkout
20 # FORCE_REPO_UPDATE : discard existing directories
21 #
22 # BMO_RUN_LOCAL : run the baremetal operator locally (not in Kubernetes cluster)
23
24 M3PATH="${GOPATH}/src/github.com/metal3-io"
25 BMOPATH="${M3PATH}/baremetal-operator"
26
27 BMOREPO="${BMOREPO:-https://github.com/metal3-io/baremetal-operator.git}"
28 BMOBRANCH="${BMOBRANCH:-10eb5aa3e614d0fdc6315026ebab061cbae6b929}"
29 FORCE_REPO_UPDATE="${FORCE_REPO_UPDATE:-true}"
30
31 BMO_RUN_LOCAL="${BMO_RUN_LOCAL:-false}"
32 COMPUTE_NODE_PASSWORD="${COMPUTE_NODE_PASSWORD:-mypasswd}"
33 BM_IMAGE=${BM_IMAGE:-"bionic-server-cloudimg-amd64.img"}
34 IMAGE_URL=http://172.22.0.1/images/${BM_IMAGE}
35 IMAGE_CHECKSUM=http://172.22.0.1/images/${BM_IMAGE}.md5sum
36
37 function clone_repos {
38     mkdir -p "${M3PATH}"
39     if [[ -d ${BMOPATH} && "${FORCE_REPO_UPDATE}" == "true" ]]; then
40       rm -rf "${BMOPATH}"
41     fi
42     if [ ! -d "${BMOPATH}" ] ; then
43         pushd "${M3PATH}"
44         git clone "${BMOREPO}"
45         popd
46     fi
47     pushd "${BMOPATH}"
48     git checkout "${BMOBRANCH}"
49     git pull -r || true
50     popd
51 }
52
53 function launch_baremetal_operator {
54     docker pull $IRONIC_BAREMETAL_IMAGE
55     kubectl apply -f $BMODIR/namespace/namespace.yaml
56     kubectl apply -f $BMODIR/rbac/service_account.yaml -n metal3
57     kubectl apply -f $BMODIR/rbac/role.yaml -n metal3
58     kubectl apply -f $BMODIR/rbac/role_binding.yaml
59     kubectl apply -f $BMODIR/crds/metal3.io_baremetalhosts_crd.yaml
60     kubectl apply -f $BMODIR/operator/no_ironic/operator.yaml -n metal3
61 }
62
63 # documentation for the values below may be found at
64 # https://cloudinit.readthedocs.io/en/latest/topics/modules.html
65 function create_userdata {
66     name="$1"
67     COMPUTE_NODE_FQDN="$name.akraino.icn.org"
68     printf "#cloud-config\n" > $name-userdata.yaml
69     if [ -n "$COMPUTE_NODE_PASSWORD" ]; then
70         printf "password: ""%s" "$COMPUTE_NODE_PASSWORD" >>  $name-userdata.yaml
71         printf "\nchpasswd: {expire: False}\n" >>  $name-userdata.yaml
72         printf "ssh_pwauth: True\n" >>  $name-userdata.yaml
73     fi
74
75     if [ -n "$COMPUTE_NODE_FQDN" ]; then
76         printf "fqdn: ""%s" "$COMPUTE_NODE_FQDN" >>  $name-userdata.yaml
77         printf "\n" >>  $name-userdata.yaml
78     fi
79     printf "disable_root: false\n" >> $name-userdata.yaml
80     printf "ssh_authorized_keys:\n  - " >> $name-userdata.yaml
81
82     if [ ! -f $HOME/.ssh/id_rsa.pub ]; then
83         yes y | ssh-keygen -t rsa -N "" -f $HOME/.ssh/id_rsa
84     fi
85
86     cat $HOME/.ssh/id_rsa.pub >> $name-userdata.yaml
87     cloud_init_scripts >> $name-userdata.yaml
88     printf "\n" >> $name-userdata.yaml
89 }
90
91 function cloud_init_scripts {
92     # set_dhcp_indentifier.sh:
93     #   The IP address assigned to the provisioning NIC will change
94     #   due to IPA using the MAC address as the client ID and systemd
95     #   using a different ID.  Tell systemd to use the MAC as the
96     #   client ID.  We can't do this in the network data as only the
97     #   JSON format is supported by metal3, and the JSON format does
98     #   not support the dhcp-identifier field.
99     cat << 'EOF'
100 write_files:
101 - path: /var/lib/cloud/scripts/per-instance/set_dhcp_identifier.sh
102   owner: root:root
103   permissions: '0777'
104   content: |
105     #!/usr/bin/env bash
106     set -eux -o pipefail
107     sed -i -e '/dhcp4: true$/!b' -e 'h;s/\S.*/dhcp-identifier: mac/;H;g' /etc/netplan/50-cloud-init.yaml
108     netplan apply
109 EOF
110 }
111
112 function apply_userdata_credential {
113     name="$1"
114     cat <<EOF > ./$name-user-data-credential.yaml
115 apiVersion: v1
116 data:
117   userData: $(base64 -w 0 $name-userdata.yaml)
118 kind: Secret
119 metadata:
120   name: $name-user-data
121   namespace: metal3
122 type: Opaque
123 EOF
124     kubectl apply -n metal3 -f $name-user-data-credential.yaml
125 }
126
127 function create_networkdata {
128     name="$1"
129     node_networkdata $name > $name-networkdata.json
130 }
131
132 function apply_networkdata_credential {
133     name="$1"
134     cat <<EOF > ./$name-network-data-credential.yaml
135 apiVersion: v1
136 data:
137   networkData: $(base64 -w 0 $name-networkdata.json)
138 kind: Secret
139 metadata:
140   name: $name-network-data
141   namespace: metal3
142 type: Opaque
143 EOF
144     kubectl apply -n metal3 -f $name-network-data-credential.yaml
145 }
146
147 function make_bm_hosts {
148     while IFS=',' read -r name address user password mac; do
149         create_userdata $name
150         apply_userdata_credential $name
151         create_networkdata $name
152         apply_networkdata_credential $name
153         GO111MODULE=auto go run "${BMOPATH}"/cmd/make-bm-worker/main.go \
154            -address "$address" \
155            -password "$password" \
156            -user "$user" \
157            -boot-mac "$mac" \
158            "$name" > $name-bm-node.yaml
159         printf "  image:" >> $name-bm-node.yaml
160         printf "\n    url: ""%s" "${IMAGE_URL}" >> $name-bm-node.yaml
161         printf "\n    checksum: ""%s" "${IMAGE_CHECKSUM}" >> $name-bm-node.yaml
162         printf "\n  userData:" >> $name-bm-node.yaml
163         printf "\n    name: ""%s" "$name""-user-data" >> $name-bm-node.yaml
164         printf "\n    namespace: metal3" >> $name-bm-node.yaml
165         printf "\n  networkData:" >> $name-bm-node.yaml
166         printf "\n    name: ""%s" "$name""-network-data" >> $name-bm-node.yaml
167         printf "\n    namespace: metal3" >> $name-bm-node.yaml
168         printf "\n  rootDeviceHints:" >> $name-bm-node.yaml
169         printf "\n    minSizeGigabytes: 48\n" >> $name-bm-node.yaml
170         kubectl apply -f $name-bm-node.yaml -n metal3
171     done
172 }
173
174 function apply_bm_hosts {
175     list_nodes | make_bm_hosts
176 }
177
178 clone_repos
179 launch_baremetal_operator
180 apply_bm_hosts