The goal of GeoCode is to get students exploring geological threats using the practice of programming. In this case, we are using Blockly to generate code that runs a simple tephra distribution model and seismic activity model.
Students change eruption parameters using Blockly.
master
: The latest work, may not have been fully-QA'd so should not be used in production. http://geocode-app.concord.org/branch/master/index.htmlproduction
: http://geocode-app.concord.org/[tag]
(e.g. 2.1.0): Versioned releases. http://geocode-app.concord.org/version/2.1.0/index.html[feature-branch]
: Feature branch. The built url will strip off any leading numbers in the branch (e.g. "1234-feature" -> "feature"). http://geocode-app.concord.org/branch/[feature]/index.html
- Blockly Block programming tool developed by Google.
- JS-interpreter An isolated javascript evaluation engine.
- mobx Simple, scalable state management.
- mobx-state-tree Transactional, MobX powered state container.
- styled-components CSS only components via ES6.
- PixiJS 2D Graphics and scene-graph utlity.
- React-Pixi Use react state to drive Pixi scenes.
- React Web Components.
- TypeScript Javascript that scales.
- D3 JavaScript library used for representing data.
- ReactFauxDom Integrates D3 with React.
- Clone this repo and
cd
into it - Run
npm install
to pull dependencies. - Run
npm start
to runwebpack-dev-server
in development mode with hot module replacement.
If you want to build a local version run npm build
, it will create the files in the dist
folder.
You do not need to build to deploy the code, that is automatic. See more info in the Deployment section below.
- Find a story from the GeoCode - Pivotal Tracker
- Assign yourself the role story owner.
- Add
npaessel
orlblagg
as a code reviewer for the story. - Copy the Story ID, you will need it in the next step.
- Create a git topic branch in the format of
#####-pt-story-headline-abbreviated
where the PT story ID is thefirst part of the branch name. - In your commit comment-body reference the PT Story ID and URL.
e.g.:
[#1234567] https://www.pivotaltracker.com/story/show/1234567
Commits should have concise headlines with details in the body. - Test and lint your branch.
- Push it to GitHub e.g.
git push --set-upstream origin #####-pt-story-headline-abbreviated
- After a few moments, your deployed branch should be available at
http://geocode-app.concord.org/branch/<branchname>/index.html
. The built url will strip off any leading numbers in the branch (e.g. "1234-feature" -> "feature"). Verify that your deploy worked, and copy the URL. - When you are satisfied with your commits, and the deployment looks good, submit
a pull request for your branch in GitHub, adding
npaessel
orlublagg
as code reviewers. In your pull request summarize the work, reference the PT story, and provide a link to the deployed demo branch. - Update the PT story with a link to the demo deployment and the GitHub pull request.
- Data related to simulations is saved in
stores/tephra-simulation-store.ts
,stores/seismic-simulation-store.ts
, andstores/sample-collections-store.ts
. - Data related to charts (used in tephra unit) is saved in
stores/charts-store.ts
. - Data related to UI options available to authors is saved in
stores/ui-store.ts
. - Data related to the code interpreter is stored in
stores/blockly-store.ts
.
The simulation is a simple tephra calculation function defined in tephra2.ts
.
This function will be improved, modified with the help of our volcanologist partners.
We use D3.js to create graphs and charts representing data from simulations. D3 allows you to bind arbitrary data to the DOM, and then apply data-driven transformations to the document. Integrating D3 with React can be challenging since both want direct access to the DOM. To help with potential conflicts, we use ReactFauxDOM, a DOM-like data structure that can be mutated by D3 and then rendered to React elements.
The Blockly configurations and blocks are specified in the GeoCode project in the following locations:
src/blockly-blocks/blocks.js
Imports the files used for the custom blocks. Individual files are located insrc/blockly-blocks
(e.g.,src/blockly-blocks/block-add-town.js
,src/blockly-blocks/block-add-volcano.js
, etc.).src/assets/blockly-authoring/code/basic-setup.xml
andsrc/assets/blockly-authoring/code/nested-loops.xml
These are the two default programs that are loaded in the Blocks panel (which file is used depends on settings).src/assets/blockly-authoring/toolbox/first-toolbox.xml
andsrc/assets/blockly-authoring/toolbox/full-toolbox.xml
These are the list of blocks available in the toolbox (which file is used depends on settings).
You can create new blocks using the Blockly Developer Tools block factory or by copying and modifying an existing block in the GeoCode project.
If using the Block Factory, use the browser interface to create your new block.
If copying and modifying an existing block, start with one of the blocks located in src/blockly-blocks
(e.g., src/blockly-blocks/block-add-town.js
, src/blockly-blocks/block-add-volcano.js
, etc.). Make a copy of one of the block files, and then make any needed changes or additions.
Whether using the Block Factory or copying and modifying an existing block, each new block must be added to the GeoCode project. To add a block to the project, create a new JavaScript module containing the block's UI and code generation methods. Name the new JavaScript module using kebab-case (e.g., block-my-example-block.js
). Add the new JavaScript module to src\blockly-blocks
, and add an import for the new JavaScript module in blocks.js
. A new block must have a unique camelCase key (e.g., myExampleBlock
).
Each existing GeoCode block has UI and code generation methods defined in its JavaScript module. Therefore, we must also define UI and code generation methods for our new block in the newly created JavaScript module that we added to src\blockly-blocks
and imported in blocks.js
. If you are using the Blockly Developer Tools Block Factory, take the Block Definition JavaScript and the Generator stub JavaScript and add it to your new block's JavaScript module. If copying and modifying an existing block, then be sure to make changes to both of these sections.
The UI generation for the new block is defined in the Blockly.Blocks
global object. This is done in the block's JavaScript module. Blocks register themselves in this object using a unique key (the block name
field if you are using the Block Factory), such as Blockly.Blocks['print']
.
Similarly, the code generation for the new block is defined in the block's JavaScript module in a global variable named Blockly.JavaScript
.
Any custom functions added to a block's code generation must be defined in interpreter.js
in the function named
makeInterpreterFunc
. Look at an example to see how the function is registered:
addFunc("setWindspeed", (speed) => {
simulation.setWindSpeed(speed);
});
The above method adds "setWindspeed" as a new javascript function that the blockly generated code can use. In this example, it modifies a parameter in the simulation store.
If a function needs to return data, in order to plug it into another block, the
return value must be wrapped in a { data: any }
object:
addFunc("add", (params: {a: number, b: number}) => {
const val = params.a + params.b;
return {
data: val
};
});
For a new block to be available in the list of blocks that a user can add from the Blocks panel, it must be added to one or both of the toolboxes located in src/assets/blockly-authoring/toolbox/first-toolbox.xml
and src/assets/blockly-authoring/toolbox/full-toolbox.xml
. Modify one or both of these XML files to include your new block. Create a new XML block
element nested inside the proper category
element. Be sure to set the type
to the unique camelCase key that was created for your block (e.g., myExampleBlock
). For example, the add volcano block is added to the volcano category by inserting <block type="addVolcano"></block>
inside of the volcano category element. If your block requires default fields, then add the appropriate field
elements inside of the block
element.
To change the default block program that is shown in the Blocks panel when the app is loaded, modify the contents of src/assets/blockly-authoring/code/basic-setup.xml
or src/assets/blockly-authoring/code/nested-loops.xml
. Depending on your settings, the program specified in one of these XML files will be loaded as the default program.
You can also create a program in the Blocks panel (from eg http://localhost:8080/), access a copy of the XML defining the program, and copy it into one of the above files. To access the XML of the current program in the Blocks panel by doing the following:
- in GeoCode open the model options and press Save current code to local storage
- open the browser developer tools
- click on the "Application" tab
- under "Storage" open "Local Storage"
- the program XML is stored in local storage under the key "blockly-workspace"
- Make sure if you are using Visual Studio Code that you use the workspace version of TypeScript.
To ensure that you are open a TypeScript file in VSC and then click on the version number next to
TypeScript React
in the status bar and select 'Use Workspace Version' in the popup menu.
See the seismic data readme, and the fetch-data
scripts in package.json.
The GeoCoder model has many authoring options that affect how the model runs and what the model shows users. Most of the authoring options are shown in the “Model Options” menu in the top right corner. A document describing authoring controls can be found here.
Production releases to S3 are based on the contents of the /dist folder and are built automatically by Travis for each branch pushed to GitHub and each merge into production.
Merges into production are deployed to http://geocode-app.concord.org.
Other branches are deployed to http://geocode-app.concord.org/branch/.
You can view the status of all the branch deploys here.
The report-item-interactive is deployed next to the main app in a file called report-item.html
eg: http://geocode-app.concord.org/branch//report-item.html
To deploy a production release:
- Make sure all of the stories for the release are accepted and PRs are merged.
- Run
npm version [major.minor.patch]
(for example,npm version 5.8.0
) to update the version number in package.json and package-lock.json, create a commit with this change, and create a tag. It isn't necessary to add thev
prefix, the version command does that automatically. This will make a tag calledv5.8.0
- Push the updated tag:
git push
andgit push origin v[major.minor.patch]
- GitHub actions will automatically start building and deploying the branch when it is pushed, and the tag. You can track the progress here: /~https://github.com/concord-consortium/collaborative-learning/actions
- Generate the release notes.
- You need to check out: /~https://github.com/concord-consortium/dev-templates
- Go into the scripts folder
cd scripts
and runnpm i
- Create a
.env
file in the scripts folder to include aPT_TOKEN=<token>
. - You can get the PT token from https://www.pivotaltracker.com/profile and then copy the API TOKEN.
- Run the release notes script:
npm run release-notes <pt label>
. Thept label
is usuallyclue-[major.minor.patch]
. This output is for the GitHub release. - Run the release notes script again for slack:
npm run release-notes <pt label> slack
. This changes the formatting so it can be pasted into Slack.
- Create a GitHub release and add the release notes.
- Go to /~https://github.com/concord-consortium/collaborative-learning/releases
- Click "Draft a new release"
- Choose the tag that you pushed before
v[major.minor.patch]
- Give it a title like: "Version [major.minor.patch] - Released [Date of planned release]"
- Paste in the non-slack release notes from the script for the description
- Click publish release
- Do the release: run a GitHub action with the tag name.
- Go to /~https://github.com/concord-consortium/collaborative-learning/actions
- Click on "Release Production"
- Click on "Run Workflow"
- Fill the dialog with the tag
v[major.minor.patch]
- Click the "Run Workflow"
- Open an activity that is using the interactive and ensure that it works as expected.
- If everything works correctly, notify the slack channel #releases.
The message should be the slack formatted release notes along with the date and time of the planned release. To paste the message into slack and get the formatting working correctly you have to into your slack settings/advanced section and tick the box:
Format messages with markup
. Usually we include a link to the github release notes in the slack message.
Run npm test
to run jest tests. Run npm run test:full
to run jest and Cypress tests.
Inside of your package.json
file:
--browser browser-name
: define browser for running tests--group group-name
: assign a group name for tests running--spec
: define the spec files to run--headed
: show cypress test runner GUI while running test (will exit by default when done)--no-exit
: keep cypress test runner GUI open when done running--record
: decide whether or not tests will have video recordings--key
: specify your secret record key--reporter
: specify a mocha reporter
cypress run --browser chrome
will run cypress in a chrome browsercypress run --headed --no-exit
will open cypress test runner when tests begin to run, and it will remain open when tests are finished running.cypress run --spec 'cypress/integration/examples/smoke-test.js'
will point to a smoke-test file rather than running all of the test files for a project.
- hide-model-options: when included, the model options authoring dialog in the upper-right is hidden.
- unit={name}: configure application for {unit}. Unit name can currently be set to
Seismic
orTephra
(e.g.,unit=Seismic
). Values are case sensitive.
This project also includes source files and a webpack target for a customized Portal report view, called a report item interactive.
The code deployed as the report item interactive is located in src/report-item
.
When an interactive use the Geocode report item interactive (set in LARA authoring using the "Report Item URL" field),
the report will render the component src/report-item/report-item.tsx
as an Iframe. This report item uses the
LARA Interactive API to register a report-item-answer-listener. That listener will render custom HTML for
each student answer that is solicited by rendering src/report-item/studnent-answer-view.tsx
to text using
renderToStaticMarkup
in the studentAnswerHtml
method.
GeoCode is Copyright 2018 (c) by the Concord Consortium and is distributed under the MIT license.
See license.md for the complete license text.