Update readme.md

This commit is contained in:
Lukas_Weberruß 2025-07-18 10:50:39 +00:00
parent adf144ab30
commit 32a13476f5

219
readme.md
View file

@ -1,16 +1,17 @@
# Umfassende Anleitung: Moodle auf STACKIT Cloud Foundry # Umfassende Anleitung: Moodle auf STACKIT Cloud Foundry
Diese Anleitung dokumentiert den gesamten Prozess, um eine Moodle-Instanz auf STACKIT Cloud Foundry zu deployen. Sie umfasst die Ersteinrichtung der CF-Umgebung, die Vorbereitung der Moodle-Anwendung inklusive aller notwendigen Anpassungen für den Proxy- und CDN-Betrieb sowie die Konfiguration für eine externe Datenbank, S3-Dateispeicher, Redis-Caching und Autoskalierung. Diese Anleitung dokumentiert den gesamten Prozess, um eine Moodle-Instanz auf STACKIT Cloud Foundry zu deployen. Sie ist als allgemeingültige Vorlage konzipiert und verwendet Platzhalter, die an Ihre Umgebung angepasst werden müssen.
## Architektur ## Architektur
Die folgende Abbildung zeigt die Architektur der Moodle-Umgebung auf STACKIT als Mermaid-Diagramm. Die folgende Abbildung zeigt die Architektur der Moodle-Umgebung. CDN und Redis sind als optionale, aber für den Produktivbetrieb empfohlene Komponenten dargestellt.
```mermaid ```mermaid
graph TD graph TD
A[Anwender] --> B(STACKIT DNS); A[Anwender] --> B(STACKIT DNS);
B --> C(STACKIT CDN); B --> C(STACKIT CDN<br/>(Optional));
C --> D{Cloud Foundry Router}; C -.-> D{Cloud Foundry Router};
B --> D;
subgraph Cloud Foundry Plattform subgraph Cloud Foundry Plattform
D --> E1[Moodle Instanz 1]; D --> E1[Moodle Instanz 1];
@ -19,16 +20,16 @@ graph TD
subgraph Backing Services subgraph Backing Services
F[(MariaDB<br/>Datenbank)]; F[(MariaDB<br/>Datenbank)];
G[(Redis<br/>Sessions & Locks)]; G[(Redis<br/>Sessions & Locks<br/>(Optional))];
H[(S3 Object Storage<br/>Dateien)]; H[(S3 Object Storage<br/>Dateien)];
end end
E1 --> F; E1 --> F;
E1 --> G; E1 -.-> G;
E1 --> H; E1 --> H;
E2 --> F; E2 --> F;
E2 --> G; E2 -.-> G;
E2 --> H; E2 --> H;
```` ````
@ -36,20 +37,16 @@ graph TD
### 1.1. CLIs installieren ### 1.1. CLIs installieren
Für diese Anleitung werden zwei Command Line Interfaces (CLIs) benötigt:
* **Cloud Foundry CLI:** Das primäre Werkzeug zur Interaktion mit der Cloud Foundry Plattform. * **Cloud Foundry CLI:** Das primäre Werkzeug zur Interaktion mit der Cloud Foundry Plattform.
* **Download:** [github.com/cloudfoundry/cli/releases](https://github.com/cloudfoundry/cli/releases) * **Download:** `https://github.com/cloudfoundry/cli/releases`
* **STACKIT CLI:** Wird benötigt, um STACKIT-spezifische Services wie Object Storage zu verwalten. * **STACKIT CLI:** Wird benötigt, um STACKIT-spezifische Services wie DNS, Object Storage und CDN zu verwalten.
* **Installationsanleitung:** Befolgen Sie die offizielle Anleitung unter [docs.stackit.cloud](https://docs.stackit.cloud/stackit/en/service-plans-redis-75142939.html). * **Installationsanleitung:** `https://docs.stackit.cloud/stackit/de/automatisieren/cli/cli-installieren-200720436.html`
### 1.2. Bei STACKIT Cloud Foundry anmelden ### 1.2. Bei STACKIT Cloud Foundry anmelden
Nach der Installation des CF CLI müssen Sie sich bei der STACKIT Cloud Foundry API anmelden.
* **API-Endpunkt:** `https://api.cf.eu01.stackit.cloud` * **API-Endpunkt:** `https://api.cf.eu01.stackit.cloud`
* **Login-Befehl:** * **Login-Befehl:**
@ -60,24 +57,10 @@ Nach der Installation des CF CLI müssen Sie sich bei der STACKIT Cloud Foundry
### 1.3. Organisation und Space erstellen ### 1.3. Organisation und Space erstellen
Eine "Organisation" ist der höchste Mandant, ein "Space" eine Umgebung (z.B. Entwicklung, Produktion).
1. **Organisation erstellen:**
```bash ```bash
cf create-org DEINE_ORGANISATION cf create-org <DEINE_ORGANISATION>
``` cf create-space <DEIN_SPACE> -o <DEINE_ORGANISATION>
cf target -o <DEINE_ORGANISATION> -s <DEIN_SPACE>
2. **Space erstellen:**
```bash
cf create-space produktion -o DEINE_ORGANISATION
```
3. **Ziel festlegen:**
```bash
cf target -o DEINE_ORGANISATION -s produktion
``` ```
## 2\. Moodle-Projekt lokal vorbereiten ## 2\. Moodle-Projekt lokal vorbereiten
@ -89,9 +72,9 @@ Eine "Organisation" ist der höchste Mandant, ein "Space" eine Umgebung (z.B. En
wget https://packaging.moodle.org/stable500/moodle-5.0.1.tgz wget https://packaging.moodle.org/stable500/moodle-5.0.1.tgz
tar -xzf moodle-5.0.1.tgz tar -xzf moodle-5.0.1.tgz
# Redis-Plugin für Moodle installieren # Redis-Plugin für Moodle installieren (Version passend zu Moodle 5.x)
wget https://moodle.org/plugins/download.php/29037/cachestore_redis_moodle41_2022112800.zip wget https://moodle.org/plugins/download.php/30019/cachestore_redis_moodle44_2024051300.zip
unzip cachestore_redis_moodle41_2022112800.zip unzip cachestore_redis_moodle44_2024051300.zip
mv redis moodle/cache/stores/ mv redis moodle/cache/stores/
# S3 Object File System Plugin installieren # S3 Object File System Plugin installieren
@ -102,8 +85,6 @@ cd ..
### 2.2. Buildpack- & System-Workarounds einrichten ### 2.2. Buildpack- & System-Workarounds einrichten
Diese Schritte sind für das `cloudfoundry/php-buildpack` notwendig.
```bash ```bash
# Web-Verzeichnis festlegen und PHP-Version auf 8.2 erzwingen # Web-Verzeichnis festlegen und PHP-Version auf 8.2 erzwingen
echo '{"WEBDIR": ".", "PHP_VERSION": "{PHP_82_LATEST}"}' > moodle/.bp-config/options.json echo '{"WEBDIR": ".", "PHP_VERSION": "{PHP_82_LATEST}"}' > moodle/.bp-config/options.json
@ -128,42 +109,105 @@ ln -s /tmp/php-fpm.pid moodle/php/var/run/php-fpm.pid
#### 3.1. Object Storage Bucket erstellen (STACKIT CLI) #### 3.1. Object Storage Bucket erstellen (STACKIT CLI)
Verwenden Sie das STACKIT CLI, um einen Bucket zu erstellen.
```bash ```bash
# Erstellt einen neuen Bucket stackit object-storage bucket create --name <DEIN_S3_BUCKET_NAME>
stackit object-storage bucket create --name <BUCKET_NAME>
``` ```
Anschließend benötigen Sie Zugangsdaten, um auf den Bucket zugreifen zu können.
```bash ```bash
# Erstellt neue Zugangsdaten für Ihr Projekt
# Sie müssen Ihre STACKIT Projekt-ID angeben.
stackit object-storage credential create --project-id <DEINE_PROJEKT_ID> stackit object-storage credential create --project-id <DEINE_PROJEKT_ID>
``` ```
**Notieren Sie sich den ausgegebenen `accessKey` und `secretAccessKey`.** Sie werden im nächsten Schritt benötigt. **Notieren Sie sich den ausgegebenen `accessKey` und `secretAccessKey`.**
#### 3.2. Cloud Foundry Services erstellen (CF CLI) #### 3.2. CDN-Distribution erstellen (STACKIT CLI, Optional)
Diese Befehle legen die für Moodle notwendigen CF-Services an. 1. **CDN-Distribution erstellen:**
```bash
# Ersetzen Sie <DEINE_PROJEKT_ID>, <DEINE_INTERNE_CF_ROUTE> und <DEINE_MOODLE_DOMAIN>.
stackit curl -X POST --body '{
"name": "<DEIN_CDN_DISTRIBUTION_NAME>",
"origin": {
"path": "/",
"hostname": "<DEINE_INTERNE_CF_ROUTE>",
"port": 443
},
"originRequestHeaders": [
{
"name": "Host",
"values": [
"<DEINE_MOODLE_DOMAIN>"
]
},
{
"name": "Cache-Control",
"values": [
"no-cache, no-store, must-revalidate"
]
}
],
"enabled": true
}' /cdn/v1beta/projects/<DEINE_PROJEKT_ID>/distributions
```
**Notieren Sie sich die `id` der Distribution aus der Antwort.**
2. **CDN-Standard-Domain auslesen:**
```bash
# Ersetzen Sie <DEINE_PROJEKT_ID> und <DISTRIBUTION_ID>.
stackit curl -X GET /cdn/v1beta/projects/<DEINE_PROJEKT_ID>/distributions/<DISTRIBUTION_ID>
```
**Notieren Sie sich den Wert des `domain`-Feldes aus der Antwort. Dies ist Ihr CNAME-Ziel.**
3. **Benutzerdefinierte Domain hinzufügen:**
```bash
# Ersetzen Sie <DEINE_PROJEKT_ID> und <DISTRIBUTION_ID>.
stackit curl -X POST --body '{
"domain": "<DEINE_MOODLE_DOMAIN>"
}' /cdn/v1beta/projects/<DEINE_PROJEKT_ID>/distributions/<DISTRIBUTION_ID>/domains
```
#### 3.3. DNS-Einträge konfigurieren (STACKIT CLI)
1. **Zone-ID ermitteln:**
```bash
stackit dns zone list
```
Notieren Sie sich die `ID` Ihrer Domain-Zone.
2. **CNAME-Eintrag erstellen:**
```bash
# Für den Betrieb MIT CDN
# <DEIN_CDN_ENDPOINT> ist die Standard-Domain, die Sie in Schritt 3.2 ausgelesen haben.
stackit dns record-set create --zone-id <DEINE_ZONE_ID> --name <DEINE_MOODLE_DOMAIN> --type CNAME --ttl 300 --records "<DEIN_CDN_ENDPOINT>."
# Für den Betrieb OHNE CDN (falls benötigt)
stackit dns record-set create --zone-id <DEINE_ZONE_ID> --name <DEINE_INTERNE_CF_ROUTE> --type CNAME --ttl 300 --records "console.apps.01.cf.eu01.stackit.cloud."
```
#### 3.4. Cloud Foundry Services erstellen (CF CLI)
```bash ```bash
# Datenbank-Service # Datenbank-Service
cf create-service <dein-mariadb-service> <plan> moodle-demo-mariadb cf create-service <dein-mariadb-service> <plan> <DEINE_MARIADB_INSTANZ>
# Redis-Service (Optional) # Redis-Service (Optional)
cf create-service appcloud-redis7 redis-4.16.100-replica moodle-redis cf create-service appcloud-redis7 redis-4.16.100-replica <DEINE_REDIS_INSTANZ>
# Autoscaler-Service # Autoscaler-Service
cf create-service autoscaler autoscaler-free-plan moodle-autoscaler cf create-service autoscaler autoscaler-free-plan <DEIN_AUTOSCALER_INSTANZ>
# User-Provided Service für die S3-Credentials aus dem vorherigen Schritt # User-Provided Service für die S3-Credentials
cf create-user-provided-service moodle-s3-credentials -p '{ cf create-user-provided-service <DEINE_S3_CUPS_INSTANZ> -p '{
"access_key": "DEIN_ZUVOR_GENERIERTER_ACCESS_KEY", "access_key": "<DEIN_ZUVOR_GENERIERTER_ACCESS_KEY>",
"secret_key": "DEIN_ZUVOR_GENERIERTER_SECRET_KEY", "secret_key": "<DEIN_ZUVOR_GENERIERTER_SECRET_KEY>",
"bucket": "DEIN_ZUVOR_GENERIERTES_BUCKET", "bucket": "<DEIN_S3_BUCKET_NAME>",
"endpoint": "https://object.storage.eu01.onstackit.cloud/" "endpoint": "https://object.storage.eu01.onstackit.cloud/"
}' }'
``` ```
@ -172,8 +216,6 @@ cf create-user-provided-service moodle-s3-credentials -p '{
#### a) Manifest-Dateien #### a) Manifest-Dateien
Erstellen Sie zwei Manifest-Dateien, um einfach zwischen dem Betrieb mit und ohne CDN wechseln zu können.
**`manifest-cdn.yml` (Produktion):** **`manifest-cdn.yml` (Produktion):**
```yaml ```yaml
@ -185,17 +227,17 @@ applications:
buildpack: https://github.com/cloudfoundry/php-buildpack.git buildpack: https://github.com/cloudfoundry/php-buildpack.git
path: moodle path: moodle
routes: routes:
- route: moodle.cf.demo.stackit.rocks - route: <DEINE_INTERNE_CF_ROUTE>
- route: moodle.demo.stackit.rocks - route: <DEINE_MOODLE_DOMAIN>
env: env:
USE_CDN: 'true' USE_CDN: 'true'
WWW_ROOT: "https://moodle.demo.stackit.rocks" WWW_ROOT: "https://<DEINE_MOODLE_DOMAIN>"
BP_PHP_LOG_LEVEL: WARNING BP_PHP_LOG_LEVEL: WARNING
services: services:
- moodle-demo-mariadb - <DEINE_MARIADB_INSTANZ>
- moodle-redis - <DEINE_REDIS_INSTANZ>
- moodle-autoscaler - <DEIN_AUTOSCALER_INSTANZ>
- moodle-s3-credentials - <DEINE_S3_CUPS_INSTANZ>
autoscaling: autoscaling:
min_instances: 2 min_instances: 2
max_instances: 4 max_instances: 4
@ -225,16 +267,16 @@ applications:
buildpack: https://github.com/cloudfoundry/php-buildpack.git buildpack: https://github.com/cloudfoundry/php-buildpack.git
path: moodle path: moodle
routes: routes:
- route: moodle.cf.demo.stackit.rocks - route: <DEINE_INTERNE_CF_ROUTE>
env: env:
USE_CDN: 'false' USE_CDN: 'false'
WWW_ROOT: "https://moodle.cf.demo.stackit.rocks" WWW_ROOT: "https://<DEINE_INTERNE_CF_ROUTE>"
BP_PHP_LOG_LEVEL: WARNING BP_PHP_LOG_LEVEL: WARNING
services: services:
- moodle-demo-mariadb - <DEINE_MARIADB_INSTANZ>
- moodle-redis - <DEINE_REDIS_INSTANZ>
- moodle-autoscaler - <DEIN_AUTOSCALER_INSTANZ>
- moodle-s3-credentials - <DEINE_S3_CUPS_INSTANZ>
autoscaling: autoscaling:
min_instances: 1 min_instances: 1
max_instances: 4 max_instances: 4
@ -243,14 +285,11 @@ applications:
#### b) Dynamische `config.php` #### b) Dynamische `config.php`
Kopieren Sie `moodle/config-dist.php` zu `moodle/config.php` und ersetzen Sie den gesamten Inhalt mit dieser finalen, dynamischen Vorlage. Kopieren Sie `moodle/config-dist.php` zu `moodle/config.php` und ersetzen Sie den Inhalt mit dieser Vorlage.
```php ```php
<?php // Moodle configuration file <?php // Moodle configuration file
// =========================================================================
// UMWELTSPEZIFISCHE KONFIGURATION
// =========================================================================
$useCdn = (getenv('USE_CDN') === 'true'); $useCdn = (getenv('USE_CDN') === 'true');
$wwwRoot = getenv('WWW_ROOT'); $wwwRoot = getenv('WWW_ROOT');
@ -258,38 +297,28 @@ unset($CFG);
global $CFG; global $CFG;
$CFG = new stdClass(); $CFG = new stdClass();
//=========================================================================
// 1. WEBSEITE, VERZEICHNISSE & PROXY EINSTELLUNGEN
//=========================================================================
$CFG->wwwroot = $wwwRoot; $CFG->wwwroot = $wwwRoot;
$CFG->dirroot = __DIR__; $CFG->dirroot = __DIR__;
$CFG->dataroot = '/tmp/moodledata'; $CFG->dataroot = '/tmp/moodledata';
$CFG->admin = 'admin'; $CFG->admin = 'admin';
$CFG->directorypermissions = 0777; $CFG->directorypermissions = 0777;
// --- Proxy-Einstellungen ---
$CFG->sslproxy = 1; $CFG->sslproxy = 1;
$CFG->slasharguments = false; $CFG->slasharguments = false;
if ($useCdn) { if ($useCdn) {
// Diese Einstellungen sind NUR für den CDN-Betrieb notwendig, um den "reverseproxyabused"-Fehler zu vermeiden.
$CFG->reverseproxy = 1; $CFG->reverseproxy = 1;
$CFG->trustedproxies = '127.0.0.1, 10.0.0.0/8, 172.16.0.0/12, 192.168.0.0/16, 185.124.192.185'; $CFG->trustedproxies = '127.0.0.1, 10.0.0.0/8, 172.16.0.0/12, 192.168.0.0/16, <IP_DES_CDN_KNOTENS>';
} }
// --- Workaround: Manuelles Erstellen des dataroot-Verzeichnisses ---
if (!file_exists($CFG->dataroot)) { if (!file_exists($CFG->dataroot)) {
@mkdir($CFG->dataroot, $CFG->directorypermissions, true); @mkdir($CFG->dataroot, $CFG->directorypermissions, true);
} }
//=========================================================================
// 2. DATENBANK-SETUP
//=========================================================================
$CFG->dbtype = 'mariadb'; $CFG->dbtype = 'mariadb';
$CFG->dblibrary = 'native'; $CFG->dblibrary = 'native';
$CFG->prefix = 'mdl_'; $CFG->prefix = 'mdl_';
$CFG->dboptions = array ('dbpersist' => 0, 'dbcollation' => 'utf8mb4_unicode_ci'); $CFG->dboptions = array ('dbpersist' => 0, 'dbcollation' => 'utf8mb4_unicode_ci');
// --- Lese Service-Konfigurationen aus VCAP_SERVICES ---
$vcap_services_json = getenv('VCAP_SERVICES'); $vcap_services_json = getenv('VCAP_SERVICES');
if ($vcap_services_json) { if ($vcap_services_json) {
$vcap_services = json_decode($vcap_services_json, true); $vcap_services = json_decode($vcap_services_json, true);
@ -310,9 +339,9 @@ if ($vcap_services_json) {
} }
} }
// Redis (Optional, für bessere Performance) // Redis
$redis_service_key = null; $redis_service_key = null;
if (isset($vcap_services['appcloud-redis60'])) { $redis_service_key = 'appcloud-redis60'; } if (isset($vcap_services['appcloud-redis7'])) { $redis_service_key = 'appcloud-redis7'; }
if ($redis_service_key && isset($vcap_services[$redis_service_key][0]['credentials'])) { if ($redis_service_key && isset($vcap_services[$redis_service_key][0]['credentials'])) {
$redis_creds = $vcap_services[$redis_service_key][0]['credentials']; $redis_creds = $vcap_services[$redis_service_key][0]['credentials'];
$CFG->session_handler_class = '\cachestore_redis\session_handler'; $CFG->session_handler_class = '\cachestore_redis\session_handler';
@ -325,7 +354,7 @@ if ($vcap_services_json) {
// S3 // S3
if (isset($vcap_services['user-provided'])) { if (isset($vcap_services['user-provided'])) {
foreach ($vcap_services['user-provided'] as $service) { foreach ($vcap_services['user-provided'] as $service) {
if ($service['name'] === 'moodle-s3-credentials') { if ($service['name'] === '<DEINE_S3_CUPS_INSTANZ>') {
$s3_creds = $service['credentials']; $s3_creds = $service['credentials'];
$CFG->objectfs_s3_key = $s3_creds['access_key']; $CFG->objectfs_s3_key = $s3_creds['access_key'];
$CFG->objectfs_s3_secret = $s3_creds['secret_key']; $CFG->objectfs_s3_secret = $s3_creds['secret_key'];
@ -336,24 +365,17 @@ if ($vcap_services_json) {
} }
} }
} }
// Fallback auf Datenbank-Sessions, falls Redis nicht konfiguriert ist
if (empty($CFG->session_handler_class)) { if (empty($CFG->session_handler_class)) {
$CFG->session_handler_class = '\core\session\database'; $CFG->session_handler_class = '\core\session\database';
} }
//=========================================================================
// SESSION COOKIE HANDLING (bedingt durch CDN)
//=========================================================================
if ($useCdn) { if ($useCdn) {
$CFG->cookiepath = '/'; $CFG->cookiepath = '/';
$CFG->cookiedomain = 'moodle.demo.stackit.rocks'; $CFG->cookiedomain = '<DEINE_MOODLE_DOMAIN>';
$CFG->cookiesecure = 1; $CFG->cookiesecure = 1;
$CFG->cookiehttponly = 1; $CFG->cookiehttponly = 1;
} }
//=========================================================================
// DO NOT ADD ANYTHING BELOW THIS LINE
//=========================================================================
require_once(__DIR__ . '/lib/setup.php'); require_once(__DIR__ . '/lib/setup.php');
``` ```
@ -367,8 +389,6 @@ Wählen Sie das gewünschte Manifest für Ihr Deployment aus.
cf push -f manifest-cdn.yml cf push -f manifest-cdn.yml
``` ```
**Hinweis:** Der CDN-Fall hat sich als problematisch erwiesen. Wenn Fehler auftreten, liegt es wahrscheinlich an den Caching- oder Header-Einstellungen im CDN selbst, die hier nicht behandelt werden können.
**Ohne CDN (Test / Direkter Zugriff):** **Ohne CDN (Test / Direkter Zugriff):**
```bash ```bash
@ -376,6 +396,3 @@ cf push -f manifest-no-cdn.yml
``` ```
Nach Abschluss des Deployments und dem Durchlaufen der Moodle-Web-Installation ist Ihre Instanz einsatzbereit. Nach Abschluss des Deployments und dem Durchlaufen der Moodle-Web-Installation ist Ihre Instanz einsatzbereit.
```
```