-
Notifications
You must be signed in to change notification settings - Fork 99
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: Huy Mai <[email protected]>
- Loading branch information
Showing
13 changed files
with
1,399 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
name: build-fkas-images-action | ||
|
||
on: | ||
push: | ||
branches: | ||
- 'main' | ||
paths: | ||
- 'hack/fake-apiserver/**' | ||
|
||
permissions: | ||
contents: read | ||
|
||
jobs: | ||
build_FKAS: | ||
name: Build Metal3-FKAS image | ||
if: github.repository == 'metal3-io/cluster-api-provider-metal3' | ||
uses: metal3-io/project-infra/.github/workflows/container-image-build.yml@main | ||
with: | ||
image-name: "metal3-fkas" | ||
pushImage: true | ||
dockerfile-directory: hack/fake-apiserver | ||
secrets: | ||
QUAY_USERNAME: ${{ secrets.QUAY_USERNAME }} | ||
QUAY_PASSWORD: ${{ secrets.QUAY_PASSWORD }} | ||
SLACK_WEBHOOK: ${{ secrets.SLACK_WEBHOOK }} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
# Copyright 2024 The Kubernetes Authors. | ||
# | ||
# Licensed under the Apache License, Version 2.0 (the "License"); | ||
# you may not use this file except in compliance with the License. | ||
# You may obtain a copy of the License at | ||
# | ||
# http://www.apache.org/licenses/LICENSE-2.0 | ||
# | ||
# Unless required by applicable law or agreed to in writing, software | ||
# distributed under the License is distributed on an "AS IS" BASIS, | ||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
# See the License for the specific language governing permissions and | ||
# limitations under the License. | ||
|
||
# Support FROM override | ||
ARG BUILD_IMAGE=docker.io/golang:1.22.7@sha256:192683db8982323952988c7b86c098ee7ecc6cbeb202bf7c113ff9be5358367c | ||
ARG BASE_IMAGE=gcr.io/distroless/static:nonroot@sha256:9ecc53c269509f63c69a266168e4a687c7eb8c0cfd753bd8bfcaa4f58a90876f | ||
|
||
# Build the fkas binary on golang image | ||
FROM $BUILD_IMAGE AS base | ||
WORKDIR /workspace | ||
|
||
# Run this with docker build --build_arg $(go env GOPROXY) to override the goproxy | ||
ARG goproxy=https://proxy.golang.org | ||
ENV GOPROXY=$goproxy | ||
|
||
# Copy the Go Modules manifests | ||
COPY go.mod go.sum ./ | ||
|
||
# Cache deps before building and copying source so that we don't need to re-download as much | ||
# and so that source changes don't invalidate our downloaded layer | ||
RUN go mod download | ||
|
||
# Build Fkas | ||
FROM base AS build-fkas | ||
|
||
# Copy the sources | ||
COPY cmd/metal3-fkas/*.go . | ||
|
||
# Build | ||
ARG ARCH=amd64 | ||
RUN CGO_ENABLED=0 GOOS=linux GOARCH=${ARCH} \ | ||
go build -a -ldflags '-extldflags "-static"' \ | ||
-o fkas . | ||
|
||
# Build fkas-reconciler | ||
FROM base AS build-fkas-reconciler | ||
|
||
# Copy the sources | ||
COPY cmd/metal3-fkas-reconciler/*.go . | ||
|
||
# Build | ||
ARG ARCH=amd64 | ||
RUN CGO_ENABLED=0 GOOS=linux GOARCH=${ARCH} \ | ||
go build -a -ldflags '-extldflags "-static"' \ | ||
-o reconciler . | ||
|
||
# Copy the controller-manager into a thin image | ||
FROM $BASE_IMAGE | ||
WORKDIR / | ||
# Use uid of nonroot user (65532) because kubernetes expects numeric user when applying pod security policies | ||
COPY --from=build-fkas /workspace/fkas . | ||
COPY --from=build-fkas-reconciler /workspace/reconciler . | ||
USER 65532 | ||
ENTRYPOINT ["/fkas"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,262 @@ | ||
# Metal3 Fake Kubernetes API server system (Metal3-FKAS) | ||
|
||
## FKAS | ||
|
||
Metal3-FKAS tool for testing CAPI-related projects. | ||
When being asked, it generates new fake kubernetes api endpoint, that responds | ||
to all typical requests that CAPI sends to a newly provisioned cluster. | ||
|
||
Despite being developed for Metal3 ecosystem, FKAS is provider-free. It can be adopted | ||
and used by any CAPI provider with intention to test the provider provisioning ability, | ||
without using real nodes. | ||
|
||
### Purpose | ||
|
||
After a CAPI Infrastructure provisions a new cluster, CAPI will send queries | ||
towards the newly launched cluster's API server to verify that the cluster is | ||
fully up and running. | ||
|
||
In a simulated provisioning process, there are no real nodes, hence we cannot | ||
have an actual API server running inside the node. Booting up a real kubelet | ||
and etcd servers elsewhere is possible, but these processes are likely to consume | ||
a lot of resources. | ||
|
||
FKAS is useful in this situation. When a request is sent towards `/register` endpoint, | ||
it will spawn a new simulated kubernetes API server with *unique* a host and | ||
port pair. | ||
User can, then, inject the address into cluster template consumed by | ||
CAPI with any infra provider. | ||
|
||
### How to use | ||
|
||
You can build the `metal3-fkas` image that is suitable for | ||
your local environment with | ||
|
||
```shell | ||
make build-fkas | ||
``` | ||
|
||
The result is an image with label `quay.io/metal3-io/metal3-fkas:<your-arch-name>` | ||
|
||
Alternatively, you can also build a custom image with | ||
|
||
```shell | ||
cd hack/fake-apiserver | ||
docker build -t <custom tag> . | ||
``` | ||
|
||
For local tests, it's normally needed to load the image into the cluster. | ||
For e.g. with `minikube` | ||
|
||
```shell | ||
docker image save -o /tmp/api-server.tar <image-name> | ||
minikube image load /tmp/api-server.tar | ||
``` | ||
|
||
Now you can deploy this container to the cluster, for e.g. with a deployment | ||
|
||
```yaml | ||
# fkas-deployment.yaml | ||
--- | ||
apiVersion: apps/v1 | ||
kind: Deployment | ||
metadata: | ||
name: fkas | ||
namespace: default | ||
spec: | ||
replicas: 1 | ||
selector: | ||
matchLabels: | ||
strategy: | ||
app: metal3-fkas | ||
type: Recreate | ||
template: | ||
metadata: | ||
labels: | ||
app: metal3-fkas | ||
spec: | ||
containers: | ||
- image: quay.io/metal3-io/metal3-fkas:amd64 | ||
imagePullPolicy: IfNotPresent | ||
name: fkas | ||
env: | ||
- name: POD_IP | ||
valueFrom: | ||
fieldRef: | ||
fieldPath: status.podIP | ||
name: apiserver | ||
``` | ||
```shell | ||
kubectl apply -f fkas-deployment.yaml | ||
``` | ||
|
||
After building the container image and deploy it to the bootstrap kubernetes cluster, | ||
you need to create a tunnel to send request to it and get response, by using | ||
a LoadBalancer, or a simple port-forward | ||
|
||
```shell | ||
fkas_pod_name=$(kubectl get pods -l app=metal3-fkas -o jsonpath="{.items[0].metadata.name}") | ||
kubectl port-forward pod/${fkas_pod_name} 3333:3333 2>/dev/null& | ||
``` | ||
|
||
Now, you can generate a fake API server endpoint by sending | ||
a GET request to the fake API server. But first, let's generate some needed certificates | ||
|
||
```shell | ||
openssl req -x509 -subj "/CN=Kubernetes API" -new -newkey rsa:2048 \ | ||
-nodes -keyout "/tmp/ca.key" -sha256 -days 3650 -out "/tmp/ca.crt" | ||
openssl req -x509 -subj "/CN=ETCD CA" -new -newkey rsa:2048 \ | ||
-nodes -keyout "/tmp/etcd.key" -sha256 -days 3650 -out "/tmp/etcd.crt" | ||
``` | ||
|
||
```shell | ||
caKeyEncoded=$(cat /tmp/ca.key | base64 -w 0) | ||
caCertEncoded=$(cat /tmp/ca.crt | base64 -w 0) | ||
etcdKeyEncoded=$(cat /tmp/etcd.key | base64 -w 0) | ||
etcdCertEncoded=$(cat /tmp/etcd.crt | base64 -w 0) | ||
namespace=<cluster-namespace> | ||
cluster_name=<cluster-name> | ||
|
||
cluster_endpoint=$(curl -X POST "localhost:3333/register" \ | ||
-H "Content-Type: application/json" -d '{ | ||
"resource": "'$namespace/$cluster_name'", | ||
"caKey": "'$caKeyEncoded'", | ||
"caCert": "'$caCertEncoded'", | ||
"etcdKey": "'$etcdKeyEncoded'", | ||
"etcdCert": "'$etcdCertEncoded'" | ||
}') | ||
``` | ||
|
||
The fake API server will return a response with the ip and port of the newly | ||
generated api server. For example: | ||
|
||
```json | ||
{ | ||
Host: "10.244.0.83", | ||
Port: 20000 | ||
} | ||
``` | ||
|
||
We also need to manually create the `ca-secret` and `etcd-secret` of the new cluster, | ||
so that they match the certs used by the api server. | ||
|
||
```shell | ||
host=$(echo ${cluster_endpoints} | jq -r ".Host") | ||
port=$(echo ${cluster_endpoints} | jq -r ".Port") | ||
|
||
cat <<EOF > "/tmp/${cluster}-ca-secrets.yaml" | ||
apiVersion: v1 | ||
kind: Secret | ||
metadata: | ||
labels: | ||
cluster.x-k8s.io/cluster-name: ${cluster} | ||
name: ${cluster}-ca | ||
namespace: ${namespace} | ||
type: kubernetes.io/tls | ||
data: | ||
tls.crt: ${caCertEncoded} | ||
tls.key: ${caKeyEncoded} | ||
EOF | ||
|
||
kubectl -n ${namespace} apply -f /tmp/${cluster}-ca-secrets.yaml | ||
|
||
cat <<EOF > "/tmp/${cluster}-etcd-secrets.yaml" | ||
apiVersion: v1 | ||
kind: Secret | ||
metadata: | ||
labels: | ||
cluster.x-k8s.io/cluster-name: ${cluster} | ||
name: ${cluster}-etcd | ||
namespace: ${namespace} | ||
type: kubernetes.io/tls | ||
data: | ||
tls.crt: ${etcdCertEncoded} | ||
tls.key: ${etcdKeyEncoded} | ||
EOF | ||
|
||
kubectl -n ${namespace} apply -f /tmp/${cluster}-etcd-secrets.yaml | ||
``` | ||
|
||
A new cluster can be provisioned by feeding a CAPI infrastructure provider | ||
(for e.g. CAPM3) with the host and port we got from FKAS. | ||
|
||
```shell | ||
# Injecting the new api address into the cluster | ||
export CLUSTER_APIENDPOINT_HOST="${host}" | ||
export CLUSTER_APIENDPOINT_PORT="${port}" | ||
|
||
# | ||
clusterctl generate cluster "${cluster}" \ | ||
--from "${CLUSTER_TEMPLATE}" \ | ||
--target-namespace "${namespace}" > /tmp/${cluster}-cluster.yaml | ||
kubectl apply -f /tmp/${cluster}-cluster.yaml | ||
``` | ||
|
||
After the cluster is created, CAPI will expect that information like node name | ||
and provider ID is registered in the API server. Since our API server doesn't | ||
live inside the node, we will need to feed the info to it, by sending a | ||
GET request to `/updateNode` endpoint: | ||
|
||
```shell | ||
curl -X POST "localhost:3333/updateNode" -H "Content-Type: application/json" -d '{ | ||
"resource": "${namespace}/${cluster}", | ||
"nodeName": "<machine-object-name>", | ||
"providerID": "<provider-id>", | ||
"uuid": "<node-uuid>", | ||
"nodeType": "<node-type>" | ||
}' | ||
``` | ||
|
||
Here `nodeType` should be either `control-plane` or `worker` (In fact, | ||
anything not equals to `control-plane` will be treated as a worker) | ||
|
||
### Acknowledgements | ||
|
||
This was developed thanks to the implementation of | ||
[Cluster API Provider In Memory (CAPIM)](https://github.com/kubernetes-sigs/cluster-api/tree/main/test/infrastructure/inmemory). | ||
|
||
## Metal3 FKAS System | ||
|
||
### FKAS in Metal3 | ||
|
||
In metal3 ecosystem, currently we have two ways of simulating a workflow without | ||
using any baremetal or virtual machines: | ||
|
||
- [FakeIPA container](https://github.com/metal3-io/utility-images/tree/main/fake-ipa) | ||
- BMO simulation mode | ||
|
||
In both of these cases, the "nodes" are not able to boot up any kubernetes api server, | ||
hence the needs of having mock API servers on-demands. | ||
|
||
Similar to the general case, after having BMHs provisioned to `available` state, | ||
the user can send a request towards the Fake API server endpoint `/register`, | ||
which will spawn a new API server, with an unique `Host` and `Port` pair. | ||
|
||
User can, then, use this IP address to feed the cluster template, by exporting | ||
`CLUSTER_APIENDPOINT_HOST` and `CLUSTER_APIENDPOINT_PORT` variables. | ||
|
||
There is no need of manually check and send node info to `/updateNode`, as we have | ||
another tool to automate that part. | ||
|
||
### Metal3-FKAS-Reconciler | ||
|
||
This tool runs as a side-car container alongside FKAS, and works specifically | ||
for Metal3. It eliminates the needs of user to manually fetch the nodes information | ||
and send to `/updateNode` (as described earlier), by constantly watch the changes | ||
in BMH objects, notice if a BMH is being provisioned to a kubernetes node, and | ||
send request to `updateNode` with appropriate information. | ||
|
||
### Deployment | ||
|
||
The `metal3-fkas-system` (including `fkas` and `metal3-fkas-reconciler`) | ||
can be deployed with the `k8s/metal3-fkas-system-deployment.yaml` file. | ||
|
||
```shell | ||
kubectl apply -f k8s/metal3-fkas-system-deployment.yaml | ||
``` | ||
|
||
## Disclaimer | ||
|
||
This is intended for development environments only. | ||
Do **NOT** use it in production. |
Oops, something went wrong.