-
Notifications
You must be signed in to change notification settings - Fork 89
Home
Welcome to the Mozilla-Learning wiki! This page will take you through the general information about how the learning.mozilla.org codebase is structured, see the nav menu on the right for more specific additional pages.
The following is an onboarding document that describes how the learning site works from a code and run-time perspective.
-
route checks:
- redirect -> redirect client
- route based on lib/routes.jsx
- render page to HTML
- serve response
- try to resolve as WP stub
- try to resolve as static resource
- check if there's a locale in the path. If not, figure out which one to add and start route checks again with a redirect (see below)
- treat as 404
-
locale negotiation: When a path is requested that doesn't exactly match an existing route,
app.js
hands off the path to mofo-localize, along with the json in the build step generated from the .properties files and theaccept-language
http request header. Mofo-localize does some best guesses at which locale to send to the user based on the available ones. If it returns a redirect string,app.js
handles that and starts the route matching process again.
- all
<Route>
entries are nested as content to thecomponents/page.jsx
component,<Page>
.-
<Page>
is the master "template" component, and loads all component scaffolding.
-
- Specific pages are loaded from the
./pages
directory, and are set asthis.props.children
content to the Page template.- technically, they are cloned because additional props need to be added in, which cannot be done directly on the
this.props.children
value, but can be done usingReact.cloneElement
.
- technically, they are cloned because additional props need to be added in, which cannot be done directly on the
- Specific pages all specify a static
pageClassName
value, which is used by the<Page>
component to set the outermost CSS class. - Specific pages all specify a static
pageTitle
value, which is used by the<Page>
component to set the appropriate page title.
Broadly speaking, content is modelled using React, and React code is split over two locations.
-
./pages
houses all "webpage" files, where in the ideal case each one file mirrors one page of the learning website. -
./components
houses all the modular code used by one or more pages for specific kinds of content. Note that the<Page>
template is considered a component, not a page on its own.
Practically, some pages contain code either in their own file or in sibling files that should really be components, but were kept near their page so that they were easy to find or easy to edit.
One special component is the <Sidebar>
component, which acts as navigation element for the learning site.
Due to legacy reasons, it does not build a sidebar based on information in the routes.jsx
file, but has a hardcoded list of top level navigation items and nested sub items.
If you're on a page associated with some route, and this route allows for page content that swaps out the original page without affecting the URL users see, then our users have no way of returning to the base page content other than to first navigate "somewhere else" and then come back. In order to deal with this, there is a ./lib/resetreload
library that can be required in, with a one line API:
resetreload.shouldResetOnReload(true);
Calling this in your code (anywhere) will result in the Sidebar's entry for the current page to bypass normal React-Router behaviour (which is "you are navigating to the page you are already on, I will do nothing") and will instead call the page component's reset()
function, if it has one defined. If the page component has no reset()
function, nothing happens.
Note that anything can instruct the sidebar to reset the current page, but the reset function needs to live on the component that is loaded in for the route you are on, based on routes.jsx
. In the /clubs
example, while something deep inside a component inside a component on the page could call resetreload.shouldResetOnReload(true)
, the reset function must be defined for in the ClubsPage.jsx
component code, because that is the entry point for the /clubs
route.
CSS is managed through LESS, housed in the ./less
directory and aggregrated through ./less/index.less
. There are a few common files (most notably variables.less
for the site colouring) and then for actual site content we use the same split for components and pages, to keep the style/code relation clear for developers.
The LESS compile is managed by gulp
via the less task that runs as part of ./gulp/shared/minimal.build.tasks.js
, defined in ./gulp/less.task.js
, which writes the compiled less as styles.css
into the ./dist
dir, where it will sit next to the site code bundle.
The learning site does not do its own user/session management, instead deferring to the Teach-API for this work.
The ./lib/teach-api.js
file is a connector that is loaded as part of the <Page>
template and is used to check whether the client should currently be treated as logged in or not: if the teach-api's getLoginInfo
returns a username, the client is to be considered logged in, as the indicated user. If the return is null
, there is no active session and the user is an anonymous user.
The Teach-APi itself is a Django API server that, itself, defers actual user authentication to the id.webmaker.org
service. When a user wants to log into the learning website, the following series of events occur:
- User clicks
sign in
orsign up
on a page on the learning website. - These links redirect the user to the Teach-API, with a reference to the learning site URL they came from
- the Teach-API immediately redirects the user to
id.webmaker.org
("id.wmo") with a query argument that ensures the original learning site location is not lost - the user runs through the id.wmo motions, filling in their username and password, and id.wmo performs its authentication routine.
- when the user's credentials are correct, id.wmo performs a callback to the Teach-API, not the learning site, informing it that the user is authenticated
- the Teach-API marks the user as having an active session using Django session management, and then immediately redirects the user's client to the learning site URL the user originally came from.
- At this point the flow is the same as the next section
- The user loads a page, which loads the
teach-api.js
code, which checks to see if a localStorage entry exists for a teach api session. - If one exists, the user is considered logged in and they will see their personal information as part of the site navigation.
- If no entry exists, the
teach-api.js
code contacts the Teach-API to see if, based on cookie information, the Django session manager thinks there should be an active session. If so, theteach-api.js
code builds the localStorage entry and the user is considered logged in. - If no Django session exists, the user is considered anonymous and they will see
sign in
/sign up
options as part of the site navigation.
A significant part of the learning site is around Mozilla clubs, which are saved and loaded via the Teach-API, which stores club information as user-submitted structured data.
Club information is consulted and manipulated through the teach-api.js
interface.
Locked behind a ENABLE_BADGES
feature flag, the learning site has a series of pages for Credly badges that are tied to user accounts.
Badge data is negotiated through the ./lib/badges-api.js
code. This code, like the clubs code, relies on the Teach-API, but because of contracting work specifically around badges, ended up having its own accessor for talking to the Teach-API.
(this file may be merged into the teach-api.js
file at some point, once Badges have been fully integrated into the learning website)
Here be dragons.
As a project with a fair bit of legacy heritage, the build system is no on par with most of our other projects, relying on several different layers of technology, some no longer used, some used better in newer projects. testing involves one highly convoluted test pass, and one fairly straight forward on.
mocha
is used for straight-forward testing using standard describe
/ it.should(...)
assertion tests, running over all files in the ./test
directory.
phantom
is used for straight forward browser testing using test/browser/run-in-phantom.js
as entry point. It loads the website from the dist
bundle into a headless browser and then runs a whole bunch of assertion tests against the site "as loaded in a browser and used by a user".
Here be the true dragons.
The gulp process is a bit more convoluted because it used to be based on pregenerating the entire site as static content, which we no longer do.
gulp test
starts by executing the copy-static-files
, less
, and webpack
tasks, which builds the project into the dist
directory.
Once done, it starts the test-react-warnings
task defined in the ./gulp/test.tasks.js
file. This test task runs the createIndexFileStream
function from ./gulp/shared/index.functions.js
, which builds an indexStaticWatcher
and runs its build function, which in turns reports any errors/warnings that occur during that.
indexStaticWatcher
is defined in ./lib/build/index-static-watcher.js
, and its build()
function effectively kicks off a custom webpack compile run based on the configuration file located in ./config/index.static.config
. The resulting bundle is then used to build a new IndexFileStream
, and its errors are the ones that are passed through into the test task for determining whether or not any failures occured.
The IndexFileStream
is defined in ./lib/build/gulp-index-file-stream
, and loads the custom webpack-generated site bundle, extracting the URL information (which still come from routes.jsx), converting each url to a virtual file, which can be loaded in as if it were a real file for serving/testing. Which we don't do, because the only thing we catch is warnings/errors during the file generation process (the stream of virtual files is discarded once it's done generating).
-
mocha
runs regular tests withsinon
assertion testing. -
pantom
runs regular browser tests withsinon
assertion testing. -
gulp
runs a webpack build of the site and checks whether all URLs compile to HTML properly, using spaghetti logic.
This is simply a matter of running npm start
which is an alias for node app.js
. There be no dragons here.
Please see https://github.com/mozilla/learning.mozilla.org/blob/develop/L10N.md for now on how we do localisation in this project.
In order to run learning.mozilla.org completely via localhost, you will need the following projects and dependencies:
- Node.js, npm, python, pip, virtualenv, PostgreSQL
- learning.mozilla.org
- teach-api
- id.webmaker.org
- login.webmaker.org
Most of the dependencies are easily installed, PostgreSQL might be a bit of a jerk depending on the OS you use.
The git repos all come with their own instructions in their respective README.md files, and it's easy to forget but the learning.mozilla.org repo needs an .env
file made with the content:
TEACH_API_URL=http://localhost:8000
and the Teach API needs the following enviroment values set (or ./teach/settings.py
updated):
IDAPI_URL = os.environ.get('IDAPI_URL', 'http://localhost:1234')
IDAPI_CLIENT_ID = os.environ.get('IDAPI_CLIENT_ID', 'test')
IDAPI_CLIENT_SECRET = os.environ.get('IDAPI_CLIENT_SECRET', 'test')
LOGINAPI_URL = os.environ.get('LOGINAPI_URL', 'http://localhost:3000')
TEACH_SITE_URL = os.environ.get('TEACH_SITE_URL', 'http://localhost:8008')
Finally, there is one crucial bit missing pertaining to the id.webmaker.org system:
In order to use learning.mozilla.org with webmaker user accounts, we need the teach-api installed to manage user sessions, id.webmaker.org installed to broker user authentication, and login.webmaker.org installed to act as user database endpoint. However, for the teach-api to be able to negotiate with id.webmaker.org it needs a "client id" so that id.webmaker.org can verify that the authentication calls are being made by an authorized service.
The README.md does not currently cover how to do that, so: after installing id.webmaker.org, create a new database through bash (or whatever terminal you use) using the createdb
command that you get when installing PostgreSQL:
$> createdb webmaker_oauth_test
After this, tell id.webmaker.org to run its tests using npm test
. This will connect to the database, populate it, clear most of it, but leave three client ids for testing purposes. We need to pick one of these to act as the teach-api's client id.
Start the postgres CLI client and connect to the webmaker_oauth_test database:
$> psql webmaker_oauth_test
note: if you are a Windows user, you will probably get an error because there is "no role " defined. Using the postgresql administrative username and password you picked when you installed postgresql, run the createuser
CLI utilily:
$> createuser -s -U YourPostgresAdminUsername -W YourWindowsUserName
This creates a new superuser with your windows user name (-s
setting you use as superuser, -U ...
being the flag that runs the CLI utility as the postgres admin, and -W
being the flag that requires you to type the admin password). You should now be able to run the psql webmaker_oauth_test
command properly.
Once connected with the postgres CLI, issue the following update command:
psql> update clients set redirect_uri = 'http://localhost:8000/auth/oauth2/callback' where client_id = 'test';
This sets the callback URL for the test
clientid to the URL that the teach-api exposes for authentication purposes. With this done, in the teach-api project open the ./teach/settings.py
file and update the IDAPI_CLIENT_ID
and IDAPI_CLIENT_SECRET
to say "test"
if they don't already.
You should now be able to run the whole shebang locally by:
- starting the login.wmo service
- starting the id.wmo service
- starting the teach-api (in its virtualenv)
- starting learning.moz
- navigating to http://localhost:8008
- sign in/up (which only works if everything's set up correctly)
- finally getting things done ✨
Whether you are renaming, removing, or changing the default value for an existing environment variable, make sure that:
- The environment variables section on README.md is up to date, and note that:
- we're following 12-factor practices, so your variable should have a default value that "just works"(tm), and also note that:
-
config/webpack.config.js
reflects your changes, and: - your
.env
file reflects the overrides on the default that you need.