6
0
mirror of https://github.com/ChristianLight/tutor.git synced 2025-01-24 22:08:24 +00:00

fix: upgrade MySQL from 5.7 to 8.1 first and then to 8.4 (#1149)

* fix: upgrade MySQL from 5.7 to 8.1 first and then to 8.4
This is required when upgrading from Tutor v15 to v18 directly
MySQL does not allow direct upgrades from v5.7 to v8.4

* fix: run MySQL 8.1 as a separate container during upgrade from Olive to Redwood (#1140)
We do this because MySQL 8.1 does not have the --mysql-native-password option
We have this option turned on for backward compatibility
This commit is contained in:
Danyal Faheem 2024-10-30 17:00:48 +05:00 committed by GitHub
parent dbb91cb38f
commit e2786afa58
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 236 additions and 6 deletions

View File

@ -0,0 +1,2 @@
- [Bugfix] Do not directly upgrade MySQL from v5.7 to v8.4 when upgrading from quince as MySQL does not allow that. First, upgrade to v8.1 and then to v8.4. (by @Danyal-Faheem)
This process should be automatic for most users. However, if you are running a third-party MySQL (i.e., RUN_MYSQL=false), you are expected to perform this process yourself. Please refer to the third-party provider's documentation for detailed instructions. Ensuring that your MySQL version is up-to-date is crucial for maintaining compatibility and security.

View File

@ -0,0 +1 @@
- [Bugfix] Run MySQL 8.1 as a separate container during upgrade from Olive to Redwood as it crashed otherwise due to the `--mysql-native-password` option not being present. (by @Danyal-Faheem)

View File

@ -1,10 +1,13 @@
from __future__ import annotations from __future__ import annotations
from typing import Optional
import click import click
from packaging import version
from tutor import config as tutor_config from tutor import config as tutor_config
from tutor import fmt, plugins from tutor import fmt, plugins
from tutor.types import Config from tutor.types import Config, get_typed
def upgrade_from_lilac(config: Config) -> None: def upgrade_from_lilac(config: Config) -> None:
@ -60,6 +63,34 @@ def get_mongo_upgrade_parameters(
return mongo_version, admin_command return mongo_version, admin_command
def get_intermediate_mysql_upgrade(config: Config) -> Optional[str]:
"""
Checks if a MySQL upgrade is needed based on the Tutor version and MySQL setup.
This method ensures that MySQL is running and determines if the upgrade
process should proceed based on the Tutor version. It is intended for upgrades
from Tutor version 15 to version 18 or later. Manual upgrade steps are not
required for versions 16 or 17.
Returns:
Optional[str]: The docker image of MySQL to upgrade to or None if not applicable
"""
if not get_typed(config, "RUN_MYSQL", bool):
fmt.echo_info(
"You are not running MySQL (RUN_MYSQL=false). It is your "
"responsibility to upgrade your MySQL instance to v8.4. There is "
"nothing left to do to upgrade from Olive."
)
return None
image_tag = get_typed(config, "DOCKER_IMAGE_MYSQL", str).split(":")[-1]
# If latest image, we assign a constant value to invalidate the condition
# as we know that the latest image will always be greater than 8.1.0
target_version = (
version.Version("8.1.1") if image_tag == "latest" else version.parse(image_tag)
)
return "docker.io/mysql:8.1.0" if target_version > version.parse("8.1.0") else None
PALM_RENAME_ORA2_FOLDER_COMMAND = """ PALM_RENAME_ORA2_FOLDER_COMMAND = """
if stat '/openedx/data/ora2/SET-ME-PLEASE (ex. bucket-name)' 2> /dev/null; then if stat '/openedx/data/ora2/SET-ME-PLEASE (ex. bucket-name)' 2> /dev/null; then
echo "Renaming ora2 folder..." echo "Renaming ora2 folder..."

View File

@ -4,8 +4,9 @@ import click
from tutor import config as tutor_config from tutor import config as tutor_config
from tutor import env as tutor_env from tutor import env as tutor_env
from tutor import hooks
from tutor import fmt from tutor import fmt
from tutor.commands import compose from tutor.commands import compose, jobs
from tutor.types import Config from tutor.types import Config
from . import common as common_upgrade from . import common as common_upgrade
@ -158,6 +159,63 @@ def upgrade_from_olive(context: click.Context, config: Config) -> None:
upgrade_mongodb(context, config, "4.2.17", "4.2") upgrade_mongodb(context, config, "4.2.17", "4.2")
upgrade_mongodb(context, config, "4.4.22", "4.4") upgrade_mongodb(context, config, "4.4.22", "4.4")
intermediate_mysql_docker_image = common_upgrade.get_intermediate_mysql_upgrade(
config
)
if not intermediate_mysql_docker_image:
return
click.echo(fmt.title(f"Upgrading MySQL to {intermediate_mysql_docker_image}"))
# We start up a mysql-8.1 container to build data dictionary to preserve
# the upgrade order of 5.7 -> 8.1 -> 8.4
# Use the mysql-8.1 context so that we can clear these filters later on
with hooks.Contexts.app("mysql-8.1").enter():
hooks.Filters.ENV_PATCHES.add_items(
[
(
"local-docker-compose-services",
"""
mysql-8.1:
extends: mysql
image: docker.io/mysql:8.1.0
command: >
mysqld
--character-set-server=utf8mb3
--collation-server=utf8mb3_general_ci
--binlog-expire-logs-seconds=259200
""",
),
(
"local-docker-compose-jobs-services",
"""
mysql-8.1-job:
image: docker.io/mysql:8.1.0
depends_on: {{ [("mysql-8.1", RUN_MYSQL)]|list_if }}
""",
),
]
)
hooks.Filters.CONFIG_DEFAULTS.add_item(("MYSQL_HOST", "mysql-8.1"))
hooks.Filters.CLI_DO_INIT_TASKS.add_item(
("mysql-8.1", tutor_env.read_core_template_file("jobs", "init", "mysql.sh"))
)
tutor_env.save(context.obj.root, config)
# Run the init command to make sure MySQL is ready for connections
context.invoke(jobs.initialise, limit="mysql-8.1")
context.invoke(compose.stop, services=["mysql-8.1"])
# Clear the filters added for mysql-8.1 as we don't need them anymore
hooks.clear_all(context="app:mysql-8.1")
# Save environment and run init for mysql 8.4 to make sure MySQL is ready
tutor_env.save(context.obj.root, config)
context.invoke(jobs.initialise, limit="mysql")
context.invoke(compose.stop, services=["mysql"])
def upgrade_from_quince(context: click.Context, config: Config) -> None: def upgrade_from_quince(context: click.Context, config: Config) -> None:
click.echo(fmt.title("Upgrading from Quince")) click.echo(fmt.title("Upgrading from Quince"))

View File

@ -2,7 +2,7 @@ import click
from tutor import config as tutor_config from tutor import config as tutor_config
from tutor import env as tutor_env from tutor import env as tutor_env
from tutor import fmt from tutor import fmt, hooks
from tutor.commands import k8s from tutor.commands import k8s
from tutor.commands.context import Context from tutor.commands.context import Context
from tutor.types import Config from tutor.types import Config
@ -39,7 +39,7 @@ def upgrade_from(context: click.Context, from_release: str) -> None:
running_release = "olive" running_release = "olive"
if running_release == "olive": if running_release == "olive":
upgrade_from_olive(context.obj, config) upgrade_from_olive(context, config)
running_release = "palm" running_release = "palm"
if running_release == "palm": if running_release == "palm":
@ -148,11 +148,11 @@ def upgrade_from_maple(context: Context, config: Config) -> None:
) )
def upgrade_from_olive(context: Context, config: Config) -> None: def upgrade_from_olive(context: click.Context, config: Config) -> None:
# Note that we need to exec because the ora2 folder is not bind-mounted in the job # Note that we need to exec because the ora2 folder is not bind-mounted in the job
# services. # services.
k8s.kubectl_apply( k8s.kubectl_apply(
context.root, context.obj.root,
"--selector", "--selector",
"app.kubernetes.io/name=lms", "app.kubernetes.io/name=lms",
) )
@ -165,6 +165,144 @@ def upgrade_from_olive(context: Context, config: Config) -> None:
upgrade_mongodb(config, "4.2.17", "4.2") upgrade_mongodb(config, "4.2.17", "4.2")
upgrade_mongodb(config, "4.4.22", "4.4") upgrade_mongodb(config, "4.4.22", "4.4")
intermediate_mysql_docker_image = common_upgrade.get_intermediate_mysql_upgrade(
config
)
if not intermediate_mysql_docker_image:
return
click.echo(fmt.title(f"Upgrading MySQL to {intermediate_mysql_docker_image}"))
# We start up a mysql-8.1 container to build data dictionary to preserve
# the upgrade order of 5.7 -> 8.1 -> 8.4
# Use the mysql-8.1 context so that we can clear these filters later on
with hooks.Contexts.app("mysql-8.1").enter():
hooks.Filters.ENV_PATCHES.add_items(
[
(
"k8s-deployments",
"""
---
apiVersion: apps/v1
kind: Deployment
metadata:
name: mysql-81
labels:
app.kubernetes.io/name: mysql-81
spec:
selector:
matchLabels:
app.kubernetes.io/name: mysql-81
strategy:
type: Recreate
template:
metadata:
labels:
app.kubernetes.io/name: mysql-81
spec:
securityContext:
runAsUser: 999
runAsGroup: 999
fsGroup: 999
fsGroupChangePolicy: "OnRootMismatch"
containers:
- name: mysql-81
image: docker.io/mysql:8.1.0
args:
- "mysqld"
- "--character-set-server=utf8mb3"
- "--collation-server=utf8mb3_general_ci"
- "--binlog-expire-logs-seconds=259200"
env:
- name: MYSQL_ROOT_PASSWORD
value: "{{ MYSQL_ROOT_PASSWORD }}"
ports:
- containerPort: 3306
volumeMounts:
- mountPath: /var/lib/mysql
name: data
securityContext:
allowPrivilegeEscalation: false
volumes:
- name: data
persistentVolumeClaim:
claimName: mysql
""",
),
(
"k8s-jobs",
"""
---
apiVersion: batch/v1
kind: Job
metadata:
name: mysql-81-job
labels:
app.kubernetes.io/component: job
spec:
template:
spec:
restartPolicy: Never
containers:
- name: mysql-81
image: docker.io/mysql:8.1.0
""",
),
]
)
hooks.Filters.ENV_PATCHES.add_item(
(
"k8s-services",
"""
---
apiVersion: v1
kind: Service
metadata:
name: mysql-81
labels:
app.kubernetes.io/name: mysql-81
spec:
type: ClusterIP
ports:
- port: 3306
protocol: TCP
selector:
app.kubernetes.io/name: mysql-81
""",
)
)
hooks.Filters.CONFIG_DEFAULTS.add_item(("MYSQL_HOST", "mysql-81"))
hooks.Filters.CLI_DO_INIT_TASKS.add_item(
("mysql-81", tutor_env.read_core_template_file("jobs", "init", "mysql.sh"))
)
tutor_env.save(context.obj.root, config)
# Run the init command to make sure MySQL is ready for connections
k8s.kubectl_apply(
context.obj.root,
"--selector",
"app.kubernetes.io/name=mysql-81",
)
k8s.wait_for_deployment_ready(config, "mysql-81")
context.invoke(k8s.do.commands["init"], limit="mysql-8.1")
context.invoke(k8s.stop, names=["mysql-81"])
# Clear the filters added for mysql-8.1 as we don't need them anymore
hooks.clear_all(context="app:mysql-8.1")
# Save environment and run init for mysql 8.4 to make sure MySQL is ready
tutor_env.save(context.obj.root, config)
k8s.kubectl_apply(
context.obj.root,
"--selector",
"app.kubernetes.io/name=mysql",
)
k8s.wait_for_deployment_ready(config, "mysql")
context.invoke(k8s.do.commands["init"], limit="mysql")
context.invoke(k8s.stop, names=["mysql"])
def upgrade_from_quince(config: Config) -> None: def upgrade_from_quince(config: Config) -> None:
click.echo(fmt.title("Upgrading from Quince")) click.echo(fmt.title("Upgrading from Quince"))