My teaching website, built with Docusaurus. Visit at https://teach.silasberger.ch/.
yarn install
: Install dependencies.SITE=teach yarn run start
: Start the dev server for theteach
site.SITE=teach yarn build
: Create a release build for theteach
site.
For more examples, check out the CI workflows in .github
.
A good place to start is to run SITE=drafts yarn start
and then visit http://localhost:3000/default/Components/.
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
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
, changeTEST_USERNAME
to a different user ./runClone.sh
- In a private window, open http://localhost:3001
SITE
: Which site should be built; must correspond to an entry inconfig/siteProperties/siteProperties.ts
.LOG_LEVEL
: Only applicable during build stage; 1=WARN
, 2=INFO
, 3=DEBUG
.
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>
.
- 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 inmySite.scriptsConfigs.yaml
. - Create a site properties file
config/siteProperties/mySite.siteProperties.ts
. - Add an entry for
mySite
inconfig/siteProperties/siteProperties.ts
. - Create a directory
content/sites/mySite
. - If required: Add the new site to the deployment pipeline's matrix definition.
- Add a script config object
"someScript": {markers: {}, mappings: []}
toconfig/scriptsConfigs/mySite.scriptsConfigs.yaml
. - Define markers and mappings as needed.
- Add an entry for
someScript
toconfig/sidebars/mySite.sidebars.ts
. - Make sure to include a mapping with
{"section": "/index.md", ...}
to have a proper entry page for the script.
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 thehello-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": {...}
)
- Page path:
- 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
- Page path:
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:
- The candidate that was explicitly (i.e. not recursively) mapped by a mapping entry in the script config.
- Out of the candidates which explicitly (i.e. not recursively) matched a marker, the one with the highest specificity (=lower specificity number).
- The candidate that was implicitly mapped my a mapping entry in the script config (i.e. the child of an explicitly mapped parent directory).
- 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).
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"]
}
]
}
}
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 filehello.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 specificity3
and a markerbar
with specificity1
, then the source candidatehello.[foo,bar,baz].md
will receive a specificity of1
. - 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 asSome-Topic.[bar]
per-se. However, it will nevertheless discover and include itsfoo
-marked children, such asSome-Topic.[bar]/A-Sub-Topic.[foo]/*
orSome-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 todrafts.scriptsConfigs.yaml
, or mark a file or directory withdraft
(or any other) draft-specific marker defined indrafts.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 markerdraft
.
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)
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
.
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.
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
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 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.
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.
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
).