How To Update The Release Version Of An Open edX Deployment
Open edX Releases
Open edX cuts new named releases approximately every 6 months. Each of our deployments needs to be able to independently upgrade the respective version on their own schedule. MITx Residential upgrades twice per year, roughly aligned with the release timing of the upstream Open edX platform, whereas xPro has typically upgraded on an annual basis in the December timeframe.
Our MITx Online deployment of Open edX is a special case in that it deploys directly from the master branches of the different applications and has a weekly release schedule.
Open Learning's Open edX Deployments
At Open Learning we have four distinct installations of Open edX, each with their own release cadence and configuration requirements. For each of those deployments we have multiple environment stages (e.g. CI, QA, Production). Each of those deployment * stages combinations needs to be able to have their versions managed independently, with newer versions being progressively promoted to higher environments.
This is further complicated by the long testing cycles for new releases in a given deployment. We need to be able to deploy a newer release to a lower environment stage, while maintaining the ability to deploy patches of the current release in a given time period to subsequent environments. For example, as we start testing the Palm release for the MITx Residential deployment we need to get it installed in the CI environment. Meanwhile, we may have a bug-fix or security patch that needs to be tested in the QA environment and propagated to Production.
In addition to the cartesian product of (versions * deployments * env stages) we also have to manage these values across the different components of the Open edX ecosystem. For example, the core of the platform is called edx-platform, which in turn relies on: - a forum service - the edx-notes API service - a codejail REST API service - and myriad MFEs (Micro-FrontEnds).
For any of these different components we may need to be able to override the repository and/or branch that we are building/deploying from.
Versioning of Build and Deploy
In order to support this sea of complexity we have a module of bridge.settings.openedx
that has a combination of data structures and helper functions to control what
applications and versions get deployed where and when. These values are then used in the
build stages of the different components, as well as driving the deployment pipelines in
Concourse that are responsible for orchestrating all of this.
Managing Supported Releases
We only want to support the build and deployment of a minimal subset of Open edX
releases. This is controlled by the Enum OpenEdxSupportedRelease
in the
bridge.settings.openedx.types
module. When there is a new release created it needs to
be added to this Enum. For example, to add support for the Palm release a new line is
added of the format palm = ("palm", "open-release/palm.master")
When we have upgraded all deployments past a given release we remove it from that Enum so that we no longer need to worry about maintaining configuration/code/etc. for that release.
Performing An Upgrade
There are two data structures that control the applications and versions that get
included in a given deployment and which version to use in the respective environment
stage. The OpenLearningOpenEdxDeployment
Enum is the top level that sets the release
name for the environment stage of a given deployment.
There are situations where we need to customize a component that is being deployed. In
those cases we typically create a fork of the upstream repository where we manage the
patches that we require. The ReleaseMap
dictionary is used to manage any overrides of
repository and branch for a given component of the platform, as well as which components
are included in that deployment. The OpenEdxApplicationVersion
structure will map to
the default repositories and branches for a given component, but supplies a
branch_override
and origin_override
parameter to manage these customizations.
For example, to upgrade our MITx Residential deployments to start testing the Palm
release we change the CI
stages of the mitx
and mitx-staging
deployments to use
the palm
value for the OpenEdxSupportedRelease
@@ -13,7 +13,7 @@ class OpenLearningOpenEdxDeployment(Enum):
mitx = DeploymentEnvRelease(
deployment_name="mitx",
env_release_map=[
- EnvRelease("CI", OpenEdxSupportedRelease["olive"]),
+ EnvRelease("CI", OpenEdxSupportedRelease["palm"]),
EnvRelease("QA", OpenEdxSupportedRelease["olive"]),
EnvRelease("Production", OpenEdxSupportedRelease["olive"]),
],
@@ -21,7 +21,7 @@ class OpenLearningOpenEdxDeployment(Enum):
mitx_staging = DeploymentEnvRelease(
deployment_name="mitx-staging",
env_release_map=[
- EnvRelease("CI", OpenEdxSupportedRelease["olive"]),
+ EnvRelease("CI", OpenEdxSupportedRelease["palm"]),
EnvRelease("QA", OpenEdxSupportedRelease["olive"]),
EnvRelease("Production", OpenEdxSupportedRelease["olive"]),
],
Because Palm is a new release for these deployments we also need to add a palm
key to
the ReleaseMap
dictionary that contains the applications that are associated with
those deployments and the appropriate OpenEdxApplicationVersion
records for that
deployment.
@@ -61,6 +61,122 @@ ReleaseMap: dict[
OpenEdxSupportedRelease,
dict[OpenEdxDeploymentName, list[OpenEdxApplicationVersion]],
] = {
+ "palm": {
+ "mitx": [
+ OpenEdxApplicationVersion(
+ application="edx-platform", # type: ignore
+ application_type="IDA",
+ release="palm",
+ branch_override="mitx/palm",
+ origin_override="https://github.com/mitodl/edx-platform",
+ ),
...
+ ],
+ "mitx-staging": [
+ OpenEdxApplicationVersion(
+ application="edx-platform", # type: ignore
+ application_type="IDA",
+ release="palm",
+ branch_override="mitx/palm",
+ origin_override="https://github.com/mitodl/edx-platform",
+ ),
...
+ ],
+ },
"olive": {
"mitx": [
OpenEdxApplicationVersion(
All of the deployment
pipelines
for these application components are managed by a corresponding meta
pipeline that
will automatically update the build and pipeline configuration based on the changed
version information as soon as it is merged into the master
branch of
ol-infrastructure
.
Supporting New Applications
There are two categories of applications that comprise the overall Open edX platform. These are "IDA"s (Independently Deployable Applications), and "MFE"s (Micro Front-Ends). An IDA is a Django application that can be deployed as a backend service and typically integrates with the edx-platform (LMS and CMS) via OAuth. An MFE is a ReactJS application that is deployed as a standalone site which then interacts with LMS and/or CMS via REST APIs.
In order to ensure that we have visibility into which of these components we are
deploying there is an Enum of
OpenEdxApplication
and
OpenEdxMicroFrontend
respectively which captures the details of which elements of the overall edX ecosystem
we are supporting in our deployments. These Enums are then used as an attribute of the
OpenEdxApplicationVersion
model which captures the details of a specific instance of one of those
components. These are then used in the ReleaseMap
dict to show which versions and
deployments include the given instance of that application and version.
To add support for a new IDA or MFE you need to add a new entry to the appropriate Enum,
and then include an instance of an OpenEdxApplicationVersion
that points to that
application in the ReleaseMap
at the appropriate location.
Note - TMM 2023-04-14
- The current configuration of our meta pipelines means that before the updates to the
bridge.settings.openedx.version_matrix
module can be picked up by themeta
pipelines theol-infrastructure
docker image needs to be built and pushed to our registry so that it can be loaded. This means that once the ol-infrastructure image pipeline completes it might be necessary to manually trigger the meta pipelines again. - Once a new release is deployed to a given environment stage for the first time it may be necessary to manually ensure that all database migrations are run properly. It will be attempted automatically on deployment, but there are often conflicts between the existing database state and the migration logic that require intervention.
Updating An MITOL OpenEdX Branch With The Latest
This involves updating our edx-platform fork and rebasing the correct branch.
For instance, to update our forked OpenEdX Sumac branch for MITxOnline, you would:
- Check out our fork of the edx-platform repo -
gh repo clone mitodl/edx-platform
- Change to the right branch -
git checkout mitx/sumac
- Update all the things -
git fetch --all
- Rebase from upstream -
git pull --rebase upstream open-release/sumac.master
- Check the results, and if correct push the branch -
git push origin mitx/sumac --force
Troubleshooting
OpenEdX Redwood
With each new OpenEdX release, inevitably there will be problems. For instance, in the Redwood release, we noticed that login for MITX CI was failing with a 500 response code.
So we ran the following Grafana query using the query builder:
{environment="mitx-ci", application="edxapp"} |= `` | json | line_format `{{.message}}
That yielded a problem with JWT signing keys as the exception cited a signing failure.
So we had to log onto an EC2 instance appropriate to MITX CI and run the following command from the lms container shell:
manage.py generate_jwt_signing_key
That produced public and private signing key JSON blobs, which we could then add to SOPS and then through the pipeline into Vault where they'll be picked up as Redwood deploys to the various apps and environments.