7.1 KiB
Live Events
Canvas includes the ability to push a subset of real-time events to a
Kinesis stream, which can then be consumed for various analytics
purposes. This is not a full-fidelity feed of all changes to the
database, but a targetted set of interesting actions such as
grade_change
, login
, etc.
Development and Testing
To enabled Live Events, you need to configure the plugin in the /plugins
interface. If using the docker-compose dev setup, there is a "fake
kinesis" available in docker-compose/kinesis.override.yml available for
use. Once it's up, make sure you have the aws
cli installed, and run
the following command to create a stream (with canvas running):
AWS_ACCESS_KEY_ID=key AWS_SECRET_ACCESS_KEY=secret aws --endpoint-url http://kinesis.canvaslms.docker/ kinesis create-stream --stream-name=live-events --shard-count=1 --region=us-east-1
Once the stream is created, configure your Canvas (by going to the /plugins url on your running Canvas) to use it:
Setting Name | Value |
---|---|
Kinesis Stream Name | live-events |
AWS Region | us-east-1 |
AWS Endpoint | http://kinesis:4567 |
AWS Access Key ID | key |
AWS Secret Access Key | secret |
- or -
Setting Name | Value |
---|---|
Configure using consul | checked |
Restart Canvas, and events should start flowing to your kinesis stream.
You can view the stream with the tail_kinesis
tool:
docker-compose run --rm web script/tail_kinesis http://kinesis:4567 live-events
Contract Tests
We use Pact to ensure our live events messages don't regress. For contract testing live events we use the pact-messages gem. Here's some helpful terminology to get started:
Provider
: the live events producerConsumer
: the live events subscriberPact file
: the json file in which the consumer-dictated contract is definedPact Broker
: the web app that serves as a permanent storage repository for sharing Pact files between consumers and providers. Instructure has its own internal Pact Broker.
Canvas LMS Live Events Consumers
Canvas LMS emits live events to multiple subscribers including Quiz LTI and Gauge, among others.
You can view which consumers have contract tests in place for Canvas LMS live
events in this repo's spec/contracts/service_consumers/live_events/
directory.
Any consumers who desire to have contract tests with Canvas LMS live events will need to initiate that effort as Canvas LMS cannot run the tests without the consumer's Pact file.
Development Workflow:
This example uses the quiz_lti
repo, which you can substitute with any other
live events consumer.
- If you're adding a new consumer to the contract tests suite, open
spec/contracts/service_consumers/pact_config.rb
and add a new constant to theConsumers
module. The new constant's value must match the one defined in the new consumer's Pact file(s)! For example, if the consumer calls itself "Quiz LTI" in its Pact file(s) then the new constant should beQUIZ_LTI = 'Quiz LTI'.freeze
- In the quiz_pact_broker repo, spin up a Pact Broker with
bin/dev-setup
- In the
quiz_lti
repo, spin up the Quiz LTI service withbin/dev-setup
- In the
quiz_lti
repo'sspec/contracts/service_providers/live_events
directory, write or modify a live events contract test - In the
quiz_lti
repo, runbin/contracts-generate
to generate a Pact file and publish it to the local Pact Broker - In this repo, open
spec/contracts/service_consumers/pact_config.rb
and comment out the consumers you won't be testing. For example:
module Consumers
# GAUGE = 'Gauge'.freeze
QUIZ_LTI = 'Quiz LTI'.freeze
All = Consumers.constants.map { |c| Consumers.const_get(c) }.freeze
end
- In this repo, spin up the Canvas LMS service with
script/docker_dev_setup.sh
- In this repo, run
bin/contracts-verify-live-events
to pull the new Pact files from the local Pact Broker and run the contract tests against the Canvas LMS service. The new tests will most likely fail. That's OK! This is a TDD workflow. - In this repo, write or modify Pact::Messages RSpec tests in the
spec/contracts/service_consumers/live_events/
directory for the new live events contracts. We've written a module to help you get started. See thequiz_lti/assignment_created_spec.rb
for an example. - Run
bin/contracts-verify-live-events
again - Repeat steps 8 and 9 until all tests pass
Bonus: You can view the Pact file(s) in the Pact Broker at http://pact:broker@pact-broker.docker along with an API dependency graph!
Debugging Failures
Pact has some basic RSpec output for failed specs. It also keeps a log in
log/pact.log
and offers general pointers for debugging.
Above all, learn the Pact basics.
What should live events contract tests cover?
The aim of contract testing here is not for every consumer to verify the entirety of every live events message emitted by Canvas LMS. Rather, the goal is to ensure changes to the Canvas LMS live events service won't break its consumers. We can best accomplish this when each consumer's Pact file defines only the message contents on which the consumer relies.
For example, let's say both Quiz LTI and Gauge subscribe to the Canvas LMS live
event, quizzes.item_created
. Let's say Quiz LTI only relies on the following
hash in the live event message:
{
root_account_uuid: 'abcd-example-uuid',
outcome_alignment_set_guid: '1234-example-guid'
}
And let's say Gauge only relies on this hash:
{
root_account_uuid: 'abcd-example-uuid',
item_id: '1'
}
But the actual quizzes.item_created
live event message looks like this:
{
root_account_uuid: 'abcd-example-uuid',
outcome_alignment_set_guid: '1234-example-guid',
item_id: '1',
scoring_algorithm: 'Equalized'
}
Therefore Gauge's Pact file shouldn't include outcome_alignment_set_guid
or
scoring_algorithm
, and Quiz LTI's Pact file shouldn't include item_id
or
scoring_algorithm
. Each Pact file defines the minimum contract required. This
way if Canvas LMS were to change its Live Events service such that it no longer
emits the item_id
, then Canvas LMS will know right away that merging the code
change would break Quiz LTI but not Gauge.
Changing Contracts
When a Pact spec fails in Canvas LMS this means your code change isn't compatible with what a consumer expects from Canvas LMS. These consumer expectations are defined in the given consumer code base and communicated via its Pact file(s).
If your code change is absolutely necessary and you need to break the consumer contract, then you'll first need to talk with that team and work with them to change the contract in their repo. After that change is merged then you can merge your new changes to Canvas LMS.