Entwicklung, Staging und Produktion so ähnlich wie möglich halten
##X. Dev-Prod-Vergleichbarkeit
Historisch gibt es große Lücken zwischen Entwicklung (wo ein Entwickler live an einem lokalen Deploy der App Änderungen macht) und Produktion (ein laufender Deploy der App, auf den Endbenutzer zugreifen). Diese Lücken zeigen sich auf drei Gebieten:
- Die Zeit-Lücke Ein Entwickler arbeitet an Code der Tage, Wochen oder sogar Monate braucht um in Produktion zu gehen.
- Die Personal-Lücke: Entwickler schreiben Code, Operatoren deployen ihn.
- Die Werkzeug-Lücke: Entwickler nutzen vielleicht einen Stack wie Nginx, SQLite und OS X - die Produktion nutzt Apache, MySQL und Linux.
Die Zwölf-Faktor-App ist ausgelegt auf Continuous Deployment indem sie die Lücke zwischen Entwicklung und Produktion klein hält. Mit Blick auf die oben beschriebenen drei Lücken:
- Die Zeit-Lücke klein halten: Ein Entwickler kann Code schreiben, der Stunden oder sogar Minuten später deployed wird.
- Die Personal-Lücke klein halten: Entwickler die Code schreiben sind intensiv am Deployment und der Überwachung des Verhaltens auf Produktion beteiligt.
- Die Werkzeug-Lücke klein halten: Entwicklung und Produktion so ähnlich wie möglich halten.
Das Gesagte in einer Tabelle:
Traditionelle App | Zwölf-Faktor-App | |
---|---|---|
Zeit zwischen Deployments | Wochen | Stunden |
Code-Autoren und Code-Deployer | Andere Menschen | Dieselben Menschen |
Entwicklungs- und Produktions-Umgebung | Unterschiedlich | So ähnlich wie möglich |
Im Bereich der unterstützenden Dienste wie der Datenbank der App, dem Queue-System oder dem Cache ist die Dev-Prod-Vergleichbarkeit wichtig. Viele Sprachen bieten Bibliotheken, die den Zugriff auf die unterstützenden Dienste vereinfachen und ebenso Adapter für unterschiedliche Arten von Diensten.
Art | Sprache | Bibliothek | Adapter |
---|---|---|---|
Datenbank | Ruby/Rails | ActiveRecord | MySQL, PostgreSQL, SQLite |
Queue | Python/Django | Celery | RabbitMQ, Beanstalkd, Redis |
Cache | Ruby/Rails | ActiveSupport::Cache | Speicher, Dateisystem, Memcached |
Für Entwickler ist es sehr elegant, einen leichtgewichtigen unterstützenden Dienst in der lokalen Umgebung zu benutzen, während ein ernst zu nehmender und robuster unterstützender Dienst in Produktion verwendet wird. So kann man SQLite lokal und PostgreSQL in Produktion benutzen; oder zum Cachen den lokalen Speicher in Entwicklung und Memcached in Produktion.
Der Zwölf-Faktor-Entwickler widersteht dem Drang, verschiedene unterstützende Dienste in Entwicklung und Produktion zu verwenden, selbst wenn Adapter über alle Unterschiede hinweg abstrahieren. Unterschiede zwischen den unterstützenden Diensten verursachen kleinste Inkompatiblitäten, und Code, der in Entwicklung oder Staging funktioniert und Tests besteht, scheitert in Produktion. Diese Reibungskosten und die dann notwendige Dämpfung des Continuous Deployment sind sehr hoch, wenn man sie über die Lebensdauer einer App aggregiert.
Leichtgewichtige lokale Dienste überzeugen weniger als früher. Moderne unterstützende Dienste wie Memcached, PostgreSQL und RabbitMQ sind dank moderner Paketierungs-Systeme wie Homebrew und apt-get nicht schwierig zu installieren und zu starten. Auch deklarative Installationssysteme wie Chef oder Puppet in Kombination mit leichtgewichtigen virtuellen Umgebungen wie Vagrant setzen Entwickler in den Stand, lokale Umgebungen ans Laufen zu bringen, die nahe an Produktionsumgebungen herankommen. Die Installations- und Betriebskosten dieser Systeme sind gering verglichen mit dem Nutzen der Dev-Prod-Vergleichbarkeit und einem Continuous Deployment.
Adapter für unterschiedliche unterstützende Dienste sind weiterhin von Nutzen, weil sie das Portieren auf andere unterstützende Dienste schmerzlos machen. Aber alle Deploys der App (Entwicklungsumgebungen, Staging, Produktion) sollten denselben Typ und dieselbe Version eines jeden unterstützenden Diensts benutzen.