Rust in Openshift deployen

von Dr. Liv Fischer | 27. Juli 2018 | Allgemein, Deutsch, Software Engineering

Ich mag keinen Bloat. Programme, die essentiell nichts tun und ein Gigabyte Speicher und eine halbe CPU belegen gehen mir nicht in den Sinn. Jedes belegte Byte und jeder ausgeführte Takt sind ein potentielles Versteck für Bugs. Je weniger ausgeführt wird, desto sicherer kann man sich sein, dass das Programm auch in jedem Fall das Richtige ™ tut. Code ist kein Asset, sondern eine Liability: kleiner ist besser. Und Code, in den man nicht reingucken kann, ist der Schlimmste. Ich will also unbedingt weg von Java und der unsäglichen proprietären Oracle JVM mit ihren Unwägbarkeiten, seltsamem GC-Verhalten und ihrem perversen Speicherhunger. Kurz: meine Quest ist Rust! Ergo habe ich mich beim gestrigen Senacor-24h-Sprint auch dieses Themas angenommen, dieses Mal allerdings mit dem Fokus darauf, die Anwendung auch tatsächlich im Netzwerk des Kunden zu deployen, um meine Pitches mit direkten Vergleichen von Performance und Ressourcennutzung untermauern zu können.

Das Programm

Für dieses Mal habe ich einen simplen Netzwerkservice gebaut und auf Rocket verzichtet, da ich für Produktionscode den stabilen Releasekanal von Rust verwenden möchte. Das Programm nutzt daher Actix Web als Webframework. Es ist ausgesprochen minimal und kann mit

rustup self update
rustup update stable
cargo init rust-service
cd rust-service

aufgesetzt werden. Anschließend kann die Cargo.toml editiert werden, um Actix hinzuzufügen:

[package]
name = "rust-service"
version = "0.1.0"
authors = ["Fischer, Maxi <maxi.fischer@senacor.com>"]
[dependencies]
actix-web = "0.7"

Das eigentliche Programm besteht in unserem Fall aus einer sehr schlanken main.rs:

extern crate actix_web;
use actix_web::{server, App, HttpRequest};
fn hello(_req: &HttpRequest) -> &'static str {
    "Hello world!\n"
}
fn main() {
    server::new(|| App::new().resource("/hello", |r| r.f(hello)))
        .bind("127.0.0.1:6000")
        .unwrap()
        .run();
}

Hierbei ist /hello unser Endpunkt, der lediglich einen String zurück liefert. Mit dem Schnipsel Gluecode in fn main teilen wir Actix mit, dass die Funktion fn hello der Handler dafür sein soll und dass wir auf TCP/IP-Port 6000 auf HTTP-Anfragen warten wollen. Mit .unwrap() sagen wir dem Programm, es möge sich ordentlich beenden, falls dabei ein Fehler (wie z.B. dass der Port schon belegt ist) auftritt. Die letzte Zeile in fn main schließlich startet den Server.

Ausprobieren kann man dieses simple Programm jetzt ganz leicht mit

cargo run

Beim ersten Ausführen wird Cargo automatisch alle Abhängigkeiten herunterladen und kompilieren. Für erfahrene Javaprogrammierer wird das Kompilieren etwas länger dauern als sie gewohnt sind, aber dafür erhalten wir auch eine Binärdatei, die nativ auf unserem System läuft und keine zig Megabyte große Runtime benötigt, auf deren Quellcode wir keinen Zugriff haben. Als zusätzlicher Bonus ist die Speicherverwaltung komplett statisch und unsere Performance daher besser vorhersagbar.

Bauen in Openshift

Da der Jenkins meines Kunden keine Rust-Toolchain zur Verfügung stellt, habe ich mich entschieden, den Service direkt in Openshift zu bauen. Um im produktiv deployten Pod jedoch keine Toolchain herumschleppen zu müssen, die nutzlos Platz verschwendet und ein potentielles Sicherheitsrisiko darstellt, würde ich am liebsten intermediäre Dockercontainer im Build nutzen. Da OpenShift 3.9 aber leider kein Docker 17.05 und damit keine Multi-Stage Builds unterstützt, müssen wir zwei Dockerfiles und entsprechend verkettete Buildkonfigurationen nutzen.

Bauen

Um Rust in unserem privaten Namespace nutzen zu können, können wir einen ImageStream erstellen und dem offiziellen RedHat-Stream folgen lassen.

apiVersion: image.openshift.io/v1
kind: ImageStream
metadata:
  name: rust
  namespace: build
spec:
  dockerImageRepository: registry.access.redhat.com/devtools/rust-toolset-7-rhel7

Momentan kann Openshift diese noch nicht automatisch synchronisieren; dazu müssen wir ein Kommando mit dem Kommandozeilenclient ausführen:

oc import-image rust

Dockerfile.build:

FROM devtools/rust-toolset-7-rhel7:1.26.2-7
USER 0
COPY ./ ./
RUN scl enable rust-toolset-7 'cargo build --release'
RUN mkdir -p /build-out && cp target/release/rust-service /build-out/app
RUN ls -lh /build-out/

In Redhat müssen wir die Buildtools mit scl enable rust-toolset-7 erst aktivieren, bevor wir sie nutzen können. Das fertige Artefakt kopieren wir nach /build-out/app, von wo der nächste Schritt es abholen wird. Die FROM Zeile wird dabei von Openshift ignoriert und durch den entsprechenden Eintrag in unserer Buildconfig ersetzt. Diese sieht so aus:

apiVersion: build.openshift.io/v1
kind: BuildConfig
metadata:
  labels:
    build: rust-service-build
  name: rust-service-build
  namespace: my-namespace
spec:
  output:
    to:
      kind: ImageStreamTag
      name: 'rust-service:latest'
  resources:
    requests:
      cpu: 250m
      memory: 1Gi
    limits:
      cpu: 2000m
      memory: 4Gi
  runPolicy: Serial
  source:
    contextDir: rust-service
    git:
      ref: master
      uri: 'https://git.company.com/rust.git'
    sourceSecret:
      name: builder-secret
    type: Git
  strategy:
    dockerStrategy:
      dockerfilePath: Dockerfile.build
      from:
        kind: ImageStreamTag
        name: 'rust:latest'
        namespace: build
    type: Docker

Ein entsprechender ImageStream rust-service muss dabei ebenfalls existieren. Wichtig ist hier, die Resource-Limits nicht zu klein anzusetzen, rustc ist doch recht speicherhungrig. Diese Kosten zahlt man dafür nur beim Bauen und nicht beim Betreiben der Anwendung!

Deployen

Nun haben wir einen ImageStream mit unseren fertigen Artefakten. Diese müssen wir jetzt noch mit Hilfe eines Deploymentbuilds in einen schlanken Container verpacken. Dazu benötigen wir noch ein (sehr schlankes) Dockerfile.run,

FROM scratch
COPY ./build-out/app /
CMD ["/app"]

sowie eine dazugehörige Buildkonfiguration:

apiVersion: build.openshift.io/v1
kind: BuildConfig
metadata:
  labels:
    build: rust-deployment
  name: rust-deployment
  namespace: build
spec:
  failedBuildsHistoryLimit: 5
  output:
    to:
      kind: ImageStreamTag
      name: 'rust-deployment:latest'
  resources:
    limits:
      cpu: '1'
      memory: 1Gi
    requests:
      cpu: 250m
      memory: 1Gi
  runPolicy: Serial
  source:
    contextDir: rust-service
    git:
      ref: master
      uri: 'https://git.company.com/rust.git'
    images:
      - from:
          kind: ImageStreamTag
          name: 'rust-service:latest'
          namespace: build
        paths:
          - destinationDir: rust-service
            sourcePath: /build-out
    type: Git
  strategy:
    dockerStrategy:
      dockerfilePath: Dockerfile.run
      from:
        kind: ImageStreamTag
        name: 'nn-rhel:7.4-129'
        namespace: build
    type: Docker
  successfulBuildsHistoryLimit: 5
  triggers:
    - imageChange:
      type: ImageChange

Der Trigger sorgt dafür, dass der Deploymentbuild angestoßen wird, wenn der „Buildbuild“ erfolgreich durchgelaufen ist. Mit dem from Teil importieren wir den Stream aus dem vorherigen Kapitel und stellen dessen Inhalt dem neuen Dockerbuild zur Verfügung. Das destinationDir berücksichtigt dabei allerdings nicht das contextDir, dieses muss also nochmal explizit angegeben werden.

Nun aber wirklich wirklich deployen

Jetzt haben wir tatsächlich einen echten, nutzbaren Container. Fehlt nur noch die Deploymentkonfiguration:

apiVersion: apps.openshift.io/v1
kind: DeploymentConfig
metadata:
  labels:
    app: rust-app
    deploymentconfig: rust-app
  name: rust-app
  namespace: prod
spec:
  replicas: 1
  revisionHistoryLimit: 10
  selector:
    app: rust-app
    deploymentconfig: rust-app
  strategy:
    activeDeadlineSeconds: 21600
    recreateParams:
      timeoutSeconds: 100
    resources:
      requests:
        cpu: 300m
        memory: 128Mi
    rollingParams:
      intervalSeconds: 1
      maxSurge: 1
      maxUnavailable: 0
      timeoutSeconds: 100
      updatePeriodSeconds: 1
    type: Rolling
  template:
    metadata:
      labels:
        app: rust-app
        deploymentconfig: rust-app
    spec:
      containers:
        - image: 'docker-registry.default.svc:5000/build/rust-deployment:latest'
          imagePullPolicy: Always
          livenessProbe:
            failureThreshold: 10
            httpGet:
              path: /rust
              port: 6000
              scheme: HTTP
            initialDelaySeconds: 20
            periodSeconds: 15
            successThreshold: 1
            timeoutSeconds: 20
          name: rusty-advk-mock
          ports:
            - containerPort: 8080
              protocol: TCP
          readinessProbe:
            failureThreshold: 5
            httpGet:
              path: /rust
              port: 6000
              scheme: HTTP
            initialDelaySeconds: 20
            periodSeconds: 10
            successThreshold: 1
            timeoutSeconds: 20
          resources:
            limits:
              cpu: '1'
              memory: 256Mi
            requests:
              cpu: 200m
              memory: 128Mi
          securityContext:
            capabilities: {}
            privileged: false
          terminationMessagePath: /dev/termination-log
          terminationMessagePolicy: File
      dnsPolicy: ClusterFirst
      restartPolicy: Always
      schedulerName: default-scheduler
      securityContext: {}
      terminationGracePeriodSeconds: 30
  triggers:
    - type: ConfigChange

Where does the newborn go from here

Nun da wir eine erfolgreich laufende Anwendung haben, sollten wir sie mit Leben füllen. Wir setzen hier im Projekt auf Schnittstellenbeschreibungen mit Swagger und sind es aus dem Spring Boot Umfeld gewohnt, uns Servercode mittels Delegatepattern generieren zu lassen, da generierte Stubs zu viele Nachteile mit sich bringen. Momentan gibt es zwar Codegeneratoren für Rust, diese sind aber in einem so jämmerlichen Zustand, dass wir hier noch viel Arbeit vor uns haben. Doch das wird Inhalt eines zukünftigen Artikels sein.