| # Copyright 2018-present Open Networking Foundation |
| # |
| # SPDX-License-Identifier: Apache-2.0 |
| |
| SHELL := /bin/bash |
| BUILD ?= /tmp/build |
| M ?= $(BUILD)/milestones |
| MAKEDIR := $(dir $(realpath $(firstword $(MAKEFILE_LIST)))) |
| SCRIPTDIR := $(MAKEDIR)/scripts |
| RESOURCEDIR := $(MAKEDIR)/resources |
| WORKSPACE ?= $(HOME) |
| VENV ?= $(BUILD)/venv/aiab |
| |
| 4G_CORE_VALUES ?= $(MAKEDIR)/sd-core-4g-values.yaml |
| 5G_CORE_VALUES ?= $(MAKEDIR)/sd-core-5g-values.yaml |
| OAISIM_VALUES ?= $(MAKEDIR)/oaisim-values.yaml |
| ROC_VALUES ?= $(MAKEDIR)/roc-values.yaml |
| ROC_4G_MODELS ?= $(MAKEDIR)/roc-4g-models.json |
| ROC_5G_MODELS ?= $(MAKEDIR)/roc-5g-models.json |
| TEST_APP_VALUES?= $(MAKEDIR)/5g-test-apps-values.yaml |
| |
| KUBESPRAY_VERSION ?= release-2.17 |
| DOCKER_VERSION ?= '20.10' |
| K8S_VERSION ?= v1.20.11 |
| HELM_VERSION ?= v3.6.3 |
| |
| ENABLE_ROUTER ?= true |
| ENABLE_OAISIM ?= true |
| ENABLE_GNBSIM ?= true |
| ENABLE_SUBSCRIBER_PROXY ?= false |
| GNBSIM_COLORS ?= true |
| |
| DATA_IFACE ?= data |
| ifeq ($(DATA_IFACE), data) |
| RAN_SUBNET := 192.168.251.0/24 |
| else |
| RAN_SUBNET := $(shell ip route | grep $${DATA_IFACE} | awk '/kernel/ {print $$1}') |
| endif |
| |
| NODE_IP ?= $(shell ip route get 8.8.8.8 | grep -oP 'src \K\S+') |
| ifndef NODE_IP |
| $(error NODE_IP is not set) |
| endif |
| |
| MME_IP ?= |
| |
| HELM_GLOBAL_ARGS ?= |
| |
| # Allow installing local charts or specific versions of published charts. |
| # E.g., to install the Aether 1.5 release: |
| # CHARTS=release-1.5 make test |
| # Default is to install from the local charts. |
| CHARTS ?= local |
| CONFIGFILE := configs/$(CHARTS) |
| include $(CONFIGFILE) |
| |
| cpu_family := $(shell lscpu | grep 'CPU family:' | awk '{print $$3}') |
| cpu_model := $(shell lscpu | grep 'Model:' | awk '{print $$2}') |
| os_vendor := $(shell lsb_release -i -s) |
| os_release := $(shell lsb_release -r -s) |
| USER := $(shell whoami) |
| |
| .PHONY: 4g-core 5g-core oaisim test reset-test reset-ue reset-5g-test clean |
| |
| $(M): |
| mkdir -p $(M) |
| |
| $(M)/system-check: | $(M) |
| @if [[ $(cpu_family) -eq 6 ]]; then \ |
| if [[ $(cpu_model) -lt 60 ]]; then \ |
| echo "FATAL: haswell CPU or newer is required."; \ |
| exit 1; \ |
| fi \ |
| else \ |
| echo "FATAL: unsupported CPU family."; \ |
| exit 1; \ |
| fi |
| @if [[ $(os_vendor) =~ (Ubuntu) ]]; then \ |
| if [[ ! $(os_release) =~ (18.04) ]]; then \ |
| echo "WARN: $(os_vendor) $(os_release) has not been tested."; \ |
| fi; \ |
| if dpkg --compare-versions 4.15 gt $(shell uname -r); then \ |
| echo "FATAL: kernel 4.15 or later is required."; \ |
| echo "Please upgrade your kernel by running" \ |
| "apt install --install-recommends linux-generic-hwe-$(os_release)"; \ |
| exit 1; \ |
| fi \ |
| else \ |
| echo "FAIL: unsupported OS."; \ |
| exit 1; \ |
| fi |
| touch $@ |
| |
| $(M)/setup: | $(M) |
| sudo $(SCRIPTDIR)/cloudlab-disksetup.sh |
| sudo apt update; sudo apt install -y software-properties-common python3 python3-pip python3-venv jq httpie ipvsadm |
| touch $@ |
| |
| $(BUILD)/kubespray: | $(M)/setup |
| mkdir -p $(BUILD) |
| cd $(BUILD); git clone https://github.com/kubernetes-incubator/kubespray.git -b $(KUBESPRAY_VERSION) |
| |
| $(VENV)/bin/activate: | $(M)/setup |
| python3 -m venv $(VENV) |
| source "$(VENV)/bin/activate" && \ |
| python -m pip install -U pip && \ |
| deactivate |
| |
| $(M)/kubespray-requirements: $(BUILD)/kubespray | $(VENV)/bin/activate |
| source "$(VENV)/bin/activate" && \ |
| pip install -r $(BUILD)/kubespray/requirements.txt |
| touch $@ |
| |
| $(M)/k8s-ready: | $(M)/setup $(BUILD)/kubespray $(VENV)/bin/activate $(M)/kubespray-requirements |
| source "$(VENV)/bin/activate" && cd $(BUILD)/kubespray; \ |
| ansible-playbook -b -i inventory/local/hosts.ini \ |
| -e "{'override_system_hostname' : False, 'disable_swap' : True}" \ |
| -e "{'docker_version' : $(DOCKER_VERSION)}" \ |
| -e "{'docker_iptables_enabled' : True}" \ |
| -e "{'kube_version' : $(K8S_VERSION)}" \ |
| -e "{'kube_network_plugin_multus' : True, 'multus_version' : stable, 'multus_cni_version' : 0.3.1}" \ |
| -e "{'kube_proxy_metrics_bind_address' : 0.0.0.0:10249}" \ |
| -e "{'kube_pods_subnet' : 192.168.84.0/24, 'kube_service_addresses' : 192.168.85.0/24}" \ |
| -e "{'kube_apiserver_node_port_range' : 2000-36767}" \ |
| -e "{'kubeadm_enabled': True}" \ |
| -e "{'kube_feature_gates' : [SCTPSupport=True]}" \ |
| -e "{'kubelet_custom_flags' : [--allowed-unsafe-sysctls=net.*, --node-ip=$(NODE_IP)]}" \ |
| -e "{'dns_min_replicas' : 1}" \ |
| -e "{'helm_enabled' : True, 'helm_version' : $(HELM_VERSION)}" \ |
| cluster.yml |
| mkdir -p $(HOME)/.kube |
| sudo cp -f /etc/kubernetes/admin.conf $(HOME)/.kube/config |
| sudo chown $(shell id -u):$(shell id -g) $(HOME)/.kube/config |
| kubectl wait pod -n kube-system --for=condition=Ready --all |
| sudo adduser $(USER) docker |
| touch $@ |
| |
| $(M)/helm-ready: | $(M)/k8s-ready |
| helm repo add incubator https://charts.helm.sh/incubator |
| helm repo add cord https://charts.opencord.org |
| helm repo add atomix https://charts.atomix.io |
| helm repo add onosproject https://charts.onosproject.org |
| helm repo add aether https://charts.aetherproject.org |
| touch $@ |
| |
| /opt/cni/bin/static: | $(M)/k8s-ready |
| mkdir -p $(BUILD)/cni-plugins; cd $(BUILD)/cni-plugins; \ |
| wget https://github.com/containernetworking/plugins/releases/download/v0.8.2/cni-plugins-linux-amd64-v0.8.2.tgz && \ |
| tar xvfz cni-plugins-linux-amd64-v0.8.2.tgz |
| sudo cp $(BUILD)/cni-plugins/static /opt/cni/bin/ |
| |
| node-prep: | $(M)/helm-ready /opt/cni/bin/static |
| |
| $(M)/router-pod: |
| sudo ip link add $(DATA_IFACE) type dummy || true; |
| sudo ip link set $(DATA_IFACE) up || true; |
| DATA_IFACE=$(DATA_IFACE) envsubst < $(RESOURCEDIR)/router.yaml | kubectl apply -f - |
| kubectl wait pod -n default --for=condition=Ready -l app=router --timeout=300s |
| @touch $@ |
| |
| $(M)/router-host: |
| sudo ip link add core link $(DATA_IFACE) type macvlan mode bridge |
| sudo ip link set core up |
| sudo ip addr add 192.168.250.1/24 dev core |
| sudo ip link add access link $(DATA_IFACE) type macvlan mode bridge |
| sudo ip link set access up |
| sudo ip addr add 192.168.252.1/24 dev access |
| sudo sysctl -w net.ipv4.ip_forward=1; |
| $(eval oiface := $(shell ip route list default | awk -F 'dev' '{ print $$2; exit }' | awk '{ print $$1 }')) |
| sudo ip route add 172.250.0.0/16 via 192.168.250.3 |
| sudo iptables -t nat -A POSTROUTING -s 172.250.0.0/16 -o $(oiface) -j MASQUERADE |
| @touch $@ |
| |
| 4g-core: node-prep |
| ifeq ($(ENABLE_ROUTER),true) |
| ifeq ($(ENABLE_OAISIM),true) |
| 4g-core: $(M)/router-pod |
| else |
| 4g-core: $(M)/router-host |
| endif |
| endif |
| 4g-core: $(M)/omec |
| $(M)/omec: |
| @if [[ "${CHARTS}" == "local" || "${CHARTS}" == "local-sdcore" ]]; then \ |
| helm dep up $(SD_CORE_CHART); \ |
| else \ |
| helm repo update; \ |
| fi |
| NODE_IP=${NODE_IP} DATA_IFACE=${DATA_IFACE} RAN_SUBNET=${RAN_SUBNET} envsubst < $(4G_CORE_VALUES) | \ |
| helm upgrade --create-namespace --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace omec \ |
| --values - \ |
| sd-core \ |
| $(SD_CORE_CHART) |
| @if [[ "${ENABLE_OAISIM}" == "false" ]]; then \ |
| $(eval mme_ip := $(shell ip -4 -o addr show $${DATA_IFACE} | awk '{print $$4}' | cut -d'/' -f1)) \ |
| echo "Your MME IP is $(mme_ip)"; \ |
| fi |
| @touch $@ |
| |
| 5g-core: node-prep |
| ifeq ($(ENABLE_ROUTER),true) |
| ifeq ($(ENABLE_GNBSIM),true) |
| 5g-core: $(M)/router-pod |
| else |
| 5g-core: $(M)/router-host |
| endif |
| endif |
| 5g-core: $(M)/5g-core |
| $(M)/5g-core: |
| @if [[ "${CHARTS}" == "local" || "${CHARTS}" == "local-sdcore" ]]; then \ |
| helm dep up $(SD_CORE_CHART); \ |
| else \ |
| helm repo update; \ |
| fi |
| NODE_IP=${NODE_IP} DATA_IFACE=${DATA_IFACE} RAN_SUBNET=${RAN_SUBNET} envsubst < $(5G_CORE_VALUES) | \ |
| helm upgrade --create-namespace --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace omec \ |
| --values - \ |
| sd-core \ |
| $(SD_CORE_CHART) |
| touch $@ |
| |
| # UE images includes kernel module, ue_ip.ko |
| # which should be built in the exactly same kernel version of the host machine |
| $(BUILD)/openairinterface: | $(M)/setup |
| mkdir -p $(BUILD) |
| cd $(BUILD); git clone https://github.com/opencord/openairinterface.git |
| |
| download-ue-image: | $(M)/k8s-ready |
| sg docker -c "docker pull ${OAISIM_UE_IMAGE} && \ |
| docker tag ${OAISIM_UE_IMAGE} omecproject/lte-uesoftmodem:1.1.0" |
| touch $(M)/ue-image |
| |
| $(M)/ue-image: | $(M)/k8s-ready $(BUILD)/openairinterface |
| cd $(BUILD)/openairinterface; \ |
| sg docker -c "docker build . --target lte-uesoftmodem \ |
| --build-arg build_base=omecproject/oai-base:1.1.0 \ |
| --file Dockerfile.ue \ |
| --tag omecproject/lte-uesoftmodem:1.1.0" |
| touch $@ |
| |
| $(M)/oaisim-lo: |
| @sudo ip addr add 127.0.0.2/8 dev lo || true |
| @touch $@ |
| |
| oaisim-standalone: | $(M)/helm-ready $(M)/ue-image $(M)/oaisim-lo |
| @ip link show $(DATA_IFACE) > /dev/null || (echo DATA_IFACE is not set or does not exist; exit 1) |
| @if [[ "${MME_IP}" == "" ]]; then \ |
| echo MME_IP is not set; \ |
| exit 1; \ |
| else \ |
| ping -c 3 $(MME_IP) > /dev/null || (echo MME $(MME_IP) is not reachable; exit 1) \ |
| fi |
| sudo ip route add 192.168.252.0/24 via $(MME_IP) |
| helm repo update |
| helm upgrade --create-namespace --install $(HELM_GLOBAL_ARGS) --namespace omec oaisim cord/oaisim -f $(OAISIM_VALUES) \ |
| --set config.enb.networks.s1u.interface=$(DATA_IFACE) \ |
| --set config.enb.networks.s1_mme.interface=$(DATA_IFACE) \ |
| --set config.enb.mme.address=$(MME_IP) \ |
| --set config.enb.mme.isLocal=false \ |
| --set images.pullPolicy=IfNotPresent |
| kubectl rollout status -n omec statefulset ue |
| @echo "Test: registration" |
| @timeout 60s bash -c \ |
| "until ip addr show oip1 | grep -q inet; \ |
| do \ |
| echo 'Waiting for UE 1 gets IP address'; \ |
| sleep 3; \ |
| done" |
| @echo "Test: ping from UE to 8.8.8.8" |
| ping -I oip1 8.8.8.8 -c 3 |
| @touch $(M)/oaisim $(M)/omec |
| |
| oaisim: | $(M)/oaisim |
| $(M)/oaisim: | $(M)/ue-image $(M)/router-pod $(M)/oaisim-lo |
| sudo ip link add enb link $(DATA_IFACE) type macvlan mode bridge || true |
| sudo ip link set enb up || true |
| sudo ip addr add 192.168.251.3/24 dev enb || true |
| sudo ip route add 192.168.252.0/24 via 192.168.251.1 || true |
| helm upgrade --create-namespace --install $(HELM_GLOBAL_ARGS) --namespace omec oaisim cord/oaisim -f $(OAISIM_VALUES) \ |
| --set images.pullPolicy=IfNotPresent |
| kubectl rollout status -n omec statefulset ue |
| @timeout 60s bash -c \ |
| "until ip addr show oip1 | grep -q inet; \ |
| do \ |
| echo 'Waiting for UE 1 gets IP address'; \ |
| sleep 3; \ |
| done" |
| touch $@ |
| |
| roc: $(M)/roc |
| $(M)/roc: $(M)/helm-ready |
| kubectl get namespace aether-roc 2> /dev/null || kubectl create namespace aether-roc |
| helm repo update |
| if [ "$(CHARTS)" == "local" ]; then helm dep up $(AETHER_ROC_UMBRELLA_CHART); fi |
| helm upgrade --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace kube-system \ |
| --values $(ROC_VALUES) \ |
| atomix-controller \ |
| $(ATOMIX_CONTROLLER_CHART) |
| helm upgrade --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace kube-system \ |
| --values $(ROC_VALUES) \ |
| atomix-raft-storage \ |
| $(ATOMIX_RAFT_STORAGE_CHART) |
| helm upgrade --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace kube-system \ |
| --values $(ROC_VALUES) \ |
| onos-operator \ |
| $(ONOS_OPERATOR_CHART) |
| helm upgrade --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace aether-roc \ |
| --values $(ROC_VALUES) \ |
| aether-roc-umbrella \ |
| $(AETHER_ROC_UMBRELLA_CHART) |
| touch $@ |
| |
| # Load the ROC 4G models. Disable loading network slice from SimApp. |
| roc-4g-models: $(M)/roc |
| sed -i 's/provision-network-slice: true/provision-network-slice: false/' $(4G_CORE_VALUES) |
| sed -i 's/# syncUrl/syncUrl/' $(4G_CORE_VALUES) |
| if [ "${ENABLE_SUBSCRIBER_PROXY}" == "true" ] ; then \ |
| sed -i 's/# sub-proxy-endpt:/sub-proxy-endpt:/' $(4G_CORE_VALUES) ; \ |
| sed -i 's/# addr: sub/ addr: sub/' $(4G_CORE_VALUES) ; \ |
| sed -i 's/# port: 5000/ port: 5000/' $(4G_CORE_VALUES) ; \ |
| fi |
| $(eval ONOS_CLI_POD := $(shell kubectl -n aether-roc get pods -l name=onos-cli -o name)) |
| echo "ONOS CLI pod: ${ONOS_CLI_POD}" |
| until kubectl -n aether-roc exec ${ONOS_CLI_POD} -- \ |
| curl -s -f -L -X PATCH "http://aether-roc-api:8181/aether-roc-api" \ |
| --header 'Content-Type: application/json' \ |
| --data-raw "$$(cat ${ROC_4G_MODELS})"; do sleep 5; done |
| |
| # Load the ROC 5G models. Disable loading network slice from SimApp. |
| roc-5g-models: $(M)/roc |
| sed -i 's/provision-network-slice: true/provision-network-slice: false/' $(5G_CORE_VALUES) |
| sed -i 's/# syncUrl/syncUrl/' $(5G_CORE_VALUES) |
| if [ "${ENABLE_SUBSCRIBER_PROXY}" == "true" ] ; then \ |
| sed -i 's/# sub-proxy-endpt:/sub-proxy-endpt:/' $(5G_CORE_VALUES) ; \ |
| sed -i 's/# addr: sub/ addr: sub/' $(5G_CORE_VALUES) ; \ |
| sed -i 's/# port: 5000/ port: 5000/' $(5G_CORE_VALUES) ; \ |
| fi |
| $(eval ONOS_CLI_POD := $(shell kubectl -n aether-roc get pods -l name=onos-cli -o name)) |
| echo "ONOS CLI pod: ${ONOS_CLI_POD}" |
| until kubectl -n aether-roc exec ${ONOS_CLI_POD} -- \ |
| curl -s -f -L -X PATCH "http://aether-roc-api:8181/aether-roc-api" \ |
| --header 'Content-Type: application/json' \ |
| --data-raw "$$(cat ${ROC_5G_MODELS})"; do sleep 5; done |
| |
| roc-clean: |
| @echo "This could take 2-3 minutes..." |
| sed -i 's/provision-network-slice: false/provision-network-slice: true/' $(4G_CORE_VALUES) |
| sed -i 's/ syncUrl/ # syncUrl/' $(4G_CORE_VALUES) |
| sed -i 's/ sub-proxy-endpt:/ # sub-proxy-endpt:/' $(4G_CORE_VALUES) |
| sed -i 's/ addr: sub/ # addr: sub/' $(4G_CORE_VALUES) |
| sed -i 's/ port: 5000/ # port: 5000/' $(4G_CORE_VALUES) |
| sed -i 's/provision-network-slice: false/provision-network-slice: true/' $(5G_CORE_VALUES) |
| sed -i 's/ syncUrl/ # syncUrl/' $(5G_CORE_VALUES) |
| sed -i 's/ sub-proxy-endpt:/ # sub-proxy-endpt:/' $(5G_CORE_VALUES) |
| sed -i 's/ addr: sub/ # addr: sub/' $(5G_CORE_VALUES) |
| sed -i 's/ port: 5000/ # port: 5000/' $(5G_CORE_VALUES) |
| kubectl delete namespace aether-roc || true |
| rm -rf $(M)/roc |
| |
| omec-clean: |
| helm delete -n omec $$(helm -n omec ls -qa) || true |
| @echo "" |
| @echo "Wait for all pods to terminate..." |
| kubectl wait -n omec --for=delete --all=true -l app!=ue pod --timeout=180s || true |
| |
| router-clean: |
| @kubectl delete net-attach-def router-net 2>/dev/null || true |
| @kubectl delete po router 2>/dev/null || true |
| kubectl wait --for=delete -l app=router pod --timeout=180s 2>/dev/null || true |
| sudo ip link del access || true |
| sudo ip link del core || true |
| $(eval oiface := $(shell ip route list default | awk -F 'dev' '{ print $$2; exit }' | awk '{ print $$1 }')) |
| sudo iptables -t nat -D POSTROUTING -s 172.250.0.0/16 -o $(oiface) -j MASQUERADE || true |
| @sudo ip link del data 2>/dev/null || true |
| @cd $(M); rm -f router-pod router-host |
| |
| oaisim-clean: reset-ue |
| @sudo ip addr del 127.0.0.2/8 dev lo 2>/dev/null || true |
| @sudo ip link del enb 2>/dev/null || true |
| @sudo ip route del 192.168.252.0/24 || true |
| @cd $(M); rm -f oaisim-lo |
| |
| test: | 4g-core $(M)/oaisim |
| @sleep 5 |
| @echo "Test1: ping from UE to SGI network gateway" |
| ping -I oip1 192.168.250.1 -c 15 |
| @echo "Test2: ping from UE to 8.8.8.8" |
| ping -I oip1 8.8.8.8 -c 3 |
| @echo "Test3: ping from UE to google.com" |
| ping -I oip1 google.com -c 3 |
| @echo "Finished to test" |
| |
| 5g-test: | 5g-core |
| @if [[ "${CHARTS}" == "release-1.6" ]]; then echo "[NOTE] 5G Test not supported for Aether 1.6, exiting..."; exit 1; fi |
| @echo "Test: Registration + UE initiated PDU Session Establishment + User Data packets" |
| @sleep 60 |
| @rm -f /tmp/gnbsim.out |
| @if [[ ${GNBSIM_COLORS} == "true" ]]; then \ |
| kubectl -n omec exec gnbsim-0 -- ./gnbsim 2>&1 | tee /tmp/gnbsim.out; \ |
| else \ |
| kubectl -n omec exec gnbsim-0 -- ./gnbsim 2>&1 | sed -u "s,\x1B\[[0-9;]*[a-zA-Z],,g" | tee /tmp/gnbsim.out; \ |
| fi |
| @grep -q "Simulation Result: PASS" /tmp/gnbsim.out |
| |
| reset-test: | oaisim-clean omec-clean router-clean |
| @cd $(M); rm -f omec oaisim 5g-core |
| |
| reset-ue: |
| helm delete -n omec oaisim || true |
| kubectl wait -n omec --for=delete pod enb-0 || true |
| kubectl wait -n omec --for=delete pod ue-0 || true |
| cd $(M); rm -f oaisim |
| |
| reset-5g-test: omec-clean |
| cd $(M); rm -f 5g-core |
| |
| reset-dbtestapp: |
| helm uninstall --namespace omec 5g-test-app |
| |
| dbtestapp: |
| helm repo update |
| if [ "$(CHARTS)" == "local" ]; then helm dep up $(5G_TEST_APPS_CHART); fi |
| helm upgrade --install --wait $(HELM_GLOBAL_ARGS) \ |
| --namespace omec \ |
| 5g-test-app \ |
| --values $(TEST_APP_VALUES) \ |
| $(5G_TEST_APPS_CHART) |
| @echo "Finished to dbtestapp" |
| |
| clean: | oaisim-clean router-clean |
| source "$(VENV)/bin/activate" && cd $(BUILD)/kubespray; \ |
| ansible-playbook -b -i inventory/local/hosts.ini reset.yml --extra-vars "reset_confirmation=yes" |
| @if [ -d /usr/local/etc/emulab ]; then \ |
| mount | grep /mnt/extra/kubelet/pods | cut -d" " -f3 | sudo xargs umount; \ |
| sudo rm -rf /mnt/extra/kubelet; \ |
| fi |
| rm -rf $(M) |