Skip to content

SilasBerger/teaching-website

Repository files navigation

Website

My teaching website, built with Docusaurus. Visit at https://teach.silasberger.ch/.

Install and build

  • yarn install: Install dependencies.
  • SITE=teach yarn run start: Start the dev server for the teach site.
  • SITE=teach yarn build: Create a release build for the teach site.

For more examples, check out the CI workflows in .github.

Where do I start?

A good place to start is to run SITE=drafts yarn start and then visit http://localhost:3000/default/Components/.

Basic Dev Setup

API

Repo: teaching-api

  • To use local login, set NO_AUTH=true in .env
  • To start with a fresh db, run scripts/purge_dev_services.sh
  • Run dev_services.compose.yml
  • yarn install
  • yarn run db:migrate
  • yarn run db:seed
  • yarn run dev
  • Open http://localhost:3000

Frontend

Repo: teaching-website

  • To work with local login, set TEST_USERNAME="[email protected]" (for a user that exists in the DB)
    • Set this user to admin in the DB
  • yarn install
  • SITE=gbsl yarn run start

To run a clone (especially with local login):

  • Create a second clone of this repo
  • yarn install
  • In .env, change TEST_USERNAME to a different user
  • ./runClone.sh
  • In a private window, open http://localhost:3001

General configuration

Environment variables

  • SITE: Which site should be built; must correspond to an entry in config/siteProperties/siteProperties.ts.
  • LOG_LEVEL: Only applicable during build stage; 1=WARN, 2=INFO, 3=DEBUG.

Config files

  • config/builderConfig.ts: Set global material and script root directories.
  • config/siteProperties/<site>.siteProperties.ts: Define general properties and global elements for <site>.
  • config/siteProperties/siteProperties.ts: Collect all <site>.siteProperties definitions.
  • config/sidebars/<site>.sidebars.ts: Define the sidebar configuration for <site>.
  • config/scriptsConfigs/<site>.scriptsConfigs.yaml: Define the scripts configurations for <site>.

Examples: Common configuration workflows

Configuring a new site called mySite

  • Create a scripts configs file config/scriptsConfigs/mySite.scriptsConfigs.yaml.
  • Create a sidebar file config/sidebars/mySite.sidebars.ts with one entry for each script defined in mySite.scriptsConfigs.yaml.
  • Create a site properties file config/siteProperties/mySite.siteProperties.ts.
  • Add an entry for mySite in config/siteProperties/siteProperties.ts.
  • Create a directory content/sites/mySite.
  • If required: Add the new site to the deployment pipeline's matrix definition.

Configuring a new script someScript for a site mySite

  • Add a script config object "someScript": {markers: {}, mappings: []} to config/scriptsConfigs/mySite.scriptsConfigs.yaml.
  • Define markers and mappings as needed.
  • Add an entry for someScript to config/sidebars/mySite.sidebars.ts.
  • Make sure to include a mapping with {"section": "/index.md", ...} to have a proper entry page for the script.

Content configuration

Pages

The pages root is defined in the SiteProperties#pagesRoot property in config/siteProperties/<site>.siteProperties.ts for a given site. The contents of the pages root directory are served /.

Warning: Make sure that the URL paths for pages and scrips do not collide! Examples:

  • Example 1: Okay, no collision; index.md and the hello-in-10-languages script root are siblings in /greetings/
    • Page path: /greetings/index.md
    • Script root: /greetings/hello-in-10-languages (from a script definition "greetings/hello-in-10-languages": {...})
  • Example 2: Problematic, collision; index.md sits inside the /hello-in-10-languages script root
    • Page path: /greetings/hello-in-10-languages/index.md
    • Script root: /greetings/hello-in-10-languages

Scripts

A script is a unique collection of elements from the material library. As such, it is a function of the material library and a set of configurations, executed and applied by the sync mechanism. From a technical point of view, the sync mechanism is a parameterized file copy process, where the sources and destinations are defined through configuration.

The root directory for the material library is defined by builderConfig.ts#MATERIAL_ROOT. The root directory for the scripts output directory is defined by builderConfig.ts#SCRIPTS_ROOT. The latter contains all script roots for a given site after building for that site (SITE=mySite yarn run build).

There are two ways to specify which material library elements should be copied to a particular script: the scripts config and _markers. These two approaches can be combined can be combined at will, whereby multiple library elements may point to the same destination in the script. In such cases, the following precedence rules define which of these library elements (in this case, called source candidates) will be copied the contested destination:

  1. The candidate that was explicitly (i.e. not recursively) mapped by a mapping entry in the script config.
  2. Out of the candidates which explicitly (i.e. not recursively) matched a marker, the one with the highest specificity (=lower specificity number).
  3. The candidate that was implicitly mapped my a mapping entry in the script config (i.e. the child of an explicitly mapped parent directory).
  4. Out of the candidates which implicitly matched a marker (i.e. children of a parent directory that matched a marker), the one with the highest specificity (=lower specificity number).

The scripts configs file

An excerpt of the scripts configs file <mySite>.scriptsConfigs.yaml for a site mySite might look as follows:

{
  "english": { # this object defines a script named "english"; its URL path will be /english
    "markers": { # markers definition (see below)
      "english": 0, # marker "english" has specificity 0 (the lower the number, the higher the precedence)
      "languages": 1 # marker "languages" has specificity 1
    },
    "mappings": [ # mappings definition
      # recursively copy the /General-Information directory from the material library to /01-General-Information in
      # the script 
      {"section": "/01-General-Information", "material": "/General-Information"},
      {"section": "/02-Units/01-Unit-1", "material": "/English/Units/Unit-1"},
      {"section": "/02-Units/02-Unit-2", "material": "/English/Units/Unit-2"}
    ]
  },
  "programming": {
    "markers": {
      "programming": 0
    },
    "mappings": [
      {"section": "/01-General-Information", "material": "/General-Information"},
      {
        # recursively copy the /Computer-Science/Programming/01-Introduction directory from the material library
        # to /02-Introduction-to-Programming in the script, but...
        "section": "/02-Introduction-to-Programming",
        "material": "/Computer-Science/Programming/01-Introduction",
        # ...from /Computer-Science/Programming/01-Introduction, ignore the file 04-Loops/03-do-while.mdx and the
        # entire "06-Exceptions" directory
        "ignore": ["04-Loops/03-do-while.mdx", "06-Exceptions"]
      }
    ]
  }
}

Markers

If a directory or filename contains a segment of the form .[], it is considered marked. Any comma-separated values within the two brackets are considered markers. For example, the file hello.[foo,bar].md has two markers: foo and bar.

During the sync process for a given script, the following rules apply to all marked files (and directories):

  • If none of the script's markers (as defined by its markers definition) match any of that file's markers, then that file is ignored.
  • If any of the file's markers match any of the script's markers, the file considered a source candidate for a destination file with the same name, but with the marker section removed. For example: a marked source file hello.[foo,bar].md would become a source candidate for a destination file hello.md.
  • Multiple marked files may be considered source candidates for the same destination file.
  • Source candidates inherit their specificity from the specificity of their most specific applicable marker (=lowest number). For example: a script defines a marker foo with specificity 3 and a marker bar with specificity 1, then the source candidate hello.[foo,bar,baz].md will receive a specificity of 1.
  • Multiple applicable markers on a given file may not have equal specificity within a given script.
  • Multiple source candidates for a given destination file may not yield equal specificity.

Additional useful information about the behavior of markers:

  • Directories with unmatched markers do not prevent their children with matched markers from being used as source candidates. For example: If a script has a marker [foo], it will ignore a directory such as Some-Topic.[bar] per-se. However, it will nevertheless discover and include its foo-marked children, such as Some-Topic.[bar]/A-Sub-Topic.[foo]/* or Some-Topic.[bar]/an-article.[foo].mdx
  • File or directory names with an empty marker list (e.g. some-article.[].md, Some-Topic.[]) are still considered marked, even though they will not match any applicable markers. This behavior can be used to "mark" a file or directory as an unpublished draft.
  • When working on a page that should not (yet) be mapped into any scripts (e.g. a placeholder page while developing a new component), it is recommended to use the drafts site as a drafting and development playground. Either add an explicit mapping to drafts.scriptsConfigs.yaml, or mark a file or directory with draft (or any other) draft-specific marker defined in drafts.scriptsConfigs.yaml), e.g. Path/To/Some/future-article.[draft].md. Since marked elements are discovered even when nested in a non-marked path, that file (and its parent hierarchy) will be included in any draft script with marker draft.

The final result

Assuming a material tree that looks as follows:

β”œβ”€β”€ General-Information
β”‚   β”œβ”€β”€ Required-Materials.[english]
β”‚   β”‚   β”œβ”€β”€ 01-Stationary.md
β”‚   β”‚   └── 02-Books.md
β”‚   β”œβ”€β”€ index.md
β”‚   β”œβ”€β”€ 01-Organizational-Matters.md
β”‚   β”œβ”€β”€ 02-Class-Rules.md
β”‚   β”œβ”€β”€ 02-Class-Rules.[languages].md
β”‚   β”œβ”€β”€ 03-Semester-Agenda.[programming].md
β”‚   └── 03-Semester-Agenda.[english].md
β”œβ”€β”€ Digital-Tools
β”‚   β”œβ”€β”€ word-processing.md
β”‚   └── programming-environment.[programming].md 
β”œβ”€β”€ English
β”‚   └── Units
β”‚       β”œβ”€β”€ Unit-1
β”‚       β”‚   └── ...
β”‚       └── Unit-2
β”‚           └── ...
└── Computer-Science
    β”œβ”€β”€ Programming
    β”‚   β”œβ”€β”€ 01-Introduction
    β”‚   β”‚   β”œβ”€β”€ 01-Hello-World
    β”‚   β”‚   β”‚   └── ...
    β”‚   β”‚   β”œβ”€β”€ 02-Variables
    β”‚   β”‚   β”‚   └── ...
    β”‚   β”‚   β”œβ”€β”€ 03-Conditionals
    β”‚   β”‚   β”‚   └── ...
    β”‚   β”‚   β”œβ”€β”€ 04-Loops
    β”‚   β”‚   β”‚   β”œβ”€β”€ index.md
    β”‚   β”‚   β”‚   β”œβ”€β”€ 01-for.mdx
    β”‚   β”‚   β”‚   β”œβ”€β”€ 02-while.mdx
    β”‚   β”‚   β”‚   └── 03-do-while.mdx
    β”‚   β”‚   β”œβ”€β”€ 05-Functions
    β”‚   β”‚   β”‚   └── ...
    β”‚   β”‚   └── 06-Exceptions
    β”‚   β”‚       └── ...
    β”‚   └── 02-Classes-and-Objects
    β”œβ”€β”€ Algorithms
    β”‚   └── ...
    └── Networks
        └── ...

The final english script at /path/to/scripts/english will look as follows:

β”œβ”€β”€ 01-General-Information  # explicitly mapped, include all non-marked contents
β”‚   β”œβ”€β”€ Required-Materials  # include directory because it was marked with [languages]
β”‚   β”‚   β”œβ”€β”€ 01-Stationary.md
β”‚   β”‚   └── 02-Books.md
β”‚   β”œβ”€β”€ index.md
β”‚   β”œβ”€β”€ 01-Organizational-Matters.md
β”‚   β”œβ”€β”€ 02-Class-Rules.md  # source file: 02-Class-Rules.[languages].md
β”‚   └── 03-Semester-Agenda.md  # source file: 03-Semester-Agenda.[english].md
└── 02-Units  # explicitly mapped, include all non-marked contents
    β”œβ”€β”€ 01-Unit-1 
    β”‚   └── ...
    └── 02-Unit-2
        └── ..

The final programming script at /path/to/scripts/programming will look as follows:

β”œβ”€β”€ 01-General-Information
β”‚   β”œβ”€β”€ index.md
β”‚   β”œβ”€β”€ 01-Organizational-Matters.md
β”‚   β”œβ”€β”€ 02-Class-Rules.md  # source file: 02-Class-Rules.md (unmarked)
β”‚   └── 03-Semester-Agenda  # source file: 03-Semester-Agenda.[programming].md
β”œβ”€β”€ Digital-Tools  # parent included because child was marked with [programming]
β”‚   └── programming-environment.md  # source file: programming-environment.[programming].md
└── 02-Introduction-to-Programming  # explicitly mapped, include all non-marked contents 
    β”œβ”€β”€ 01-Hello-World
    β”‚   └── ...
    β”œβ”€β”€ 02-Variables
    β”‚   └── ...
    β”œβ”€β”€ 03-Conditionals
    β”‚   └── ...
    β”œβ”€β”€ 04-Loops
    β”‚   β”œβ”€β”€ index.md
    β”‚   β”œβ”€β”€ 01-for.mdx
    β”‚   β”œβ”€β”€ 02-while.mdx
    β”‚   └── # missing: 03-do-while.mdx (ignored)
    β”œβ”€β”€ 05-Functions
    β”‚   └── ...
    └──  # missing: 06-Exceptions (ignored)

Concepts: Material, sites, pages, scripts, and the sync mechanism

Material

The material library is a central collection of sections and articles in the form of Markdown files. For every site, these sections and articles are selectively assembled into individual scripts as needed. The root directory for the material library is defined by builderConfig.ts#MATERIAL_ROOT.

Sites

A site represents...

  • from a technical point of view: a unique build target.
  • from the website creator's point of view: a specific deployment configuration, defined by its unique collection of pages and scripts.
  • from the website user's point of view: a unique website.

A site is defined by a set of configuration files and entries as described in the General configuration section.

Pages

The Docusaurus pages plugin is used to create on-off standalone pages. These are individual Markdown files that are not part of any particular script. Pages can for instance be used to create landing pages or entrypoints for navigating to scripts.

Each site defines a pages root directory in its config/siteProperties/<site>.siteProperties.ts configuration file. The contents of that directory are served at /. Looking at the following folder structure:

content
└── pages
    └── mysite
        β”œβ”€β”€ index.md  # page
        └── greetings
            └── hello-world.md  # page

Assuming that we are building the site mysite, and that the pagesRoot for mysite is set to content/pages/mysite. We would then see the resulting HTML files for our markdown file served as follows:

  • for index.md: / and /index.html
  • for hello-world.md: /greetings/hello-world and /greetings/hello-world.html

Scripts

A script is a unique collection of elements from the material library. It is defined by an entry in the site's *.scriptsConfigs.yaml file. The contents of the script are arranged via mapping entries in that scripts config entry, as well as through matching markers between the scripts config and filenames in the material library.

When building the respective site, each script definition yields a docs root for the Docusaurus Docs plugin. On that site, that docs root is served at /<scriptId>.

The sync mechanism

The sync mechanism is used to assemble contents from the material library into each individual script, as defined by that scripts configuration in the site's *.scriptsConfigs.yaml file and any applicable markers in the material filenames. docusaurus.config.ts marks the entrypoint into the build and sync process.

Deployment

Legacy sites

Some legacy sites may still be available in the codebase (i.e. have a .siteProperties.ts-file, etc.) but are no longer included in the deployment build matrix and are hence no longer being deployed on push. Currently, this is the case for the lerbermatt site.

Environment secrets (GitHub Actions)

The build_and_deploy_sites task in check_build_deploy.yml specifies the following environment:

environment: deploy_${{ matrix.site }}

From that, the runner automatically created an environment named deploy_${{ matrix.site }} on the first run with this config. Note that legacy environments (i.e. environments for sites which are no longer in the build matrix) will not be automatically removed. Environments can be modified here.

The secrets within these environments are available in the secrets context to use within the workflow spec. They must be explicitly included and specified as environment variables for the respective jobs, e.g.

steps:
    run: SITE=${{ matrix.site }} yarn run build
    env:
      CLIENT_ID: ${{ secrets.CLIENT_ID }}
      TENANT_ID: ${{ secrets.TENANT_ID }}
      APP_URL: ${{ secrets.APP_URL }}
      BACKEND_URL: ${{ secrets.BACKEND_URL }}
      API_URI: ${{ secrets.API_URI }}

If a field FOO is added as a variable, rather than a secret, it is instead available in the vars context: ${{ vars.FOO }}}.

If an environment for a particular site does not provide a given secret or variable, that value (and hence, the environment variable) will be undefined. If, at some point, we need a different login strategy for another site (e.g. username / password for teach), we would add additional environment variables / secrets to the workflow config, define the required secrets in the corresponding environment (in this example, the ones for username / pw in teach), and make sure that the build process can handle one "set" being undefined (the MSAL one for teach and the username / pw one for gbsl).