Using Tampermonkey to add question banks

After being annoyed that there is no question bank API, I thought about using requests to the Canvas UI using Tampermonkey. It turns out that this is complicated because although in the end, it is just POST of a title (i.e., the name of a question bank) via a form. Unfortunately, this form uses an authenticity token to prevent cross site request forgery (CSRF). So the solution was to find the form on a page (when logged into the course where you want to create the question banks), fill-in the name of the question bank and send the form. The results (and script) are shown  below:


// ==UserScript==
// @name        Add Question Bank
// @description Generates a question bank of a given name
// @include     http://*/courses/*/question_banks
// @require     https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/1.3.3/FileSaver.js
// @require     http://code.jquery.com/jquery-3.3.1.min.js
// @require     https://code.jquery.com/ui/1.11.4/jquery-ui.min.js
// @version     10
// @grant       none
// @run-at      document-idle
// ==/UserScript==
/* globals $ */
(function() {
  'use strict';

  // get the validation token and do a post

  var userData = {};
  
  createBank('Test bank20');
  createBank('Test bank21');
  createBank('Test bank22');
  createBank('Test bank23');


  function createBank(title) {
      var token=""
      var courseid=getCourseId()
      var url='http://canvas.docker/courses/'+courseid+'/question_banks'

      if ($('#right-side-wrapper').length) {
          $('#right-side-wrapper').click() // click the button
          token=$("#edit_bank_form input[name=authenticity_token]").val() // output the token just to see it
          console.info("token", token);
          $("#edit_bank_form #assessment_question_bank_title").val(title) // set the title of the new question bank
          $.post( url, $('form#edit_bank_form').serialize(), function(data) {
              var result = data;
              var id=result.assessment_question_bank.id;
              console.info("back from POST, the new id is ", id, " and the full result is ", result)
          },
                 'json' // I expect a JSON response
          );
    return;
    }
  }
 
  function getCourseId() {
    var courseId = null;
    try {
      var courseRegex = new RegExp('/courses/([0-9]+)');
      var matches = courseRegex.exec(window.location.href);
      if (matches) {
        courseId = matches[1];
      } else {
        throw new Error('Unable to detect Course ID');
      }
    } catch (e) {
      errorHandler(e);
    }
    return courseId;
  }


  function errorHandler(e) {
    console.log(e.name + ': ' + e.message);
  }
})();

The resulting set of question banks:

Question-banks-20190715a.png

Console output:

Question-banks-20190715-console-output.png

It would have been much nicer to have had an API such as:

POST /api/v1/courses/:course_id/question_banks

with a JSON payload of the form:

payload={'assessment_question_bank[title]': title}

I would like to thanks James Jones Links to an external site. for his script access-report-instructor.user.js (from https://github.com/jamesjonesmath/canvancement Links to an external site. ) as an inspiration. It helped get me started with the script I wrote (which is not very elegant, but shows that is is possible to create question banks without having to manually click at the GUI).